| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | package checker | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 	"io" | 
					
						
							|  |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2015-07-12 00:25:42 +02:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/debug" | 
					
						
							| 
									
										
										
										
											2018-10-28 21:12:15 +01:00
										 |  |  | 	"github.com/restic/restic/internal/errors" | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/pack" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/repository" | 
					
						
							| 
									
										
										
										
											2018-10-28 21:12:15 +01:00
										 |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							| 
									
										
										
										
											2020-11-04 14:11:29 +01:00
										 |  |  | 	"github.com/restic/restic/internal/ui/progress" | 
					
						
							| 
									
										
										
										
											2018-10-28 21:12:15 +01:00
										 |  |  | 	"golang.org/x/sync/errgroup" | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Checker runs various checks on a repository. It is advisable to create an | 
					
						
							|  |  |  | // exclusive Lock in the repository before running any checks. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // A Checker only tests for internal errors within the data structures of the | 
					
						
							|  |  |  | // repository (e.g. missing blobs), and needs a valid Repository to work on. | 
					
						
							|  |  |  | type Checker struct { | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	packs    map[restic.ID]int64 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	blobRefs struct { | 
					
						
							|  |  |  | 		sync.Mutex | 
					
						
							| 
									
										
										
										
											2020-11-06 23:41:04 +01:00
										 |  |  | 		M restic.BlobSet | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-11-07 00:07:32 +01:00
										 |  |  | 	trackUnused bool | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-12 22:34:12 +02:00
										 |  |  | 	masterIndex *repository.MasterIndex | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-31 23:07:50 +02:00
										 |  |  | 	repo restic.Repository | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // New returns a new checker which runs on repo. | 
					
						
							| 
									
										
										
										
											2020-11-07 00:07:32 +01:00
										 |  |  | func New(repo restic.Repository, trackUnused bool) *Checker { | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	c := &Checker{ | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 		packs:       make(map[restic.ID]int64), | 
					
						
							| 
									
										
										
										
											2015-10-12 22:34:12 +02:00
										 |  |  | 		masterIndex: repository.NewMasterIndex(), | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 		repo:        repo, | 
					
						
							| 
									
										
										
										
											2020-11-07 00:07:32 +01:00
										 |  |  | 		trackUnused: trackUnused, | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 23:41:04 +01:00
										 |  |  | 	c.blobRefs.M = restic.NewBlobSet() | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return c | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 21:49:30 +01:00
										 |  |  | const defaultParallelism = 5 | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | // ErrDuplicatePacks is returned when a pack is found in more than one index. | 
					
						
							|  |  |  | type ErrDuplicatePacks struct { | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	PackID  restic.ID | 
					
						
							|  |  |  | 	Indexes restic.IDSet | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e ErrDuplicatePacks) Error() string { | 
					
						
							|  |  |  | 	return fmt.Sprintf("pack %v contained in several indexes: %v", e.PackID.Str(), e.Indexes) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-08 21:50:48 +01:00
										 |  |  | // ErrOldIndexFormat is returned when an index with the old format is | 
					
						
							|  |  |  | // found. | 
					
						
							|  |  |  | type ErrOldIndexFormat struct { | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	restic.ID | 
					
						
							| 
									
										
										
										
											2015-11-08 21:50:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (err ErrOldIndexFormat) Error() string { | 
					
						
							|  |  |  | 	return fmt.Sprintf("index %v has old format", err.ID.Str()) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | // LoadIndex loads all index files. | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | func (c *Checker) LoadIndex(ctx context.Context) (hints []error, errs []error) { | 
					
						
							| 
									
										
										
										
											2016-09-27 22:35:08 +02:00
										 |  |  | 	debug.Log("Start") | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// track spawned goroutines using wg, create a new context which is | 
					
						
							|  |  |  | 	// cancelled as soon as an error occurs. | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	wg, wgCtx := errgroup.WithContext(ctx) | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	type FileInfo struct { | 
					
						
							|  |  |  | 		restic.ID | 
					
						
							|  |  |  | 		Size int64 | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	type Result struct { | 
					
						
							|  |  |  | 		*repository.Index | 
					
						
							|  |  |  | 		restic.ID | 
					
						
							|  |  |  | 		Err error | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	ch := make(chan FileInfo) | 
					
						
							|  |  |  | 	resultCh := make(chan Result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// send list of index files through ch, which is closed afterwards | 
					
						
							|  |  |  | 	wg.Go(func() error { | 
					
						
							|  |  |  | 		defer close(ch) | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 		return c.repo.List(wgCtx, restic.IndexFile, func(id restic.ID, size int64) error { | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 			select { | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 			case <-wgCtx.Done(): | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 				return nil | 
					
						
							|  |  |  | 			case ch <- FileInfo{id, size}: | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2015-11-08 21:50:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	// a worker receives an index ID from ch, loads the index, and sends it to indexCh | 
					
						
							|  |  |  | 	worker := func() error { | 
					
						
							| 
									
										
										
										
											2019-03-24 22:12:38 +01:00
										 |  |  | 		var buf []byte | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 		for fi := range ch { | 
					
						
							|  |  |  | 			debug.Log("worker got file %v", fi.ID.Str()) | 
					
						
							| 
									
										
										
										
											2019-03-24 22:12:38 +01:00
										 |  |  | 			var err error | 
					
						
							|  |  |  | 			var idx *repository.Index | 
					
						
							| 
									
										
										
										
											2020-10-13 20:39:54 +02:00
										 |  |  | 			oldFormat := false | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 			buf, err = c.repo.LoadAndDecrypt(wgCtx, buf[:0], restic.IndexFile, fi.ID) | 
					
						
							| 
									
										
										
										
											2020-10-13 20:39:54 +02:00
										 |  |  | 			if err == nil { | 
					
						
							| 
									
										
										
										
											2020-10-17 09:06:10 +02:00
										 |  |  | 				idx, oldFormat, err = repository.DecodeIndex(buf, fi.ID) | 
					
						
							| 
									
										
										
										
											2020-10-13 20:39:54 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if oldFormat { | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 				debug.Log("index %v has old format", fi.ID.Str()) | 
					
						
							|  |  |  | 				hints = append(hints, ErrOldIndexFormat{fi.ID}) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 			err = errors.Wrapf(err, "error loading index %v", fi.ID.Str()) | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 			select { | 
					
						
							|  |  |  | 			case resultCh <- Result{idx, fi.ID, err}: | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 			case <-wgCtx.Done(): | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	// run workers on ch | 
					
						
							|  |  |  | 	wg.Go(func() error { | 
					
						
							| 
									
										
										
										
											2020-10-14 14:23:06 +02:00
										 |  |  | 		defer close(resultCh) | 
					
						
							|  |  |  | 		return repository.RunWorkers(defaultParallelism, worker) | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	// receive decoded indexes | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	packToIndex := make(map[restic.ID]restic.IDSet) | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	wg.Go(func() error { | 
					
						
							|  |  |  | 		for res := range resultCh { | 
					
						
							|  |  |  | 			debug.Log("process index %v, err %v", res.ID, res.Err) | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 			if res.Err != nil { | 
					
						
							|  |  |  | 				errs = append(errs, res.Err) | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 			c.masterIndex.Insert(res.Index) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			debug.Log("process blobs") | 
					
						
							|  |  |  | 			cnt := 0 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 			for blob := range res.Index.Each(wgCtx) { | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 				cnt++ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if _, ok := packToIndex[blob.PackID]; !ok { | 
					
						
							|  |  |  | 					packToIndex[blob.PackID] = restic.NewIDSet() | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				packToIndex[blob.PackID].Insert(res.ID) | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			debug.Log("%d blobs processed", cnt) | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	err := wg.Wait() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		errs = append(errs, err) | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	// Merge index before computing pack sizes, as this needs removed duplicates | 
					
						
							|  |  |  | 	c.masterIndex.MergeFinalIndexes() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// compute pack size using index entries | 
					
						
							| 
									
										
										
										
											2020-11-16 04:57:43 +01:00
										 |  |  | 	c.packs = c.masterIndex.PackSize(ctx, false) | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-27 22:35:08 +02:00
										 |  |  | 	debug.Log("checking for duplicate packs") | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | 	for packID := range c.packs { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 		debug.Log("  check pack %v: contained in %d indexes", packID, len(packToIndex[packID])) | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | 		if len(packToIndex[packID]) > 1 { | 
					
						
							| 
									
										
										
										
											2015-10-25 16:26:50 +01:00
										 |  |  | 			hints = append(hints, ErrDuplicatePacks{ | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | 				PackID:  packID, | 
					
						
							|  |  |  | 				Indexes: packToIndex[packID], | 
					
						
							| 
									
										
										
										
											2015-10-25 16:26:50 +01:00
										 |  |  | 			}) | 
					
						
							| 
									
										
										
										
											2015-10-25 16:00:06 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 21:27:28 +01:00
										 |  |  | 	err = c.repo.SetIndex(c.masterIndex) | 
					
						
							| 
									
										
										
										
											2018-03-31 10:02:09 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		debug.Log("SetIndex returned error: %v", err) | 
					
						
							|  |  |  | 		errs = append(errs, err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-25 16:26:50 +01:00
										 |  |  | 	return hints, errs | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | // PackError describes an error with a specific pack. | 
					
						
							|  |  |  | type PackError struct { | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	ID       restic.ID | 
					
						
							| 
									
										
										
										
											2015-07-12 17:09:48 +02:00
										 |  |  | 	Orphaned bool | 
					
						
							|  |  |  | 	Err      error | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e PackError) Error() string { | 
					
						
							| 
									
										
										
										
											2018-04-07 10:07:54 +02:00
										 |  |  | 	return "pack " + e.ID.Str() + ": " + e.Err.Error() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // IsOrphanedPack returns true if the error describes a pack which is not | 
					
						
							|  |  |  | // contained in any index. | 
					
						
							|  |  |  | func IsOrphanedPack(err error) bool { | 
					
						
							|  |  |  | 	if e, ok := errors.Cause(err).(PackError); ok && e.Orphaned { | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return false | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 00:25:42 +02:00
										 |  |  | // Packs checks that all packs referenced in the index are still available and | 
					
						
							| 
									
										
										
										
											2015-07-12 01:44:19 +02:00
										 |  |  | // there are no packs that aren't in an index. errChan is closed after all | 
					
						
							|  |  |  | // packs have been checked. | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | func (c *Checker) Packs(ctx context.Context, errChan chan<- error) { | 
					
						
							| 
									
										
										
										
											2015-07-12 01:44:19 +02:00
										 |  |  | 	defer close(errChan) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-27 22:35:08 +02:00
										 |  |  | 	debug.Log("checking for %d packs", len(c.packs)) | 
					
						
							| 
									
										
										
										
											2015-07-12 00:25:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-11 21:00:48 -05:00
										 |  |  | 	debug.Log("listing repository packs") | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	repoPacks := make(map[restic.ID]int64) | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-16 11:16:38 +02:00
										 |  |  | 	err := c.repo.List(ctx, restic.PackFile, func(id restic.ID, size int64) error { | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 		repoPacks[id] = size | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		errChan <- err | 
					
						
							| 
									
										
										
										
											2015-07-12 00:25:42 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	for id, size := range c.packs { | 
					
						
							|  |  |  | 		reposize, ok := repoPacks[id] | 
					
						
							|  |  |  | 		// remove from repoPacks so we can find orphaned packs | 
					
						
							|  |  |  | 		delete(repoPacks, id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// missing: present in c.packs but not in the repo | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							|  |  |  | 			select { | 
					
						
							|  |  |  | 			case <-ctx.Done(): | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			case errChan <- PackError{ID: id, Err: errors.New("does not exist")}: | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// size not matching: present in c.packs and in the repo, but sizes do not match | 
					
						
							|  |  |  | 		if size != reposize { | 
					
						
							|  |  |  | 			select { | 
					
						
							|  |  |  | 			case <-ctx.Done(): | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			case errChan <- PackError{ID: id, Err: errors.Errorf("unexpected file size: got %d, expected %d", reposize, size)}: | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-01-11 21:00:48 -05:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-07-12 00:25:42 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	// orphaned: present in the repo but not in c.packs | 
					
						
							|  |  |  | 	for orphanID := range repoPacks { | 
					
						
							| 
									
										
										
										
											2018-01-11 21:00:48 -05:00
										 |  |  | 		select { | 
					
						
							|  |  |  | 		case <-ctx.Done(): | 
					
						
							|  |  |  | 			return | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 		case errChan <- PackError{ID: orphanID, Orphaned: true, Err: errors.New("not referenced in any index")}: | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Error is an error that occurred while checking a repository. | 
					
						
							|  |  |  | type Error struct { | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	TreeID restic.ID | 
					
						
							|  |  |  | 	BlobID restic.ID | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 	Err    error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e Error) Error() string { | 
					
						
							| 
									
										
										
										
											2016-02-26 23:06:25 +01:00
										 |  |  | 	if !e.BlobID.IsNull() && !e.TreeID.IsNull() { | 
					
						
							| 
									
										
										
										
											2015-10-11 19:25:02 +02:00
										 |  |  | 		msg := "tree " + e.TreeID.Str() | 
					
						
							|  |  |  | 		msg += ", blob " + e.BlobID.Str() | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		msg += ": " + e.Err.Error() | 
					
						
							|  |  |  | 		return msg | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-26 23:06:25 +01:00
										 |  |  | 	if !e.TreeID.IsNull() { | 
					
						
							| 
									
										
										
										
											2015-10-11 19:25:02 +02:00
										 |  |  | 		return "tree " + e.TreeID.Str() + ": " + e.Err.Error() | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return e.Err.Error() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-11 19:13:35 +02:00
										 |  |  | // TreeError collects several errors that occurred while processing a tree. | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | type TreeError struct { | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	ID     restic.ID | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	Errors []error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e TreeError) Error() string { | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 	return fmt.Sprintf("tree %v: %v", e.ID.Str(), e.Errors) | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type treeJob struct { | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	restic.ID | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	error | 
					
						
							|  |  |  | 	*restic.Tree | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // loadTreeWorker loads trees from repo and sends them to out. | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | func loadTreeWorker(ctx context.Context, repo restic.Repository, | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 	in <-chan restic.ID, out chan<- treeJob) { | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 	for treeID := range in { | 
					
						
							|  |  |  | 		tree, err := repo.LoadTree(ctx, treeID) | 
					
						
							|  |  |  | 		debug.Log("load tree %v (%v) returned err: %v", tree, treeID, err) | 
					
						
							|  |  |  | 		job := treeJob{ID: treeID, error: err, Tree: tree} | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 01:44:19 +02:00
										 |  |  | 		select { | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | 		case <-ctx.Done(): | 
					
						
							| 
									
										
										
										
											2015-07-12 01:44:19 +02:00
										 |  |  | 			return | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 		case out <- job: | 
					
						
							| 
									
										
										
										
											2015-07-12 01:44:19 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | // checkTreeWorker checks the trees received and sends out errors to errChan. | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | func (c *Checker) checkTreeWorker(ctx context.Context, in <-chan treeJob, out chan<- error) { | 
					
						
							|  |  |  | 	for job := range in { | 
					
						
							|  |  |  | 		debug.Log("check tree %v (tree %v, err %v)", job.ID, job.Tree, job.error) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var errs []error | 
					
						
							|  |  |  | 		if job.error != nil { | 
					
						
							|  |  |  | 			errs = append(errs, job.error) | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			errs = c.checkTree(job.ID, job.Tree) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 		if len(errs) == 0 { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		treeError := TreeError{ID: job.ID, Errors: errs} | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		select { | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | 		case <-ctx.Done(): | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 			return | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 		case out <- treeError: | 
					
						
							| 
									
										
										
										
											2016-09-27 22:35:08 +02:00
										 |  |  | 			debug.Log("tree %v: sent %d errors", treeError.ID, len(treeError.Errors)) | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 18:34:55 +02:00
										 |  |  | func (c *Checker) filterTrees(ctx context.Context, backlog restic.IDs, loaderChan chan<- restic.ID, in <-chan treeJob, out chan<- treeJob) { | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		inCh                    = in | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 		outCh                   chan<- treeJob | 
					
						
							|  |  |  | 		loadCh                  chan<- restic.ID | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		job                     treeJob | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 		nextTreeID              restic.ID | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		outstandingLoadTreeJobs = 0 | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	for { | 
					
						
							|  |  |  | 		if loadCh == nil && len(backlog) > 0 { | 
					
						
							| 
									
										
										
										
											2019-07-06 20:01:48 +02:00
										 |  |  | 			// process last added ids first, that is traverse the tree in depth-first order | 
					
						
							|  |  |  | 			ln := len(backlog) - 1 | 
					
						
							|  |  |  | 			nextTreeID, backlog = backlog[ln], backlog[:ln] | 
					
						
							| 
									
										
										
										
											2019-07-13 18:34:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// use a separate flag for processed trees to ensure that check still processes trees | 
					
						
							|  |  |  | 			// even when a file references a tree blob | 
					
						
							|  |  |  | 			c.blobRefs.Lock() | 
					
						
							| 
									
										
										
										
											2020-04-18 19:46:33 +02:00
										 |  |  | 			h := restic.BlobHandle{ID: nextTreeID, Type: restic.TreeBlob} | 
					
						
							| 
									
										
										
										
											2020-11-06 23:41:04 +01:00
										 |  |  | 			blobReferenced := c.blobRefs.M.Has(h) | 
					
						
							| 
									
										
										
										
											2020-11-06 23:54:20 +01:00
										 |  |  | 			// noop if already referenced | 
					
						
							|  |  |  | 			c.blobRefs.M.Insert(h) | 
					
						
							| 
									
										
										
										
											2019-07-13 18:34:55 +02:00
										 |  |  | 			c.blobRefs.Unlock() | 
					
						
							| 
									
										
										
										
											2020-11-06 23:41:04 +01:00
										 |  |  | 			if blobReferenced { | 
					
						
							| 
									
										
										
										
											2019-07-06 18:35:03 +02:00
										 |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-07-13 18:34:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 			loadCh = loaderChan | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		if loadCh == nil && outCh == nil && outstandingLoadTreeJobs == 0 { | 
					
						
							| 
									
										
										
										
											2016-09-27 22:35:08 +02:00
										 |  |  | 			debug.Log("backlog is empty, all channels nil, exiting") | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 			return | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		select { | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | 		case <-ctx.Done(): | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 			return | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		case loadCh <- nextTreeID: | 
					
						
							|  |  |  | 			outstandingLoadTreeJobs++ | 
					
						
							|  |  |  | 			loadCh = nil | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		case j, ok := <-inCh: | 
					
						
							|  |  |  | 			if !ok { | 
					
						
							| 
									
										
										
										
											2016-09-27 22:35:08 +02:00
										 |  |  | 				debug.Log("input channel closed") | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 				inCh = nil | 
					
						
							|  |  |  | 				in = nil | 
					
						
							|  |  |  | 				continue | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			outstandingLoadTreeJobs-- | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 			debug.Log("input job tree %v", j.ID) | 
					
						
							| 
									
										
										
										
											2015-10-11 18:45:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if j.error != nil { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 				debug.Log("received job with error: %v (tree %v, ID %v)", j.error, j.Tree, j.ID) | 
					
						
							| 
									
										
										
										
											2015-10-11 18:45:16 +02:00
										 |  |  | 			} else if j.Tree == nil { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 				debug.Log("received job with nil tree pointer: %v (ID %v)", j.error, j.ID) | 
					
						
							| 
									
										
										
										
											2020-03-07 14:10:34 +01:00
										 |  |  | 				// send a new job with the new error instead of the old one | 
					
						
							|  |  |  | 				j = treeJob{ID: j.ID, error: errors.New("tree is nil and error is nil")} | 
					
						
							| 
									
										
										
										
											2015-10-11 18:45:16 +02:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2019-07-06 18:12:48 +02:00
										 |  |  | 				subtrees := j.Tree.Subtrees() | 
					
						
							|  |  |  | 				debug.Log("subtrees for tree %v: %v", j.ID, subtrees) | 
					
						
							| 
									
										
										
										
											2019-07-06 20:01:48 +02:00
										 |  |  | 				// iterate backwards over subtree to compensate backwards traversal order of nextTreeID selection | 
					
						
							|  |  |  | 				for i := len(subtrees) - 1; i >= 0; i-- { | 
					
						
							|  |  |  | 					id := subtrees[i] | 
					
						
							| 
									
										
										
										
											2015-10-11 18:45:16 +02:00
										 |  |  | 					if id.IsNull() { | 
					
						
							| 
									
										
										
										
											2015-10-11 19:13:35 +02:00
										 |  |  | 						// We do not need to raise this error here, it is | 
					
						
							|  |  |  | 						// checked when the tree is checked. Just make sure | 
					
						
							|  |  |  | 						// that we do not add any null IDs to the backlog. | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 						debug.Log("tree %v has nil subtree", j.ID) | 
					
						
							| 
									
										
										
										
											2015-10-11 18:45:16 +02:00
										 |  |  | 						continue | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					backlog = append(backlog, id) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 			job = j | 
					
						
							|  |  |  | 			outCh = out | 
					
						
							|  |  |  | 			inCh = nil | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case outCh <- job: | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 			debug.Log("tree sent to check: %v", job.ID) | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 			outCh = nil | 
					
						
							|  |  |  | 			inCh = in | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 09:32:06 +01:00
										 |  |  | func loadSnapshotTreeIDs(ctx context.Context, repo restic.Repository) (ids restic.IDs, errs []error) { | 
					
						
							|  |  |  | 	err := restic.ForAllSnapshots(ctx, repo, nil, func(id restic.ID, sn *restic.Snapshot, err error) error { | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			errs = append(errs, err) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		treeID := *sn.Tree | 
					
						
							|  |  |  | 		debug.Log("snapshot %v has tree %v", id, treeID) | 
					
						
							|  |  |  | 		ids = append(ids, treeID) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		errs = append(errs, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ids, errs | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | // Structure checks that for all snapshots all referenced data blobs and | 
					
						
							|  |  |  | // subtrees are available in the index. errChan is closed after all trees have | 
					
						
							|  |  |  | // been traversed. | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | func (c *Checker) Structure(ctx context.Context, errChan chan<- error) { | 
					
						
							|  |  |  | 	trees, errs := loadSnapshotTreeIDs(ctx, c.repo) | 
					
						
							| 
									
										
										
										
											2016-09-27 22:35:08 +02:00
										 |  |  | 	debug.Log("need to check %d trees from snapshots, %d errs returned", len(trees), len(errs)) | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	for _, err := range errs { | 
					
						
							|  |  |  | 		select { | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | 		case <-ctx.Done(): | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 			return | 
					
						
							|  |  |  | 		case errChan <- err: | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 	loaderChan := make(chan restic.ID) | 
					
						
							|  |  |  | 	loadedTreeChan := make(chan treeJob) | 
					
						
							|  |  |  | 	treeStream := make(chan treeJob) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wg, ctx := errgroup.WithContext(ctx) | 
					
						
							|  |  |  | 	var loadTreeWg sync.WaitGroup | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	for i := 0; i < defaultParallelism; i++ { | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 		loadTreeWg.Add(1) | 
					
						
							|  |  |  | 		wg.Go(func() error { | 
					
						
							|  |  |  | 			defer loadTreeWg.Done() | 
					
						
							|  |  |  | 			loadTreeWorker(ctx, c.repo, loaderChan, loadedTreeChan) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 	// close once all loadTreeWorkers have completed | 
					
						
							|  |  |  | 	wg.Go(func() error { | 
					
						
							|  |  |  | 		loadTreeWg.Wait() | 
					
						
							|  |  |  | 		close(loadedTreeChan) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 00:23:45 +01:00
										 |  |  | 	defer close(errChan) | 
					
						
							|  |  |  | 	for i := 0; i < defaultParallelism; i++ { | 
					
						
							|  |  |  | 		wg.Go(func() error { | 
					
						
							|  |  |  | 			c.checkTreeWorker(ctx, treeStream, errChan) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wg.Go(func() error { | 
					
						
							|  |  |  | 		defer close(loaderChan) | 
					
						
							|  |  |  | 		defer close(treeStream) | 
					
						
							|  |  |  | 		c.filterTrees(ctx, trees, loaderChan, loadedTreeChan, treeStream) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	wg.Wait() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | func (c *Checker) checkTree(id restic.ID, tree *restic.Tree) (errs []error) { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 	debug.Log("checking tree %v", id) | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-11 20:55:28 +02:00
										 |  |  | 	for _, node := range tree.Nodes { | 
					
						
							| 
									
										
										
										
											2016-09-01 21:20:03 +02:00
										 |  |  | 		switch node.Type { | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		case "file": | 
					
						
							| 
									
										
										
										
											2016-04-10 16:51:16 +02:00
										 |  |  | 			if node.Content == nil { | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 				errs = append(errs, Error{TreeID: id, Err: errors.Errorf("file %q has nil blob list", node.Name)}) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:51:16 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-11 20:55:28 +02:00
										 |  |  | 			for b, blobID := range node.Content { | 
					
						
							|  |  |  | 				if blobID.IsNull() { | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 					errs = append(errs, Error{TreeID: id, Err: errors.Errorf("file %q blob %d has null ID", node.Name, b)}) | 
					
						
							| 
									
										
										
										
											2015-10-11 20:55:28 +02:00
										 |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2020-12-20 08:48:31 +01:00
										 |  |  | 				// Note that we do not use the blob size. The "obvious" check | 
					
						
							|  |  |  | 				// whether the sum of the blob sizes matches the file size | 
					
						
							|  |  |  | 				// unfortunately fails in some cases that are not resolveable | 
					
						
							|  |  |  | 				// by users, so we omit this check, see #1887 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				_, found := c.repo.LookupBlobSize(blobID, restic.DataBlob) | 
					
						
							| 
									
										
										
										
											2018-03-31 13:22:25 +02:00
										 |  |  | 				if !found { | 
					
						
							| 
									
										
										
										
											2020-11-06 23:32:31 +01:00
										 |  |  | 					debug.Log("tree %v references blob %v which isn't contained in index", id, blobID) | 
					
						
							|  |  |  | 					errs = append(errs, Error{TreeID: id, Err: errors.Errorf("file %q blob %v not found in index", node.Name, blobID)}) | 
					
						
							| 
									
										
										
										
											2018-03-31 13:22:25 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-11-07 00:07:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if c.trackUnused { | 
					
						
							|  |  |  | 				// loop a second time to keep the locked section as short as possible | 
					
						
							|  |  |  | 				c.blobRefs.Lock() | 
					
						
							|  |  |  | 				for _, blobID := range node.Content { | 
					
						
							|  |  |  | 					if blobID.IsNull() { | 
					
						
							|  |  |  | 						continue | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					h := restic.BlobHandle{ID: blobID, Type: restic.DataBlob} | 
					
						
							|  |  |  | 					c.blobRefs.M.Insert(h) | 
					
						
							|  |  |  | 					debug.Log("blob %v is referenced", blobID) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				c.blobRefs.Unlock() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		case "dir": | 
					
						
							|  |  |  | 			if node.Subtree == nil { | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 				errs = append(errs, Error{TreeID: id, Err: errors.Errorf("dir node %q has no subtree", node.Name)}) | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-10-11 18:46:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if node.Subtree.IsNull() { | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 				errs = append(errs, Error{TreeID: id, Err: errors.Errorf("dir node %q subtree id is null", node.Name)}) | 
					
						
							| 
									
										
										
										
											2015-10-11 18:46:26 +02:00
										 |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-04-10 16:51:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-20 20:46:57 +02:00
										 |  |  | 		case "symlink", "socket", "chardev", "dev", "fifo": | 
					
						
							| 
									
										
										
										
											2016-05-08 23:16:01 +02:00
										 |  |  | 			// nothing to check | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-10 16:51:16 +02:00
										 |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2016-09-01 21:20:03 +02:00
										 |  |  | 			errs = append(errs, Error{TreeID: id, Err: errors.Errorf("node %q with invalid type %q", node.Name, node.Type)}) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:51:16 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if node.Name == "" { | 
					
						
							|  |  |  | 			errs = append(errs, Error{TreeID: id, Err: errors.New("node with empty name")}) | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return errs | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // UnusedBlobs returns all blobs that have never been referenced. | 
					
						
							| 
									
										
										
										
											2020-11-06 23:41:04 +01:00
										 |  |  | func (c *Checker) UnusedBlobs(ctx context.Context) (blobs restic.BlobHandles) { | 
					
						
							| 
									
										
										
										
											2020-11-07 00:07:32 +01:00
										 |  |  | 	if !c.trackUnused { | 
					
						
							|  |  |  | 		panic("only works when tracking blob references") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-07-12 16:42:22 +02:00
										 |  |  | 	c.blobRefs.Lock() | 
					
						
							|  |  |  | 	defer c.blobRefs.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-13 18:34:55 +02:00
										 |  |  | 	debug.Log("checking %d blobs", len(c.blobRefs.M)) | 
					
						
							| 
									
										
										
										
											2020-11-06 23:41:04 +01:00
										 |  |  | 	ctx, cancel := context.WithCancel(ctx) | 
					
						
							|  |  |  | 	defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for blob := range c.repo.Index().Each(ctx) { | 
					
						
							|  |  |  | 		h := restic.BlobHandle{ID: blob.ID, Type: blob.Type} | 
					
						
							|  |  |  | 		if !c.blobRefs.M.Has(h) { | 
					
						
							|  |  |  | 			debug.Log("blob %v not referenced", h) | 
					
						
							|  |  |  | 			blobs = append(blobs, h) | 
					
						
							| 
									
										
										
										
											2015-07-11 16:00:49 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return blobs | 
					
						
							| 
									
										
										
										
											2015-06-29 00:22:25 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-07-12 17:09:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 17:29:31 +01:00
										 |  |  | // CountPacks returns the number of packs in the repository. | 
					
						
							|  |  |  | func (c *Checker) CountPacks() uint64 { | 
					
						
							|  |  |  | 	return uint64(len(c.packs)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 00:38:14 -05:00
										 |  |  | // GetPacks returns IDSet of packs in the repository | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | func (c *Checker) GetPacks() map[restic.ID]int64 { | 
					
						
							| 
									
										
										
										
											2018-01-02 00:38:14 -05:00
										 |  |  | 	return c.packs | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | // checkPack reads a pack and checks the integrity of all blobs. | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | func checkPack(ctx context.Context, r restic.Repository, id restic.ID, size int64) error { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 	debug.Log("checking pack %v", id) | 
					
						
							| 
									
										
										
										
											2020-08-16 11:16:38 +02:00
										 |  |  | 	h := restic.Handle{Type: restic.PackFile, Name: id.String()} | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	packfile, hash, realSize, err := repository.DownloadAndHash(ctx, r.Backend(), h) | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-02-11 22:41:59 -05:00
										 |  |  | 		return errors.Wrap(err, "checkPack") | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							| 
									
										
										
										
											2017-10-25 12:03:55 -04:00
										 |  |  | 		_ = packfile.Close() | 
					
						
							|  |  |  | 		_ = os.Remove(packfile.Name()) | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 	debug.Log("hash for pack %v is %v", id, hash) | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-04 22:54:26 +00:00
										 |  |  | 	if !hash.Equal(id) { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 		debug.Log("Pack ID does not match, want %v, got %v", id, hash) | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 		return errors.Errorf("Pack ID does not match, want %v, got %v", id.Str(), hash.Str()) | 
					
						
							| 
									
										
										
										
											2016-02-04 22:54:26 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	if realSize != size { | 
					
						
							|  |  |  | 		debug.Log("Pack size does not match, want %v, got %v", size, realSize) | 
					
						
							|  |  |  | 		return errors.Errorf("Pack size does not match, want %v, got %v", size, realSize) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-16 04:57:43 +01:00
										 |  |  | 	blobs, hdrSize, err := pack.List(r.Key(), packfile, size) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var errs []error | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 	var buf []byte | 
					
						
							| 
									
										
										
										
											2020-11-16 04:57:43 +01:00
										 |  |  | 	sizeFromBlobs := uint(hdrSize) | 
					
						
							| 
									
										
										
										
											2020-11-01 16:30:20 +01:00
										 |  |  | 	idx := r.Index() | 
					
						
							| 
									
										
										
										
											2016-08-25 21:08:16 +02:00
										 |  |  | 	for i, blob := range blobs { | 
					
						
							| 
									
										
										
										
											2020-11-16 04:57:43 +01:00
										 |  |  | 		sizeFromBlobs += blob.Length | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 		debug.Log("  check blob %d: %v", i, blob) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		buf = buf[:cap(buf)] | 
					
						
							|  |  |  | 		if uint(len(buf)) < blob.Length { | 
					
						
							|  |  |  | 			buf = make([]byte, blob.Length) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		buf = buf[:blob.Length] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		_, err := packfile.Seek(int64(blob.Offset), 0) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return errors.Errorf("Seek(%v): %v", blob.Offset, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		_, err = io.ReadFull(packfile, buf) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 			debug.Log("  error loading blob %v: %v", blob.ID, err) | 
					
						
							| 
									
										
										
										
											2017-02-06 21:17:52 +01:00
										 |  |  | 			errs = append(errs, errors.Errorf("blob %v: %v", i, err)) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-29 11:33:57 +01:00
										 |  |  | 		nonce, ciphertext := buf[:r.Key().NonceSize()], buf[r.Key().NonceSize():] | 
					
						
							|  |  |  | 		plaintext, err := r.Key().Open(ciphertext[:0], nonce, ciphertext, nil) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 			debug.Log("  error decrypting blob %v: %v", blob.ID, err) | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 			errs = append(errs, errors.Errorf("blob %v: %v", i, err)) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-29 11:33:57 +01:00
										 |  |  | 		hash := restic.Hash(plaintext) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 		if !hash.Equal(blob.ID) { | 
					
						
							| 
									
										
										
										
											2018-01-25 20:49:41 +01:00
										 |  |  | 			debug.Log("  Blob ID does not match, want %v, got %v", blob.ID, hash) | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 			errs = append(errs, errors.Errorf("Blob ID does not match, want %v, got %v", blob.ID.Str(), hash.Str())) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-11-01 16:30:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 		// Check if blob is contained in index and position is correct | 
					
						
							| 
									
										
										
										
											2020-11-01 16:30:20 +01:00
										 |  |  | 		idxHas := false | 
					
						
							| 
									
										
										
										
											2020-11-05 22:18:00 +01:00
										 |  |  | 		for _, pb := range idx.Lookup(blob.BlobHandle) { | 
					
						
							| 
									
										
										
										
											2020-11-10 14:35:59 +01:00
										 |  |  | 			if pb.PackID == id && pb.Offset == blob.Offset && pb.Length == blob.Length { | 
					
						
							| 
									
										
										
										
											2020-11-01 16:30:20 +01:00
										 |  |  | 				idxHas = true | 
					
						
							|  |  |  | 				break | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if !idxHas { | 
					
						
							| 
									
										
										
										
											2020-11-10 14:35:59 +01:00
										 |  |  | 			errs = append(errs, errors.Errorf("Blob %v is not contained in index or position is incorrect", blob.ID.Str())) | 
					
						
							| 
									
										
										
										
											2020-11-01 16:30:20 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-16 04:57:43 +01:00
										 |  |  | 	if int64(sizeFromBlobs) != size { | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 		debug.Log("Pack size does not match, want %v, got %v", size, sizeFromBlobs) | 
					
						
							|  |  |  | 		errs = append(errs, errors.Errorf("Pack size does not match, want %v, got %v", size, sizeFromBlobs)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 	if len(errs) > 0 { | 
					
						
							| 
									
										
										
										
											2016-08-21 17:48:36 +02:00
										 |  |  | 		return errors.Errorf("pack %v contains %v errors: %v", id.Str(), len(errs), errs) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ReadData loads all data from the repository and checks the integrity. | 
					
						
							| 
									
										
										
										
											2020-11-08 21:03:59 +01:00
										 |  |  | func (c *Checker) ReadData(ctx context.Context, errChan chan<- error) { | 
					
						
							|  |  |  | 	c.ReadPacks(ctx, c.packs, nil, errChan) | 
					
						
							| 
									
										
										
										
											2018-01-02 00:38:14 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ReadPacks loads data from specified packs and checks the integrity. | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | func (c *Checker) ReadPacks(ctx context.Context, packs map[restic.ID]int64, p *progress.Counter, errChan chan<- error) { | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 	defer close(errChan) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 	g, ctx := errgroup.WithContext(ctx) | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	type packsize struct { | 
					
						
							|  |  |  | 		id   restic.ID | 
					
						
							|  |  |  | 		size int64 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ch := make(chan packsize) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 	// run workers | 
					
						
							|  |  |  | 	for i := 0; i < defaultParallelism; i++ { | 
					
						
							|  |  |  | 		g.Go(func() error { | 
					
						
							|  |  |  | 			for { | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 				var ps packsize | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 				var ok bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				select { | 
					
						
							|  |  |  | 				case <-ctx.Done(): | 
					
						
							|  |  |  | 					return nil | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 				case ps, ok = <-ch: | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 					if !ok { | 
					
						
							|  |  |  | 						return nil | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 				err := checkPack(ctx, c.repo, ps.id, ps.size) | 
					
						
							| 
									
										
										
										
											2020-11-04 14:11:29 +01:00
										 |  |  | 				p.Add(1) | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 				if err == nil { | 
					
						
							|  |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				select { | 
					
						
							|  |  |  | 				case <-ctx.Done(): | 
					
						
							|  |  |  | 					return nil | 
					
						
							|  |  |  | 				case errChan <- err: | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2015-12-06 17:09:06 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-12-06 17:09:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 00:38:14 -05:00
										 |  |  | 	// push packs to ch | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 	for pack, size := range packs { | 
					
						
							| 
									
										
										
										
											2018-01-02 00:38:14 -05:00
										 |  |  | 		select { | 
					
						
							| 
									
										
										
										
											2020-11-10 08:16:47 +01:00
										 |  |  | 		case ch <- packsize{id: pack, size: size}: | 
					
						
							| 
									
										
										
										
											2018-01-02 00:38:14 -05:00
										 |  |  | 		case <-ctx.Done(): | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	close(ch) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-21 17:25:36 +01:00
										 |  |  | 	err := g.Wait() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case <-ctx.Done(): | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		case errChan <- err: | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-12-06 17:09:06 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-12-02 22:40:36 +01:00
										 |  |  | } |