2021-07-13 17:14:25 +00:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2021-07-26 16:34:26 +00:00
|
|
|
"context"
|
2021-07-13 17:14:25 +00:00
|
|
|
"encoding/json"
|
2021-08-08 18:19:47 +00:00
|
|
|
"fmt"
|
2021-07-13 17:14:25 +00:00
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
|
2021-08-08 18:19:47 +00:00
|
|
|
"github.com/gorilla/mux"
|
2021-07-13 17:14:25 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2021-08-14 17:56:03 +00:00
|
|
|
|
|
|
|
"github.com/christianselig/apollo-backend/internal/domain"
|
2022-03-12 17:50:05 +00:00
|
|
|
"github.com/christianselig/apollo-backend/internal/reddit"
|
2021-07-13 17:14:25 +00:00
|
|
|
)
|
|
|
|
|
2021-10-17 16:41:12 +00:00
|
|
|
type accountNotificationsRequest struct {
|
2022-03-12 17:50:05 +00:00
|
|
|
InboxNotifications bool `json:"inbox_notifications"`
|
|
|
|
WatcherNotifications bool `json:"watcher_notifications"`
|
2022-03-26 16:39:28 +00:00
|
|
|
GlobalMute bool `json:"global_mute"`
|
2021-10-17 16:41:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *api) notificationsAccountHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
anr := &accountNotificationsRequest{}
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(anr); err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
apns := vars["apns"]
|
|
|
|
rid := vars["redditID"]
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
dev, err := a.deviceRepo.GetByAPNSToken(ctx, apns)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
acct, err := a.accountRepo.GetByRedditID(ctx, rid)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-26 16:39:28 +00:00
|
|
|
if err := a.deviceRepo.SetNotifiable(ctx, &dev, &acct, anr.InboxNotifications, anr.WatcherNotifications, anr.GlobalMute); err != nil {
|
2021-10-17 16:41:12 +00:00
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
}
|
|
|
|
|
2022-03-12 17:50:05 +00:00
|
|
|
func (a *api) getNotificationsAccountHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
apns := vars["apns"]
|
|
|
|
rid := vars["redditID"]
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
dev, err := a.deviceRepo.GetByAPNSToken(ctx, apns)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
acct, err := a.accountRepo.GetByRedditID(ctx, rid)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-26 16:39:28 +00:00
|
|
|
inbox, watchers, global, err := a.deviceRepo.GetNotifiable(ctx, &dev, &acct)
|
2022-03-12 17:50:05 +00:00
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
|
2022-03-26 16:39:28 +00:00
|
|
|
an := &accountNotificationsRequest{InboxNotifications: inbox, WatcherNotifications: watchers, GlobalMute: global}
|
2022-03-12 17:50:05 +00:00
|
|
|
_ = json.NewEncoder(w).Encode(an)
|
|
|
|
}
|
|
|
|
|
2021-08-08 18:19:47 +00:00
|
|
|
func (a *api) disassociateAccountHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
apns := vars["apns"]
|
|
|
|
rid := vars["redditID"]
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
dev, err := a.deviceRepo.GetByAPNSToken(ctx, apns)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
acct, err := a.accountRepo.GetByRedditID(ctx, rid)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := a.accountRepo.Disassociate(ctx, &acct, &dev); err != nil {
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *api) upsertAccountsHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
apns := vars["apns"]
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
dev, err := a.deviceRepo.GetByAPNSToken(ctx, apns)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
laccs, err := a.accountRepo.GetByAPNSToken(ctx, apns)
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
accsMap := map[string]domain.Account{}
|
|
|
|
for _, acc := range laccs {
|
|
|
|
accsMap[acc.NormalizedUsername()] = acc
|
|
|
|
}
|
|
|
|
|
|
|
|
var raccs []domain.Account
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&raccs); err != nil {
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, acc := range raccs {
|
|
|
|
delete(accsMap, acc.NormalizedUsername())
|
|
|
|
|
2022-03-12 17:50:05 +00:00
|
|
|
ac := a.reddit.NewAuthenticatedClient(reddit.SkipRateLimiting, acc.RefreshToken, acc.AccessToken)
|
2021-08-08 18:19:47 +00:00
|
|
|
tokens, err := ac.RefreshTokens()
|
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset expiration timer
|
|
|
|
acc.ExpiresAt = time.Now().Unix() + 3540
|
|
|
|
acc.RefreshToken = tokens.RefreshToken
|
|
|
|
acc.AccessToken = tokens.AccessToken
|
|
|
|
|
2022-03-12 17:50:05 +00:00
|
|
|
ac = a.reddit.NewAuthenticatedClient(reddit.SkipRateLimiting, acc.RefreshToken, acc.AccessToken)
|
2021-08-08 18:19:47 +00:00
|
|
|
me, err := ac.Me()
|
|
|
|
|
2021-09-25 13:19:42 +00:00
|
|
|
if err != nil {
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-08 18:19:47 +00:00
|
|
|
if me.NormalizedUsername() != acc.NormalizedUsername() {
|
|
|
|
a.errorResponse(w, r, 422, "nice try")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-14 15:21:17 +00:00
|
|
|
// Set account ID from Reddit
|
|
|
|
acc.AccountID = me.ID
|
|
|
|
|
2021-08-08 18:19:47 +00:00
|
|
|
if err := a.accountRepo.CreateOrUpdate(ctx, &acc); err != nil {
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-09-25 13:19:42 +00:00
|
|
|
_ = a.accountRepo.Associate(ctx, &acc, &dev)
|
2021-08-08 18:19:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, acc := range accsMap {
|
|
|
|
fmt.Println(acc.NormalizedUsername())
|
2021-09-25 13:19:42 +00:00
|
|
|
_ = a.accountRepo.Disassociate(ctx, &acc, &dev)
|
2021-08-08 18:19:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *api) upsertAccountHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
|
2021-07-26 16:34:26 +00:00
|
|
|
ctx := context.Background()
|
|
|
|
var acct domain.Account
|
|
|
|
|
2021-08-08 18:19:47 +00:00
|
|
|
if err := json.NewDecoder(r.Body).Decode(&acct); err != nil {
|
2021-07-13 17:14:25 +00:00
|
|
|
a.logger.WithFields(logrus.Fields{
|
|
|
|
"err": err,
|
|
|
|
}).Info("failed to parse request json")
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Here we check whether the account is supplied with a valid token.
|
2022-03-12 17:50:05 +00:00
|
|
|
ac := a.reddit.NewAuthenticatedClient(reddit.SkipRateLimiting, acct.RefreshToken, acct.AccessToken)
|
2021-07-13 17:14:25 +00:00
|
|
|
tokens, err := ac.RefreshTokens()
|
|
|
|
if err != nil {
|
|
|
|
a.logger.WithFields(logrus.Fields{
|
|
|
|
"err": err,
|
|
|
|
}).Info("failed to refresh token")
|
|
|
|
a.errorResponse(w, r, 422, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset expiration timer
|
|
|
|
acct.ExpiresAt = time.Now().Unix() + 3540
|
|
|
|
acct.RefreshToken = tokens.RefreshToken
|
|
|
|
acct.AccessToken = tokens.AccessToken
|
|
|
|
|
2022-03-12 17:50:05 +00:00
|
|
|
ac = a.reddit.NewAuthenticatedClient(reddit.SkipRateLimiting, acct.RefreshToken, acct.AccessToken)
|
2021-07-13 17:14:25 +00:00
|
|
|
me, err := ac.Me()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
a.logger.WithFields(logrus.Fields{
|
|
|
|
"err": err,
|
|
|
|
}).Info("failed to grab user details from Reddit")
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if me.NormalizedUsername() != acct.NormalizedUsername() {
|
|
|
|
a.logger.WithFields(logrus.Fields{
|
|
|
|
"err": err,
|
|
|
|
}).Info("user is not who they say they are")
|
|
|
|
a.errorResponse(w, r, 422, "nice try")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set account ID from Reddit
|
|
|
|
acct.AccountID = me.ID
|
|
|
|
|
|
|
|
// Associate
|
2021-08-08 18:19:47 +00:00
|
|
|
dev, err := a.deviceRepo.GetByAPNSToken(ctx, vars["apns"])
|
2021-07-13 17:14:25 +00:00
|
|
|
if err != nil {
|
|
|
|
a.logger.WithFields(logrus.Fields{
|
|
|
|
"err": err,
|
|
|
|
}).Info("failed fetching device from database")
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Upsert account
|
2021-07-26 16:34:26 +00:00
|
|
|
if err := a.accountRepo.CreateOrUpdate(ctx, &acct); err != nil {
|
2021-07-13 17:14:25 +00:00
|
|
|
a.logger.WithFields(logrus.Fields{
|
|
|
|
"err": err,
|
|
|
|
}).Info("failed updating account in database")
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-08 18:19:47 +00:00
|
|
|
if err := a.accountRepo.Associate(ctx, &acct, &dev); err != nil {
|
2021-07-13 17:14:25 +00:00
|
|
|
a.logger.WithFields(logrus.Fields{
|
|
|
|
"err": err,
|
|
|
|
}).Info("failed associating account with device")
|
|
|
|
a.errorResponse(w, r, 500, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
}
|