aboutsummaryrefslogtreecommitdiff
path: root/user/xp.go
blob: 246b097de662637a8a1c47d06eb525998c9747a9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
package user

import (
	"context"
	"log/slog"
	"slices"
	"sync"

	"git.anhgelus.world/anhgelus/les-copaings-bot/config"
	"git.anhgelus.world/anhgelus/les-copaings-bot/exp"
	"github.com/anhgelus/gokord"
	"github.com/nyttikord/gokord/bot"
	"github.com/nyttikord/gokord/user"
)

type cXP struct {
	Cxp     uint
	copaing *Copaing
}

func (c *cXP) Copaing() *Copaing {
	return c.copaing
}

func (c *cXP) GetXP() uint {
	return c.Cxp
}

func (cc *CopaingCached) AddXP(ctx context.Context, s bot.Session, m *user.Member, xp uint, fn func(uint, uint)) {
	old := cc.XPs
	pastLevel := exp.Level(old)
	s.Logger().Debug("adding xp", "user", m.DisplayName(), "old", old, "to add", xp)
	cc.XPs += xp
	cc.XPToAdd += xp
	if err := cc.Save(ctx); err != nil {
		s.Logger().Error("saving user in state", "error", err, "user", m.DisplayName(), "xp", xp, "guild", cc.GuildID)
		return
	}
	newLevel := exp.Level(old + xp)
	if newLevel > pastLevel {
		fn(old+xp, newLevel)
		onNewLevel(s, m, newLevel)
	}
}

func (c *Copaing) GetXP(logger *slog.Logger) (uint, error) {
	cfg := config.GetGuildConfig(c.GuildID)
	return c.GetXPForDays(logger, cfg.DaysXPRemains)
}

func (c *Copaing) GetXPForDays(logger *slog.Logger, n uint) (uint, error) {
	xp := uint(0)
	rows, err := gokord.DB.
		Model(&CopaingXP{}).
		Where(
			"created_at >= ? and guild_id = ? and copaing_id = ?",
			exp.TimeStampNDaysBefore(n),
			c.GuildID,
			c.ID,
		).
		Rows()
	if err != nil {
		return 0, err
	}
	defer rows.Close()
	for rows.Next() {
		var cxp CopaingXP
		err = gokord.DB.ScanRows(rows, &cxp)
		if err != nil {
			logger.Error("scanning rows", "error", err, "copaing", c.ID, "guild", c.GuildID)
			continue
		}
		xp += cxp.XP
	}
	return xp, nil
}

// GetBestXP returns n Copaing with the best XP within d days (d <= cfg.DaysXPRemain; d < 0 <=> d = cfg.DaysXPRemain)
//
// This function is slow
func GetBestXP(logger *slog.Logger, guildId string, n uint, d int) ([]CopaingAccess, error) {
	if d < 0 {
		cfg := config.GetGuildConfig(guildId)
		d = int(cfg.DaysXPRemains)
	}
	rows, err := gokord.DB.Model(&Copaing{}).Where("guild_id = ?", guildId).Rows()
	if err != nil {
		return nil, err
	}
	defer rows.Close()
	var l []*cXP
	var wg sync.WaitGroup
	for rows.Next() {
		var c Copaing
		err = gokord.DB.ScanRows(rows, &c)
		if err != nil {
			logger.Error("scanning rows", "error", err, "copaing", c.ID, "guild", c.GuildID)
			continue
		}
		wg.Add(1)
		go func() {
			defer wg.Done()
			xp, err := c.GetXPForDays(logger, uint(d))
			if err != nil {
				logger.Error("fetching xp", "error", err, "copaing", c.ID, "guild", c.GuildID)
				return
			}
			l = append(l, &cXP{Cxp: xp, copaing: &c})
		}()
	}
	wg.Wait()
	slices.SortFunc(l, func(a, b *cXP) int {
		// desc order
		return int(b.Cxp) - int(a.Cxp)
	})
	m := min(len(l), int(n))
	cs := make([]CopaingAccess, m)
	for i, c := range l[:m] {
		cs[i] = c
	}
	return cs, nil
}