mirror of
https://github.com/steveiliop56/tinyauth.git
synced 2026-03-22 22:47:52 +00:00
122 lines
2.5 KiB
Go
122 lines
2.5 KiB
Go
package service
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"github.com/steveiliop56/tinyauth/internal/config"
|
|
"github.com/steveiliop56/tinyauth/internal/utils/tlog"
|
|
)
|
|
|
|
type GithubEmailResponse []struct {
|
|
Email string `json:"email"`
|
|
Primary bool `json:"primary"`
|
|
}
|
|
|
|
type GithubUserInfoResponse struct {
|
|
Login string `json:"login"`
|
|
Name string `json:"name"`
|
|
ID int `json:"id"`
|
|
}
|
|
|
|
func defaultExtractor(client *http.Client, url string) (config.Claims, error) {
|
|
var claims config.Claims
|
|
|
|
res, err := client.Get(url)
|
|
if err != nil {
|
|
return config.Claims{}, err
|
|
}
|
|
defer res.Body.Close()
|
|
|
|
if res.StatusCode < 200 || res.StatusCode >= 300 {
|
|
return config.Claims{}, fmt.Errorf("request failed with status: %s", res.Status)
|
|
}
|
|
|
|
body, err := io.ReadAll(res.Body)
|
|
if err != nil {
|
|
return config.Claims{}, err
|
|
}
|
|
|
|
tlog.App.Trace().Str("body", string(body)).Msg("Userinfo response body")
|
|
|
|
err = json.Unmarshal(body, &claims)
|
|
if err != nil {
|
|
return config.Claims{}, err
|
|
}
|
|
|
|
return claims, nil
|
|
}
|
|
|
|
func githubExtractor(client *http.Client, url string) (config.Claims, error) {
|
|
var user config.Claims
|
|
|
|
userInfo, err := githubRequest[GithubUserInfoResponse](client, "https://api.github.com/user")
|
|
if err != nil {
|
|
return config.Claims{}, err
|
|
}
|
|
|
|
userEmails, err := githubRequest[GithubEmailResponse](client, "https://api.github.com/user/emails")
|
|
if err != nil {
|
|
return config.Claims{}, err
|
|
}
|
|
|
|
if len(userEmails) == 0 {
|
|
return user, errors.New("no emails found")
|
|
}
|
|
|
|
for _, email := range userEmails {
|
|
if email.Primary {
|
|
user.Email = email.Email
|
|
break
|
|
}
|
|
}
|
|
|
|
// Use first available email if no primary email was found
|
|
if user.Email == "" {
|
|
user.Email = userEmails[0].Email
|
|
}
|
|
|
|
user.PreferredUsername = userInfo.Login
|
|
user.Name = userInfo.Name
|
|
user.Sub = strconv.Itoa(userInfo.ID)
|
|
|
|
return user, nil
|
|
}
|
|
|
|
func githubRequest[T any](client *http.Client, url string) (T, error) {
|
|
var githubRes T
|
|
|
|
req, err := http.NewRequest("GET", "https://api.github.com/user", nil)
|
|
if err != nil {
|
|
return githubRes, err
|
|
}
|
|
|
|
req.Header.Set("Accept", "application/vnd.github+json")
|
|
|
|
res, err := client.Do(req)
|
|
if err != nil {
|
|
return githubRes, err
|
|
}
|
|
defer res.Body.Close()
|
|
|
|
if res.StatusCode < 200 || res.StatusCode >= 300 {
|
|
return githubRes, fmt.Errorf("request failed with status: %s", res.Status)
|
|
}
|
|
|
|
body, err := io.ReadAll(res.Body)
|
|
if err != nil {
|
|
return githubRes, err
|
|
}
|
|
|
|
err = json.Unmarshal(body, &githubRes)
|
|
if err != nil {
|
|
return githubRes, err
|
|
}
|
|
|
|
return githubRes, nil
|
|
}
|