utils.go 4.87 KB
Newer Older
1 2 3 4
package dagutils

import (
	"errors"
Jeromy's avatar
Jeromy committed
5
	"strings"
6

7 8
	ds "github.com/ipfs/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-datastore"
	syncds "github.com/ipfs/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-datastore/sync"
9 10
	context "github.com/ipfs/go-ipfs/Godeps/_workspace/src/golang.org/x/net/context"

11 12 13
	bstore "github.com/ipfs/go-ipfs/blocks/blockstore"
	bserv "github.com/ipfs/go-ipfs/blockservice"
	offline "github.com/ipfs/go-ipfs/exchange/offline"
14 15 16
	dag "github.com/ipfs/go-ipfs/merkledag"
)

Jeromy's avatar
Jeromy committed
17 18
type Editor struct {
	root *dag.Node
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33

	// tmp is a temporary in memory (for now) dagstore for all of the
	// intermediary nodes to be stored in
	tmp dag.DAGService

	// src is the dagstore with *all* of the data on it, it is used to pull
	// nodes from for modification (nil is a valid value)
	src dag.DAGService
}

func NewMemoryDagService() dag.DAGService {
	// build mem-datastore for editor's intermediary nodes
	bs := bstore.NewBlockstore(syncds.MutexWrap(ds.NewMapDatastore()))
	bsrv := bserv.New(bs, offline.Exchange(bs))
	return dag.NewDAGService(bsrv)
Jeromy's avatar
Jeromy committed
34 35
}

36 37
// root is the node to be modified, source is the dagstore to pull nodes from (optional)
func NewDagEditor(root *dag.Node, source dag.DAGService) *Editor {
Jeromy's avatar
Jeromy committed
38 39
	return &Editor{
		root: root,
40 41
		tmp:  NewMemoryDagService(),
		src:  source,
Jeromy's avatar
Jeromy committed
42 43 44 45 46 47 48
	}
}

func (e *Editor) GetNode() *dag.Node {
	return e.root.Copy()
}

49
func (e *Editor) GetDagService() dag.DAGService {
50
	return e.tmp
Jeromy's avatar
Jeromy committed
51 52
}

53
func addLink(ctx context.Context, ds dag.DAGService, root *dag.Node, childname string, childnd *dag.Node) (*dag.Node, error) {
54 55 56 57
	if childname == "" {
		return nil, errors.New("cannot create link with no name!")
	}

58 59
	// ensure that the node we are adding is in the dagservice
	_, err := ds.Add(childnd)
60 61 62 63
	if err != nil {
		return nil, err
	}

Jeromy's avatar
Jeromy committed
64 65
	_ = ds.Remove(root)

66 67 68
	// ensure no link with that name already exists
	_ = root.RemoveNodeLink(childname) // ignore error, only option is ErrNotFound

69
	if err := root.AddNodeLinkClean(childname, childnd); err != nil {
70 71 72
		return nil, err
	}

73
	if _, err := ds.Add(root); err != nil {
74 75 76 77 78
		return nil, err
	}
	return root, nil
}

79
func (e *Editor) InsertNodeAtPath(ctx context.Context, path string, toinsert *dag.Node, create func() *dag.Node) error {
Jeromy's avatar
Jeromy committed
80
	splpath := strings.Split(path, "/")
81
	nd, err := e.insertNodeAtPath(ctx, e.root, splpath, toinsert, create)
Jeromy's avatar
Jeromy committed
82 83 84 85 86 87 88
	if err != nil {
		return err
	}
	e.root = nd
	return nil
}

89
func (e *Editor) insertNodeAtPath(ctx context.Context, root *dag.Node, path []string, toinsert *dag.Node, create func() *dag.Node) (*dag.Node, error) {
90
	if len(path) == 1 {
91
		return addLink(ctx, e.tmp, root, path[0], toinsert)
92 93
	}

94
	nd, err := root.GetLinkedNode(ctx, e.tmp, path[0])
95 96
	if err != nil {
		// if 'create' is true, we create directories on the way down as needed
97
		if err == dag.ErrLinkNotFound && create != nil {
Jeromy's avatar
Jeromy committed
98
			nd = create()
99 100 101 102 103 104
			err = nil // no longer an error case
		} else if err == dag.ErrNotFound {
			nd, err = root.GetLinkedNode(ctx, e.src, path[0])
		}

		if err != nil {
105 106 107 108
			return nil, err
		}
	}

109
	ndprime, err := e.insertNodeAtPath(ctx, nd, path[1:], toinsert, create)
110 111 112 113
	if err != nil {
		return nil, err
	}

114
	_ = e.tmp.Remove(root)
Jeromy's avatar
Jeromy committed
115

116 117 118 119 120 121
	_ = root.RemoveNodeLink(path[0])
	err = root.AddNodeLinkClean(path[0], ndprime)
	if err != nil {
		return nil, err
	}

122
	_, err = e.tmp.Add(root)
123 124 125 126 127 128 129
	if err != nil {
		return nil, err
	}

	return root, nil
}

Jeromy's avatar
Jeromy committed
130 131
func (e *Editor) RmLink(ctx context.Context, path string) error {
	splpath := strings.Split(path, "/")
132
	nd, err := e.rmLink(ctx, e.root, splpath)
Jeromy's avatar
Jeromy committed
133 134 135 136 137 138 139
	if err != nil {
		return err
	}
	e.root = nd
	return nil
}

140
func (e *Editor) rmLink(ctx context.Context, root *dag.Node, path []string) (*dag.Node, error) {
141 142 143 144 145 146 147
	if len(path) == 1 {
		// base case, remove node in question
		err := root.RemoveNodeLink(path[0])
		if err != nil {
			return nil, err
		}

148
		_, err = e.tmp.Add(root)
149 150 151 152 153 154 155
		if err != nil {
			return nil, err
		}

		return root, nil
	}

156 157 158 159 160
	nd, err := root.GetLinkedNode(ctx, e.tmp, path[0])
	if err == dag.ErrNotFound {
		nd, err = root.GetLinkedNode(ctx, e.src, path[0])
	}

161 162 163 164
	if err != nil {
		return nil, err
	}

165
	nnode, err := e.rmLink(ctx, nd, path[1:])
166 167 168 169
	if err != nil {
		return nil, err
	}

170
	_ = e.tmp.Remove(root)
Jeromy's avatar
Jeromy committed
171

172 173 174 175 176 177
	_ = root.RemoveNodeLink(path[0])
	err = root.AddNodeLinkClean(path[0], nnode)
	if err != nil {
		return nil, err
	}

178
	_, err = e.tmp.Add(root)
179 180 181 182 183 184
	if err != nil {
		return nil, err
	}

	return root, nil
}
185

186 187 188 189
func (e *Editor) Finalize(ds dag.DAGService) (*dag.Node, error) {
	nd := e.GetNode()
	err := copyDag(nd, e.tmp, ds)
	return nd, err
190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
}

func copyDag(nd *dag.Node, from, to dag.DAGService) error {
	_, err := to.Add(nd)
	if err != nil {
		return err
	}

	for _, lnk := range nd.Links {
		child, err := lnk.GetNode(context.Background(), from)
		if err != nil {
			if err == dag.ErrNotFound {
				// not found means we didnt modify it, and it should
				// already be in the target datastore
				continue
			}
			return err
		}

		err = copyDag(child, from, to)
		if err != nil {
			return err
		}
	}
	return nil
}