2018-06-19 15:30:26 +02:00
|
|
|
// Copyright 2014 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// Package clientcredentials implements the OAuth2.0 "client credentials" token flow,
|
|
|
|
// also known as the "two-legged OAuth 2.0".
|
|
|
|
//
|
|
|
|
// This should be used when the client is acting on its own behalf or when the client
|
|
|
|
// is the resource owner. It may also be used when requesting access to protected
|
|
|
|
// resources based on an authorization previously arranged with the authorization
|
|
|
|
// server.
|
|
|
|
//
|
|
|
|
// See https://tools.ietf.org/html/rfc6749#section-4.4
|
|
|
|
package clientcredentials // import "golang.org/x/oauth2/clientcredentials"
|
|
|
|
|
|
|
|
import (
|
2019-03-17 20:19:56 +01:00
|
|
|
"context"
|
2018-06-19 15:30:26 +02:00
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"golang.org/x/oauth2"
|
|
|
|
"golang.org/x/oauth2/internal"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Config describes a 2-legged OAuth2 flow, with both the
|
|
|
|
// client application information and the server's endpoint URLs.
|
|
|
|
type Config struct {
|
|
|
|
// ClientID is the application's ID.
|
|
|
|
ClientID string
|
|
|
|
|
|
|
|
// ClientSecret is the application's secret.
|
|
|
|
ClientSecret string
|
|
|
|
|
|
|
|
// TokenURL is the resource server's token endpoint
|
|
|
|
// URL. This is a constant specific to each server.
|
|
|
|
TokenURL string
|
|
|
|
|
|
|
|
// Scope specifies optional requested permissions.
|
|
|
|
Scopes []string
|
|
|
|
|
|
|
|
// EndpointParams specifies additional parameters for requests to the token endpoint.
|
|
|
|
EndpointParams url.Values
|
2019-03-17 20:19:56 +01:00
|
|
|
|
|
|
|
// AuthStyle optionally specifies how the endpoint wants the
|
|
|
|
// client ID & client secret sent. The zero value means to
|
|
|
|
// auto-detect.
|
|
|
|
AuthStyle oauth2.AuthStyle
|
2018-06-19 15:30:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Token uses client credentials to retrieve a token.
|
2019-03-17 20:19:56 +01:00
|
|
|
//
|
|
|
|
// The provided context optionally controls which HTTP client is used. See the oauth2.HTTPClient variable.
|
2018-06-19 15:30:26 +02:00
|
|
|
func (c *Config) Token(ctx context.Context) (*oauth2.Token, error) {
|
|
|
|
return c.TokenSource(ctx).Token()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Client returns an HTTP client using the provided token.
|
2019-03-17 20:19:56 +01:00
|
|
|
// The token will auto-refresh as necessary.
|
|
|
|
//
|
|
|
|
// The provided context optionally controls which HTTP client
|
|
|
|
// is returned. See the oauth2.HTTPClient variable.
|
|
|
|
//
|
|
|
|
// The returned Client and its Transport should not be modified.
|
2018-06-19 15:30:26 +02:00
|
|
|
func (c *Config) Client(ctx context.Context) *http.Client {
|
|
|
|
return oauth2.NewClient(ctx, c.TokenSource(ctx))
|
|
|
|
}
|
|
|
|
|
|
|
|
// TokenSource returns a TokenSource that returns t until t expires,
|
|
|
|
// automatically refreshing it as necessary using the provided context and the
|
|
|
|
// client ID and client secret.
|
|
|
|
//
|
|
|
|
// Most users will use Config.Client instead.
|
|
|
|
func (c *Config) TokenSource(ctx context.Context) oauth2.TokenSource {
|
|
|
|
source := &tokenSource{
|
|
|
|
ctx: ctx,
|
|
|
|
conf: c,
|
|
|
|
}
|
|
|
|
return oauth2.ReuseTokenSource(nil, source)
|
|
|
|
}
|
|
|
|
|
|
|
|
type tokenSource struct {
|
|
|
|
ctx context.Context
|
|
|
|
conf *Config
|
|
|
|
}
|
|
|
|
|
|
|
|
// Token refreshes the token by using a new client credentials request.
|
|
|
|
// tokens received this way do not include a refresh token
|
|
|
|
func (c *tokenSource) Token() (*oauth2.Token, error) {
|
|
|
|
v := url.Values{
|
|
|
|
"grant_type": {"client_credentials"},
|
|
|
|
}
|
|
|
|
if len(c.conf.Scopes) > 0 {
|
|
|
|
v.Set("scope", strings.Join(c.conf.Scopes, " "))
|
|
|
|
}
|
|
|
|
for k, p := range c.conf.EndpointParams {
|
2019-03-17 20:19:56 +01:00
|
|
|
// Allow grant_type to be overridden to allow interoperability with
|
|
|
|
// non-compliant implementations.
|
|
|
|
if _, ok := v[k]; ok && k != "grant_type" {
|
2018-06-19 15:30:26 +02:00
|
|
|
return nil, fmt.Errorf("oauth2: cannot overwrite parameter %q", k)
|
|
|
|
}
|
|
|
|
v[k] = p
|
|
|
|
}
|
2019-03-17 20:19:56 +01:00
|
|
|
|
|
|
|
tk, err := internal.RetrieveToken(c.ctx, c.conf.ClientID, c.conf.ClientSecret, c.conf.TokenURL, v, internal.AuthStyle(c.conf.AuthStyle))
|
2018-06-19 15:30:26 +02:00
|
|
|
if err != nil {
|
|
|
|
if rErr, ok := err.(*internal.RetrieveError); ok {
|
|
|
|
return nil, (*oauth2.RetrieveError)(rErr)
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
t := &oauth2.Token{
|
|
|
|
AccessToken: tk.AccessToken,
|
|
|
|
TokenType: tk.TokenType,
|
|
|
|
RefreshToken: tk.RefreshToken,
|
|
|
|
Expiry: tk.Expiry,
|
|
|
|
}
|
|
|
|
return t.WithExtra(tk.Raw), nil
|
|
|
|
}
|