2014-05-13 14:06:46 -04:00
|
|
|
// Copyright 2014 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.
|
|
|
|
|
2014-05-05 17:54:23 -04:00
|
|
|
// Package google provides support for making
|
|
|
|
// OAuth2 authorized and authenticated HTTP requests
|
2014-05-10 07:50:51 -04:00
|
|
|
// to Google APIs. It supports Web server, client-side,
|
|
|
|
// service accounts, Google Compute Engine service accounts,
|
|
|
|
// and Google App Engine service accounts authorization
|
|
|
|
// and authentications flows:
|
2014-05-05 17:54:23 -04:00
|
|
|
//
|
|
|
|
// For more information, please read
|
|
|
|
// https://developers.google.com/accounts/docs/OAuth2.
|
|
|
|
//
|
|
|
|
// Example usage:
|
|
|
|
// // Web server flow usage:
|
|
|
|
// // Specify your configuration.
|
|
|
|
// // Your credentials should be obtained from the Google
|
|
|
|
// // Developer Console (https://console.developers.google.com).
|
|
|
|
// var config = google.NewConfig(&oauth2.Opts{
|
|
|
|
// ClientID: YOUR_CLIENT_ID,
|
|
|
|
// ClientSecret: YOUR_CLIENT_SECRET,
|
|
|
|
// RedirectURL: "http://you.example.org/handler",
|
|
|
|
// Scopes: []string{ "scope1", "scope2" },
|
|
|
|
// })
|
|
|
|
//
|
|
|
|
// // A landing page redirects to Google to get the auth code.
|
|
|
|
// func landing(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// http.Redirect(w, r, config.AuthCodeURL(""), http.StatusFound)
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// // The user will be redirected back to this handler, that takes the
|
|
|
|
// // "code" query parameter and Exchanges it for an access token.
|
|
|
|
// func handler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// t, err := config.NewTransportWithCode(r.FormValue("code"))
|
|
|
|
// // The Transport now has a valid Token. Create an *http.Client
|
|
|
|
// // with which we can make authenticated API requests.
|
|
|
|
// c := t.Client()
|
|
|
|
// c.Post(...)
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// // Service accounts usage:
|
|
|
|
// // Google Developer Console will provide a p12 file contains
|
|
|
|
// // a private key. You need to export it to the pem format.
|
|
|
|
// // Run the following command to generate a pem file that
|
|
|
|
// // contains your private key:
|
|
|
|
// // $ openssl pkcs12 -in /path/to/p12key.p12 -out key.pem -nodes
|
|
|
|
// // Then, specify your configuration.
|
|
|
|
// var config = google.NewServiceAccountConfig(&oauth2.JWTOpts{
|
|
|
|
// Email: "xxx@developer.gserviceaccount.com",
|
|
|
|
// PemFilename: "/path/to/key.pem",
|
|
|
|
// Scopes: []string{
|
|
|
|
// "https://www.googleapis.com/auth/drive.readonly"
|
|
|
|
// },
|
|
|
|
// })
|
|
|
|
//
|
|
|
|
// // Create a transport.
|
|
|
|
// t, err := config.NewTransport()
|
|
|
|
// // Or, you can create a transport that impersonates
|
|
|
|
// // a Google user.
|
|
|
|
// t, err := config.NewTransportWithUser(googleUserEmail)
|
|
|
|
//
|
|
|
|
// // Create a client to make authorized requests.
|
|
|
|
// c := t.Client()
|
|
|
|
// c.Post(...)
|
|
|
|
//
|
|
|
|
package google
|
|
|
|
|
|
|
|
import (
|
2014-05-11 11:06:03 -04:00
|
|
|
"encoding/json"
|
|
|
|
"net/http"
|
|
|
|
"path"
|
|
|
|
"time"
|
|
|
|
|
2014-05-05 17:54:23 -04:00
|
|
|
"github.com/rakyll/oauth2"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// Google endpoints.
|
|
|
|
uriGoogleAuth = "https://accounts.google.com/o/oauth2/auth"
|
|
|
|
uriGoogleToken = "https://accounts.google.com/o/oauth2/token"
|
|
|
|
)
|
|
|
|
|
2014-05-11 11:06:03 -04:00
|
|
|
type metaTokenRespBody struct {
|
|
|
|
AccessToken string `json:"access_token"`
|
|
|
|
ExpiresIn time.Duration `json:"expires_in"`
|
|
|
|
TokenType string `json:"token_type"`
|
|
|
|
}
|
|
|
|
|
2014-05-10 03:41:39 -04:00
|
|
|
// ComputeEngineConfig represents a OAuth 2.0 consumer client
|
|
|
|
// running on Google Compute Engine.
|
2014-05-11 11:06:03 -04:00
|
|
|
type ComputeEngineConfig struct {
|
|
|
|
account string
|
|
|
|
}
|
2014-05-10 03:41:39 -04:00
|
|
|
|
2014-05-05 17:54:23 -04:00
|
|
|
// NewConfig creates a new OAuth2 config that uses Google
|
|
|
|
// endpoints.
|
2014-05-10 07:16:50 -04:00
|
|
|
func NewConfig(opts *oauth2.Options) (*oauth2.Config, error) {
|
2014-05-05 17:54:23 -04:00
|
|
|
return oauth2.NewConfig(opts, uriGoogleAuth, uriGoogleToken)
|
|
|
|
}
|
|
|
|
|
2014-05-10 03:41:39 -04:00
|
|
|
// NewServiceAccountConfig creates a new JWT config that can
|
|
|
|
// fetch Bearer JWT tokens from Google endpoints.
|
2014-05-10 07:16:50 -04:00
|
|
|
func NewServiceAccountConfig(opts *oauth2.JWTOptions) (*oauth2.JWTConfig, error) {
|
2014-05-10 03:41:39 -04:00
|
|
|
return oauth2.NewJWTConfig(opts, uriGoogleToken)
|
|
|
|
}
|
|
|
|
|
2014-05-05 17:54:23 -04:00
|
|
|
// NewComputeEngineConfig creates a new config that can fetch tokens
|
2014-05-11 11:06:03 -04:00
|
|
|
// from Google Compute Engine instance's metaserver. If no account is
|
|
|
|
// provided, default is used.
|
|
|
|
func NewComputeEngineConfig(account string) (*ComputeEngineConfig, error) {
|
|
|
|
return &ComputeEngineConfig{account: account}, nil
|
2014-05-05 17:54:23 -04:00
|
|
|
}
|
|
|
|
|
2014-05-10 03:41:39 -04:00
|
|
|
// NewTransport creates an authorized transport.
|
|
|
|
func (c *ComputeEngineConfig) NewTransport() (oauth2.Transport, error) {
|
|
|
|
return oauth2.NewAuthorizedTransport(c, nil), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// FetchToken retrieves a new access token via metadata server.
|
2014-05-11 11:06:03 -04:00
|
|
|
func (c *ComputeEngineConfig) FetchToken(existing *oauth2.Token) (token *oauth2.Token, err error) {
|
|
|
|
account := "default"
|
|
|
|
if c.account != "" {
|
|
|
|
account = c.account
|
|
|
|
}
|
|
|
|
u := "http://" + path.Join("metadata/computeMetadata/v1/instance/service-accounts", account, "token")
|
|
|
|
req, err := http.NewRequest("GET", u, nil)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
req.Header.Add("X-Google-Metadata-Request", "True")
|
|
|
|
resp, err := (&http.Client{Transport: oauth2.DefaultTransport}).Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
var tokenResp metaTokenRespBody
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&tokenResp)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
token = &oauth2.Token{
|
|
|
|
AccessToken: tokenResp.AccessToken,
|
|
|
|
TokenType: tokenResp.TokenType,
|
|
|
|
Expiry: time.Now().Add(tokenResp.ExpiresIn * time.Second),
|
|
|
|
}
|
|
|
|
return
|
2014-05-05 17:54:23 -04:00
|
|
|
}
|