resolver.go 3.82 KB
Newer Older
1
// Package path implements utilities for resolving paths within ipfs.
Jeromy's avatar
Jeromy committed
2 3 4
package path

import (
5
	"errors"
Jeromy's avatar
Jeromy committed
6
	"fmt"
Jeromy's avatar
Jeromy committed
7
	"time"
Jeromy's avatar
Jeromy committed
8

9
	mh "gx/ipfs/QmYf7ng2hG5XBtJA3tN34DQ2GUN5HNksEw1rLDkmr6vGku/go-multihash"
10
	"gx/ipfs/QmZy2y8t9zQH2a1b8q2ZSLKp17ATuJoCNxxyMFG5qFExpt/go-net/context"
11

12
	key "github.com/ipfs/go-ipfs/blocks/key"
13
	merkledag "github.com/ipfs/go-ipfs/merkledag"
Jeromy's avatar
Jeromy committed
14
	logging "gx/ipfs/Qmazh5oNUVsDZTs2g59rq8aYQqwpss8tcUWQzor5sCCEuH/go-log"
Jeromy's avatar
Jeromy committed
15 16
)

Jeromy's avatar
Jeromy committed
17
var log = logging.Logger("path")
Jeromy's avatar
Jeromy committed
18

19 20 21 22
// Paths after a protocol must contain at least one component
var ErrNoComponents = errors.New(
	"path must contain at least one component")

23 24
// ErrNoLink is returned when a link is not found in a path
type ErrNoLink struct {
25 26
	Name string
	Node mh.Multihash
27 28 29
}

func (e ErrNoLink) Error() string {
30
	return fmt.Sprintf("no link named %q under %s", e.Name, e.Node.B58String())
31 32
}

Jeromy's avatar
Jeromy committed
33 34 35 36 37 38
// Resolver provides path resolution to IPFS
// It has a pointer to a DAGService, which is uses to resolve nodes.
type Resolver struct {
	DAG merkledag.DAGService
}

39 40 41 42
// SplitAbsPath clean up and split fpath. It extracts the first component (which
// must be a Multihash) and return it separately.
func SplitAbsPath(fpath Path) (mh.Multihash, []string, error) {

Jeromy's avatar
Jeromy committed
43 44 45 46 47 48 49 50 51
	log.Debugf("Resolve: '%s'", fpath)

	parts := fpath.Segments()
	if parts[0] == "ipfs" {
		parts = parts[1:]
	}

	// if nothing, bail.
	if len(parts) == 0 {
52
		return nil, nil, ErrNoComponents
Jeromy's avatar
Jeromy committed
53 54 55 56 57 58
	}

	// first element in the path is a b58 hash (for now)
	h, err := mh.FromB58String(parts[0])
	if err != nil {
		log.Debug("given path element is not a base58 string.\n")
59 60 61 62 63 64 65 66
		return nil, nil, err
	}

	return h, parts[1:], nil
}

// ResolvePath fetches the node for given path. It returns the last item
// returned by ResolvePathComponents.
67
func (s *Resolver) ResolvePath(ctx context.Context, fpath Path) (*merkledag.Node, error) {
68 69 70 71 72
	// validate path
	if err := fpath.IsValid(); err != nil {
		return nil, err
	}

73
	nodes, err := s.ResolvePathComponents(ctx, fpath)
74 75 76
	if err != nil || nodes == nil {
		return nil, err
	}
77
	return nodes[len(nodes)-1], err
78 79 80 81 82
}

// 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.
83
func (s *Resolver) ResolvePathComponents(ctx context.Context, fpath Path) ([]*merkledag.Node, error) {
84 85
	h, parts, err := SplitAbsPath(fpath)
	if err != nil {
Jeromy's avatar
Jeromy committed
86 87 88
		return nil, err
	}

89
	log.Debug("Resolve dag get.")
90
	nd, err := s.DAG.Get(ctx, key.Key(h))
Jeromy's avatar
Jeromy committed
91 92 93 94
	if err != nil {
		return nil, err
	}

95
	return s.ResolveLinks(ctx, nd, parts)
Jeromy's avatar
Jeromy committed
96 97 98 99
}

// ResolveLinks iteratively resolves names by walking the link hierarchy.
// Every node is fetched from the DAGService, resolving the next name.
100 101
// Returns the list of nodes forming the path, starting with ndd. This list is
// guaranteed never to be empty.
Jeromy's avatar
Jeromy committed
102 103 104
//
// ResolveLinks(nd, []string{"foo", "bar", "baz"})
// would retrieve "baz" in ("bar" in ("foo" in nd.Links).Links).Links
105
func (s *Resolver) ResolveLinks(ctx context.Context, ndd *merkledag.Node, names []string) ([]*merkledag.Node, error) {
Jeromy's avatar
Jeromy committed
106

107
	result := make([]*merkledag.Node, 0, len(names)+1)
108 109
	result = append(result, ndd)
	nd := ndd // dup arg workaround
Jeromy's avatar
Jeromy committed
110 111 112 113 114 115 116 117 118 119 120 121 122

	// for each of the path components
	for _, name := range names {

		var nlink *merkledag.Link
		// for each of the links in nd, the current object
		for _, link := range nd.Links {
			if link.Name == name {
				nlink = link
				break
			}
		}

123
		if nlink == nil || len(nlink.Hash) == 0 {
124
			n, _ := nd.Multihash()
125
			return result, ErrNoLink{Name: name, Node: n}
Jeromy's avatar
Jeromy committed
126 127
		}

128 129 130 131 132 133
		if nlink.GetCachedNode() == nil {
			var cancel context.CancelFunc
			ctx, cancel = context.WithTimeout(ctx, time.Minute)
			defer cancel()
		}

134
		var err error
135
		nd, err = nlink.GetNodeAndCache(ctx, s.DAG)
136 137
		if err != nil {
			return append(result, nd), err
Jeromy's avatar
Jeromy committed
138
		}
139

140
		result = append(result, nd)
Jeromy's avatar
Jeromy committed
141
	}
142
	return result, nil
Jeromy's avatar
Jeromy committed
143
}