Phil is a TypeScript/Node.js Discord bot designed specifically for the Official HiJack Server. Originally, the bot started out with two purposes:
- Reduce the number of bots that we were using for one-off purposes (having a bot on the server just to get a single function from them);
- Manage the daily writing prompts, both allowing users to suggest new prompts and then posting the prompt for the day automatically.
Over time, Phil's functionality and usefulness began to grow and he's been given more and more duties on the server.
Phil also served as a useful reentry point to working with JavaScript. It's been a number of years since I've worked on a large JavaScript project, and I'd never really worked with Node.js before, so Phil was a great learning project for figuring out how to structure a Node.js project, how to run a Node.js project, how to deploy one, and then how to work with the latest in ECMAScript 7. Over the various versions, I've used Phil as a vehicle to expand and update my knowledge of JavaScript, introducing for the first time (to me) such new tools as: await/async, TypeScript, and yarn.
Phil is designed to be run/hosted on Heroku. Running him locally is also possible as part of the Heroku Toolbelt. There are a couple of dependencies you'll need to manually secure in order to run him, but they're all pretty standard run-of-the-mill.
- Node.js
- Heroku CLI (and Heroku account)
- yarn
- Docker
All of his other dependencies are secured through yarn and can be done by running yarn install
.
To set up the database, you'll want to create a new table in Postgres and then run all of the .sql files that are in the database/ directory in order. It's not too difficult, and each subsequent file is meant as a patch against the base one. Pretty straightforward.
Lastly, if you're going to run locally (or remotely) you'll need to set up your environment variables using various IDs from Discord. You can't run Phil without having all of the environment variables declared. The file .env.template
provides all of the environment variables that you'll need to provide in order for Phil to start up.
You'll need to register a bot account for Discord. That's pretty simple and can be done here. You'll also need a Heroku account. A free one is more than enough, though I'd recommend putting in your credit card information. Phil will never require services that will charge you any money, and Heroku definitely seems to be on the up-and-up about it. The only reason I suggest registering your credit card is it'll massively bump the number of hours that your account can have stuff hosted to 1,000 hours per month, which 31 days * 24 hours/day = 744 hours per month (744 < 1000, usually), which guarantees you that Phil will never go down partway through the month.
Here's (more or less) a flow to get Phil started locally. It'll probably need some individual tweaking, but this gives you the general idea I think:
git clone [email protected]:ahlec/Phil.git
cd Phil
cp .env.template .env
[take a moment to configure .env]
yarn install
yarn db-migrate
yarn start
This will launch Phil running from your local machine. Subsequently, you can get Phil running using just yarn start
.
Phil is now written entirely in TypeScript, which means that we need to transpile our TypeScript code to JavaScript in order for changes to show up. The process of transpilation will happen automatically before starting up when using yarn start
. When it comes time to push him to Heroku and have him running on the cloud rather than on your local machine, you simply need to git push heroku master
and it'll push everything in your git repo to the live environment on Heroku. You only need to push your TypeScript file changes, however. The JavaScript gets transpiled automatically when pushed to Heroku as part of the build process. It WON'T push your .env file, though, so you'll need to configure your environment variables on Heroku. Easiest way to do that is just using their dashboard to modify them.
Phil is made up of a couple of different components. I won't go into all of the details about what makes Phil tick, because more or less it's straightforward (well, as straight-forward as reading through anybody else's source code ever is). However, there are some constructs that I'd like to touch upon.
Commands. These are the most obvious of the components. Commands are any function that a user is capable of invoking by using the command prefix + the name of a command (ie p!prompt
or p!help
if the command prefix is p!
). Commands can only be run in server messages (they cannot be run in direct messages) because they require the command prefix in order to be interpreted; starting with Version 13 the command prefix is configured exclusively on a per-server basis, so commands could not be used in direct messages, which exist outside of servers.
Chronos. "Chronos" is actually the plural; an individual one of these is a chrono. Chronos are Phil's equivalent of a chron job. It's something that Phil will run at a certain point in time to perform some kind of action. Some of these can be posting something on a regular interval in the chat; others can be performing some kind of cleanup operation on a routine basis. Unlike commands, individual chronos cannot be triggered or named by any user, and they happen even if nobody says anything (they're based on time rather than on user input). These grew out of the original foundation for posting a new daily prompt. Added in Version 2.
Direct Message Processors. Direct message processors do pretty much exactly what you would expect them to do: they process direct messages. A number of tasks with Phil, such as suggesting new prompts for a suggestion bucket or setting one's timezone, are done through direct messages. Public message commands are capable of offloading processes to direct messages, where processors will pick up the work from there without the need to listen for commands. Only one processor is capable of being active at a time, and there is an intrinsic hierarchy of precedence if multiple processors were to vie for the status of being active at once. Direct message processors were introduced in Version 13, having been reworked from an older system known as Analyzers.