package gcp import ( "net/http" "strings" "github.com/minio/minio-go/v7" "github.com/minio/minio-go/v7/pkg/credentials" "github.com/pkg/errors" "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 *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 if !t.currentToken.Valid() { var err error t.currentToken, err = t.tokenSrc.Token() if err != nil { return nil, errors.Wrap(err, "failed to acquire token") } } req.Header.Set("Authorization", "Bearer "+t.currentToken.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) }