| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
- // SPDX-License-Identifier: Apache-2.0
- package oauth2 // import "miniflux.app/v2/internal/oauth2"
- import (
- "context"
- "encoding/json"
- "fmt"
- "net/http"
- "miniflux.app/v2/internal/model"
- "golang.org/x/oauth2"
- )
- // Google OAuth2 API documentation: https://developers.google.com/identity/protocols/oauth2
- const (
- googleAuthURL = "https://accounts.google.com/o/oauth2/v2/auth"
- googleTokenURL = "https://oauth2.googleapis.com/token"
- googleUserInfoURL = "https://www.googleapis.com/oauth2/v3/userinfo"
- )
- type googleProfile struct {
- Sub string `json:"sub"`
- Email string `json:"email"`
- }
- type googleProvider struct {
- clientID string
- clientSecret string
- redirectURL string
- }
- // NewGoogleProvider returns a Provider that authenticates users via Google OAuth2.
- func NewGoogleProvider(clientID, clientSecret, redirectURL string) Provider {
- return &googleProvider{clientID: clientID, clientSecret: clientSecret, redirectURL: redirectURL}
- }
- func (g *googleProvider) Config() *oauth2.Config {
- return &oauth2.Config{
- RedirectURL: g.redirectURL,
- ClientID: g.clientID,
- ClientSecret: g.clientSecret,
- Scopes: []string{"email"},
- Endpoint: oauth2.Endpoint{
- AuthURL: googleAuthURL,
- TokenURL: googleTokenURL,
- },
- }
- }
- func (g *googleProvider) UserExtraKey() string {
- return "google_id"
- }
- func (g *googleProvider) Profile(ctx context.Context, code, codeVerifier string) (*UserProfile, error) {
- conf := g.Config()
- token, err := conf.Exchange(ctx, code, oauth2.SetAuthURLParam("code_verifier", codeVerifier))
- if err != nil {
- return nil, fmt.Errorf("google: failed to exchange token: %w", err)
- }
- client := conf.Client(ctx, token)
- resp, err := client.Get(googleUserInfoURL)
- if err != nil {
- return nil, fmt.Errorf("google: failed to get user info: %w", err)
- }
- defer resp.Body.Close()
- if resp.StatusCode != http.StatusOK {
- return nil, fmt.Errorf("google: unexpected status code %d from userinfo endpoint", resp.StatusCode)
- }
- var user googleProfile
- decoder := json.NewDecoder(resp.Body)
- if err := decoder.Decode(&user); err != nil {
- return nil, fmt.Errorf("google: unable to unserialize Google profile: %w", err)
- }
- return &UserProfile{Key: g.UserExtraKey(), ID: user.Sub, Username: user.Email}, nil
- }
- func (g *googleProvider) PopulateUserCreationWithProfileID(user *model.UserCreationRequest, profile *UserProfile) {
- user.GoogleID = profile.ID
- }
- func (g *googleProvider) PopulateUserWithProfileID(user *model.User, profile *UserProfile) {
- user.GoogleID = profile.ID
- }
- func (g *googleProvider) UserProfileID(user *model.User) string {
- return user.GoogleID
- }
- func (g *googleProvider) UnsetUserProfileID(user *model.User) {
- user.GoogleID = ""
- }
|