| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | package swift | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2020-12-19 12:49:58 +01:00
										 |  |  | 	"crypto/md5" | 
					
						
							|  |  |  | 	"encoding/hex" | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2020-12-19 12:39:48 +01:00
										 |  |  | 	"hash" | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	"path" | 
					
						
							| 
									
										
										
										
											2020-12-18 23:38:09 +01:00
										 |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/backend" | 
					
						
							|  |  |  | 	"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" | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-17 12:47:53 +01:00
										 |  |  | 	"github.com/cenkalti/backoff/v4" | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	"github.com/ncw/swift/v2" | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // beSwift is a backend which stores the data on a swift endpoint. | 
					
						
							|  |  |  | type beSwift struct { | 
					
						
							| 
									
										
										
										
											2021-08-07 22:20:49 +02:00
										 |  |  | 	conn        *swift.Connection | 
					
						
							|  |  |  | 	connections uint | 
					
						
							|  |  |  | 	sem         *backend.Semaphore | 
					
						
							|  |  |  | 	container   string // Container name | 
					
						
							|  |  |  | 	prefix      string // Prefix of object names in the container | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	backend.Layout | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | // ensure statically that *beSwift implements restic.Backend. | 
					
						
							|  |  |  | var _ restic.Backend = &beSwift{} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | // Open opens the swift backend at a container in region. The container is | 
					
						
							|  |  |  | // created if it does not exist yet. | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | func Open(ctx context.Context, cfg Config, rt http.RoundTripper) (restic.Backend, error) { | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	debug.Log("config %#v", cfg) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 00:33:25 +02:00
										 |  |  | 	sem, err := backend.NewSemaphore(cfg.Connections) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	be := &beSwift{ | 
					
						
							|  |  |  | 		conn: &swift.Connection{ | 
					
						
							| 
									
										
										
										
											2019-01-29 17:06:20 +01:00
										 |  |  | 			UserName:                    cfg.UserName, | 
					
						
							| 
									
										
										
										
											2020-12-11 19:09:32 +01:00
										 |  |  | 			UserId:                      cfg.UserID, | 
					
						
							| 
									
										
										
										
											2019-01-29 17:06:20 +01:00
										 |  |  | 			Domain:                      cfg.Domain, | 
					
						
							| 
									
										
										
										
											2020-12-11 19:09:32 +01:00
										 |  |  | 			DomainId:                    cfg.DomainID, | 
					
						
							| 
									
										
										
										
											2019-01-29 17:06:20 +01:00
										 |  |  | 			ApiKey:                      cfg.APIKey, | 
					
						
							|  |  |  | 			AuthUrl:                     cfg.AuthURL, | 
					
						
							|  |  |  | 			Region:                      cfg.Region, | 
					
						
							|  |  |  | 			Tenant:                      cfg.Tenant, | 
					
						
							|  |  |  | 			TenantId:                    cfg.TenantID, | 
					
						
							|  |  |  | 			TenantDomain:                cfg.TenantDomain, | 
					
						
							| 
									
										
										
										
											2020-12-11 19:09:32 +01:00
										 |  |  | 			TenantDomainId:              cfg.TenantDomainID, | 
					
						
							| 
									
										
										
										
											2019-01-29 17:06:20 +01:00
										 |  |  | 			TrustId:                     cfg.TrustID, | 
					
						
							|  |  |  | 			StorageUrl:                  cfg.StorageURL, | 
					
						
							| 
									
										
										
										
											2021-08-04 22:56:18 +02:00
										 |  |  | 			AuthToken:                   cfg.AuthToken.Unwrap(), | 
					
						
							| 
									
										
										
										
											2019-01-29 17:06:20 +01:00
										 |  |  | 			ApplicationCredentialId:     cfg.ApplicationCredentialID, | 
					
						
							|  |  |  | 			ApplicationCredentialName:   cfg.ApplicationCredentialName, | 
					
						
							| 
									
										
										
										
											2021-08-04 22:56:18 +02:00
										 |  |  | 			ApplicationCredentialSecret: cfg.ApplicationCredentialSecret.Unwrap(), | 
					
						
							| 
									
										
										
										
											2019-01-29 17:06:20 +01:00
										 |  |  | 			ConnectTimeout:              time.Minute, | 
					
						
							|  |  |  | 			Timeout:                     time.Minute, | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 20:04:23 +02:00
										 |  |  | 			Transport: rt, | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-08-07 22:20:49 +02:00
										 |  |  | 		connections: cfg.Connections, | 
					
						
							|  |  |  | 		sem:         sem, | 
					
						
							|  |  |  | 		container:   cfg.Container, | 
					
						
							|  |  |  | 		prefix:      cfg.Prefix, | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 		Layout: &backend.DefaultLayout{ | 
					
						
							|  |  |  | 			Path: cfg.Prefix, | 
					
						
							|  |  |  | 			Join: path.Join, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Authenticate if needed | 
					
						
							|  |  |  | 	if !be.conn.Authenticated() { | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 		if err := be.conn.Authenticate(ctx); err != nil { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 			return nil, errors.Wrap(err, "conn.Authenticate") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Ensure container exists | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	switch _, _, err := be.conn.Container(ctx, be.container); err { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	case nil: | 
					
						
							|  |  |  | 		// Container exists | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case swift.ContainerNotFound: | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 		err = be.createContainer(ctx, cfg.DefaultContainerPolicy) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, errors.Wrap(err, "beSwift.createContainer") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return nil, errors.Wrap(err, "conn.Container") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	return be, nil | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | func (be *beSwift) createContainer(ctx context.Context, policy string) error { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	var h swift.Headers | 
					
						
							|  |  |  | 	if policy != "" { | 
					
						
							|  |  |  | 		h = swift.Headers{ | 
					
						
							|  |  |  | 			"X-Storage-Policy": policy, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	return be.conn.ContainerCreate(ctx, be.container, h) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-07 22:20:49 +02:00
										 |  |  | func (be *beSwift) Connections() uint { | 
					
						
							|  |  |  | 	return be.connections | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | // Location returns this backend's location (the container name). | 
					
						
							|  |  |  | func (be *beSwift) Location() string { | 
					
						
							|  |  |  | 	return be.container | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 12:39:48 +01:00
										 |  |  | // Hasher may return a hash function for calculating a content hash for the backend | 
					
						
							|  |  |  | func (be *beSwift) Hasher() hash.Hash { | 
					
						
							| 
									
										
										
										
											2020-12-19 12:49:58 +01:00
										 |  |  | 	return md5.New() | 
					
						
							| 
									
										
										
										
											2020-12-19 12:39:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-01 20:07:29 +02:00
										 |  |  | // HasAtomicReplace returns whether Save() can atomically replace files | 
					
						
							|  |  |  | func (be *beSwift) HasAtomicReplace() bool { | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  | func (be *beSwift) Load(ctx context.Context, h restic.Handle, length int, offset int64, fn func(rd io.Reader) error) error { | 
					
						
							|  |  |  | 	return backend.DefaultLoad(ctx, h, length, offset, be.openReader, fn) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (be *beSwift) openReader(ctx context.Context, h restic.Handle, length int, offset int64) (io.ReadCloser, error) { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	debug.Log("Load %v, length %v, offset %v", h, length, offset) | 
					
						
							|  |  |  | 	if err := h.Valid(); err != nil { | 
					
						
							| 
									
										
										
										
											2020-12-17 12:47:53 +01:00
										 |  |  | 		return nil, backoff.Permanent(err) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if offset < 0 { | 
					
						
							|  |  |  | 		return nil, errors.New("offset is negative") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if length < 0 { | 
					
						
							|  |  |  | 		return nil, errors.Errorf("invalid length %d", length) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	objName := be.Filename(h) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	headers := swift.Headers{} | 
					
						
							|  |  |  | 	if offset > 0 { | 
					
						
							|  |  |  | 		headers["Range"] = fmt.Sprintf("bytes=%d-", offset) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	if length > 0 { | 
					
						
							|  |  |  | 		headers["Range"] = fmt.Sprintf("bytes=%d-%d", offset, offset+int64(length)-1) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	if _, ok := headers["Range"]; ok { | 
					
						
							|  |  |  | 		debug.Log("Load(%v) send range %v", h, headers["Range"]) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-02 12:38:17 -04:00
										 |  |  | 	be.sem.GetToken() | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	obj, _, err := be.conn.ObjectOpen(ctx, be.container, objName, false, headers) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 		debug.Log("  err %v", err) | 
					
						
							| 
									
										
										
										
											2017-11-02 12:38:17 -04:00
										 |  |  | 		be.sem.ReleaseToken() | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 		return nil, errors.Wrap(err, "conn.ObjectOpen") | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-02 12:38:17 -04:00
										 |  |  | 	return be.sem.ReleaseTokenOnClose(obj, nil), nil | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Save stores data in the backend at the handle. | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | func (be *beSwift) Save(ctx context.Context, h restic.Handle, rd restic.RewindReader) error { | 
					
						
							|  |  |  | 	if err := h.Valid(); err != nil { | 
					
						
							| 
									
										
										
										
											2020-12-17 12:47:53 +01:00
										 |  |  | 		return backoff.Permanent(err) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	objName := be.Filename(h) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	debug.Log("Save %v at %v", h, objName) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-02 18:29:32 -04:00
										 |  |  | 	be.sem.GetToken() | 
					
						
							|  |  |  | 	defer be.sem.ReleaseToken() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	encoding := "binary/octet-stream" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	debug.Log("PutObject(%v, %v, %v)", be.container, objName, encoding) | 
					
						
							| 
									
										
										
										
											2020-12-18 23:38:09 +01:00
										 |  |  | 	hdr := swift.Headers{"Content-Length": strconv.FormatInt(rd.Length(), 10)} | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	_, err := be.conn.ObjectPut(ctx, | 
					
						
							|  |  |  | 		be.container, objName, rd, true, hex.EncodeToString(rd.Hash()), | 
					
						
							|  |  |  | 		encoding, hdr) | 
					
						
							| 
									
										
										
										
											2020-12-18 23:38:09 +01:00
										 |  |  | 	// swift does not return the upload length | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	debug.Log("%v, err %#v", objName, err) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return errors.Wrap(err, "client.PutObject") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Stat returns information about a blob. | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | func (be *beSwift) Stat(ctx context.Context, h restic.Handle) (bi restic.FileInfo, err error) { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	debug.Log("%v", h) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	objName := be.Filename(h) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-02 18:29:32 -04:00
										 |  |  | 	be.sem.GetToken() | 
					
						
							|  |  |  | 	defer be.sem.ReleaseToken() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	obj, _, err := be.conn.Object(ctx, be.container, objName) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		debug.Log("Object() err %v", err) | 
					
						
							|  |  |  | 		return restic.FileInfo{}, errors.Wrap(err, "conn.Object") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 	return restic.FileInfo{Size: obj.Bytes, Name: h.Name}, nil | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Test returns true if a blob of the given type and name exists in the backend. | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | func (be *beSwift) Test(ctx context.Context, h restic.Handle) (bool, error) { | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	objName := be.Filename(h) | 
					
						
							| 
									
										
										
										
											2017-11-02 18:29:32 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	be.sem.GetToken() | 
					
						
							|  |  |  | 	defer be.sem.ReleaseToken() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	switch _, _, err := be.conn.Object(ctx, be.container, objName); err { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	case nil: | 
					
						
							|  |  |  | 		return true, nil | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case swift.ObjectNotFound: | 
					
						
							|  |  |  | 		return false, nil | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return false, errors.Wrap(err, "conn.Object") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Remove removes the blob with the given name and type. | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | func (be *beSwift) Remove(ctx context.Context, h restic.Handle) error { | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	objName := be.Filename(h) | 
					
						
							| 
									
										
										
										
											2017-11-02 18:29:32 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	be.sem.GetToken() | 
					
						
							|  |  |  | 	defer be.sem.ReleaseToken() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	err := be.conn.ObjectDelete(ctx, be.container, objName) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	debug.Log("Remove(%v) -> err %v", h, err) | 
					
						
							|  |  |  | 	return errors.Wrap(err, "conn.ObjectDelete") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  | func (be *beSwift) List(ctx context.Context, t restic.FileType, fn func(restic.FileInfo) error) error { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	debug.Log("listing %v", t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-14 19:13:01 +01:00
										 |  |  | 	prefix, _ := be.Basedir(t) | 
					
						
							|  |  |  | 	prefix += "/" | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 	err := be.conn.ObjectsWalk(ctx, be.container, &swift.ObjectsOpts{Prefix: prefix}, | 
					
						
							|  |  |  | 		func(ctx context.Context, opts *swift.ObjectsOpts) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 			be.sem.GetToken() | 
					
						
							| 
									
										
										
										
											2021-04-09 11:24:28 +02:00
										 |  |  | 			newObjects, err := be.conn.Objects(ctx, be.container, opts) | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 			be.sem.ReleaseToken() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, errors.Wrap(err, "conn.ObjectNames") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			for _, obj := range newObjects { | 
					
						
							|  |  |  | 				m := path.Base(strings.TrimPrefix(obj.Name, prefix)) | 
					
						
							|  |  |  | 				if m == "" { | 
					
						
							|  |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				fi := restic.FileInfo{ | 
					
						
							|  |  |  | 					Name: m, | 
					
						
							|  |  |  | 					Size: obj.Bytes, | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 				err := fn(fi) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 				if err != nil { | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 					return nil, err | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-09-24 21:19:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if ctx.Err() != nil { | 
					
						
							|  |  |  | 					return nil, ctx.Err() | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			return newObjects, nil | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 	return ctx.Err() | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Remove keys for a specified backend type. | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | func (be *beSwift) removeKeys(ctx context.Context, t restic.FileType) error { | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 	return be.List(ctx, t, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 		return be.Remove(ctx, restic.Handle{Type: t, Name: fi.Name}) | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | // IsNotExist returns true if the error is caused by a not existing file. | 
					
						
							|  |  |  | func (be *beSwift) IsNotExist(err error) bool { | 
					
						
							|  |  |  | 	if e, ok := errors.Cause(err).(*swift.Error); ok { | 
					
						
							|  |  |  | 		return e.StatusCode == http.StatusNotFound | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return false | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | // Delete removes all restic objects in the container. | 
					
						
							|  |  |  | // It will not remove the container itself. | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | func (be *beSwift) Delete(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	alltypes := []restic.FileType{ | 
					
						
							| 
									
										
										
										
											2020-08-16 11:16:38 +02:00
										 |  |  | 		restic.PackFile, | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 		restic.KeyFile, | 
					
						
							|  |  |  | 		restic.LockFile, | 
					
						
							|  |  |  | 		restic.SnapshotFile, | 
					
						
							|  |  |  | 		restic.IndexFile} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, t := range alltypes { | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		err := be.removeKeys(ctx, t) | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	err := be.Remove(ctx, restic.Handle{Type: restic.ConfigFile}) | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 	if err != nil && !be.IsNotExist(err) { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Close does nothing | 
					
						
							|  |  |  | func (be *beSwift) Close() error { return nil } |