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
|
package user
import (
"context"
"database/sql"
"errors"
"time"
"git.anhgelus.world/anhgelus/les-copaings-bot/common"
)
type Copaing struct {
ID uint64
CopaingXPs []CopaingXP
GuildID uint64
lastSaved int
}
type CopaingXP struct {
XP uint
CopaingID uint64
GuildID uint64
CreatedAt time.Time
}
type CopaingAccess interface {
Copaing() *Copaing
GetXP() uint
}
func GetCopaing(ctx context.Context, discordID, guildID uint64) *CopaingCached {
state := GetState(ctx)
cc, err := state.Copaing(guildID, discordID)
if err != nil {
c := Copaing{ID: discordID, GuildID: guildID}
if err := c.load(ctx); err != nil {
panic(err)
}
cc = FromCopaing(&c)
}
return cc
}
func (c *Copaing) load(ctx context.Context) error {
db := common.GetDB(ctx)
row := db.QueryRowContext(
ctx,
`SELECT id, guild_id FROM copaings WHERE id = ? AND guild_id = ?`,
c.ID, c.GuildID,
)
c.CopaingXPs = make([]CopaingXP, 0)
c.lastSaved = 0
err := row.Err()
if err != nil {
if !errors.Is(err, sql.ErrNoRows) {
return err
}
_, err = db.ExecContext(
ctx,
`INSERT INTO copaings (id, guild_id) VALUES (?, ?)`,
c.ID, c.GuildID,
)
return err
}
rows, err := db.QueryContext(
ctx,
`SELECT xp, created_at FROM copaing_xps WHERE copaing_id = ? AND guild_id = ?`,
c.ID, c.GuildID,
)
defer rows.Close()
if err != nil {
if !errors.Is(err, sql.ErrNoRows) {
return err
}
return nil
}
for rows.Next() {
var xp CopaingXP
err = rows.Scan(&xp.XP, &xp.CreatedAt)
if err != nil {
return err
}
xp.CopaingID = c.ID
xp.GuildID = c.GuildID
c.CopaingXPs = append(c.CopaingXPs, xp)
}
c.lastSaved = len(c.CopaingXPs)
return nil
}
func (c *Copaing) Save(ctx context.Context) error {
if c.lastSaved == len(c.CopaingXPs) {
return nil
}
db := common.GetDB(ctx)
for _, xp := range c.CopaingXPs[c.lastSaved:] {
_, err := db.ExecContext(
ctx,
`INSERT INTO copaing_xps (copaing_id, guild_id, xp) VALUES (?, ?, ?)`,
c.ID, c.GuildID, xp.XP,
)
if err != nil {
return err
}
c.lastSaved++
}
return nil
}
func (c *Copaing) Delete(ctx context.Context) error {
_, err := common.GetDB(ctx).ExecContext(
ctx,
`DELETE FROM copaings WHERE copaing_id = ? AND guild_id = ?`,
c.ID, c.GuildID,
)
c.lastSaved = 0
c.CopaingXPs = nil
return err
}
|