diff bot and room settings

This commit is contained in:
Aine
2022-08-30 14:37:19 +03:00
parent f97ebb604a
commit 5ed3a53223
9 changed files with 127 additions and 102 deletions

View File

@@ -28,7 +28,7 @@ func (b *Bot) allowOwner(actorID id.UserID, targetRoomID id.RoomID) bool {
} }
} }
cfg, err := b.getSettings(targetRoomID) cfg, err := b.getRoomSettings(targetRoomID)
if err != nil { if err != nil {
b.Error(context.Background(), targetRoomID, "failed to retrieve settings: %v", err) b.Error(context.Background(), targetRoomID, "failed to retrieve settings: %v", err)
return false return false

View File

@@ -23,7 +23,8 @@ type Bot struct {
allowedAdmins []*regexp.Regexp allowedAdmins []*regexp.Regexp
commands commandList commands commandList
rooms sync.Map rooms sync.Map
cfg cache.Cache[settings] botcfg cache.Cache[botsettings]
cfg cache.Cache[roomsettings]
log *logger.Logger log *logger.Logger
lp *linkpearl.Linkpearl lp *linkpearl.Linkpearl
mu map[id.RoomID]*sync.Mutex mu map[id.RoomID]*sync.Mutex
@@ -43,7 +44,8 @@ func New(
prefix: prefix, prefix: prefix,
domain: domain, domain: domain,
rooms: sync.Map{}, rooms: sync.Map{},
cfg: cache.NewLRU[settings](1000), botcfg: cache.NewLRU[botsettings](1),
cfg: cache.NewLRU[roomsettings](1000),
log: log, log: log,
lp: lp, lp: lp,
mu: map[id.RoomID]*sync.Mutex{}, mu: map[id.RoomID]*sync.Mutex{},

View File

@@ -193,7 +193,7 @@ func (b *Bot) sendIntroduction(ctx context.Context, roomID id.RoomID) {
func (b *Bot) sendHelp(ctx context.Context) { func (b *Bot) sendHelp(ctx context.Context) {
evt := eventFromContext(ctx) evt := eventFromContext(ctx)
cfg, serr := b.getSettings(evt.RoomID) cfg, serr := b.getRoomSettings(evt.RoomID)
if serr != nil { if serr != nil {
b.log.Error("cannot retrieve settings: %v", serr) b.log.Error("cannot retrieve settings: %v", serr)
} }

View File

@@ -13,7 +13,7 @@ import (
func (b *Bot) sendMailboxes(ctx context.Context) { func (b *Bot) sendMailboxes(ctx context.Context) {
evt := eventFromContext(ctx) evt := eventFromContext(ctx)
mailboxes := map[string]settings{} mailboxes := map[string]roomsettings{}
slice := []string{} slice := []string{}
b.rooms.Range(func(key any, value any) bool { b.rooms.Range(func(key any, value any) bool {
if key == nil { if key == nil {
@@ -31,7 +31,7 @@ func (b *Bot) sendMailboxes(ctx context.Context) {
if !ok { if !ok {
return true return true
} }
config, err := b.getSettings(roomID) config, err := b.getRoomSettings(roomID)
if err != nil { if err != nil {
b.log.Error("cannot retrieve settings: %v", err) b.log.Error("cannot retrieve settings: %v", err)
} }
@@ -79,7 +79,7 @@ func (b *Bot) runDelete(ctx context.Context, commandSlice []string) {
roomID := v.(id.RoomID) roomID := v.(id.RoomID)
b.rooms.Delete(mailbox) b.rooms.Delete(mailbox)
err := b.setSettings(roomID, settings{}) err := b.setRoomSettings(roomID, roomsettings{})
if err != nil { if err != nil {
b.Error(ctx, evt.RoomID, "cannot update settings: %v", err) b.Error(ctx, evt.RoomID, "cannot update settings: %v", err)
return return

View File

@@ -7,7 +7,7 @@ import (
func (b *Bot) runStop(ctx context.Context) { func (b *Bot) runStop(ctx context.Context) {
evt := eventFromContext(ctx) evt := eventFromContext(ctx)
cfg, err := b.getSettings(evt.RoomID) cfg, err := b.getRoomSettings(evt.RoomID)
if err != nil { if err != nil {
b.Error(ctx, evt.RoomID, "failed to retrieve settings: %v", err) b.Error(ctx, evt.RoomID, "failed to retrieve settings: %v", err)
return return
@@ -21,7 +21,7 @@ func (b *Bot) runStop(ctx context.Context) {
b.rooms.Delete(mailbox) b.rooms.Delete(mailbox)
err = b.setSettings(evt.RoomID, settings{}) err = b.setRoomSettings(evt.RoomID, roomsettings{})
if err != nil { if err != nil {
b.Error(ctx, evt.RoomID, "cannot update settings: %v", err) b.Error(ctx, evt.RoomID, "cannot update settings: %v", err)
return return
@@ -40,7 +40,7 @@ func (b *Bot) handleOption(ctx context.Context, cmd []string) {
func (b *Bot) getOption(ctx context.Context, name string) { func (b *Bot) getOption(ctx context.Context, name string) {
evt := eventFromContext(ctx) evt := eventFromContext(ctx)
cfg, err := b.getSettings(evt.RoomID) cfg, err := b.getRoomSettings(evt.RoomID)
if err != nil { if err != nil {
b.Error(ctx, evt.RoomID, "failed to retrieve settings: %v", err) b.Error(ctx, evt.RoomID, "failed to retrieve settings: %v", err)
return return
@@ -74,7 +74,7 @@ func (b *Bot) setOption(ctx context.Context, name, value string) {
} }
} }
cfg, err := b.getSettings(evt.RoomID) cfg, err := b.getRoomSettings(evt.RoomID)
if err != nil { if err != nil {
b.Error(ctx, evt.RoomID, "failed to retrieve settings: %v", err) b.Error(ctx, evt.RoomID, "failed to retrieve settings: %v", err)
return return
@@ -92,7 +92,7 @@ func (b *Bot) setOption(ctx context.Context, name, value string) {
value = fmt.Sprintf("%s@%s", value, b.domain) value = fmt.Sprintf("%s@%s", value, b.domain)
} }
err = b.setSettings(evt.RoomID, cfg) err = b.setRoomSettings(evt.RoomID, cfg)
if err != nil { if err != nil {
b.Error(ctx, evt.RoomID, "cannot update settings: %v", err) b.Error(ctx, evt.RoomID, "cannot update settings: %v", err)
return return

View File

@@ -4,15 +4,11 @@ import (
"strings" "strings"
"maunium.net/go/mautrix/id" "maunium.net/go/mautrix/id"
"gitlab.com/etke.cc/postmoogle/utils"
) )
// account data keys // account data keys
const ( const (
messagekey = "cc.etke.postmoogle.message" messagekey = "cc.etke.postmoogle.message"
settingskey = "cc.etke.postmoogle.settings"
botconfigkey = "cc.etke.postmoogle.config"
) )
// event keys // event keys
@@ -21,19 +17,6 @@ const (
eventInReplyToKey = "cc.etke.postmoogle.inReplyTo" eventInReplyToKey = "cc.etke.postmoogle.inReplyTo"
) )
// option keys
const (
optionOwner = "owner"
optionMailbox = "mailbox"
optionNoSender = "nosender"
optionNoSubject = "nosubject"
optionNoHTML = "nohtml"
optionNoThreads = "nothreads"
optionNoFiles = "nofiles"
botOptionUsers = "users"
)
var migrations = []string{} var migrations = []string{}
func (b *Bot) migrate() error { func (b *Bot) migrate() error {
@@ -72,7 +55,7 @@ func (b *Bot) syncRooms() error {
} }
for _, roomID := range resp.JoinedRooms { for _, roomID := range resp.JoinedRooms {
b.migrateSettings(roomID) b.migrateSettings(roomID)
cfg, serr := b.getSettings(roomID) cfg, serr := b.getRoomSettings(roomID)
if serr != nil { if serr != nil {
b.log.Warn("cannot get %s settings: %v", roomID, err) b.log.Warn("cannot get %s settings: %v", roomID, err)
continue continue
@@ -113,44 +96,3 @@ func (b *Bot) setThreadID(roomID id.RoomID, messageID string, eventID id.EventID
} }
} }
} }
// TODO: remove after migration
func (b *Bot) migrateBotSettings(users []string) error {
config := b.getBotSettings()
cfgUsers := config.Users()
if len(users) > 0 && len(cfgUsers) == 0 {
_, err := parseMXIDpatterns(users, "")
if err != nil {
return err
}
config.Set(botOptionUsers, strings.Join(users, " "))
return b.setBotSettings(config)
}
return nil
}
func (b *Bot) getBotSettings() settings {
cfg := b.cfg.Get(botconfigkey)
if cfg != nil {
return cfg
}
config := settings{}
err := b.lp.GetClient().GetAccountData(botconfigkey, &config)
if err != nil {
if strings.Contains(err.Error(), "M_NOT_FOUND") {
err = nil
}
b.log.Error("cannot get bot settings: %v", utils.UnwrapError(err))
} else {
b.cfg.Set(botconfigkey, config)
}
return config
}
func (b *Bot) setBotSettings(cfg settings) error {
b.cfg.Set(botconfigkey, cfg)
return utils.UnwrapError(b.lp.GetClient().SetAccountData(botconfigkey, cfg))
}

View File

@@ -12,7 +12,7 @@ import (
"gitlab.com/etke.cc/postmoogle/utils" "gitlab.com/etke.cc/postmoogle/utils"
) )
func email2content(email *utils.Email, cfg settings, threadID id.EventID) *event.Content { func email2content(email *utils.Email, cfg roomsettings, threadID id.EventID) *event.Content {
var text strings.Builder var text strings.Builder
if !cfg.NoSender() { if !cfg.NoSender() {
text.WriteString("From: ") text.WriteString("From: ")
@@ -66,7 +66,7 @@ func (b *Bot) Send(ctx context.Context, email *utils.Email) error {
b.lock(roomID) b.lock(roomID)
defer b.unlock(roomID) defer b.unlock(roomID)
cfg, err := b.getSettings(roomID) cfg, err := b.getRoomSettings(roomID)
if err != nil { if err != nil {
b.Error(ctx, roomID, "cannot get settings: %v", err) b.Error(ctx, roomID, "cannot get settings: %v", err)
} }

73
bot/settings_bot.go Normal file
View File

@@ -0,0 +1,73 @@
package bot
import (
"strings"
"gitlab.com/etke.cc/postmoogle/utils"
)
// account data key
const botsettingskey = "cc.etke.postmoogle.config"
// bot options keys
const (
botOptionUsers = "users"
)
type botsettings map[string]string
// Get option
func (s botsettings) Get(key string) string {
return s[strings.ToLower(strings.TrimSpace(key))]
}
// Set option
func (s botsettings) Set(key, value string) {
s[strings.ToLower(strings.TrimSpace(key))] = value
}
// Users option
func (s botsettings) Users() []string {
return strings.Split(s.Get(botOptionUsers), " ")
}
// TODO: remove after migration
func (b *Bot) migrateBotSettings(users []string) error {
config := b.getBotSettings()
cfgUsers := config.Users()
if len(users) > 0 && len(cfgUsers) == 0 {
_, err := parseMXIDpatterns(users, "")
if err != nil {
return err
}
config.Set(botOptionUsers, strings.Join(users, " "))
return b.setBotSettings(config)
}
return nil
}
func (b *Bot) getBotSettings() botsettings {
cfg := b.botcfg.Get(botsettingskey)
if cfg != nil {
return cfg
}
config := botsettings{}
err := b.lp.GetClient().GetAccountData(botsettingskey, &config)
if err != nil {
if strings.Contains(err.Error(), "M_NOT_FOUND") {
err = nil
}
b.log.Error("cannot get bot settings: %v", utils.UnwrapError(err))
} else {
b.botcfg.Set(botsettingskey, config)
}
return config
}
func (b *Bot) setBotSettings(cfg botsettings) error {
b.botcfg.Set(botsettingskey, cfg)
return utils.UnwrapError(b.lp.GetClient().SetAccountData(botsettingskey, cfg))
}

View File

@@ -9,8 +9,21 @@ import (
"gitlab.com/etke.cc/postmoogle/utils" "gitlab.com/etke.cc/postmoogle/utils"
) )
// settings of a room // account data key
type settings map[string]string const roomsettingskey = "cc.etke.postmoogle.settings"
// option keys
const (
optionOwner = "owner"
optionMailbox = "mailbox"
optionNoSender = "nosender"
optionNoSubject = "nosubject"
optionNoHTML = "nohtml"
optionNoThreads = "nothreads"
optionNoFiles = "nofiles"
)
type roomsettings map[string]string
// settingsOld of a room // settingsOld of a room
type settingsOld struct { type settingsOld struct {
@@ -20,52 +33,47 @@ type settingsOld struct {
} }
// Get option // Get option
func (s settings) Get(key string) string { func (s roomsettings) Get(key string) string {
return s[strings.ToLower(strings.TrimSpace(key))] return s[strings.ToLower(strings.TrimSpace(key))]
} }
func (s settings) Mailbox() string { // Set option
func (s roomsettings) Set(key, value string) {
s[strings.ToLower(strings.TrimSpace(key))] = value
}
func (s roomsettings) Mailbox() string {
return s.Get(optionMailbox) return s.Get(optionMailbox)
} }
func (s settings) Owner() string { func (s roomsettings) Owner() string {
return s.Get(optionOwner) return s.Get(optionOwner)
} }
func (s settings) NoSender() bool { func (s roomsettings) NoSender() bool {
return utils.Bool(s.Get(optionNoSender)) return utils.Bool(s.Get(optionNoSender))
} }
func (s settings) NoSubject() bool { func (s roomsettings) NoSubject() bool {
return utils.Bool(s.Get(optionNoSubject)) return utils.Bool(s.Get(optionNoSubject))
} }
func (s settings) NoHTML() bool { func (s roomsettings) NoHTML() bool {
return utils.Bool(s.Get(optionNoHTML)) return utils.Bool(s.Get(optionNoHTML))
} }
func (s settings) NoThreads() bool { func (s roomsettings) NoThreads() bool {
return utils.Bool(s.Get(optionNoThreads)) return utils.Bool(s.Get(optionNoThreads))
} }
func (s settings) NoFiles() bool { func (s roomsettings) NoFiles() bool {
return utils.Bool(s.Get(optionNoFiles)) return utils.Bool(s.Get(optionNoFiles))
} }
// Users is bot/admin option
func (s settings) Users() []string {
return strings.Split(s.Get(botOptionUsers), " ")
}
// Set option
func (s settings) Set(key, value string) {
s[strings.ToLower(strings.TrimSpace(key))] = value
}
// TODO: remove after migration // TODO: remove after migration
func (b *Bot) migrateSettings(roomID id.RoomID) { func (b *Bot) migrateSettings(roomID id.RoomID) {
var config settingsOld var config settingsOld
err := b.lp.GetClient().GetRoomAccountData(roomID, settingskey, &config) err := b.lp.GetClient().GetRoomAccountData(roomID, roomsettingskey, &config)
if err != nil { if err != nil {
// any error = no need to migrate // any error = no need to migrate
return return
@@ -74,25 +82,25 @@ func (b *Bot) migrateSettings(roomID id.RoomID) {
if config.Mailbox == "" { if config.Mailbox == "" {
return return
} }
cfg := settings{} cfg := roomsettings{}
cfg.Set(optionMailbox, config.Mailbox) cfg.Set(optionMailbox, config.Mailbox)
cfg.Set(optionOwner, config.Owner.String()) cfg.Set(optionOwner, config.Owner.String())
cfg.Set(optionNoSender, strconv.FormatBool(config.NoSender)) cfg.Set(optionNoSender, strconv.FormatBool(config.NoSender))
err = b.setSettings(roomID, cfg) err = b.setRoomSettings(roomID, cfg)
if err != nil { if err != nil {
b.log.Error("cannot migrate settings: %v", err) b.log.Error("cannot migrate settings: %v", err)
} }
} }
func (b *Bot) getSettings(roomID id.RoomID) (settings, error) { func (b *Bot) getRoomSettings(roomID id.RoomID) (roomsettings, error) {
cfg := b.cfg.Get(roomID.String()) cfg := b.cfg.Get(roomID.String())
if cfg != nil { if cfg != nil {
return cfg, nil return cfg, nil
} }
config := settings{} config := roomsettings{}
err := b.lp.GetClient().GetRoomAccountData(roomID, settingskey, &config) err := b.lp.GetClient().GetRoomAccountData(roomID, roomsettingskey, &config)
if err != nil { if err != nil {
if strings.Contains(err.Error(), "M_NOT_FOUND") { if strings.Contains(err.Error(), "M_NOT_FOUND") {
// Suppress `M_NOT_FOUND (HTTP 404): Room account data not found` errors. // Suppress `M_NOT_FOUND (HTTP 404): Room account data not found` errors.
@@ -107,7 +115,7 @@ func (b *Bot) getSettings(roomID id.RoomID) (settings, error) {
return config, utils.UnwrapError(err) return config, utils.UnwrapError(err)
} }
func (b *Bot) setSettings(roomID id.RoomID, cfg settings) error { func (b *Bot) setRoomSettings(roomID id.RoomID, cfg roomsettings) error {
b.cfg.Set(roomID.String(), cfg) b.cfg.Set(roomID.String(), cfg)
return utils.UnwrapError(b.lp.GetClient().SetRoomAccountData(roomID, settingskey, cfg)) return utils.UnwrapError(b.lp.GetClient().SetRoomAccountData(roomID, roomsettingskey, cfg))
} }