telegabber/xmpp/component.go

141 lines
3 KiB
Go
Raw Normal View History

2019-10-22 16:36:54 +00:00
package xmpp
import (
"github.com/pkg/errors"
"time"
2019-10-22 16:36:54 +00:00
"dev.narayana.im/narayana/telegabber/config"
"dev.narayana.im/narayana/telegabber/persistence"
"dev.narayana.im/narayana/telegabber/telegram"
2019-11-24 17:10:29 +00:00
"dev.narayana.im/narayana/telegabber/xmpp/gateway"
2019-10-22 16:36:54 +00:00
log "github.com/sirupsen/logrus"
2019-10-22 16:36:54 +00:00
"gosrc.io/xmpp"
)
var tgConf config.TelegramConfig
2019-11-24 17:10:29 +00:00
var sessions map[string]*telegram.Client
2019-12-04 15:55:15 +00:00
var db *persistence.SessionsYamlDB
2019-10-29 01:23:57 +00:00
// NewComponent starts a new component and wraps it in
// a stream manager that you should start yourself
2019-11-19 20:25:14 +00:00
func NewComponent(conf config.XMPPConfig, tc config.TelegramConfig) (*xmpp.StreamManager, *xmpp.Component, error) {
var err error
2019-11-24 17:10:29 +00:00
gateway.Jid, err = xmpp.NewJid(conf.Jid)
if err != nil {
2019-11-19 20:25:14 +00:00
return nil, nil, err
}
tgConf = tc
2019-10-22 16:36:54 +00:00
options := xmpp.ComponentOptions{
Address: conf.Host + ":" + conf.Port,
Domain: conf.Jid,
Secret: conf.Password,
Name: "telegabber",
}
router := xmpp.NewRouter()
router.HandleFunc("iq", HandleIq)
router.HandleFunc("presence", HandlePresence)
2019-10-22 16:36:54 +00:00
router.HandleFunc("message", HandleMessage)
component, err := xmpp.NewComponent(options, router)
if err != nil {
2019-11-19 20:25:14 +00:00
return nil, nil, err
2019-10-22 16:36:54 +00:00
}
2019-11-24 17:10:29 +00:00
err = loadSessions(conf.Db, component)
if err != nil {
return nil, nil, err
}
2019-12-04 15:55:15 +00:00
sm := xmpp.NewStreamManager(component, func(s xmpp.Sender) {
go heartbeat(component)
})
2019-11-19 20:25:14 +00:00
return sm, component, nil
2019-10-22 16:36:54 +00:00
}
func heartbeat(component *xmpp.Component) {
var err error
probeType := gateway.SPType("probe")
2019-11-14 20:11:04 +00:00
for jid := range sessions {
2019-12-04 15:55:15 +00:00
err = gateway.SendPresence(component, jid, probeType)
if err != nil {
log.Error(err)
}
2019-11-14 20:11:04 +00:00
}
log.Info("Starting heartbeat queue")
for {
2019-11-29 00:51:41 +00:00
for key, presence := range gateway.Queue {
err = component.Send(presence)
if err != nil {
gateway.LogBadPresence(err, presence)
} else {
2019-11-29 00:51:41 +00:00
delete(gateway.Queue, key)
}
}
time.Sleep(60e9)
}
}
2019-11-24 17:10:29 +00:00
func loadSessions(dbPath string, component *xmpp.Component) error {
var err error
2019-11-24 17:10:29 +00:00
sessions = make(map[string]*telegram.Client)
db, err = persistence.LoadSessions(dbPath)
if err != nil {
return err
}
db.Transaction(func() bool {
for jid, session := range db.Data.Sessions {
2019-11-24 17:10:29 +00:00
getTelegramInstance(jid, &session, component)
}
return false
}, persistence.SessionMarshaller)
return nil
}
2019-11-24 17:10:29 +00:00
func getTelegramInstance(jid string, savedSession *persistence.Session, component *xmpp.Component) (*telegram.Client, bool) {
var err error
session, ok := sessions[jid]
if !ok {
2019-11-24 17:10:29 +00:00
session, err = telegram.NewClient(tgConf, jid, component, savedSession)
if err != nil {
log.Error(errors.Wrap(err, "TDlib initialization failure"))
return session, false
}
sessions[jid] = session
}
return session, true
}
2019-11-14 20:11:04 +00:00
2019-11-19 20:25:14 +00:00
// Close gracefully terminates the component and saves active sessions
func Close(component *xmpp.Component) {
log.Error("Disconnecting...")
for _, session := range sessions {
session.Disconnect()
}
db.Transaction(func() bool {
for jid, session := range sessions {
db.Data.Sessions[jid] = *session.Session
}
return true
}, persistence.SessionMarshaller)
component.Disconnect()
}