2022-07-06 23:19:05 +02:00
|
|
|
package auth
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-07-15 17:52:37 +02:00
|
|
|
"fmt"
|
2022-12-16 17:06:03 +02:00
|
|
|
"io"
|
2022-07-06 23:19:05 +02:00
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"golang.org/x/oauth2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// baseProvider defines common fields and methods used by OAuth2 client providers.
|
|
|
|
type baseProvider struct {
|
|
|
|
scopes []string
|
|
|
|
clientId string
|
|
|
|
clientSecret string
|
|
|
|
redirectUrl string
|
|
|
|
authUrl string
|
|
|
|
tokenUrl string
|
|
|
|
userApiUrl string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scopes implements Provider.Scopes interface.
|
|
|
|
func (p *baseProvider) Scopes() []string {
|
|
|
|
return p.scopes
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetScopes implements Provider.SetScopes interface.
|
|
|
|
func (p *baseProvider) SetScopes(scopes []string) {
|
|
|
|
p.scopes = scopes
|
|
|
|
}
|
|
|
|
|
|
|
|
// ClientId implements Provider.ClientId interface.
|
|
|
|
func (p *baseProvider) ClientId() string {
|
|
|
|
return p.clientId
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetClientId implements Provider.SetClientId interface.
|
|
|
|
func (p *baseProvider) SetClientId(clientId string) {
|
|
|
|
p.clientId = clientId
|
|
|
|
}
|
|
|
|
|
|
|
|
// ClientSecret implements Provider.ClientSecret interface.
|
|
|
|
func (p *baseProvider) ClientSecret() string {
|
|
|
|
return p.clientSecret
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetClientSecret implements Provider.SetClientSecret interface.
|
|
|
|
func (p *baseProvider) SetClientSecret(secret string) {
|
|
|
|
p.clientSecret = secret
|
|
|
|
}
|
|
|
|
|
|
|
|
// RedirectUrl implements Provider.RedirectUrl interface.
|
|
|
|
func (p *baseProvider) RedirectUrl() string {
|
|
|
|
return p.redirectUrl
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetRedirectUrl implements Provider.SetRedirectUrl interface.
|
|
|
|
func (p *baseProvider) SetRedirectUrl(url string) {
|
|
|
|
p.redirectUrl = url
|
|
|
|
}
|
|
|
|
|
|
|
|
// AuthUrl implements Provider.AuthUrl interface.
|
|
|
|
func (p *baseProvider) AuthUrl() string {
|
|
|
|
return p.authUrl
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetAuthUrl implements Provider.SetAuthUrl interface.
|
|
|
|
func (p *baseProvider) SetAuthUrl(url string) {
|
|
|
|
p.authUrl = url
|
|
|
|
}
|
|
|
|
|
|
|
|
// TokenUrl implements Provider.TokenUrl interface.
|
|
|
|
func (p *baseProvider) TokenUrl() string {
|
|
|
|
return p.tokenUrl
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetTokenUrl implements Provider.SetTokenUrl interface.
|
|
|
|
func (p *baseProvider) SetTokenUrl(url string) {
|
|
|
|
p.tokenUrl = url
|
|
|
|
}
|
|
|
|
|
|
|
|
// UserApiUrl implements Provider.UserApiUrl interface.
|
|
|
|
func (p *baseProvider) UserApiUrl() string {
|
|
|
|
return p.userApiUrl
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetUserApiUrl implements Provider.SetUserApiUrl interface.
|
|
|
|
func (p *baseProvider) SetUserApiUrl(url string) {
|
|
|
|
p.userApiUrl = url
|
|
|
|
}
|
|
|
|
|
|
|
|
// BuildAuthUrl implements Provider.BuildAuthUrl interface.
|
|
|
|
func (p *baseProvider) BuildAuthUrl(state string, opts ...oauth2.AuthCodeOption) string {
|
|
|
|
return p.oauth2Config().AuthCodeURL(state, opts...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FetchToken implements Provider.FetchToken interface.
|
|
|
|
func (p *baseProvider) FetchToken(code string, opts ...oauth2.AuthCodeOption) (*oauth2.Token, error) {
|
|
|
|
return p.oauth2Config().Exchange(context.Background(), code, opts...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Client implements Provider.Client interface.
|
|
|
|
func (p *baseProvider) Client(token *oauth2.Token) *http.Client {
|
|
|
|
return p.oauth2Config().Client(context.Background(), token)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FetchRawUserData implements Provider.FetchRawUserData interface.
|
2022-11-30 15:16:09 +02:00
|
|
|
func (p *baseProvider) FetchRawUserData(token *oauth2.Token) ([]byte, error) {
|
2022-11-13 14:20:11 +02:00
|
|
|
req, err := http.NewRequest("GET", p.userApiUrl, nil)
|
|
|
|
if err != nil {
|
2022-11-30 15:16:09 +02:00
|
|
|
return nil, err
|
2022-11-13 14:20:11 +02:00
|
|
|
}
|
|
|
|
|
2022-11-30 15:16:09 +02:00
|
|
|
return p.sendRawUserDataRequest(req, token)
|
2022-11-13 14:20:11 +02:00
|
|
|
}
|
|
|
|
|
2022-11-30 15:16:09 +02:00
|
|
|
// sendRawUserDataRequest sends the specified user data request and return its raw response body.
|
|
|
|
func (p *baseProvider) sendRawUserDataRequest(req *http.Request, token *oauth2.Token) ([]byte, error) {
|
2022-07-06 23:19:05 +02:00
|
|
|
client := p.Client(token)
|
|
|
|
|
2022-11-13 14:20:11 +02:00
|
|
|
response, err := client.Do(req)
|
2022-07-06 23:19:05 +02:00
|
|
|
if err != nil {
|
2022-11-30 15:16:09 +02:00
|
|
|
return nil, err
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
defer response.Body.Close()
|
|
|
|
|
2022-12-16 17:06:03 +02:00
|
|
|
result, err := io.ReadAll(response.Body)
|
2022-07-06 23:19:05 +02:00
|
|
|
if err != nil {
|
2022-11-30 15:16:09 +02:00
|
|
|
return nil, err
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
2022-07-15 17:52:37 +02:00
|
|
|
// http.Client.Get doesn't treat non 2xx responses as error
|
|
|
|
if response.StatusCode >= 400 {
|
2022-11-30 15:16:09 +02:00
|
|
|
return nil, fmt.Errorf(
|
2022-07-15 17:52:37 +02:00
|
|
|
"Failed to fetch OAuth2 user profile via %s (%d):\n%s",
|
|
|
|
p.userApiUrl,
|
|
|
|
response.StatusCode,
|
2022-11-30 15:16:09 +02:00
|
|
|
string(result),
|
2022-07-15 17:52:37 +02:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-11-30 15:16:09 +02:00
|
|
|
return result, nil
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// oauth2Config constructs a oauth2.Config instance based on the provider settings.
|
|
|
|
func (p *baseProvider) oauth2Config() *oauth2.Config {
|
|
|
|
return &oauth2.Config{
|
|
|
|
RedirectURL: p.redirectUrl,
|
|
|
|
ClientID: p.clientId,
|
|
|
|
ClientSecret: p.clientSecret,
|
|
|
|
Scopes: p.scopes,
|
|
|
|
Endpoint: oauth2.Endpoint{
|
|
|
|
AuthURL: p.authUrl,
|
|
|
|
TokenURL: p.tokenUrl,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|