2020-02-26 15:07:33 +00:00
|
|
|
package main
|
2020-02-21 13:27:42 +00:00
|
|
|
|
|
|
|
import (
|
2020-02-26 19:21:32 +00:00
|
|
|
"crypto/rand"
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
2020-02-21 13:27:42 +00:00
|
|
|
"fmt"
|
2020-02-26 14:59:33 +00:00
|
|
|
"unicode"
|
2020-02-21 13:27:42 +00:00
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
2020-02-26 19:21:32 +00:00
|
|
|
"golang.org/x/crypto/nacl/secretbox"
|
2020-02-21 13:27:42 +00:00
|
|
|
|
|
|
|
. "git.deuxfleurs.fr/Deuxfleurs/easybridge/connector"
|
2020-02-26 20:36:35 +00:00
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/connector/irc"
|
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/connector/mattermost"
|
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/connector/xmpp"
|
2020-02-21 13:27:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const EASYBRIDGE_SYSTEM_PROTOCOL string = "✯◡✯"
|
|
|
|
|
|
|
|
func ezbrMxId() string {
|
|
|
|
return fmt.Sprintf("@%s:%s", registration.SenderLocalpart, config.MatrixDomain)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ezbrSystemRoom(user_mx_id string) (string, error) {
|
|
|
|
return dbGetMxPmRoom(EASYBRIDGE_SYSTEM_PROTOCOL, UserID("Easybridge"), ezbrMxId(), user_mx_id, "easybridge")
|
|
|
|
}
|
|
|
|
|
|
|
|
func ezbrSystemSend(user_mx_id string, msg string) {
|
|
|
|
mx_room_id, err := ezbrSystemRoom(user_mx_id)
|
|
|
|
if err == nil {
|
|
|
|
err = mx.SendMessageAs(mx_room_id, "m.text", msg, ezbrMxId())
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
log.Warnf("(%s) %s", user_mx_id, msg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ezbrSystemSendf(user_mx_id string, format string, args ...interface{}) {
|
|
|
|
ezbrSystemSend(user_mx_id, fmt.Sprintf(format, args...))
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----
|
|
|
|
|
|
|
|
func roomAlias(protocol string, id RoomID) string {
|
|
|
|
return fmt.Sprintf("_ezbr__%s__%s", safeStringForId(string(id)), protocol)
|
|
|
|
}
|
|
|
|
|
|
|
|
func userMxId(protocol string, id UserID) string {
|
|
|
|
return fmt.Sprintf("_ezbr__%s__%s", safeStringForId(string(id)), protocol)
|
|
|
|
}
|
|
|
|
|
|
|
|
func safeStringForId(in string) string {
|
2020-02-26 14:59:33 +00:00
|
|
|
id2 := ""
|
|
|
|
for _, c := range in {
|
|
|
|
if c == '@' {
|
|
|
|
id2 += "__"
|
|
|
|
} else if c == ':' {
|
|
|
|
id2 += "_"
|
2020-02-26 16:03:49 +00:00
|
|
|
} else if unicode.IsDigit(c) || unicode.IsLetter(c) || c == '.' || c == '-' || c == '_' {
|
2020-02-26 14:59:33 +00:00
|
|
|
id2 += string(c)
|
|
|
|
}
|
|
|
|
}
|
2020-02-21 13:27:42 +00:00
|
|
|
return id2
|
|
|
|
}
|
2020-02-26 19:21:32 +00:00
|
|
|
|
|
|
|
// ---- Encoding and encryption of account config
|
|
|
|
|
|
|
|
func encryptAccountConfig(config map[string]string, key *[32]byte) string {
|
|
|
|
bytes, err := json.Marshal(config)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var nonce [24]byte
|
|
|
|
_, err = rand.Read(nonce[:])
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
crypto := secretbox.Seal([]byte{}, bytes, &nonce, key)
|
|
|
|
all := append(nonce[:], crypto...)
|
|
|
|
return base64.StdEncoding.EncodeToString(all)
|
|
|
|
}
|
|
|
|
|
|
|
|
func decryptAccountConfig(data string, key *[32]byte) (map[string]string, error) {
|
|
|
|
bytes, err := base64.StdEncoding.DecodeString(data)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var nonce [24]byte
|
|
|
|
copy(nonce[:], bytes[:24])
|
|
|
|
|
|
|
|
decoded, ok := secretbox.Open([]byte{}, bytes[24:], &nonce, key)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("Invalid key")
|
|
|
|
}
|
|
|
|
|
|
|
|
var config map[string]string
|
|
|
|
err = json.Unmarshal(decoded, &config)
|
|
|
|
return config, err
|
|
|
|
}
|
2020-02-26 20:36:35 +00:00
|
|
|
|
|
|
|
// ----
|
|
|
|
|
|
|
|
func createConnector(protocol string) Connector {
|
|
|
|
switch protocol {
|
|
|
|
case "irc":
|
|
|
|
return &irc.IRC{}
|
|
|
|
case "xmpp":
|
|
|
|
return &xmpp.XMPP{}
|
|
|
|
case "mattermost":
|
|
|
|
return &mattermost.Mattermost{}
|
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|