blob: 659a586a414c22669dc77ebfa97127e169ce7874 [file] [log] [blame]
Jiri Simsa5293dcb2014-05-10 09:56:38 -07001package vsync
2
3// Tests for the Veyron Sync DAG component.
4
5import (
6 "errors"
7 "fmt"
Jiri Simsa5293dcb2014-05-10 09:56:38 -07008 "os"
9 "reflect"
10 "testing"
11 "time"
12
Jiri Simsa870ddd62014-06-27 14:56:58 -070013 "veyron/lib/testutil"
14
Jiri Simsa5293dcb2014-05-10 09:56:38 -070015 "veyron2/storage"
16)
17
18// dagFilename generates a filename for a temporary (per unit test) DAG file.
19// Do not replace this function with TempFile because TempFile creates the new
20// file and the tests must verify that the DAG can create a non-existing file.
21func dagFilename() string {
22 return fmt.Sprintf("%s/sync_dag_test_%d_%d", os.TempDir(), os.Getpid(), time.Now().UnixNano())
23}
24
25// fileSize returns the size of a file.
26func fileSize(fname string) int64 {
27 finfo, err := os.Stat(fname)
28 if err != nil {
29 return -1
30 }
31 return finfo.Size()
32}
33
34// TestDAGOpen tests the creation of a DAG, closing and re-opening it. It also
35// verifies that its backing file is created and that a 2nd close is safe.
36func TestDAGOpen(t *testing.T) {
37 dagfile := dagFilename()
38 defer os.Remove(dagfile)
39
40 dag, err := openDAG(dagfile)
41 if err != nil {
42 t.Fatalf("Cannot open new DAG file %s", dagfile)
43 }
44
45 fsize := fileSize(dagfile)
46 if fsize < 0 {
47 t.Fatalf("DAG file %s not created", dagfile)
48 }
49
50 dag.flush()
51 oldfsize := fsize
52 fsize = fileSize(dagfile)
53 if fsize <= oldfsize {
54 t.Fatalf("DAG file %s not flushed", dagfile)
55 }
56
57 dag.close()
58
59 dag, err = openDAG(dagfile)
60 if err != nil {
61 t.Fatalf("Cannot re-open existing DAG file %s", dagfile)
62 }
63
64 oldfsize = fsize
65 fsize = fileSize(dagfile)
66 if fsize != oldfsize {
67 t.Fatalf("DAG file %s size changed across re-open", dagfile)
68 }
69
70 dag.close()
71 dag.close() // multiple closes should be a safe NOP
72
73 fsize = fileSize(dagfile)
74 if fsize != oldfsize {
75 t.Fatalf("DAG file %s size changed across close", dagfile)
76 }
77
78 // Fail opening a DAG in a non-existent directory.
79 _, err = openDAG("/not/really/there/junk.dag")
80 if err == nil {
81 t.Fatalf("openDAG() did not fail when using a bad pathname")
82 }
83}
84
85// TestInvalidDAG tests using DAG methods on an invalid (closed) DAG.
86func TestInvalidDAG(t *testing.T) {
87 dagfile := dagFilename()
88 defer os.Remove(dagfile)
89
90 dag, err := openDAG(dagfile)
91 if err != nil {
92 t.Fatalf("Cannot open new DAG file %s", dagfile)
93 }
94
95 dag.close()
96
97 oid, err := strToObjID("6789")
98 if err != nil {
99 t.Error(err)
100 }
101
102 err = dag.addNode(oid, 4, false, []storage.Version{2, 3}, "foobar")
103 if err == nil || err.Error() != "invalid DAG" {
104 t.Errorf("addNode() did not fail on a closed DAG: %v", err)
105 }
106
107 err = dag.moveHead(oid, 4)
108 if err == nil || err.Error() != "invalid DAG" {
109 t.Errorf("moveHead() did not fail on a closed DAG: %v", err)
110 }
111
112 _, _, _, _, err = dag.hasConflict(oid)
113 if err == nil || err.Error() != "invalid DAG" {
114 t.Errorf("hasConflict() did not fail on a closed DAG: %v", err)
115 }
116
117 _, err = dag.getLogrec(oid, 4)
118 if err == nil || err.Error() != "invalid DAG" {
119 t.Errorf("getLogrec() did not fail on a closed DAG: %v", err)
120 }
121
122 err = dag.prune(oid, 4, func(lr string) error {
123 return nil
124 })
125 if err == nil || err.Error() != "invalid DAG" {
126 t.Errorf("prune() did not fail on a closed DAG: %v", err)
127 }
128
129 node := &dagNode{Level: 15, Parents: []storage.Version{444, 555}, Logrec: "logrec-23"}
130 err = dag.setNode(oid, 4, node)
131 if err == nil || err.Error() != "invalid DAG" {
132 t.Errorf("setNode() did not fail on a closed DAG: %v", err)
133 }
134
135 _, err = dag.getNode(oid, 4)
136 if err == nil || err.Error() != "invalid DAG" {
137 t.Errorf("getNode() did not fail on a closed DAG: %v", err)
138 }
139
140 err = dag.delNode(oid, 4)
141 if err == nil || err.Error() != "invalid DAG" {
142 t.Errorf("delNode() did not fail on a closed DAG: %v", err)
143 }
144
Raja Daoudf123da82014-06-12 12:03:21 -0700145 err = dag.addParent(oid, 4, 2, true)
146 if err == nil || err.Error() != "invalid DAG" {
147 t.Errorf("addParent() did not fail on a closed DAG: %v", err)
148 }
149
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700150 err = dag.setHead(oid, 4)
151 if err == nil || err.Error() != "invalid DAG" {
152 t.Errorf("setHead() did not fail on a closed DAG: %v", err)
153 }
154
155 _, err = dag.getHead(oid)
156 if err == nil || err.Error() != "invalid DAG" {
157 t.Errorf("getHead() did not fail on a closed DAG: %v", err)
158 }
159
160 err = dag.compact()
161 if err == nil || err.Error() != "invalid DAG" {
162 t.Errorf("compact() did not fail on a closed DAG: %v", err)
163 }
164
165 // These calls should be harmless NOPs.
166 dag.clearGraft()
167 dag.flush()
168 dag.close()
169 if dag.hasNode(oid, 4) {
170 t.Errorf("hasNode() found an object on a closed DAG")
171 }
172 if pmap := dag.getParentMap(oid); len(pmap) != 0 {
173 t.Errorf("getParentMap() found data on a closed DAG: %v", pmap)
174 }
Raja Daoudf123da82014-06-12 12:03:21 -0700175 if hmap, gmap := dag.getGraftNodes(oid); hmap != nil || gmap != nil {
176 t.Errorf("getGraftNodes() found data on a closed DAG: head map: %v, graft map: %v", hmap, gmap)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700177 }
178}
179
180// TestSetNode tests setting and getting a DAG node across DAG open/close/reopen.
181func TestSetNode(t *testing.T) {
182 dagfile := dagFilename()
183 defer os.Remove(dagfile)
184
185 dag, err := openDAG(dagfile)
186 if err != nil {
187 t.Fatalf("Cannot open new DAG file %s", dagfile)
188 }
189
190 version := storage.Version(0)
191 oid, err := strToObjID("111")
192 if err != nil {
193 t.Fatal(err)
194 }
195
196 node, err := dag.getNode(oid, version)
197 if err == nil || node != nil {
198 t.Errorf("Found non-existent object %d:%d in DAG file %s: %v", oid, version, dagfile, node)
199 }
200
201 if dag.hasNode(oid, version) {
202 t.Errorf("hasNode() found non-existent object %d:%d in DAG file %s", oid, version, dagfile)
203 }
204
205 if logrec, err := dag.getLogrec(oid, version); err == nil || logrec != "" {
206 t.Errorf("Non-existent object %d:%d has a logrec in DAG file %s: %v", oid, version, dagfile, logrec)
207 }
208
209 node = &dagNode{Level: 15, Parents: []storage.Version{444, 555}, Logrec: "logrec-23"}
210 if err = dag.setNode(oid, version, node); err != nil {
211 t.Fatalf("Cannot set object %d:%d (%v) in DAG file %s", oid, version, node, dagfile)
212 }
213
214 for i := 0; i < 2; i++ {
215 node2, err := dag.getNode(oid, version)
216 if err != nil || node2 == nil {
217 t.Errorf("Cannot find stored object %d:%d (i=%d) in DAG file %s", oid, version, i, dagfile)
218 }
219
220 if !dag.hasNode(oid, version) {
221 t.Errorf("hasNode() did not find object %d:%d (i=%d) in DAG file %s", oid, version, i, dagfile)
222 }
223
224 if !reflect.DeepEqual(node, node2) {
225 t.Errorf("Object %d:%d has wrong data (i=%d) in DAG file %s: %v instead of %v",
226 oid, version, i, dagfile, node2, node)
227 }
228
229 if logrec, err := dag.getLogrec(oid, version); err != nil || logrec != "logrec-23" {
230 t.Errorf("Object %d:%d has wrong logrec (i=%d) in DAG file %s: %v",
231 oid, version, i, dagfile, logrec)
232 }
233
234 if i == 0 {
235 dag.flush()
236 dag.close()
237 dag, err = openDAG(dagfile)
238 if err != nil {
239 t.Fatalf("Cannot re-open DAG file %s", dagfile)
240 }
241 }
242 }
243
244 dag.close()
245}
246
247// TestDelNode tests deleting a DAG node across DAG open/close/reopen.
248func TestDelNode(t *testing.T) {
249 dagfile := dagFilename()
250 defer os.Remove(dagfile)
251
252 dag, err := openDAG(dagfile)
253 if err != nil {
254 t.Fatalf("Cannot open new DAG file %s", dagfile)
255 }
256
257 version := storage.Version(1)
258 oid, err := strToObjID("222")
259 if err != nil {
260 t.Fatal(err)
261 }
262
263 node := &dagNode{Level: 123, Parents: []storage.Version{333}, Logrec: "logrec-789"}
264 if err = dag.setNode(oid, version, node); err != nil {
265 t.Fatalf("Cannot set object %d:%d (%v) in DAG file %s", oid, version, node, dagfile)
266 }
267
268 dag.flush()
269
270 err = dag.delNode(oid, version)
271 if err != nil {
272 t.Fatalf("Cannot delete object %d:%d in DAG file %s", oid, version, dagfile)
273 }
274
275 dag.flush()
276
277 for i := 0; i < 2; i++ {
278 node2, err := dag.getNode(oid, version)
279 if err == nil || node2 != nil {
280 t.Errorf("Found deleted object %d:%d (%v) (i=%d) in DAG file %s", oid, version, node2, i, dagfile)
281 }
282
283 if dag.hasNode(oid, version) {
284 t.Errorf("hasNode() found deleted object %d:%d (i=%d) in DAG file %s", oid, version, i, dagfile)
285 }
286
287 if logrec, err := dag.getLogrec(oid, version); err == nil || logrec != "" {
288 t.Errorf("Deleted object %d:%d (i=%d) has logrec in DAG file %s: %v", oid, version, i, dagfile, logrec)
289 }
290
291 if i == 0 {
292 dag.close()
293 dag, err = openDAG(dagfile)
294 if err != nil {
295 t.Fatalf("Cannot re-open DAG file %s", dagfile)
296 }
297 }
298 }
299
300 dag.close()
301}
302
Raja Daoudf123da82014-06-12 12:03:21 -0700303// TestAddParent tests adding parents to a DAG node.
304func TestAddParent(t *testing.T) {
305 dagfile := dagFilename()
306 defer os.Remove(dagfile)
307
308 dag, err := openDAG(dagfile)
309 if err != nil {
310 t.Fatalf("Cannot open new DAG file %s", dagfile)
311 }
312
313 version := storage.Version(7)
314 oid, err := strToObjID("789")
315 if err != nil {
316 t.Fatal(err)
317 }
318
319 if err = dag.addParent(oid, version, 1, true); err == nil {
320 t.Errorf("addParent() did not fail for an unknown object %d:%d in DAG file %s", oid, version, dagfile)
321 }
322
323 node := &dagNode{Level: 15, Logrec: "logrec-22"}
324 if err = dag.setNode(oid, version, node); err != nil {
325 t.Fatalf("Cannot set object %d:%d (%v) in DAG file %s", oid, version, node, dagfile)
326 }
327
328 for _, parent := range []storage.Version{1, 2, 3} {
329 if err = dag.addParent(oid, version, parent, true); err == nil {
330 t.Errorf("addParent() did not reject invalid parent %d for object %d:%d in DAG file %s",
331 parent, oid, version, dagfile)
332 }
333
334 pnode := &dagNode{Level: 11, Logrec: fmt.Sprint("logrec-%d", parent)}
335 if err = dag.setNode(oid, parent, pnode); err != nil {
336 t.Fatalf("Cannot set parent object %d:%d (%v) in DAG file %s", oid, parent, pnode, dagfile)
337 }
338
339 remote := parent%2 == 0
340 for i := 0; i < 2; i++ {
341 if err = dag.addParent(oid, version, parent, remote); err != nil {
342 t.Errorf("addParent() failed on parent %d, remote %d (i=%d) for object %d:%d in DAG file %s: %v",
343 parent, remote, i, oid, version, dagfile, err)
344 }
345 }
346 }
347
348 node2, err := dag.getNode(oid, version)
349 if err != nil || node2 == nil {
350 t.Errorf("Cannot find stored object %d:%d in DAG file %s", oid, version, dagfile)
351 }
352
353 expParents := []storage.Version{1, 2, 3}
354 if !reflect.DeepEqual(node2.Parents, expParents) {
355 t.Errorf("invalid parents for object %d:%d in DAG file %s: %v instead of %v",
356 oid, version, dagfile, node2.Parents, expParents)
357 }
358
359 dag.close()
360}
361
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700362// TestSetHead tests setting and getting a DAG head node across DAG open/close/reopen.
363func TestSetHead(t *testing.T) {
364 dagfile := dagFilename()
365 defer os.Remove(dagfile)
366
367 dag, err := openDAG(dagfile)
368 if err != nil {
369 t.Fatalf("Cannot open new DAG file %s", dagfile)
370 }
371
372 oid, err := strToObjID("333")
373 if err != nil {
374 t.Fatal(err)
375 }
376
377 version, err := dag.getHead(oid)
378 if err == nil {
379 t.Errorf("Found non-existent object head %d in DAG file %s: %d", oid, dagfile, version)
380 }
381
382 version = 555
383 if err = dag.setHead(oid, version); err != nil {
384 t.Fatalf("Cannot set object head %d (%d) in DAG file %s", oid, version, dagfile)
385 }
386
387 dag.flush()
388
389 for i := 0; i < 3; i++ {
390 version2, err := dag.getHead(oid)
391 if err != nil {
392 t.Errorf("Cannot find stored object head %d (i=%d) in DAG file %s", oid, i, dagfile)
393 }
394 if version != version2 {
395 t.Errorf("Object %d has wrong head data (i=%d) in DAG file %s: %d instead of %d",
396 oid, i, dagfile, version2, version)
397 }
398
399 if i == 0 {
400 dag.close()
401 dag, err = openDAG(dagfile)
402 if err != nil {
403 t.Fatalf("Cannot re-open DAG file %s", dagfile)
404 }
405 } else if i == 1 {
406 version = 888
407 if err = dag.setHead(oid, version); err != nil {
408 t.Fatalf("Cannot set new object head %d (%d) in DAG file %s", oid, version, dagfile)
409 }
410 dag.flush()
411 }
412 }
413
414 dag.close()
415}
416
417// checkEndOfSync simulates and check the end-of-sync operations: clear the
418// node grafting metadata and verify that it is empty and that HasConflict()
419// detects this case and fails, then close the DAG.
420func checkEndOfSync(d *dag, oid storage.ID) error {
421 // Clear grafting info; this happens at the end of a sync log replay.
422 d.clearGraft()
423
424 // There should be no grafting info, and hasConflict() should fail.
Raja Daoudf123da82014-06-12 12:03:21 -0700425 newHeads, grafts := d.getGraftNodes(oid)
426 if newHeads != nil || grafts != nil {
427 return fmt.Errorf("Object %d: graft info not cleared: newHeads (%v), grafts (%v)", oid, newHeads, grafts)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700428 }
429
430 isConflict, newHead, oldHead, ancestor, errConflict := d.hasConflict(oid)
431 if errConflict == nil {
432 return fmt.Errorf("Object %d: conflict did not fail: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
433 oid, isConflict, newHead, oldHead, ancestor, errConflict)
434 }
435
436 d.close()
437 return nil
438}
439
440// TestLocalUpdates tests the sync handling of initial local updates: an object
441// is created (v0) and updated twice (v1, v2) on this device. The DAG should
442// show: v0 -> v1 -> v2 and the head should point to v2.
443func TestLocalUpdates(t *testing.T) {
444 dagfile := dagFilename()
445 defer os.Remove(dagfile)
446
447 dag, err := openDAG(dagfile)
448 if err != nil {
449 t.Fatalf("Cannot open new DAG file %s", dagfile)
450 }
451
452 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
453 t.Fatal(err)
454 }
455
456 // The head must have moved to "v2" and the parent map shows the updated DAG.
457 oid, err := strToObjID("12345")
458 if err != nil {
459 t.Fatal(err)
460 }
461
462 if head, e := dag.getHead(oid); e != nil || head != 2 {
463 t.Errorf("Invalid object %d head in DAG file %s: %d", oid, dagfile, head)
464 }
465
466 pmap := dag.getParentMap(oid)
467
468 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}}
469
470 if !reflect.DeepEqual(pmap, exp) {
471 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
472 }
473
474 // Make sure an existing node cannot be added again.
475 if err = dag.addNode(oid, 1, false, []storage.Version{0, 2}, "foobar"); err == nil {
476 t.Errorf("addNode() did not fail when given an existing node")
477 }
478
479 // Make sure a new node cannot have more than 2 parents.
480 if err = dag.addNode(oid, 3, false, []storage.Version{0, 1, 2}, "foobar"); err == nil {
481 t.Errorf("addNode() did not fail when given 3 parents")
482 }
483
484 // Make sure a new node cannot have an invalid parent.
485 if err = dag.addNode(oid, 3, false, []storage.Version{0, 555}, "foobar"); err == nil {
486 t.Errorf("addNode() did not fail when using an invalid parent")
487 }
488
489 // Make sure a new root node (no parents) cannot be added once a root exists.
490 // For the parents array, check both the "nil" and the empty array as input.
491 if err = dag.addNode(oid, 6789, false, nil, "foobar"); err == nil {
492 t.Errorf("Adding a 2nd root node (nil parents) for object %d in DAG file %s did not fail", oid, dagfile)
493 }
494 if err = dag.addNode(oid, 6789, false, []storage.Version{}, "foobar"); err == nil {
495 t.Errorf("Adding a 2nd root node (empty parents) for object %d in DAG file %s did not fail", oid, dagfile)
496 }
497
498 if err := checkEndOfSync(dag, oid); err != nil {
499 t.Fatal(err)
500 }
501}
502
503// TestRemoteUpdates tests the sync handling of initial remote updates:
504// an object is created (v0) and updated twice (v1, v2) on another device and
505// we learn about it during sync. The updated DAG should show: v0 -> v1 -> v2
506// and report no conflicts with the new head pointing at v2.
507func TestRemoteUpdates(t *testing.T) {
508 dagfile := dagFilename()
509 defer os.Remove(dagfile)
510
511 dag, err := openDAG(dagfile)
512 if err != nil {
513 t.Fatalf("Cannot open new DAG file %s", dagfile)
514 }
515
516 if err = dagReplayCommands(dag, "remote-init-00.sync"); err != nil {
517 t.Fatal(err)
518 }
519
520 // The head must not have moved (i.e. still undefined) and the parent
521 // map shows the newly grafted DAG fragment.
522 oid, err := strToObjID("12345")
523 if err != nil {
524 t.Fatal(err)
525 }
526
527 if head, e := dag.getHead(oid); e == nil {
528 t.Errorf("Object %d head found in DAG file %s: %d", oid, dagfile, head)
529 }
530
531 pmap := dag.getParentMap(oid)
532
533 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}}
534
535 if !reflect.DeepEqual(pmap, exp) {
536 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
537 }
538
539 // Verify the grafting of remote nodes.
Raja Daoudf123da82014-06-12 12:03:21 -0700540 newHeads, grafts := dag.getGraftNodes(oid)
541
542 expNewHeads := map[storage.Version]struct{}{2: struct{}{}}
543 if !reflect.DeepEqual(newHeads, expNewHeads) {
544 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700545 }
546
547 expgrafts := map[storage.Version]uint64{}
548 if !reflect.DeepEqual(grafts, expgrafts) {
549 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
550 }
551
552 // There should be no conflict.
553 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
554 if !(!isConflict && newHead == 2 && oldHead == 0 && ancestor == 0 && errConflict == nil) {
555 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
556 oid, isConflict, newHead, oldHead, ancestor, errConflict)
557 }
558
559 if logrec, e := dag.getLogrec(oid, newHead); e != nil || logrec != "logrec-02" {
560 t.Errorf("Invalid logrec for newhead object %d:%d in DAG file %s: %v", oid, newHead, dagfile, logrec)
561 }
562
563 // Make sure an unknown node cannot become the new head.
564 if err = dag.moveHead(oid, 55); err == nil {
565 t.Errorf("moveHead() did not fail on an invalid node")
566 }
567
568 // Then we can move the head and clear the grafting data.
569 if err = dag.moveHead(oid, newHead); err != nil {
570 t.Errorf("Object %d cannot move head to %d in DAG file %s: %v", oid, newHead, dagfile, err)
571 }
572
573 if err := checkEndOfSync(dag, oid); err != nil {
574 t.Fatal(err)
575 }
576}
577
578// TestRemoteNoConflict tests sync of remote updates on top of a local initial
579// state without conflict. An object is created locally and updated twice
580// (v0 -> v1 -> v2). Another device, having gotten this info, makes 3 updates
581// on top of that (v2 -> v3 -> v4 -> v5) and sends this info in a later sync.
582// The updated DAG should show (v0 -> v1 -> v2 -> v3 -> v4 -> v5) and report
583// no conflicts with the new head pointing at v5. It should also report v2 as
584// the graft point on which the new fragment (v3 -> v4 -> v5) gets attached.
585func TestRemoteNoConflict(t *testing.T) {
586 dagfile := dagFilename()
587 defer os.Remove(dagfile)
588
589 dag, err := openDAG(dagfile)
590 if err != nil {
591 t.Fatalf("Cannot open new DAG file %s", dagfile)
592 }
593
594 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
595 t.Fatal(err)
596 }
597 if err = dagReplayCommands(dag, "remote-noconf-00.sync"); err != nil {
598 t.Fatal(err)
599 }
600
601 // The head must not have moved (i.e. still at v2) and the parent map
602 // shows the newly grafted DAG fragment on top of the prior DAG.
603 oid, err := strToObjID("12345")
604 if err != nil {
605 t.Fatal(err)
606 }
607
608 if head, e := dag.getHead(oid); e != nil || head != 2 {
609 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
610 }
611
612 pmap := dag.getParentMap(oid)
613
614 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}, 3: {2}, 4: {3}, 5: {4}}
615
616 if !reflect.DeepEqual(pmap, exp) {
617 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
618 }
619
620 // Verify the grafting of remote nodes.
Raja Daoudf123da82014-06-12 12:03:21 -0700621 newHeads, grafts := dag.getGraftNodes(oid)
622
623 expNewHeads := map[storage.Version]struct{}{5: struct{}{}}
624 if !reflect.DeepEqual(newHeads, expNewHeads) {
625 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700626 }
627
628 expgrafts := map[storage.Version]uint64{2: 2}
629 if !reflect.DeepEqual(grafts, expgrafts) {
630 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
631 }
632
633 // There should be no conflict.
634 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
635 if !(!isConflict && newHead == 5 && oldHead == 2 && ancestor == 0 && errConflict == nil) {
636 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
637 oid, isConflict, newHead, oldHead, ancestor, errConflict)
638 }
639
640 if logrec, e := dag.getLogrec(oid, oldHead); e != nil || logrec != "logrec-02" {
641 t.Errorf("Invalid logrec for oldhead object %d:%d in DAG file %s: %v", oid, oldHead, dagfile, logrec)
642 }
643 if logrec, e := dag.getLogrec(oid, newHead); e != nil || logrec != "logrec-05" {
644 t.Errorf("Invalid logrec for newhead object %d:%d in DAG file %s: %v", oid, newHead, dagfile, logrec)
645 }
646
647 // Then we can move the head and clear the grafting data.
648 if err = dag.moveHead(oid, newHead); err != nil {
649 t.Errorf("Object %d cannot move head to %d in DAG file %s: %v", oid, newHead, dagfile, err)
650 }
651
652 // Clear the grafting data and verify that hasConflict() fails without it.
653 dag.clearGraft()
654 isConflict, newHead, oldHead, ancestor, errConflict = dag.hasConflict(oid)
655 if errConflict == nil {
656 t.Errorf("hasConflict() did not fail w/o graft info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
657 oid, isConflict, newHead, oldHead, ancestor, errConflict)
658 }
659
660 if err := checkEndOfSync(dag, oid); err != nil {
661 t.Fatal(err)
662 }
663}
664
665// TestRemoteConflict tests sync handling remote updates that build on the
666// local initial state and trigger a conflict. An object is created locally
667// and updated twice (v0 -> v1 -> v2). Another device, having only gotten
668// the v0 -> v1 history, makes 3 updates on top of v1 (v1 -> v3 -> v4 -> v5)
669// and sends this info during a later sync. Separately, the local device
670// makes a conflicting (concurrent) update v1 -> v2. The updated DAG should
671// show the branches: (v0 -> v1 -> v2) and (v0 -> v1 -> v3 -> v4 -> v5) and
672// report the conflict between v2 and v5 (current and new heads). It should
673// also report v1 as the graft point and the common ancestor in the conflict.
674// The conflict is resolved locally by creating v6 that is derived from both
675// v2 and v5 and it becomes the new head.
676func TestRemoteConflict(t *testing.T) {
677 dagfile := dagFilename()
678 defer os.Remove(dagfile)
679
680 dag, err := openDAG(dagfile)
681 if err != nil {
682 t.Fatalf("Cannot open new DAG file %s", dagfile)
683 }
684
685 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
686 t.Fatal(err)
687 }
688 if err = dagReplayCommands(dag, "remote-conf-00.sync"); err != nil {
689 t.Fatal(err)
690 }
691
692 // The head must not have moved (i.e. still at v2) and the parent map
693 // shows the newly grafted DAG fragment on top of the prior DAG.
694 oid, err := strToObjID("12345")
695 if err != nil {
696 t.Fatal(err)
697 }
698
699 if head, e := dag.getHead(oid); e != nil || head != 2 {
700 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
701 }
702
703 pmap := dag.getParentMap(oid)
704
705 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}, 3: {1}, 4: {3}, 5: {4}}
706
707 if !reflect.DeepEqual(pmap, exp) {
708 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
709 }
710
711 // Verify the grafting of remote nodes.
Raja Daoudf123da82014-06-12 12:03:21 -0700712 newHeads, grafts := dag.getGraftNodes(oid)
713
714 expNewHeads := map[storage.Version]struct{}{2: struct{}{}, 5: struct{}{}}
715 if !reflect.DeepEqual(newHeads, expNewHeads) {
716 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700717 }
718
719 expgrafts := map[storage.Version]uint64{1: 1}
720 if !reflect.DeepEqual(grafts, expgrafts) {
721 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
722 }
723
724 // There should be a conflict between v2 and v5 with v1 as ancestor.
725 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
726 if !(isConflict && newHead == 5 && oldHead == 2 && ancestor == 1 && errConflict == nil) {
727 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
728 oid, isConflict, newHead, oldHead, ancestor, errConflict)
729 }
730
731 if logrec, e := dag.getLogrec(oid, oldHead); e != nil || logrec != "logrec-02" {
732 t.Errorf("Invalid logrec for oldhead object %d:%d in DAG file %s: %v", oid, oldHead, dagfile, logrec)
733 }
734 if logrec, e := dag.getLogrec(oid, newHead); e != nil || logrec != "logrec-05" {
735 t.Errorf("Invalid logrec for newhead object %d:%d in DAG file %s: %v", oid, newHead, dagfile, logrec)
736 }
737 if logrec, e := dag.getLogrec(oid, ancestor); e != nil || logrec != "logrec-01" {
738 t.Errorf("Invalid logrec for ancestor object %d:%d in DAG file %s: %v", oid, ancestor, dagfile, logrec)
739 }
740
741 // Resolve the conflict by adding a new local v6 derived from v2 and v5 (this replay moves the head).
742 if err = dagReplayCommands(dag, "local-resolve-00.sync"); err != nil {
743 t.Fatal(err)
744 }
745
746 // Verify that the head moved to v6 and the parent map shows the resolution.
747 if head, e := dag.getHead(oid); e != nil || head != 6 {
748 t.Errorf("Object %d has wrong head after conflict resolution in DAG file %s: %d", oid, dagfile, head)
749 }
750
751 exp[6] = []storage.Version{2, 5}
752 pmap = dag.getParentMap(oid)
753 if !reflect.DeepEqual(pmap, exp) {
754 t.Errorf("Invalid object %d parent map after conflict resolution in DAG file %s: (%v) instead of (%v)",
755 oid, dagfile, pmap, exp)
756 }
757
758 if err := checkEndOfSync(dag, oid); err != nil {
759 t.Fatal(err)
760 }
761}
762
763// TestRemoteConflictTwoGrafts tests sync handling remote updates that build
764// on the local initial state and trigger a conflict with 2 graft points.
765// An object is created locally and updated twice (v0 -> v1 -> v2). Another
766// device, first learns about v0 and makes it own conflicting update v0 -> v3.
767// That remote device later learns about v1 and resolves the v1/v3 confict by
768// creating v4. Then it makes a last v4 -> v5 update -- which will conflict
769// with v2 but it doesn't know that.
770// Now the sync order is reversed and the local device learns all of what
771// happened on the remote device. The local DAG should get be augmented by
772// a subtree with 2 graft points: v0 and v1. It receives this new branch:
773// v0 -> v3 -> v4 -> v5. Note that v4 is also derived from v1 as a remote
774// conflict resolution. This should report a conflict between v2 and v5
775// (current and new heads), with v0 and v1 as graft points, and v1 as the
776// most-recent common ancestor for that conflict. The conflict is resolved
777// locally by creating v6, derived from both v2 and v5, becoming the new head.
778func TestRemoteConflictTwoGrafts(t *testing.T) {
779 dagfile := dagFilename()
780 defer os.Remove(dagfile)
781
782 dag, err := openDAG(dagfile)
783 if err != nil {
784 t.Fatalf("Cannot open new DAG file %s", dagfile)
785 }
786
787 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
788 t.Fatal(err)
789 }
790 if err = dagReplayCommands(dag, "remote-conf-01.sync"); err != nil {
791 t.Fatal(err)
792 }
793
794 // The head must not have moved (i.e. still at v2) and the parent map
795 // shows the newly grafted DAG fragment on top of the prior DAG.
796 oid, err := strToObjID("12345")
797 if err != nil {
798 t.Fatal(err)
799 }
800
801 if head, e := dag.getHead(oid); e != nil || head != 2 {
802 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
803 }
804
805 pmap := dag.getParentMap(oid)
806
807 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}, 3: {0}, 4: {1, 3}, 5: {4}}
808
809 if !reflect.DeepEqual(pmap, exp) {
810 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
811 }
812
813 // Verify the grafting of remote nodes.
Raja Daoudf123da82014-06-12 12:03:21 -0700814 newHeads, grafts := dag.getGraftNodes(oid)
815
816 expNewHeads := map[storage.Version]struct{}{2: struct{}{}, 5: struct{}{}}
817 if !reflect.DeepEqual(newHeads, expNewHeads) {
818 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
Jiri Simsa5293dcb2014-05-10 09:56:38 -0700819 }
820
821 expgrafts := map[storage.Version]uint64{0: 0, 1: 1}
822 if !reflect.DeepEqual(grafts, expgrafts) {
823 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
824 }
825
826 // There should be a conflict between v2 and v5 with v1 as ancestor.
827 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
828 if !(isConflict && newHead == 5 && oldHead == 2 && ancestor == 1 && errConflict == nil) {
829 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
830 oid, isConflict, newHead, oldHead, ancestor, errConflict)
831 }
832
833 if logrec, e := dag.getLogrec(oid, oldHead); e != nil || logrec != "logrec-02" {
834 t.Errorf("Invalid logrec for oldhead object %d:%d in DAG file %s: %v", oid, oldHead, dagfile, logrec)
835 }
836 if logrec, e := dag.getLogrec(oid, newHead); e != nil || logrec != "logrec-05" {
837 t.Errorf("Invalid logrec for newhead object %d:%d in DAG file %s: %v", oid, newHead, dagfile, logrec)
838 }
839 if logrec, e := dag.getLogrec(oid, ancestor); e != nil || logrec != "logrec-01" {
840 t.Errorf("Invalid logrec for ancestor object %d:%d in DAG file %s: %v", oid, ancestor, dagfile, logrec)
841 }
842
843 // Resolve the conflict by adding a new local v6 derived from v2 and v5 (this replay moves the head).
844 if err = dagReplayCommands(dag, "local-resolve-00.sync"); err != nil {
845 t.Fatal(err)
846 }
847
848 // Verify that the head moved to v6 and the parent map shows the resolution.
849 if head, e := dag.getHead(oid); e != nil || head != 6 {
850 t.Errorf("Object %d has wrong head after conflict resolution in DAG file %s: %d", oid, dagfile, head)
851 }
852
853 exp[6] = []storage.Version{2, 5}
854 pmap = dag.getParentMap(oid)
855 if !reflect.DeepEqual(pmap, exp) {
856 t.Errorf("Invalid object %d parent map after conflict resolution in DAG file %s: (%v) instead of (%v)",
857 oid, dagfile, pmap, exp)
858 }
859
860 if err := checkEndOfSync(dag, oid); err != nil {
861 t.Fatal(err)
862 }
863}
864
865// TestAncestorIterator checks that the iterator goes over the correct set
866// of ancestor nodes for an object given a starting node. It should traverse
867// reconvergent DAG branches only visiting each ancestor once:
868// v0 -> v1 -> v2 -> v4 -> v5 -> v7 -> v8
869// |--> v3 ---| |
870// +--> v6 ---------------+
871// - Starting at v0 it should only cover v0.
872// - Starting at v2 it should only cover v0-v2.
873// - Starting at v5 it should only cover v0-v5.
874// - Starting at v8 it should cover all nodes (v0-v8).
875func TestAncestorIterator(t *testing.T) {
876 dagfile := dagFilename()
877 defer os.Remove(dagfile)
878
879 dag, err := openDAG(dagfile)
880 if err != nil {
881 t.Fatalf("Cannot open new DAG file %s", dagfile)
882 }
883
884 if err = dagReplayCommands(dag, "local-init-01.sync"); err != nil {
885 t.Fatal(err)
886 }
887
888 oid, err := strToObjID("12345")
889 if err != nil {
890 t.Fatal(err)
891 }
892
893 // Loop checking the iteration behavior for different starting nodes.
894 for _, start := range []storage.Version{0, 2, 5, 8} {
895 visitCount := make(map[storage.Version]int)
896 err = dag.ancestorIter(oid, []storage.Version{start},
897 func(oid storage.ID, v storage.Version, node *dagNode) error {
898 visitCount[v]++
899 return nil
900 })
901
902 // Check that all prior nodes are visited only once.
903 for i := storage.Version(0); i < (start + 1); i++ {
904 if visitCount[i] != 1 {
905 t.Errorf("wrong visit count for iter on object %d node %d starting from node %d: %d instead of 1",
906 oid, i, start, visitCount[i])
907 }
908 }
909 }
910
911 // Make sure an error in the callback is returned through the iterator.
912 cbErr := errors.New("callback error")
913 err = dag.ancestorIter(oid, []storage.Version{8}, func(oid storage.ID, v storage.Version, node *dagNode) error {
914 if v == 0 {
915 return cbErr
916 }
917 return nil
918 })
919 if err != cbErr {
920 t.Errorf("wrong error returned from callback: %v instead of %v", err, cbErr)
921 }
922
923 if err = checkEndOfSync(dag, oid); err != nil {
924 t.Fatal(err)
925 }
926}
927
928// TestPruning tests sync pruning of the DAG for an object with 3 concurrent
929// updates (i.e. 2 conflict resolution convergent points). The pruning must
930// get rid of the DAG branches across the reconvergence points:
931// v0 -> v1 -> v2 -> v4 -> v5 -> v7 -> v8
932// |--> v3 ---| |
933// +--> v6 ---------------+
934// By pruning at v0, nothing is deleted.
935// Then by pruning at v1, only v0 is deleted.
936// Then by pruning at v5, v1-v4 are deleted leaving v5 and "v6 -> v7 -> v8".
937// Then by pruning at v7, v5-v6 are deleted leaving "v7 -> v8".
938// Then by pruning at v8, v7 is deleted leaving v8 as the head.
939// Then by pruning again at v8 nothing changes.
940func TestPruning(t *testing.T) {
941 dagfile := dagFilename()
942 defer os.Remove(dagfile)
943
944 dag, err := openDAG(dagfile)
945 if err != nil {
946 t.Fatalf("Cannot open new DAG file %s", dagfile)
947 }
948
949 if err = dagReplayCommands(dag, "local-init-01.sync"); err != nil {
950 t.Fatal(err)
951 }
952
953 oid, err := strToObjID("12345")
954 if err != nil {
955 t.Fatal(err)
956 }
957
958 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}, 3: {1}, 4: {2, 3}, 5: {4}, 6: {1}, 7: {5, 6}, 8: {7}}
959
960 // Loop pruning at an invalid version (333) then at v0, v5, v8 and again at v8.
961 testVersions := []storage.Version{333, 0, 1, 5, 7, 8, 8}
962 delCounts := []int{0, 0, 1, 4, 2, 1, 0}
963
964 for i, version := range testVersions {
965 del := 0
966 err = dag.prune(oid, version, func(lr string) error {
967 del++
968 return nil
969 })
970
971 if i == 0 && err == nil {
972 t.Errorf("pruning non-existent object %d:%d did not fail in DAG file %s", oid, version, dagfile)
973 } else if i > 0 && err != nil {
974 t.Errorf("pruning object %d:%d failed in DAG file %s: %v", oid, version, dagfile, err)
975 }
976
977 if del != delCounts[i] {
978 t.Errorf("pruning object %d:%d deleted %d log records instead of %d", oid, version, del, delCounts[i])
979 }
980
981 if head, err := dag.getHead(oid); err != nil || head != 8 {
982 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
983 }
984
985 // Remove pruned nodes from the expected parent map used to validate
986 // and set the parents of the pruned node to nil.
987 if version < 10 {
988 for j := storage.Version(0); j < version; j++ {
989 delete(exp, j)
990 }
991 exp[version] = nil
992 }
993
994 pmap := dag.getParentMap(oid)
995 if !reflect.DeepEqual(pmap, exp) {
996 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
997 }
998 }
999
1000 if err = checkEndOfSync(dag, oid); err != nil {
1001 t.Fatal(err)
1002 }
1003}
1004
1005// TestPruningCallbackError tests sync pruning of the DAG when the callback
1006// function returns an error. The pruning must try to delete as many nodes
1007// and log records as possible and properly adjust the parent pointers of
1008// the pruning node. The object DAG is:
1009// v0 -> v1 -> v2 -> v4 -> v5 -> v7 -> v8
1010// |--> v3 ---| |
1011// +--> v6 ---------------+
1012// By pruning at v8 and having the callback function fail for v3, all other
1013// nodes must be deleted and only v8 remains as the head.
1014func TestPruningCallbackError(t *testing.T) {
1015 dagfile := dagFilename()
1016 defer os.Remove(dagfile)
1017
1018 dag, err := openDAG(dagfile)
1019 if err != nil {
1020 t.Fatalf("Cannot open new DAG file %s", dagfile)
1021 }
1022
1023 if err = dagReplayCommands(dag, "local-init-01.sync"); err != nil {
1024 t.Fatal(err)
1025 }
1026
1027 oid, err := strToObjID("12345")
1028 if err != nil {
1029 t.Fatal(err)
1030 }
1031
1032 exp := map[storage.Version][]storage.Version{8: nil}
1033
1034 // Prune at v8 with a callback function that fails for v3.
1035 del, expDel := 0, 8
1036 version := storage.Version(8)
1037 err = dag.prune(oid, version, func(lr string) error {
1038 del++
1039 if lr == "logrec-03" {
1040 return fmt.Errorf("refuse to delete %s", lr)
1041 }
1042 return nil
1043 })
1044
1045 if err == nil {
1046 t.Errorf("pruning object %d:%d did not fail in DAG file %s", oid, version, dagfile)
1047 }
1048 if del != expDel {
1049 t.Errorf("pruning object %d:%d deleted %d log records instead of %d", oid, version, del, expDel)
1050 }
1051
1052 if head, err := dag.getHead(oid); err != nil || head != 8 {
1053 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
1054 }
1055
1056 pmap := dag.getParentMap(oid)
1057 if !reflect.DeepEqual(pmap, exp) {
1058 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
1059 }
1060
1061 if err = checkEndOfSync(dag, oid); err != nil {
1062 t.Fatal(err)
1063 }
1064}
1065
1066// TestDAGCompact tests compacting of dag's kvdb file.
1067func TestDAGCompact(t *testing.T) {
1068 dagfile := dagFilename()
1069 defer os.Remove(dagfile)
1070
1071 dag, err := openDAG(dagfile)
1072 if err != nil {
1073 t.Fatalf("Cannot open new DAG file %s", dagfile)
1074 }
1075
1076 // Put some data in "heads" table.
1077 headMap := make(map[storage.ID]storage.Version)
1078 for i := 0; i < 10; i++ {
1079 // Generate a random object id in [0, 1000).
Jiri Simsa870ddd62014-06-27 14:56:58 -07001080 oid, err := strToObjID(fmt.Sprintf("%d", testutil.Rand.Intn(1000)))
Jiri Simsa5293dcb2014-05-10 09:56:38 -07001081 if err != nil {
1082 t.Fatal(err)
1083 }
1084 // Generate a random version number for this object.
Jiri Simsa870ddd62014-06-27 14:56:58 -07001085 vers := storage.Version(testutil.Rand.Intn(5000))
Jiri Simsa5293dcb2014-05-10 09:56:38 -07001086
1087 // Cache this <oid,version> pair to verify with getHead().
1088 headMap[oid] = vers
1089
1090 if err = dag.setHead(oid, vers); err != nil {
1091 t.Fatalf("Cannot set object head %d (%d) in DAG file %s", oid, vers, dagfile)
1092 }
1093
1094 // Flush immediately to let the kvdb file grow.
1095 dag.flush()
1096 }
1097
1098 // Put some data in "nodes" table.
1099 type nodeKey struct {
1100 oid storage.ID
1101 vers storage.Version
1102 }
1103 nodeMap := make(map[nodeKey]*dagNode)
1104 for oid, vers := range headMap {
1105 // Generate a random dag node for this <oid, vers>.
Jiri Simsa870ddd62014-06-27 14:56:58 -07001106 l := uint64(testutil.Rand.Intn(20))
1107 p1 := storage.Version(testutil.Rand.Intn(5000))
1108 p2 := storage.Version(testutil.Rand.Intn(5000))
1109 log := fmt.Sprintf("%d", testutil.Rand.Intn(1000))
Jiri Simsa5293dcb2014-05-10 09:56:38 -07001110 node := &dagNode{Level: l, Parents: []storage.Version{p1, p2}, Logrec: log}
1111
1112 // Cache this <oid,version, dagNode> to verify with getNode().
1113 key := nodeKey{oid: oid, vers: vers}
1114 nodeMap[key] = node
1115
1116 if err = dag.setNode(oid, vers, node); err != nil {
1117 t.Fatalf("Cannot set object %d:%d (%v) in DAG file %s", oid, vers, node, dagfile)
1118 }
1119
1120 // Flush immediately to let the kvdb file grow.
1121 dag.flush()
1122 }
1123
1124 // Get size before compaction.
1125 oldSize := fileSize(dagfile)
1126 if oldSize < 0 {
1127 t.Fatalf("DAG file %s not created", dagfile)
1128 }
1129
1130 if err = dag.compact(); err != nil {
1131 t.Fatalf("Cannot compact DAG file %s", dagfile)
1132 }
1133
1134 // Verify size of kvdb file is reduced.
1135 size := fileSize(dagfile)
1136 if size < 0 {
1137 t.Fatalf("DAG file %s not created", dagfile)
1138 }
1139 if size > oldSize {
1140 t.Fatalf("DAG file %s not compacted", dagfile)
1141 }
1142
1143 // Check data exists after compaction.
1144 for oid, vers := range headMap {
1145 vers2, err := dag.getHead(oid)
1146 if err != nil {
1147 t.Errorf("Cannot find stored object head %d in DAG file %s", oid, dagfile)
1148 }
1149 if vers != vers2 {
1150 t.Errorf("Object %d has wrong head data in DAG file %s: %d instead of %d",
1151 oid, dagfile, vers2, vers)
1152 }
1153 }
1154 for key, node := range nodeMap {
1155 node2, err := dag.getNode(key.oid, key.vers)
1156 if err != nil || node2 == nil {
1157 t.Errorf("Cannot find stored object %d:%d in DAG file %s", key.oid, key.vers, dagfile)
1158 }
1159 if !reflect.DeepEqual(node, node2) {
1160 t.Errorf("Object %d:%d has wrong data in DAG file %s: %v instead of %v",
1161 key.oid, key.vers, dagfile, node2, node)
1162 }
1163 }
1164 dag.close()
1165}
Raja Daoudf123da82014-06-12 12:03:21 -07001166
1167// TestRemoteLinkedNoConflictSameHead tests sync of remote updates that contain
1168// linked nodes (conflict resolution by selecting an existing version) on top of
1169// a local initial state without conflict. An object is created locally and
1170// updated twice (v0 -> v1 -> v2). Another device has learned about v0, created
1171// (v0 -> v3), then learned about (v0 -> v1) and resolved that conflict by selecting
1172// v1 over v3. Now it sends that new info (v3 and the v1/v3 link) back to the
1173// original (local) device. Instead of a v2/v3 conflict, the device sees that
1174// v1 was chosen over v3 and resolves it as a no-conflict case.
1175func TestRemoteLinkedNoConflictSameHead(t *testing.T) {
1176 dagfile := dagFilename()
1177 defer os.Remove(dagfile)
1178
1179 dag, err := openDAG(dagfile)
1180 if err != nil {
1181 t.Fatalf("Cannot open new DAG file %s", dagfile)
1182 }
1183
1184 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
1185 t.Fatal(err)
1186 }
1187 if err = dagReplayCommands(dag, "remote-noconf-link-00.log.sync"); err != nil {
1188 t.Fatal(err)
1189 }
1190
1191 // The head must not have moved (i.e. still at v2) and the parent map
1192 // shows the newly grafted DAG fragment on top of the prior DAG.
1193 oid, err := strToObjID("12345")
1194 if err != nil {
1195 t.Fatal(err)
1196 }
1197
1198 if head, e := dag.getHead(oid); e != nil || head != 2 {
1199 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
1200 }
1201
1202 pmap := dag.getParentMap(oid)
1203
1204 exp := map[storage.Version][]storage.Version{0: nil, 1: {0, 3}, 2: {1}, 3: {0}}
1205
1206 if !reflect.DeepEqual(pmap, exp) {
1207 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
1208 }
1209
1210 // Verify the grafting of remote nodes.
1211 newHeads, grafts := dag.getGraftNodes(oid)
1212
1213 expNewHeads := map[storage.Version]struct{}{2: struct{}{}}
1214 if !reflect.DeepEqual(newHeads, expNewHeads) {
1215 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
1216 }
1217
1218 expgrafts := map[storage.Version]uint64{0: 0, 3: 1}
1219 if !reflect.DeepEqual(grafts, expgrafts) {
1220 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
1221 }
1222
1223 // There should be no conflict.
1224 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
1225 if !(!isConflict && newHead == 2 && oldHead == 2 && ancestor == 0 && errConflict == nil) {
1226 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1227 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1228 }
1229
1230 // Clear the grafting data and verify that hasConflict() fails without it.
1231 dag.clearGraft()
1232 isConflict, newHead, oldHead, ancestor, errConflict = dag.hasConflict(oid)
1233 if errConflict == nil {
1234 t.Errorf("hasConflict() did not fail w/o graft info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1235 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1236 }
1237
1238 if err := checkEndOfSync(dag, oid); err != nil {
1239 t.Fatal(err)
1240 }
1241}
1242
1243// TestRemoteLinkedConflict tests sync of remote updates that contain linked
1244// nodes (conflict resolution by selecting an existing version) on top of a local
1245// initial state triggering a local conflict. An object is created locally and
1246// updated twice (v0 -> v1 -> v2). Another device has along the way learned about v0,
1247// created (v0 -> v3), then learned about (v0 -> v1) and resolved that conflict by
1248// selecting v3 over v1. Now it sends that new info (v3 and the v3/v1 link) back
1249// to the original (local) device. The device sees a v2/v3 conflict.
1250func TestRemoteLinkedConflict(t *testing.T) {
1251 dagfile := dagFilename()
1252 defer os.Remove(dagfile)
1253
1254 dag, err := openDAG(dagfile)
1255 if err != nil {
1256 t.Fatalf("Cannot open new DAG file %s", dagfile)
1257 }
1258
1259 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
1260 t.Fatal(err)
1261 }
1262 if err = dagReplayCommands(dag, "remote-conf-link.log.sync"); err != nil {
1263 t.Fatal(err)
1264 }
1265
1266 // The head must not have moved (i.e. still at v2) and the parent map
1267 // shows the newly grafted DAG fragment on top of the prior DAG.
1268 oid, err := strToObjID("12345")
1269 if err != nil {
1270 t.Fatal(err)
1271 }
1272
1273 if head, e := dag.getHead(oid); e != nil || head != 2 {
1274 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
1275 }
1276
1277 pmap := dag.getParentMap(oid)
1278
1279 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}, 3: {0, 1}}
1280
1281 if !reflect.DeepEqual(pmap, exp) {
1282 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
1283 }
1284
1285 // Verify the grafting of remote nodes.
1286 newHeads, grafts := dag.getGraftNodes(oid)
1287
1288 expNewHeads := map[storage.Version]struct{}{2: struct{}{}, 3: struct{}{}}
1289 if !reflect.DeepEqual(newHeads, expNewHeads) {
1290 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
1291 }
1292
1293 expgrafts := map[storage.Version]uint64{0: 0, 1: 1}
1294 if !reflect.DeepEqual(grafts, expgrafts) {
1295 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
1296 }
1297
1298 // There should be no conflict.
1299 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
1300 if !(isConflict && newHead == 3 && oldHead == 2 && ancestor == 1 && errConflict == nil) {
1301 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1302 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1303 }
1304
1305 // Clear the grafting data and verify that hasConflict() fails without it.
1306 dag.clearGraft()
1307 isConflict, newHead, oldHead, ancestor, errConflict = dag.hasConflict(oid)
1308 if errConflict == nil {
1309 t.Errorf("hasConflict() did not fail w/o graft info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1310 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1311 }
1312
1313 if err := checkEndOfSync(dag, oid); err != nil {
1314 t.Fatal(err)
1315 }
1316}
1317
1318// TestRemoteLinkedNoConflictNewHead tests sync of remote updates that contain
1319// linked nodes (conflict resolution by selecting an existing version) on top of
1320// a local initial state without conflict, but moves the head node to a new one.
1321// An object is created locally and updated twice (v0 -> v1 -> v2). Another device
1322// has along the way learned about v0, created (v0 -> v3), then learned about
1323// (v0 -> v1 -> v2) and resolved that conflict by selecting v3 over v2. Now it
1324// sends that new info (v3 and the v3/v2 link) back to the original (local) device.
1325// The device sees that the new head v3 is "derived" from v2 thus no conflict.
1326func TestRemoteLinkedConflictNewHead(t *testing.T) {
1327 dagfile := dagFilename()
1328 defer os.Remove(dagfile)
1329
1330 dag, err := openDAG(dagfile)
1331 if err != nil {
1332 t.Fatalf("Cannot open new DAG file %s", dagfile)
1333 }
1334
1335 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
1336 t.Fatal(err)
1337 }
1338 if err = dagReplayCommands(dag, "remote-noconf-link-01.log.sync"); err != nil {
1339 t.Fatal(err)
1340 }
1341
1342 // The head must not have moved (i.e. still at v2) and the parent map
1343 // shows the newly grafted DAG fragment on top of the prior DAG.
1344 oid, err := strToObjID("12345")
1345 if err != nil {
1346 t.Fatal(err)
1347 }
1348
1349 if head, e := dag.getHead(oid); e != nil || head != 2 {
1350 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
1351 }
1352
1353 pmap := dag.getParentMap(oid)
1354
1355 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1}, 3: {0, 2}}
1356
1357 if !reflect.DeepEqual(pmap, exp) {
1358 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
1359 }
1360
1361 // Verify the grafting of remote nodes.
1362 newHeads, grafts := dag.getGraftNodes(oid)
1363
1364 expNewHeads := map[storage.Version]struct{}{3: struct{}{}}
1365 if !reflect.DeepEqual(newHeads, expNewHeads) {
1366 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
1367 }
1368
1369 expgrafts := map[storage.Version]uint64{0: 0, 2: 2}
1370 if !reflect.DeepEqual(grafts, expgrafts) {
1371 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
1372 }
1373
1374 // There should be no conflict.
1375 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
1376 if !(!isConflict && newHead == 3 && oldHead == 2 && ancestor == 0 && errConflict == nil) {
1377 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1378 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1379 }
1380
1381 // Clear the grafting data and verify that hasConflict() fails without it.
1382 dag.clearGraft()
1383 isConflict, newHead, oldHead, ancestor, errConflict = dag.hasConflict(oid)
1384 if errConflict == nil {
1385 t.Errorf("hasConflict() did not fail w/o graft info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1386 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1387 }
1388
1389 if err := checkEndOfSync(dag, oid); err != nil {
1390 t.Fatal(err)
1391 }
1392}
1393
1394// TestRemoteLinkedNoConflictNewHeadOvertake tests sync of remote updates that
1395// contain linked nodes (conflict resolution by selecting an existing version)
1396// on top of a local initial state without conflict, but moves the head node
1397// to a new one that overtook the linked node.
1398// An object is created locally and updated twice (v0 -> v1 -> v2). Another
1399// device has along the way learned about v0, created (v0 -> v3), then learned
1400// about (v0 -> v1 -> v2) and resolved that conflict by selecting v2 over v3.
1401// Then it creates a new update v4 from v2 (v2 -> v4). Now it sends that new
1402// info (v3, the v2/v3 link, and v4) back to the original (local) device.
1403// The device sees that the new head v4 is "derived" from v2 thus no conflict.
1404func TestRemoteLinkedConflictNewHeadOvertake(t *testing.T) {
1405 dagfile := dagFilename()
1406 defer os.Remove(dagfile)
1407
1408 dag, err := openDAG(dagfile)
1409 if err != nil {
1410 t.Fatalf("Cannot open new DAG file %s", dagfile)
1411 }
1412
1413 if err = dagReplayCommands(dag, "local-init-00.sync"); err != nil {
1414 t.Fatal(err)
1415 }
1416 if err = dagReplayCommands(dag, "remote-noconf-link-02.log.sync"); err != nil {
1417 t.Fatal(err)
1418 }
1419
1420 // The head must not have moved (i.e. still at v2) and the parent map
1421 // shows the newly grafted DAG fragment on top of the prior DAG.
1422 oid, err := strToObjID("12345")
1423 if err != nil {
1424 t.Fatal(err)
1425 }
1426
1427 if head, e := dag.getHead(oid); e != nil || head != 2 {
1428 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
1429 }
1430
1431 pmap := dag.getParentMap(oid)
1432
1433 exp := map[storage.Version][]storage.Version{0: nil, 1: {0}, 2: {1, 3}, 3: {0}, 4: {2}}
1434
1435 if !reflect.DeepEqual(pmap, exp) {
1436 t.Errorf("Invalid object %d parent map in DAG file %s: (%v) instead of (%v)", oid, dagfile, pmap, exp)
1437 }
1438
1439 // Verify the grafting of remote nodes.
1440 newHeads, grafts := dag.getGraftNodes(oid)
1441
1442 expNewHeads := map[storage.Version]struct{}{4: struct{}{}}
1443 if !reflect.DeepEqual(newHeads, expNewHeads) {
1444 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
1445 }
1446
1447 expgrafts := map[storage.Version]uint64{0: 0, 2: 2, 3: 1}
1448 if !reflect.DeepEqual(grafts, expgrafts) {
1449 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
1450 }
1451
1452 // There should be no conflict.
1453 isConflict, newHead, oldHead, ancestor, errConflict := dag.hasConflict(oid)
1454 if !(!isConflict && newHead == 4 && oldHead == 2 && ancestor == 0 && errConflict == nil) {
1455 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1456 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1457 }
1458
1459 // Then we can move the head and clear the grafting data.
1460 if err = dag.moveHead(oid, newHead); err != nil {
1461 t.Errorf("Object %d cannot move head to %d in DAG file %s: %v", oid, newHead, dagfile, err)
1462 }
1463
1464 // Clear the grafting data and verify that hasConflict() fails without it.
1465 dag.clearGraft()
1466 isConflict, newHead, oldHead, ancestor, errConflict = dag.hasConflict(oid)
1467 if errConflict == nil {
1468 t.Errorf("hasConflict() did not fail w/o graft info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1469 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1470 }
1471
1472 // Now new info comes from another device repeating the v2/v3 link.
1473 // Verify that it is a NOP (no changes).
1474 if err = dagReplayCommands(dag, "remote-noconf-link-repeat.log.sync"); err != nil {
1475 t.Fatal(err)
1476 }
1477
1478 if head, e := dag.getHead(oid); e != nil || head != 4 {
1479 t.Errorf("Object %d has wrong head in DAG file %s: %d", oid, dagfile, head)
1480 }
1481
1482 newHeads, grafts = dag.getGraftNodes(oid)
1483 if !reflect.DeepEqual(newHeads, expNewHeads) {
1484 t.Errorf("Object %d has invalid newHeads in DAG file %s: (%v) instead of (%v)", oid, dagfile, newHeads, expNewHeads)
1485 }
1486
1487 expgrafts = map[storage.Version]uint64{}
1488 if !reflect.DeepEqual(grafts, expgrafts) {
1489 t.Errorf("Invalid object %d graft in DAG file %s: (%v) instead of (%v)", oid, dagfile, grafts, expgrafts)
1490 }
1491
1492 isConflict, newHead, oldHead, ancestor, errConflict = dag.hasConflict(oid)
1493 if !(!isConflict && newHead == 4 && oldHead == 4 && ancestor == 0 && errConflict == nil) {
1494 t.Errorf("Object %d wrong conflict info: flag %t, newHead %d, oldHead %d, ancestor %d, err %v",
1495 oid, isConflict, newHead, oldHead, ancestor, errConflict)
1496 }
1497
1498 if err := checkEndOfSync(dag, oid); err != nil {
1499 t.Fatal(err)
1500 }
1501}