2016-01-17 13:00:44 -05:00
// © 2013 the CatBase Authors under the WTFPL. See AUTHORS for the list of authors.
2013-12-10 18:37:07 -05:00
2012-08-17 16:38:15 -04:00
package bot
import (
2016-01-15 01:12:26 -05:00
"database/sql"
2012-10-11 16:28:00 -04:00
"errors"
2012-08-17 16:38:15 -04:00
"fmt"
2016-01-15 01:12:26 -05:00
"log"
2012-08-26 15:14:09 -04:00
"math/rand"
2013-03-04 10:43:23 -05:00
"regexp"
2012-08-26 15:14:09 -04:00
"strconv"
2012-08-17 18:09:29 -04:00
"strings"
2012-08-26 15:14:09 -04:00
"time"
2016-04-01 10:20:03 -04:00
"github.com/velour/catbase/bot/msg"
2012-08-17 16:38:15 -04:00
)
2016-01-15 08:59:26 -05:00
// Handles incomming PRIVMSG requests
2016-04-01 10:20:03 -04:00
func ( b * bot ) MsgReceived ( msg msg . Message ) {
2016-03-10 21:11:52 -05:00
log . Println ( "Received message: " , msg )
2016-01-15 08:59:26 -05:00
2016-03-10 13:37:07 -05:00
// msg := b.buildMessage(client, inMsg)
// do need to look up user and fix it
2016-01-15 08:59:26 -05:00
2016-03-21 18:05:51 -04:00
if strings . HasPrefix ( msg . Body , "help " ) && msg . Command {
2016-01-15 08:59:26 -05:00
parts := strings . Fields ( strings . ToLower ( msg . Body ) )
b . checkHelp ( msg . Channel , parts )
goto RET
}
2017-06-28 09:18:41 -04:00
go ( func ( ) {
2017-06-28 10:51:48 -04:00
done := make ( chan bool , 1 )
go ( func ( ) {
for _ , name := range b . pluginOrdering {
p := b . plugins [ name ]
if p . Message ( msg ) {
break
}
2017-06-28 09:18:41 -04:00
}
2017-06-28 10:51:48 -04:00
done <- true
} ) ( )
select {
case <- done :
break
case <- time . After ( 5 * time . Second ) :
failMsg := fmt . Sprintf ( "I thought about it for a really long time, but ultimately I couldn't really comprehend the magnitude of this message...\n```%+v```\nI'm going to take a forever nap now. I'm so sorry that you won't know what great things could have come from that message. It was a really good message.." , msg )
b . SendMessage ( msg . Channel , failMsg )
log . Fatal ( failMsg )
2016-01-15 08:59:26 -05:00
}
2017-06-28 09:18:41 -04:00
} ) ( )
2016-01-15 08:59:26 -05:00
RET :
b . logIn <- msg
return
}
// Handle incoming events
2016-04-01 10:20:03 -04:00
func ( b * bot ) EventReceived ( msg msg . Message ) {
2016-03-10 21:11:52 -05:00
log . Println ( "Received event: " , msg )
2016-03-10 13:37:07 -05:00
//msg := b.buildMessage(conn, inMsg)
2016-03-30 10:00:20 -04:00
for _ , name := range b . pluginOrdering {
p := b . plugins [ name ]
2016-03-10 13:37:07 -05:00
if p . Event ( msg . Body , msg ) { // TODO: could get rid of msg.Body
2016-01-15 08:59:26 -05:00
break
}
}
}
2016-03-30 10:00:20 -04:00
func ( b * bot ) SendMessage ( channel , message string ) {
b . conn . SendMessage ( channel , message )
2016-03-10 13:37:07 -05:00
}
2016-03-30 10:00:20 -04:00
func ( b * bot ) SendAction ( channel , message string ) {
b . conn . SendAction ( channel , message )
2012-08-17 16:38:15 -04:00
}
2017-06-17 16:20:06 -04:00
func ( b * bot ) React ( channel , reaction string , message msg . Message ) {
b . conn . React ( channel , reaction , message )
}
2012-08-17 21:39:26 -04:00
// Checks to see if the user is asking for help, returns true if so and handles the situation.
2016-03-30 10:00:20 -04:00
func ( b * bot ) checkHelp ( channel string , parts [ ] string ) {
2012-08-17 22:30:04 -04:00
if len ( parts ) == 1 {
2012-08-17 21:39:26 -04:00
// just print out a list of help topics
2012-08-26 20:30:23 -04:00
topics := "Help topics: about variables"
2016-03-30 10:00:20 -04:00
for name , _ := range b . plugins {
2012-08-17 21:39:26 -04:00
topics = fmt . Sprintf ( "%s, %s" , topics , name )
}
b . SendMessage ( channel , topics )
} else {
// trigger the proper plugin's help response
if parts [ 1 ] == "about" {
b . Help ( channel , parts )
return
}
2012-08-26 20:30:23 -04:00
if parts [ 1 ] == "variables" {
b . listVars ( channel , parts )
return
}
2016-03-30 10:00:20 -04:00
plugin := b . plugins [ parts [ 1 ] ]
2012-08-17 21:39:26 -04:00
if plugin != nil {
plugin . Help ( channel , parts )
} else {
msg := fmt . Sprintf ( "I'm sorry, I don't know what %s is!" , parts [ 1 ] )
b . SendMessage ( channel , msg )
}
}
}
2016-04-01 10:20:03 -04:00
func ( b * bot ) LastMessage ( channel string ) ( msg . Message , error ) {
2012-10-11 16:28:00 -04:00
log := <- b . logOut
if len ( log ) == 0 {
2016-04-01 10:20:03 -04:00
return msg . Message { } , errors . New ( "No messages found." )
2012-10-11 16:28:00 -04:00
}
2013-06-16 22:40:30 -04:00
for i := len ( log ) - 1 ; i >= 0 ; i -- {
msg := log [ i ]
if strings . ToLower ( msg . Channel ) == strings . ToLower ( channel ) {
return msg , nil
2013-06-16 21:03:43 -04:00
}
}
2016-04-01 10:20:03 -04:00
return msg . Message { } , errors . New ( "No messages found." )
2012-10-11 16:28:00 -04:00
}
2012-08-17 16:38:15 -04:00
// Take an input string and mutate it based on $vars in the string
2016-04-01 10:20:03 -04:00
func ( b * bot ) Filter ( message msg . Message , input string ) string {
2012-08-26 15:14:09 -04:00
rand . Seed ( time . Now ( ) . Unix ( ) )
2012-08-17 18:09:29 -04:00
if strings . Contains ( input , "$NICK" ) {
nick := strings . ToUpper ( message . User . Name )
input = strings . Replace ( input , "$NICK" , nick , - 1 )
2012-08-26 15:14:09 -04:00
}
2012-08-26 19:23:51 -04:00
// Let's be bucket compatible for this var
2012-08-26 20:33:33 -04:00
input = strings . Replace ( input , "$who" , "$nick" , - 1 )
2012-08-26 15:14:09 -04:00
if strings . Contains ( input , "$nick" ) {
2012-08-17 18:09:29 -04:00
nick := message . User . Name
input = strings . Replace ( input , "$nick" , nick , - 1 )
}
2012-08-26 15:14:09 -04:00
2013-08-30 22:09:45 -04:00
for strings . Contains ( input , "$someone" ) {
2013-06-01 21:59:55 -04:00
nicks := b . Who ( message . Channel )
someone := nicks [ rand . Intn ( len ( nicks ) ) ] . Name
2013-08-30 22:09:45 -04:00
input = strings . Replace ( input , "$someone" , someone , 1 )
2012-08-26 15:14:09 -04:00
}
for strings . Contains ( input , "$digit" ) {
num := strconv . Itoa ( rand . Intn ( 9 ) )
input = strings . Replace ( input , "$digit" , num , 1 )
}
for strings . Contains ( input , "$nonzero" ) {
num := strconv . Itoa ( rand . Intn ( 8 ) + 1 )
input = strings . Replace ( input , "$nonzero" , num , 1 )
}
2012-08-26 19:23:51 -04:00
r , err := regexp . Compile ( "\\$[A-z]+" )
if err != nil {
panic ( err )
}
varname := r . FindString ( input )
blacklist := make ( map [ string ] bool )
blacklist [ "$and" ] = true
for len ( varname ) > 0 && ! blacklist [ varname ] {
2016-01-15 01:12:26 -05:00
text , err := b . getVar ( varname )
if err != nil {
2012-08-26 19:23:51 -04:00
blacklist [ varname ] = true
continue
}
2016-01-15 01:12:26 -05:00
input = strings . Replace ( input , varname , text , 1 )
2012-08-26 19:23:51 -04:00
varname = r . FindString ( input )
}
2012-08-17 16:38:15 -04:00
return input
}
2016-03-30 10:00:20 -04:00
func ( b * bot ) getVar ( varName string ) ( string , error ) {
2016-01-15 01:12:26 -05:00
var text string
2016-05-20 16:28:48 -04:00
err := b . db . Get ( & text , ` select value from variables where name=? order by random() limit 1 ` , varName )
2016-01-15 01:12:26 -05:00
switch {
case err == sql . ErrNoRows :
return "" , fmt . Errorf ( "No factoid found" )
case err != nil :
2016-01-15 08:59:26 -05:00
log . Fatal ( "getVar error: " , err )
2016-01-15 01:12:26 -05:00
}
return text , nil
}
2016-03-30 10:00:20 -04:00
func ( b * bot ) listVars ( channel string , parts [ ] string ) {
2016-05-20 16:28:48 -04:00
var variables [ ] string
err := b . db . Select ( & variables , ` select name from variables group by name ` )
2012-08-26 20:30:23 -04:00
if err != nil {
2016-01-15 01:12:26 -05:00
log . Fatal ( err )
2012-08-26 20:30:23 -04:00
}
msg := "I know: $who, $someone, $digit, $nonzero"
2016-05-20 16:28:48 -04:00
if len ( variables ) > 0 {
msg += ", " + strings . Join ( variables , ", " )
2012-08-26 20:30:23 -04:00
}
b . SendMessage ( channel , msg )
}
2016-03-30 10:00:20 -04:00
func ( b * bot ) Help ( channel string , parts [ ] string ) {
2012-08-17 18:56:44 -04:00
msg := fmt . Sprintf ( "Hi, I'm based on godeepintir version %s. I'm written in Go, and you " +
"can find my source code on the internet here: " +
2016-03-30 10:00:20 -04:00
"http://github.com/velour/catbase" , b . version )
2012-08-17 18:56:44 -04:00
b . SendMessage ( channel , msg )
}
2012-08-25 00:46:13 -04:00
2013-05-07 20:08:18 -04:00
// Send our own musings to the plugins
2016-03-30 10:00:20 -04:00
func ( b * bot ) selfSaid ( channel , message string , action bool ) {
2016-04-01 10:20:03 -04:00
msg := msg . Message {
2016-03-30 10:00:20 -04:00
User : & b . me , // hack
2013-05-07 20:08:18 -04:00
Channel : channel ,
Body : message ,
Raw : message , // hack
2013-06-17 00:04:10 -04:00
Action : action ,
2013-05-07 20:08:18 -04:00
Command : false ,
Time : time . Now ( ) ,
Host : "0.0.0.0" , // hack
}
2016-03-30 10:00:20 -04:00
for _ , name := range b . pluginOrdering {
p := b . plugins [ name ]
2013-05-07 20:08:18 -04:00
if p . BotMessage ( msg ) {
break
}
}
}