| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | package archiver | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"runtime" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/errors" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							| 
									
										
										
										
											2022-08-20 11:56:00 +02:00
										 |  |  | 	"github.com/restic/restic/internal/test" | 
					
						
							| 
									
										
										
										
											2022-05-27 19:08:50 +02:00
										 |  |  | 	"golang.org/x/sync/errgroup" | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-10 23:41:11 +01:00
										 |  |  | func treeSaveHelper(_ context.Context, _ restic.BlobType, buf *Buffer, _ string, cb func(res SaveBlobResponse)) { | 
					
						
							| 
									
										
										
										
											2022-10-07 20:23:38 +02:00
										 |  |  | 	cb(SaveBlobResponse{ | 
					
						
							| 
									
										
										
										
											2022-08-19 23:08:13 +02:00
										 |  |  | 		id:         restic.NewRandomID(), | 
					
						
							|  |  |  | 		known:      false, | 
					
						
							| 
									
										
										
										
											2022-10-08 21:29:32 +02:00
										 |  |  | 		length:     len(buf.Data), | 
					
						
							|  |  |  | 		sizeInRepo: len(buf.Data), | 
					
						
							| 
									
										
										
										
											2022-10-07 20:23:38 +02:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2022-08-19 23:08:13 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-08 21:29:32 +02:00
										 |  |  | func setupTreeSaver() (context.Context, context.CancelFunc, *TreeSaver, func() error) { | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 	ctx, cancel := context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2022-05-27 19:08:50 +02:00
										 |  |  | 	wg, ctx := errgroup.WithContext(ctx) | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-21 00:31:26 +02:00
										 |  |  | 	errFn := func(snPath string, err error) error { | 
					
						
							| 
									
										
										
										
											2022-08-19 23:08:13 +02:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-08 21:29:32 +02:00
										 |  |  | 	b := NewTreeSaver(ctx, wg, uint(runtime.NumCPU()), treeSaveHelper, errFn) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	shutdown := func() error { | 
					
						
							|  |  |  | 		b.TriggerShutdown() | 
					
						
							|  |  |  | 		return wg.Wait() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ctx, cancel, b, shutdown | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestTreeSaver(t *testing.T) { | 
					
						
							|  |  |  | 	ctx, cancel, b, shutdown := setupTreeSaver() | 
					
						
							|  |  |  | 	defer cancel() | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-29 11:57:10 +02:00
										 |  |  | 	var results []FutureNode | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < 20; i++ { | 
					
						
							|  |  |  | 		node := &restic.Node{ | 
					
						
							|  |  |  | 			Name: fmt.Sprintf("file-%d", i), | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 23:08:13 +02:00
										 |  |  | 		fb := b.Save(ctx, join("/", node.Name), node.Name, node, nil, nil) | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 		results = append(results, fb) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, tree := range results { | 
					
						
							| 
									
										
										
										
											2022-05-29 11:57:10 +02:00
										 |  |  | 		tree.take(ctx) | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-08 21:29:32 +02:00
										 |  |  | 	err := shutdown() | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestTreeSaverError(t *testing.T) { | 
					
						
							|  |  |  | 	var tests = []struct { | 
					
						
							|  |  |  | 		trees  int | 
					
						
							| 
									
										
										
										
											2022-08-19 23:08:13 +02:00
										 |  |  | 		failAt int | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 	}{ | 
					
						
							|  |  |  | 		{1, 1}, | 
					
						
							|  |  |  | 		{20, 2}, | 
					
						
							|  |  |  | 		{20, 5}, | 
					
						
							|  |  |  | 		{20, 15}, | 
					
						
							|  |  |  | 		{200, 150}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	errTest := errors.New("test error") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, test := range tests { | 
					
						
							|  |  |  | 		t.Run("", func(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2022-10-08 21:29:32 +02:00
										 |  |  | 			ctx, cancel, b, shutdown := setupTreeSaver() | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 			defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-29 11:57:10 +02:00
										 |  |  | 			var results []FutureNode | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			for i := 0; i < test.trees; i++ { | 
					
						
							|  |  |  | 				node := &restic.Node{ | 
					
						
							|  |  |  | 					Name: fmt.Sprintf("file-%d", i), | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2022-08-19 23:08:13 +02:00
										 |  |  | 				nodes := []FutureNode{ | 
					
						
							|  |  |  | 					newFutureNodeWithResult(futureNodeResult{node: &restic.Node{ | 
					
						
							|  |  |  | 						Name: fmt.Sprintf("child-%d", i), | 
					
						
							|  |  |  | 					}}), | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				if (i + 1) == test.failAt { | 
					
						
							|  |  |  | 					nodes = append(nodes, newFutureNodeWithResult(futureNodeResult{ | 
					
						
							|  |  |  | 						err: errTest, | 
					
						
							|  |  |  | 					})) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 23:08:13 +02:00
										 |  |  | 				fb := b.Save(ctx, join("/", node.Name), node.Name, node, nodes, nil) | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 				results = append(results, fb) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for _, tree := range results { | 
					
						
							| 
									
										
										
										
											2022-05-29 11:57:10 +02:00
										 |  |  | 				tree.take(ctx) | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-08 21:29:32 +02:00
										 |  |  | 			err := shutdown() | 
					
						
							| 
									
										
										
										
											2018-05-12 21:40:31 +02:00
										 |  |  | 			if err == nil { | 
					
						
							|  |  |  | 				t.Errorf("expected error not found") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if err != errTest { | 
					
						
							|  |  |  | 				t.Fatalf("unexpected error found: %v", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-20 11:56:00 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestTreeSaverDuplicates(t *testing.T) { | 
					
						
							|  |  |  | 	for _, identicalNodes := range []bool{true, false} { | 
					
						
							|  |  |  | 		t.Run("", func(t *testing.T) { | 
					
						
							|  |  |  | 			ctx, cancel, b, shutdown := setupTreeSaver() | 
					
						
							|  |  |  | 			defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			node := &restic.Node{ | 
					
						
							|  |  |  | 				Name: "file", | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			nodes := []FutureNode{ | 
					
						
							|  |  |  | 				newFutureNodeWithResult(futureNodeResult{node: &restic.Node{ | 
					
						
							|  |  |  | 					Name: "child", | 
					
						
							|  |  |  | 				}}), | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if identicalNodes { | 
					
						
							|  |  |  | 				nodes = append(nodes, newFutureNodeWithResult(futureNodeResult{node: &restic.Node{ | 
					
						
							|  |  |  | 					Name: "child", | 
					
						
							|  |  |  | 				}})) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				nodes = append(nodes, newFutureNodeWithResult(futureNodeResult{node: &restic.Node{ | 
					
						
							|  |  |  | 					Name: "child", | 
					
						
							|  |  |  | 					Size: 42, | 
					
						
							|  |  |  | 				}})) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			fb := b.Save(ctx, join("/", node.Name), node.Name, node, nodes, nil) | 
					
						
							|  |  |  | 			fb.take(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			err := shutdown() | 
					
						
							|  |  |  | 			if identicalNodes { | 
					
						
							|  |  |  | 				test.Assert(t, err == nil, "unexpected error found: %v", err) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				test.Assert(t, err != nil, "expected error not found") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |