// Copyright 2018 Google Inc. All Rights Reserved. // // 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 util import ( "bytes" "context" "encoding/json" "fmt" "runtime" "time" "github.com/GoogleCloudPlatform/cloudsql-proxy/logging" "golang.org/x/oauth2" exec "golang.org/x/sys/execabs" ) // GcloudConfigData represents the data returned by `gcloud config config-helper`. type GcloudConfigData struct { Configuration struct { Properties struct { Core struct { Project string Account string } } } Credential struct { AccessToken string `json:"access_token"` TokenExpiry time.Time `json:"token_expiry"` } } func (cfg *GcloudConfigData) oauthToken() *oauth2.Token { return &oauth2.Token{ AccessToken: cfg.Credential.AccessToken, Expiry: cfg.Credential.TokenExpiry, } } type GcloudStatusCode int const ( GcloudOk GcloudStatusCode = iota GcloudNotFound // generic execution failure error not specified above. GcloudExecErr ) type GcloudError struct { GcloudError error Status GcloudStatusCode } func (e *GcloudError) Error() string { return e.GcloudError.Error() } // GcloudConfig returns a GcloudConfigData object or an error of type *GcloudError. func GcloudConfig() (*GcloudConfigData, error) { gcloudCmd := "gcloud" if runtime.GOOS == "windows" { gcloudCmd = gcloudCmd + ".cmd" } if _, err := exec.LookPath(gcloudCmd); err != nil { return nil, &GcloudError{err, GcloudNotFound} } buf, errbuf := new(bytes.Buffer), new(bytes.Buffer) cmd := exec.Command(gcloudCmd, "--format", "json", "config", "config-helper", "--min-expiry", "1h") cmd.Stdout = buf cmd.Stderr = errbuf if err := cmd.Run(); err != nil { err = fmt.Errorf("error reading config: %v; stderr was:\n%v", err, errbuf) logging.Errorf("GcloudConfig: %v", err) return nil, &GcloudError{err, GcloudExecErr} } data := &GcloudConfigData{} if err := json.Unmarshal(buf.Bytes(), data); err != nil { logging.Errorf("Failed to unmarshal bytes from gcloud: %v", err) logging.Errorf(" gcloud returned:\n%s", buf) return nil, &GcloudError{err, GcloudExecErr} } return data, nil } // gcloudTokenSource implements oauth2.TokenSource via the `gcloud config config-helper` command. type gcloudTokenSource struct { } // Token helps gcloudTokenSource implement oauth2.TokenSource. func (src *gcloudTokenSource) Token() (*oauth2.Token, error) { cfg, err := GcloudConfig() if err != nil { return nil, err } return cfg.oauthToken(), nil } func GcloudTokenSource(ctx context.Context) (oauth2.TokenSource, error) { src := &gcloudTokenSource{} tok, err := src.Token() if err != nil { return nil, err } return oauth2.ReuseTokenSource(tok, src), nil }