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-29 09:51:09 +00:00
|
|
|
"strings"
|
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"
|
|
|
|
)
|
|
|
|
|
|
|
|
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) {
|
2020-02-28 16:32:09 +00:00
|
|
|
mx_room_id, err := dbGetMxPmRoom(EASYBRIDGE_SYSTEM_PROTOCOL, UserID("Easybridge"), ezbrMxId(), user_mx_id, "easybridge")
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
widget_kv_key := "ezbr_widget_on:" + mx_room_id
|
2020-03-13 10:29:03 +00:00
|
|
|
if config.WebURL != "" && dbKvGet(widget_kv_key) != "yes" {
|
2020-02-28 16:32:09 +00:00
|
|
|
widget := map[string]interface{}{
|
|
|
|
"type": "m.easybridge",
|
|
|
|
"url": config.WebURL,
|
|
|
|
"name": "Easybridge account configuration dashboard",
|
|
|
|
}
|
|
|
|
err = mx.PutStateAs(mx_room_id, "im.vector.modular.widgets", "ezbr_widget", widget, ezbrMxId())
|
2020-03-13 10:29:03 +00:00
|
|
|
if err == nil {
|
|
|
|
dbKvPut(widget_kv_key, "yes")
|
2020-02-28 16:32:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mx_room_id, nil
|
2020-02-21 13:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2020-02-29 09:51:09 +00:00
|
|
|
what := fmt.Sprintf("%s_%s", safeStringForId(string(id)), protocol)
|
|
|
|
return strings.Replace(config.NameFormat, "{}", what, 1)
|
2020-02-21 13:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func userMxId(protocol string, id UserID) string {
|
2020-02-29 09:51:09 +00:00
|
|
|
what := fmt.Sprintf("%s_%s", safeStringForId(string(id)), protocol)
|
|
|
|
return strings.Replace(config.NameFormat, "{}", what, 1)
|
2020-02-21 13:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
2020-02-29 09:51:09 +00:00
|
|
|
func isBridgedIdentifier(mxid string) bool {
|
|
|
|
if mxid[0] == '@' || mxid[0] == '#' {
|
|
|
|
return isBridgedIdentifier(mxid[1:])
|
|
|
|
}
|
|
|
|
|
|
|
|
if strings.Contains(mxid, ":") {
|
|
|
|
sp := strings.Split(mxid, ":")
|
|
|
|
return (sp[1] == config.MatrixDomain) && isBridgedIdentifier(sp[0])
|
|
|
|
}
|
|
|
|
|
|
|
|
nameformat_fixed_part := strings.Replace(config.NameFormat, "{}", "", 1)
|
|
|
|
if strings.HasPrefix(config.NameFormat, "{}") {
|
|
|
|
return strings.HasSuffix(mxid, nameformat_fixed_part)
|
|
|
|
} else if strings.HasSuffix(config.NameFormat, "{}") {
|
|
|
|
return strings.HasPrefix(mxid, nameformat_fixed_part)
|
|
|
|
} else {
|
|
|
|
// This is not supported
|
|
|
|
log.Fatal("Invalid name format %s, please put {} at the beginning or at the end", config.NameFormat)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
}
|