1
0
Fork 0
mirror of https://git.sr.ht/~ashkeel/strimertul synced 2024-09-20 02:00:49 +00:00
strimertul/modules/loyalty/manager.go

204 lines
4.8 KiB
Go
Raw Normal View History

2021-05-02 12:29:43 +00:00
package loyalty
import (
"context"
"errors"
"sync"
2021-05-02 12:29:43 +00:00
"github.com/dgraph-io/badger/v3"
jsoniter "github.com/json-iterator/go"
2021-05-07 16:36:23 +00:00
"github.com/sirupsen/logrus"
2021-05-10 23:54:55 +00:00
kv "github.com/strimertul/kilovolt/v3"
"github.com/strimertul/strimertul/database"
2021-05-02 12:29:43 +00:00
)
type Manager struct {
Config Config
Rewards RewardStorage
Goals GoalStorage
RedeemQueue RedeemQueueStorage
points PointStorage
pointsmu sync.Mutex
hub *kv.Hub
logger logrus.FieldLogger
2021-05-02 12:29:43 +00:00
}
func NewManager(db *database.DB, hub *kv.Hub, log logrus.FieldLogger) (*Manager, error) {
if log == nil {
log = logrus.New()
}
2021-05-02 12:29:43 +00:00
manager := &Manager{
logger: log,
hub: hub,
pointsmu: sync.Mutex{},
2021-05-02 12:29:43 +00:00
}
// Ger data from DB
if err := db.GetJSON(ConfigKey, &manager.Config); err != nil {
2021-05-02 12:29:43 +00:00
if err == badger.ErrKeyNotFound {
log.Warn("missing configuration for loyalty (but it's enabled). Please make sure to set it up properly!")
2021-05-02 12:29:43 +00:00
} else {
return nil, err
}
}
if err := db.GetJSON(PointsKey, &manager.points); err != nil {
2021-05-02 12:29:43 +00:00
if err == badger.ErrKeyNotFound {
manager.points = make(PointStorage)
2021-05-02 12:29:43 +00:00
} else {
return nil, err
}
}
if err := db.GetJSON(RewardsKey, &manager.Rewards); err != nil {
2021-05-02 12:29:43 +00:00
if err != badger.ErrKeyNotFound {
return nil, err
}
}
if err := db.GetJSON(GoalsKey, &manager.Goals); err != nil {
2021-05-02 12:29:43 +00:00
if err != badger.ErrKeyNotFound {
return nil, err
}
}
if err := db.GetJSON(QueueKey, &manager.RedeemQueue); err != nil {
2021-05-02 12:29:43 +00:00
if err != badger.ErrKeyNotFound {
return nil, err
}
}
// Subscribe for changes
go func() {
db.Subscribe(context.Background(), manager.update, "loyalty/")
2021-05-02 12:29:43 +00:00
}()
return manager, nil
}
func (m *Manager) update(kvs []database.ModifiedKV) error {
for _, kv := range kvs {
2021-05-02 12:29:43 +00:00
var err error
switch string(kv.Key) {
case ConfigKey:
err = jsoniter.ConfigFastest.Unmarshal(kv.Data, &m.Config)
2021-05-02 12:29:43 +00:00
case PointsKey:
m.pointsmu.Lock()
err = jsoniter.ConfigFastest.Unmarshal(kv.Data, &m.points)
m.pointsmu.Unlock()
2021-05-02 12:29:43 +00:00
case GoalsKey:
err = jsoniter.ConfigFastest.Unmarshal(kv.Data, &m.Goals)
2021-05-02 12:29:43 +00:00
case RewardsKey:
err = jsoniter.ConfigFastest.Unmarshal(kv.Data, &m.Rewards)
2021-05-02 12:29:43 +00:00
case QueueKey:
err = jsoniter.ConfigFastest.Unmarshal(kv.Data, &m.RedeemQueue)
case CreateRedeemRPC:
var redeem Redeem
err = jsoniter.ConfigFastest.Unmarshal(kv.Data, &redeem)
if err == nil {
err = m.AddRedeem(redeem)
}
case RemoveRedeemRPC:
var redeem Redeem
err = jsoniter.ConfigFastest.Unmarshal(kv.Data, &redeem)
if err == nil {
err = m.RemoveRedeem(redeem)
}
2021-05-02 12:29:43 +00:00
}
if err != nil {
m.logger.WithFields(logrus.Fields{
"key": string(kv.Key),
"error": err.Error(),
}).Error("subscribe error: invalid JSON received on key")
2021-05-02 12:29:43 +00:00
} else {
m.logger.WithField("key", string(kv.Key)).Debug("updated key")
2021-05-02 12:29:43 +00:00
}
}
return nil
}
func (m *Manager) SavePoints() error {
m.pointsmu.Lock()
defer m.pointsmu.Unlock()
data, _ := jsoniter.ConfigFastest.Marshal(m.points)
2021-05-02 12:29:43 +00:00
return m.hub.WriteKey(PointsKey, string(data))
}
func (m *Manager) GetPoints(user string) int64 {
m.pointsmu.Lock()
defer m.pointsmu.Unlock()
points, ok := m.points[user]
if ok {
return points
}
return 0
}
func (m *Manager) SetPoints(user string, points int64) {
m.pointsmu.Lock()
defer m.pointsmu.Unlock()
m.points[user] = points
}
2021-05-02 12:29:43 +00:00
func (m *Manager) GivePoints(pointsToGive map[string]int64) error {
// Add points to each user
for user, points := range pointsToGive {
balance := m.GetPoints(user)
m.SetPoints(user, balance+points)
2021-05-02 12:29:43 +00:00
}
// Save points
return m.SavePoints()
}
func (m *Manager) TakePoints(pointsToTake map[string]int64) error {
// Add points to each user
for user, points := range pointsToTake {
balance := m.GetPoints(user)
m.SetPoints(user, balance-points)
2021-05-02 12:29:43 +00:00
}
// Save points
return m.SavePoints()
}
func (m *Manager) SaveQueue() error {
data, _ := jsoniter.ConfigFastest.Marshal(m.RedeemQueue)
return m.hub.WriteKey(QueueKey, string(data))
}
func (m *Manager) AddRedeem(redeem Redeem) error {
// Add to local list
m.RedeemQueue = append(m.RedeemQueue, redeem)
// Send redeem event
data, _ := jsoniter.ConfigFastest.Marshal(redeem)
m.hub.WriteKey(RedeemEvent, string(data))
2021-05-02 12:29:43 +00:00
// Save points
return m.SaveQueue()
}
func (m *Manager) RemoveRedeem(redeem Redeem) error {
for index, queued := range m.RedeemQueue {
if queued.When == redeem.When && queued.Username == redeem.Username && queued.Reward.ID == redeem.Reward.ID {
// Remove redemption from list
m.RedeemQueue = append(m.RedeemQueue[:index], m.RedeemQueue[index+1:]...)
// Save points
return m.SaveQueue()
}
}
return errors.New("redeem not found")
}
func (m *Manager) SaveGoals() error {
data, _ := jsoniter.ConfigFastest.Marshal(m.Goals)
return m.hub.WriteKey(GoalsKey, string(data))
}
func (m *Manager) ContributeGoal(goal *Goal, user string, points int64) error {
goal.Contributed += points
goal.Contributors[user] += points
return m.SaveGoals()
}