databag/net/server/internal/api_setCardStatus.go
2022-02-08 12:24:42 -08:00

134 lines
3.5 KiB
Go

package databag
import (
"errors"
"strconv"
"net/http"
"encoding/hex"
"gorm.io/gorm"
"github.com/gorilla/mux"
"databag/internal/store"
"github.com/theckman/go-securerandom"
)
func SetCardStatus(w http.ResponseWriter, r *http.Request) {
var res error
account, code, err := BearerAppToken(r, false);
if err != nil {
ErrResponse(w, code, err)
return
}
// scan parameters
params := mux.Vars(r)
cardId := params["cardId"]
token := r.FormValue("token")
// scan revisions
var viewRevision int64
view := r.FormValue("viewRevision")
if view != "" {
if viewRevision, res = strconv.ParseInt(view, 10, 64); res != nil {
ErrResponse(w, http.StatusBadRequest, res)
return
}
}
var articleRevision int64
article := r.FormValue("articleRevision")
if article != "" {
if articleRevision, res = strconv.ParseInt(article, 10, 64); res != nil {
ErrResponse(w, http.StatusBadRequest, res)
return
}
}
var channelRevision int64
channel := r.FormValue("channelRevision")
if channel != "" {
if channelRevision, res = strconv.ParseInt(channel, 10, 64); res != nil {
ErrResponse(w, http.StatusBadRequest, res)
return
}
}
var profileRevision int64
profile := r.FormValue("profileRevision")
if profile != "" {
if profileRevision, res = strconv.ParseInt(profile, 10, 64); res != nil {
ErrResponse(w, http.StatusBadRequest, res)
return
}
}
var status string
if err := ParseRequest(r, w, &status); err != nil {
ErrResponse(w, http.StatusBadRequest, err)
return
}
if !AppCardStatus(status) {
ErrResponse(w, http.StatusBadRequest, errors.New("unknown status"))
return
}
if status == APP_CARDCONNECTED && token == "" {
ErrResponse(w, http.StatusBadRequest, errors.New("connected token not set"))
return
}
// load referenced card
var slot store.CardSlot
if err := store.DB.Preload("Card.Groups").Where("account_id = ? AND card_slot_id = ?", account.ID, cardId).First(&slot).Error; err != nil {
if !errors.Is(err, gorm.ErrRecordNotFound) {
ErrResponse(w, http.StatusInternalServerError, err)
} else {
ErrResponse(w, http.StatusNotFound, err)
}
return
}
if slot.Card == nil {
ErrResponse(w, http.StatusNotFound, errors.New("card has been deleted"))
return
}
// update card
slot.Revision = account.CardRevision + 1
if token != "" {
slot.Card.OutToken = token
}
if status == APP_CARDCONNECTING {
if slot.Card.Status != APP_CARDCONNECTING && slot.Card.Status != APP_CARDCONNECTED {
data, err := securerandom.Bytes(APP_TOKENSIZE)
if err != nil {
ErrResponse(w, http.StatusInternalServerError, err)
return
}
slot.Card.InToken = hex.EncodeToString(data)
}
}
slot.Card.Status = status
slot.Card.NotifiedView = viewRevision
slot.Card.NotifiedArticle = articleRevision
slot.Card.NotifiedChannel = channelRevision
slot.Card.NotifiedProfile = profileRevision
// save and update contact revision
err = store.DB.Transaction(func(tx *gorm.DB) error {
if res := tx.Save(&slot.Card).Error; res != nil {
return res
}
if res := tx.Preload("Card").Save(&slot).Error; res != nil {
return res
}
if res := tx.Model(&account).Update("card_revision", account.CardRevision + 1).Error; res != nil {
return res
}
return nil
})
if err != nil {
ErrResponse(w, http.StatusInternalServerError, err)
return
}
SetStatus(account)
WriteResponse(w, getCardModel(&slot));
}