2020-02-16 15:26:55 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2020-02-16 18:30:49 +00:00
|
|
|
"crypto/rand"
|
|
|
|
"encoding/hex"
|
2020-02-17 18:02:26 +00:00
|
|
|
"encoding/json"
|
2020-02-16 18:30:49 +00:00
|
|
|
"flag"
|
2020-02-17 18:02:26 +00:00
|
|
|
"fmt"
|
|
|
|
_ "fmt"
|
2020-02-16 18:30:49 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2020-02-17 18:02:26 +00:00
|
|
|
_ "strings"
|
2020-02-16 18:30:49 +00:00
|
|
|
_ "time"
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-02-17 08:41:08 +00:00
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/appservice"
|
2020-02-16 15:26:55 +00:00
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/connector"
|
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/connector/irc"
|
2020-02-16 16:53:31 +00:00
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/connector/xmpp"
|
2020-02-17 22:17:13 +00:00
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/connector/mattermost"
|
2020-02-17 18:02:26 +00:00
|
|
|
"git.deuxfleurs.fr/Deuxfleurs/easybridge/mxlib"
|
2020-02-16 15:26:55 +00:00
|
|
|
)
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
type ConfigAccount struct {
|
2020-02-17 18:02:26 +00:00
|
|
|
Protocol string `json:"protocol"`
|
|
|
|
Rooms []string `json:"rooms"`
|
|
|
|
Config map[string]string `json:"config"`
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
type ConfigFile struct {
|
2020-02-17 18:02:26 +00:00
|
|
|
HttpBindAddr string `json:"http_bind_addr"`
|
|
|
|
Registration string `json:"registration"`
|
|
|
|
Server string `json:"homeserver_url"`
|
|
|
|
DbType string `json:"db_type"`
|
|
|
|
DbPath string `json:"db_path"`
|
|
|
|
MatrixDomain string `json:"matrix_domain"`
|
|
|
|
Accounts map[string]map[string]ConfigAccount `json:"accounts"`
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
var configFlag = flag.String("config", "./config.json", "Configuration file path")
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
var config *ConfigFile
|
|
|
|
var registration *mxlib.Registration
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
func readConfig() ConfigFile {
|
|
|
|
config_file := ConfigFile{
|
|
|
|
HttpBindAddr: "0.0.0.0:8321",
|
|
|
|
Registration: "./registration.yaml",
|
2020-02-17 18:02:26 +00:00
|
|
|
Server: "http://localhost:8008",
|
|
|
|
DbType: "sqlite3",
|
|
|
|
DbPath: "easybridge.db",
|
|
|
|
Accounts: map[string]map[string]ConfigAccount{},
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
_, err := os.Stat(*configFlag)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
// Generate default config file
|
|
|
|
log.Printf("Generating default config file as %s", *configFlag)
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
bytes, err := json.MarshalIndent(&config_file, "", " ")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ioutil.WriteFile(*configFlag, bytes, 0644)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return config_file
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2020-02-16 18:30:49 +00:00
|
|
|
log.Fatal(err)
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
bytes, err := ioutil.ReadFile(*configFlag)
|
2020-02-16 15:26:55 +00:00
|
|
|
if err != nil {
|
2020-02-16 18:30:49 +00:00
|
|
|
log.Fatal(err)
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
err = json.Unmarshal(bytes, &config_file)
|
2020-02-16 15:26:55 +00:00
|
|
|
if err != nil {
|
2020-02-16 18:30:49 +00:00
|
|
|
log.Fatal(err)
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
return config_file
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
2020-02-16 16:53:31 +00:00
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
func readRegistration(file string) mxlib.Registration {
|
|
|
|
rnd := make([]byte, 64)
|
|
|
|
n, err := rand.Read(rnd)
|
|
|
|
if err != nil || n != 64 {
|
|
|
|
log.Fatal(err)
|
2020-02-16 16:53:31 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
reg := mxlib.Registration{
|
2020-02-17 18:02:26 +00:00
|
|
|
Id: "Easybridge",
|
|
|
|
Url: "http://localhost:8321",
|
|
|
|
AsToken: hex.EncodeToString(rnd[:32]),
|
|
|
|
HsToken: hex.EncodeToString(rnd[32:]),
|
2020-02-17 14:30:01 +00:00
|
|
|
SenderLocalpart: "_ezbr_",
|
2020-02-21 18:28:00 +00:00
|
|
|
RateLimited: false,
|
2020-02-16 18:30:49 +00:00
|
|
|
Namespaces: mxlib.RegistrationNamespaceSet{
|
|
|
|
Users: []mxlib.RegistrationNamespace{
|
|
|
|
mxlib.RegistrationNamespace{
|
|
|
|
Exclusive: true,
|
2020-02-17 18:02:26 +00:00
|
|
|
Regex: "@_ezbr_.*",
|
2020-02-16 18:30:49 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
Aliases: []mxlib.RegistrationNamespace{
|
|
|
|
mxlib.RegistrationNamespace{
|
|
|
|
Exclusive: true,
|
2020-02-17 18:02:26 +00:00
|
|
|
Regex: "#_ezbr_.*",
|
2020-02-16 18:30:49 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
Rooms: []mxlib.RegistrationNamespace{},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = os.Stat(file)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
// Generate default config file
|
|
|
|
log.Printf("Generating default registration file as %s", file)
|
|
|
|
|
|
|
|
bytes, err := yaml.Marshal(®)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ioutil.WriteFile(file, bytes, 0644)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return reg
|
2020-02-16 16:53:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2020-02-16 18:30:49 +00:00
|
|
|
log.Fatal(err)
|
2020-02-16 16:53:31 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
bytes, err := ioutil.ReadFile(file)
|
2020-02-16 16:53:31 +00:00
|
|
|
if err != nil {
|
2020-02-16 18:30:49 +00:00
|
|
|
log.Fatal(err)
|
2020-02-16 16:53:31 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
err = yaml.Unmarshal(bytes, ®)
|
2020-02-16 16:53:31 +00:00
|
|
|
if err != nil {
|
2020-02-16 18:30:49 +00:00
|
|
|
log.Fatal(err)
|
2020-02-16 16:53:31 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
return reg
|
2020-02-16 16:53:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
2020-02-17 08:41:08 +00:00
|
|
|
log.SetLevel(log.DebugLevel)
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
config_file := readConfig()
|
|
|
|
config = &config_file
|
|
|
|
|
|
|
|
reg_file := readRegistration(config.Registration)
|
|
|
|
registration = ®_file
|
|
|
|
|
|
|
|
as_config := &appservice.Config{
|
|
|
|
HttpBindAddr: config.HttpBindAddr,
|
2020-02-17 18:02:26 +00:00
|
|
|
Server: config.Server,
|
|
|
|
DbType: config.DbType,
|
|
|
|
DbPath: config.DbPath,
|
2020-02-16 21:07:41 +00:00
|
|
|
MatrixDomain: config.MatrixDomain,
|
2020-02-16 18:30:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
errch, err := appservice.Start(registration, as_config)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for user, accounts := range config.Accounts {
|
|
|
|
for name, params := range accounts {
|
|
|
|
var conn connector.Connector
|
|
|
|
switch params.Protocol {
|
|
|
|
case "irc":
|
|
|
|
conn = &irc.IRC{}
|
|
|
|
case "xmpp":
|
|
|
|
conn = &xmpp.XMPP{}
|
2020-02-17 22:17:13 +00:00
|
|
|
case "mattermost":
|
|
|
|
conn = &mattermost.Mattermost{}
|
2020-02-17 15:28:32 +00:00
|
|
|
default:
|
|
|
|
log.Fatalf("Invalid protocol %s", params.Protocol)
|
2020-02-16 18:30:49 +00:00
|
|
|
}
|
|
|
|
account := &appservice.Account{
|
2020-02-17 18:02:26 +00:00
|
|
|
MatrixUser: fmt.Sprintf("@%s:%s", user, config.MatrixDomain),
|
2020-02-16 18:30:49 +00:00
|
|
|
AccountName: name,
|
2020-02-17 18:02:26 +00:00
|
|
|
Protocol: params.Protocol,
|
|
|
|
Conn: conn,
|
2020-02-16 22:27:03 +00:00
|
|
|
JoinedRooms: map[connector.RoomID]bool{},
|
2020-02-16 18:30:49 +00:00
|
|
|
}
|
|
|
|
conn.SetHandler(account)
|
2020-02-16 22:27:03 +00:00
|
|
|
appservice.AddAccount(account)
|
2020-02-21 14:12:22 +00:00
|
|
|
go connectAndJoin(account, params)
|
2020-02-16 18:30:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = <-errch
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-21 14:12:22 +00:00
|
|
|
func connectAndJoin(account *appservice.Account, params ConfigAccount) {
|
2020-02-16 18:30:49 +00:00
|
|
|
log.Printf("Connecting to %s", params.Protocol)
|
2020-02-21 14:12:22 +00:00
|
|
|
err := account.Conn.Configure(params.Config)
|
2020-02-16 18:30:49 +00:00
|
|
|
if err != nil {
|
2020-02-17 08:41:08 +00:00
|
|
|
log.Printf("Could not connect to %s: %s", params.Protocol, err)
|
2020-02-16 18:30:49 +00:00
|
|
|
} else {
|
2020-02-17 08:41:08 +00:00
|
|
|
log.Printf("Connected to %s, now joining %#v", params.Protocol, params.Rooms)
|
2020-02-16 18:30:49 +00:00
|
|
|
for _, room := range params.Rooms {
|
2020-02-21 14:12:22 +00:00
|
|
|
err := account.Conn.Join(connector.RoomID(room))
|
2020-02-16 18:30:49 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("Could not join %s: %s", room, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-02-16 16:53:31 +00:00
|
|
|
}
|