Anitrack/SimklUserFunctions.go

232 lines
5.8 KiB
Go
Raw Permalink Normal View History

2024-07-30 12:57:08 -04:00
package main
2024-07-30 12:59:06 -04:00
import (
"bytes"
"context"
"encoding/json"
2025-01-17 20:39:18 -05:00
"errors"
2024-07-30 12:59:06 -04:00
"fmt"
"io"
"log"
"net/http"
"sync"
"github.com/99designs/keyring"
"github.com/wailsapp/wails/v2/pkg/runtime"
)
var simklJwt SimklJWT
var simklRing, _ = keyring.Open(keyring.Config{
2024-10-18 22:06:33 -04:00
ServiceName: "AniTrack",
KeychainName: "AniTrack",
KeychainSynchronizable: false,
KeychainTrustApplication: true,
KeychainAccessibleWhenUnlocked: true,
2024-07-30 12:59:06 -04:00
})
var simklCtxShutdown, simklCancel = context.WithCancel(context.Background())
func (a *App) CheckIfSimklLoggedIn() bool {
if (SimklJWT{} == simklJwt) {
2024-12-15 00:19:48 -05:00
tokenType, tokenTypeErr := simklRing.Get("SimklTokenType")
accessToken, accessTokenErr := simklRing.Get("SimklAccessToken")
scope, scopeErr := simklRing.Get("SimklScope")
if (tokenTypeErr != nil || accessTokenErr != nil || scopeErr != nil) || len(accessToken.Data) == 0 {
return false
} else {
simklJwt.TokenType = string(tokenType.Data)
simklJwt.AccessToken = string(accessToken.Data)
simklJwt.Scope = string(scope.Data)
return true
}
} else {
return true
}
}
2024-07-30 12:59:06 -04:00
func (a *App) SimklLogin() {
2024-12-15 00:19:48 -05:00
if !a.CheckIfSimklLoggedIn() {
tokenType, tokenTypeErr := simklRing.Get("SimklTokenType")
accessToken, accessTokenErr := simklRing.Get("SimklAccessToken")
scope, scopeErr := simklRing.Get("SimklScope")
if (tokenTypeErr != nil || accessTokenErr != nil || scopeErr != nil) || len(accessToken.Data) == 0 {
getSimklCodeUrl := "https://simkl.com/oauth/authorize?response_type=code&client_id=" + Environment.SIMKL_CLIENT_ID + "&redirect_uri=" + Environment.SIMKL_CALLBACK_URI
2024-09-07 22:35:51 -04:00
runtime.BrowserOpenURL(*wailsContext, getSimklCodeUrl)
2024-07-30 12:59:06 -04:00
serverDone := &sync.WaitGroup{}
serverDone.Add(1)
a.handleSimklCallback(serverDone)
2024-07-30 12:59:06 -04:00
serverDone.Wait()
} else {
simklJwt.TokenType = string(tokenType.Data)
simklJwt.AccessToken = string(accessToken.Data)
simklJwt.Scope = string(scope.Data)
}
}
}
func (a *App) handleSimklCallback(wg *sync.WaitGroup) {
mux := http.NewServeMux()
srv := &http.Server{Addr: ":6734", Handler: mux}
mux.HandleFunc("/callback", func(w http.ResponseWriter, r *http.Request) {
2024-07-30 12:59:06 -04:00
select {
case <-simklCtxShutdown.Done():
fmt.Println("Shutting down...")
return
default:
}
content := r.FormValue("code")
if content != "" {
simklJwt = getSimklAuthorizationToken(content)
_ = simklRing.Set(keyring.Item{
Key: "SimklTokenType",
Data: []byte(simklJwt.TokenType),
})
_ = simklRing.Set(keyring.Item{
Key: "SimklAccessToken",
Data: []byte(simklJwt.AccessToken),
})
_ = simklRing.Set(keyring.Item{
Key: "SimklScope",
Data: []byte(simklJwt.Scope),
})
2024-09-07 22:35:51 -04:00
_, err := runtime.MessageDialog(*wailsContext, runtime.MessageDialogOptions{
Title: "Simkl Authorization",
Message: "It is now safe to close your browser tab",
})
if err != nil {
log.Println(err)
}
2024-07-30 12:59:06 -04:00
fmt.Println("Shutting down...")
simklCancel()
err = srv.Shutdown(context.Background())
2024-07-30 12:59:06 -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()
2025-01-17 20:39:18 -05:00
if err := srv.ListenAndServe(); err != nil && !errors.Is(err, http.ErrServerClosed) {
2024-07-30 12:59:06 -04:00
log.Fatalf("listen: %s\n", err)
}
fmt.Println("Shutting down...")
}()
}
func getSimklAuthorizationToken(content string) SimklJWT {
data := 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"`
}{
GrantType: "authorization_code",
ClientID: Environment.SIMKL_CLIENT_ID,
ClientSecret: Environment.SIMKL_CLIENT_SECRET,
RedirectURI: Environment.SIMKL_CALLBACK_URI,
2024-07-30 12:59:06 -04:00
Code: content,
}
jsonData, err := json.Marshal(data)
if err != nil {
log.Fatal(err)
}
response, err := http.NewRequest("POST", "https://api.simkl.com/oauth/token", bytes.NewBuffer(jsonData))
if err != nil {
log.Printf("Failed at response, %s\n", err)
}
response.Header.Add("Content-Type", "application/json")
client := &http.Client{}
2025-01-17 20:39:18 -05:00
res, resErr := client.Do(response)
if resErr != nil {
2024-07-30 12:59:06 -04:00
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-07-30 12:59:06 -04:00
var post SimklJWT
err = json.Unmarshal(returnedBody, &post)
if err != nil {
log.Printf("Failed at unmarshal, %s\n", err)
}
return post
}
func (a *App) GetSimklLoggedInUser() SimklUser {
2024-07-30 12:59:06 -04:00
a.SimklLogin()
client := &http.Client{}
req, _ := http.NewRequest("POST", "https://api.simkl.com/users/settings", nil)
req.Header.Add("Content-Type", "application/json")
req.Header.Add("Authorization", "Bearer "+simklJwt.AccessToken)
req.Header.Add("simkl-api-key", Environment.SIMKL_CLIENT_ID)
response, err := client.Do(req)
if err != nil {
log.Printf("Failed at request, %s\n", err)
return SimklUser{}
2024-07-30 12:59:06 -04:00
}
defer response.Body.Close()
respBody, _ := io.ReadAll(response.Body)
2024-08-18 17:23:09 -04:00
var errCheck struct {
Error string `json:"error"`
Message string `json:"message"`
}
err = json.Unmarshal(respBody, &errCheck)
if err != nil {
log.Printf("Failed at unmarshal, %s\n", err)
}
if errCheck.Error != "" {
a.LogoutSimkl()
return SimklUser{}
}
var user SimklUser
err = json.Unmarshal(respBody, &user)
2024-07-30 12:59:06 -04:00
if err != nil {
log.Printf("Failed at unmarshal, %s\n", err)
}
return user
2024-07-30 12:59:06 -04:00
}
2024-08-15 16:16:40 -04:00
func (a *App) LogoutSimkl() string {
if (SimklJWT{} != simklJwt) {
2024-12-15 00:19:48 -05:00
tokenTypeErr := simklRing.Remove("SimklTokenType")
accessTokenErr := simklRing.Remove("SimklAccessToken")
scopeErr := simklRing.Remove("SimklScope")
2024-08-15 16:16:40 -04:00
2024-12-15 00:19:48 -05:00
if tokenTypeErr != nil || accessTokenErr != nil || scopeErr != nil {
fmt.Println("Simkl Logout Failed")
2024-08-15 16:16:40 -04:00
}
simklJwt = SimklJWT{}
2024-08-15 16:16:40 -04:00
}
return "Simkl Logged Out Successfully"
}