twitter-bot-manager
TypeScript icon, indicating that this package has built-in type declarations

1.0.0-beta-11 • Public • Published

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

  1. Ensure you have a Twitter Developer Account, if you don't, make one, cause this package is useless without it.

  2. 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.

  3. 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

  1. Init a new Node project, fill out the prompts.
mkdir my-really-cool-bot-project
cd my-really-cool-bot-project
npm init
  1. Install the bot-manager package
npm i twitter-bot-manager
  1. Setup the bot-server in your project's entry file
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",
   }),
});
  1. 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.

What's that second parameter, self, for?

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.

  1. Fork the Project
  2. Create your Feature Branch (git checkout -b feature/AmazingFeature)
  3. Commit your Changes (git commit -m 'Add some AmazingFeature')
  4. Push to the Branch (git push origin feature/AmazingFeature)
  5. Open a Pull Request

License

Distributed under the MIT License. See LICENSE for more information.

Readme

Keywords

none

Package Sidebar

Install

npm i twitter-bot-manager

Weekly Downloads

0

Version

1.0.0-beta-11

License

MIT

Unpacked Size

115 kB

Total Files

16

Last publish

Collaborators

  • christophergomez