2016-02-20 22:05:48 +01:00
|
|
|
package rest
|
|
|
|
|
|
|
|
import (
|
2017-06-03 17:39:57 +02:00
|
|
|
"context"
|
2016-02-20 22:05:48 +01:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2020-12-19 12:39:48 +01:00
|
|
|
"hash"
|
2016-02-20 22:05:48 +01:00
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"path"
|
2016-02-21 16:35:25 +01:00
|
|
|
"strings"
|
2016-02-20 22:05:48 +01:00
|
|
|
|
2022-10-15 16:23:39 +02:00
|
|
|
"github.com/restic/restic/internal/backend/layout"
|
2022-06-12 17:45:34 +02:00
|
|
|
"github.com/restic/restic/internal/backend/sema"
|
2017-07-23 14:21:03 +02:00
|
|
|
"github.com/restic/restic/internal/debug"
|
|
|
|
"github.com/restic/restic/internal/errors"
|
2017-07-24 17:42:25 +02:00
|
|
|
"github.com/restic/restic/internal/restic"
|
2016-08-21 17:46:23 +02:00
|
|
|
|
2020-12-17 12:47:53 +01:00
|
|
|
"github.com/cenkalti/backoff/v4"
|
2016-02-20 22:05:48 +01:00
|
|
|
)
|
|
|
|
|
2017-01-22 12:32:20 +01:00
|
|
|
// make sure the rest backend implements restic.Backend
|
2018-03-13 22:30:41 +01:00
|
|
|
var _ restic.Backend = &Backend{}
|
2017-01-22 12:32:20 +01:00
|
|
|
|
2018-03-13 22:30:41 +01:00
|
|
|
// Backend uses the REST protocol to access data stored on a server.
|
|
|
|
type Backend struct {
|
2021-08-07 22:20:49 +02:00
|
|
|
url *url.URL
|
|
|
|
connections uint
|
2022-06-12 17:45:34 +02:00
|
|
|
sem sema.Semaphore
|
2022-10-09 10:21:30 +02:00
|
|
|
client http.Client
|
2022-10-15 16:23:39 +02:00
|
|
|
layout.Layout
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2018-03-13 22:22:35 +01:00
|
|
|
// the REST API protocol version is decided by HTTP request headers, these are the constants.
|
2018-01-23 23:12:52 +01:00
|
|
|
const (
|
2018-03-13 22:22:35 +01:00
|
|
|
ContentTypeV1 = "application/vnd.x.restic.rest.v1"
|
|
|
|
ContentTypeV2 = "application/vnd.x.restic.rest.v2"
|
2018-01-23 23:12:52 +01:00
|
|
|
)
|
|
|
|
|
2016-02-20 22:05:48 +01:00
|
|
|
// Open opens the REST backend with the given config.
|
2018-03-13 22:30:41 +01:00
|
|
|
func Open(cfg Config, rt http.RoundTripper) (*Backend, error) {
|
2022-06-12 17:45:34 +02:00
|
|
|
sem, err := sema.New(cfg.Connections)
|
2017-06-06 00:25:22 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-04-11 21:47:57 +02:00
|
|
|
// use url without trailing slash for layout
|
|
|
|
url := cfg.URL.String()
|
|
|
|
if url[len(url)-1] == '/' {
|
|
|
|
url = url[:len(url)-1]
|
|
|
|
}
|
|
|
|
|
2018-03-13 22:30:41 +01:00
|
|
|
be := &Backend{
|
2021-08-07 22:20:49 +02:00
|
|
|
url: cfg.URL,
|
2022-10-09 10:21:30 +02:00
|
|
|
client: http.Client{Transport: rt},
|
2022-10-15 16:23:39 +02:00
|
|
|
Layout: &layout.RESTLayout{URL: url, Join: path.Join},
|
2021-08-07 22:20:49 +02:00
|
|
|
connections: cfg.Connections,
|
|
|
|
sem: sem,
|
2017-04-11 21:47:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return be, nil
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2017-03-16 21:50:26 +01:00
|
|
|
// Create creates a new REST on server configured in config.
|
2020-04-10 12:08:52 +02:00
|
|
|
func Create(ctx context.Context, cfg Config, rt http.RoundTripper) (*Backend, error) {
|
2017-09-24 20:04:23 +02:00
|
|
|
be, err := Open(cfg, rt)
|
2017-03-16 21:50:26 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-04-10 12:08:52 +02:00
|
|
|
_, err = be.Stat(ctx, restic.Handle{Type: restic.ConfigFile})
|
2017-03-16 21:50:26 +01:00
|
|
|
if err == nil {
|
|
|
|
return nil, errors.Fatal("config file already exists")
|
|
|
|
}
|
|
|
|
|
|
|
|
url := *cfg.URL
|
|
|
|
values := url.Query()
|
|
|
|
values.Set("create", "true")
|
|
|
|
url.RawQuery = values.Encode()
|
|
|
|
|
2017-11-25 20:56:40 +01:00
|
|
|
resp, err := be.client.Post(url.String(), "binary/octet-stream", strings.NewReader(""))
|
2017-03-16 21:50:26 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return nil, errors.Fatalf("server response unexpected: %v (%v)", resp.Status, resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
2022-12-02 19:36:43 +01:00
|
|
|
_, err = io.Copy(io.Discard, resp.Body)
|
2017-03-16 21:50:26 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = resp.Body.Close()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return be, nil
|
|
|
|
}
|
|
|
|
|
2021-08-07 22:20:49 +02:00
|
|
|
func (b *Backend) Connections() uint {
|
|
|
|
return b.connections
|
|
|
|
}
|
|
|
|
|
2016-02-20 22:05:48 +01:00
|
|
|
// Location returns this backend's location (the server's URL).
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) Location() string {
|
2016-02-20 22:05:48 +01:00
|
|
|
return b.url.String()
|
|
|
|
}
|
|
|
|
|
2020-12-19 12:39:48 +01:00
|
|
|
// Hasher may return a hash function for calculating a content hash for the backend
|
|
|
|
func (b *Backend) Hasher() hash.Hash {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-05-01 20:07:29 +02:00
|
|
|
// HasAtomicReplace returns whether Save() can atomically replace files
|
|
|
|
func (b *Backend) HasAtomicReplace() bool {
|
|
|
|
// rest-server prevents overwriting
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2016-02-20 22:05:48 +01:00
|
|
|
// Save stores data in the backend at the handle.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) Save(ctx context.Context, h restic.Handle, rd restic.RewindReader) error {
|
2016-02-20 22:05:48 +01:00
|
|
|
if err := h.Valid(); err != nil {
|
2020-12-17 12:47:53 +01:00
|
|
|
return backoff.Permanent(err)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2017-06-03 17:39:57 +02:00
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
|
|
defer cancel()
|
|
|
|
|
2017-06-15 15:58:23 +02:00
|
|
|
// make sure that client.Post() cannot close the reader by wrapping it
|
2022-10-09 10:21:30 +02:00
|
|
|
req, err := http.NewRequestWithContext(ctx,
|
2022-12-02 19:36:43 +01:00
|
|
|
http.MethodPost, b.Filename(h), io.NopCloser(rd))
|
2018-01-23 23:12:52 +01:00
|
|
|
if err != nil {
|
2022-10-09 10:21:30 +02:00
|
|
|
return errors.WithStack(err)
|
2018-01-23 23:12:52 +01:00
|
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "application/octet-stream")
|
2018-03-13 22:22:35 +01:00
|
|
|
req.Header.Set("Accept", ContentTypeV2)
|
2018-01-23 23:12:52 +01:00
|
|
|
|
2018-03-08 22:14:12 +01:00
|
|
|
// explicitly set the content length, this prevents chunked encoding and
|
|
|
|
// let's the server know what's coming.
|
|
|
|
req.ContentLength = rd.Length()
|
|
|
|
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.GetToken()
|
2022-10-09 10:21:30 +02:00
|
|
|
resp, err := b.client.Do(req)
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.ReleaseToken()
|
2016-02-20 22:05:48 +01:00
|
|
|
|
2021-01-30 19:25:04 +01:00
|
|
|
var cerr error
|
2016-02-20 22:05:48 +01:00
|
|
|
if resp != nil {
|
2022-12-02 19:36:43 +01:00
|
|
|
_, _ = io.Copy(io.Discard, resp.Body)
|
2021-01-30 19:25:04 +01:00
|
|
|
cerr = resp.Body.Close()
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2022-10-09 10:21:30 +02:00
|
|
|
return errors.WithStack(err)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
2017-05-28 12:31:19 +02:00
|
|
|
return errors.Errorf("server response unexpected: %v (%v)", resp.Status, resp.StatusCode)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2021-01-30 19:25:04 +01:00
|
|
|
return errors.Wrap(cerr, "Close")
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2022-06-13 20:35:37 +02:00
|
|
|
// notExistError is returned whenever the requested file does not exist on the
|
2017-06-15 13:40:27 +02:00
|
|
|
// server.
|
2022-06-13 20:35:37 +02:00
|
|
|
type notExistError struct {
|
2017-06-15 13:40:27 +02:00
|
|
|
restic.Handle
|
|
|
|
}
|
|
|
|
|
2022-06-13 20:35:37 +02:00
|
|
|
func (e *notExistError) Error() string {
|
2017-06-15 13:40:27 +02:00
|
|
|
return fmt.Sprintf("%v does not exist", e.Handle)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsNotExist returns true if the error was caused by a non-existing file.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) IsNotExist(err error) bool {
|
2022-06-13 20:35:37 +02:00
|
|
|
var e *notExistError
|
|
|
|
return errors.As(err, &e)
|
2017-06-15 13:40:27 +02:00
|
|
|
}
|
|
|
|
|
2018-01-16 23:59:16 -05:00
|
|
|
// Load runs fn with a reader that yields the contents of the file at h at the
|
|
|
|
// given offset.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) Load(ctx context.Context, h restic.Handle, length int, offset int64, fn func(rd io.Reader) error) error {
|
2020-10-03 02:23:34 +03:00
|
|
|
r, err := b.openReader(ctx, h, length, offset)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = fn(r)
|
|
|
|
if err != nil {
|
|
|
|
_ = r.Close() // ignore error here
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note: readerat.ReadAt() (the fn) uses io.ReadFull() that doesn't
|
|
|
|
// wait for EOF after reading body. Due to HTTP/2 stream multiplexing
|
|
|
|
// and goroutine timings the EOF frame arrives from server (eg. rclone)
|
|
|
|
// with a delay after reading body. Immediate close might trigger
|
|
|
|
// HTTP/2 stream reset resulting in the *stream closed* error on server,
|
|
|
|
// so we wait for EOF before closing body.
|
|
|
|
var buf [1]byte
|
|
|
|
_, err = r.Read(buf[:])
|
|
|
|
if err == io.EOF {
|
|
|
|
err = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if e := r.Close(); err == nil {
|
|
|
|
err = e
|
|
|
|
}
|
|
|
|
return err
|
2018-01-16 23:59:16 -05:00
|
|
|
}
|
|
|
|
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) openReader(ctx context.Context, h restic.Handle, length int, offset int64) (io.ReadCloser, error) {
|
2017-01-22 22:01:12 +01:00
|
|
|
if err := h.Valid(); err != nil {
|
2020-12-17 12:47:53 +01:00
|
|
|
return nil, backoff.Permanent(err)
|
2017-01-22 22:01:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if offset < 0 {
|
|
|
|
return nil, errors.New("offset is negative")
|
|
|
|
}
|
|
|
|
|
|
|
|
if length < 0 {
|
|
|
|
return nil, errors.Errorf("invalid length %d", length)
|
|
|
|
}
|
|
|
|
|
2022-10-09 10:21:30 +02:00
|
|
|
req, err := http.NewRequestWithContext(ctx, "GET", b.Filename(h), nil)
|
2017-01-22 22:01:12 +01:00
|
|
|
if err != nil {
|
2022-10-09 10:21:30 +02:00
|
|
|
return nil, errors.WithStack(err)
|
2017-01-22 22:01:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
byteRange := fmt.Sprintf("bytes=%d-", offset)
|
|
|
|
if length > 0 {
|
|
|
|
byteRange = fmt.Sprintf("bytes=%d-%d", offset, offset+int64(length)-1)
|
|
|
|
}
|
2018-01-23 23:12:52 +01:00
|
|
|
req.Header.Set("Range", byteRange)
|
2018-03-13 22:22:35 +01:00
|
|
|
req.Header.Set("Accept", ContentTypeV2)
|
2017-01-23 18:11:10 +01:00
|
|
|
debug.Log("Load(%v) send range %v", h, byteRange)
|
2017-01-22 22:01:12 +01:00
|
|
|
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.GetToken()
|
2022-10-09 10:21:30 +02:00
|
|
|
resp, err := b.client.Do(req)
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.ReleaseToken()
|
2017-01-22 22:01:12 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
if resp != nil {
|
2022-12-02 19:36:43 +01:00
|
|
|
_, _ = io.Copy(io.Discard, resp.Body)
|
2017-06-03 17:39:57 +02:00
|
|
|
_ = resp.Body.Close()
|
2017-01-22 22:01:12 +01:00
|
|
|
}
|
|
|
|
return nil, errors.Wrap(err, "client.Do")
|
|
|
|
}
|
|
|
|
|
2017-06-15 13:40:27 +02:00
|
|
|
if resp.StatusCode == http.StatusNotFound {
|
|
|
|
_ = resp.Body.Close()
|
2022-06-13 20:35:37 +02:00
|
|
|
return nil, ¬ExistError{h}
|
2017-06-15 13:40:27 +02:00
|
|
|
}
|
|
|
|
|
2017-01-22 22:01:12 +01:00
|
|
|
if resp.StatusCode != 200 && resp.StatusCode != 206 {
|
2017-06-03 17:39:57 +02:00
|
|
|
_ = resp.Body.Close()
|
2017-05-28 12:31:19 +02:00
|
|
|
return nil, errors.Errorf("unexpected HTTP response (%v): %v", resp.StatusCode, resp.Status)
|
2017-01-22 22:01:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return resp.Body, nil
|
|
|
|
}
|
|
|
|
|
2016-02-20 22:05:48 +01:00
|
|
|
// Stat returns information about a blob.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) Stat(ctx context.Context, h restic.Handle) (restic.FileInfo, error) {
|
2016-02-20 22:05:48 +01:00
|
|
|
if err := h.Valid(); err != nil {
|
2020-12-17 12:47:53 +01:00
|
|
|
return restic.FileInfo{}, backoff.Permanent(err)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2022-10-09 10:21:30 +02:00
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodHead, b.Filename(h), nil)
|
2018-01-23 23:12:52 +01:00
|
|
|
if err != nil {
|
2022-10-09 10:21:30 +02:00
|
|
|
return restic.FileInfo{}, errors.WithStack(err)
|
2018-01-23 23:12:52 +01:00
|
|
|
}
|
2018-03-13 22:22:35 +01:00
|
|
|
req.Header.Set("Accept", ContentTypeV2)
|
2018-01-23 23:12:52 +01:00
|
|
|
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.GetToken()
|
2022-10-09 10:21:30 +02:00
|
|
|
resp, err := b.client.Do(req)
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.ReleaseToken()
|
2016-02-20 22:05:48 +01:00
|
|
|
if err != nil {
|
2022-10-09 10:21:30 +02:00
|
|
|
return restic.FileInfo{}, errors.WithStack(err)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2022-12-02 19:36:43 +01:00
|
|
|
_, _ = io.Copy(io.Discard, resp.Body)
|
2016-02-20 22:05:48 +01:00
|
|
|
if err = resp.Body.Close(); err != nil {
|
2016-08-31 22:51:35 +02:00
|
|
|
return restic.FileInfo{}, errors.Wrap(err, "Close")
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2017-06-15 13:40:27 +02:00
|
|
|
if resp.StatusCode == http.StatusNotFound {
|
|
|
|
_ = resp.Body.Close()
|
2022-06-13 20:35:37 +02:00
|
|
|
return restic.FileInfo{}, ¬ExistError{h}
|
2017-06-15 13:40:27 +02:00
|
|
|
}
|
|
|
|
|
2016-02-20 22:05:48 +01:00
|
|
|
if resp.StatusCode != 200 {
|
2017-05-28 12:31:19 +02:00
|
|
|
return restic.FileInfo{}, errors.Errorf("unexpected HTTP response (%v): %v", resp.StatusCode, resp.Status)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if resp.ContentLength < 0 {
|
2016-08-31 22:51:35 +02:00
|
|
|
return restic.FileInfo{}, errors.New("negative content length")
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2016-08-31 22:51:35 +02:00
|
|
|
bi := restic.FileInfo{
|
2016-02-20 22:05:48 +01:00
|
|
|
Size: resp.ContentLength,
|
2018-01-20 19:34:38 +01:00
|
|
|
Name: h.Name,
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return bi, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove removes the blob with the given name and type.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) Remove(ctx context.Context, h restic.Handle) error {
|
2016-02-20 22:05:48 +01:00
|
|
|
if err := h.Valid(); err != nil {
|
2020-12-17 12:47:53 +01:00
|
|
|
return backoff.Permanent(err)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2022-10-09 10:21:30 +02:00
|
|
|
req, err := http.NewRequestWithContext(ctx, "DELETE", b.Filename(h), nil)
|
2016-02-20 22:05:48 +01:00
|
|
|
if err != nil {
|
2022-10-09 10:21:30 +02:00
|
|
|
return errors.WithStack(err)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
2018-03-13 22:22:35 +01:00
|
|
|
req.Header.Set("Accept", ContentTypeV2)
|
2018-01-23 23:12:52 +01:00
|
|
|
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.GetToken()
|
2022-10-09 10:21:30 +02:00
|
|
|
resp, err := b.client.Do(req)
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.ReleaseToken()
|
2016-02-20 22:05:48 +01:00
|
|
|
|
|
|
|
if err != nil {
|
2016-08-29 21:54:50 +02:00
|
|
|
return errors.Wrap(err, "client.Do")
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2017-06-15 13:40:27 +02:00
|
|
|
if resp.StatusCode == http.StatusNotFound {
|
|
|
|
_ = resp.Body.Close()
|
2022-06-13 20:35:37 +02:00
|
|
|
return ¬ExistError{h}
|
2017-06-15 13:40:27 +02:00
|
|
|
}
|
|
|
|
|
2016-02-20 22:05:48 +01:00
|
|
|
if resp.StatusCode != 200 {
|
2017-03-16 21:50:26 +01:00
|
|
|
return errors.Errorf("blob not removed, server response: %v (%v)", resp.Status, resp.StatusCode)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2022-12-02 19:36:43 +01:00
|
|
|
_, err = io.Copy(io.Discard, resp.Body)
|
2017-06-03 17:39:57 +02:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Copy")
|
|
|
|
}
|
|
|
|
|
|
|
|
return errors.Wrap(resp.Body.Close(), "Close")
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2018-01-20 19:34:38 +01:00
|
|
|
// List runs fn for each file in the backend which has the type t. When an
|
|
|
|
// error occurs (or fn returns an error), List stops and returns it.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) List(ctx context.Context, t restic.FileType, fn func(restic.FileInfo) error) error {
|
2017-04-11 21:47:57 +02:00
|
|
|
url := b.Dirname(restic.Handle{Type: t})
|
2016-02-21 16:35:25 +01:00
|
|
|
if !strings.HasSuffix(url, "/") {
|
|
|
|
url += "/"
|
|
|
|
}
|
|
|
|
|
2022-10-09 10:21:30 +02:00
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
2018-01-23 23:12:52 +01:00
|
|
|
if err != nil {
|
2022-10-09 10:21:30 +02:00
|
|
|
return errors.WithStack(err)
|
2018-01-23 23:12:52 +01:00
|
|
|
}
|
2018-03-13 22:22:35 +01:00
|
|
|
req.Header.Set("Accept", ContentTypeV2)
|
2018-01-23 23:12:52 +01:00
|
|
|
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.GetToken()
|
2022-10-09 10:21:30 +02:00
|
|
|
resp, err := b.client.Do(req)
|
2017-06-06 00:25:22 +02:00
|
|
|
b.sem.ReleaseToken()
|
2016-02-20 22:05:48 +01:00
|
|
|
|
|
|
|
if err != nil {
|
2018-03-08 10:22:43 +00:00
|
|
|
return errors.Wrap(err, "List")
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return errors.Errorf("List failed, server response: %v (%v)", resp.Status, resp.StatusCode)
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2018-03-13 22:22:35 +01:00
|
|
|
if resp.Header.Get("Content-Type") == ContentTypeV2 {
|
2018-01-23 23:12:52 +01:00
|
|
|
return b.listv2(ctx, t, resp, fn)
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.listv1(ctx, t, resp, fn)
|
|
|
|
}
|
|
|
|
|
|
|
|
// listv1 uses the REST protocol v1, where a list HTTP request (e.g. `GET
|
|
|
|
// /data/`) only returns the names of the files, so we need to issue an HTTP
|
|
|
|
// HEAD request for each file.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) listv1(ctx context.Context, t restic.FileType, resp *http.Response, fn func(restic.FileInfo) error) error {
|
2018-01-23 23:12:52 +01:00
|
|
|
debug.Log("parsing API v1 response")
|
2016-02-20 22:05:48 +01:00
|
|
|
dec := json.NewDecoder(resp.Body)
|
|
|
|
var list []string
|
2018-01-23 23:12:52 +01:00
|
|
|
if err := dec.Decode(&list); err != nil {
|
2018-01-20 19:34:38 +01:00
|
|
|
return errors.Wrap(err, "Decode")
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2018-01-20 19:34:38 +01:00
|
|
|
for _, m := range list {
|
|
|
|
fi, err := b.Stat(ctx, restic.Handle{Name: m, Type: t})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2018-01-20 19:34:38 +01:00
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
|
|
|
|
|
|
|
fi.Name = m
|
|
|
|
err = fn(fi)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ctx.Err()
|
2016-02-20 22:05:48 +01:00
|
|
|
}
|
|
|
|
|
2018-01-23 23:12:52 +01:00
|
|
|
// listv2 uses the REST protocol v2, where a list HTTP request (e.g. `GET
|
|
|
|
// /data/`) returns the names and sizes of all files.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) listv2(ctx context.Context, t restic.FileType, resp *http.Response, fn func(restic.FileInfo) error) error {
|
2018-01-23 23:12:52 +01:00
|
|
|
debug.Log("parsing API v2 response")
|
|
|
|
dec := json.NewDecoder(resp.Body)
|
|
|
|
|
|
|
|
var list []struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
Size int64 `json:"size"`
|
|
|
|
}
|
|
|
|
if err := dec.Decode(&list); err != nil {
|
|
|
|
return errors.Wrap(err, "Decode")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, item := range list {
|
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
|
|
|
|
|
|
|
fi := restic.FileInfo{
|
|
|
|
Name: item.Name,
|
|
|
|
Size: item.Size,
|
|
|
|
}
|
|
|
|
|
|
|
|
err := fn(fi)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
|
|
|
|
2016-02-20 22:05:48 +01:00
|
|
|
// Close closes all open files.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) Close() error {
|
2016-02-20 22:05:48 +01:00
|
|
|
// this does not need to do anything, all open files are closed within the
|
|
|
|
// same function.
|
|
|
|
return nil
|
|
|
|
}
|
2017-10-14 13:38:17 +02:00
|
|
|
|
|
|
|
// Remove keys for a specified backend type.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) removeKeys(ctx context.Context, t restic.FileType) error {
|
2018-01-20 19:34:38 +01:00
|
|
|
return b.List(ctx, t, func(fi restic.FileInfo) error {
|
|
|
|
return b.Remove(ctx, restic.Handle{Type: t, Name: fi.Name})
|
|
|
|
})
|
2017-10-14 13:38:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delete removes all data in the backend.
|
2018-03-13 22:30:41 +01:00
|
|
|
func (b *Backend) Delete(ctx context.Context) error {
|
2017-10-14 13:38:17 +02:00
|
|
|
alltypes := []restic.FileType{
|
2020-08-16 11:16:38 +02:00
|
|
|
restic.PackFile,
|
2017-10-14 13:38:17 +02:00
|
|
|
restic.KeyFile,
|
|
|
|
restic.LockFile,
|
|
|
|
restic.SnapshotFile,
|
|
|
|
restic.IndexFile}
|
|
|
|
|
|
|
|
for _, t := range alltypes {
|
|
|
|
err := b.removeKeys(ctx, t)
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-14 15:56:38 +02:00
|
|
|
err := b.Remove(ctx, restic.Handle{Type: restic.ConfigFile})
|
|
|
|
if err != nil && b.IsNotExist(err) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
2017-10-14 13:38:17 +02:00
|
|
|
}
|