mirror of
https://github.com/octoleo/restic.git
synced 2025-01-05 16:12:29 +00:00
debug: Remove manual Str() call Log()
This commit is contained in:
parent
ed99f53786
commit
663c57ab4d
@ -182,11 +182,11 @@ type Finder struct {
|
|||||||
|
|
||||||
func (f *Finder) findInTree(ctx context.Context, treeID restic.ID, prefix string) error {
|
func (f *Finder) findInTree(ctx context.Context, treeID restic.ID, prefix string) error {
|
||||||
if f.notfound.Has(treeID) {
|
if f.notfound.Has(treeID) {
|
||||||
debug.Log("%v skipping tree %v, has already been checked", prefix, treeID.Str())
|
debug.Log("%v skipping tree %v, has already been checked", prefix, treeID)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("%v checking tree %v\n", prefix, treeID.Str())
|
debug.Log("%v checking tree %v\n", prefix, treeID)
|
||||||
|
|
||||||
tree, err := f.repo.LoadTree(ctx, treeID)
|
tree, err := f.repo.LoadTree(ctx, treeID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -186,7 +186,7 @@ func pruneRepository(gopts GlobalOptions, repo restic.Repository) error {
|
|||||||
bar = newProgressMax(!gopts.Quiet, uint64(len(snapshots)), "snapshots")
|
bar = newProgressMax(!gopts.Quiet, uint64(len(snapshots)), "snapshots")
|
||||||
bar.Start()
|
bar.Start()
|
||||||
for _, sn := range snapshots {
|
for _, sn := range snapshots {
|
||||||
debug.Log("process snapshot %v", sn.ID().Str())
|
debug.Log("process snapshot %v", sn.ID())
|
||||||
|
|
||||||
err = restic.FindUsedBlobs(ctx, repo, *sn.Tree, usedBlobs, seenBlobs)
|
err = restic.FindUsedBlobs(ctx, repo, *sn.Tree, usedBlobs, seenBlobs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -197,7 +197,7 @@ func pruneRepository(gopts GlobalOptions, repo restic.Repository) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("processed snapshot %v", sn.ID().Str())
|
debug.Log("processed snapshot %v", sn.ID())
|
||||||
bar.Report(restic.Stat{Blobs: 1})
|
bar.Report(restic.Stat{Blobs: 1})
|
||||||
}
|
}
|
||||||
bar.Done()
|
bar.Done()
|
||||||
|
@ -82,7 +82,7 @@ func changeTags(ctx context.Context, repo *repository.Repository, sn *restic.Sna
|
|||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("new snapshot saved as %v", id.Str())
|
debug.Log("new snapshot saved as %v", id)
|
||||||
|
|
||||||
if err = repo.Flush(ctx); err != nil {
|
if err = repo.Flush(ctx); err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
|
@ -59,9 +59,9 @@ func (r *Reader) Archive(ctx context.Context, name string, rd io.Reader, p *rest
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, restic.ID{}, err
|
return nil, restic.ID{}, err
|
||||||
}
|
}
|
||||||
debug.Log("saved blob %v (%d bytes)\n", id.Str(), chunk.Length)
|
debug.Log("saved blob %v (%d bytes)\n", id, chunk.Length)
|
||||||
} else {
|
} else {
|
||||||
debug.Log("blob %v already saved in the repo\n", id.Str())
|
debug.Log("blob %v already saved in the repo\n", id)
|
||||||
}
|
}
|
||||||
|
|
||||||
freeBuf(chunk.Data)
|
freeBuf(chunk.Data)
|
||||||
@ -94,14 +94,14 @@ func (r *Reader) Archive(ctx context.Context, name string, rd io.Reader, p *rest
|
|||||||
return nil, restic.ID{}, err
|
return nil, restic.ID{}, err
|
||||||
}
|
}
|
||||||
sn.Tree = &treeID
|
sn.Tree = &treeID
|
||||||
debug.Log("tree saved as %v", treeID.Str())
|
debug.Log("tree saved as %v", treeID)
|
||||||
|
|
||||||
id, err := repo.SaveJSONUnpacked(ctx, restic.SnapshotFile, sn)
|
id, err := repo.SaveJSONUnpacked(ctx, restic.SnapshotFile, sn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, restic.ID{}, err
|
return nil, restic.ID{}, err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("snapshot saved as %v", id.Str())
|
debug.Log("snapshot saved as %v", id)
|
||||||
|
|
||||||
err = repo.Flush(ctx)
|
err = repo.Flush(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -95,20 +95,20 @@ func (arch *Archiver) isKnownBlob(id restic.ID, t restic.BlobType) bool {
|
|||||||
|
|
||||||
// Save stores a blob read from rd in the repository.
|
// Save stores a blob read from rd in the repository.
|
||||||
func (arch *Archiver) Save(ctx context.Context, t restic.BlobType, data []byte, id restic.ID) error {
|
func (arch *Archiver) Save(ctx context.Context, t restic.BlobType, data []byte, id restic.ID) error {
|
||||||
debug.Log("Save(%v, %v)\n", t, id.Str())
|
debug.Log("Save(%v, %v)\n", t, id)
|
||||||
|
|
||||||
if arch.isKnownBlob(id, restic.DataBlob) {
|
if arch.isKnownBlob(id, restic.DataBlob) {
|
||||||
debug.Log("blob %v is known\n", id.Str())
|
debug.Log("blob %v is known\n", id)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := arch.repo.SaveBlob(ctx, t, data, id)
|
_, err := arch.repo.SaveBlob(ctx, t, data, id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log("Save(%v, %v): error %v\n", t, id.Str(), err)
|
debug.Log("Save(%v, %v): error %v\n", t, id, err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("Save(%v, %v): new blob\n", t, id.Str())
|
debug.Log("Save(%v, %v): new blob\n", t, id)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -170,7 +170,7 @@ func (arch *Archiver) saveChunk(ctx context.Context, chunk chunker.Chunk, p *res
|
|||||||
err := arch.Save(ctx, restic.DataBlob, chunk.Data, id)
|
err := arch.Save(ctx, restic.DataBlob, chunk.Data, id)
|
||||||
// TODO handle error
|
// TODO handle error
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log("Save(%v) failed: %v", id.Str(), err)
|
debug.Log("Save(%v) failed: %v", id, err)
|
||||||
fmt.Printf("\nerror while saving data to the repo: %+v\n", err)
|
fmt.Printf("\nerror while saving data to the repo: %+v\n", err)
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
@ -204,7 +204,7 @@ func updateNodeContent(node *restic.Node, results []saveResult) error {
|
|||||||
node.Content[i] = b.id
|
node.Content[i] = b.id
|
||||||
bytes += b.bytes
|
bytes += b.bytes
|
||||||
|
|
||||||
debug.Log(" adding blob %s, %d bytes", b.id.Str(), b.bytes)
|
debug.Log(" adding blob %s, %d bytes", b.id, b.bytes)
|
||||||
}
|
}
|
||||||
|
|
||||||
if bytes != node.Size {
|
if bytes != node.Size {
|
||||||
@ -304,7 +304,7 @@ func (arch *Archiver) fileWorker(ctx context.Context, wg *sync.WaitGroup, p *res
|
|||||||
contentMissing := false
|
contentMissing := false
|
||||||
for _, blob := range oldNode.Content {
|
for _, blob := range oldNode.Content {
|
||||||
if !arch.repo.Index().Has(blob, restic.DataBlob) {
|
if !arch.repo.Index().Has(blob, restic.DataBlob) {
|
||||||
debug.Log(" %v not using old data, %v is missing", e.Path(), blob.Str())
|
debug.Log(" %v not using old data, %v is missing", e.Path(), blob)
|
||||||
contentMissing = true
|
contentMissing = true
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@ -437,7 +437,7 @@ func (arch *Archiver) dirWorker(ctx context.Context, wg *sync.WaitGroup, p *rest
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
debug.Log("save tree for %s: %v", dir.Path(), id.Str())
|
debug.Log("save tree for %s: %v", dir.Path(), id)
|
||||||
if id.IsNull() {
|
if id.IsNull() {
|
||||||
panic("invalid null subtree restic.ID return from SaveTreeJSON()")
|
panic("invalid null subtree restic.ID return from SaveTreeJSON()")
|
||||||
}
|
}
|
||||||
@ -770,7 +770,7 @@ func (arch *Archiver) Snapshot(ctx context.Context, p *restic.Progress, paths, t
|
|||||||
|
|
||||||
// receive the top-level tree
|
// receive the top-level tree
|
||||||
root := (<-resCh).(*restic.Node)
|
root := (<-resCh).(*restic.Node)
|
||||||
debug.Log("root node received: %v", root.Subtree.Str())
|
debug.Log("root node received: %v", root.Subtree)
|
||||||
sn.Tree = root.Subtree
|
sn.Tree = root.Subtree
|
||||||
|
|
||||||
// load top-level tree again to see if it is empty
|
// load top-level tree again to see if it is empty
|
||||||
@ -798,7 +798,7 @@ func (arch *Archiver) Snapshot(ctx context.Context, p *restic.Progress, paths, t
|
|||||||
return nil, restic.ID{}, err
|
return nil, restic.ID{}, err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("saved snapshot %v", id.Str())
|
debug.Log("saved snapshot %v", id)
|
||||||
|
|
||||||
return sn, id, nil
|
return sn, id, nil
|
||||||
}
|
}
|
||||||
|
@ -90,7 +90,7 @@ func (c *Checker) LoadIndex(ctx context.Context) (hints []error, errs []error) {
|
|||||||
debug.Log("worker got index %v", id)
|
debug.Log("worker got index %v", id)
|
||||||
idx, err := repository.LoadIndexWithDecoder(ctx, c.repo, id, repository.DecodeIndex)
|
idx, err := repository.LoadIndexWithDecoder(ctx, c.repo, id, repository.DecodeIndex)
|
||||||
if errors.Cause(err) == repository.ErrOldIndexFormat {
|
if errors.Cause(err) == repository.ErrOldIndexFormat {
|
||||||
debug.Log("index %v has old format", id.Str())
|
debug.Log("index %v has old format", id)
|
||||||
hints = append(hints, ErrOldIndexFormat{id})
|
hints = append(hints, ErrOldIndexFormat{id})
|
||||||
|
|
||||||
idx, err = repository.LoadIndexWithDecoder(ctx, c.repo, id, repository.DecodeOldIndex)
|
idx, err = repository.LoadIndexWithDecoder(ctx, c.repo, id, repository.DecodeOldIndex)
|
||||||
@ -158,7 +158,7 @@ func (c *Checker) LoadIndex(ctx context.Context) (hints []error, errs []error) {
|
|||||||
|
|
||||||
debug.Log("checking for duplicate packs")
|
debug.Log("checking for duplicate packs")
|
||||||
for packID := range c.packs {
|
for packID := range c.packs {
|
||||||
debug.Log(" check pack %v: contained in %d indexes", packID.Str(), len(packToIndex[packID]))
|
debug.Log(" check pack %v: contained in %d indexes", packID, len(packToIndex[packID]))
|
||||||
if len(packToIndex[packID]) > 1 {
|
if len(packToIndex[packID]) > 1 {
|
||||||
hints = append(hints, ErrDuplicatePacks{
|
hints = append(hints, ErrDuplicatePacks{
|
||||||
PackID: packID,
|
PackID: packID,
|
||||||
@ -247,12 +247,12 @@ func (e Error) Error() string {
|
|||||||
func loadTreeFromSnapshot(ctx context.Context, repo restic.Repository, id restic.ID) (restic.ID, error) {
|
func loadTreeFromSnapshot(ctx context.Context, repo restic.Repository, id restic.ID) (restic.ID, error) {
|
||||||
sn, err := restic.LoadSnapshot(ctx, repo, id)
|
sn, err := restic.LoadSnapshot(ctx, repo, id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log("error loading snapshot %v: %v", id.Str(), err)
|
debug.Log("error loading snapshot %v: %v", id, err)
|
||||||
return restic.ID{}, err
|
return restic.ID{}, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if sn.Tree == nil {
|
if sn.Tree == nil {
|
||||||
debug.Log("snapshot %v has no tree", id.Str())
|
debug.Log("snapshot %v has no tree", id)
|
||||||
return restic.ID{}, errors.Errorf("snapshot %v has no tree", id)
|
return restic.ID{}, errors.Errorf("snapshot %v has no tree", id)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -277,7 +277,7 @@ func loadSnapshotTreeIDs(ctx context.Context, repo restic.Repository) (restic.ID
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("load snapshot %v", id.Str())
|
debug.Log("load snapshot %v", id)
|
||||||
|
|
||||||
treeID, err := loadTreeFromSnapshot(ctx, repo, id)
|
treeID, err := loadTreeFromSnapshot(ctx, repo, id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -287,7 +287,7 @@ func loadSnapshotTreeIDs(ctx context.Context, repo restic.Repository) (restic.ID
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("snapshot %v has tree %v", id.Str(), treeID.Str())
|
debug.Log("snapshot %v has tree %v", id, treeID)
|
||||||
trees.Lock()
|
trees.Lock()
|
||||||
trees.IDs = append(trees.IDs, treeID)
|
trees.IDs = append(trees.IDs, treeID)
|
||||||
trees.Unlock()
|
trees.Unlock()
|
||||||
@ -345,16 +345,16 @@ func loadTreeWorker(ctx context.Context, repo restic.Repository,
|
|||||||
if !ok {
|
if !ok {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
debug.Log("load tree %v", treeID.Str())
|
debug.Log("load tree %v", treeID)
|
||||||
|
|
||||||
tree, err := repo.LoadTree(ctx, treeID)
|
tree, err := repo.LoadTree(ctx, treeID)
|
||||||
debug.Log("load tree %v (%v) returned err: %v", tree, treeID.Str(), err)
|
debug.Log("load tree %v (%v) returned err: %v", tree, treeID, err)
|
||||||
job = treeJob{ID: treeID, error: err, Tree: tree}
|
job = treeJob{ID: treeID, error: err, Tree: tree}
|
||||||
outCh = out
|
outCh = out
|
||||||
inCh = nil
|
inCh = nil
|
||||||
|
|
||||||
case outCh <- job:
|
case outCh <- job:
|
||||||
debug.Log("sent tree %v", job.ID.Str())
|
debug.Log("sent tree %v", job.ID)
|
||||||
outCh = nil
|
outCh = nil
|
||||||
inCh = in
|
inCh = in
|
||||||
}
|
}
|
||||||
@ -394,14 +394,14 @@ func (c *Checker) checkTreeWorker(ctx context.Context, in <-chan treeJob, out ch
|
|||||||
alreadyChecked = true
|
alreadyChecked = true
|
||||||
}
|
}
|
||||||
c.blobRefs.M[id]++
|
c.blobRefs.M[id]++
|
||||||
debug.Log("tree %v refcount %d", job.ID.Str(), c.blobRefs.M[id])
|
debug.Log("tree %v refcount %d", job.ID, c.blobRefs.M[id])
|
||||||
c.blobRefs.Unlock()
|
c.blobRefs.Unlock()
|
||||||
|
|
||||||
if alreadyChecked {
|
if alreadyChecked {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("check tree %v (tree %v, err %v)", job.ID.Str(), job.Tree, job.error)
|
debug.Log("check tree %v (tree %v, err %v)", job.ID, job.Tree, job.error)
|
||||||
|
|
||||||
var errs []error
|
var errs []error
|
||||||
if job.error != nil {
|
if job.error != nil {
|
||||||
@ -411,7 +411,7 @@ func (c *Checker) checkTreeWorker(ctx context.Context, in <-chan treeJob, out ch
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(errs) > 0 {
|
if len(errs) > 0 {
|
||||||
debug.Log("checked tree %v: %v errors", job.ID.Str(), len(errs))
|
debug.Log("checked tree %v: %v errors", job.ID, len(errs))
|
||||||
treeError = TreeError{ID: job.ID, Errors: errs}
|
treeError = TreeError{ID: job.ID, Errors: errs}
|
||||||
outCh = out
|
outCh = out
|
||||||
inCh = nil
|
inCh = nil
|
||||||
@ -473,23 +473,23 @@ func filterTrees(ctx context.Context, backlog restic.IDs, loaderChan chan<- rest
|
|||||||
|
|
||||||
outstandingLoadTreeJobs--
|
outstandingLoadTreeJobs--
|
||||||
|
|
||||||
debug.Log("input job tree %v", j.ID.Str())
|
debug.Log("input job tree %v", j.ID)
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
if j.error != nil {
|
if j.error != nil {
|
||||||
debug.Log("received job with error: %v (tree %v, ID %v)", j.error, j.Tree, j.ID.Str())
|
debug.Log("received job with error: %v (tree %v, ID %v)", j.error, j.Tree, j.ID)
|
||||||
} else if j.Tree == nil {
|
} else if j.Tree == nil {
|
||||||
debug.Log("received job with nil tree pointer: %v (ID %v)", j.error, j.ID.Str())
|
debug.Log("received job with nil tree pointer: %v (ID %v)", j.error, j.ID)
|
||||||
err = errors.New("tree is nil and error is nil")
|
err = errors.New("tree is nil and error is nil")
|
||||||
} else {
|
} else {
|
||||||
debug.Log("subtrees for tree %v: %v", j.ID.Str(), j.Tree.Subtrees())
|
debug.Log("subtrees for tree %v: %v", j.ID, j.Tree.Subtrees())
|
||||||
for _, id := range j.Tree.Subtrees() {
|
for _, id := range j.Tree.Subtrees() {
|
||||||
if id.IsNull() {
|
if id.IsNull() {
|
||||||
// We do not need to raise this error here, it is
|
// We do not need to raise this error here, it is
|
||||||
// checked when the tree is checked. Just make sure
|
// checked when the tree is checked. Just make sure
|
||||||
// that we do not add any null IDs to the backlog.
|
// that we do not add any null IDs to the backlog.
|
||||||
debug.Log("tree %v has nil subtree", j.ID.Str())
|
debug.Log("tree %v has nil subtree", j.ID)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
backlog = append(backlog, id)
|
backlog = append(backlog, id)
|
||||||
@ -506,7 +506,7 @@ func filterTrees(ctx context.Context, backlog restic.IDs, loaderChan chan<- rest
|
|||||||
inCh = nil
|
inCh = nil
|
||||||
|
|
||||||
case outCh <- job:
|
case outCh <- job:
|
||||||
debug.Log("tree sent to check: %v", job.ID.Str())
|
debug.Log("tree sent to check: %v", job.ID)
|
||||||
outCh = nil
|
outCh = nil
|
||||||
inCh = in
|
inCh = in
|
||||||
}
|
}
|
||||||
@ -547,7 +547,7 @@ func (c *Checker) Structure(ctx context.Context, errChan chan<- error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (c *Checker) checkTree(id restic.ID, tree *restic.Tree) (errs []error) {
|
func (c *Checker) checkTree(id restic.ID, tree *restic.Tree) (errs []error) {
|
||||||
debug.Log("checking tree %v", id.Str())
|
debug.Log("checking tree %v", id)
|
||||||
|
|
||||||
var blobs []restic.ID
|
var blobs []restic.ID
|
||||||
|
|
||||||
@ -591,11 +591,11 @@ func (c *Checker) checkTree(id restic.ID, tree *restic.Tree) (errs []error) {
|
|||||||
for _, blobID := range blobs {
|
for _, blobID := range blobs {
|
||||||
c.blobRefs.Lock()
|
c.blobRefs.Lock()
|
||||||
c.blobRefs.M[blobID]++
|
c.blobRefs.M[blobID]++
|
||||||
debug.Log("blob %v refcount %d", blobID.Str(), c.blobRefs.M[blobID])
|
debug.Log("blob %v refcount %d", blobID, c.blobRefs.M[blobID])
|
||||||
c.blobRefs.Unlock()
|
c.blobRefs.Unlock()
|
||||||
|
|
||||||
if !c.blobs.Has(blobID) {
|
if !c.blobs.Has(blobID) {
|
||||||
debug.Log("tree %v references blob %v which isn't contained in index", id.Str(), blobID.Str())
|
debug.Log("tree %v references blob %v which isn't contained in index", id, blobID)
|
||||||
|
|
||||||
errs = append(errs, Error{TreeID: id, BlobID: blobID, Err: errors.New("not found in index")})
|
errs = append(errs, Error{TreeID: id, BlobID: blobID, Err: errors.New("not found in index")})
|
||||||
}
|
}
|
||||||
@ -612,7 +612,7 @@ func (c *Checker) UnusedBlobs() (blobs restic.IDs) {
|
|||||||
debug.Log("checking %d blobs", len(c.blobs))
|
debug.Log("checking %d blobs", len(c.blobs))
|
||||||
for id := range c.blobs {
|
for id := range c.blobs {
|
||||||
if c.blobRefs.M[id] == 0 {
|
if c.blobRefs.M[id] == 0 {
|
||||||
debug.Log("blob %v not referenced", id.Str())
|
debug.Log("blob %v not referenced", id)
|
||||||
blobs = append(blobs, id)
|
blobs = append(blobs, id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -627,7 +627,7 @@ func (c *Checker) CountPacks() uint64 {
|
|||||||
|
|
||||||
// checkPack reads a pack and checks the integrity of all blobs.
|
// checkPack reads a pack and checks the integrity of all blobs.
|
||||||
func checkPack(ctx context.Context, r restic.Repository, id restic.ID) error {
|
func checkPack(ctx context.Context, r restic.Repository, id restic.ID) error {
|
||||||
debug.Log("checking pack %v", id.Str())
|
debug.Log("checking pack %v", id)
|
||||||
h := restic.Handle{Type: restic.DataFile, Name: id.String()}
|
h := restic.Handle{Type: restic.DataFile, Name: id.String()}
|
||||||
|
|
||||||
rd, err := r.Backend().Load(ctx, h, 0, 0)
|
rd, err := r.Backend().Load(ctx, h, 0, 0)
|
||||||
@ -656,10 +656,10 @@ func checkPack(ctx context.Context, r restic.Repository, id restic.ID) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
hash := restic.IDFromHash(hrd.Sum(nil))
|
hash := restic.IDFromHash(hrd.Sum(nil))
|
||||||
debug.Log("hash for pack %v is %v", id.Str(), hash.Str())
|
debug.Log("hash for pack %v is %v", id, hash)
|
||||||
|
|
||||||
if !hash.Equal(id) {
|
if !hash.Equal(id) {
|
||||||
debug.Log("Pack ID does not match, want %v, got %v", id.Str(), hash.Str())
|
debug.Log("Pack ID does not match, want %v, got %v", id, hash)
|
||||||
return errors.Errorf("Pack ID does not match, want %v, got %v", id.Str(), hash.Str())
|
return errors.Errorf("Pack ID does not match, want %v, got %v", id.Str(), hash.Str())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -686,7 +686,7 @@ func checkPack(ctx context.Context, r restic.Repository, id restic.ID) error {
|
|||||||
|
|
||||||
_, err = io.ReadFull(packfile, buf)
|
_, err = io.ReadFull(packfile, buf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log(" error loading blob %v: %v", blob.ID.Str(), err)
|
debug.Log(" error loading blob %v: %v", blob.ID, err)
|
||||||
errs = append(errs, errors.Errorf("blob %v: %v", i, err))
|
errs = append(errs, errors.Errorf("blob %v: %v", i, err))
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@ -694,14 +694,14 @@ func checkPack(ctx context.Context, r restic.Repository, id restic.ID) error {
|
|||||||
nonce, ciphertext := buf[:r.Key().NonceSize()], buf[r.Key().NonceSize():]
|
nonce, ciphertext := buf[:r.Key().NonceSize()], buf[r.Key().NonceSize():]
|
||||||
plaintext, err := r.Key().Open(ciphertext[:0], nonce, ciphertext, nil)
|
plaintext, err := r.Key().Open(ciphertext[:0], nonce, ciphertext, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log(" error decrypting blob %v: %v", blob.ID.Str(), err)
|
debug.Log(" error decrypting blob %v: %v", blob.ID, err)
|
||||||
errs = append(errs, errors.Errorf("blob %v: %v", i, err))
|
errs = append(errs, errors.Errorf("blob %v: %v", i, err))
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
hash := restic.Hash(plaintext)
|
hash := restic.Hash(plaintext)
|
||||||
if !hash.Equal(blob.ID) {
|
if !hash.Equal(blob.ID) {
|
||||||
debug.Log(" Blob ID does not match, want %v, got %v", blob.ID.Str(), hash.Str())
|
debug.Log(" Blob ID does not match, want %v, got %v", blob.ID, hash)
|
||||||
errs = append(errs, errors.Errorf("Blob ID does not match, want %v, got %v", blob.ID.Str(), hash.Str()))
|
errs = append(errs, errors.Errorf("Blob ID does not match, want %v, got %v", blob.ID.Str(), hash.Str()))
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -34,10 +34,10 @@ func cleanupNodeName(name string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func newDir(ctx context.Context, root *Root, inode, parentInode uint64, node *restic.Node) (*dir, error) {
|
func newDir(ctx context.Context, root *Root, inode, parentInode uint64, node *restic.Node) (*dir, error) {
|
||||||
debug.Log("new dir for %v (%v)", node.Name, node.Subtree.Str())
|
debug.Log("new dir for %v (%v)", node.Name, node.Subtree)
|
||||||
tree, err := root.repo.LoadTree(ctx, *node.Subtree)
|
tree, err := root.repo.LoadTree(ctx, *node.Subtree)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log(" error loading tree %v: %v", node.Subtree.Str(), err)
|
debug.Log(" error loading tree %v: %v", node.Subtree, err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
items := make(map[string]*restic.Node)
|
items := make(map[string]*restic.Node)
|
||||||
@ -74,10 +74,10 @@ func replaceSpecialNodes(ctx context.Context, repo restic.Repository, node *rest
|
|||||||
}
|
}
|
||||||
|
|
||||||
func newDirFromSnapshot(ctx context.Context, root *Root, inode uint64, snapshot *restic.Snapshot) (*dir, error) {
|
func newDirFromSnapshot(ctx context.Context, root *Root, inode uint64, snapshot *restic.Snapshot) (*dir, error) {
|
||||||
debug.Log("new dir for snapshot %v (%v)", snapshot.ID().Str(), snapshot.Tree.Str())
|
debug.Log("new dir for snapshot %v (%v)", snapshot.ID(), snapshot.Tree)
|
||||||
tree, err := root.repo.LoadTree(ctx, *snapshot.Tree)
|
tree, err := root.repo.LoadTree(ctx, *snapshot.Tree)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log(" loadTree(%v) failed: %v", snapshot.ID().Str(), err)
|
debug.Log(" loadTree(%v) failed: %v", snapshot.ID(), err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
items := make(map[string]*restic.Node)
|
items := make(map[string]*restic.Node)
|
||||||
|
@ -90,7 +90,7 @@ type indexJSON struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func loadIndexJSON(ctx context.Context, repo restic.Repository, id restic.ID) (*indexJSON, error) {
|
func loadIndexJSON(ctx context.Context, repo restic.Repository, id restic.ID) (*indexJSON, error) {
|
||||||
debug.Log("process index %v\n", id.Str())
|
debug.Log("process index %v\n", id)
|
||||||
|
|
||||||
var idx indexJSON
|
var idx indexJSON
|
||||||
err := repo.LoadJSONUnpacked(ctx, restic.IndexFile, id, &idx)
|
err := repo.LoadJSONUnpacked(ctx, restic.IndexFile, id, &idx)
|
||||||
@ -116,7 +116,7 @@ func Load(ctx context.Context, repo restic.Repository, p *restic.Progress) (*Ind
|
|||||||
err := repo.List(ctx, restic.IndexFile, func(id restic.ID, size int64) error {
|
err := repo.List(ctx, restic.IndexFile, func(id restic.ID, size int64) error {
|
||||||
p.Report(restic.Stat{Blobs: 1})
|
p.Report(restic.Stat{Blobs: 1})
|
||||||
|
|
||||||
debug.Log("Load index %v", id.Str())
|
debug.Log("Load index %v", id)
|
||||||
idx, err := loadIndexJSON(ctx, repo, id)
|
idx, err := loadIndexJSON(ctx, repo, id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -125,7 +125,7 @@ func Load(ctx context.Context, repo restic.Repository, p *restic.Progress) (*Ind
|
|||||||
res := make(map[restic.ID]Pack)
|
res := make(map[restic.ID]Pack)
|
||||||
supersedes[id] = restic.NewIDSet()
|
supersedes[id] = restic.NewIDSet()
|
||||||
for _, sid := range idx.Supersedes {
|
for _, sid := range idx.Supersedes {
|
||||||
debug.Log(" index %v supersedes %v", id.Str(), sid)
|
debug.Log(" index %v supersedes %v", id, sid)
|
||||||
supersedes[id].Insert(sid)
|
supersedes[id].Insert(sid)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,7 +161,7 @@ func Load(ctx context.Context, repo restic.Repository, p *restic.Progress) (*Ind
|
|||||||
if _, ok := results[indexID]; !ok {
|
if _, ok := results[indexID]; !ok {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
debug.Log(" removing index %v, superseded by %v", indexID.Str(), superID.Str())
|
debug.Log(" removing index %v, superseded by %v", indexID, superID)
|
||||||
fmt.Fprintf(os.Stderr, "index %v can be removed, superseded by index %v\n", indexID.Str(), superID.Str())
|
fmt.Fprintf(os.Stderr, "index %v can be removed, superseded by index %v\n", indexID.Str(), superID.Str())
|
||||||
delete(results, indexID)
|
delete(results, indexID)
|
||||||
}
|
}
|
||||||
|
@ -398,7 +398,7 @@ func (idx *Index) SetID(id restic.ID) error {
|
|||||||
return errors.New("ID already set")
|
return errors.New("ID already set")
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("ID set to %v", id.Str())
|
debug.Log("ID set to %v", id)
|
||||||
idx.id = id
|
idx.id = id
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
@ -550,7 +550,7 @@ func DecodeOldIndex(buf []byte) (idx *Index, err error) {
|
|||||||
|
|
||||||
// LoadIndexWithDecoder loads the index and decodes it with fn.
|
// LoadIndexWithDecoder loads the index and decodes it with fn.
|
||||||
func LoadIndexWithDecoder(ctx context.Context, repo restic.Repository, id restic.ID, fn func([]byte) (*Index, error)) (idx *Index, err error) {
|
func LoadIndexWithDecoder(ctx context.Context, repo restic.Repository, id restic.ID, fn func([]byte) (*Index, error)) (idx *Index, err error) {
|
||||||
debug.Log("Loading index %v", id.Str())
|
debug.Log("Loading index %v", id)
|
||||||
|
|
||||||
buf, err := repo.LoadAndDecrypt(ctx, restic.IndexFile, id)
|
buf, err := repo.LoadAndDecrypt(ctx, restic.IndexFile, id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -246,7 +246,7 @@ func (mi *MasterIndex) RebuildIndex(packBlacklist restic.IDSet) (*Index, error)
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("adding index id %v to supersedes field", id.Str())
|
debug.Log("adding index id %v to supersedes field", id)
|
||||||
|
|
||||||
err = newIndex.AddToSupersedes(id)
|
err = newIndex.AddToSupersedes(id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -138,7 +138,7 @@ func (r *Repository) savePacker(ctx context.Context, t restic.BlobType, p *Packe
|
|||||||
|
|
||||||
// update blobs in the index
|
// update blobs in the index
|
||||||
for _, b := range p.Packer.Blobs() {
|
for _, b := range p.Packer.Blobs() {
|
||||||
debug.Log(" updating blob %v to pack %v", b.ID.Str(), id.Str())
|
debug.Log(" updating blob %v to pack %v", b.ID, id)
|
||||||
r.idx.Store(restic.PackedBlob{
|
r.idx.Store(restic.PackedBlob{
|
||||||
Blob: restic.Blob{
|
Blob: restic.Blob{
|
||||||
Type: b.Type,
|
Type: b.Type,
|
||||||
|
@ -48,7 +48,7 @@ func Repack(ctx context.Context, repo restic.Repository, packs restic.IDSet, kee
|
|||||||
}
|
}
|
||||||
|
|
||||||
hash := restic.IDFromHash(hrd.Sum(nil))
|
hash := restic.IDFromHash(hrd.Sum(nil))
|
||||||
debug.Log("pack %v loaded (%d bytes), hash %v", packID.Str(), packLength, hash.Str())
|
debug.Log("pack %v loaded (%d bytes), hash %v", packID, packLength, hash)
|
||||||
|
|
||||||
if !packID.Equal(hash) {
|
if !packID.Equal(hash) {
|
||||||
return nil, errors.Errorf("hash does not match id: want %v, got %v", packID, hash)
|
return nil, errors.Errorf("hash does not match id: want %v, got %v", packID, hash)
|
||||||
@ -64,7 +64,7 @@ func Repack(ctx context.Context, repo restic.Repository, packs restic.IDSet, kee
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("processing pack %v, blobs: %v", packID.Str(), len(blobs))
|
debug.Log("processing pack %v, blobs: %v", packID, len(blobs))
|
||||||
var buf []byte
|
var buf []byte
|
||||||
for _, entry := range blobs {
|
for _, entry := range blobs {
|
||||||
h := restic.BlobHandle{ID: entry.ID, Type: entry.Type}
|
h := restic.BlobHandle{ID: entry.ID, Type: entry.Type}
|
||||||
@ -109,7 +109,7 @@ func Repack(ctx context.Context, repo restic.Repository, packs restic.IDSet, kee
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log(" saved blob %v", entry.ID.Str())
|
debug.Log(" saved blob %v", entry.ID)
|
||||||
|
|
||||||
keepBlobs.Delete(h)
|
keepBlobs.Delete(h)
|
||||||
}
|
}
|
||||||
|
@ -66,7 +66,7 @@ func (r *Repository) PrefixLength(t restic.FileType) (int, error) {
|
|||||||
// LoadAndDecrypt loads and decrypts data identified by t and id from the
|
// LoadAndDecrypt loads and decrypts data identified by t and id from the
|
||||||
// backend.
|
// backend.
|
||||||
func (r *Repository) LoadAndDecrypt(ctx context.Context, t restic.FileType, id restic.ID) (buf []byte, err error) {
|
func (r *Repository) LoadAndDecrypt(ctx context.Context, t restic.FileType, id restic.ID) (buf []byte, err error) {
|
||||||
debug.Log("load %v with id %v", t, id.Str())
|
debug.Log("load %v with id %v", t, id)
|
||||||
|
|
||||||
h := restic.Handle{Type: t, Name: id.String()}
|
h := restic.Handle{Type: t, Name: id.String()}
|
||||||
buf, err = backend.LoadAll(ctx, r.be, h)
|
buf, err = backend.LoadAll(ctx, r.be, h)
|
||||||
@ -112,12 +112,12 @@ func (r *Repository) sortCachedPacks(blobs []restic.PackedBlob) []restic.PackedB
|
|||||||
// pack from the backend, the result is stored in plaintextBuf, which must be
|
// pack from the backend, the result is stored in plaintextBuf, which must be
|
||||||
// large enough to hold the complete blob.
|
// large enough to hold the complete blob.
|
||||||
func (r *Repository) loadBlob(ctx context.Context, id restic.ID, t restic.BlobType, plaintextBuf []byte) (int, error) {
|
func (r *Repository) loadBlob(ctx context.Context, id restic.ID, t restic.BlobType, plaintextBuf []byte) (int, error) {
|
||||||
debug.Log("load %v with id %v (buf len %v, cap %d)", t, id.Str(), len(plaintextBuf), cap(plaintextBuf))
|
debug.Log("load %v with id %v (buf len %v, cap %d)", t, id, len(plaintextBuf), cap(plaintextBuf))
|
||||||
|
|
||||||
// lookup packs
|
// lookup packs
|
||||||
blobs, found := r.idx.Lookup(id, t)
|
blobs, found := r.idx.Lookup(id, t)
|
||||||
if !found {
|
if !found {
|
||||||
debug.Log("id %v not found in index", id.Str())
|
debug.Log("id %v not found in index", id)
|
||||||
return 0, errors.Errorf("id %v not found in repository", id)
|
return 0, errors.Errorf("id %v not found in repository", id)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -126,7 +126,7 @@ func (r *Repository) loadBlob(ctx context.Context, id restic.ID, t restic.BlobTy
|
|||||||
|
|
||||||
var lastError error
|
var lastError error
|
||||||
for _, blob := range blobs {
|
for _, blob := range blobs {
|
||||||
debug.Log("blob %v/%v found: %v", t, id.Str(), blob)
|
debug.Log("blob %v/%v found: %v", t, id, blob)
|
||||||
|
|
||||||
if blob.Type != t {
|
if blob.Type != t {
|
||||||
debug.Log("blob %v has wrong block type, want %v", blob, t)
|
debug.Log("blob %v has wrong block type, want %v", blob, t)
|
||||||
@ -206,7 +206,7 @@ func (r *Repository) SaveAndEncrypt(ctx context.Context, t restic.BlobType, data
|
|||||||
id = &hashedID
|
id = &hashedID
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("save id %v (%v, %d bytes)", id.Str(), t, len(data))
|
debug.Log("save id %v (%v, %d bytes)", id, t, len(data))
|
||||||
|
|
||||||
// get buf from the pool
|
// get buf from the pool
|
||||||
ciphertext := getBuf()
|
ciphertext := getBuf()
|
||||||
@ -353,7 +353,7 @@ func (r *Repository) saveIndex(ctx context.Context, indexes ...*Index) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("Saved index %d as %v", i, sid.Str())
|
debug.Log("Saved index %d as %v", i, sid)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
@ -575,7 +575,7 @@ func (r *Repository) Close() error {
|
|||||||
// be large enough to hold the encrypted blob, since it is used as scratch
|
// be large enough to hold the encrypted blob, since it is used as scratch
|
||||||
// space.
|
// space.
|
||||||
func (r *Repository) LoadBlob(ctx context.Context, t restic.BlobType, id restic.ID, buf []byte) (int, error) {
|
func (r *Repository) LoadBlob(ctx context.Context, t restic.BlobType, id restic.ID, buf []byte) (int, error) {
|
||||||
debug.Log("load blob %v into buf (len %v, cap %v)", id.Str(), len(buf), cap(buf))
|
debug.Log("load blob %v into buf (len %v, cap %v)", id, len(buf), cap(buf))
|
||||||
size, found := r.idx.LookupSize(id, t)
|
size, found := r.idx.LookupSize(id, t)
|
||||||
if !found {
|
if !found {
|
||||||
return 0, errors.Errorf("id %v not found in repository", id)
|
return 0, errors.Errorf("id %v not found in repository", id)
|
||||||
@ -608,7 +608,7 @@ func (r *Repository) SaveBlob(ctx context.Context, t restic.BlobType, buf []byte
|
|||||||
|
|
||||||
// LoadTree loads a tree from the repository.
|
// LoadTree loads a tree from the repository.
|
||||||
func (r *Repository) LoadTree(ctx context.Context, id restic.ID) (*restic.Tree, error) {
|
func (r *Repository) LoadTree(ctx context.Context, id restic.ID) (*restic.Tree, error) {
|
||||||
debug.Log("load tree %v", id.Str())
|
debug.Log("load tree %v", id)
|
||||||
|
|
||||||
size, found := r.idx.LookupSize(id, restic.TreeBlob)
|
size, found := r.idx.LookupSize(id, restic.TreeBlob)
|
||||||
if !found {
|
if !found {
|
||||||
|
@ -225,7 +225,7 @@ func (l *Lock) Stale() bool {
|
|||||||
// Refresh refreshes the lock by creating a new file in the backend with a new
|
// Refresh refreshes the lock by creating a new file in the backend with a new
|
||||||
// timestamp. Afterwards the old lock is removed.
|
// timestamp. Afterwards the old lock is removed.
|
||||||
func (l *Lock) Refresh(ctx context.Context) error {
|
func (l *Lock) Refresh(ctx context.Context) error {
|
||||||
debug.Log("refreshing lock %v", l.lockID.Str())
|
debug.Log("refreshing lock %v", l.lockID)
|
||||||
id, err := l.createLock(ctx)
|
id, err := l.createLock(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -236,7 +236,7 @@ func (l *Lock) Refresh(ctx context.Context) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("new lock ID %v", id.Str())
|
debug.Log("new lock ID %v", id)
|
||||||
l.lockID = &id
|
l.lockID = &id
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
@ -42,10 +42,10 @@ func NewRestorer(repo Repository, id ID) (*Restorer, error) {
|
|||||||
// restoreTo restores a tree from the repo to a destination. target is the path in
|
// restoreTo restores a tree from the repo to a destination. target is the path in
|
||||||
// the file system, location within the snapshot.
|
// the file system, location within the snapshot.
|
||||||
func (res *Restorer) restoreTo(ctx context.Context, target, location string, treeID ID, idx *HardlinkIndex) error {
|
func (res *Restorer) restoreTo(ctx context.Context, target, location string, treeID ID, idx *HardlinkIndex) error {
|
||||||
debug.Log("%v %v %v", target, location, treeID.Str())
|
debug.Log("%v %v %v", target, location, treeID)
|
||||||
tree, err := res.repo.LoadTree(ctx, treeID)
|
tree, err := res.repo.LoadTree(ctx, treeID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log("error loading tree %v: %v", treeID.Str(), err)
|
debug.Log("error loading tree %v: %v", treeID, err)
|
||||||
return res.Error(location, nil, err)
|
return res.Error(location, nil, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,8 +36,8 @@ func NewTreeWalker(ch chan<- loadTreeJob, out chan<- TreeJob) *TreeWalker {
|
|||||||
// Walk starts walking the tree given by id. When the channel done is closed,
|
// Walk starts walking the tree given by id. When the channel done is closed,
|
||||||
// processing stops.
|
// processing stops.
|
||||||
func (tw *TreeWalker) Walk(ctx context.Context, path string, id restic.ID) {
|
func (tw *TreeWalker) Walk(ctx context.Context, path string, id restic.ID) {
|
||||||
debug.Log("starting on tree %v for %v", id.Str(), path)
|
debug.Log("starting on tree %v for %v", id, path)
|
||||||
defer debug.Log("done walking tree %v for %v", id.Str(), path)
|
defer debug.Log("done walking tree %v for %v", id, path)
|
||||||
|
|
||||||
resCh := make(chan loadTreeResult, 1)
|
resCh := make(chan loadTreeResult, 1)
|
||||||
tw.ch <- loadTreeJob{
|
tw.ch <- loadTreeJob{
|
||||||
@ -141,10 +141,10 @@ func loadTreeWorker(ctx context.Context, wg *sync.WaitGroup, in <-chan loadTreeJ
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.Log("received job to load tree %v", job.id.Str())
|
debug.Log("received job to load tree %v", job.id)
|
||||||
tree, err := load(job.id)
|
tree, err := load(job.id)
|
||||||
|
|
||||||
debug.Log("tree %v loaded, error %v", job.id.Str(), err)
|
debug.Log("tree %v loaded, error %v", job.id, err)
|
||||||
|
|
||||||
select {
|
select {
|
||||||
case job.res <- loadTreeResult{tree, err}:
|
case job.res <- loadTreeResult{tree, err}:
|
||||||
@ -168,7 +168,7 @@ const loadTreeWorkers = 10
|
|||||||
// file and directory it finds. When the channel done is closed, processing
|
// file and directory it finds. When the channel done is closed, processing
|
||||||
// stops.
|
// stops.
|
||||||
func Tree(ctx context.Context, repo TreeLoader, id restic.ID, jobCh chan<- TreeJob) {
|
func Tree(ctx context.Context, repo TreeLoader, id restic.ID, jobCh chan<- TreeJob) {
|
||||||
debug.Log("start on %v, start workers", id.Str())
|
debug.Log("start on %v, start workers", id)
|
||||||
|
|
||||||
load := func(id restic.ID) (*restic.Tree, error) {
|
load := func(id restic.ID) (*restic.Tree, error) {
|
||||||
tree, err := repo.LoadTree(ctx, id)
|
tree, err := repo.LoadTree(ctx, id)
|
||||||
|
Loading…
Reference in New Issue
Block a user