mirror of
https://github.com/status-im/op-geth.git
synced 2025-01-22 20:50:11 +00:00
2c110c81ee
* swarm/storage/mru: Adaptive Frequency swarm/storage/mru/lookup: fixed getBaseTime Added NewEpoch constructor swarm/api/client: better error handling in GetResource() swarm/storage/mru: Renamed structures. Renamed ResourceMetadata to ResourceID. Renamed ResourceID.Name to ResourceID.Topic swarm/storage/mru: Added binarySerializer interface and test tools swarm/storage/mru/lookup: Changed base time to time and + marshallers swarm/storage/mru: Added ResourceID (former resourceMetadata) swarm/storage/mru: Added ResourceViewId and serialization tests swarm/storage/mru/lookup: fixed epoch unmarshaller. Added Epoch Equals swarm/storage/mru: Fixes as per review comments cmd/swarm: reworded resource create/update help text regarding topic swarm/storage/mru: Added UpdateLookup and serializer tests swarm/storage/mru: Added UpdateHeader, serializers and tests swarm/storage/mru: changed UpdateAddr / epoch to Base() swarm/storage/mru: Added resourceUpdate serializer and tests swarm/storage/mru: Added SignedResourceUpdate tests and serializers swarm/storage/mru/lookup: fixed GetFirstEpoch bug swarm/storage/mru: refactor, comments, cleanup Also added tests for Topic swarm/storage/mru: handler tests pass swarm/storage/mru: all resource package tests pass swarm/storage/mru: resource test pass after adding timestamp checking support swarm/storage/mru: Added JSON serializers to ResourceIDView structures swarm/storage/mru: Sever, client, API test pass swarm/storage/mru: server test pass swarm/storage/mru: Added topic length check swarm/storage/mru: removed some literals, improved "previous lookup" test case swarm/storage/mru: some fixes and comments as per review swarm/storage/mru: first working version without metadata chunk swarm/storage/mru: Various fixes as per review swarm/storage/mru: client test pass swarm/storage/mru: resource query strings and manifest-less queries swarm/storage/mru: simplify naming swarm/storage/mru: first autofreq working version swarm/storage/mru: renamed ToValues to AppendValues swarm/resource/mru: Added ToValues / FromValues for URL query strings swarm/storage/mru: Changed POST resource to work with query strings. No more JSON. swarm/storage/mru: removed resourceid swarm/storage/mru: Opened up structures swarm/storage/mru: Merged Request and SignedResourceUpdate swarm/storage/mru: removed initial data from CLI resource create swarm/storage/mru: Refactor Topic as a direct fixed-length array swarm/storage/mru/lookup: Comprehensive GetNextLevel tests swarm/storage/mru: Added comments Added length checks in Topic swarm/storage/mru: fixes in tests and some code comments swarm/storage/mru/lookup: new optimized lookup algorithm swarm/api: moved getResourceView to api out of server swarm/storage/mru: Lookup algorithm working swarm/storage/mru: comments and renamed NewLookupParams Deleted commented code swarm/storage/mru/lookup: renamed Epoch.LaterThan to After swarm/storage/mru/lookup: Comments and tidying naming swarm/storage/mru: fix lookup algorithm swarm/storage/mru: exposed lookup hint removed updateheader swarm/storage/mru/lookup: changed GetNextEpoch for initial values swarm/storage/mru: resource tests pass swarm/storage/mru: valueSerializer interface and tests swarm/storage/mru/lookup: Comments, improvements, fixes, more tests swarm/storage/mru: renamed UpdateLookup to ID, LookupParams to Query swarm/storage/mru: renamed query receiver var swarm/cmd: MRU CLI tests * cmd/swarm: remove rogue fmt * swarm/storage/mru: Add version / header for future use * swarm/storage/mru: Fixes/comments as per review cmd/swarm: remove rogue fmt swarm/storage/mru: Add version / header for future use- * swarm/storage/mru: fix linter errors * cmd/swarm: Speeded up TestCLIResourceUpdate
299 lines
9.4 KiB
Go
299 lines
9.4 KiB
Go
// Copyright 2018 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/>.
|
|
|
|
// Handler is the API for Mutable Resources
|
|
// It enables creating, updating, syncing and retrieving resources and their update data
|
|
package mru
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/swarm/storage/mru/lookup"
|
|
|
|
"github.com/ethereum/go-ethereum/swarm/log"
|
|
"github.com/ethereum/go-ethereum/swarm/storage"
|
|
)
|
|
|
|
type Handler struct {
|
|
chunkStore *storage.NetStore
|
|
HashSize int
|
|
resources map[uint64]*cacheEntry
|
|
resourceLock sync.RWMutex
|
|
storeTimeout time.Duration
|
|
queryMaxPeriods uint32
|
|
}
|
|
|
|
// HandlerParams pass parameters to the Handler constructor NewHandler
|
|
// Signer and TimestampProvider are mandatory parameters
|
|
type HandlerParams struct {
|
|
}
|
|
|
|
// hashPool contains a pool of ready hashers
|
|
var hashPool sync.Pool
|
|
|
|
// init initializes the package and hashPool
|
|
func init() {
|
|
hashPool = sync.Pool{
|
|
New: func() interface{} {
|
|
return storage.MakeHashFunc(resourceHashAlgorithm)()
|
|
},
|
|
}
|
|
}
|
|
|
|
// NewHandler creates a new Mutable Resource API
|
|
func NewHandler(params *HandlerParams) *Handler {
|
|
rh := &Handler{
|
|
resources: make(map[uint64]*cacheEntry),
|
|
}
|
|
|
|
for i := 0; i < hasherCount; i++ {
|
|
hashfunc := storage.MakeHashFunc(resourceHashAlgorithm)()
|
|
if rh.HashSize == 0 {
|
|
rh.HashSize = hashfunc.Size()
|
|
}
|
|
hashPool.Put(hashfunc)
|
|
}
|
|
|
|
return rh
|
|
}
|
|
|
|
// SetStore sets the store backend for the Mutable Resource API
|
|
func (h *Handler) SetStore(store *storage.NetStore) {
|
|
h.chunkStore = store
|
|
}
|
|
|
|
// Validate is a chunk validation method
|
|
// If it looks like a resource update, the chunk address is checked against the userAddr of the update's signature
|
|
// It implements the storage.ChunkValidator interface
|
|
func (h *Handler) Validate(chunkAddr storage.Address, data []byte) bool {
|
|
dataLength := len(data)
|
|
if dataLength < minimumSignedUpdateLength {
|
|
return false
|
|
}
|
|
|
|
// check if it is a properly formatted update chunk with
|
|
// valid signature and proof of ownership of the resource it is trying
|
|
// to update
|
|
|
|
// First, deserialize the chunk
|
|
var r Request
|
|
if err := r.fromChunk(chunkAddr, data); err != nil {
|
|
log.Debug("Invalid resource chunk", "addr", chunkAddr.Hex(), "err", err.Error())
|
|
return false
|
|
}
|
|
|
|
// Verify signatures and that the signer actually owns the resource
|
|
// If it fails, it means either the signature is not valid, data is corrupted
|
|
// or someone is trying to update someone else's resource.
|
|
if err := r.Verify(); err != nil {
|
|
log.Debug("Invalid signature", "err", err)
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// GetContent retrieves the data payload of the last synced update of the Mutable Resource
|
|
func (h *Handler) GetContent(view *View) (storage.Address, []byte, error) {
|
|
if view == nil {
|
|
return nil, nil, NewError(ErrInvalidValue, "view is nil")
|
|
}
|
|
rsrc := h.get(view)
|
|
if rsrc == nil {
|
|
return nil, nil, NewError(ErrNotFound, "resource does not exist")
|
|
}
|
|
return rsrc.lastKey, rsrc.data, nil
|
|
}
|
|
|
|
// NewRequest prepares a Request structure with all the necessary information to
|
|
// just add the desired data and sign it.
|
|
// The resulting structure can then be signed and passed to Handler.Update to be verified and sent
|
|
func (h *Handler) NewRequest(ctx context.Context, view *View) (request *Request, err error) {
|
|
if view == nil {
|
|
return nil, NewError(ErrInvalidValue, "view cannot be nil")
|
|
}
|
|
|
|
now := TimestampProvider.Now().Time
|
|
request = new(Request)
|
|
request.Header.Version = ProtocolVersion
|
|
|
|
query := NewQueryLatest(view, lookup.NoClue)
|
|
|
|
rsrc, err := h.Lookup(ctx, query)
|
|
if err != nil {
|
|
if err.(*Error).code != ErrNotFound {
|
|
return nil, err
|
|
}
|
|
// not finding updates means that there is a network error
|
|
// or that the resource really does not have updates
|
|
}
|
|
|
|
request.View = *view
|
|
|
|
// if we already have an update, then find next epoch
|
|
if rsrc != nil {
|
|
request.Epoch = lookup.GetNextEpoch(rsrc.Epoch, now)
|
|
} else {
|
|
request.Epoch = lookup.GetFirstEpoch(now)
|
|
}
|
|
|
|
return request, nil
|
|
}
|
|
|
|
// Lookup retrieves a specific or latest version of the resource
|
|
// Lookup works differently depending on the configuration of `ID`
|
|
// See the `ID` documentation and helper functions:
|
|
// `LookupLatest` and `LookupBefore`
|
|
// When looking for the latest update, it starts at the next period after the current time.
|
|
// upon failure tries the corresponding keys of each previous period until one is found
|
|
// (or startTime is reached, in which case there are no updates).
|
|
func (h *Handler) Lookup(ctx context.Context, query *Query) (*cacheEntry, error) {
|
|
|
|
timeLimit := query.TimeLimit
|
|
if timeLimit == 0 { // if time limit is set to zero, the user wants to get the latest update
|
|
timeLimit = TimestampProvider.Now().Time
|
|
}
|
|
|
|
if query.Hint == lookup.NoClue { // try to use our cache
|
|
entry := h.get(&query.View)
|
|
if entry != nil && entry.Epoch.Time <= timeLimit { // avoid bad hints
|
|
query.Hint = entry.Epoch
|
|
}
|
|
}
|
|
|
|
// we can't look for anything without a store
|
|
if h.chunkStore == nil {
|
|
return nil, NewError(ErrInit, "Call Handler.SetStore() before performing lookups")
|
|
}
|
|
|
|
var ul ID
|
|
ul.View = query.View
|
|
var readCount int
|
|
|
|
// Invoke the lookup engine.
|
|
// The callback will be called every time the lookup algorithm needs to guess
|
|
requestPtr, err := lookup.Lookup(timeLimit, query.Hint, func(epoch lookup.Epoch, now uint64) (interface{}, error) {
|
|
readCount++
|
|
ul.Epoch = epoch
|
|
ctx, cancel := context.WithTimeout(ctx, defaultRetrieveTimeout)
|
|
defer cancel()
|
|
|
|
chunk, err := h.chunkStore.Get(ctx, ul.Addr())
|
|
if err != nil { // TODO: check for catastrophic errors other than chunk not found
|
|
return nil, nil
|
|
}
|
|
|
|
var request Request
|
|
if err := request.fromChunk(chunk.Address(), chunk.Data()); err != nil {
|
|
return nil, nil
|
|
}
|
|
if request.Time <= timeLimit {
|
|
return &request, nil
|
|
}
|
|
return nil, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
log.Info(fmt.Sprintf("Resource lookup finished in %d lookups", readCount))
|
|
|
|
request, _ := requestPtr.(*Request)
|
|
if request == nil {
|
|
return nil, NewError(ErrNotFound, "no updates found")
|
|
}
|
|
return h.updateCache(request)
|
|
|
|
}
|
|
|
|
// update mutable resource cache map with specified content
|
|
func (h *Handler) updateCache(request *Request) (*cacheEntry, error) {
|
|
|
|
updateAddr := request.Addr()
|
|
log.Trace("resource cache update", "topic", request.Topic.Hex(), "updatekey", updateAddr, "epoch time", request.Epoch.Time, "epoch level", request.Epoch.Level)
|
|
|
|
rsrc := h.get(&request.View)
|
|
if rsrc == nil {
|
|
rsrc = &cacheEntry{}
|
|
h.set(&request.View, rsrc)
|
|
}
|
|
|
|
// update our rsrcs entry map
|
|
rsrc.lastKey = updateAddr
|
|
rsrc.ResourceUpdate = request.ResourceUpdate
|
|
rsrc.Reader = bytes.NewReader(rsrc.data)
|
|
return rsrc, nil
|
|
}
|
|
|
|
// Update adds an actual data update
|
|
// Uses the Mutable Resource metadata currently loaded in the resources map entry.
|
|
// It is the caller's responsibility to make sure that this data is not stale.
|
|
// Note that a Mutable Resource update cannot span chunks, and thus has a MAX NET LENGTH 4096, INCLUDING update header data and signature. An error will be returned if the total length of the chunk payload will exceed this limit.
|
|
// Update can only check if the caller is trying to overwrite the very last known version, otherwise it just puts the update
|
|
// on the network.
|
|
func (h *Handler) Update(ctx context.Context, r *Request) (updateAddr storage.Address, err error) {
|
|
|
|
// we can't update anything without a store
|
|
if h.chunkStore == nil {
|
|
return nil, NewError(ErrInit, "Call Handler.SetStore() before updating")
|
|
}
|
|
|
|
rsrc := h.get(&r.View)
|
|
if rsrc != nil && rsrc.Epoch.Equals(r.Epoch) { // This is the only cheap check we can do for sure
|
|
return nil, NewError(ErrInvalidValue, "A former update in this epoch is already known to exist")
|
|
}
|
|
|
|
chunk, err := r.toChunk() // Serialize the update into a chunk. Fails if data is too big
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// send the chunk
|
|
h.chunkStore.Put(ctx, chunk)
|
|
log.Trace("resource update", "updateAddr", r.idAddr, "epoch time", r.Epoch.Time, "epoch level", r.Epoch.Level, "data", chunk.Data())
|
|
// update our resources map cache entry if the new update is older than the one we have, if we have it.
|
|
if rsrc != nil && r.Epoch.After(rsrc.Epoch) {
|
|
rsrc.Epoch = r.Epoch
|
|
rsrc.data = make([]byte, len(r.data))
|
|
rsrc.lastKey = r.idAddr
|
|
copy(rsrc.data, r.data)
|
|
rsrc.Reader = bytes.NewReader(rsrc.data)
|
|
}
|
|
|
|
return r.idAddr, nil
|
|
}
|
|
|
|
// Retrieves the resource cache value for the given nameHash
|
|
func (h *Handler) get(view *View) *cacheEntry {
|
|
mapKey := view.mapKey()
|
|
h.resourceLock.RLock()
|
|
defer h.resourceLock.RUnlock()
|
|
rsrc := h.resources[mapKey]
|
|
return rsrc
|
|
}
|
|
|
|
// Sets the resource cache value for the given View
|
|
func (h *Handler) set(view *View, rsrc *cacheEntry) {
|
|
mapKey := view.mapKey()
|
|
h.resourceLock.Lock()
|
|
defer h.resourceLock.Unlock()
|
|
h.resources[mapKey] = rsrc
|
|
}
|