SimpleCloudNotifier/scnserver/api/handler/compat.go

935 lines
29 KiB
Go
Raw Normal View History

2022-11-13 19:17:07 +01:00
package handler
import (
"blackforestbytes.com/simplecloudnotifier/api/apierr"
hl "blackforestbytes.com/simplecloudnotifier/api/apihighlight"
2022-12-20 13:55:09 +01:00
"blackforestbytes.com/simplecloudnotifier/api/ginresp"
ct "blackforestbytes.com/simplecloudnotifier/db/cursortoken"
primarydb "blackforestbytes.com/simplecloudnotifier/db/impl/primary"
2022-11-13 19:17:07 +01:00
"blackforestbytes.com/simplecloudnotifier/logic"
2022-11-19 15:13:47 +01:00
"blackforestbytes.com/simplecloudnotifier/models"
2022-11-20 01:28:32 +01:00
"database/sql"
"fmt"
2022-11-13 19:17:07 +01:00
"github.com/gin-gonic/gin"
2022-11-20 01:28:32 +01:00
"gogs.mikescher.com/BlackForestBytes/goext/dataext"
"gogs.mikescher.com/BlackForestBytes/goext/langext"
"net/http"
2022-11-13 19:17:07 +01:00
)
type CompatHandler struct {
2022-11-20 01:28:32 +01:00
app *logic.Application
database *primarydb.Database
2022-11-13 19:17:07 +01:00
}
func NewCompatHandler(app *logic.Application) CompatHandler {
return CompatHandler{
2022-11-20 01:28:32 +01:00
app: app,
database: app.Database.Primary,
2022-11-13 19:17:07 +01:00
}
}
// SendMessageCompat swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Deprecated
//
2023-04-21 21:45:16 +02:00
// @Summary Send a new message (compatibility)
// @Description All parameter can be set via query-parameter or form-data body. Only UserID, UserKey and Title are required
// @Tags External
//
2023-04-21 21:45:16 +02:00
// @Param query_data query handler.SendMessageCompat.combined false " "
// @Param form_data formData handler.SendMessageCompat.combined false " "
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.SendMessageCompat.response
// @Failure 400 {object} ginresp.apiError
// @Failure 401 {object} ginresp.apiError
// @Failure 403 {object} ginresp.apiError
// @Failure 500 {object} ginresp.apiError
//
2023-04-21 21:45:16 +02:00
// @Router /send.php [POST]
func (h MessageHandler) SendMessageCompat(g *gin.Context) ginresp.HTTPResponse {
type combined struct {
UserID *int64 `json:"user_id" form:"user_id"`
UserKey *string `json:"user_key" form:"user_key"`
Title *string `json:"title" form:"title"`
Content *string `json:"content" form:"content"`
Priority *int `json:"priority" form:"priority"`
UserMessageID *string `json:"msg_id" form:"msg_id"`
SendTimestamp *float64 `json:"timestamp" form:"timestamp"`
}
type response struct {
Success bool `json:"success"`
ErrorID apierr.APIError `json:"error"`
ErrorHighlight int `json:"errhighlight"`
Message string `json:"message"`
SuppressSend bool `json:"suppress_send"`
MessageCount int `json:"messagecount"`
Quota int `json:"quota"`
IsPro bool `json:"is_pro"`
QuotaMax int `json:"quota_max"`
SCNMessageID int64 `json:"scn_msg_id"`
}
var f combined
var q combined
ctx, errResp := h.app.StartRequest(g, nil, &q, nil, &f, logic.RequestOptions{IgnoreWrongContentType: true})
if errResp != nil {
return *errResp
}
defer ctx.Cancel()
data := dataext.ObjectMerge(f, q)
newid, err := h.database.ConvertCompatID(ctx, langext.Coalesce(data.UserID, -1), "userid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query userid<old>", err)
}
if newid == nil {
return ginresp.SendAPIError(g, 400, apierr.USER_NOT_FOUND, hl.USER_ID, "User not found (compat)", nil)
}
2023-04-21 21:45:16 +02:00
okResp, errResp := h.sendMessageInternal(g, ctx, langext.Ptr(models.UserID(*newid)), data.UserKey, nil, data.Title, data.Content, data.Priority, data.UserMessageID, data.SendTimestamp, nil)
if errResp != nil {
return *errResp
} else {
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
ErrorID: apierr.NO_ERROR,
ErrorHighlight: -1,
Message: langext.Conditional(okResp.MessageIsOld, "Message already sent", "Message sent"),
SuppressSend: okResp.MessageIsOld,
MessageCount: okResp.User.MessagesSent,
Quota: okResp.User.QuotaUsedToday(),
IsPro: okResp.User.IsPro,
QuotaMax: okResp.User.QuotaPerDay(),
SCNMessageID: okResp.CompatMessageID,
}))
}
}
2022-11-13 19:17:07 +01:00
// Register swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Summary Register a new account
// @ID compat-register
// @Tags API-v1
2022-11-23 19:32:23 +01:00
//
2023-04-21 21:45:16 +02:00
// @Deprecated
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param fcm_token query string true "the (android) fcm token"
// @Param pro query string true "if the user is a paid account" Enums(true, false)
// @Param pro_token query string true "the (android) IAP token"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param fcm_token formData string true "the (android) fcm token"
// @Param pro formData string true "if the user is a paid account" Enums(true, false)
// @Param pro_token formData string true "the (android) IAP token"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.Register.response
// @Failure default {object} ginresp.compatAPIError
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Router /api/register.php [get]
2022-11-13 19:17:07 +01:00
func (h CompatHandler) Register(g *gin.Context) ginresp.HTTPResponse {
type query struct {
FCMToken *string `json:"fcm_token" form:"fcm_token"`
Pro *string `json:"pro" form:"pro"`
ProToken *string `json:"pro_token" form:"pro_token"`
2022-11-13 19:17:07 +01:00
}
type response struct {
Success bool `json:"success"`
2022-11-13 19:17:07 +01:00
Message string `json:"message"`
2022-11-20 01:28:32 +01:00
UserID int64 `json:"user_id"`
2022-11-13 19:17:07 +01:00
UserKey string `json:"user_key"`
QuotaUsed int `json:"quota"`
QuotaMax int `json:"quota_max"`
2023-01-17 22:32:13 +01:00
IsPro bool `json:"is_pro"`
2022-11-13 19:17:07 +01:00
}
2022-11-20 01:28:32 +01:00
var datq query
var datb query
ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true})
2022-11-20 01:28:32 +01:00
if errResp != nil {
return *errResp
}
defer ctx.Cancel()
data := dataext.ObjectMerge(datb, datq)
if data.FCMToken == nil {
return ginresp.CompatAPIError(0, "Missing parameter [[fcm_token]]")
}
if data.Pro == nil {
return ginresp.CompatAPIError(0, "Missing parameter [[pro]]")
}
if data.ProToken == nil {
return ginresp.CompatAPIError(0, "Missing parameter [[pro_token]]")
}
2023-01-17 22:03:27 +01:00
if data.ProToken != nil {
data.ProToken = langext.Ptr("ANDROID|v1|" + *data.ProToken)
}
2022-11-20 01:28:32 +01:00
if *data.Pro != "true" {
data.ProToken = nil
}
if data.ProToken != nil {
2023-01-17 22:03:27 +01:00
ptok, err := h.app.VerifyProToken(ctx, *data.ProToken)
2022-11-20 01:28:32 +01:00
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query purchase status")
}
if !ptok {
return ginresp.CompatAPIError(0, "Purchase token could not be verified")
}
}
adminKey := h.app.GenerateRandomAuthKey()
err := h.database.ClearFCMTokens(ctx, *data.FCMToken)
if err != nil {
return ginresp.CompatAPIError(0, "Failed to clear existing fcm tokens")
}
if data.ProToken != nil {
err := h.database.ClearProTokens(ctx, *data.ProToken)
if err != nil {
2023-01-17 22:03:27 +01:00
return ginresp.CompatAPIError(0, "Failed to clear existing pro tokens")
2022-11-20 01:28:32 +01:00
}
}
2023-04-21 21:45:16 +02:00
user, err := h.database.CreateUser(ctx, data.ProToken, nil)
2022-11-20 01:28:32 +01:00
if err != nil {
return ginresp.CompatAPIError(0, "Failed to create user in db")
}
2023-04-21 21:45:16 +02:00
_, err = h.database.CreateKeyToken(ctx, "CompatKey", user.UserID, true, make([]models.ChannelID, 0), models.TokenPermissionList{models.PermAdmin}, adminKey)
if err != nil {
return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create admin-key in db", err)
}
2023-01-17 22:03:27 +01:00
_, err = h.database.CreateClient(ctx, user.UserID, models.ClientTypeAndroid, *data.FCMToken, "compat", "compat")
2022-11-20 01:28:32 +01:00
if err != nil {
2022-11-24 12:53:27 +01:00
return ginresp.CompatAPIError(0, "Failed to create client in db")
2022-11-20 01:28:32 +01:00
}
oldid, err := h.database.CreateCompatID(ctx, "userid", user.UserID.String())
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to create userid<old>", err)
}
2022-11-20 01:28:32 +01:00
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
Message: "New user registered",
UserID: oldid,
2023-04-21 21:45:16 +02:00
UserKey: adminKey,
2022-11-20 01:28:32 +01:00
QuotaUsed: user.QuotaUsedToday(),
QuotaMax: user.QuotaPerDay(),
2023-01-17 22:32:13 +01:00
IsPro: user.IsPro,
2022-11-20 01:28:32 +01:00
}))
2022-11-13 19:17:07 +01:00
}
// Info swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Summary Get information about the current user
// @ID compat-info
// @Tags API-v1
2022-11-23 19:32:23 +01:00
//
2023-04-21 21:45:16 +02:00
// @Deprecated
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id query string true "the user_id"
// @Param user_key query string true "the user_key"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id formData string true "the user_id"
// @Param user_key formData string true "the user_key"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.Info.response
// @Failure default {object} ginresp.compatAPIError
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Router /api/info.php [get]
2022-11-13 19:17:07 +01:00
func (h CompatHandler) Info(g *gin.Context) ginresp.HTTPResponse {
type query struct {
UserID *int64 `json:"user_id" form:"user_id"`
UserKey *string `json:"user_key" form:"user_key"`
2022-11-13 19:17:07 +01:00
}
type response struct {
2022-11-20 01:28:32 +01:00
Success bool `json:"success"`
2022-11-13 19:17:07 +01:00
Message string `json:"message"`
2022-11-20 01:28:32 +01:00
UserID int64 `json:"user_id"`
2022-11-13 19:17:07 +01:00
UserKey string `json:"user_key"`
2022-11-20 01:28:32 +01:00
QuotaUsed int `json:"quota"`
QuotaMax int `json:"quota_max"`
IsPro int `json:"is_pro"`
2022-11-13 19:17:07 +01:00
FCMSet bool `json:"fcm_token_set"`
UnackCount int `json:"unack_count"`
}
2022-11-20 01:28:32 +01:00
var datq query
var datb query
ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true})
2022-11-20 01:28:32 +01:00
if errResp != nil {
return *errResp
}
defer ctx.Cancel()
data := dataext.ObjectMerge(datb, datq)
if data.UserID == nil {
return ginresp.CompatAPIError(101, "Missing parameter [[user_id]]")
}
if data.UserKey == nil {
return ginresp.CompatAPIError(102, "Missing parameter [[user_key]]")
}
useridCompNew, err := h.database.ConvertCompatID(ctx, *data.UserID, "userid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query userid<old>", err)
}
if useridCompNew == nil {
return ginresp.SendAPIError(g, 400, apierr.USER_NOT_FOUND, hl.USER_ID, "User not found (compat)", nil)
}
user, err := h.database.GetUser(ctx, models.UserID(*useridCompNew))
2022-11-20 01:28:32 +01:00
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(201, "User not found")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2022-11-13 19:17:07 +01:00
2023-04-21 21:45:16 +02:00
keytok, err := h.database.GetKeyTokenByToken(ctx, *data.UserKey)
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(204, "Authentification failed")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query token")
}
if !keytok.IsAdmin(user.UserID) {
2022-11-20 01:28:32 +01:00
return ginresp.CompatAPIError(204, "Authentification failed")
}
clients, err := h.database.ListClients(ctx, user.UserID)
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query clients")
}
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
Message: "ok",
UserID: *data.UserID,
2023-04-21 21:45:16 +02:00
UserKey: keytok.Token,
2022-11-20 01:28:32 +01:00
QuotaUsed: user.QuotaUsedToday(),
QuotaMax: user.QuotaPerDay(),
IsPro: langext.Conditional(user.IsPro, 1, 0),
2023-05-28 23:25:18 +02:00
FCMSet: len(clients) > 0,
2022-11-20 01:28:32 +01:00
UnackCount: 0,
}))
2022-11-13 19:17:07 +01:00
}
// Ack swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Summary Acknowledge that a message was received
// @ID compat-ack
// @Tags API-v1
2022-11-23 19:32:23 +01:00
//
2023-04-21 21:45:16 +02:00
// @Deprecated
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id query string true "the user_id"
// @Param user_key query string true "the user_key"
// @Param scn_msg_id query string true "the message id"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id formData string true "the user_id"
// @Param user_key formData string true "the user_key"
// @Param scn_msg_id formData string true "the message id"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.Ack.response
// @Failure default {object} ginresp.compatAPIError
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Router /api/ack.php [get]
2022-11-13 19:17:07 +01:00
func (h CompatHandler) Ack(g *gin.Context) ginresp.HTTPResponse {
type query struct {
UserID *int64 `json:"user_id" form:"user_id"`
UserKey *string `json:"user_key" form:"user_key"`
MessageID *int64 `json:"scn_msg_id" form:"scn_msg_id"`
2022-11-13 19:17:07 +01:00
}
type response struct {
2022-11-20 01:28:32 +01:00
Success bool `json:"success"`
2022-11-13 19:17:07 +01:00
Message string `json:"message"`
PrevAckValue int `json:"prev_ack"`
NewAckValue int `json:"new_ack"`
}
2022-11-20 01:28:32 +01:00
var datq query
var datb query
ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true})
2022-11-20 01:28:32 +01:00
if errResp != nil {
return *errResp
}
defer ctx.Cancel()
data := dataext.ObjectMerge(datb, datq)
if data.UserID == nil {
return ginresp.CompatAPIError(101, "Missing parameter [[user_id]]")
}
if data.UserKey == nil {
return ginresp.CompatAPIError(102, "Missing parameter [[user_key]]")
}
if data.MessageID == nil {
return ginresp.CompatAPIError(103, "Missing parameter [[scn_msg_id]]")
}
useridCompNew, err := h.database.ConvertCompatID(ctx, *data.UserID, "userid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query userid<old>", err)
}
if useridCompNew == nil {
return ginresp.SendAPIError(g, 400, apierr.USER_NOT_FOUND, hl.USER_ID, "User not found (compat)", nil)
}
user, err := h.database.GetUser(ctx, models.UserID(*useridCompNew))
2022-11-20 01:28:32 +01:00
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(201, "User not found")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2023-04-21 21:45:16 +02:00
keytok, err := h.database.GetKeyTokenByToken(ctx, *data.UserKey)
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(204, "Authentification failed")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query token")
}
if !keytok.IsAdmin(user.UserID) {
2022-11-20 01:28:32 +01:00
return ginresp.CompatAPIError(204, "Authentification failed")
}
2022-11-13 19:17:07 +01:00
messageIdComp, err := h.database.ConvertCompatID(ctx, *data.MessageID, "messageid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query messageid<old>", err)
}
if useridCompNew == nil {
return ginresp.SendAPIError(g, 400, apierr.MESSAGE_NOT_FOUND, hl.USER_ID, "Message not found (compat)", nil)
}
ackBefore, err := h.database.GetAck(ctx, models.MessageID(*messageIdComp))
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query ack", err)
}
if !ackBefore {
err = h.database.SetAck(ctx, user.UserID, models.MessageID(*messageIdComp))
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to set ack", err)
}
}
2022-11-20 01:28:32 +01:00
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
Message: "ok",
PrevAckValue: langext.Conditional(ackBefore, 1, 0),
2022-11-20 01:28:32 +01:00
NewAckValue: 1,
}))
2022-11-13 19:17:07 +01:00
}
// Requery swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Summary Return all not-acknowledged messages
// @ID compat-requery
// @Tags API-v1
2022-11-23 19:32:23 +01:00
//
2023-04-21 21:45:16 +02:00
// @Deprecated
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id query string true "the user_id"
// @Param user_key query string true "the user_key"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id formData string true "the user_id"
// @Param user_key formData string true "the user_key"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.Requery.response
// @Failure default {object} ginresp.compatAPIError
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Router /api/requery.php [get]
2022-11-13 19:17:07 +01:00
func (h CompatHandler) Requery(g *gin.Context) ginresp.HTTPResponse {
type query struct {
UserID *int64 `json:"user_id" form:"user_id"`
UserKey *string `json:"user_key" form:"user_key"`
2022-11-13 19:17:07 +01:00
}
type response struct {
2022-11-20 01:28:32 +01:00
Success bool `json:"success"`
2022-11-13 19:17:07 +01:00
Message string `json:"message"`
Count int `json:"count"`
Data []models.CompatMessage `json:"data"`
}
2022-11-20 01:28:32 +01:00
var datq query
var datb query
ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true})
2022-11-20 01:28:32 +01:00
if errResp != nil {
return *errResp
}
defer ctx.Cancel()
2022-11-13 19:17:07 +01:00
2022-11-20 01:28:32 +01:00
data := dataext.ObjectMerge(datb, datq)
if data.UserID == nil {
return ginresp.CompatAPIError(101, "Missing parameter [[user_id]]")
}
if data.UserKey == nil {
return ginresp.CompatAPIError(102, "Missing parameter [[user_key]]")
}
useridCompNew, err := h.database.ConvertCompatID(ctx, *data.UserID, "userid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query userid<old>", err)
}
if useridCompNew == nil {
return ginresp.SendAPIError(g, 400, apierr.USER_NOT_FOUND, hl.USER_ID, "User not found (compat)", nil)
}
user, err := h.database.GetUser(ctx, models.UserID(*useridCompNew))
2022-11-20 01:28:32 +01:00
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(201, "User not found")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2023-04-21 21:45:16 +02:00
keytok, err := h.database.GetKeyTokenByToken(ctx, *data.UserKey)
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(204, "Authentification failed")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query token")
}
if !keytok.IsAdmin(user.UserID) {
2022-11-20 01:28:32 +01:00
return ginresp.CompatAPIError(204, "Authentification failed")
}
filter := models.MessageFilter{
Owner: langext.Ptr([]models.UserID{user.UserID}),
CompatAcknowledged: langext.Ptr(false),
}
2023-05-27 23:54:14 +02:00
msgs, _, err := h.database.ListMessages(ctx, filter, langext.Ptr(16), ct.Start())
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
compMsgs := make([]models.CompatMessage, 0, len(msgs))
for _, v := range msgs {
messageIdComp, err := h.database.ConvertToCompatIDOrCreate(ctx, v.MessageID.String(), "messageid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query/create messageid<old>", err)
}
compMsgs = append(compMsgs, models.CompatMessage{
Title: compatizeMessageTitle(ctx, h.app, v),
Body: v.Content,
Priority: v.Priority,
Timestamp: v.Timestamp().Unix(),
UserMessageID: v.UserMessageID,
SCNMessageID: messageIdComp,
Trimmed: nil,
})
}
2022-11-20 01:28:32 +01:00
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
Message: "ok",
Count: len(compMsgs),
Data: compMsgs,
2022-11-20 01:28:32 +01:00
}))
2022-11-13 19:17:07 +01:00
}
// Update swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Summary Set the fcm-token (android)
// @ID compat-update
// @Tags API-v1
2022-11-23 19:32:23 +01:00
//
2023-04-21 21:45:16 +02:00
// @Deprecated
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id query string true "the user_id"
// @Param user_key query string true "the user_key"
// @Param fcm_token query string true "the (android) fcm token"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id formData string true "the user_id"
// @Param user_key formData string true "the user_key"
// @Param fcm_token formData string true "the (android) fcm token"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.Update.response
// @Failure default {object} ginresp.compatAPIError
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Router /api/update.php [get]
2022-11-13 19:17:07 +01:00
func (h CompatHandler) Update(g *gin.Context) ginresp.HTTPResponse {
type query struct {
UserID *int64 `json:"user_id" form:"user_id"`
UserKey *string `json:"user_key" form:"user_key"`
FCMToken *string `json:"fcm_token" form:"fcm_token"`
2022-11-13 19:17:07 +01:00
}
type response struct {
2022-11-20 01:28:32 +01:00
Success bool `json:"success"`
2022-11-13 19:17:07 +01:00
Message string `json:"message"`
2022-11-20 01:28:32 +01:00
UserID int64 `json:"user_id"`
2022-11-13 19:17:07 +01:00
UserKey string `json:"user_key"`
2022-11-20 01:28:32 +01:00
QuotaUsed int `json:"quota"`
QuotaMax int `json:"quota_max"`
IsPro int `json:"is_pro"`
}
var datq query
var datb query
ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true})
2022-11-20 01:28:32 +01:00
if errResp != nil {
return *errResp
}
defer ctx.Cancel()
data := dataext.ObjectMerge(datb, datq)
if data.UserID == nil {
return ginresp.CompatAPIError(101, "Missing parameter [[user_id]]")
}
if data.UserKey == nil {
return ginresp.CompatAPIError(102, "Missing parameter [[user_key]]")
}
useridCompNew, err := h.database.ConvertCompatID(ctx, *data.UserID, "userid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query userid<old>", err)
}
if useridCompNew == nil {
return ginresp.SendAPIError(g, 400, apierr.USER_NOT_FOUND, hl.USER_ID, "User not found (compat)", nil)
}
user, err := h.database.GetUser(ctx, models.UserID(*useridCompNew))
2022-11-20 01:28:32 +01:00
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(201, "User not found")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2023-04-21 21:45:16 +02:00
keytok, err := h.database.GetKeyTokenByToken(ctx, *data.UserKey)
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(204, "Authentification failed")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query token")
}
if !keytok.IsAdmin(user.UserID) {
2022-11-20 01:28:32 +01:00
return ginresp.CompatAPIError(204, "Authentification failed")
}
clients, err := h.database.ListClients(ctx, user.UserID)
if err != nil {
return ginresp.CompatAPIError(0, "Failed to list clients")
}
newAdminKey := h.app.GenerateRandomAuthKey()
2023-04-21 21:45:16 +02:00
_, err = h.database.CreateKeyToken(ctx, "CompatKey", user.UserID, true, make([]models.ChannelID, 0), models.TokenPermissionList{models.PermAdmin}, newAdminKey)
if err != nil {
return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create admin-key in db", err)
}
err = h.database.DeleteKeyToken(ctx, keytok.KeyTokenID)
2022-11-20 01:28:32 +01:00
if err != nil {
return ginresp.CompatAPIError(0, "Failed to update keys")
}
if data.FCMToken != nil {
for _, client := range clients {
err = h.database.DeleteClient(ctx, client.ClientID)
if err != nil {
return ginresp.CompatAPIError(0, "Failed to delete client")
}
}
2023-01-17 22:03:27 +01:00
_, err = h.database.CreateClient(ctx, user.UserID, models.ClientTypeAndroid, *data.FCMToken, "compat", "compat")
2022-11-20 01:28:32 +01:00
if err != nil {
return ginresp.CompatAPIError(0, "Failed to delete client")
}
2022-11-13 19:17:07 +01:00
}
2022-11-20 01:28:32 +01:00
user, err = h.database.GetUser(ctx, user.UserID)
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2022-11-13 19:17:07 +01:00
2022-11-20 01:28:32 +01:00
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
Message: "user updated",
UserID: *data.UserID,
2023-04-21 21:45:16 +02:00
UserKey: newAdminKey,
2022-11-20 01:28:32 +01:00
QuotaUsed: user.QuotaUsedToday(),
QuotaMax: user.QuotaPerDay(),
IsPro: langext.Conditional(user.IsPro, 1, 0),
}))
2022-11-13 19:17:07 +01:00
}
// Expand swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Summary Get a whole (potentially truncated) message
// @ID compat-expand
// @Tags API-v1
2022-11-23 19:32:23 +01:00
//
2023-04-21 21:45:16 +02:00
// @Deprecated
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id query string true "The user_id"
// @Param user_key query string true "The user_key"
// @Param scn_msg_id query string true "The message-id"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id formData string true "The user_id"
// @Param user_key formData string true "The user_key"
// @Param scn_msg_id formData string true "The message-id"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.Expand.response
// @Failure default {object} ginresp.compatAPIError
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Router /api/expand.php [get]
2022-11-13 19:17:07 +01:00
func (h CompatHandler) Expand(g *gin.Context) ginresp.HTTPResponse {
type query struct {
UserID *int64 `json:"user_id" form:"user_id"`
UserKey *string `json:"user_key" form:"user_key"`
MessageID *int64 `json:"scn_msg_id" form:"scn_msg_id"`
2022-11-13 19:17:07 +01:00
}
type response struct {
2022-11-20 01:28:32 +01:00
Success bool `json:"success"`
Message string `json:"message"`
Data models.CompatMessage `json:"data"`
}
var datq query
var datb query
ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true})
2022-11-20 01:28:32 +01:00
if errResp != nil {
return *errResp
}
defer ctx.Cancel()
data := dataext.ObjectMerge(datb, datq)
if data.UserID == nil {
return ginresp.CompatAPIError(101, "Missing parameter [[user_id]]")
}
if data.UserKey == nil {
return ginresp.CompatAPIError(102, "Missing parameter [[user_key]]")
}
if data.MessageID == nil {
return ginresp.CompatAPIError(103, "Missing parameter [[scn_msg_id]]")
2022-11-13 19:17:07 +01:00
}
useridCompNew, err := h.database.ConvertCompatID(ctx, *data.UserID, "userid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query userid<old>", err)
}
if useridCompNew == nil {
return ginresp.SendAPIError(g, 400, apierr.USER_NOT_FOUND, hl.USER_ID, "User not found (compat)", nil)
}
user, err := h.database.GetUser(ctx, models.UserID(*useridCompNew))
2022-11-20 01:28:32 +01:00
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(201, "User not found")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2022-11-13 19:17:07 +01:00
2023-04-21 21:45:16 +02:00
keytok, err := h.database.GetKeyTokenByToken(ctx, *data.UserKey)
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(204, "Authentification failed")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query token")
}
if !keytok.IsAdmin(user.UserID) {
2022-11-20 01:28:32 +01:00
return ginresp.CompatAPIError(204, "Authentification failed")
}
messageCompNew, err := h.database.ConvertCompatID(ctx, *data.MessageID, "messageid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query messagid<old>", err)
}
if messageCompNew == nil {
return ginresp.CompatAPIError(301, "Message not found")
}
msg, err := h.database.GetMessage(ctx, models.MessageID(*messageCompNew), false)
2022-11-20 01:28:32 +01:00
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(301, "Message not found")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query message")
}
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
Message: "ok",
Data: models.CompatMessage{
Title: compatizeMessageTitle(ctx, h.app, msg),
2023-01-17 21:30:53 +01:00
Body: msg.Content,
2022-11-20 01:28:32 +01:00
Trimmed: langext.Ptr(false),
Priority: msg.Priority,
Timestamp: msg.Timestamp().Unix(),
UserMessageID: msg.UserMessageID,
SCNMessageID: *data.MessageID,
2022-11-20 01:28:32 +01:00
},
}))
2022-11-13 19:17:07 +01:00
}
// Upgrade swaggerdoc
//
2023-04-21 21:45:16 +02:00
// @Summary Upgrade a free account to a paid account
// @ID compat-upgrade
// @Tags API-v1
2022-11-23 19:32:23 +01:00
//
2023-04-21 21:45:16 +02:00
// @Deprecated
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id query string true "the user_id"
// @Param user_key query string true "the user_key"
// @Param pro query string true "if the user is a paid account" Enums(true, false)
// @Param pro_token query string true "the (android) IAP token"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Param user_id formData string true "the user_id"
// @Param user_key formData string true "the user_key"
// @Param pro formData string true "if the user is a paid account" Enums(true, false)
// @Param pro_token formData string true "the (android) IAP token"
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Success 200 {object} handler.Upgrade.response
// @Failure default {object} ginresp.compatAPIError
2022-11-20 13:18:09 +01:00
//
2023-04-21 21:45:16 +02:00
// @Router /api/upgrade.php [get]
2022-11-13 19:17:07 +01:00
func (h CompatHandler) Upgrade(g *gin.Context) ginresp.HTTPResponse {
type query struct {
UserID *int64 `json:"user_id" form:"user_id"`
UserKey *string `json:"user_key" form:"user_key"`
Pro *string `json:"pro" form:"pro"`
ProToken *string `json:"pro_token" form:"pro_token"`
2022-11-13 19:17:07 +01:00
}
type response struct {
2022-11-20 01:28:32 +01:00
Success bool `json:"success"`
Message string `json:"message"`
UserID int64 `json:"user_id"`
QuotaUsed int `json:"quota"`
QuotaMax int `json:"quota_max"`
IsPro bool `json:"is_pro"`
}
var datq query
var datb query
ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true})
2022-11-20 01:28:32 +01:00
if errResp != nil {
return *errResp
2022-11-13 19:17:07 +01:00
}
2022-11-20 01:28:32 +01:00
defer ctx.Cancel()
data := dataext.ObjectMerge(datb, datq)
2022-11-13 19:17:07 +01:00
2022-11-20 01:28:32 +01:00
if data.UserID == nil {
return ginresp.CompatAPIError(101, "Missing parameter [[user_id]]")
}
if data.UserKey == nil {
return ginresp.CompatAPIError(102, "Missing parameter [[user_key]]")
}
if data.Pro == nil {
return ginresp.CompatAPIError(103, "Missing parameter [[pro]]")
}
if data.ProToken == nil {
return ginresp.CompatAPIError(104, "Missing parameter [[pro_token]]")
}
useridCompNew, err := h.database.ConvertCompatID(ctx, *data.UserID, "userid")
if err != nil {
return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to query userid<old>", err)
}
if useridCompNew == nil {
return ginresp.SendAPIError(g, 400, apierr.USER_NOT_FOUND, hl.USER_ID, "User not found (compat)", nil)
}
user, err := h.database.GetUser(ctx, models.UserID(*useridCompNew))
2022-11-20 01:28:32 +01:00
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(201, "User not found")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2023-04-21 21:45:16 +02:00
keytok, err := h.database.GetKeyTokenByToken(ctx, *data.UserKey)
if err == sql.ErrNoRows {
return ginresp.CompatAPIError(204, "Authentification failed")
}
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query token")
}
if !keytok.IsAdmin(user.UserID) {
2022-11-20 01:28:32 +01:00
return ginresp.CompatAPIError(204, "Authentification failed")
}
if data.ProToken != nil {
data.ProToken = langext.Ptr("ANDROID|v1|" + *data.ProToken)
}
2022-11-20 01:28:32 +01:00
if *data.Pro != "true" {
data.ProToken = nil
}
if data.ProToken != nil {
ptok, err := h.app.VerifyProToken(ctx, *data.ProToken)
2022-11-20 01:28:32 +01:00
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query purchase status")
}
if !ptok {
return ginresp.CompatAPIError(0, "Purchase token could not be verified")
}
err = h.database.ClearProTokens(ctx, *data.ProToken)
if err != nil {
return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to clear existing fcm tokens", err)
}
err = h.database.UpdateUserProToken(ctx, user.UserID, langext.Ptr(*data.ProToken))
2022-11-20 01:28:32 +01:00
if err != nil {
return ginresp.CompatAPIError(0, "Failed to update user")
}
} else {
err = h.database.UpdateUserProToken(ctx, user.UserID, nil)
if err != nil {
return ginresp.CompatAPIError(0, "Failed to update user")
}
}
user, err = h.database.GetUser(ctx, user.UserID)
if err != nil {
return ginresp.CompatAPIError(0, "Failed to query user")
}
2022-11-13 19:17:07 +01:00
2022-11-20 01:28:32 +01:00
return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{
Success: true,
Message: "user updated",
UserID: *data.UserID,
2022-11-20 01:28:32 +01:00
QuotaUsed: user.QuotaUsedToday(),
QuotaMax: user.QuotaPerDay(),
IsPro: user.IsPro,
}))
2022-11-13 19:17:07 +01:00
}
func compatizeMessageTitle(ctx *logic.AppContext, app *logic.Application, msg models.Message) string {
if msg.ChannelInternalName == "main" {
return msg.Title
}
channel, err := app.Database.Primary.GetChannelByID(ctx, msg.ChannelID)
if err != nil {
return fmt.Sprintf("[%s] %s", "%SCN-ERR%", msg.Title)
}
return fmt.Sprintf("[%s] %s", channel.DisplayName, msg.Title)
}