README
About The Project
This Node package is a small but powerful tool to bypass the lengthy Twitter bot setup stage and get straight to defining your bot actions. While some initial setup is still required, this package aims to abstract away the network and bot lifecycle management.
Built With
Getting Started
To get a local copy up and running follow these simple steps.
Prerequisites
- Node.js
- npm
npm install npm@latest -g
- Twitter Developer Account
Installation
Add it to any project or build a new project around it with NPM
Install from npm
npm i twitter-bot-manager
That's it.
Or you can download it directly from here and add it to any project, just don't forget to install it's dependencies first.
Clone the repo into your project
git clone https://github.com/christophgomez/twitter-bot-manager.git
Change directories
cd twitter-bot-manager
Install NPM packages
npm i
Usage
Here's a small step-by-step to get a simple bot up and running.
Step 1: Twitter
Ensure you have a Twitter Developer Account, if you don't, make one, cause this package is useless without it.
Once you're registered as a developer, create a new app, you'll need the app keys and tokens later. If you want your bot to respond to DMs, ensure your app has Read, Write, and Direct Messages permissions.
Create a Dev Environment for the app you just created using the Account Activity API/Sandbox option at the bottom of that page. Ensure the environment is labeled 'dev'.
Step 2: Node
- Init a new Node project, fill out the prompts.
mkdir my-really-cool-bot-project
cd my-really-cool-bot-project
npm init
- Install the bot-manager package
npm i twitter-bot-manager
- Setup the bot-server in your project's entry file
import TwitterBotServer from "twitter-bot-manager";
TwitterBotServer({
bot: new TwitterBot{
name: "My Bot",
consumer_key: "YOUR_TWITTER_APP_CONSUMER_KEY",
consumer_secret: "YOUR_TWITTER_APP_CONSUMER_SECRET",
token: "YOUR_TWITTER_APP_ACCESS_TOKEN",
token_secret: "YOUR_TWITTER_APP_ACCESS_TOKEN_SECRET",
},
});
- Run your file, if your entry file is index.js, for example:
node index.js
The package should handle registering your bot with the Twitter servers, and if all went well, you should see a final output of:
Waiting to receive and process [bot name]'s account events...
Bot eventActions
This is where the magic happens. To make your bot actually do things, you must define that behavior in the eventActions function parameter. Whenever your bot's account receives an event, it will go through this function, and your bot will act accordingly.
Let's break it down a little. You can define the function wherever you'd like; in the object itself, at the top of the file, or if the function start's to get very large, maybe in it's own file for better maintainability. For now, lets define it within the object itself:
TwitterBotServer({
bot: new TwitterBot({
name: "My Bot",
consumer_key: "YOUR_TWITTER_APP_CONSUMER_KEY",
consumer_secret: "YOUR_TWITTER_APP_CONSUMER_SECRET",
token: "YOUR_TWITTER_APP_ACCESS_TOKEN",
token_secret: "YOUR_TWITTER_APP_ACCESS_TOKEN_SECRET",
eventActions: (event, self) => {
console.log("Incoming event for My Bot");
console.log(event);
},
}),
});
Whenever that account receives an event, the server will simply log the event, seeing what it contains and it's structure may be useful. Let's expand on this:
(event, self) => {
console.log("Incoming event for My Bot");
console.log(event);
switch (true) {
case "tweet_create_events" in event:
break;
case "favorite_events" in event:
break;
case "follow_events" in event:
break;
case "unfollow_events" in event:
break;
case "block_events" in event:
break;
case "unblock_events" in event:
break;
case "mute_events" in event:
break;
case "unmute_events" in event:
break;
case "user_event" in event:
break;
case "direct_message_events" in event:
break;
case "direct_message_indicate_typing_events" in event:
break;
case "direct_message_mark_read_events" in event:
break;
case "tweet_delete_events" in event:
break;
}
};
Now, depending on the type of event your bot receives, you can do something accordingly.
self
, for?
What's that second parameter, Your function will be passed a reference to the bot itself. You can do whatever you please with this reference, in response to any event that comes from Twitter
If you want to make your bot do something Twitter-y, use the bot's oauth to make a call to the Twitter API. You can use the inluded API object, it's still a work in progress, and requires more function definitions for all the API endpoints, so any contributions to that are welcome.
Here's an example of getting all user tweets from someone who favorited one of your bot's tweets
Tip #1: You can mark the event handler async
Tip #2: You can name the params whatever you want
import { TwitterAPI } from "twitter-bot-manager";
// you can also write this in a seperate file and import it
export const action = async (e, bot) => {
console.log("Incoming event for " + bot.name);
console.log(e);
const api = new TwitterAPI(bot.oauth);
switch (true) {
case "favorite_events" in e:
e = event.favorite_events[0];
console.log("Getting user " + e.user.id + "'s tweets...");
const tweets = await api.getAllUserTweets(e.user.id);
console.log("Number of tweets imported: " + tweets.length);
console.log(tweets);
break;
}
};
All in all, your entry file could end up looking something like this:
import TwitterBotServer, { TwitterBot, TwitterAPI } from "twitter-bot-manager";
TwitterBotServer({
bot: new TwitterBot({
name: "My Bot",
consumer_key: "YOUR_TWITTER_APP_CONSUMER_KEY",
consumer_secret: "YOUR_TWITTER_APP_CONSUMER_SECRET",
token: "YOUR_TWITTER_APP_ACCESS_TOKEN",
token_secret: "YOUR_TWITTER_APP_ACCESS_TOKEN_SECRET",
eventActions: async (event, self) => {
console.log("Incoming event for " + self.name);
console.log(event);
const api = new TwitterAPI(self.oauth);
switch (true) {
case "direct_message_events" in event:
event = event.favorite_events[0];
console.log("Getting user " + event.user.id + "'s tweets...");
const tweets = await api.getAllUserTweets(
event.user.id,
self.oauth
);
console.log("Number of tweets imported: " + tweets.length);
console.log(tweets);
break;
}
},
}),
});
Or maybe you want to drill down, and only handle a specific event, you can do that too.
import TwitterBotServer, { TwitterBot } from "twitter-bot-manager";
TwitterBotServer({
bot: new TwitterBot({
name: "My Bot",
consumer_key: "YOUR_TWITTER_APP_CONSUMER_KEY",
consumer_secret: "YOUR_TWITTER_APP_CONSUMER_SECRET",
token: "YOUR_TWITTER_APP_ACCESS_TOKEN",
token_secret: "YOUR_TWITTER_APP_ACCESS_TOKEN_SECRET",
eventActions: {
alwaysRunDefault: true,
actions: {
default: (event, bot) => {
console.log("All events will pass through this action");
},
direct_message_indicate_typing_events: (event, bot) => {
console.log(
"Only the direct message typing events will pass through here"
);
},
},
},
}),
});
If you choose to omit alwaysRunDefault
, the default action will only run when an action is not found for an incoming event.
Bot Jobs
You can also specify any number of jobs when creating your bot, these jobs will run at the provided interval. The interval must be valid cron syntax.
import TwitterBotServer, { TwitterBot } from "twitter-bot-manager";
TwitterBotServer({
bot: new TwitterBot({
name: "My Bot",
consumer_key: "YOUR_TWITTER_APP_CONSUMER_KEY",
consumer_secret: "YOUR_TWITTER_APP_CONSUMER_SECRET",
token: "YOUR_TWITTER_APP_ACCESS_TOKEN",
token_secret: "YOUR_TWITTER_APP_ACCESS_TOKEN_SECRET",
jobs: [
{
interval: "* * * * *",
jobAction: (self) => {
// A job that will run every minute
},
},
{
interval: "*/2 * * * * ",
jobAction: (self) => {
// A job that will run every minute
},
},
],
}),
});
Again, you have access to the bot itself in this function, so you can whatever you want with the bot, and not have to worry about keeping your own references.
Specify a timezone to run the job at the provided interval respective to a timezone differing from your OS's.
job: {
interval: string,
jobAction: (self) => any,
timezone?: string
}
Manage Multiple Bots
This package's true power comes from the fact that it allows you to spin up and manage multiple bots at once. Here's how to do that.
import TwitterBotServer, { TwitterBotManager } from "twitter-bot-manager";
const bots = {
firstBot: {
name: "My First Bot",
consumer_key: "FIRST_BOT_APP_CONSUMER_KEY",
consumer_secret: "FIRST_BOT_APP_CONSUMER_SECRET",
token: "FIRST_BOT_APP_ACCESS_TOKEN",
token_secret: "FIRST_BOT_APP_ACCESS_TOKEN_SECRET",
eventActions: (event, self) => {
// A function to handle incoming Twitter events
},
jobs: [
{
interval: "* * * * *",
jobAction: (self) => {
// A job that will run every minute
},
},
],
},
secondBot: {
name: "My Second Bot",
consumer_key: "SECOND_BOT_APP_CONSUMER_KEY",
consumer_secret: "SECOND_BOT_APP_CONSUMER_SECRET",
token: "SECOND_BOT_APP_ACCESS_TOKEN",
token_secret: "SECOND_BOT_ACCESS_TOKEN_SECRET",
eventActions: (event, self) => {
// A function to handle incoming Twitter events
},
jobs: [
{
interval: "* * * * *",
jobAction: (self) => {
// A job that will run every minute
},
},
],
},
};
TwitterBotServer({
botManager: new TwitterBotManager(bots),
});
If you want to keep references to the bots and manager you can do that too:
import TwitterBotServer, {
TwitterBotManager,
TwitterBot,
} from "twitter-bot-manager";
const firstBot = new TwitterBot({
name: "My First Bot",
consumer_key: "FIRST_BOT_APP_CONSUMER_KEY",
consumer_secret: "FIRST_BOT_APP_CONSUMER_SECRET",
token: "FIRST_BOT_APP_ACCESS_TOKEN",
token_secret: "FIRST_BOT_APP_ACCESS_TOKEN_SECRET",
eventActions: (event, self) => {
// A function to handle incoming Twitter events
},
jobs: [
{
interval: "* * * * *",
jobAction: (self) => {
// A job that will run every minute
},
},
],
});
const secondBot = new TwitterBot({
name: "My Second Bot",
consumer_key: "SECOND_BOT_APP_CONSUMER_KEY",
consumer_secret: "SECOND_BOT_APP_CONSUMER_SECRET",
token: "SECOND_BOT_APP_ACCESS_TOKEN",
token_secret: "SECOND_BOT_ACCESS_TOKEN_SECRET",
eventActions: (event, self) => {
// A function to handle incoming Twitter events
},
jobs: [
{
interval: "* * * * *",
jobAction: (self) => {
// A job that will run every minute
},
},
],
});
const bots = { firstBot, secondBot };
const manager = new TwitterBotManager(bots);
TwitterBotServer({
botManager: manager,
});
This will eventually become unmaintainable, so I suggest writing and exporting your bots in another file, and importing them into your entry file.
Protect your keys
Seriously, protect your keys!! All the examples have been written with the keys directly pasted into the code, but you shouldn't do that, unless you want your keys stolen and your bot hacked, especially if your project is going to be committed to GitHub or deployed to the web.
Instead, you should create some sort of .env or a keys file that will be added to your .gitignore!
Obviously, the key's file will never be committed, so you'd only use that locally, and in a production environment, you'd be forced to use environment vars.
For example:
import TwitterBotServer, { TwitterBot } from "twitter-bot-manager";
const accountName = "MyFirstBot";
const eventActions = (event, self) => {
console.log("Incoming event for " + accountName);
console.log(event);
};
let bot;
if (process.env.NODE_ENV === "production") {
bot = new TwitterBot({
name: accountName,
consumer_key: process.env[accountName + "_consumer_key"],
consumer_secret: process.env[accountName + "_consumer_secret"],
token: process.env[accountName + "_token"],
token_secret: process.env[accountName + "_token_secret"],
eventActions: eventActions,
});
} else {
const keys = require("./keys").default;
bot = new TwitterBot({
name: accountName,
consumer_key: keys[accountName]["consumer_key"],
consumer_secret: keys[accountName]["consumer_secret"],
token: keys[accountName]["token"],
token_secret: keys[accountName]["token_secret"],
eventActions: eventActions,
});
}
TwitterBotServer({
account,
});
Where keys.js is:
const keys = {
MyFirstBot: {
consumer_key: "",
consumer_secret: "",
token: "",
token_secret: "",
},
};
export default keys;
Roadmap
See the open issues for a list of proposed features (and known issues).
Contributing
Contributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are greatly appreciated.
- Fork the Project
- Create your Feature Branch (
git checkout -b feature/AmazingFeature
) - Commit your Changes (
git commit -m 'Add some AmazingFeature'
) - Push to the Branch (
git push origin feature/AmazingFeature
) - Open a Pull Request
License
Distributed under the MIT License. See LICENSE
for more information.