forked from TrueCloudLab/distribution
fix Context issue
Signed-off-by: Shawn Chen <chen8132@gmail.com>
This commit is contained in:
parent
ae91d1f429
commit
3390f32aec
6 changed files with 259 additions and 24 deletions
|
@ -1,12 +1,13 @@
|
||||||
package middleware
|
package middleware
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/docker/distribution/context"
|
dcontext "github.com/docker/distribution/context"
|
||||||
storagedriver "github.com/docker/distribution/registry/storage/driver"
|
storagedriver "github.com/docker/distribution/registry/storage/driver"
|
||||||
storagemiddleware "github.com/docker/distribution/registry/storage/driver/middleware"
|
storagemiddleware "github.com/docker/distribution/registry/storage/driver/middleware"
|
||||||
|
|
||||||
|
@ -99,7 +100,7 @@ func newAliCDNStorageMiddleware(storageDriver storagedriver.StorageDriver, optio
|
||||||
func (ac *aliCDNStorageMiddleware) URLFor(ctx context.Context, path string, options map[string]interface{}) (string, error) {
|
func (ac *aliCDNStorageMiddleware) URLFor(ctx context.Context, path string, options map[string]interface{}) (string, error) {
|
||||||
|
|
||||||
if ac.StorageDriver.Name() != "oss" {
|
if ac.StorageDriver.Name() != "oss" {
|
||||||
context.GetLogger(ctx).Warn("the AliCDN middleware does not support this backend storage driver")
|
dcontext.GetLogger(ctx).Warn("the AliCDN middleware does not support this backend storage driver")
|
||||||
return ac.StorageDriver.URLFor(ctx, path, options)
|
return ac.StorageDriver.URLFor(ctx, path, options)
|
||||||
}
|
}
|
||||||
acURL, err := ac.urlSigner.Sign(ac.baseURL+path, time.Now().Add(ac.duration))
|
acURL, err := ac.urlSigner.Sign(ac.baseURL+path, time.Now().Add(ac.duration))
|
||||||
|
|
97
vendor/github.com/denverdino/aliyungo/cdn/auth/random_uuid.go
generated
vendored
Normal file
97
vendor/github.com/denverdino/aliyungo/cdn/auth/random_uuid.go
generated
vendored
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
package auth
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/rand"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"syscall"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Bits is the number of bits in a UUID
|
||||||
|
Bits = 128
|
||||||
|
|
||||||
|
// Size is the number of bytes in a UUID
|
||||||
|
Size = Bits / 8
|
||||||
|
|
||||||
|
format = "%08x%04x%04x%04x%012x"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// Loggerf can be used to override the default logging destination. Such
|
||||||
|
// log messages in this library should be logged at warning or higher.
|
||||||
|
Loggerf = func(format string, args ...interface{}) {}
|
||||||
|
)
|
||||||
|
|
||||||
|
// UUID represents a UUID value. UUIDs can be compared and set to other values
|
||||||
|
// and accessed by byte.
|
||||||
|
type UUID [Size]byte
|
||||||
|
|
||||||
|
// GenerateUUID creates a new, version 4 uuid.
|
||||||
|
func GenerateUUID() (u UUID) {
|
||||||
|
const (
|
||||||
|
// ensures we backoff for less than 450ms total. Use the following to
|
||||||
|
// select new value, in units of 10ms:
|
||||||
|
// n*(n+1)/2 = d -> n^2 + n - 2d -> n = (sqrt(8d + 1) - 1)/2
|
||||||
|
maxretries = 9
|
||||||
|
backoff = time.Millisecond * 10
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
totalBackoff time.Duration
|
||||||
|
count int
|
||||||
|
retries int
|
||||||
|
)
|
||||||
|
|
||||||
|
for {
|
||||||
|
// This should never block but the read may fail. Because of this,
|
||||||
|
// we just try to read the random number generator until we get
|
||||||
|
// something. This is a very rare condition but may happen.
|
||||||
|
b := time.Duration(retries) * backoff
|
||||||
|
time.Sleep(b)
|
||||||
|
totalBackoff += b
|
||||||
|
|
||||||
|
n, err := io.ReadFull(rand.Reader, u[count:])
|
||||||
|
if err != nil {
|
||||||
|
if retryOnError(err) && retries < maxretries {
|
||||||
|
count += n
|
||||||
|
retries++
|
||||||
|
Loggerf("error generating version 4 uuid, retrying: %v", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Any other errors represent a system problem. What did someone
|
||||||
|
// do to /dev/urandom?
|
||||||
|
panic(fmt.Errorf("error reading random number generator, retried for %v: %v", totalBackoff.String(), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
u[6] = (u[6] & 0x0f) | 0x40 // set version byte
|
||||||
|
u[8] = (u[8] & 0x3f) | 0x80 // set high order byte 0b10{8,9,a,b}
|
||||||
|
|
||||||
|
return u
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u UUID) String() string {
|
||||||
|
return fmt.Sprintf(format, u[:4], u[4:6], u[6:8], u[8:10], u[10:])
|
||||||
|
}
|
||||||
|
|
||||||
|
// retryOnError tries to detect whether or not retrying would be fruitful.
|
||||||
|
func retryOnError(err error) bool {
|
||||||
|
switch err := err.(type) {
|
||||||
|
case *os.PathError:
|
||||||
|
return retryOnError(err.Err) // unpack the target error
|
||||||
|
case syscall.Errno:
|
||||||
|
if err == syscall.EPERM {
|
||||||
|
// EPERM represents an entropy pool exhaustion, a condition under
|
||||||
|
// which we backoff and retry.
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
80
vendor/github.com/denverdino/aliyungo/cdn/auth/sign_url.go
generated
vendored
Normal file
80
vendor/github.com/denverdino/aliyungo/cdn/auth/sign_url.go
generated
vendored
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
package auth
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/md5"
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// An URLSigner provides URL signing utilities to sign URLs for Aliyun CDN
|
||||||
|
// resources.
|
||||||
|
// authentication document: https://help.aliyun.com/document_detail/85117.html
|
||||||
|
type URLSigner struct {
|
||||||
|
authType string
|
||||||
|
privKey string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewURLSigner returns a new signer object.
|
||||||
|
func NewURLSigner(authType string, privKey string) *URLSigner {
|
||||||
|
return &URLSigner{
|
||||||
|
authType: authType,
|
||||||
|
privKey: privKey,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sign returns a signed aliyuncdn url based on authentication type
|
||||||
|
func (s URLSigner) Sign(uri string, expires time.Time) (string, error) {
|
||||||
|
r, err := url.Parse(uri)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("unable to parse url: %s", uri)
|
||||||
|
}
|
||||||
|
|
||||||
|
switch s.authType {
|
||||||
|
case "a":
|
||||||
|
return aTypeSign(r, s.privKey, expires), nil
|
||||||
|
case "b":
|
||||||
|
return bTypeSign(r, s.privKey, expires), nil
|
||||||
|
case "c":
|
||||||
|
return cTypeSign(r, s.privKey, expires), nil
|
||||||
|
default:
|
||||||
|
return "", fmt.Errorf("invalid authentication type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// sign by A type authentication method.
|
||||||
|
// authentication document: https://help.aliyun.com/document_detail/85113.html
|
||||||
|
func aTypeSign(r *url.URL, privateKey string, expires time.Time) string {
|
||||||
|
//rand is a random uuid without "-"
|
||||||
|
rand := GenerateUUID().String()
|
||||||
|
// not use, "0" by default
|
||||||
|
uid := "0"
|
||||||
|
secret := fmt.Sprintf("%s-%d-%s-%s-%s", r.Path, expires.Unix(), rand, uid, privateKey)
|
||||||
|
hashValue := md5.Sum([]byte(secret))
|
||||||
|
authKey := fmt.Sprintf("%d-%s-%s-%x", expires.Unix(), rand, uid, hashValue)
|
||||||
|
if r.RawQuery == "" {
|
||||||
|
return fmt.Sprintf("%s?auth_key=%s", r.String(), authKey)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s&auth_key=%s", r.String(), authKey)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// sign by B type authentication method.
|
||||||
|
// authentication document: https://help.aliyun.com/document_detail/85114.html
|
||||||
|
func bTypeSign(r *url.URL, privateKey string, expires time.Time) string {
|
||||||
|
formatExp := expires.Format("200601021504")
|
||||||
|
secret := privateKey + formatExp + r.Path
|
||||||
|
hashValue := md5.Sum([]byte(secret))
|
||||||
|
signURL := fmt.Sprintf("%s://%s/%s/%x%s?%s", r.Scheme, r.Host, formatExp, hashValue, r.Path, r.RawQuery)
|
||||||
|
return signURL
|
||||||
|
}
|
||||||
|
|
||||||
|
// sign by C type authentication method.
|
||||||
|
// authentication document: https://help.aliyun.com/document_detail/85115.html
|
||||||
|
func cTypeSign(r *url.URL, privateKey string, expires time.Time) string {
|
||||||
|
hexExp := fmt.Sprintf("%x", expires.Unix())
|
||||||
|
secret := privateKey + r.Path + hexExp
|
||||||
|
hashValue := md5.Sum([]byte(secret))
|
||||||
|
signURL := fmt.Sprintf("%s://%s/%x/%s%s?%s", r.Scheme, r.Host, hashValue, hexExp, r.Path, r.RawQuery)
|
||||||
|
return signURL
|
||||||
|
}
|
14
vendor/github.com/denverdino/aliyungo/oss/client.go
generated
vendored
14
vendor/github.com/denverdino/aliyungo/oss/client.go
generated
vendored
|
@ -851,6 +851,17 @@ func (b *Bucket) SignedURLWithArgs(path string, expires time.Time, params url.Va
|
||||||
return b.SignedURLWithMethod("GET", path, expires, params, headers)
|
return b.SignedURLWithMethod("GET", path, expires, params, headers)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Bucket) SignedURLWithMethodForAssumeRole(method, path string, expires time.Time, params url.Values, headers http.Header) string {
|
||||||
|
var uv = url.Values{}
|
||||||
|
if params != nil {
|
||||||
|
uv = params
|
||||||
|
}
|
||||||
|
if len(b.Client.SecurityToken) != 0 {
|
||||||
|
uv.Set("security-token", b.Client.SecurityToken)
|
||||||
|
}
|
||||||
|
return b.SignedURLWithMethod(method, path, expires, params, headers)
|
||||||
|
}
|
||||||
|
|
||||||
// SignedURLWithMethod returns a signed URL that allows anyone holding the URL
|
// SignedURLWithMethod returns a signed URL that allows anyone holding the URL
|
||||||
// to either retrieve the object at path or make a HEAD request against it. The signature is valid until expires.
|
// to either retrieve the object at path or make a HEAD request against it. The signature is valid until expires.
|
||||||
func (b *Bucket) SignedURLWithMethod(method, path string, expires time.Time, params url.Values, headers http.Header) string {
|
func (b *Bucket) SignedURLWithMethod(method, path string, expires time.Time, params url.Values, headers http.Header) string {
|
||||||
|
@ -1039,7 +1050,8 @@ func partiallyEscapedPath(path string) string {
|
||||||
func (client *Client) prepare(req *request) error {
|
func (client *Client) prepare(req *request) error {
|
||||||
// Copy so they can be mutated without affecting on retries.
|
// Copy so they can be mutated without affecting on retries.
|
||||||
headers := copyHeader(req.headers)
|
headers := copyHeader(req.headers)
|
||||||
if len(client.SecurityToken) != 0 {
|
// security-token should be in either Params or Header, cannot be in both
|
||||||
|
if len(req.params.Get("security-token")) == 0 && len(client.SecurityToken) != 0 {
|
||||||
headers.Set("x-oss-security-token", client.SecurityToken)
|
headers.Set("x-oss-security-token", client.SecurityToken)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
48
vendor/github.com/denverdino/aliyungo/oss/signature.go
generated
vendored
48
vendor/github.com/denverdino/aliyungo/oss/signature.go
generated
vendored
|
@ -13,26 +13,44 @@ const HeaderOSSPrefix = "x-oss-"
|
||||||
|
|
||||||
var ossParamsToSign = map[string]bool{
|
var ossParamsToSign = map[string]bool{
|
||||||
"acl": true,
|
"acl": true,
|
||||||
|
"append": true,
|
||||||
|
"bucketInfo": true,
|
||||||
|
"cname": true,
|
||||||
|
"comp": true,
|
||||||
|
"cors": true,
|
||||||
"delete": true,
|
"delete": true,
|
||||||
|
"endTime": true,
|
||||||
|
"img": true,
|
||||||
|
"lifecycle": true,
|
||||||
|
"live": true,
|
||||||
"location": true,
|
"location": true,
|
||||||
"logging": true,
|
"logging": true,
|
||||||
"notification": true,
|
"objectMeta": true,
|
||||||
"partNumber": true,
|
"partNumber": true,
|
||||||
"policy": true,
|
"position": true,
|
||||||
"requestPayment": true,
|
"qos": true,
|
||||||
"torrent": true,
|
"referer": true,
|
||||||
"uploadId": true,
|
"replication": true,
|
||||||
"uploads": true,
|
"replicationLocation": true,
|
||||||
"versionId": true,
|
"replicationProgress": true,
|
||||||
"versioning": true,
|
|
||||||
"versions": true,
|
|
||||||
"response-content-type": true,
|
|
||||||
"response-content-language": true,
|
|
||||||
"response-expires": true,
|
|
||||||
"response-cache-control": true,
|
"response-cache-control": true,
|
||||||
"response-content-disposition": true,
|
"response-content-disposition": true,
|
||||||
"response-content-encoding": true,
|
"response-content-encoding": true,
|
||||||
"bucketInfo": true,
|
"response-content-language": true,
|
||||||
|
"response-content-type": true,
|
||||||
|
"response-expires": true,
|
||||||
|
"security-token": true,
|
||||||
|
"startTime": true,
|
||||||
|
"status": true,
|
||||||
|
"style": true,
|
||||||
|
"styleName": true,
|
||||||
|
"symlink": true,
|
||||||
|
"tagging": true,
|
||||||
|
"uploadId": true,
|
||||||
|
"uploads": true,
|
||||||
|
"vod": true,
|
||||||
|
"website": true,
|
||||||
|
"x-oss-process": true,
|
||||||
}
|
}
|
||||||
|
|
||||||
func (client *Client) signRequest(request *request) {
|
func (client *Client) signRequest(request *request) {
|
||||||
|
@ -62,7 +80,7 @@ func (client *Client) signRequest(request *request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(params) > 0 {
|
if len(params) > 0 {
|
||||||
resource = resource + "?" + util.Encode(params)
|
resource = resource + "?" + util.EncodeWithoutEscape(params)
|
||||||
}
|
}
|
||||||
|
|
||||||
canonicalizedResource := resource
|
canonicalizedResource := resource
|
||||||
|
@ -74,7 +92,7 @@ func (client *Client) signRequest(request *request) {
|
||||||
//log.Println("stringToSign: ", stringToSign)
|
//log.Println("stringToSign: ", stringToSign)
|
||||||
signature := util.CreateSignature(stringToSign, client.AccessKeySecret)
|
signature := util.CreateSignature(stringToSign, client.AccessKeySecret)
|
||||||
|
|
||||||
if query.Get("OSSAccessKeyId") != "" {
|
if urlSignature {
|
||||||
query.Set("Signature", signature)
|
query.Set("Signature", signature)
|
||||||
} else {
|
} else {
|
||||||
headers.Set("Authorization", "OSS "+client.AccessKeyId+":"+signature)
|
headers.Set("Authorization", "OSS "+client.AccessKeyId+":"+signature)
|
||||||
|
|
37
vendor/github.com/denverdino/aliyungo/util/util.go
generated
vendored
37
vendor/github.com/denverdino/aliyungo/util/util.go
generated
vendored
|
@ -4,13 +4,13 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
srand "crypto/rand"
|
srand "crypto/rand"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"sort"
|
"sort"
|
||||||
"time"
|
"time"
|
||||||
"fmt"
|
|
||||||
"encoding/json"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const dictionary = "_0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"
|
const dictionary = "_0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"
|
||||||
|
@ -66,6 +66,34 @@ func Encode(v url.Values) string {
|
||||||
return buf.String()
|
return buf.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Like Encode, but key and value are not escaped
|
||||||
|
func EncodeWithoutEscape(v url.Values) string {
|
||||||
|
if v == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
var buf bytes.Buffer
|
||||||
|
keys := make([]string, 0, len(v))
|
||||||
|
for k := range v {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
sort.Strings(keys)
|
||||||
|
for _, k := range keys {
|
||||||
|
vs := v[k]
|
||||||
|
prefix := k
|
||||||
|
for _, v := range vs {
|
||||||
|
if buf.Len() > 0 {
|
||||||
|
buf.WriteByte('&')
|
||||||
|
}
|
||||||
|
buf.WriteString(prefix)
|
||||||
|
if v != "" {
|
||||||
|
buf.WriteString("=")
|
||||||
|
buf.WriteString(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return buf.String()
|
||||||
|
}
|
||||||
|
|
||||||
func GetGMTime() string {
|
func GetGMTime() string {
|
||||||
return time.Now().UTC().Format(http.TimeFormat)
|
return time.Now().UTC().Format(http.TimeFormat)
|
||||||
}
|
}
|
||||||
|
@ -148,11 +176,10 @@ func GenerateRandomECSPassword() string {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
func PrettyJson(object interface{}) string {
|
func PrettyJson(object interface{}) string {
|
||||||
b,err := json.MarshalIndent(object,"", " ")
|
b, err := json.MarshalIndent(object, "", " ")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("ERROR: PrettyJson, %v\n %s\n",err,b)
|
fmt.Printf("ERROR: PrettyJson, %v\n %s\n", err, b)
|
||||||
}
|
}
|
||||||
return string(b)
|
return string(b)
|
||||||
}
|
}
|
Loading…
Reference in a new issue