catbase/plugins/achievements/achievements.go

287 lines
7.9 KiB
Go
Raw Normal View History

2020-04-20 09:56:49 +00:00
package achievements
import (
"fmt"
2020-04-24 20:24:29 +00:00
"regexp"
"strings"
"time"
2020-04-20 09:56:49 +00:00
"github.com/jmoiron/sqlx"
2020-04-24 20:24:29 +00:00
"github.com/rs/zerolog/log"
2020-04-20 09:56:49 +00:00
"github.com/velour/catbase/bot"
"github.com/velour/catbase/bot/msg"
"github.com/velour/catbase/config"
)
// A plugin to track our misdeeds
type AchievementsPlugin struct {
bot bot.Bot
cfg *config.Config
db *sqlx.DB
}
func New(b bot.Bot) *AchievementsPlugin {
2020-04-24 20:24:29 +00:00
ap := &AchievementsPlugin{
bot: b,
cfg: b.Config(),
db: b.DB(),
}
err := ap.mkDB()
if err != nil {
log.Fatal().Err(err).Msg("unable to create achievements tables")
2020-04-20 09:56:49 +00:00
}
2021-02-01 03:14:55 +00:00
2021-06-17 20:21:30 +00:00
ap.register()
2021-02-01 03:14:55 +00:00
2020-04-24 20:24:29 +00:00
return ap
2020-04-20 09:56:49 +00:00
}
2021-06-17 20:21:30 +00:00
var grantRegex = regexp.MustCompile(`(?i)grant (?P<emojy>(?::[[:word:][:punct:]]+:\s?)+) to :?(?P<nick>[[:word:]]+):?`)
var createRegex = regexp.MustCompile(`(?i)create trophy (?P<emojy>(?::[[:word:][:punct:]]+:\s?)+) (?P<description>.+)`)
var greatRegex = regexp.MustCompile(`(?i)how great (?:am i|is :?(?P<who>[[:word:]]+))[[:punct:]]*`)
var listRegex = regexp.MustCompile(`(?i)^list (?P<whos>.+)?\s?trophies$`)
func (p *AchievementsPlugin) register() {
p.bot.RegisterRegexCmd(p, bot.Message, grantRegex, p.grantCmd)
p.bot.RegisterRegexCmd(p, bot.Message, createRegex, p.createCmd)
p.bot.RegisterRegexCmd(p, bot.Message, greatRegex, p.greatCmd)
p.bot.RegisterRegexCmd(p, bot.Message, listRegex, p.listCmd)
p.bot.Register(p, bot.Help, p.help)
}
2020-04-20 09:56:49 +00:00
func (p *AchievementsPlugin) mkDB() error {
2020-04-24 20:24:29 +00:00
trophiesTable := `create table if not exists trophies (
emojy string primary key,
description string,
creator string
);`
awardsTable := `create table if not exists awards (
id integer primary key,
2020-04-24 20:24:29 +00:00
emojy string references trophies(emojy) on delete restrict on update cascade,
holder string,
amount integer default 0,
granted timestamp CURRENT_TIMESTAMP
);`
2020-04-20 09:56:49 +00:00
tx, err := p.db.Beginx()
if err != nil {
return err
}
2020-04-24 20:24:29 +00:00
if _, err = tx.Exec(trophiesTable); err != nil {
return err
}
if _, err = tx.Exec(awardsTable); err != nil {
2020-04-20 09:56:49 +00:00
return err
}
err = tx.Commit()
if err != nil {
return err
}
return nil
}
2020-04-24 20:24:29 +00:00
func (p *AchievementsPlugin) GetAwards(nick string) []Award {
var awards []Award
q := `select * from awards inner join trophies on awards.emojy=trophies.emojy where holder=?`
if err := p.db.Select(&awards, q, nick); err != nil {
log.Error().Err(err).Msg("could not select awards")
}
return awards
}
2021-06-17 20:21:30 +00:00
func (p *AchievementsPlugin) listCmd(r bot.Request) bool {
log.Debug().Msgf("Values: %+v", r.Values)
whos := strings.TrimSpace(r.Values["whos"])
if whos == "my" {
whos = r.Msg.User.Name
}
log.Debug().Msgf("whos = %s", whos)
ts, err := p.AllTrophies()
if err != nil {
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, "Some problem happened.")
log.Error().Err(err).Msg("could not retrieve trophies")
return true
}
if len(ts) == 0 {
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, "There are no trophies.")
return true
}
msg := "Current trophies:\n"
log.Debug().Msgf("Trophies: %s", ts)
for _, t := range ts {
log.Debug().Msgf("t.Creator = %s", t.Creator)
if strings.TrimSpace(t.Creator) == whos {
log.Debug().Msgf("adding %s", t)
msg += fmt.Sprintf("%s - %s\n", t.Emojy, t.Description)
} else if whos == "" {
msg += fmt.Sprintf("%s - %s (by %s)\n", t.Emojy, t.Description, t.Creator)
}
}
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, strings.TrimSpace(msg))
return true
}
2020-04-24 20:24:29 +00:00
2021-02-01 03:14:55 +00:00
func (p *AchievementsPlugin) grantCmd(r bot.Request) bool {
nick := r.Msg.User.Name
emojy := r.Values["emojy"]
receiver := r.Values["nick"]
trophy, err := p.FindTrophy(emojy)
if err != nil {
log.Error().Err(err).Msg("could not find trophy")
msg := fmt.Sprintf("The %s award doesn't exist.", emojy)
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, msg)
2020-04-24 20:24:29 +00:00
return true
}
2021-02-01 03:14:55 +00:00
if nick == trophy.Creator {
a, err := p.Grant(receiver, emojy)
2020-04-24 20:24:29 +00:00
if err != nil {
2021-02-01 03:14:55 +00:00
log.Error().Err(err).Msg("could not award trophy")
2020-04-24 20:24:29 +00:00
}
2021-02-01 03:14:55 +00:00
msg := fmt.Sprintf("Congrats %s. You just got the %s award for %s.",
receiver, emojy, a.Description)
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, msg)
} else {
msg := fmt.Sprintf("Sorry, %s. %s owns that trophy.", nick, trophy.Creator)
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, msg)
2020-04-24 20:24:29 +00:00
}
2021-02-01 03:14:55 +00:00
return true
}
func (p *AchievementsPlugin) createCmd(r bot.Request) bool {
nick := r.Msg.User.Name
emojy := r.Values["emojy"]
description := r.Values["description"]
t, err := p.Create(emojy, description, nick)
if err != nil {
log.Error().Err(err).Msg("could not create trophy")
if strings.Contains(err.Error(), "exists") {
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, err.Error())
2020-04-24 20:24:29 +00:00
return true
}
2021-02-01 03:14:55 +00:00
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, "I'm too humble to ever award that trophy")
2020-04-24 20:24:29 +00:00
return true
}
2021-02-01 03:14:55 +00:00
resp := fmt.Sprintf("Okay %s. I have crafted a one-of-a-kind %s trophy to give for %s",
nick, t.Emojy, t.Description)
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, resp)
return true
}
func (p *AchievementsPlugin) greatCmd(r bot.Request) bool {
nick := r.Msg.User.Name
who := r.Values["who"]
if who == "" {
who = nick
}
awards := p.GetAwards(who)
if len(awards) == 0 {
m := fmt.Sprintf("%s has no achievements to their name. They really suck.", who)
if who == nick {
m = fmt.Sprintf("You have no achievements to your name. "+
"You are a sad and terrible specimen of the human condition, %s.", who)
}
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, m)
} else {
m := fmt.Sprintf("Wow, let's all clap for %s. Look at these awards:", who)
for _, a := range awards {
m += fmt.Sprintf("\n%s - %s", a.Emojy, a.Description)
}
p.bot.Send(r.Conn, bot.Message, r.Msg.Channel, m)
}
return true
2020-04-20 09:56:49 +00:00
}
func (p *AchievementsPlugin) help(c bot.Connector, kind bot.Kind, message msg.Message, args ...any) bool {
2020-04-20 09:56:49 +00:00
ch := message.Channel
me := p.bot.WhoAmI()
2020-04-24 20:24:29 +00:00
msg := "The achievements plugins awards trophies."
msg += fmt.Sprintf("\nYou can create a trophy with `%s, create trophy <emojy> <description>`", me)
msg += fmt.Sprintf("\nYou can award a trophy with `%s, grant <emojy> to <nick>`", me)
msg += "\nYou can see others awards with `How great is <nick>`"
msg += "\nYou can see your awards with `How great am I?`"
2020-04-20 09:56:49 +00:00
p.bot.Send(c, bot.Message, ch, msg)
return true
}
// Award is used by other plugins to register a particular award for a user
2020-04-24 20:24:29 +00:00
func (p *AchievementsPlugin) Grant(nick, emojy string) (Award, error) {
empty := Award{}
q := `insert into awards (emojy,holder) values (?, ?)`
tx, err := p.db.Beginx()
if err != nil {
return empty, err
}
if _, err := tx.Exec(q, emojy, nick); err != nil {
tx.Rollback()
return empty, err
}
if err := tx.Commit(); err != nil {
return empty, err
}
return p.FindAward(emojy)
}
func (p *AchievementsPlugin) Create(emojy, description, creator string) (Trophy, error) {
t, err := p.FindTrophy(emojy)
if err == nil {
return t, fmt.Errorf("the trophy %s already exists", emojy)
}
q := `insert into trophies (emojy,description,creator) values (?,?,?)`
tx, err := p.db.Beginx()
if err != nil {
return Trophy{}, err
}
_, err = tx.Exec(q, emojy, description, creator)
if err != nil {
tx.Rollback()
return Trophy{}, err
}
err = tx.Commit()
return Trophy{
Emojy: emojy,
Description: description,
Creator: creator,
}, err
}
type Trophy struct {
Emojy string
Description string
Creator string
}
type Award struct {
2020-04-24 20:24:29 +00:00
Trophy
ID int64
Holder string
Amount int
Granted *time.Time
}
func (a *Award) Save() error {
2020-04-20 09:56:49 +00:00
return nil
}
2020-04-24 20:24:29 +00:00
2021-06-17 20:21:30 +00:00
func (p *AchievementsPlugin) AllTrophies() ([]Trophy, error) {
q := `select * from trophies order by creator`
var t []Trophy
err := p.db.Select(&t, q)
return t, err
}
2020-04-24 20:24:29 +00:00
func (p *AchievementsPlugin) FindTrophy(emojy string) (Trophy, error) {
q := `select * from trophies where emojy=?`
var t Trophy
err := p.db.Get(&t, q, emojy)
return t, err
}
func (p *AchievementsPlugin) FindAward(emojy string) (Award, error) {
q := `select * from awards inner join trophies on awards.emojy=trophies.emojy where trophies.emojy=?`
var a Award
err := p.db.Get(&a, q, emojy)
return a, err
}