Implementation of Ruby YAML::Store and read/save of the session database

This commit is contained in:
bodqhrohro 2019-11-11 01:50:50 +02:00
parent 7185d4ac9b
commit fbe99c65ec
5 changed files with 126 additions and 1 deletions

1
.gitignore vendored
View file

@ -1,3 +1,4 @@
config.yml config.yml
telegabber telegabber
sessions/ sessions/
session.dat

View file

@ -21,4 +21,4 @@
:host: '127.0.0.1' :host: '127.0.0.1'
:port: 8899 :port: 8899
:password: 'password' :password: 'password'
:db: 'sessions.dat' :db: 'session.dat'

69
persistence/sessions.go Normal file
View file

@ -0,0 +1,69 @@
package persistence
import (
"github.com/pkg/errors"
"io/ioutil"
"dev.narayana.im/narayana/telegabber/yamldb"
"gopkg.in/yaml.v2"
)
// SessionsYamlDB wraps YamlDB with Session
type SessionsYamlDB struct {
yamldb.YamlDB
Data *SessionsMap
}
// SessionsMap is for :sessions: subtree
type SessionsMap struct {
Sessions map[string]Session `yaml:":sessions"`
}
// Session is a key-values subtree
type Session struct {
Login string `yaml:":login"`
}
var sessionDB SessionsYamlDB
// Marshaller implementation for YamlDB
func Marshaller() ([]byte, error) {
return yaml.Marshal(sessionDB.Data)
}
// LoadSessions restores TDlib sessions from the previous run
func LoadSessions(path string) (SessionsYamlDB, error) {
var sessionData SessionsMap
sessionDB, err := initYamlDB(path, &sessionData)
if err != nil {
return sessionDB, errors.Wrap(err, "Sessions restore error")
}
sessionDB.Transaction(func() {
}, Marshaller)
return sessionDB, nil
}
func initYamlDB(path string, dataPtr *SessionsMap) (SessionsYamlDB, error) {
file, err := ioutil.ReadFile(path)
if err == nil {
err = yaml.Unmarshal(file, dataPtr)
if err != nil {
return SessionsYamlDB{}, errors.Wrap(err, "YamlDB is corrupted")
}
} else {
// DB file does not exist, create an empty DB
dataPtr.Sessions = make(map[string]Session)
}
return SessionsYamlDB{
YamlDB: yamldb.YamlDB{
Path: path,
PathNew: path + ".new",
},
Data: dataPtr,
}, nil
}

View file

@ -2,6 +2,7 @@ package xmpp
import ( import (
"dev.narayana.im/narayana/telegabber/config" "dev.narayana.im/narayana/telegabber/config"
"dev.narayana.im/narayana/telegabber/persistence"
"dev.narayana.im/narayana/telegabber/telegram" "dev.narayana.im/narayana/telegabber/telegram"
"gosrc.io/xmpp" "gosrc.io/xmpp"
@ -10,11 +11,13 @@ import (
var jid *xmpp.Jid var jid *xmpp.Jid
var tgConf config.TelegramConfig var tgConf config.TelegramConfig
var sessions map[string]telegram.Client var sessions map[string]telegram.Client
var db persistence.SessionsYamlDB
// NewComponent starts a new component and wraps it in // NewComponent starts a new component and wraps it in
// a stream manager that you should start yourself // a stream manager that you should start yourself
func NewComponent(conf config.XMPPConfig, tc config.TelegramConfig) (*xmpp.StreamManager, error) { func NewComponent(conf config.XMPPConfig, tc config.TelegramConfig) (*xmpp.StreamManager, error) {
var err error var err error
jid, err = xmpp.NewJid(conf.Jid) jid, err = xmpp.NewJid(conf.Jid)
if err != nil { if err != nil {
return nil, err return nil, err
@ -23,6 +26,11 @@ func NewComponent(conf config.XMPPConfig, tc config.TelegramConfig) (*xmpp.Strea
tgConf = tc tgConf = tc
sessions = make(map[string]telegram.Client) sessions = make(map[string]telegram.Client)
db, err = persistence.LoadSessions(conf.Db)
if err != nil {
return nil, err
}
options := xmpp.ComponentOptions{ options := xmpp.ComponentOptions{
Address: conf.Host + ":" + conf.Port, Address: conf.Host + ":" + conf.Port,
Domain: conf.Jid, Domain: conf.Jid,

47
yamldb/yamldb.go Normal file
View file

@ -0,0 +1,47 @@
package yamldb
import (
"github.com/pkg/errors"
"io/ioutil"
"os"
"sync"
log "github.com/sirupsen/logrus"
)
// YamlDB represents a YAML file database instance
type YamlDB struct {
Path string
PathNew string
lock sync.Mutex
}
// Transaction executes the given callback and safely saves
// the data after they are modified within the callback
func (db *YamlDB) Transaction(callback func(), marshaller func() ([]byte, error)) error {
var err error
log.Debug("Enter transaction")
db.lock.Lock()
defer func() {
db.lock.Unlock()
log.Debug("Exit transaction")
}()
callback()
yamlData, err := marshaller()
if err != nil {
return errors.Wrap(err, "Data marshalling error")
}
err = ioutil.WriteFile(db.PathNew, yamlData, 0644)
if err != nil {
return errors.Wrap(err, "YamlDB write failure")
}
err = os.Rename(db.PathNew, db.Path)
if err != nil {
return errors.Wrap(err, "Couldn't rewrite an old YamlDB file")
}
return nil
}