This repository has been archived on 2020-09-27. You can view files and clone it, but cannot push or open issues or pull requests.
thrempp/component/threema/send.go

195 lines
4.4 KiB
Go
Raw Normal View History

package threema
import (
2019-09-06 04:34:13 +02:00
"encoding/base32"
2019-10-03 23:44:42 +02:00
"encoding/xml"
"strconv"
2019-09-06 04:34:13 +02:00
"strings"
"github.com/bdlm/log"
"github.com/o3ma/o3"
2019-10-03 23:44:42 +02:00
"gosrc.io/xmpp"
2019-06-28 03:03:38 +02:00
"gosrc.io/xmpp/stanza"
)
2019-10-03 23:44:42 +02:00
type PresMUCUserItem struct {
XMLName xml.Name `xml:"item"`
Affiliation string `xml:"affiliation,attr"`
Role string `xml:"role,attr"`
}
type PresMUCUserStatus struct {
XMLName xml.Name `xml:"status"`
Code int `xml:"code,attr"`
}
type PresMUCUserList struct {
XMLName xml.Name `xml:"http://jabber.org/protocol/muc#user x"`
Items []PresMUCUserItem
Status *PresMUCUserStatus
}
func (a *Account) handlePresence(p stanza.Presence) error {
logger := log.WithFields(map[string]interface{}{
"from": p.Attrs.From,
"to": p.Attrs.To,
})
_, header := jidToThreemaGroup(p.To)
if header == nil {
logger.Debug("no group presence")
return nil
}
from, _ := xmpp.NewJid(p.From)
to, _ := xmpp.NewJid(p.To)
if a.XMPPResource[to.Node] == nil {
a.XMPPResource[to.Node] = make(map[string]bool)
}
a.XMPPResource[to.Node][from.Resource] = true
ownsender := strings.ToLower(a.ThreemaID.ID.String())
//TODO list current users
senders := []string{header.CreatorID.String(), ownsender}
for _, sender := range senders {
sender = strings.ToLower(sender)
pres := stanza.NewPresence(stanza.Attrs{To: p.Attrs.From, From: jidFromThreemaGroup(sender, header)})
presMUCUserList := PresMUCUserList{
Items: []PresMUCUserItem{
{
Affiliation: "admin",
Role: "moderator",
},
},
}
if sender == ownsender {
presMUCUserList.Status = &PresMUCUserStatus{Code: 110}
}
pres.Extensions = append(pres.Extensions, presMUCUserList)
a.xmpp <- pres
}
return nil
}
2019-06-28 03:03:38 +02:00
func (a *Account) Send(to string, msg stanza.Message) error {
m, err := a.sending(to, msg)
if err != nil {
return err
}
if m != nil {
a.send <- m
}
return nil
}
2019-06-28 03:03:38 +02:00
func (a *Account) sending(to string, msg stanza.Message) (o3.Message, error) {
2019-09-06 03:08:58 +02:00
from := string(a.AccountThreema.TID)
2019-06-06 22:54:45 +02:00
logger := log.WithFields(map[string]interface{}{
2019-09-06 03:08:58 +02:00
"from": a.XMPP.String(),
"from_t": from,
"to": to,
2019-06-06 22:54:45 +02:00
})
2019-09-06 03:57:27 +02:00
msg3ID := o3.NewMsgID()
header := &o3.MessageHeader{
Sender: o3.NewIDString(from),
ID: msg3ID,
Recipient: o3.NewIDString(to),
PubNick: a.ThreemaID.Nick,
}
2019-09-06 04:34:13 +02:00
var groupHeader *o3.GroupMessageHeader
if msg.Type == stanza.MessageTypeGroupchat {
toA := strings.SplitN(to, "-", 2)
gid, err := base32.StdEncoding.DecodeString(toA[1])
if err != nil {
return nil, err
}
groupHeader = &o3.GroupMessageHeader{
CreatorID: o3.NewIDString(toA[0]),
}
copy(groupHeader.GroupID[:], gid)
}
2019-06-06 22:05:33 +02:00
chatState := false
chatStateComposing := false
2019-06-06 22:54:45 +02:00
2019-06-06 22:05:33 +02:00
msgStateID := ""
msgStateRead := false
for _, el := range msg.Extensions {
switch ex := el.(type) {
2019-06-06 22:54:45 +02:00
2019-06-28 03:03:38 +02:00
case *stanza.StateActive:
2019-06-06 22:05:33 +02:00
chatState = true
2019-06-28 03:03:38 +02:00
case *stanza.StateComposing:
2019-06-06 22:05:33 +02:00
chatState = true
chatStateComposing = true
2019-06-28 03:03:38 +02:00
case *stanza.StateGone:
2019-06-06 22:05:33 +02:00
chatState = true
2019-06-28 03:03:38 +02:00
case *stanza.StateInactive:
2019-06-06 22:05:33 +02:00
chatState = true
2019-06-28 03:03:38 +02:00
case *stanza.StatePaused:
2019-06-06 22:05:33 +02:00
chatState = true
2019-06-06 22:54:45 +02:00
2019-06-28 03:03:38 +02:00
case *stanza.ReceiptReceived:
2019-06-06 22:05:33 +02:00
msgStateID = ex.ID
2019-06-28 03:03:38 +02:00
case *stanza.MarkReceived:
2019-06-06 22:05:33 +02:00
msgStateID = ex.ID
2019-06-06 22:54:45 +02:00
2019-06-28 03:03:38 +02:00
case *stanza.MarkDisplayed:
2019-06-06 22:05:33 +02:00
msgStateRead = true
msgStateID = ex.ID
}
}
2019-06-06 22:05:33 +02:00
if msg.Body == "" {
if msgStateID != "" {
id, err := strconv.ParseUint(msgStateID, 10, 64)
if err != nil {
return nil, err
}
2019-09-06 03:08:58 +02:00
drm := &o3.DeliveryReceiptMessage{
2019-09-06 03:57:27 +02:00
MessageHeader: header,
Status: o3.MSGDELIVERED,
MessageID: id,
2019-06-06 22:05:33 +02:00
}
2019-08-30 17:28:21 +02:00
if msgStateRead {
drm.Status = o3.MSGREAD
2019-06-06 22:05:33 +02:00
}
2019-06-06 22:54:45 +02:00
logger.WithFields(map[string]interface{}{
2019-06-06 22:05:33 +02:00
"msg_id": id,
2019-08-30 17:28:21 +02:00
"type": drm.Status,
2019-06-06 22:05:33 +02:00
}).Debug("update status of threema message")
return drm, nil
}
2019-06-06 22:05:33 +02:00
if chatState {
2019-09-06 03:08:58 +02:00
tnm := &o3.TypingNotificationMessage{
2019-09-06 03:57:27 +02:00
MessageHeader: header,
2019-08-30 17:28:21 +02:00
}
2019-06-06 22:54:45 +02:00
if chatStateComposing {
tnm.OnOff = 0x1
}
logger.WithFields(map[string]interface{}{
2019-09-06 03:57:27 +02:00
"on": tnm.OnOff,
2019-09-06 03:08:58 +02:00
}).Debug("send typing")
return tnm, nil
}
}
// send text message
2019-09-06 03:08:58 +02:00
msg3 := &o3.TextMessage{
GroupMessageHeader: groupHeader,
MessageHeader: header,
Body: msg.Body,
}
2019-09-06 04:34:13 +02:00
logger = logger.WithFields(map[string]interface{}{
2019-06-06 22:54:45 +02:00
"x_id": msg.Id,
2019-08-30 17:28:21 +02:00
"t_id": msg3ID,
2019-06-06 22:54:45 +02:00
"text": msg.Body,
2019-09-06 04:34:13 +02:00
})
if groupHeader != nil {
logger.Debug("send grouptext")
// TODO iterate of all occupants
2019-10-03 23:44:42 +02:00
//msg3.GroupMessageHeader.Recipient: o3.NewIDString(to),
return msg3, nil
2019-09-06 04:34:13 +02:00
}
a.deliveredMSG[msg3ID] = msg.Id
a.readedMSG[msg3ID] = msg.Id
logger.Debug("send text")
return msg3, nil
}