blob: 1a02d37d1c738b7df7ccd9bb28e97b0fe2731977 [file] [log] [blame]
// Copyright 2017 The LUCI Authors.
//
// 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.
package internal
import (
"context"
"fmt"
"net/http"
"golang.org/x/oauth2"
"google.golang.org/api/googleapi"
"go.chromium.org/luci/common/gcloud/iam"
"go.chromium.org/luci/common/retry/transient"
)
type iamTokenProvider struct {
actAs string
scopes []string
transport http.RoundTripper
cacheKey CacheKey
}
// NewIAMTokenProvider returns TokenProvider that uses generateAccessToken IAM
// API to grab tokens belonging to some service account.
func NewIAMTokenProvider(ctx context.Context, actAs string, scopes []string, transport http.RoundTripper) (TokenProvider, error) {
return &iamTokenProvider{
actAs: actAs,
scopes: scopes,
transport: transport,
cacheKey: CacheKey{
Key: fmt.Sprintf("iam/%s", actAs),
Scopes: scopes,
},
}, nil
}
func (p *iamTokenProvider) RequiresInteraction() bool {
return false
}
func (p *iamTokenProvider) Lightweight() bool {
return false
}
func (p *iamTokenProvider) Email() string {
return p.actAs
}
func (p *iamTokenProvider) CacheKey(ctx context.Context) (*CacheKey, error) {
return &p.cacheKey, nil
}
func (p *iamTokenProvider) MintToken(ctx context.Context, base *Token) (*Token, error) {
client := &iam.Client{
Client: &http.Client{
Transport: &tokenInjectingTransport{
transport: p.transport,
token: &base.Token,
},
},
}
tok, err := client.GenerateAccessToken(ctx, p.actAs, p.scopes, nil, 0)
if err == nil {
return &Token{
Token: *tok,
Email: p.Email(),
}, nil
}
// Any 4** HTTP response is a fatal error. Everything else is transient.
if apiErr, _ := err.(*googleapi.Error); apiErr != nil && apiErr.Code < 500 {
return nil, err
}
return nil, transient.Tag.Apply(err)
}
func (p *iamTokenProvider) RefreshToken(ctx context.Context, prev, base *Token) (*Token, error) {
// Service account tokens are self sufficient, there's no need for refresh
// token. Minting a token and "refreshing" it is a same thing.
return p.MintToken(ctx, base)
}
////////////////////////////////////////////////////////////////////////////////
type tokenInjectingTransport struct {
transport http.RoundTripper
token *oauth2.Token
}
func (t *tokenInjectingTransport) RoundTrip(req *http.Request) (*http.Response, error) {
clone := *req
clone.Header = make(http.Header, len(req.Header)+1)
for k, v := range req.Header {
clone.Header[k] = v
}
t.token.SetAuthHeader(&clone)
return t.transport.RoundTrip(&clone)
}