func (p *Plugin) detailTask(msg *slick.Message, id string) { todo := p.store.Get(msg.Channel) index, err := getTaskIndex(id, todo) if err != nil { msg.ReplyMention("Task not found...") return } task := todo[index] msg.Reply(printTaskDetails(task)) }
func (p *Plugin) deleteTask(msg *slick.Message, id string) { todo := p.store.Get(msg.Channel) index, err := getTaskIndex(id, todo) if err != nil { msg.ReplyMention("Task not found...") return } todo = append(todo[:index], todo[index+1:]...) p.store.Put(msg.Channel, todo) msg.Reply(fmt.Sprintf("Deleted task `%s`", id)) }
func (p *Plugin) replyHelp(msg *slick.Message, extra string) { answer := extra + ` Here's how you can get things orgnz'ed™: !todo add sometin to get done !todo list !todo strike ID !todo remove ID !todo help !todo ID ` msg.Reply(answer) return }
func (p *Plugin) closeTask(msg *slick.Message, id string) { todo := p.store.Get(msg.Channel) index, err := getTaskIndex(id, todo) if err != nil { msg.ReplyMention("Task not found...") return } parts := strings.Split(msg.Match[0], " ") task := todo[index] task.Closed = true task.ClosedAt = time.Now() if len(parts) > 3 { task.ClosingNote = strings.Join(parts[3:], " ") } p.store.Put(msg.Channel, todo) msg.Reply("`" + task.ID + "` ~" + strings.Join(task.Text, " // ") + "~ " + task.ClosingNote) }
func (p *Plugin) listTasks(msg *slick.Message, includeClosed bool) { todo := p.store.Get(msg.Channel) var answer []string for _, task := range todo { if task.Closed && !includeClosed { continue } // TODO format if task is closed text := "`" + task.ID + "` " + strings.Join(task.Text, " // ") answer = append(answer, text) } if len(answer) == 0 { msg.Reply("Nothing to do... Coffee time?") } else { msg.Reply(strings.Join(answer, "\n")) } }
func (bugger *Bugger) ChatHandler(listen *slick.Listener, msg *slick.Message) { if !msg.MentionsMe { return } if msg.ContainsAny([]string{"bug report", "bug count"}) && msg.ContainsAny([]string{"how", "help"}) { var report string if msg.Contains("bug report") { report = "bug report" } else { report = "bug count" } mention := bugger.bot.Config.Nickname msg.Reply(fmt.Sprintf( `Usage: %s, [give me a | insert demand] <%s> [from the | syntax filler] [last | past] [n] [days | weeks] examples: %s, please give me a %s over the last 5 days %s, produce a %s (7 day default) %s, I want a %s from the past 2 weeks %s, %s from the past week`, mention, report, mention, report, mention, report, mention, report, mention, report)) } else if msg.Contains("bug report") { days := util.GetDaysFromQuery(msg.Text) bugger.messageReport(days, msg, listen, func() string { reporter := bugger.makeBugReporter(days) return reporter.printReport(days) }) } else if msg.Contains("bug count") { days := util.GetDaysFromQuery(msg.Text) bugger.messageReport(days, msg, listen, func() string { reporter := bugger.makeBugReporter(days) return reporter.printCount(days) }) } return }
func (p *Plugin) createTask(msg *slick.Message) { var text []string text = append(text, strings.TrimPrefix(msg.Match[0], "!todo add ")) todo := p.store.Get(msg.Channel) if len(todo) > 600 { msg.ReplyMention("Gosh you have over 600 tasks!!! Clean some up first.") return } id := p.generateRandomID(todo) task := &Task{ ID: id, CreatedAt: time.Now(), User: msg.FromUser.ID, Text: text, Closed: false, } todo = append(todo, task) p.store.Put(msg.Channel, todo) msg.Reply("`" + task.ID + "` added to the todo") }
func (bugger *Bugger) messageReport(days int, msg *slick.Message, listen *slick.Listener, genReport func() string) { if days > 31 { msg.Reply(fmt.Sprintf("Whaoz, %d is too much data to compile - well maybe not, I am just scared", days)) return } msg.Reply(bugger.bot.WithMood("Building report - one moment please", "Whaooo! Pinging those githubbers - Let's do this!")) msg.Reply(genReport()) }
func (wicked *Wicked) ChatHandler(listen *slick.Listener, msg *slick.Message) { bot := listen.Bot uuidNow := time.Now() if strings.HasPrefix(msg.Text, "!wicked ") { fromRoom := "" if msg.FromChannel != nil { fromRoom = msg.FromChannel.ID } availableRoom := wicked.FindAvailableRoom(fromRoom) if availableRoom == nil { msg.Reply("No available Wicked Confroom for a meeting! Seems you'll need to create new Wicked Confrooms !") goto continueLogging } id := wicked.NextMeetingID() meeting := NewMeeting(id, msg.FromUser, msg.Text[7:], bot, availableRoom, uuidNow) wicked.pastMeetings = append(wicked.pastMeetings, meeting) wicked.meetings[availableRoom.ID] = meeting if availableRoom.ID == fromRoom { meeting.sendToRoom(fmt.Sprintf(`*** Starting wicked meeting W%s in here.`, meeting.ID)) } else { msg.Reply(fmt.Sprintf(`*** Starting wicked meeting W%s in room "%s". Join with !join W%s`, meeting.ID, availableRoom.Name, meeting.ID)) initiatedFrom := "" if fromRoom != "" { initiatedFrom = fmt.Sprintf(` in "%s"`, msg.FromChannel.Name) } meeting.sendToRoom(fmt.Sprintf(`*** Wicked meeting initiated by @%s%s. Goal: %s`, msg.FromUser.Name, initiatedFrom, meeting.Goal)) } meeting.sendToRoom(fmt.Sprintf(`*** Access report at %s/wicked/%s.html`, wicked.bot.Config.WebBaseURL, meeting.ID)) meeting.setTopic(fmt.Sprintf(`[Running] W%s goal: %s`, meeting.ID, meeting.Goal)) } else if strings.HasPrefix(msg.Text, "!join") { match := joinMatcher.FindStringSubmatch(msg.Text) if match == nil { msg.ReplyMention(`invalid !join syntax. Use something like "!join W123"`) } else { for _, meeting := range wicked.meetings { if match[1] == meeting.ID { meeting.sendToRoom(fmt.Sprintf(`*** @%s asked to join`, msg.FromUser.Name)) } } } } continueLogging: // // Public commands and messages // if msg.FromChannel == nil { return } room := msg.FromChannel.ID meeting, meetingExists := wicked.meetings[room] if !meetingExists { return } user := meeting.ImportUser(msg.FromUser) if strings.HasPrefix(msg.Text, "!proposition ") { decision := meeting.AddDecision(user, msg.Text[12:], uuidNow) if decision == nil { msg.Reply("Whoops, wrong syntax for !proposition") } else { msg.Reply(fmt.Sprintf("Proposition added, ref: D%s", decision.ID)) } } else if strings.HasPrefix(msg.Text, "!ref ") { meeting.AddReference(user, msg.Text[4:], uuidNow) msg.Reply("Ref. added") } else if strings.HasPrefix(msg.Text, "!conclude") { meeting.Conclude() // TODO: kill all waiting goroutines dealing with messaging delete(wicked.meetings, room) meeting.sendToRoom("Concluding Wicked meeting, that's all folks!") meeting.setTopic(fmt.Sprintf(`[Concluded] W%s goal: %s`, meeting.ID, meeting.Goal)) } else if match := decisionMatcher.FindStringSubmatch(msg.Text); match != nil { decision := meeting.GetDecisionByID(match[1]) if decision != nil { decision.RecordPlusplus(user) msg.ReplyMention("noted") } } // Log message newMessage := &Message{ From: user, Timestamp: uuidNow, Text: msg.Text, } meeting.Logs = append(meeting.Logs, newMessage) }
func (funny *Funny) ChatHandler(listen *slick.Listener, msg *slick.Message) { bot := listen.Bot if msg.Contains("mama") { listen.Bot.Listen(&slick.Listener{ ListenDuration: time.Duration(10 * time.Second), MessageHandlerFunc: func(listen *slick.Listener, msg *slick.Message) { if strings.Contains(msg.Text, "papa") { msg.Reply("3s", "yo rocker").DeleteAfter("3s") msg.AddReaction("wink") go func() { time.Sleep(3 * time.Second) msg.AddReaction("beer") time.Sleep(1 * time.Second) msg.RemoveReaction("wink") }() } }, }) } if msg.MentionsMe { if msg.Contains("you're funny") { if bot.Mood == slick.Happy { msg.Reply("/me blushes") } else { msg.Reply("here's another one") msg.Reply(slick.RandomString("robot jokes")) } } else if msg.ContainsAny([]string{"dumb ass", "dumbass"}) { msg.Reply("don't say such things") } else if msg.ContainsAny([]string{"thanks", "thank you", "thx", "thnks"}) { msg.Reply(bot.WithMood("my pleasure", "any time, just ask, I'm here for you, ffiieeewww!get a life")) } else if msg.Contains("how are you") && msg.MentionsMe { msg.ReplyMention(bot.WithMood("good, and you ?", "I'm wild today!! wadabout you ?")) bot.Listen(&slick.Listener{ ListenDuration: 60 * time.Second, FromUser: msg.FromUser, FromChannel: msg.FromChannel, MentionsMeOnly: true, MessageHandlerFunc: func(listen *slick.Listener, msg *slick.Message) { msg.ReplyMention(bot.WithMood("glad to hear it!", "zwweeeeeeeeet !")) listen.Close() }, TimeoutFunc: func(listen *slick.Listener) { msg.ReplyMention("well, we can catch up later") listen.Close() }, }) } } if msg.ContainsAny([]string{"lot of excitement", "that's exciting", "how exciting", "much excitement"}) { msg.Reply("http://static.fjcdn.com/gifs/Japanese+kids+spongebob+toys_0ad21b_3186721.gif") } else if msg.ContainsAny([]string{"what is your problem", "what's your problem", "is there a problem", "which problem"}) { msg.Reply("http://media4.giphy.com/media/19hU0m3TJe6I/200w.gif") } else if msg.Contains("force push") { url := slick.RandomString("forcePush") msg.Reply(url) } else if msg.ContainsAny([]string{"there is a bug", "there's a bug"}) { msg.Reply("https://s3.amazonaws.com/pushbullet-uploads/ujy7DF0U8wm-9YYvLZkmSM8pMYcxCXXig8LjJORE9Xzt/The-life-of-a-coder.jpg") } else if msg.ContainsAny([]string{"oh yeah", "approved"}) { msg.Reply("https://i.chzbgr.com/maxW250/4496881920/h9C58F860.gif") } else if msg.Contains("ice cream") { msg.Reply("http://i.giphy.com/IGyLuFXIGSJj2.gif") msg.Reply("I love ice cream too") } else if msg.ContainsAny([]string{"lot of tension", "some tension", " tensed"}) { msg.Reply("http://thumbpress.com/wp-content/uploads/2014/01/funny-gif-meeting-strangers-girl-scared1.gif") msg.Reply("tensed, like that ?") } else if msg.Contains("quick fix") { msg.Reply("http://blog.pgi.com/wp-content/uploads/2013/02/jim-carey.gif") msg.Reply("make it real quick") } else if msg.ContainsAny([]string{"crack an egg", "crack something", "to crack"}) { msg.Reply("http://s3-ec.buzzfed.com/static/enhanced/webdr02/2012/11/8/18/anigif_enhanced-buzz-31656-1352415875-9.gif") msg.Reply("crack an egg, yeah") } else if msg.ContainsAny([]string{"i'm stuck", "I'm stuck", "we're stuck"}) { msg.Reply("http://media.giphy.com/media/RVlWx1msxnf7W/giphy.gif") msg.Reply("I'm stuck too!") } else if msg.ContainsAny([]string{"watching tv", "watch tv"}) { msg.Reply("http://i0.kym-cdn.com/photos/images/newsfeed/000/495/040/9ab.gif") msg.Reply("like that ?") } else if msg.ContainsAny([]string{"spider", "pee on", "inappropriate"}) { msg.Reply("https://i.chzbgr.com/maxW500/5626597120/hB2E11E61.gif") } else if msg.ContainsAny([]string{"a meeting", "an interview"}) { msg.Reply("like this one") msg.Reply("https://i.chzbgr.com/maxW500/6696664320/hFC69678C.gif") } else if msg.ContainsAny([]string{"it's odd", "it is odd", "that's odd", "that is odd", "it's awkward", "it is awkward", "that's awkward", "that is awkward"}) { term := "awkward" if msg.Contains("odd") { term = "odd" } msg.Reply(fmt.Sprintf("THAT's %s", term)) msg.Reply("https://i.chzbgr.com/maxW500/8296294144/h7AC1001C.gif") } else if msg.Text == "ls" { msg.Reply("/code deploy/ Contributors-Guide/ image_server/ sheep_porn/ streambed/\nstreamhead/ README.md") } else if msg.ContainsAny([]string{"that's really cool", "that is really cool", "really happy"}) { msg.Reply("http://media.giphy.com/media/BlVnrxJgTGsUw/giphy.gif") } else if msg.ContainsAny([]string{"difficult problem", "hard problem"}) { msg.Reply("naming things, cache invalidation and off-by-1 errors are the two most difficult computer science problems") } else if msg.Contains("in theory") { msg.Reply("yeah, theory and practice perfectly match... in theory.") } else if msg.Contains("dishes") { msg.Reply(slick.RandomString("dishes")) } else if msg.Contains(" bean") { msg.Reply("http://media3.giphy.com/media/c35RMDO6luMaQ/500w.gif") } else if msg.Contains("steak") { msg.Reply("http://media.tumblr.com/tumblr_me6r52h1md1r6nno1.gif") } else if msg.ContainsAny([]string{"booze", "alcohol", "martini", " dog "}) { msg.Reply("http://media2.giphy.com/media/ZmJBjPdd44gXS/200w.gif") } else if msg.ContainsAny([]string{"internet", " tube "}) { msg.Reply("https://pbs.twimg.com/media/By0J3YHCcAA4UBo.jpg:large") } }
func (dep *Deployer) ChatHandler(listen *slick.Listener, msg *slick.Message) { bot := listen.Bot // Discard non "mention_name, " prefixed messages if !strings.HasPrefix(msg.Text, fmt.Sprintf("%s, ", bot.Config.Nickname)) { return } if match := deployFormat.FindStringSubmatch(msg.Text); match != nil { if dep.lockedBy != "" { msg.Reply(fmt.Sprintf("Deployment was locked by %s. Unlock with '%s, unlock deployment' if they're OK with it.", dep.lockedBy, dep.bot.Config.Nickname)) return } if dep.runningJob != nil { params := dep.runningJob.params msg.Reply(fmt.Sprintf("@%s Deploy currently running: %s", msg.FromUser.Name, params)) return } else { params := &DeployParams{ Environment: match[3], Branch: match[2], Tags: match[8], DeploymentBranch: match[5], InitiatedBy: msg.FromUser.RealName, From: "chat", initiatedByChat: msg, } go dep.handleDeploy(params) } return } else if msg.Contains("cancel deploy") { if dep.runningJob == nil { msg.Reply("No deploy running, sorry man..") } else { if dep.runningJob.killing == true { msg.Reply("deploy: Interrupt signal already sent, waiting to die") return } else { msg.Reply("deploy: Sending Interrupt signal...") dep.runningJob.killing = true dep.runningJob.kill <- true } } return } else if msg.Contains("in the pipe") { url := dep.getCompareUrl("prod", dep.config.DefaultStreambedBranch) mention := msg.FromUser.Name if url != "" { msg.Reply(fmt.Sprintf("@%s in %s branch, waiting to reach prod: %s", mention, dep.config.DefaultStreambedBranch, url)) } else { msg.Reply(fmt.Sprintf("@%s couldn't get current revision on prod", mention)) } } else if msg.Contains("unlock deploy") { dep.lockedBy = "" msg.Reply(fmt.Sprintf("Deployment is now unlocked.")) bot.Notify(dep.config.AnnounceRoom, "purple", "text", fmt.Sprintf("%s has unlocked deployment", msg.FromUser.Name), true) } else if msg.Contains("lock deploy") { dep.lockedBy = msg.FromUser.Name msg.Reply(fmt.Sprintf("Deployment is now locked. Unlock with '%s, unlock deployment' ASAP!", dep.bot.Config.Nickname)) bot.Notify(dep.config.AnnounceRoom, "purple", "text", fmt.Sprintf("%s has locked deployment", dep.lockedBy), true) } else if msg.Contains("deploy") || msg.Contains("push to") { mention := dep.bot.Config.Nickname msg.Reply(fmt.Sprintf(`Usage: %s, [please|insert reverence] deploy [<branch-name>] to <environment> [using <deployment-branch>][, tags: <ansible-playbook tags>, ..., ...] examples: %s, please deploy to prod %s, deploy thing-to-test to stage %s, deploy complicated-thing to stage, tags: updt_streambed, blow_up_the_sun other commands: %s, what's in the pipe? - show what's waiting to be deployed to prod %s, lock deployment - prevent deployment until it's unlocked`, mention, mention, mention, mention, mention, mention)) } }
func (v *Vote) voteHandler(listen *slick.Listener, msg *slick.Message) { v.mutex.Lock() defer v.mutex.Unlock() bot := v.bot // TODO: ok, @kat wants to survey what's for lunch, use "!vote The Food Place http://food-place.url" .. you can vote for the same place with a substring: "!vote food place" // TODO: match "!vote Mucha Dogs http://bigdogs.com" // TODO: match "!vote mucha dogs" // TODO: match "!vote Other place if msg.Text == "!what-for-lunch" || msg.Text == "!vote-for-lunch" { msg.ReplyMention("you can say `!what-for-lunch 5m` to get a vote that will last 5 minutes. `!vote-for-lunch` is an alias") return } if msg.HasPrefix("!what-for-lunch ") || msg.HasPrefix("!vote-for-lunch ") { if v.runningVotes[msg.FromChannel.ID] != nil { msg.ReplyMention("vote is already running!").DeleteAfter("3s") return } timing := strings.TrimSpace(strings.SplitN(msg.Text, " ", 2)[1]) dur, err := time.ParseDuration(timing) if err != nil { msg.ReplyMention(fmt.Sprintf("couldn't parse duration: %s", err)) return } v.runningVotes[msg.FromChannel.ID] = make([]vote, 0) go func() { time.Sleep(dur) v.mutex.Lock() defer v.mutex.Unlock() res := make(map[string]int) for _, oneVote := range v.runningVotes[msg.FromChannel.ID] { res[oneVote.vote] = res[oneVote.vote] + 1 } // TODO: print report, clear up if len(res) == 0 { msg.ReplyMention("polls closed, but no one voted") } else { out := []string{"polls closed, here are the results:"} for theVote, count := range res { plural := "" if count > 1 { plural = "s" } out = append(out, fmt.Sprintf("* %s: %d vote%s", theVote, count, plural)) } msg.ReplyMention(strings.Join(out, "\n")) } delete(v.runningVotes, msg.FromChannel.ID) }() msg.Reply("<!channel> okay, what do we eat ? Votes are open. Use `!vote The Food Place http://food-place.url` .. you can vote for the same place with a substring, ex: `!vote food place`") } if msg.HasPrefix("!vote ") { running := v.runningVotes[msg.FromChannel.ID] if running == nil { msg.Reply(bot.WithMood("what vote ?!", "oh you're so cute! voting while there's no vote going on !")) return } voteCast := strings.TrimSpace(strings.SplitN(msg.Text, " ", 2)[1]) if len(voteCast) == 0 { return } // TODO: check for dupe for _, prevVote := range running { if msg.FromUser.ID == prevVote.user { // buzz off if you voted already msg.ReplyMention(bot.WithMood("you voted already", "trying to double vote ! how charming :)")) return } } for _, prevVote := range running { if strings.Contains(strings.ToLower(prevVote.vote), strings.ToLower(voteCast)) { running = append(running, vote{msg.FromUser.ID, prevVote.vote}) v.runningVotes[msg.FromChannel.ID] = running msg.ReplyMention(bot.WithMood("okay", "hmmm kaay")).DeleteAfter("2s") return } } running = append(running, vote{msg.FromUser.ID, voteCast}) v.runningVotes[msg.FromChannel.ID] = running msg.ReplyMention(bot.WithMood("taking note", "taking note! what a creative mind...")).DeleteAfter("2s") // TODO: match "!what-for-lunch 1h|5m|50s" } }
func (plotberry *PlotBerry) ChatHandler(listen *slick.Listener, msg *slick.Message) { if msg.MentionsMe && msg.Contains("how many user") { msg.Reply(fmt.Sprintf("We got %d users!", plotberry.totalUsers)) } return }
// Handler func (healthy *Healthy) ChatHandler(listen *slick.Listener, msg *slick.Message) { log.Println("Health check. Requested by", msg.FromUser.Name) msg.Reply(healthy.CheckAll()) }