mirror of
https://github.com/octoleo/restic.git
synced 2024-11-14 09:14:07 +00:00
147 lines
4.3 KiB
Go
147 lines
4.3 KiB
Go
|
package storage
|
||
|
|
||
|
// Copyright 2017 Microsoft Corporation
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
// you may not use this file except in compliance with the License.
|
||
|
// You may obtain a copy of the License at
|
||
|
//
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
// See the License for the specific language governing permissions and
|
||
|
// limitations under the License.
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"net/url"
|
||
|
"strings"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// QueueSASOptions are options to construct a blob SAS
|
||
|
// URI.
|
||
|
// See https://docs.microsoft.com/en-us/rest/api/storageservices/constructing-a-service-sas
|
||
|
type QueueSASOptions struct {
|
||
|
QueueSASPermissions
|
||
|
SASOptions
|
||
|
}
|
||
|
|
||
|
// QueueSASPermissions includes the available permissions for
|
||
|
// a queue SAS URI.
|
||
|
type QueueSASPermissions struct {
|
||
|
Read bool
|
||
|
Add bool
|
||
|
Update bool
|
||
|
Process bool
|
||
|
}
|
||
|
|
||
|
func (q QueueSASPermissions) buildString() string {
|
||
|
permissions := ""
|
||
|
|
||
|
if q.Read {
|
||
|
permissions += "r"
|
||
|
}
|
||
|
if q.Add {
|
||
|
permissions += "a"
|
||
|
}
|
||
|
if q.Update {
|
||
|
permissions += "u"
|
||
|
}
|
||
|
if q.Process {
|
||
|
permissions += "p"
|
||
|
}
|
||
|
return permissions
|
||
|
}
|
||
|
|
||
|
// GetSASURI creates an URL to the specified queue which contains the Shared
|
||
|
// Access Signature with specified permissions and expiration time.
|
||
|
//
|
||
|
// See https://docs.microsoft.com/en-us/rest/api/storageservices/constructing-a-service-sas
|
||
|
func (q *Queue) GetSASURI(options QueueSASOptions) (string, error) {
|
||
|
canonicalizedResource, err := q.qsc.client.buildCanonicalizedResource(q.buildPath(), q.qsc.auth, true)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
// "The canonicalizedresouce portion of the string is a canonical path to the signed resource.
|
||
|
// It must include the service name (blob, table, queue or file) for version 2015-02-21 or
|
||
|
// later, the storage account name, and the resource name, and must be URL-decoded.
|
||
|
// -- https://msdn.microsoft.com/en-us/library/azure/dn140255.aspx
|
||
|
// We need to replace + with %2b first to avoid being treated as a space (which is correct for query strings, but not the path component).
|
||
|
canonicalizedResource = strings.Replace(canonicalizedResource, "+", "%2b", -1)
|
||
|
canonicalizedResource, err = url.QueryUnescape(canonicalizedResource)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
signedStart := ""
|
||
|
if options.Start != (time.Time{}) {
|
||
|
signedStart = options.Start.UTC().Format(time.RFC3339)
|
||
|
}
|
||
|
signedExpiry := options.Expiry.UTC().Format(time.RFC3339)
|
||
|
|
||
|
protocols := "https,http"
|
||
|
if options.UseHTTPS {
|
||
|
protocols = "https"
|
||
|
}
|
||
|
|
||
|
permissions := options.QueueSASPermissions.buildString()
|
||
|
stringToSign, err := queueSASStringToSign(q.qsc.client.apiVersion, canonicalizedResource, signedStart, signedExpiry, options.IP, permissions, protocols, options.Identifier)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
sig := q.qsc.client.computeHmac256(stringToSign)
|
||
|
sasParams := url.Values{
|
||
|
"sv": {q.qsc.client.apiVersion},
|
||
|
"se": {signedExpiry},
|
||
|
"sp": {permissions},
|
||
|
"sig": {sig},
|
||
|
}
|
||
|
|
||
|
if q.qsc.client.apiVersion >= "2015-04-05" {
|
||
|
sasParams.Add("spr", protocols)
|
||
|
addQueryParameter(sasParams, "sip", options.IP)
|
||
|
}
|
||
|
|
||
|
uri := q.qsc.client.getEndpoint(queueServiceName, q.buildPath(), nil)
|
||
|
sasURL, err := url.Parse(uri)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
sasURL.RawQuery = sasParams.Encode()
|
||
|
return sasURL.String(), nil
|
||
|
}
|
||
|
|
||
|
func queueSASStringToSign(signedVersion, canonicalizedResource, signedStart, signedExpiry, signedIP, signedPermissions, protocols, signedIdentifier string) (string, error) {
|
||
|
|
||
|
if signedVersion >= "2015-02-21" {
|
||
|
canonicalizedResource = "/queue" + canonicalizedResource
|
||
|
}
|
||
|
|
||
|
// https://msdn.microsoft.com/en-us/library/azure/dn140255.aspx#Anchor_12
|
||
|
if signedVersion >= "2015-04-05" {
|
||
|
return fmt.Sprintf("%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s",
|
||
|
signedPermissions,
|
||
|
signedStart,
|
||
|
signedExpiry,
|
||
|
canonicalizedResource,
|
||
|
signedIdentifier,
|
||
|
signedIP,
|
||
|
protocols,
|
||
|
signedVersion), nil
|
||
|
|
||
|
}
|
||
|
|
||
|
// reference: http://msdn.microsoft.com/en-us/library/azure/dn140255.aspx
|
||
|
if signedVersion >= "2013-08-15" {
|
||
|
return fmt.Sprintf("%s\n%s\n%s\n%s\n%s\n%s", signedPermissions, signedStart, signedExpiry, canonicalizedResource, signedIdentifier, signedVersion), nil
|
||
|
}
|
||
|
|
||
|
return "", errors.New("storage: not implemented SAS for versions earlier than 2013-08-15")
|
||
|
}
|