WIP, define new endpoints for send reaction

This commit is contained in:
René Filip 2021-11-01 18:09:21 +01:00
parent 2d80ff91a2
commit 55d63c90b6
2 changed files with 146 additions and 45 deletions

View File

@ -9,8 +9,8 @@ import (
"github.com/gabriel-vasile/mimetype" "github.com/gabriel-vasile/mimetype"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
log "github.com/sirupsen/logrus"
"github.com/gorilla/websocket" "github.com/gorilla/websocket"
log "github.com/sirupsen/logrus"
"github.com/bbernhard/signal-cli-rest-api/client" "github.com/bbernhard/signal-cli-rest-api/client"
utils "github.com/bbernhard/signal-cli-rest-api/utils" utils "github.com/bbernhard/signal-cli-rest-api/utils"
@ -29,23 +29,23 @@ const (
type GroupPermissions struct { type GroupPermissions struct {
AddMembers string `json:"add_members" enums:"only-admins,every-member"` AddMembers string `json:"add_members" enums:"only-admins,every-member"`
EditGroup string `json:"edit_group" enums:"only-admins,every-member"` EditGroup string `json:"edit_group" enums:"only-admins,every-member"`
} }
type CreateGroupRequest struct { type CreateGroupRequest struct {
Name string `json:"name"` Name string `json:"name"`
Members []string `json:"members"` Members []string `json:"members"`
Description string `json:"description"` Description string `json:"description"`
Permissions GroupPermissions `json:"permissions"` Permissions GroupPermissions `json:"permissions"`
GroupLinkState string `json:"group_link" enums:"disabled,enabled,enabled-with-approval"` GroupLinkState string `json:"group_link" enums:"disabled,enabled,enabled-with-approval"`
} }
type LoggingConfiguration struct { type LoggingConfiguration struct {
Level string `json:"Level"` Level string `json:"Level"`
} }
type Configuration struct { type Configuration struct {
Logging LoggingConfiguration `json:"logging"` Logging LoggingConfiguration `json:"logging"`
} }
type RegisterNumberRequest struct { type RegisterNumberRequest struct {
@ -57,6 +57,13 @@ type VerifyNumberSettings struct {
Pin string `json:"pin"` Pin string `json:"pin"`
} }
type Reaction struct {
Number string `json:"number"`
Timestamp int64 `json:"timestamp"`
Recipient string `json:"recipient"`
Reaction string `json:"reaction"`
}
type SendMessageV1 struct { type SendMessageV1 struct {
Number string `json:"number"` Number string `json:"number"`
Recipients []string `json:"recipients"` Recipients []string `json:"recipients"`
@ -80,8 +87,6 @@ type Error struct {
Msg string `json:"error"` Msg string `json:"error"`
} }
type CreateGroupResponse struct { type CreateGroupResponse struct {
Id string `json:"id"` Id string `json:"id"`
} }
@ -106,12 +111,12 @@ var connectionUpgrader = websocket.Upgrader{
} }
type Api struct { type Api struct {
signalClient *client.SignalClient signalClient *client.SignalClient
} }
func NewApi(signalClient *client.SignalClient) *Api { func NewApi(signalClient *client.SignalClient) *Api {
return &Api{ return &Api{
signalClient: signalClient, signalClient: signalClient,
} }
} }
@ -283,7 +288,6 @@ func (a *Api) SendV2(c *gin.Context) {
c.JSON(201, SendMessageResponse{Timestamp: strconv.FormatInt((*timestamps)[0].Timestamp, 10)}) c.JSON(201, SendMessageResponse{Timestamp: strconv.FormatInt((*timestamps)[0].Timestamp, 10)})
} }
func (a *Api) handleSignalReceive(ws *websocket.Conn, number string) { func (a *Api) handleSignalReceive(ws *websocket.Conn, number string) {
for { for {
data, err := a.signalClient.Receive(number, 0) data, err := a.signalClient.Receive(number, 0)
@ -404,7 +408,7 @@ func (a *Api) CreateGroup(c *gin.Context) {
} }
if req.GroupLinkState != "" && !utils.StringInSlice(req.GroupLinkState, []string{"enabled", "enabled-with-approval", "disabled"}) { if req.GroupLinkState != "" && !utils.StringInSlice(req.GroupLinkState, []string{"enabled", "enabled-with-approval", "disabled"}) {
c.JSON(400, Error{Msg: "Invalid group link provided - only 'enabled', 'enabled-with-approval' and 'disabled' allowed!" }) c.JSON(400, Error{Msg: "Invalid group link provided - only 'enabled', 'enabled-with-approval' and 'disabled' allowed!"})
return return
} }
@ -557,18 +561,18 @@ func (a *Api) RemoveAttachment(c *gin.Context) {
err := a.signalClient.RemoveAttachment(attachment) err := a.signalClient.RemoveAttachment(attachment)
if err != nil { if err != nil {
switch err.(type) { switch err.(type) {
case *client.InvalidNameError: case *client.InvalidNameError:
c.JSON(400, Error{Msg: err.Error()}) c.JSON(400, Error{Msg: err.Error()})
return return
case *client.NotFoundError: case *client.NotFoundError:
c.JSON(404, Error{Msg: err.Error()}) c.JSON(404, Error{Msg: err.Error()})
return return
case *client.InternalError: case *client.InternalError:
c.JSON(500, Error{Msg: err.Error()}) c.JSON(500, Error{Msg: err.Error()})
return return
default: default:
c.JSON(500, Error{Msg: err.Error()}) c.JSON(500, Error{Msg: err.Error()})
return return
} }
} }
@ -589,18 +593,18 @@ func (a *Api) ServeAttachment(c *gin.Context) {
attachmentBytes, err := a.signalClient.GetAttachment(attachment) attachmentBytes, err := a.signalClient.GetAttachment(attachment)
if err != nil { if err != nil {
switch err.(type) { switch err.(type) {
case *client.InvalidNameError: case *client.InvalidNameError:
c.JSON(400, Error{Msg: err.Error()}) c.JSON(400, Error{Msg: err.Error()})
return return
case *client.NotFoundError: case *client.NotFoundError:
c.JSON(404, Error{Msg: err.Error()}) c.JSON(404, Error{Msg: err.Error()})
return return
case *client.InternalError: case *client.InternalError:
c.JSON(500, Error{Msg: err.Error()}) c.JSON(500, Error{Msg: err.Error()})
return return
default: default:
c.JSON(500, Error{Msg: err.Error()}) c.JSON(500, Error{Msg: err.Error()})
return return
} }
} }
@ -890,6 +894,52 @@ func (a *Api) QuitGroup(c *gin.Context) {
c.Status(http.StatusNoContent) c.Status(http.StatusNoContent)
} }
// @Summary Send a reaction.
// @Tags Reactions
// @Description React to a message.
// @Accept json
// @Produce json
// @Success 201 {string} OK
// @Failure 400 {object} Error
// @Param data body Reaction true "Reaction"
// @Router /v1/react/{number} [post]
func (a *Api) SendReaction(c *gin.Context) {
var req Reaction
err := c.BindJSON(&req)
if err != nil {
c.JSON(400, Error{Msg: "Couldn't process request - invalid request"})
log.Error(err.Error())
return
}
if req.Recipient == "" {
c.JSON(400, Error{Msg: "Couldn't process request - recipient missing"})
return
}
if req.Timestamp == 0 {
c.JSON(400, Error{Msg: "Couldn't process request - timestamp missing"})
return
}
if req.Reaction == "" {
c.JSON(400, Error{Msg: "Couldn't process request - reaction missing"})
return
}
if req.Number == "" {
c.JSON(400, Error{Msg: "Couldn't process request - number missing"})
return
}
err = a.signalClient.SendReaction(req.Number, req.Recipient, req.Timestamp, req.Reaction)
if err != nil {
c.JSON(400, Error{Msg: err.Error()})
return
}
c.Status(http.StatusNoContent)
}
// @Summary Show Typing Indicator. // @Summary Show Typing Indicator.
// @Tags Messages // @Tags Messages
// @Description Show Typing Indicator. // @Description Show Typing Indicator.

View File

@ -14,9 +14,10 @@ import (
"strings" "strings"
"time" "time"
"github.com/cyphar/filepath-securejoin" securejoin "github.com/cyphar/filepath-securejoin"
"github.com/gabriel-vasile/mimetype" "github.com/gabriel-vasile/mimetype"
"github.com/h2non/filetype" "github.com/h2non/filetype"
//"github.com/sourcegraph/jsonrpc2"//"net/rpc/jsonrpc" //"github.com/sourcegraph/jsonrpc2"//"net/rpc/jsonrpc"
log "github.com/sirupsen/logrus" log "github.com/sirupsen/logrus"
@ -613,8 +614,8 @@ func (s *SignalClient) CreateGroup(number string, name string, members []string,
} }
type Response struct { type Response struct {
GroupId string `json:"groupId"` GroupId string `json:"groupId"`
Timestamp int64 `json:"timestamp"` Timestamp int64 `json:"timestamp"`
} }
var resp Response var resp Response
json.Unmarshal([]byte(rawData), &resp) json.Unmarshal([]byte(rawData), &resp)
@ -995,6 +996,56 @@ func (s *SignalClient) QuitGroup(number string, groupId string) error {
return err return err
} }
func (s *SignalClient) SendReaction(number string, recipient string, timestamp int64, reaction string) error {
var err error
recp := recipient
isGroup := false
if strings.HasPrefix(recipient, groupPrefix) {
isGroup = true
recp, err = ConvertGroupIdToInternalGroupId(recipient)
if err != nil {
return errors.New("Invalid group id")
}
}
if s.signalCliMode == JsonRpc {
type Request struct {
Recipient string `json:"recipient,omitempty"`
GroupId string `json:"group-id,omitempty"`
// TODO other fields
}
request := Request{}
if !isGroup {
request.Recipient = recp
} else {
request.GroupId = recp
}
jsonRpc2Client, err := s.getJsonRpc2Client(number)
if err != nil {
return err
}
_, err = jsonRpc2Client.getRaw("sendReaction", request)
} else {
// TODO: check CLI command again
cmd := []string{
"--config", s.signalCliConfig,
"-u", number,
"sendTyping",
"-e", reaction,
"-t", strconv.FormatInt(timestamp, 10),
}
if !isGroup {
cmd = append(cmd, recp)
} else {
cmd = append(cmd, []string{"-g", recp}...)
}
_, err = runSignalCli(true, cmd, "", s.signalCliMode)
}
return err
}
func (s *SignalClient) SendStartTyping(number string, recipient string) error { func (s *SignalClient) SendStartTyping(number string, recipient string) error {
var err error var err error
recp := recipient recp := recipient
@ -1010,7 +1061,7 @@ func (s *SignalClient) SendStartTyping(number string, recipient string) error {
if s.signalCliMode == JsonRpc { if s.signalCliMode == JsonRpc {
type Request struct { type Request struct {
Recipient string `json:"recipient,omitempty"` Recipient string `json:"recipient,omitempty"`
GroupId string `json:"group-id,omitempty"` GroupId string `json:"group-id,omitempty"`
} }
request := Request{} request := Request{}
if !isGroup { if !isGroup {
@ -1052,8 +1103,8 @@ func (s *SignalClient) SendStopTyping(number string, recipient string) error {
if s.signalCliMode == JsonRpc { if s.signalCliMode == JsonRpc {
type Request struct { type Request struct {
Recipient string `json:"recipient,omitempty"` Recipient string `json:"recipient,omitempty"`
GroupId string `json:"group-id,omitempty"` GroupId string `json:"group-id,omitempty"`
Stop bool `json:"stop"` Stop bool `json:"stop"`
} }
request := Request{Stop: true} request := Request{Stop: true}
if !isGroup { if !isGroup {