// © 2013 the CatBase Authors under the WTFPL. See AUTHORS for the list of authors. package babbler import ( "errors" "fmt" "log" "math/rand" "strings" "github.com/jmoiron/sqlx" "github.com/velour/catbase/bot" "github.com/velour/catbase/bot/msg" "github.com/velour/catbase/config" ) type BabblerPlugin struct { Bot bot.Bot db *sqlx.DB config *config.Config } func New(bot bot.Bot) *BabblerPlugin { log.SetFlags(log.LstdFlags | log.Lshortfile) if bot.DBVersion() == 1 { if _, err := bot.DB().Exec(`create table if not exists babblers ( id integer primary key, babbler string );`); err != nil { log.Fatal(err) } if _, err := bot.DB().Exec(`create table if not exists babblerWords ( id integer primary key, babblerId integer, word string, root integer, rootFrequency integer );`); err != nil { log.Fatal(err) } if _, err := bot.DB().Exec(`create table if not exists babblerArcs ( id integer primary key, fromWordId integer, toWordId interger, frequency integer );`); err != nil { log.Fatal(err) } } plugin := &BabblerPlugin{ Bot: bot, db: bot.DB(), config: bot.Config(), } return plugin } func (p *BabblerPlugin) Message(message msg.Message) bool { lowercase := strings.ToLower(message.Body) tokens := strings.Fields(lowercase) numTokens := len(tokens) if numTokens >= 2 && tokens[1] == "says" { who := tokens[0] _, err := p.getBabbler(who) if err != nil { return false } var saying string if len(tokens) == 2 { saying, _ = p.babble(who) } else { saying, _ = p.babbleSeed(who, tokens[2:]) } if saying != "" { p.Bot.SendMessage(message.Channel, saying) return true } } else if len(tokens) == 4 && strings.Index(lowercase, "initialize babbler for ") == 0 { who := tokens[3] _, err := p.getOrCreateBabbler(who) if err != nil { p.Bot.SendMessage(message.Channel, "babbler initialization failed.") return true } p.Bot.SendMessage(message.Channel, "Okay.") return true } else if strings.Index(lowercase, "batch learn for ") == 0 { who := tokens[3] babblerId, err := p.getOrCreateBabbler(who) if err != nil { p.Bot.SendMessage(message.Channel, "batch learn failed.") return true } body := strings.Join(tokens[4:], " ") body = strings.ToLower(body) for _, a := range strings.Split(body, ".") { for _, b := range strings.Split(a, "!") { for _, c := range strings.Split(b, "?") { for _, d := range strings.Split(c, "\n") { trimmed := strings.TrimSpace(d) if trimmed != "" { p.addToMarkovChain(babblerId, trimmed) } } } } } p.Bot.SendMessage(message.Channel, "phew that was tiring.") return true } else if len(tokens) == 5 && strings.Index(lowercase, "merge babbler") == 0 { if tokens[3] != "into" { p.Bot.SendMessage(message.Channel, "try using 'merge babbler [x] into [y]'") return true } who := tokens[2] into := tokens[4] if who == into { p.Bot.SendMessage(message.Channel, "that's annoying. stop it.") return true } whoBabbler, err := p.getBabbler(who) if err != nil { p.Bot.SendMessage(message.Channel, "merge failed.") return true } intoBabbler, err := p.getOrCreateBabbler(into) if err != nil { p.Bot.SendMessage(message.Channel, "merge failed.") return true } p.merge(intoBabbler, whoBabbler, into, who) p.Bot.SendMessage(message.Channel, "mooooiggged") return true } else { babblerId, err := p.getOrCreateBabbler(message.User.Name) if err == nil { p.addToMarkovChain(babblerId, lowercase) } } return false } func (p *BabblerPlugin) Help(channel string, parts []string) { p.Bot.SendMessage(channel, "initialize babbler for seabass\n\nseabass says") } func (p *BabblerPlugin) Event(kind string, message msg.Message) bool { return false } func (p *BabblerPlugin) BotMessage(message msg.Message) bool { return false } func (p *BabblerPlugin) RegisterWeb() *string { return nil } func (p *BabblerPlugin) makeBabbler(babbler string) (int64, error) { res, err := p.db.Exec(`insert into babblers (babbler) values (?);`, babbler) if err == nil { id, _ := res.LastInsertId() return id, nil } return -1, err } func (p *BabblerPlugin) getBabbler(babbler string) (int64, error) { id := int64(-1) err := p.db.Get(&id, `select id from babblers where babbler = ?`, babbler) if err != nil { } return id, err } func (p *BabblerPlugin) getOrCreateBabbler(babbler string) (int64, error) { id, err := p.getBabbler(babbler) if err != nil { id, err = p.makeBabbler(babbler) if err != nil { return id, err } query := fmt.Sprintf(`select tidbit from factoid where fact like '%s quotes';`, babbler) rows, err := p.db.Query(query) if err != nil { //we'll just ignore this but the actual creation succeeded previously return id, nil } for rows.Next() { var tidbit string err := rows.Scan(&tidbit) if err != nil { return id, err } p.addToMarkovChain(id, tidbit) } } return id, err } func (p *BabblerPlugin) getWordId(babblerId int64, word string) (int64, error) { id := int64(-1) err := p.db.Get(&id, `select id from babblerWords where babblerId = ? and word = ?`, babblerId, word) return id, err } func (p *BabblerPlugin) createNewWord(babblerId int64, word string) (int64, error) { res, err := p.db.Exec(`insert into babblerWords (babblerId, word, root, rootFrequency) values (?, ?, 0, 0);`, babblerId, word) if err != nil { return -1, err } id, _ := res.LastInsertId() return id, nil } func (p *BabblerPlugin) getOrCreateWord(babblerId int64, word string) (int64, error) { id, err := p.getWordId(babblerId, word) if err != nil { return p.createNewWord(babblerId, word) } return id, err } func (p *BabblerPlugin) incrementRootWordFrequency(babblerId int64, word string) (int64, error) { id, err := p.getOrCreateWord(babblerId, word) if err != nil { return -1, err } _, err = p.db.Exec(`update babblerWords set rootFrequency = rootFrequency + 1, root = 1 where id = ?;`, id) if err != nil { return -1, err } return id, nil } func (p *BabblerPlugin) getWordArcHelper(fromWordId, toWordId int64) (int64, error) { id := int64(-1) err := p.db.Get(&id, `select id from babblerArcs where fromWordId = ? and toWordId = ?`, fromWordId, toWordId) return id, err } func (p *BabblerPlugin) incrementWordArc(fromWordId, toWordId int64) (int64, error) { res, err := p.db.Exec(`update babblerArcs set frequency = frequency + 1 where fromWordId = ? and toWordId = ?`, fromWordId, toWordId) affectedRows := int64(0) if err == nil { affectedRows, _ = res.RowsAffected() } if err != nil || affectedRows == 0 { res, err = p.db.Exec(`insert into babblerArcs (fromWordId, toWordId, frequency) values (?, ?, 1);`, fromWordId, toWordId) if err != nil { return -1, err } } id, _ := res.LastInsertId() return id, nil } func (p *BabblerPlugin) incrementFinalWordArcHelper(wordId int64) (int64, error) { return p.incrementWordArc(wordId, -1) } func (p *BabblerPlugin) incrementWordArcHelper(babblerId, fromWordId int64, toWord string) (int64, error) { toWordId, err := p.getOrCreateWord(babblerId, toWord) if err != nil { return -1, err } _, err = p.incrementWordArc(fromWordId, toWordId) if err != nil { return -1, err } return toWordId, nil } func (p *BabblerPlugin) addToMarkovChain(babblerId int64, phrase string) { words := strings.Fields(strings.ToLower(phrase)) id, err := p.incrementRootWordFrequency(babblerId, words[0]) if err != nil { return } for i := 1; i < len(words); i++ { id, err = p.incrementWordArcHelper(babblerId, id, words[i]) if err != nil { return } } _, err = p.incrementFinalWordArcHelper(id) } func (p *BabblerPlugin) getWeightedRootWord(babblerId int64) (int64, string, error) { query := fmt.Sprintf("select id, word, rootFrequency from babblerWords where babblerId = %d and root = 1", babblerId) rows, err := p.db.Query(query) if err != nil { return -1, "", err } defer rows.Close() idToWord := map[int64]string{} idToFreq := map[int64]int64{} total := int64(0) for rows.Next() { var id int64 var word string var rootFrequency int64 err = rows.Scan(&id, &word, &rootFrequency) if err != nil { return -1, "", err } total += rootFrequency idToFreq[id] = rootFrequency idToWord[id] = word } which := rand.Int63n(total) total = 0 for id, freq := range idToFreq { if total+freq >= which { return id, idToWord[id], nil } total += freq } log.Fatalf("shouldn't happen") return -1, "", errors.New("failed to find weighted root word") } func (p *BabblerPlugin) getWeightedNextWord(fromWordId int64) (int64, string, error) { query := fmt.Sprintf("select toWordId, frequency from babblerArcs where fromWordId = %d;", fromWordId) rows, err := p.db.Query(query) if err != nil { return -1, "", err } defer rows.Close() idToFreq := map[int64]int64{} total := int64(0) for rows.Next() { var toWordId int64 var frequency int64 err = rows.Scan(&toWordId, &frequency) if err != nil { return -1, "", err } total += frequency idToFreq[toWordId] = frequency } which := rand.Int63n(total) total = 0 for id, freq := range idToFreq { if total+freq >= which { if id < 0 { return -1, "", nil } var word string err := p.db.Get(&word, `select word from babblerWords where id = ?`, id) if err != nil { return -1, "", err } return id, word, nil } total +=freq } log.Fatalf("shouldn't happen") return -1, "", errors.New("failed to find weighted next word") } func (p *BabblerPlugin) babble(who string) (string, error) { return p.babbleSeed(who, []string{}) } func (p *BabblerPlugin) babbleSeed(babbler string, seed []string) (string, error) { babblerId, err := p.getBabbler(babbler) if err != nil { return "", nil } words := seed var curWordId int64 if len(seed) == 0 { id, word, err := p.getWeightedRootWord(babblerId) if err != nil { return "", err } curWordId = id words = append(words, word) } else { id, err := p.getWordId(babblerId, seed[0]) if err != nil { return "", err } curWordId = id for i := 1; i < len(seed); i++ { nextWordId, err := p.getWordId(babblerId, seed[i]) if err != nil { return "", err } _, err = p.getWordArcHelper(curWordId, nextWordId) if err != nil { return "", err } curWordId = nextWordId } } for { id, word, err := p.getWeightedNextWord(curWordId) if err != nil { return "", err } if id < 0 { break } words = append(words, word) curWordId = id } return strings.TrimSpace(strings.Join(words, " ")), nil } func (p *BabblerPlugin) merge(intoId, otherId int64, intoName, otherName string) error { intoString := "<" + intoName + ">" otherString := "<" + otherName + ">" mapping := map[int64]int64{} query := fmt.Sprintf("select id, word, root, rootFrequency from babblerWords where babblerId = %d;", otherId) rows, err := p.db.Query(query) if err != nil { return err } defer rows.Close() type Word struct { Id int64 Word string Root int64 RootFrequency int64 } words := []Word{} for rows.Next() { word := Word{} err = rows.Scan(&word.Id, &word.Word, &word.Root, &word.RootFrequency) if err != nil { return err } words = append(words, word) } for _, word := range words { if word.Word == otherString { word.Word = intoString } doInsert := false wordId := int64(-1) if word.Root > 0 { res, err := p.db.Exec(`update babblerWords set rootFrequency = rootFrequency + ?, root = 1 where babblerId = ? and word = ? output id ;`, word.RootFrequency, intoId, word.Word) rowsAffected := int64(0) if err == nil { rowsAffected, _ = res.RowsAffected() } if err != nil || rowsAffected == 0 { doInsert = true } else { wordId, _ = res.LastInsertId() } } else { res, err := p.db.Exec(`update babblerWords set rootFrequency = rootFrequency + ? where babblerId = ? and word = ? output id;`, word.RootFrequency, intoId, word.Word) if err != nil { doInsert = true } else { wordId, _ = res.LastInsertId() } } if doInsert { res, err := p.db.Exec(`insert into babblerWords (babblerId, word, root, rootFrequency) values (?,?,?,?) ;`, intoId, word.Word, word.Root, word.RootFrequency) if err != nil { return err } wordId, _ = res.LastInsertId() } log.Printf("%s %d -> %d\n", word.Word, word.Id, wordId) mapping[word.Id] = wordId } type Arc struct { ToWordId int64 Frequency int64 } for lookup, newArcStart := range mapping { query = fmt.Sprintf("select toWordId, frequency from babblerArcs where fromWordId = %d;", lookup) rows, err := p.db.Query(query) if err != nil { return err } defer rows.Close() arcs := []Arc{} for rows.Next() { var arc Arc err = rows.Scan(&arc.ToWordId, &arc.Frequency) if err != nil { return err } arcs = append(arcs, arc) } for _, arc := range arcs { newArcEnd := int64(-1) //handle end arcs if arc.ToWordId >= 0 { newArcEnd = mapping[arc.ToWordId] } res, err := p.db.Exec(`update babblerArcs set frequency = frequency + ? where fromWordId = ? and toWordId = ?`, arc.Frequency, newArcStart, newArcEnd) rowsAffected := int64(0) if err == nil { rowsAffected, _ = res.RowsAffected() } if err != nil || rowsAffected == 0 { _, err = p.db.Exec(`insert into babblerArcs (fromWordId, toWordId, frequency) values (?, ?, ?);`, newArcStart, newArcEnd, arc.Frequency) if err != nil { return err } } } } return nil }