2021-04-26 17:52:50 +06:00
|
|
|
package azure
|
2020-11-19 20:47:18 +06:00
|
|
|
|
|
|
|
import (
|
2022-12-15 07:51:07 -05:00
|
|
|
"errors"
|
2020-11-19 20:47:18 +06:00
|
|
|
"fmt"
|
2022-12-15 07:51:07 -05:00
|
|
|
"io"
|
2020-11-19 20:47:18 +06:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2022-12-15 07:51:07 -05:00
|
|
|
"strconv"
|
2020-11-19 20:47:18 +06:00
|
|
|
"strings"
|
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
"github.com/Azure/azure-sdk-for-go/sdk/azcore"
|
|
|
|
"github.com/Azure/azure-sdk-for-go/sdk/azidentity"
|
|
|
|
"github.com/Azure/azure-sdk-for-go/sdk/storage/azblob"
|
|
|
|
"github.com/Azure/azure-sdk-for-go/sdk/storage/azblob/blob"
|
|
|
|
"github.com/Azure/azure-sdk-for-go/sdk/storage/azblob/blockblob"
|
2022-12-15 20:57:12 +06:00
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
"github.com/imgproxy/imgproxy/v3/config"
|
|
|
|
"github.com/imgproxy/imgproxy/v3/ctxreader"
|
2022-12-15 20:57:12 +06:00
|
|
|
"github.com/imgproxy/imgproxy/v3/httprange"
|
2020-11-19 20:47:18 +06:00
|
|
|
)
|
|
|
|
|
2021-04-26 17:52:50 +06:00
|
|
|
type transport struct {
|
2022-12-15 07:51:07 -05:00
|
|
|
client *azblob.Client
|
2020-11-19 20:47:18 +06:00
|
|
|
}
|
|
|
|
|
2021-04-26 17:52:50 +06:00
|
|
|
func New() (http.RoundTripper, error) {
|
2022-12-15 07:51:07 -05:00
|
|
|
var (
|
|
|
|
client *azblob.Client
|
2022-12-15 20:57:12 +06:00
|
|
|
sharedKeyCredential *azblob.SharedKeyCredential
|
2022-12-15 07:51:07 -05:00
|
|
|
defaultAzureCredential *azidentity.DefaultAzureCredential
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
2022-12-15 20:57:12 +06:00
|
|
|
if len(config.ABSName) == 0 {
|
2022-12-15 07:51:07 -05:00
|
|
|
return nil, errors.New("IMGPROXY_ABS_NAME must be set")
|
2020-11-19 20:47:18 +06:00
|
|
|
}
|
|
|
|
|
2021-04-26 17:52:50 +06:00
|
|
|
endpoint := config.ABSEndpoint
|
2020-11-19 20:47:18 +06:00
|
|
|
if len(endpoint) == 0 {
|
2021-04-26 17:52:50 +06:00
|
|
|
endpoint = fmt.Sprintf("https://%s.blob.core.windows.net", config.ABSName)
|
2020-11-19 20:47:18 +06:00
|
|
|
}
|
2022-12-15 20:57:12 +06:00
|
|
|
|
2020-11-19 20:47:18 +06:00
|
|
|
endpointURL, err := url.Parse(endpoint)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-12-15 20:57:12 +06:00
|
|
|
if len(config.ABSKey) > 0 {
|
2022-12-15 07:51:07 -05:00
|
|
|
sharedKeyCredential, err = azblob.NewSharedKeyCredential(config.ABSName, config.ABSKey)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-11-19 20:47:18 +06:00
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
client, err = azblob.NewClientWithSharedKeyCredential(endpointURL.String(), sharedKeyCredential, nil)
|
|
|
|
} else {
|
|
|
|
defaultAzureCredential, err = azidentity.NewDefaultAzureCredential(nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-11-19 20:47:18 +06:00
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
client, err = azblob.NewClient(endpointURL.String(), defaultAzureCredential, nil)
|
|
|
|
}
|
2020-11-19 20:47:18 +06:00
|
|
|
|
2022-09-07 17:09:43 +06:00
|
|
|
if err != nil {
|
2022-12-15 07:51:07 -05:00
|
|
|
return nil, err
|
2022-09-07 17:09:43 +06:00
|
|
|
}
|
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
return transport{client}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t transport) RoundTrip(req *http.Request) (*http.Response, error) {
|
|
|
|
container := req.URL.Host
|
|
|
|
key := req.URL.Path
|
|
|
|
|
2022-12-15 20:57:12 +06:00
|
|
|
statusCode := http.StatusOK
|
2022-12-15 07:51:07 -05:00
|
|
|
|
|
|
|
header := make(http.Header)
|
|
|
|
opts := &blob.DownloadStreamOptions{}
|
|
|
|
|
|
|
|
if r := req.Header.Get("Range"); len(r) != 0 {
|
|
|
|
start, end, err := httprange.Parse(r)
|
|
|
|
if err != nil {
|
|
|
|
return httprange.InvalidHTTPRangeResponse(req), err
|
|
|
|
}
|
|
|
|
|
|
|
|
if end != 0 {
|
|
|
|
length := end - start + 1
|
|
|
|
if end <= 0 {
|
|
|
|
length = blockblob.CountToEnd
|
|
|
|
}
|
|
|
|
|
|
|
|
opts.Range = blob.HTTPRange{
|
|
|
|
Offset: start,
|
|
|
|
Count: length,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
statusCode = http.StatusPartialContent
|
2022-09-07 17:09:43 +06:00
|
|
|
}
|
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
result, err := t.client.DownloadStream(req.Context(), container, strings.TrimPrefix(key, "/"), opts)
|
2020-11-19 20:47:18 +06:00
|
|
|
if err != nil {
|
2022-12-15 07:51:07 -05:00
|
|
|
if azError, ok := err.(*azcore.ResponseError); !ok || azError.StatusCode < 100 || azError.StatusCode == 301 {
|
|
|
|
return nil, err
|
|
|
|
} else {
|
|
|
|
body := strings.NewReader(azError.Error())
|
|
|
|
return &http.Response{
|
|
|
|
StatusCode: azError.StatusCode,
|
|
|
|
Proto: "HTTP/1.0",
|
|
|
|
ProtoMajor: 1,
|
|
|
|
ProtoMinor: 0,
|
|
|
|
Header: header,
|
|
|
|
ContentLength: int64(body.Len()),
|
|
|
|
Body: io.NopCloser(body),
|
|
|
|
Close: false,
|
|
|
|
Request: req,
|
|
|
|
}, nil
|
|
|
|
}
|
2020-11-19 20:47:18 +06:00
|
|
|
}
|
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
if config.ETagEnabled && result.ETag != nil {
|
|
|
|
azETag := string(*result.ETag)
|
|
|
|
header.Set("ETag", azETag)
|
2021-09-29 16:23:54 +06:00
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
if etag := req.Header.Get("If-None-Match"); len(etag) > 0 && azETag == etag {
|
2021-09-29 16:23:54 +06:00
|
|
|
return &http.Response{
|
|
|
|
StatusCode: http.StatusNotModified,
|
|
|
|
Proto: "HTTP/1.0",
|
|
|
|
ProtoMajor: 1,
|
|
|
|
ProtoMinor: 0,
|
2022-12-15 07:51:07 -05:00
|
|
|
Header: header,
|
2021-09-29 16:23:54 +06:00
|
|
|
ContentLength: 0,
|
|
|
|
Body: nil,
|
|
|
|
Close: false,
|
|
|
|
Request: req,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-15 07:51:07 -05:00
|
|
|
header.Set("Accept-Ranges", "bytes")
|
|
|
|
|
2022-12-15 20:57:12 +06:00
|
|
|
contentLength := int64(0)
|
2022-12-15 07:51:07 -05:00
|
|
|
if result.ContentLength != nil {
|
2022-12-15 20:57:12 +06:00
|
|
|
contentLength = *result.ContentLength
|
2022-12-15 07:51:07 -05:00
|
|
|
header.Set("Content-Length", strconv.FormatInt(*result.ContentLength, 10))
|
|
|
|
}
|
|
|
|
|
|
|
|
if result.ContentType != nil {
|
|
|
|
header.Set("Content-Type", *result.ContentType)
|
|
|
|
}
|
|
|
|
|
|
|
|
if result.ContentRange != nil {
|
|
|
|
header.Set("Content-Range", *result.ContentRange)
|
|
|
|
}
|
|
|
|
|
|
|
|
if result.CacheControl != nil {
|
|
|
|
header.Set("Cache-Control", *result.CacheControl)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &http.Response{
|
|
|
|
StatusCode: statusCode,
|
|
|
|
Proto: "HTTP/1.0",
|
|
|
|
ProtoMajor: 1,
|
|
|
|
ProtoMinor: 0,
|
|
|
|
Header: header,
|
2022-12-15 20:57:12 +06:00
|
|
|
ContentLength: contentLength,
|
2022-12-15 07:51:07 -05:00
|
|
|
Body: ctxreader.New(req.Context(), result.Body, true),
|
|
|
|
Close: true,
|
|
|
|
Request: req,
|
|
|
|
}, nil
|
2020-11-19 20:47:18 +06:00
|
|
|
}
|