azureblob: implement --azureblob-delete-snapshots

This flag controls what happens when we try to delete a blob with a
snapshot. The UI follows the azcopy tool.

See: https://forum.rclone.org/t/how-to-delete-undeleted-blobs-on-azure/43911/
This commit is contained in:
Nick Craig-Wood 2024-01-12 09:45:59 +00:00
parent 3df6518006
commit 519fe98e6e

View file

@ -401,6 +401,24 @@ rclone does if you know the container exists already.
Help: `If set, do not do HEAD before GET when getting objects.`, Help: `If set, do not do HEAD before GET when getting objects.`,
Default: false, Default: false,
Advanced: true, Advanced: true,
}, {
Name: "delete_snapshots",
Help: `Set to specify how to deal with snapshots on blob deletion.`,
Examples: []fs.OptionExample{
{
Value: "",
Help: "By default, the delete operation fails if a blob has snapshots",
}, {
Value: string(blob.DeleteSnapshotsOptionTypeInclude),
Help: "Specify 'include' to remove the root blob and all its snapshots",
}, {
Value: string(blob.DeleteSnapshotsOptionTypeOnly),
Help: "Specify 'only' to remove only the snapshots but keep the root blob.",
},
},
Default: "",
Exclusive: true,
Advanced: true,
}}, }},
}) })
} }
@ -437,6 +455,7 @@ type Options struct {
DirectoryMarkers bool `config:"directory_markers"` DirectoryMarkers bool `config:"directory_markers"`
NoCheckContainer bool `config:"no_check_container"` NoCheckContainer bool `config:"no_check_container"`
NoHeadObject bool `config:"no_head_object"` NoHeadObject bool `config:"no_head_object"`
DeleteSnapshots string `config:"delete_snapshots"`
} }
// Fs represents a remote azure server // Fs represents a remote azure server
@ -2356,9 +2375,10 @@ func (o *Object) Update(ctx context.Context, in io.Reader, src fs.ObjectInfo, op
// Remove an object // Remove an object
func (o *Object) Remove(ctx context.Context) error { func (o *Object) Remove(ctx context.Context) error {
blb := o.getBlobSVC() blb := o.getBlobSVC()
//only := blob.DeleteSnapshotsOptionTypeOnly opt := blob.DeleteOptions{}
opt := blob.DeleteOptions{ if o.fs.opt.DeleteSnapshots != "" {
//DeleteSnapshots: &only, action := blob.DeleteSnapshotsOptionType(o.fs.opt.DeleteSnapshots)
opt.DeleteSnapshots = &action
} }
return o.fs.pacer.Call(func() (bool, error) { return o.fs.pacer.Call(func() (bool, error) {
_, err := blb.Delete(ctx, &opt) _, err := blb.Delete(ctx, &opt)