milvus/internal/storage/gcp/gcp.go

90 lines
2.6 KiB
Go
Raw Normal View History

package gcp
import (
"net/http"
"strings"
"github.com/cockroachdb/errors"
"github.com/minio/minio-go/v7"
"github.com/minio/minio-go/v7/pkg/credentials"
"go.uber.org/atomic"
"golang.org/x/oauth2"
"golang.org/x/oauth2/google"
)
// WrapHTTPTransport wraps http.Transport, add an auth header to support GCP native auth
type WrapHTTPTransport struct {
tokenSrc oauth2.TokenSource
backend transport
currentToken atomic.Pointer[oauth2.Token]
}
// transport abstracts http.Transport to simplify test
type transport interface {
RoundTrip(req *http.Request) (*http.Response, error)
}
// NewWrapHTTPTransport constructs a new WrapHTTPTransport
func NewWrapHTTPTransport(secure bool) (*WrapHTTPTransport, error) {
tokenSrc := google.ComputeTokenSource("")
// in fact never return err
backend, err := minio.DefaultTransport(secure)
if err != nil {
return nil, errors.Wrap(err, "failed to create default transport")
}
return &WrapHTTPTransport{
tokenSrc: tokenSrc,
backend: backend,
}, nil
}
// RoundTrip wraps original http.RoundTripper by Adding a Bearer token acquired from tokenSrc
func (t *WrapHTTPTransport) RoundTrip(req *http.Request) (*http.Response, error) {
// here Valid() means the token won't be expired in 10 sec
// so the http client timeout shouldn't be longer, or we need to change the default `expiryDelta` time
currentToken := t.currentToken.Load()
if currentToken.Valid() {
req.Header.Set("Authorization", "Bearer "+currentToken.AccessToken)
} else {
newToken, err := t.tokenSrc.Token()
if err != nil {
return nil, errors.Wrap(err, "failed to acquire token")
}
t.currentToken.Store(newToken)
req.Header.Set("Authorization", "Bearer "+newToken.AccessToken)
}
return t.backend.RoundTrip(req)
}
const GcsDefaultAddress = "storage.googleapis.com"
// NewMinioClient returns a minio.Client which is compatible for GCS
func NewMinioClient(address string, opts *minio.Options) (*minio.Client, error) {
if opts == nil {
opts = &minio.Options{}
}
if address == "" {
address = GcsDefaultAddress
opts.Secure = true
}
// adhoc to remove port of gcs address to let minio-go know it's gcs
if strings.Contains(address, GcsDefaultAddress) {
address = GcsDefaultAddress
}
if opts.Creds != nil {
// if creds is set, use it directly
return minio.New(address, opts)
}
// opts.Creds == nil, assume using IAM
transport, err := NewWrapHTTPTransport(opts.Secure)
if err != nil {
return nil, errors.Wrap(err, "failed to create default transport")
}
opts.Transport = transport
opts.Creds = credentials.NewStaticV2("", "", "")
return minio.New(address, opts)
}