199 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package oldresolver
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/ipfs/go-cid"
 | 
						|
	ipld "github.com/ipfs/go-ipld-format"
 | 
						|
	logging "github.com/ipfs/go-log/v2"
 | 
						|
	dag "github.com/ipfs/go-merkledag"
 | 
						|
 | 
						|
	path "github.com/filecoin-project/lotus/lib/oldpath"
 | 
						|
)
 | 
						|
 | 
						|
var log = logging.Logger("pathresolv")
 | 
						|
 | 
						|
// ErrNoComponents is used when Paths after a protocol
 | 
						|
// do not contain at least one component
 | 
						|
var ErrNoComponents = errors.New(
 | 
						|
	"path must contain at least one component")
 | 
						|
 | 
						|
// ErrNoLink is returned when a link is not found in a path
 | 
						|
type ErrNoLink struct {
 | 
						|
	Name string
 | 
						|
	Node cid.Cid
 | 
						|
}
 | 
						|
 | 
						|
// Error implements the Error interface for ErrNoLink with a useful
 | 
						|
// human readable message.
 | 
						|
func (e ErrNoLink) Error() string {
 | 
						|
	return fmt.Sprintf("no link named %q under %s", e.Name, e.Node.String())
 | 
						|
}
 | 
						|
 | 
						|
// ResolveOnce resolves path through a single node
 | 
						|
type ResolveOnce func(ctx context.Context, ds ipld.NodeGetter, nd ipld.Node, names []string) (*ipld.Link, []string, error)
 | 
						|
 | 
						|
// Resolver provides path resolution to IPFS
 | 
						|
// It has a pointer to a DAGService, which is uses to resolve nodes.
 | 
						|
// TODO: now that this is more modular, try to unify this code with the
 | 
						|
//       the resolvers in namesys
 | 
						|
type Resolver struct {
 | 
						|
	DAG ipld.NodeGetter
 | 
						|
 | 
						|
	ResolveOnce ResolveOnce
 | 
						|
}
 | 
						|
 | 
						|
// NewBasicResolver constructs a new basic resolver.
 | 
						|
