226 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			226 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2016 The go-ethereum Authors
 | |
| // This file is part of the go-ethereum library.
 | |
| //
 | |
| // The go-ethereum library is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU Lesser General Public License as published by
 | |
| // the Free Software Foundation, either version 3 of the License, or
 | |
| // (at your option) any later version.
 | |
| //
 | |
| // The go-ethereum library is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | |
| // GNU Lesser General Public License for more details.
 | |
| //
 | |
| // You should have received a copy of the GNU Lesser General Public License
 | |
| // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| package storage
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"path/filepath"
 | |
| 	"sync"
 | |
| 
 | |
| 	"github.com/ethereum/go-ethereum/metrics"
 | |
| 	"github.com/ethereum/go-ethereum/swarm/log"
 | |
| 	"github.com/ethereum/go-ethereum/swarm/storage/mock"
 | |
| )
 | |
| 
 | |
| type LocalStoreParams struct {
 | |
| 	*StoreParams
 | |
| 	ChunkDbPath string
 | |
| 	Validators  []ChunkValidator `toml:"-"`
 | |
| }
 | |
| 
 | |
| func NewDefaultLocalStoreParams() *LocalStoreParams {
 | |
| 	return &LocalStoreParams{
 | |
| 		StoreParams: NewDefaultStoreParams(),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| //this can only finally be set after all config options (file, cmd line, env vars)
 | |
| //have been evaluated
 | |
| func (p *LocalStoreParams) Init(path string) {
 | |
| 	if p.ChunkDbPath == "" {
 | |
| 		p.ChunkDbPath = filepath.Join(path, "chunks")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // LocalStore is a combination of inmemory db over a disk persisted db
 | |
| // implements a Get/Put with fallback (caching) logic using any 2 ChunkStores
 | |
| type LocalStore struct {
 | |
| 	Validators []ChunkValidator
 | |
| 	memStore   *MemStore
 | |
| 	DbStore    *LDBStore
 | |
| 	mu         sync.Mutex
 | |
| }
 | |
| 
 | |
| // This constructor uses MemStore and DbStore as components
 | |
| func NewLocalStore(params *LocalStoreParams, mockStore *mock.NodeStore) (*LocalStore, error) {
 | |
| 	ldbparams := NewLDBStoreParams(params.StoreParams, params.ChunkDbPath)
 | |
| 	dbStore, err := NewMockDbStore(ldbparams, mockStore)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return &LocalStore{
 | |
| 		memStore:   NewMemStore(params.StoreParams, dbStore),
 | |
| 		DbStore:    dbStore,
 | |
| 		Validators: params.Validators,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func NewTestLocalStoreForAddr(params *LocalStoreParams) (*LocalStore, error) {
 | |
| 	ldbparams := NewLDBStoreParams(params.StoreParams, params.ChunkDbPath)
 | |
| 	dbStore, err := NewLDBStore(ldbparams)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	localStore := &LocalStore{
 | |
| 		memStore:   NewMemStore(params.StoreParams, dbStore),
 | |
| 		DbStore:    dbStore,
 | |
| 		Validators: params.Validators,
 | |
| 	}
 | |
| 	return localStore, nil
 | |
| }
 | |
| 
 | |
| // isValid returns true if chunk passes any of the LocalStore Validators.
 | |
| // isValid also returns true if LocalStore has no Validators.
 | |
| func (ls *LocalStore) isValid(chunk Chunk) bool {
 | |
| 	// by default chunks are valid. if we have 0 validators, then all chunks are valid.
 | |
| 	valid := true
 | |
| 
 | |
| 	// ls.Validators contains a list of one validator per chunk type.
 | |
| 	// if one validator succeeds, then the chunk is valid
 | |
| 	for _, v := range ls.Validators {
 | |
| 		if valid = v.Validate(chunk.Address(), chunk.Data()); valid {
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 	return valid
 | |
| }
 | |
| 
 | |
| // Put is responsible for doing validation and storage of the chunk
 | |
| // by using configured ChunkValidators, MemStore and LDBStore.
 | |
| // If the chunk is not valid, its GetErrored function will
 | |
| // return ErrChunkInvalid.
 | |
| // This method will check if the chunk is already in the MemStore
 | |
| // and it will return it if it is. If there is an error from
 | |
| // the MemStore.Get, it will be returned by calling GetErrored
 | |
| // on the chunk.
 | |
| // This method is responsible for closing Chunk.ReqC channel
 | |
| // when the chunk is stored in memstore.
 | |
| // After the LDBStore.Put, it is ensured that the MemStore
 | |
| // contains the chunk with the same data, but nil ReqC channel.
 | |
| func (ls *LocalStore) Put(ctx context.Context, chunk Chunk) error {
 | |
| 	if !ls.isValid(chunk) {
 | |
| 		return ErrChunkInvalid
 | |
| 	}
 | |
| 
 | |
| 	log.Trace("localstore.put", "key", chunk.Address())
 | |
| 	ls.mu.Lock()
 | |
| 	defer ls.mu.Unlock()
 | |
| 
 | |
| 	_, err := ls.memStore.Get(ctx, chunk.Address())
 | |
| 	if err == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 	if err != nil && err != ErrChunkNotFound {
 | |
| 		return err
 | |
| 	}
 | |
| 	ls.memStore.Put(ctx, chunk)
 | |
| 	err = ls.DbStore.Put(ctx, chunk)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // Get(chunk *Chunk) looks up a chunk in the local stores
 | |
| // This method is blocking until the chunk is retrieved
 | |
| // so additional timeout may be needed to wrap this call if
 | |
| // ChunkStores are remote and can have long latency
 | |
| func (ls *LocalStore) Get(ctx context.Context, addr Address) (chunk Chunk, err error) {
 | |
| 	ls.mu.Lock()
 | |
| 	defer ls.mu.Unlock()
 | |
| 
 | |
| 	return ls.get(ctx, addr)
 | |
| }
 | |
| 
 | |
| func (ls *LocalStore) get(ctx context.Context, addr Address) (chunk Chunk, err error) {
 | |
| 	chunk, err = ls.memStore.Get(ctx, addr)
 | |
| 
 | |
| 	if err != nil && err != ErrChunkNotFound {
 | |
| 		metrics.GetOrRegisterCounter("localstore.get.error", nil).Inc(1)
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	if err == nil {
 | |
| 		metrics.GetOrRegisterCounter("localstore.get.cachehit", nil).Inc(1)
 | |
| 		return chunk, nil
 | |
| 	}
 | |
| 
 | |
| 	metrics.GetOrRegisterCounter("localstore.get.cachemiss", nil).Inc(1)
 | |
| 	chunk, err = ls.DbStore.Get(ctx, addr)
 | |
| 	if err != nil {
 | |
| 		metrics.GetOrRegisterCounter("localstore.get.error", nil).Inc(1)
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	ls.memStore.Put(ctx, chunk)
 | |
| 	return chunk, nil
 | |
| }
 | |
| 
 | |
| func (ls *LocalStore) FetchFunc(ctx context.Context, addr Address) func(context.Context) error {
 | |
| 	ls.mu.Lock()
 | |
| 	defer ls.mu.Unlock()
 | |
| 
 | |
| 	_, err := ls.get(ctx, addr)
 | |
| 	if err == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return func(context.Context) error {
 | |
| 		return err
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (ls *LocalStore) BinIndex(po uint8) uint64 {
 | |
| 	return ls.DbStore.BinIndex(po)
 | |
| }
 | |
| 
 | |
| func (ls *LocalStore) Iterator(from uint64, to uint64, po uint8, f func(Address, uint64) bool) error {
 | |
| 	return ls.DbStore.SyncIterator(from, to, po, f)
 | |
| }
 | |
| 
 | |
| // Close the local store
 | |
| func (ls *LocalStore) Close() {
 | |
| 	ls.DbStore.Close()
 | |
| }
 | |
| 
 | |
| // Migrate checks the datastore schema vs the runtime schema, and runs migrations if they don't match
 | |
| func (ls *LocalStore) Migrate() error {
 | |
| 	schema, err := ls.DbStore.GetSchema()
 | |
| 	if err != nil {
 | |
| 		log.Error(err.Error())
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	log.Debug("found schema", "schema", schema, "runtime-schema", CurrentDbSchema)
 | |
| 	if schema != CurrentDbSchema {
 | |
| 		// run migrations
 | |
| 
 | |
| 		if schema == "" {
 | |
| 			log.Debug("running migrations for", "schema", schema, "runtime-schema", CurrentDbSchema)
 | |
| 
 | |
| 			// delete chunks that are not valid, i.e. chunks that do not pass any of the ls.Validators
 | |
| 			ls.DbStore.Cleanup(func(c *chunk) bool {
 | |
| 				return !ls.isValid(c)
 | |
| 			})
 | |
| 
 | |
| 			err := ls.DbStore.PutSchema(DbSchemaPurity)
 | |
| 			if err != nil {
 | |
| 				log.Error(err.Error())
 | |
| 				return err
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 |