Product
Socket Now Supports uv.lock Files
Socket now supports uv.lock files to ensure consistent, secure dependency resolution for Python projects and enhance supply chain security.
@exoplay/exobot
Advanced tools
An ES6+ chatbot. Requires Node ^6.2.
npm install --save @exoplay/exobot
To start an exobot instance, you need to import the bot itself and initialize it with plugins and chat service adapters.
const { Exobot, adapters, plugins, LogLevels } = require('@exoplay/exobot');
const { Help, Greetings } = plugins;
const BOT_ALIAS = '!e';
const BOT_NAME = 'exobot';
const LOG_LEVEL = process.env.EXOBOT_LOG_LEVEL || LogLevels.INFO;
const shell = adapters.Shell;
const bot = new Exobot(BOT_NAME, {
alias: BOT_ALIAS,
adapters: [
new shell(),
],
plugins: [
new Help(),
new Greetings(),
],
logLevel: LOG_LEVEL,
});
module.exports = bot;
$ node index.js
> Chat: hi, exobot
> exobot: hi, shell!
What did we do there?
index.js
Exobot
class, service adapters, and pluginsnode index.js
and interacted with the botThe easiest way to start is copy the example above - this will get you started with a chatbot with a shell adapter. The shell adapter will start an interactive console with which you can chat in a single "room"; Exobot will respond to messages that trigger plugins.
git init
(or source control initialization method of choice), then
npm init
to start up an NPM package. (You probably won't publish your bot
as its own package - but this will create a package.json
file that contains
your dependencies.)npm install --save @exoplay/exobot
to install the chatbot.index.js
.node index.js
. Chat with yourself for a while, then read on to learn
how to configure your chatbot, or even build your own plugins and adapters.Exobot is configured in its constructor, which takes two arguments - a bot name (a required string), and an options object.
The bot name is used for commands - if your bot's name is 'exobot'
, it will
respond
to commands beginning with 'exobot'
. You'll want this to match the
name used in your chat service (so if its name is actually 'DEATHBOT_9000'
in
Slack, you should call it that here too, or people may be confused.)
The options object contains all other configuration - such as a list of plugins and chat service adapters, log levels, and data encryption keys.
alias
- an additional way to trigger exobot commands. '/'
, ';'
, or
'hey bot'
, for example.adapters
- an array of initialized chat adapters, such as
slack,
discord,
or twitch. exobot
also comes with a shell
adapter for playing around in your terminal.plugins
- an array of initialized plugins, such as
giphy or
points. exobot also
comes with help
and greetings
plugins as examples.readFile
and writeFile
- functions called when the in-memory json db
is saved. By default, this writes a json file to cwd/data/botname.json
,
but you could also override the default local file storage to use s3 with
exobot-db-s3.dbPath
- if you're using local file storage, you can set where to save.
Defaults to cwd/data/botname.json
.Most plugins respond to chat messages - either by listen
ing to all chat
messages, or respond
ing to specific commands. exobot comes with greetings
and help
plugins, but building your own is easy. Some examples:
import { ChatPlugin } from '@exoplay/exobot';
export default class Ping extends ChatPlugin {
help = 'Says "pong" when you send it "ping"';
register (bot) {
this.respond(/ping/, this.pong);
}
pong (match, message) {
return 'pong';
}
}
In this plugin, we have extended exobot's ChatPlugin class - this gives it
functionality to respond to chat messages. We've then told it to respond
to
the regex /ping/
by firing a function, called pong
. The return
value of
the function is then sent back to the chat channel.
Chat plugins follow the following lifecycle:
First, The constructor
is called with options sent in. As the bot is
initialized with instances of plugins, this is where you would pass in
configuration options, such as:
class StatusPlugin extends ChatPlugin {
constructor (options) {
super(options);
this.endpoint = options.endpoint;
}
//...
async getStatus () {
const res = await this.http.get(this.endpoint);
return res.statusCode;
}
}
In the above example, we'd initialize the exobot instance with
plugins: [ new StatsPlugin({ endpoint: 'https://github.com' }) ]
to pass
in the options we need later on.
Next, when the bot instance begins listening, the plugin's register
method is
called, with the bot
instace passed in. Note that the constructor doesn't have
the bot yet - it doesn't exist until register
.
register
is also where you register listen
and respond
commands. listen
and respond
are the most important parts of your chat plugin - these allow the
bot to interact with chat. Each can take either a regex or a function, and if
a match is found (or, if a function, if it is truthy), it will fire the
function passed in. Functions for responding can be promises (or
ES7 async
functions) and will resolve when the promises do. This makes it
easy to write asynchronous code, such as firing http requests.
The responding function gets two arguments: a match
object, which is either
the regex's exec
response or the function return value, and a Message
object, which contains the original message, user, and whether the message is a
whisper.
You can optionally add a help
property, which exobot's help
plugin uses to
explain to useres how the plugin works.
Finally, the bot also exposes bot.http
, which is a promise-ified
superagent wrapper, to make http
calls easy to make.
class StatusPlugin extends ChatPlugin {
help = [
'Get the status of an http endpoint. Responds to `status` or listens to',
'status <http://whatever.com>.'
].join('\n');
constructor (options) {
super(options);
this.endpoint = options.endpoint;
}
register (bot) {
super.register(bot);
if (!this.endpoint) {
bot.log.warn('No endpoint passed in to StatusPlugin.');
}
this.respond(/status/, this.getStatus);
this.listen(/^status (http:\/\/\S+)/, this.getStatus);
this.listen(m => m === 'status', this.getStatus);
}
async getStatus (match, message) {
let endpoint = this.endpoint;
// if the regex succeeded, match[1] should be an http endpoint
if (match && match.length) {
endpoint = match[1];
}
const res = await this.http.get(this.endpoint);
return res.statusCode;
}
}
You can also build other types of plugins: EventPlugin
, HTTPPlugin
, or build
your own class of plugin with the Plugin
class. Documentation to come someday.
Adapters allow your bot to connect to a chat service, such as Slack or Discord. exobot comes with a shell adapter by default, but you could also build your own for your chat service of choice. Some examples:
// An example: import an API lib for your chat service, or do it with raw
// sockets or http, or whatever.
import ChatServiceLibrary from '@chatservice/lib';
import { Adapter, User } from '@exoplay/exobot';
export default class ChatServiceAdapter extends Adapter {
constructor ({ token, username }) {
super(...arguments);
this.token = token;
this.username = username;
}
register (bot) {
super.register(bot);
// Initialize the chat service lib we pulled in earlier
this.service = new ChatServiceLibrary(this.username, this.token);
// listen to some events. bind the functions to `this` to make sure we can
// access our class instance, bot, and `super`.
this.service.on('ready', this.serviceReady.bind(this));
this.service.on('message', this.serviceMessage.bind(this));
}
// the `send` funciton is defined by Adapter and called by plugins when they
// resolve (if they resolve.)
send (message) {
this.bot.log.debug(`Sending ${message.text} to ${message.channel}`);
// Send the message data to the chat service client.
this.service.sendMessage({
to: message.channel,
message: message.text,
});
}
serviceReady () {
this.status = Adapter.STATUS.CONNECTED;
this.bot.emitter.emit('connected', this.id);
this.bot.log.notice('Connected to ChatService.');
}
// We'll pretend our fake chat service lib takes a function which is called
// with message, user, and channel. We'll take these arguments and "receive"
// them, which will fire off all of the plugins so they can respond where
// necessary.
serviceMessage (user, text, channel) {
// We don't want to listen to messages from ourself.
if (user.name === this.username) { return; }
// Create a new User instance to pass along in the Message.
const user = new User(user.name, user.id);
// Check if our fake chat service lib says the channel is "private". If it
// is a private message between the user and bot, we'll make it act like a
// "respond" command instead of just a "listen".
if (channel.private) {
return super.receiveWhisper({ user, text, channel });
}
return this.receive({ user, text, channel });
}
}
Chat service adapters have a similar lifecycle to plugins:
send
, called by the bot instance when plugins resolve.You can listen and fire any arbitrary functions - for example, some chat
services may include presence information, and fire enter
and leave
events.
You can then receive
your own PresenceMessage
similar to how we receive
a
TextMessage
in the serviceMessage
in the example. (Right now, the only
Message
classes are TextMessage
and PresenceMessage
). Many adapters may
also want to make use of the Status
enum, which could be:
You may also want to use bot.log
to log important events to stdout, such as
connection or configuration errors. bot.log
can fire:
In order of ascending severity.
Exobot is loosely based on hubot, for which the author has a great deal of admiration. Hubot is more user-friendly in many ways (autoloading scripts, for example, instead of requiring the user to write their own imports and configuration). In other ways, this flexibility can be limiting; it's easier to make a pure-js bot more efficient and testable (and the author thinks that ES6, rather than Coffeescript, is a more viable choice of language; plugin-writers can always choose to opt-in to Coffeescript and export a built file if they want.)
LGPL licensed. Copyright 2016 Exoplay, LLC. See LICENSE file for more details.
FAQs
a chatbox
The npm package @exoplay/exobot receives a total of 74 weekly downloads. As such, @exoplay/exobot popularity was classified as not popular.
We found that @exoplay/exobot demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Product
Socket now supports uv.lock files to ensure consistent, secure dependency resolution for Python projects and enhance supply chain security.
Research
Security News
Socket researchers have discovered multiple malicious npm packages targeting Solana private keys, abusing Gmail to exfiltrate the data and drain Solana wallets.
Security News
PEP 770 proposes adding SBOM support to Python packages to improve transparency and catch hidden non-Python dependencies that security tools often miss.