mirror of
https://github.com/ipfs/kubo.git
synced 2026-02-23 03:17:43 +08:00
195 lines
3.6 KiB
Go
195 lines
3.6 KiB
Go
package dagutils
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"path"
|
|
|
|
dag "github.com/ipfs/go-ipfs/merkledag"
|
|
|
|
cid "gx/ipfs/QmNp85zy9RLrQ5oQD4hPyS39ezrrXpcaa7R4Y9kxdWQLLQ/go-cid"
|
|
node "gx/ipfs/QmPN7cwmpcc4DWXb4KTB9dNAJgjuPY69h3npsMfhRrQL9c/go-ipld-format"
|
|
)
|
|
|
|
const (
|
|
Add = iota
|
|
Remove
|
|
Mod
|
|
)
|
|
|
|
type Change struct {
|
|
Type int
|
|
Path string
|
|
Before *cid.Cid
|
|
After *cid.Cid
|
|
}
|
|
|
|
func (c *Change) String() string {
|
|
switch c.Type {
|
|
case Add:
|
|
return fmt.Sprintf("Added %s at %s", c.After.String(), c.Path)
|
|
case Remove:
|
|
return fmt.Sprintf("Removed %s from %s", c.Before.String(), c.Path)
|
|
case Mod:
|
|
return fmt.Sprintf("Changed %s to %s at %s", c.Before.String(), c.After.String(), c.Path)
|
|
default:
|
|
panic("nope")
|
|
}
|
|
}
|
|
|
|
func ApplyChange(ctx context.Context, ds dag.DAGService, nd *dag.ProtoNode, cs []*Change) (*dag.ProtoNode, error) {
|
|
e := NewDagEditor(nd, ds)
|
|
for _, c := range cs {
|
|
switch c.Type {
|
|
case Add:
|
|
child, err := ds.Get(ctx, c.After)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
childpb, ok := child.(*dag.ProtoNode)
|
|
if !ok {
|
|
return nil, dag.ErrNotProtobuf
|
|
}
|
|
|
|
err = e.InsertNodeAtPath(ctx, c.Path, childpb, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
case Remove:
|
|
err := e.RmLink(ctx, c.Path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
case Mod:
|
|
err := e.RmLink(ctx, c.Path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
child, err := ds.Get(ctx, c.After)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
childpb, ok := child.(*dag.ProtoNode)
|
|
if !ok {
|
|
return nil, dag.ErrNotProtobuf
|
|
}
|
|
|
|
err = e.InsertNodeAtPath(ctx, c.Path, childpb, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
|
|
return e.Finalize(ds)
|
|
}
|
|
|
|
// Diff returns a set of changes that transform node 'a' into node 'b'
|
|
func Diff(ctx context.Context, ds dag.DAGService, a, b node.Node) ([]*Change, error) {
|
|
if len(a.Links()) == 0 && len(b.Links()) == 0 {
|
|
return []*Change{
|
|
&Change{
|
|
Type: Mod,
|
|
Before: a.Cid(),
|
|
After: b.Cid(),
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
var out []*Change
|
|
clean_a := a.Copy().(*dag.ProtoNode)
|
|
clean_b := b.Copy().(*dag.ProtoNode)
|
|
|
|
// strip out unchanged stuff
|
|
for _, lnk := range a.Links() {
|
|
l, _, err := b.ResolveLink([]string{lnk.Name})
|
|
if err == nil {
|
|
if l.Cid.Equals(lnk.Cid) {
|
|
// no change... ignore it
|
|
} else {
|
|
anode, err := lnk.GetNode(ctx, ds)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
bnode, err := l.GetNode(ctx, ds)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
anodepb, ok := anode.(*dag.ProtoNode)
|
|
if !ok {
|
|
return nil, dag.ErrNotProtobuf
|
|
}
|
|
|
|
bnodepb, ok := bnode.(*dag.ProtoNode)
|
|
if !ok {
|
|
return nil, dag.ErrNotProtobuf
|
|
}
|
|
|
|
sub, err := Diff(ctx, ds, anodepb, bnodepb)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, subc := range sub {
|
|
subc.Path = path.Join(lnk.Name, subc.Path)
|
|
out = append(out, subc)
|
|
}
|
|
}
|
|
clean_a.RemoveNodeLink(l.Name)
|
|
clean_b.RemoveNodeLink(l.Name)
|
|
}
|
|
}
|
|
|
|
for _, lnk := range clean_a.Links() {
|
|
out = append(out, &Change{
|
|
Type: Remove,
|
|
Path: lnk.Name,
|
|
Before: lnk.Cid,
|
|
})
|
|
}
|
|
for _, lnk := range clean_b.Links() {
|
|
out = append(out, &Change{
|
|
Type: Add,
|
|
Path: lnk.Name,
|
|
After: lnk.Cid,
|
|
})
|
|
}
|
|
|
|
return out, nil
|
|
}
|
|
|
|
type Conflict struct {
|
|
A *Change
|
|
B *Change
|
|
}
|
|
|
|
func MergeDiffs(a, b []*Change) ([]*Change, []Conflict) {
|
|
var out []*Change
|
|
var conflicts []Conflict
|
|
paths := make(map[string]*Change)
|
|
for _, c := range a {
|
|
paths[c.Path] = c
|
|
}
|
|
|
|
for _, c := range b {
|
|
if ca, ok := paths[c.Path]; ok {
|
|
conflicts = append(conflicts, Conflict{
|
|
A: ca,
|
|
B: c,
|
|
})
|
|
} else {
|
|
out = append(out, c)
|
|
}
|
|
}
|
|
for _, c := range paths {
|
|
out = append(out, c)
|
|
}
|
|
return out, conflicts
|
|
}
|