-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathslack-bot.js
197 lines (183 loc) · 5.94 KB
/
slack-bot.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
// If this syntax looks unfamiliar, don't worry, it's just JavaScript!
// Learn more about ES2015 here: https://babeljs.io/docs/learn-es2015/
//
// Run "npm install" and then test with this command in your shell:
// SLACK_API_TOKEN=<YOUR_TOKEN_HERE> node examples/slack-bot.js
//
// Note that you'll first need a Slack API token, which you can get by going
// to your team's settings page and creating a new bot:
// https://my.slack.com/services/new/bot
const Promise = require('bluebird');
// ES2015 syntax:
// import {createSlackBot, createCommand, createParser} from 'chatter';
// ES5 syntax:
// const chatter = require('chatter');
const chatter = require('..');
const createSlackBot = chatter.createSlackBot;
const createCommand = chatter.createCommand;
const createParser = chatter.createParser;
// import {RtmClient} from '@slack/client';
const slack = require('@slack/client');
const RtmClient = slack.RtmClient;
const WebClient = slack.WebClient;
const MemoryDataStore = slack.MemoryDataStore;
// ================
// message handlers
// ================
// Respond to the word "hello"
const helloHandler = text => {
if (/hello/i.test(text)) {
return `Hello to you too!`;
}
return false;
};
// Respond, but only if the message contains the word "lol".
const lolHandler = text => {
if (/lol/i.test(text)) {
const newText = text.replace(/lol/ig, 'laugh out loud');
return `More like "${newText}" amirite`;
}
return false;
};
// A command that says something after a delay. Be careful, though! Even though
// this command yields false to indicate when it doesn't know how to handle the
// message, it does so after the delay. If possible, return false immediately!
const delayCommand = createCommand({
name: 'delay',
description: `I'll say something after a delay.`,
usage: '[yes | no]',
}, text => {
return new Promise(resolve => {
setTimeout(() => {
if (!text) {
resolve(false);
}
else if (text.toLowerCase() === 'yes') {
resolve('Awesome!');
}
else {
resolve('Bummer!');
}
}, 250);
});
});
// A command that echoes user input, as long as the user says something after
// the command name!
const echoCommand = createCommand({
name: 'echo',
description: `I'm the echo command.`,
usage: '<say anything here>',
}, text => {
if (text) {
const isAmazing = text.toLowerCase() === 'amazing';
return {
messages: [
`You said *${text}*.`,
isAmazing ? 'Which is amazing...' : 'Which is great, and all...',
isAmazing ? 'Literally!' : 'But not amazing.',
],
};
}
return false;
});
// Command that adds args into a sum. If no args were specified, return false
// to display usage information. If sum isn't a number, display a message.
const addCommand = createCommand({
name: 'add',
description: 'Adds some numbers.',
usage: 'number [ number [ number ... ] ]',
}, createParser(parsed => {
const args = parsed.args;
if (args.length === 0) {
return false;
}
const result = args.reduce((sum, n) => sum + Number(n), 0);
if (isNaN(result)) {
return `Whoops! Are you sure those were all numbers?`;
}
return `${args.join(' + ')} = ${result}`;
}));
// Command that multiplies args into a product. If no args were specified,
// return false to display usage information. If product isn't a number,
// display a message.
const multiplyCommand = createCommand({
name: 'multiply',
description: 'Multiplies some numbers.',
usage: 'number [ number [ number ... ] ]',
}, createParser(parsed => {
const args = parsed.args;
if (args.length === 0) {
return false;
}
const result = args.reduce((product, n) => product * Number(n), 1);
if (isNaN(result)) {
return `Whoops! Are you sure those were all numbers?`;
}
return `${args.join(' x ')} = ${result}`;
}));
// Parent math command that "namespaces" its sub-commands.
const mathCommand = createCommand({
name: 'math',
description: 'Math-related commands.',
}, [
addCommand,
multiplyCommand,
]);
// ================
// proper slack bot
// ================
const bot = createSlackBot({
// The bot name.
name: 'Chatter Bot',
// The getSlack function should return instances of the slack rtm and web
// clients, like so. See https://github.com/slackhq/node-slack-sdk
getSlack() {
return {
rtmClient: new RtmClient(process.env.SLACK_API_TOKEN, {
dataStore: new MemoryDataStore(),
autoReconnect: true,
logLevel: 'error',
}),
webClient: new WebClient(process.env.SLACK_API_TOKEN),
};
},
// The createMessageHandler function should return a top-level message handler
// to handle each message.
createMessageHandler(id, meta) {
const channel = meta.channel;
// Get actual bot name and aliases for the bot. If the bot's actual name in
// Slack was "test", the aliases would be "test:" "@test" "@test:", allowing
// the bot to respond to commands prefixed with any of them. If the channel
// is a DM, the bot name will be null, and the actual name will be added to
// the aliases list, so the bot can respond to both prefixed and non-
// prefixed messages. Alternately, specify your own bot name and aliases!
const nameObj = this.getBotNameAndAliases(channel.is_im);
console.log(nameObj);
// Create the top-level command.
const rootCommand = createCommand({
isParent: true,
name: nameObj.name,
aliases: nameObj.aliases,
description: `Hi, I'm the test bot!`,
}, [
delayCommand,
echoCommand,
mathCommand,
]);
// Direct message.
if (channel.is_im) {
return rootCommand;
}
// Public channel message.
return [
rootCommand,
// You can certainly combine commands and other message handlers, as long
// as the command has a name. (If the command didn't have a name, it would
// handle all messages, and message handlers after it would never run).
helloHandler,
lolHandler,
];
},
});
// Connect!
bot.login();