aboutsummaryrefslogblamecommitdiffstats
path: root/swarm/storage/localstore.go
blob: 4fa6fb2f60de64a73f634228308c8e0e9d075b91 (plain) (tree)


















                                                                                  
                 

                       

                                                 

                                                            

 


















                                                                                  



                                                                             



                                   


                                                           


                                                                                              



                               


                                                                     


              











                                                                              

 















                                                                                           











                                                                   
                                                                   
                               
                                      

         
                                                           


                            


                                                       
         

                                                  
         


                                        





                                                           
                                                                                       


                            
                                

 
                                                                                       
                                               





                                                                                
                       
                                                                                   
                                 
         
 
                                                                            
                                              
                       
                                                                                
                               
         
 
                                   
                         
 
 
                                                                                                


                            


                                   
         

                                            
         

 





                                                                                                      





                               















                                                                                                                



                                                                                                                  










                                                                   
// 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
}