func NewBasicResolver(ds ipld.DAGService) *Resolver {
 | 
						|
	return &Resolver{
 | 
						|
		DAG:         ds,
 | 
						|
		ResolveOnce: ResolveSingle,
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// ResolveToLastNode walks the given path and returns the cid of the last node
 | 
						|
// referenced by the path
 | 
						|
func (r *Resolver) ResolveToLastNode(ctx context.Context, fpath path.Path) (cid.Cid, []string, error) {
 | 
						|
	c, p, err := path.SplitAbsPath(fpath)
 | 
						|
	if err != nil {
 | 
						|
		return cid.Cid{}, nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	if len(p) == 0 {
 | 
						|
		return c, nil, nil
 | 
						|
	}
 | 
						|
 | 
						|
	nd, err := r.DAG.Get(ctx, c)
 | 
						|
	if err != nil {
 | 
						|
		return cid.Cid{}, nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	for len(p) > 0 {
 | 
						|
		lnk, rest, err := r.ResolveOnce(ctx, r.DAG, nd, p)
 | 
						|
 | 
						|
		// Note: have to drop the error here as `ResolveOnce` doesn't handle 'leaf'
 | 
						|
		// paths (so e.g. for `echo '{"foo":123}' | ipfs dag put` we wouldn't be
 | 
						|
		// able to resolve `zdpu[...]/foo`)
 | 
						|
		if lnk == nil {
 | 
						|
			break
 | 
						|
		}
 | 
						|
 | 
						|
		if err != nil {
 | 
						|
			if err == dag.ErrLinkNotFound {
 | 
						|
				err = ErrNoLink{Name: p[0], Node: nd.Cid()}
 | 
						|
			}
 | 
						|
			return cid.Cid{}, nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		next, err := lnk.GetNode(ctx, r.DAG)
 | 
						|
		if err != nil {
 | 
						|
			return cid.Cid{}, nil, err
 | 
						|
		}
 | 
						|
		nd = next
 | 
						|
		p = rest
 | 
						|
	}
 | 
						|
 | 
						|
	if len(p) == 0 {
 | 
						|
		return nd.Cid(), nil, nil
 | 
						|
	}
 | 
						|
 | 
						|
	// Confirm the path exists within the object
 | 
						|
	val, rest, err := nd.Resolve(p)
 | 
						|
	if err != nil {
 | 
						|
		if err == dag.ErrLinkNotFound {
 | 
						|
			err = ErrNoLink{Name: p[0], Node: nd.Cid()}
 | 
						|
		}
 | 
						|
		return cid.Cid{}, nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	if len(rest) > 0 {
 | 
						|
		return cid.Cid{}, nil, errors.New("path failed to resolve fully")
 | 
						|
	}
 | 
						|
	switch val.(type) {
 | 
						|
	case *ipld.Link:
 | 
						|
		return cid.Cid{}, nil, errors.New("inconsistent ResolveOnce / nd.Resolve")
 | 
						|
	default:
 | 
						|
		return nd.Cid(), p, nil
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// ResolvePath fetches the node for given path. It returns the last item
 | 
						|
// returned by ResolvePathComponents.
 | 
						|
func (r *Resolver) ResolvePath(ctx context.Context, fpath path.Path) (ipld.Node, error) {
 | 
						|
	// validate path
 | 
						|
	if err := fpath.IsValid(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	nodes, err := r.ResolvePathComponents(ctx, fpath)
 | 
						|
	if err != nil || nodes == nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return nodes[len(nodes)-1], err
 | 
						|
}
 | 
						|
 | 
						|
// ResolveSingle simply resolves one hop of a path through a graph with no
 | 
						|
// extra context (does not opaquely resolve through sharded nodes)
 | 
						|
func ResolveSingle(ctx context.Context, ds ipld.NodeGetter, nd ipld.Node, names []string) (*ipld.Link, []string, error) {
 | 
						|
	return nd.ResolveLink(names)
 | 
						|
}
 | 
						|
 | 
						|
// ResolvePathComponents fetches the nodes for each segment of the given path.
 | 
						|
// It uses the first path component as a hash (key) of the first node, then
 | 
						|
// resolves all other components walking the links, with ResolveLinks.
 | 
						|
func (r *Resolver) ResolvePathComponents(ctx context.Context, fpath path.Path) ([]ipld.Node, error) {
 | 
						|
 | 
						|
	h, parts, err := path.SplitAbsPath(fpath)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	log.Debug("resolve dag get")
 | 
						|
	nd, err := r.DAG.Get(ctx, h)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	return r.ResolveLinks(ctx, nd, parts)
 | 
						|
}
 | 
						|
 | 
						|
// ResolveLinks iteratively resolves names by walking the link hierarchy.
 | 
						|
// Every node is fetched from the DAGService, resolving the next name.
 | 
						|
// Returns the list of nodes forming the path, starting with ndd. This list is
 | 
						|
// guaranteed never to be empty.
 | 
						|
//
 | 
						|
// ResolveLinks(nd, []string{"foo", "bar", "baz"})
 | 
						|
// would retrieve "baz" in ("bar" in ("foo" in nd.Links).Links).Links
 | 
						|
func (r *Resolver) ResolveLinks(ctx context.Context, ndd ipld.Node, names []string) ([]ipld.Node, error) {
 | 
						|
	result := make([]ipld.Node, 0, len(names)+1)
 | 
						|
	result = append(result, ndd)
 | 
						|
	nd := ndd // dup arg workaround
 | 
						|
 | 
						|
	// for each of the path components
 | 
						|
	for len(names) > 0 {
 | 
						|
		var cancel context.CancelFunc
 | 
						|
		ctx, cancel = context.WithTimeout(ctx, time.Minute)
 | 
						|
		defer cancel()
 | 
						|
 | 
						|
		lnk, rest, err := r.ResolveOnce(ctx, r.DAG, nd, names)
 | 
						|
		if err == dag.ErrLinkNotFound {
 | 
						|
			return result, ErrNoLink{Name: names[0], Node: nd.Cid()}
 | 
						|
		} else if err != nil {
 | 
						|
			return result, err
 | 
						|
		}
 | 
						|
 | 
						|
		nextnode, err := lnk.GetNode(ctx, r.DAG)
 | 
						|
		if err != nil {
 | 
						|
			return result, err
 | 
						|
		}
 | 
						|
 | 
						|
		nd = nextnode
 | 
						|
		result = append(result, nextnode)
 | 
						|
		names = rest
 | 
						|
	}
 | 
						|
	return result, nil
 | 
						|
}
 |