catbase/plugins/beers.go

197 lines
4.9 KiB
Go

package plugins
import (
"bitbucket.org/phlyingpenguin/godeepintir/bot"
"fmt"
"labix.org/v2/mgo"
"labix.org/v2/mgo/bson"
"strconv"
"strings"
"time"
"math/rand"
)
// This is a skeleton plugin to serve as an example and quick copy/paste for new plugins.
type BeersPlugin struct {
Bot *bot.Bot
Coll *mgo.Collection
}
// NewBeersPlugin creates a new BeersPlugin with the Plugin interface
func NewBeersPlugin(bot *bot.Bot) *BeersPlugin {
p := BeersPlugin{
Bot: bot,
}
p.LoadData()
return &p
}
type userBeers struct {
Nick string
Beercount int
Lastdrunk time.Time
Momentum float64
New bool
}
func (u *userBeers) Save(coll *mgo.Collection) {
_, err := coll.Upsert(bson.M{"nick": u.Nick}, u)
if err != nil {
panic(err)
}
}
func getUserBeers(coll *mgo.Collection, nick string) *userBeers {
ub := userBeers{New: true}
coll.Find(bson.M{"nick": nick}).One(&ub)
if ub.New == true {
ub.New = false
ub.Nick = nick
ub.Beercount = 0
ub.Momentum = 0
ub.Save(coll)
}
return &ub
}
// Message responds to the bot hook on recieving messages.
// This function returns true if the plugin responds in a meaningful way to the users message.
// Otherwise, the function returns false and the bot continues execution of other plugins.
func (p *BeersPlugin) Message(message bot.Message) bool {
parts := strings.Fields(message.Body)
if len(parts) == 0 {
return false
}
channel := message.Channel
user := message.User
nick := user.Name
// respond to the beers type of queries
if parts[0] == "beers" {
if len(parts) == 3 {
// try to get a count out of parts[2]
count, err := strconv.Atoi(parts[2])
if err != nil {
// if it's not a number, maybe it's a nick!
p.Bot.SendMessage(channel, "Sorry, that didn't make any sense.")
}
if count < 0 {
// you can't be negative
msg := fmt.Sprintf("Sorry %s, you can't have negative beers!", nick)
p.Bot.SendMessage(channel, msg)
}
if parts[1] == "+=" {
p.setBeers(user, p.getBeers(nick)+count)
p.randomReply(channel)
} else if parts[1] == "=" {
if count == 0 {
p.puke(user, channel)
} else {
p.setBeers(user, count)
p.randomReply(channel)
}
} else {
p.Bot.SendMessage(channel, "I don't know your math.")
}
} else if len(parts) == 2 {
if p.doIKnow(parts[1]) {
p.reportCount(parts[1], channel, false)
} else {
msg := fmt.Sprintf("Sorry, I don't know %s.", parts[1])
p.Bot.SendMessage(channel, msg)
}
} else if len(parts) == 1 {
p.reportCount(nick, channel, true)
}
// no matter what, if we're in here, then we've responded
return true
} else if parts[0] == "beers++" {
p.addBeers(user)
p.randomReply(channel)
return true
} else if parts[0] == "puke" {
p.puke(user, channel)
return true
}
if message.Command && parts[0] == "imbibe" {
p.addBeers(user)
p.randomReply(channel)
return true
}
return false
}
// LoadData imports any configuration data into the plugin. This is not strictly necessary other
// than the fact that the Plugin interface demands it exist. This may be deprecated at a later
// date.
func (p *BeersPlugin) LoadData() {
p.Coll = p.Bot.Db.C("beers")
rand.Seed(time.Now().Unix())
}
// Help responds to help requests. Every plugin must implement a help function.
func (p *BeersPlugin) Help(channel string, parts []string) {
msg := "Beers: imbibe by using either beers +=,=,++ or with the !imbibe/drink " +
"commands. I'll keep a count of how many beers you've had and then if you want " +
"to reset, just !puke it all up!"
p.Bot.SendMessage(channel, msg)
}
func (p *BeersPlugin) setBeers(user *bot.User, amount int) {
ub := getUserBeers(p.Coll, user.Name)
ub.Beercount = amount
ub.Lastdrunk = time.Now()
ub.Save(p.Coll)
}
func (p *BeersPlugin) addBeers(user *bot.User) {
p.setBeers(user, p.getBeers(user.Name)+1)
}
func (p *BeersPlugin) getBeers(nick string) int {
ub := getUserBeers(p.Coll, nick)
return ub.Beercount
}
func (p *BeersPlugin) reportCount(nick, channel string, himself bool) {
beers := p.getBeers(nick)
msg := fmt.Sprintf("%s has had %d beers so far.", nick, beers)
if himself {
if beers == 0 {
msg = fmt.Sprintf("You really need to get drinkin, %s!", nick)
} else {
msg = fmt.Sprintf("You've had %d beers so far, %s.", beers, nick)
}
}
p.Bot.SendMessage(channel, msg)
fmt.Println("I should have reported a beer count!")
}
func (p *BeersPlugin) puke(user *bot.User, channel string) {
p.setBeers(user, 0)
msg := fmt.Sprintf("Ohhhhhh, and a reversal of fortune for %s!", user.Name)
p.Bot.SendMessage(channel, msg)
}
func (p *BeersPlugin) doIKnow(nick string) bool {
count, err := p.Coll.Find(bson.M{"nick": nick}).Count()
if err != nil {
panic(err)
}
return count > 0
}
// Sends random affirmation to the channel. This could be better (with a datastore for sayings)
func (p *BeersPlugin) randomReply(channel string) {
replies := []string{"ZIGGY! ZAGGY!", "HIC!", "Stay thirsty, my friend!"}
p.Bot.SendMessage(channel, replies[rand.Intn(len(replies))])
}