b2: Add new cleanup
and cleanup-hidden
backend commands.
This commit is contained in:
parent
a24aeba495
commit
070cff8a65
2 changed files with 133 additions and 14 deletions
|
@ -60,6 +60,7 @@ const (
|
||||||
defaultChunkSize = 96 * fs.Mebi
|
defaultChunkSize = 96 * fs.Mebi
|
||||||
defaultUploadCutoff = 200 * fs.Mebi
|
defaultUploadCutoff = 200 * fs.Mebi
|
||||||
largeFileCopyCutoff = 4 * fs.Gibi // 5E9 is the max
|
largeFileCopyCutoff = 4 * fs.Gibi // 5E9 is the max
|
||||||
|
defaultMaxAge = 24 * time.Hour
|
||||||
)
|
)
|
||||||
|
|
||||||
// Globals
|
// Globals
|
||||||
|
@ -1248,7 +1249,7 @@ func (f *Fs) deleteByID(ctx context.Context, ID, Name string) error {
|
||||||
// if oldOnly is true then it deletes only non current files.
|
// if oldOnly is true then it deletes only non current files.
|
||||||
//
|
//
|
||||||
// Implemented here so we can make sure we delete old versions.
|
// Implemented here so we can make sure we delete old versions.
|
||||||
func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool) error {
|
func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool, deleteHidden bool, deleteUnfinished bool, maxAge time.Duration) error {
|
||||||
bucket, directory := f.split(dir)
|
bucket, directory := f.split(dir)
|
||||||
if bucket == "" {
|
if bucket == "" {
|
||||||
return errors.New("can't purge from root")
|
return errors.New("can't purge from root")
|
||||||
|
@ -1266,7 +1267,7 @@ func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var isUnfinishedUploadStale = func(timestamp api.Timestamp) bool {
|
var isUnfinishedUploadStale = func(timestamp api.Timestamp) bool {
|
||||||
return time.Since(time.Time(timestamp)).Hours() > 24
|
return time.Since(time.Time(timestamp)) > maxAge
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete Config.Transfers in parallel
|
// Delete Config.Transfers in parallel
|
||||||
|
@ -1289,6 +1290,21 @@ func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool) error {
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
if oldOnly {
|
||||||
|
if deleteHidden && deleteUnfinished {
|
||||||
|
fs.Infof(f, "cleaning bucket %q of all hidden files, and pending multipart uploads older than %v", bucket, maxAge)
|
||||||
|
} else if deleteHidden {
|
||||||
|
fs.Infof(f, "cleaning bucket %q of all hidden files", bucket)
|
||||||
|
} else if deleteUnfinished {
|
||||||
|
fs.Infof(f, "cleaning bucket %q of pending multipart uploads older than %v", bucket, maxAge)
|
||||||
|
} else {
|
||||||
|
fs.Errorf(f, "cleaning bucket %q of nothing. This should never happen!", bucket)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fs.Infof(f, "cleaning bucket %q of all files", bucket)
|
||||||
|
}
|
||||||
|
|
||||||
last := ""
|
last := ""
|
||||||
checkErr(f.list(ctx, bucket, directory, f.rootDirectory, f.rootBucket == "", true, 0, true, false, func(remote string, object *api.File, isDirectory bool) error {
|
checkErr(f.list(ctx, bucket, directory, f.rootDirectory, f.rootBucket == "", true, 0, true, false, func(remote string, object *api.File, isDirectory bool) error {
|
||||||
if !isDirectory {
|
if !isDirectory {
|
||||||
|
@ -1299,14 +1315,14 @@ func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool) error {
|
||||||
tr := accounting.Stats(ctx).NewCheckingTransfer(oi, "checking")
|
tr := accounting.Stats(ctx).NewCheckingTransfer(oi, "checking")
|
||||||
if oldOnly && last != remote {
|
if oldOnly && last != remote {
|
||||||
// Check current version of the file
|
// Check current version of the file
|
||||||
if object.Action == "hide" {
|
if deleteHidden && object.Action == "hide" {
|
||||||
fs.Debugf(remote, "Deleting current version (id %q) as it is a hide marker", object.ID)
|
fs.Debugf(remote, "Deleting current version (id %q) as it is a hide marker", object.ID)
|
||||||
toBeDeleted <- object
|
toBeDeleted <- object
|
||||||
} else if object.Action == "start" && isUnfinishedUploadStale(object.UploadTimestamp) {
|
} else if deleteUnfinished && object.Action == "start" && isUnfinishedUploadStale(object.UploadTimestamp) {
|
||||||
fs.Debugf(remote, "Deleting current version (id %q) as it is a start marker (upload started at %s)", object.ID, time.Time(object.UploadTimestamp).Local())
|
fs.Debugf(remote, "Deleting current version (id %q) as it is a start marker (upload started at %s)", object.ID, time.Time(object.UploadTimestamp).Local())
|
||||||
toBeDeleted <- object
|
toBeDeleted <- object
|
||||||
} else {
|
} else {
|
||||||
fs.Debugf(remote, "Not deleting current version (id %q) %q", object.ID, object.Action)
|
fs.Debugf(remote, "Not deleting current version (id %q) %q dated %v (%v ago)", object.ID, object.Action, time.Time(object.UploadTimestamp).Local(), time.Since(time.Time(object.UploadTimestamp)))
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
fs.Debugf(remote, "Deleting (id %q)", object.ID)
|
fs.Debugf(remote, "Deleting (id %q)", object.ID)
|
||||||
|
@ -1328,12 +1344,17 @@ func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool) error {
|
||||||
|
|
||||||
// Purge deletes all the files and directories including the old versions.
|
// Purge deletes all the files and directories including the old versions.
|
||||||
func (f *Fs) Purge(ctx context.Context, dir string) error {
|
func (f *Fs) Purge(ctx context.Context, dir string) error {
|
||||||
return f.purge(ctx, dir, false)
|
return f.purge(ctx, dir, false, false, false, defaultMaxAge)
|
||||||
}
|
}
|
||||||
|
|
||||||
// CleanUp deletes all the hidden files.
|
// CleanUp deletes all hidden files and pending multipart uploads older than 24 hours.
|
||||||
func (f *Fs) CleanUp(ctx context.Context) error {
|
func (f *Fs) CleanUp(ctx context.Context) error {
|
||||||
return f.purge(ctx, "", true)
|
return f.purge(ctx, "", true, true, true, defaultMaxAge)
|
||||||
|
}
|
||||||
|
|
||||||
|
// cleanUp deletes all hidden files and/or pending multipart uploads older than the specified age.
|
||||||
|
func (f *Fs) cleanUp(ctx context.Context, deleteHidden bool, deleteUnfinished bool, maxAge time.Duration) (err error) {
|
||||||
|
return f.purge(ctx, "", true, deleteHidden, deleteUnfinished, maxAge)
|
||||||
}
|
}
|
||||||
|
|
||||||
// copy does a server-side copy from dstObj <- srcObj
|
// copy does a server-side copy from dstObj <- srcObj
|
||||||
|
@ -2243,8 +2264,56 @@ func (f *Fs) lifecycleCommand(ctx context.Context, name string, arg []string, op
|
||||||
return bucket.LifecycleRules, nil
|
return bucket.LifecycleRules, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var cleanupHelp = fs.CommandHelp{
|
||||||
|
Name: "cleanup",
|
||||||
|
Short: "Remove unfinished large file uploads.",
|
||||||
|
Long: `This command removes unfinished large file uploads of age greater than
|
||||||
|
max-age, which defaults to 24 hours.
|
||||||
|
|
||||||
|
Note that you can use --interactive/-i or --dry-run with this command to see what
|
||||||
|
it would do.
|
||||||
|
|
||||||
|
rclone backend cleanup b2:bucket/path/to/object
|
||||||
|
rclone backend cleanup -o max-age=7w b2:bucket/path/to/object
|
||||||
|
|
||||||
|
Durations are parsed as per the rest of rclone, 2h, 7d, 7w etc.
|
||||||
|
`,
|
||||||
|
Opts: map[string]string{
|
||||||
|
"max-age": "Max age of upload to delete",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *Fs) cleanupCommand(ctx context.Context, name string, arg []string, opt map[string]string) (out interface{}, err error) {
|
||||||
|
maxAge := defaultMaxAge
|
||||||
|
if opt["max-age"] != "" {
|
||||||
|
maxAge, err = fs.ParseDuration(opt["max-age"])
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("bad max-age: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, f.cleanUp(ctx, false, true, maxAge)
|
||||||
|
}
|
||||||
|
|
||||||
|
var cleanupHiddenHelp = fs.CommandHelp{
|
||||||
|
Name: "cleanup-hidden",
|
||||||
|
Short: "Remove old versions of files.",
|
||||||
|
Long: `This command removes any old hidden versions of files.
|
||||||
|
|
||||||
|
Note that you can use --interactive/-i or --dry-run with this command to see what
|
||||||
|
it would do.
|
||||||
|
|
||||||
|
rclone backend cleanup-hidden b2:bucket/path/to/dir
|
||||||
|
`,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *Fs) cleanupHiddenCommand(ctx context.Context, name string, arg []string, opt map[string]string) (out interface{}, err error) {
|
||||||
|
return nil, f.cleanUp(ctx, true, false, 0)
|
||||||
|
}
|
||||||
|
|
||||||
var commandHelp = []fs.CommandHelp{
|
var commandHelp = []fs.CommandHelp{
|
||||||
lifecycleHelp,
|
lifecycleHelp,
|
||||||
|
cleanupHelp,
|
||||||
|
cleanupHiddenHelp,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command the backend to run a named command
|
// Command the backend to run a named command
|
||||||
|
@ -2260,6 +2329,10 @@ func (f *Fs) Command(ctx context.Context, name string, arg []string, opt map[str
|
||||||
switch name {
|
switch name {
|
||||||
case "lifecycle":
|
case "lifecycle":
|
||||||
return f.lifecycleCommand(ctx, name, arg, opt)
|
return f.lifecycleCommand(ctx, name, arg, opt)
|
||||||
|
case "cleanup":
|
||||||
|
return f.cleanupCommand(ctx, name, arg, opt)
|
||||||
|
case "cleanup-hidden":
|
||||||
|
return f.cleanupHiddenCommand(ctx, name, arg, opt)
|
||||||
default:
|
default:
|
||||||
return nil, fs.ErrorCommandNotFound
|
return nil, fs.ErrorCommandNotFound
|
||||||
}
|
}
|
||||||
|
|
|
@ -179,14 +179,24 @@ using the `--b2-version-at` flag. This will show the file versions as they
|
||||||
were at that time, showing files that have been deleted afterwards, and
|
were at that time, showing files that have been deleted afterwards, and
|
||||||
hiding files that were created since.
|
hiding files that were created since.
|
||||||
|
|
||||||
If you wish to remove all the old versions then you can use the
|
If you wish to remove all the old versions, and unfinished large file
|
||||||
`rclone cleanup remote:bucket` command which will delete all the old
|
uploads, then you can use the `rclone cleanup remote:bucket` command
|
||||||
versions of files, leaving the current ones intact. You can also
|
which will delete all the old versions of files, leaving the current ones
|
||||||
supply a path and only old versions under that path will be deleted,
|
intact. You can also supply a path and only old versions under that path
|
||||||
e.g. `rclone cleanup remote:bucket/path/to/stuff`.
|
will be deleted, e.g. `rclone cleanup remote:bucket/path/to/stuff`.
|
||||||
|
|
||||||
Note that `cleanup` will remove partially uploaded files from the bucket
|
Note that `cleanup` will remove partially uploaded files from the bucket
|
||||||
if they are more than a day old.
|
if they are more than a day old. If you want more control over the
|
||||||
|
expiry date then run `rclone backend cleanup b2:bucket -o max-age=1h`
|
||||||
|
to remove all unfinished large file uploads older than one hour, leaving
|
||||||
|
old versions intact.
|
||||||
|
|
||||||
|
If you wish to remove all the old versions, leaving current files and
|
||||||
|
unfinished large files intact, then you can use the
|
||||||
|
[`rclone backend cleanup-hidden remote:bucket`](#cleanup-hidden)
|
||||||
|
command. You can also supply a path and only old versions under that
|
||||||
|
path will be deleted, e.g.
|
||||||
|
`rclone backend cleanup-hidden remote:bucket/path/to/stuff`.
|
||||||
|
|
||||||
When you `purge` a bucket, the current and the old versions will be
|
When you `purge` a bucket, the current and the old versions will be
|
||||||
deleted then the bucket will be deleted.
|
deleted then the bucket will be deleted.
|
||||||
|
@ -713,6 +723,42 @@ Options:
|
||||||
- "daysFromHidingToDeleting": After a file has been hidden for this many days it is deleted. 0 is off.
|
- "daysFromHidingToDeleting": After a file has been hidden for this many days it is deleted. 0 is off.
|
||||||
- "daysFromUploadingToHiding": This many days after uploading a file is hidden
|
- "daysFromUploadingToHiding": This many days after uploading a file is hidden
|
||||||
|
|
||||||
|
### cleanup
|
||||||
|
|
||||||
|
Remove unfinished large file uploads.
|
||||||
|
|
||||||
|
rclone backend cleanup remote: [options] [<arguments>+]
|
||||||
|
|
||||||
|
This command removes unfinished large file uploads of age greater than
|
||||||
|
max-age, which defaults to 24 hours.
|
||||||
|
|
||||||
|
Note that you can use --interactive/-i or --dry-run with this command to see what
|
||||||
|
it would do.
|
||||||
|
|
||||||
|
rclone backend cleanup b2:bucket/path/to/object
|
||||||
|
rclone backend cleanup -o max-age=7w b2:bucket/path/to/object
|
||||||
|
|
||||||
|
Durations are parsed as per the rest of rclone, 2h, 7d, 7w etc.
|
||||||
|
|
||||||
|
|
||||||
|
Options:
|
||||||
|
|
||||||
|
- "max-age": Max age of upload to delete
|
||||||
|
|
||||||
|
### cleanup-hidden
|
||||||
|
|
||||||
|
Remove old versions of files.
|
||||||
|
|
||||||
|
rclone backend cleanup-hidden remote: [options] [<arguments>+]
|
||||||
|
|
||||||
|
This command removes any old hidden versions of files.
|
||||||
|
|
||||||
|
Note that you can use --interactive/-i or --dry-run with this command to see what
|
||||||
|
it would do.
|
||||||
|
|
||||||
|
rclone backend cleanup-hidden b2:bucket/path/to/dir
|
||||||
|
|
||||||
|
|
||||||
{{< rem autogenerated options stop >}}
|
{{< rem autogenerated options stop >}}
|
||||||
|
|
||||||
## Limitations
|
## Limitations
|
||||||
|
|
Loading…
Reference in a new issue