2018-05-12 21:40:31 +02:00
|
|
|
package archiver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"runtime"
|
2024-02-10 23:41:11 +01:00
|
|
|
"strings"
|
2022-10-07 20:23:38 +02:00
|
|
|
"sync"
|
2018-05-12 21:40:31 +02:00
|
|
|
"sync/atomic"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/restic/restic/internal/errors"
|
|
|
|
"github.com/restic/restic/internal/restic"
|
2024-02-10 23:41:11 +01:00
|
|
|
rtest "github.com/restic/restic/internal/test"
|
2022-05-27 19:08:50 +02:00
|
|
|
"golang.org/x/sync/errgroup"
|
2018-05-12 21:40:31 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var errTest = errors.New("test error")
|
|
|
|
|
|
|
|
type saveFail struct {
|
|
|
|
cnt int32
|
|
|
|
failAt int32
|
|
|
|
}
|
|
|
|
|
2023-05-18 19:27:38 +02:00
|
|
|
func (b *saveFail) SaveBlob(_ context.Context, _ restic.BlobType, _ []byte, id restic.ID, _ bool) (restic.ID, bool, int, error) {
|
2018-05-12 21:40:31 +02:00
|
|
|
val := atomic.AddInt32(&b.cnt, 1)
|
|
|
|
if val == b.failAt {
|
2022-05-01 14:26:57 +02:00
|
|
|
return restic.ID{}, false, 0, errTest
|
2018-05-12 21:40:31 +02:00
|
|
|
}
|
|
|
|
|
2022-05-01 14:26:57 +02:00
|
|
|
return id, false, 0, nil
|
2018-05-12 21:40:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestBlobSaver(t *testing.T) {
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
2022-05-27 19:08:50 +02:00
|
|
|
wg, ctx := errgroup.WithContext(ctx)
|
2024-05-19 13:03:14 +02:00
|
|
|
saver := &saveFail{}
|
2018-05-12 21:40:31 +02:00
|
|
|
|
2024-08-27 11:26:52 +02:00
|
|
|
b := newBlobSaver(ctx, wg, saver, uint(runtime.NumCPU()))
|
2018-05-12 21:40:31 +02:00
|
|
|
|
2022-10-07 20:23:38 +02:00
|
|
|
var wait sync.WaitGroup
|
2024-08-27 11:26:52 +02:00
|
|
|
var results []saveBlobResponse
|
2022-11-10 20:19:37 +01:00
|
|
|
var lock sync.Mutex
|
2018-05-12 21:40:31 +02:00
|
|
|
|
2022-10-07 20:23:38 +02:00
|
|
|
wait.Add(20)
|
2018-05-12 21:40:31 +02:00
|
|
|
for i := 0; i < 20; i++ {
|
2024-08-27 11:26:52 +02:00
|
|
|
buf := &buffer{Data: []byte(fmt.Sprintf("foo%d", i))}
|
2022-10-07 20:23:38 +02:00
|
|
|
idx := i
|
2022-11-10 20:19:37 +01:00
|
|
|
lock.Lock()
|
2024-08-27 11:26:52 +02:00
|
|
|
results = append(results, saveBlobResponse{})
|
2022-11-10 20:19:37 +01:00
|
|
|
lock.Unlock()
|
2024-08-27 11:26:52 +02:00
|
|
|
b.Save(ctx, restic.DataBlob, buf, "file", func(res saveBlobResponse) {
|
2022-11-10 20:19:37 +01:00
|
|
|
lock.Lock()
|
2022-10-07 20:23:38 +02:00
|
|
|
results[idx] = res
|
2022-11-10 20:19:37 +01:00
|
|
|
lock.Unlock()
|
2022-10-07 20:23:38 +02:00
|
|
|
wait.Done()
|
|
|
|
})
|
2018-05-12 21:40:31 +02:00
|
|
|
}
|
|
|
|
|
2022-10-07 20:23:38 +02:00
|
|
|
wait.Wait()
|
|
|
|
for i, sbr := range results {
|
2022-05-22 15:14:25 +02:00
|
|
|
if sbr.known {
|
2018-05-12 21:40:31 +02:00
|
|
|
t.Errorf("blob %v is known, that should not be the case", i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-27 19:08:50 +02:00
|
|
|
b.TriggerShutdown()
|
2018-05-12 21:40:31 +02:00
|
|
|
|
2022-05-27 19:08:50 +02:00
|
|
|
err := wg.Wait()
|
2018-05-12 21:40:31 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBlobSaverError(t *testing.T) {
|
|
|
|
var tests = []struct {
|
|
|
|
blobs int
|
|
|
|
failAt int
|
|
|
|
}{
|
|
|
|
{20, 2},
|
|
|
|
{20, 5},
|
|
|
|
{20, 15},
|
|
|
|
{200, 150},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run("", func(t *testing.T) {
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
2022-05-27 19:08:50 +02:00
|
|
|
wg, ctx := errgroup.WithContext(ctx)
|
2018-05-12 21:40:31 +02:00
|
|
|
saver := &saveFail{
|
|
|
|
failAt: int32(test.failAt),
|
|
|
|
}
|
|
|
|
|
2024-08-27 11:26:52 +02:00
|
|
|
b := newBlobSaver(ctx, wg, saver, uint(runtime.NumCPU()))
|
2018-05-12 21:40:31 +02:00
|
|
|
|
|
|
|
for i := 0; i < test.blobs; i++ {
|
2024-08-27 11:26:52 +02:00
|
|
|
buf := &buffer{Data: []byte(fmt.Sprintf("foo%d", i))}
|
|
|
|
b.Save(ctx, restic.DataBlob, buf, "errfile", func(res saveBlobResponse) {})
|
2018-05-12 21:40:31 +02:00
|
|
|
}
|
|
|
|
|
2022-05-27 19:08:50 +02:00
|
|
|
b.TriggerShutdown()
|
2018-05-12 21:40:31 +02:00
|
|
|
|
2022-05-27 19:08:50 +02:00
|
|
|
err := wg.Wait()
|
2018-05-12 21:40:31 +02:00
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected error not found")
|
|
|
|
}
|
|
|
|
|
2024-02-10 23:41:11 +01:00
|
|
|
rtest.Assert(t, errors.Is(err, errTest), "unexpected error %v", err)
|
|
|
|
rtest.Assert(t, strings.Contains(err.Error(), "errfile"), "expected error to contain 'errfile' got: %v", err)
|
2018-05-12 21:40:31 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|