From 72a9e5e4c5cc500a9540e57f6552b80fa239a56f Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 25 Mar 2024 21:07:44 +0000 Subject: [PATCH 1/4] bugfix: fix double slashes in paths for file/files --- storage/2023-11-03/file/files/delete.go | 2 +- storage/2023-11-03/file/files/metadata_set.go | 2 +- storage/2023-11-03/file/files/properties_get.go | 2 +- storage/2023-11-03/file/files/properties_set.go | 2 +- storage/2023-11-03/file/files/range_put.go | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/storage/2023-11-03/file/files/delete.go b/storage/2023-11-03/file/files/delete.go index f3841fa..90c7737 100644 --- a/storage/2023-11-03/file/files/delete.go +++ b/storage/2023-11-03/file/files/delete.go @@ -32,7 +32,7 @@ func (c Client) Delete(ctx context.Context, shareName, path, fileName string) (r } if path != "" { - path = fmt.Sprintf("/%s/", path) + path = fmt.Sprintf("%s/", path) } opts := client.RequestOptions{ diff --git a/storage/2023-11-03/file/files/metadata_set.go b/storage/2023-11-03/file/files/metadata_set.go index 3a51a10..e1ba9d0 100644 --- a/storage/2023-11-03/file/files/metadata_set.go +++ b/storage/2023-11-03/file/files/metadata_set.go @@ -42,7 +42,7 @@ func (c Client) SetMetaData(ctx context.Context, shareName, path, fileName strin } if path != "" { - path = fmt.Sprintf("/%s/", path) + path = fmt.Sprintf("%s/", path) } opts := client.RequestOptions{ diff --git a/storage/2023-11-03/file/files/properties_get.go b/storage/2023-11-03/file/files/properties_get.go index 927ec63..129ad24 100644 --- a/storage/2023-11-03/file/files/properties_get.go +++ b/storage/2023-11-03/file/files/properties_get.go @@ -50,7 +50,7 @@ func (c Client) GetProperties(ctx context.Context, shareName, path, fileName str } if path != "" { - path = fmt.Sprintf("/%s/", path) + path = fmt.Sprintf("%s/", path) } opts := client.RequestOptions{ diff --git a/storage/2023-11-03/file/files/properties_set.go b/storage/2023-11-03/file/files/properties_set.go index 83bd841..18ecd83 100644 --- a/storage/2023-11-03/file/files/properties_set.go +++ b/storage/2023-11-03/file/files/properties_set.go @@ -89,7 +89,7 @@ func (c Client) SetProperties(ctx context.Context, shareName, path, fileName str } if path != "" { - path = fmt.Sprintf("/%s/", path) + path = fmt.Sprintf("%s/", path) } opts := client.RequestOptions{ diff --git a/storage/2023-11-03/file/files/range_put.go b/storage/2023-11-03/file/files/range_put.go index 58e6f9b..c799ffd 100644 --- a/storage/2023-11-03/file/files/range_put.go +++ b/storage/2023-11-03/file/files/range_put.go @@ -65,7 +65,7 @@ func (c Client) PutByteRange(ctx context.Context, shareName, path, fileName stri } if path != "" { - path = fmt.Sprintf("/%s/", path) + path = fmt.Sprintf("%s/", path) } opts := client.RequestOptions{ From a86d2a37305878cba6e93e03b043a9556e02d421 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Mon, 25 Mar 2024 21:29:19 +0000 Subject: [PATCH 2/4] implement retryfunc for directory deletions --- storage/2023-11-03/file/directories/delete.go | 30 +++++++++++++++++++ storage/2023-11-03/file/directories/models.go | 9 ++++++ 2 files changed, 39 insertions(+) create mode 100644 storage/2023-11-03/file/directories/models.go diff --git a/storage/2023-11-03/file/directories/delete.go b/storage/2023-11-03/file/directories/delete.go index 35b7298..eeae2c2 100644 --- a/storage/2023-11-03/file/directories/delete.go +++ b/storage/2023-11-03/file/directories/delete.go @@ -1,12 +1,17 @@ package directories import ( + "bytes" "context" + "encoding/xml" "fmt" + "io" "net/http" "strings" + "github.com/hashicorp/go-azure-helpers/lang/response" "github.com/hashicorp/go-azure-sdk/sdk/client" + "github.com/hashicorp/go-azure-sdk/sdk/odata" ) type DeleteResponse struct { @@ -32,6 +37,30 @@ func (c Client) Delete(ctx context.Context, shareName, path string) (result Dele return } + // Retry the directory deletion if the directory is not empty (deleted files take a little while to disappear) + retryFunc := func(resp *http.Response, _ *odata.OData) (bool, error) { + if resp != nil { + if response.WasStatusCode(resp, http.StatusConflict) { + // TODO: move this error response parsing to a common helper function + respBody, err := io.ReadAll(resp.Body) + if err != nil { + return false, fmt.Errorf("could not parse response body") + } + resp.Body.Close() + respBody = bytes.TrimPrefix(respBody, []byte("\xef\xbb\xbf")) + res := ErrorResponse{} + if err = xml.Unmarshal(respBody, &res); err != nil { + return false, err + } + resp.Body = io.NopCloser(bytes.NewBuffer(respBody)) + if res.Code != nil { + return strings.Contains(*res.Code, "DirectoryNotEmpty"), nil + } + } + } + return false, nil + } + opts := client.RequestOptions{ ContentType: "application/xml; charset=utf-8", ExpectedStatusCodes: []int{ @@ -40,6 +69,7 @@ func (c Client) Delete(ctx context.Context, shareName, path string) (result Dele HttpMethod: http.MethodDelete, OptionsObject: directoriesOptions{}, Path: fmt.Sprintf("/%s/%s", shareName, path), + RetryFunc: retryFunc, } req, err := c.Client.NewRequest(ctx, opts) diff --git a/storage/2023-11-03/file/directories/models.go b/storage/2023-11-03/file/directories/models.go new file mode 100644 index 0000000..85c3bde --- /dev/null +++ b/storage/2023-11-03/file/directories/models.go @@ -0,0 +1,9 @@ +package directories + +import "encoding/xml" + +type ErrorResponse struct { + XMLName xml.Name `xml:"Error"` + Code *string `xml:"Code"` + Message *string `xml:"Message"` +} From dfcf14b9bc793008c636449835d445c837f98c32 Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 26 Mar 2024 00:20:10 +0000 Subject: [PATCH 3/4] bugfix: only include path in file/files resource ID when non-empty --- storage/2023-11-03/file/files/resource_id.go | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/storage/2023-11-03/file/files/resource_id.go b/storage/2023-11-03/file/files/resource_id.go index 4928897..936b462 100644 --- a/storage/2023-11-03/file/files/resource_id.go +++ b/storage/2023-11-03/file/files/resource_id.go @@ -37,7 +37,11 @@ func NewFileID(accountId accounts.AccountId, shareName, directoryPath, fileName } func (b FileId) ID() string { - return fmt.Sprintf("%s/%s/%s/%s", b.AccountId.ID(), b.ShareName, b.DirectoryPath, b.FileName) + path := "" + if b.DirectoryPath != "" { + path = fmt.Sprintf("%s/", b.DirectoryPath) + } + return fmt.Sprintf("%s/%s/%s%s", b.AccountId.ID(), b.ShareName, path, b.FileName) } func (b FileId) String() string { From 369a2c19472d248e8974dd5a9acb0f3b670072ea Mon Sep 17 00:00:00 2001 From: Tom Bamford Date: Tue, 26 Mar 2024 00:57:52 +0000 Subject: [PATCH 4/4] file/files: resource ID can have only 2 segments when the path is empty --- storage/2023-11-03/file/files/resource_id.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/storage/2023-11-03/file/files/resource_id.go b/storage/2023-11-03/file/files/resource_id.go index 936b462..63d5c55 100644 --- a/storage/2023-11-03/file/files/resource_id.go +++ b/storage/2023-11-03/file/files/resource_id.go @@ -76,8 +76,8 @@ func ParseFileID(input, domainSuffix string) (*FileId, error) { path := strings.TrimPrefix(uri.Path, "/") segments := strings.Split(path, "/") - if len(segments) < 3 { - return nil, fmt.Errorf("expected the path to contain at least 3 segments but got %d", len(segments)) + if len(segments) < 2 { + return nil, fmt.Errorf("expected the path to contain at least 2 segments but got %d", len(segments)) } shareName := segments[0] directoryPath := strings.Join(segments[1:len(segments)-1], "/")