2024-08-13 18:54:27 -04:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/sha256"
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"math/rand"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/99designs/keyring"
|
|
|
|
"github.com/wailsapp/wails/v2/pkg/runtime"
|
|
|
|
)
|
|
|
|
|
|
|
|
var myAnimeListJwt MyAnimeListJWT
|
|
|
|
|
|
|
|
var myAnimeListRing, _ = keyring.Open(keyring.Config{
|
2024-10-18 22:06:33 -04:00
|
|
|
ServiceName: "AniTrack",
|
|
|
|
KeychainName: "AniTrack",
|
|
|
|
KeychainSynchronizable: false,
|
|
|
|
KeychainTrustApplication: true,
|
|
|
|
KeychainAccessibleWhenUnlocked: true,
|
2024-08-13 18:54:27 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
var myAnimeListCtxShutdown, myAnimeListCancel = context.WithCancel(context.Background())
|
|
|
|
|
|
|
|
type CodeVerifier struct {
|
|
|
|
Value string
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
length = 32
|
|
|
|
)
|
|
|
|
|
|
|
|
func base64URLEncode(str []byte) string {
|
|
|
|
encoded := base64.StdEncoding.EncodeToString(str)
|
|
|
|
encoded = strings.Replace(encoded, "+", "-", -1)
|
|
|
|
encoded = strings.Replace(encoded, "/", "_", -1)
|
|
|
|
encoded = strings.Replace(encoded, "=", "", -1)
|
|
|
|
return encoded
|
|
|
|
}
|
|
|
|
|
|
|
|
func verifier() (*CodeVerifier, error) {
|
|
|
|
r := rand.New(rand.NewSource(time.Now().UnixNano()))
|
2024-12-15 00:19:48 -05:00
|
|
|
b := make([]byte, length)
|
2024-08-13 18:54:27 -04:00
|
|
|
for i := 0; i < length; i++ {
|
|
|
|
b[i] = byte(r.Intn(255))
|
|
|
|
}
|
|
|
|
return CreateCodeVerifierFromBytes(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func CreateCodeVerifierFromBytes(b []byte) (*CodeVerifier, error) {
|
|
|
|
return &CodeVerifier{
|
|
|
|
Value: base64URLEncode(b),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *CodeVerifier) CodeChallengeS256() string {
|
|
|
|
h := sha256.New()
|
|
|
|
h.Write([]byte(v.Value))
|
|
|
|
return base64URLEncode(h.Sum(nil))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *App) CheckIfMyAnimeListLoggedIn() bool {
|
2024-12-15 00:19:48 -05:00
|
|
|
fmt.Println("check function reached")
|
2024-08-13 18:54:27 -04:00
|
|
|
if (MyAnimeListJWT{} == myAnimeListJwt) {
|
2024-12-15 00:19:48 -05:00
|
|
|
tokenType, tokenErr := myAnimeListRing.Get("MyAnimeListTokenType")
|
|
|
|
expiresIn, expiresInErr := myAnimeListRing.Get("MyAnimeListExpiresIn")
|
|
|
|
refreshToken, refreshTokenErr := myAnimeListRing.Get("MyAnimeListAccessToken")
|
|
|
|
accessToken, accessTokenErr := myAnimeListRing.Get("MyAnimeListRefreshToken")
|
|
|
|
if (tokenErr != nil || expiresInErr != nil || refreshTokenErr != nil || accessTokenErr != nil) || len(accessToken.Data) == 0 {
|
2024-08-13 18:54:27 -04:00
|
|
|
return false
|
|
|
|
} else {
|
2024-12-15 00:19:48 -05:00
|
|
|
var expresInConvertErr error
|
2024-08-13 18:54:27 -04:00
|
|
|
myAnimeListJwt.TokenType = string(tokenType.Data)
|
2024-12-15 00:19:48 -05:00
|
|
|
myAnimeListJwt.ExpiresIn, expresInConvertErr = strconv.Atoi(string(expiresIn.Data))
|
|
|
|
if expresInConvertErr != nil {
|
2024-08-13 18:54:27 -04:00
|
|
|
fmt.Println("unable to convert string to int")
|
|
|
|
}
|
|
|
|
myAnimeListJwt.AccessToken = string(accessToken.Data)
|
|
|
|
myAnimeListJwt.RefreshToken = string(refreshToken.Data)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *App) MyAnimeListLogin() {
|
2024-12-15 00:19:48 -05:00
|
|
|
fmt.Println("login function reached")
|
|
|
|
if !a.CheckIfMyAnimeListLoggedIn() {
|
|
|
|
fmt.Println("check logged in function failed")
|
|
|
|
tokenType, tokenErr := myAnimeListRing.Get("MyAnimeListTokenType")
|
|
|
|
expiresIn, expiresInErr := myAnimeListRing.Get("MyAnimeListExpiresIn")
|
|
|
|
refreshToken, refreshTokenErr := myAnimeListRing.Get("MyAnimeListAccessToken")
|
|
|
|
accessToken, accessTokenErr := myAnimeListRing.Get("MyAnimeListRefreshToken")
|
|
|
|
if (tokenErr != nil || expiresInErr != nil || refreshTokenErr != nil || accessTokenErr != nil) || len(accessToken.Data) == 0 {
|
2024-08-13 18:54:27 -04:00
|
|
|
verifier, _ := verifier()
|
2024-08-16 23:32:16 -04:00
|
|
|
getMyAnimeListCodeUrl := "https://myanimelist.net/v1/oauth2/authorize?response_type=code&client_id=" + Environment.MAL_CLIENT_ID + "&redirect_uri=" + Environment.MAL_CALLBACK_URI + "&code_challenge=" + verifier.Value + "&code_challenge_method=plain"
|
2024-09-07 22:35:51 -04:00
|
|
|
runtime.BrowserOpenURL(*wailsContext, getMyAnimeListCodeUrl)
|
2024-08-13 18:54:27 -04:00
|
|
|
serverDone := &sync.WaitGroup{}
|
|
|
|
serverDone.Add(1)
|
2024-08-13 20:05:25 -04:00
|
|
|
a.handleMyAnimeListCallback(serverDone, verifier)
|
2024-08-13 18:54:27 -04:00
|
|
|
serverDone.Wait()
|
|
|
|
} else {
|
2024-12-15 00:19:48 -05:00
|
|
|
var expresInConvertErr error
|
2024-08-13 18:54:27 -04:00
|
|
|
myAnimeListJwt.TokenType = string(tokenType.Data)
|
2024-12-15 00:19:48 -05:00
|
|
|
myAnimeListJwt.ExpiresIn, expresInConvertErr = strconv.Atoi(string(expiresIn.Data))
|
|
|
|
if expresInConvertErr != nil {
|
2024-08-13 18:54:27 -04:00
|
|
|
fmt.Println("unable to convert string to int in Login function")
|
|
|
|
}
|
|
|
|
myAnimeListJwt.AccessToken = string(accessToken.Data)
|
|
|
|
myAnimeListJwt.RefreshToken = string(refreshToken.Data)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-13 20:05:25 -04:00
|
|
|
func (a *App) handleMyAnimeListCallback(wg *sync.WaitGroup, verifier *CodeVerifier) {
|
2024-08-13 19:13:26 -04:00
|
|
|
mux := http.NewServeMux()
|
|
|
|
srv := &http.Server{Addr: ":6734", Handler: mux}
|
|
|
|
mux.HandleFunc("/callback", func(w http.ResponseWriter, r *http.Request) {
|
2024-08-13 18:54:27 -04:00
|
|
|
select {
|
|
|
|
case <-myAnimeListCtxShutdown.Done():
|
|
|
|
fmt.Println("Shutting down...")
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
content := r.FormValue("code")
|
|
|
|
|
|
|
|
if content != "" {
|
|
|
|
myAnimeListJwt = getMyAnimeListAuthorizationToken(content, verifier)
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListTokenType",
|
|
|
|
Data: []byte(myAnimeListJwt.TokenType),
|
|
|
|
})
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListExpiresIn",
|
|
|
|
Data: []byte(strconv.Itoa(myAnimeListJwt.ExpiresIn)),
|
|
|
|
})
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListAccessToken",
|
|
|
|
Data: []byte(myAnimeListJwt.AccessToken),
|
|
|
|
})
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListRefreshToken",
|
|
|
|
Data: []byte(myAnimeListJwt.RefreshToken),
|
|
|
|
})
|
2024-09-07 22:35:51 -04:00
|
|
|
_, err := runtime.MessageDialog(*wailsContext, runtime.MessageDialogOptions{
|
2024-08-13 20:05:25 -04:00
|
|
|
Title: "MyAnimeList Authorization",
|
|
|
|
Message: "It is now safe to close your browser tab",
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Println(err)
|
|
|
|
}
|
2024-08-13 18:54:27 -04:00
|
|
|
fmt.Println("Shutting down...")
|
|
|
|
myAnimeListCancel()
|
2024-08-13 20:05:25 -04:00
|
|
|
err = srv.Shutdown(context.Background())
|
2024-08-13 18:54:27 -04:00
|
|
|
if err != nil {
|
|
|
|
log.Println("server.Shutdown:", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
_, err := fmt.Fprintf(w, "Getting code failed.")
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
|
|
|
log.Fatalf("listen: %s\n", err)
|
|
|
|
}
|
|
|
|
fmt.Println("Shutting down...")
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
func getMyAnimeListAuthorizationToken(content string, verifier *CodeVerifier) MyAnimeListJWT {
|
|
|
|
dataForURLs := struct {
|
|
|
|
GrantType string `json:"grant_type"`
|
|
|
|
ClientID string `json:"client_id"`
|
|
|
|
ClientSecret string `json:"client_secret"`
|
|
|
|
RedirectURI string `json:"redirect_uri"`
|
|
|
|
Code string `json:"code"`
|
|
|
|
CodeVerifier *CodeVerifier `json:"code_verifier"`
|
|
|
|
}{
|
|
|
|
GrantType: "authorization_code",
|
2024-08-16 23:32:16 -04:00
|
|
|
ClientID: Environment.MAL_CLIENT_ID,
|
|
|
|
ClientSecret: Environment.MAL_CLIENT_SECRET,
|
|
|
|
RedirectURI: Environment.MAL_CALLBACK_URI,
|
2024-08-13 18:54:27 -04:00
|
|
|
Code: content,
|
|
|
|
CodeVerifier: verifier,
|
|
|
|
}
|
|
|
|
|
|
|
|
data := url.Values{}
|
|
|
|
data.Set("grant_type", dataForURLs.GrantType)
|
|
|
|
data.Set("client_id", dataForURLs.ClientID)
|
|
|
|
data.Set("client_secret", dataForURLs.ClientSecret)
|
|
|
|
data.Set("redirect_uri", dataForURLs.RedirectURI)
|
|
|
|
data.Set("code", dataForURLs.Code)
|
|
|
|
data.Set("code_verifier", dataForURLs.CodeVerifier.Value)
|
|
|
|
|
|
|
|
response, err := http.NewRequest("POST", "https://myanimelist.net/v1/oauth2/token", strings.NewReader(data.Encode()))
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed at response, %s\n", err)
|
|
|
|
}
|
|
|
|
response.Header.Add("Content-Type", "application/x-www-form-urlencoded")
|
|
|
|
|
|
|
|
client := &http.Client{}
|
|
|
|
res, reserr := client.Do(response)
|
|
|
|
if reserr != nil {
|
|
|
|
log.Printf("Failed at res, %s\n", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
defer res.Body.Close()
|
|
|
|
|
|
|
|
returnedBody, err := io.ReadAll(res.Body)
|
2024-12-15 00:19:48 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("Could not read returned body, %s\n", err)
|
|
|
|
}
|
2024-08-13 18:54:27 -04:00
|
|
|
|
|
|
|
var post MyAnimeListJWT
|
|
|
|
err = json.Unmarshal(returnedBody, &post)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed at unmarshal, %s\n", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return post
|
|
|
|
}
|
|
|
|
|
2024-09-18 14:05:41 -04:00
|
|
|
func refreshMyAnimeListAuthorizationToken() {
|
|
|
|
dataForURLs := struct {
|
|
|
|
GrantType string `json:"grant_type"`
|
|
|
|
RefreshToken string `json:"refresh_token"`
|
|
|
|
ClientID string `json:"client_id"`
|
|
|
|
ClientSecret string `json:"client_secret"`
|
|
|
|
RedirectURI string `json:"redirect_uri"`
|
|
|
|
}{
|
|
|
|
GrantType: "refresh_token",
|
|
|
|
RefreshToken: myAnimeListJwt.RefreshToken,
|
|
|
|
ClientID: Environment.MAL_CLIENT_ID,
|
|
|
|
ClientSecret: Environment.MAL_CLIENT_SECRET,
|
|
|
|
RedirectURI: Environment.MAL_CALLBACK_URI,
|
|
|
|
}
|
|
|
|
|
|
|
|
data := url.Values{}
|
|
|
|
data.Set("grant_type", dataForURLs.GrantType)
|
|
|
|
data.Set("refresh_token", dataForURLs.RefreshToken)
|
|
|
|
data.Set("client_id", dataForURLs.ClientID)
|
|
|
|
data.Set("client_secret", dataForURLs.ClientSecret)
|
|
|
|
data.Set("redirect_uri", dataForURLs.RedirectURI)
|
|
|
|
|
|
|
|
response, err := http.NewRequest("POST", "https://myanimelist.net/v1/oauth2/token", strings.NewReader(data.Encode()))
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed at response, %s\n", err)
|
|
|
|
}
|
|
|
|
response.Header.Add("Content-Type", "application/x-www-form-urlencoded")
|
|
|
|
|
|
|
|
client := &http.Client{}
|
|
|
|
res, reserr := client.Do(response)
|
|
|
|
if reserr != nil {
|
|
|
|
log.Printf("Failed at res, %s\n", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
defer res.Body.Close()
|
|
|
|
|
|
|
|
returnedBody, err := io.ReadAll(res.Body)
|
2024-12-15 00:19:48 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("Could not read returned body, %s\n", err)
|
|
|
|
}
|
2024-09-18 14:05:41 -04:00
|
|
|
|
|
|
|
err = json.Unmarshal(returnedBody, &myAnimeListJwt)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed at unmarshal, %s\n", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListTokenType",
|
|
|
|
Data: []byte(myAnimeListJwt.TokenType),
|
|
|
|
})
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListExpiresIn",
|
|
|
|
Data: []byte(strconv.Itoa(myAnimeListJwt.ExpiresIn)),
|
|
|
|
})
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListAccessToken",
|
|
|
|
Data: []byte(myAnimeListJwt.AccessToken),
|
|
|
|
})
|
|
|
|
_ = myAnimeListRing.Set(keyring.Item{
|
|
|
|
Key: "MyAnimeListRefreshToken",
|
|
|
|
Data: []byte(myAnimeListJwt.RefreshToken),
|
|
|
|
})
|
|
|
|
_, err = runtime.MessageDialog(*wailsContext, runtime.MessageDialogOptions{
|
|
|
|
Title: "MyAnimeList Authorization",
|
|
|
|
Message: "It is now safe to close your browser tab",
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-13 18:54:27 -04:00
|
|
|
func (a *App) GetMyAnimeListLoggedInUser() MyAnimeListUser {
|
|
|
|
a.MyAnimeListLogin()
|
|
|
|
|
|
|
|
client := &http.Client{}
|
|
|
|
|
|
|
|
req, _ := http.NewRequest("GET", "https://api.myanimelist.net/v2/users/@me?fields=anime_statistics", nil)
|
|
|
|
|
|
|
|
req.Header.Add("Content-Type", "application/json")
|
|
|
|
req.Header.Add("Authorization", "Bearer "+myAnimeListJwt.AccessToken)
|
2024-08-16 23:32:16 -04:00
|
|
|
req.Header.Add("myAnimeList-api-key", Environment.MAL_CLIENT_ID)
|
2024-08-13 18:54:27 -04:00
|
|
|
|
|
|
|
response, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed at request, %s\n", err)
|
|
|
|
return MyAnimeListUser{}
|
|
|
|
}
|
|
|
|
|
|
|
|
defer response.Body.Close()
|
|
|
|
|
|
|
|
var user MyAnimeListUser
|
|
|
|
|
|
|
|
respBody, _ := io.ReadAll(response.Body)
|
|
|
|
|
|
|
|
err = json.Unmarshal(respBody, &user)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Failed at unmarshal, %s\n", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return user
|
|
|
|
}
|
2024-08-15 16:16:40 -04:00
|
|
|
|
|
|
|
func (a *App) LogoutMyAnimeList() string {
|
|
|
|
if (MyAnimeListJWT{} != myAnimeListJwt) {
|
2024-12-15 00:19:48 -05:00
|
|
|
typeErr := myAnimeListRing.Remove("MyAnimeListTokenType")
|
|
|
|
expiresInErr := myAnimeListRing.Remove("MyAnimeListExpiresIn")
|
|
|
|
accessTokenErr := myAnimeListRing.Remove("MyAnimeListAccessToken")
|
|
|
|
refreshTokenErr := myAnimeListRing.Remove("MyAnimeListRefreshToken")
|
|
|
|
if typeErr != nil || expiresInErr != nil || accessTokenErr != nil || refreshTokenErr != nil {
|
|
|
|
fmt.Println("MAL Logout Failed")
|
2024-08-15 16:16:40 -04:00
|
|
|
}
|
2024-08-16 23:32:16 -04:00
|
|
|
myAnimeListJwt = MyAnimeListJWT{}
|
2024-08-15 16:16:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return "MAL Logged Out Successfully"
|
|
|
|
}
|