1
0
mirror of https://github.com/pocketbase/pocketbase.git synced 2025-03-19 14:17:48 +02:00
pocketbase/tools/auth/discord.go

92 lines
2.5 KiB
Go
Raw Permalink Normal View History

2022-08-21 19:38:42 +03:00
package auth
import (
2023-03-01 23:29:45 +02:00
"context"
"encoding/json"
2022-08-21 19:38:42 +03:00
"fmt"
"github.com/pocketbase/pocketbase/tools/types"
2022-08-21 19:38:42 +03:00
"golang.org/x/oauth2"
)
2024-09-29 19:23:19 +03:00
func init() {
Providers[NameDiscord] = wrapFactory(NewDiscordProvider)
}
2022-08-21 19:38:42 +03:00
var _ Provider = (*Discord)(nil)
// NameDiscord is the unique name of the Discord provider.
const NameDiscord string = "discord"
// Discord allows authentication via Discord OAuth2.
type Discord struct {
2024-09-29 19:23:19 +03:00
BaseProvider
2022-08-21 19:38:42 +03:00
}
// NewDiscordProvider creates a new Discord provider instance with some defaults.
func NewDiscordProvider() *Discord {
// https://discord.com/developers/docs/topics/oauth2
// https://discord.com/developers/docs/resources/user#get-current-user
2024-09-29 19:23:19 +03:00
return &Discord{BaseProvider{
ctx: context.Background(),
displayName: "Discord",
pkce: true,
scopes: []string{"identify", "email"},
2024-09-29 19:23:19 +03:00
authURL: "https://discord.com/api/oauth2/authorize",
tokenURL: "https://discord.com/api/oauth2/token",
userInfoURL: "https://discord.com/api/users/@me",
2022-08-21 19:38:42 +03:00
}}
}
// FetchAuthUser returns an AuthUser instance from Discord's user api.
//
// API reference: https://discord.com/developers/docs/resources/user#user-object
2022-08-21 19:38:42 +03:00
func (p *Discord) FetchAuthUser(token *oauth2.Token) (*AuthUser, error) {
2024-09-29 19:23:19 +03:00
data, err := p.FetchRawUserInfo(token)
if err != nil {
return nil, err
}
rawUser := map[string]any{}
if err := json.Unmarshal(data, &rawUser); err != nil {
return nil, err
}
extracted := struct {
2022-08-21 19:38:42 +03:00
Id string `json:"id"`
Username string `json:"username"`
Discriminator string `json:"discriminator"`
Avatar string `json:"avatar"`
2022-08-21 19:38:42 +03:00
Email string `json:"email"`
Verified bool `json:"verified"`
2022-08-21 19:38:42 +03:00
}{}
if err := json.Unmarshal(data, &extracted); err != nil {
2022-08-21 19:38:42 +03:00
return nil, err
}
// Build a full avatar URL using the avatar hash provided in the API response
// https://discord.com/developers/docs/reference#image-formatting
2024-09-29 19:23:19 +03:00
avatarURL := fmt.Sprintf("https://cdn.discordapp.com/avatars/%s/%s.png", extracted.Id, extracted.Avatar)
2022-08-21 19:38:42 +03:00
// Concatenate the user's username and discriminator into a single username string
username := fmt.Sprintf("%s#%s", extracted.Username, extracted.Discriminator)
2022-08-21 19:38:42 +03:00
user := &AuthUser{
2023-01-07 22:25:56 +02:00
Id: extracted.Id,
Name: username,
Username: extracted.Username,
2024-09-29 19:23:19 +03:00
AvatarURL: avatarURL,
2023-01-07 22:25:56 +02:00
RawUser: rawUser,
AccessToken: token.AccessToken,
RefreshToken: token.RefreshToken,
}
user.Expiry, _ = types.ParseDateTime(token.Expiry)
if extracted.Verified {
user.Email = extracted.Email
2022-08-21 19:38:42 +03:00
}
return user, nil
}