Quickstart | Configuration | Plugins | Adding plugins
An IRC bot that supports plugins. It was a project for learning Erlang. I'm sure the code is far from idiomatic Erlang, but it was tons of fun nonetheless.
%% From the erlang shell
%% start useless bot and connect to IRC server
> application:start(useless_irc).
%% join #testme IRC channel
> useless_irc:join("#testme").
%% login as user test
> useless_irc:login("test").
The following variables are configurable (located in ebin/useless_irc.app):
- IRC server ip address
- IRC server port
- IRC channel to join
- IRC nick to use
- you can add other configurations like tokens, keys etc..
Commands to the bot use a prefix of ,
(i will make that configurable as an environment variable).
In your IRC client of choice, you can type ,help
and that will iterate over all the installed plugins and provide some basic help on how to invoke them.
[11:56:42] <test> Plugin: timer with prefix : timer
[11:56:42] <test> Plugin: magic_8ball with prefix : 8ball
[11:56:42] <test> Plugin: weather with prefix : weather
[11:56:42] <test> Plugin: insults with prefix : insult
[11:56:42] <test> Plugin: julia with prefix : julia
Invoke the weather plugin by using ,weather
prefix. If you leave any arguments to the plugin the plugin will return you some help on how to use it as follows:
%% from your IRC client
nick> ,weather
bit> weather [forecast|conditions|astro|help] location
Invoking the plugin with help as the argument will give you the same output.
The weather plugin supports several different options for a given location (default is 95129 zipcode):
-
forecast: returns the current forecast
nick> ,weather forecast 95129 bot> Forecast for Sunday is Abundant sunshine. High 84F. Winds NW at 10 to 15 mph.
-
conditions: returns the weather conditions
nick> ,weather conditions 95129 bot> Last Updated on June 1, 8:35 AM PDT. Partly Cloudy. Temperature: 59.7 F (15.4 C). Precipitation today: 0.00 in (0 mm). Wind Calm
-
astro: returns some astronomical data
nick> ,weather astro 95129 bot> Percentage of moon illuminated: 41, Age of moon: 7, Phase: Waxing Crescent, Sunrise: 6:08, Sunset: 20:01
The location can be specified in several ways:
- using a zipcode : 95129, 60651
- a string such as
chicago
,iowa city
. You might want to be more specific in case there are multiple matches, for example:ca sunnyvale
You can invoke the insult plugin in the channel or you can send the bot a privage message, in order to generate a fresh insult for your pleasure.
%% in your IRC client as nick testnick in channel
,insult
%% IRC bot will generate a fresh insult for you
testnick is a lumpish thimbleful of fly-bitten craptacular carpet droppings
%% You can send the bot a private message as well, uselessbot is bot nick in this example
/msg uselessbot ,insult
%% the bot will respond with
You are a low-quality accumulation of vain bat toenails
Ask the magic 8 ball a question by prefixing a question with ,8ball
followed by your question.
For example:
%% in your IRC client ask the bot a question
,8ball should i learn Elixir ?
%% the magic 8 ball responds
As I see it, yes.
The timer plugin allows a user to set a timer for a time period expressed as hh:mm:ss
followed by a reminder message. The bot will send a privage message to that user after the timer expires.
%% send a private message to the bot to set a timer for 10 secs 0:0:10
%% and when the timer expires to remind me to 'wake me up'
/msg test ,timer 0:0:10 wake me up
%% after 10 seconds the bot messages the nick with the message
wake me up
The Julia
plugin allows you to evaluate Julia language
expressions.
Each user is assigned a session, the session allows the user to keep history of previous bindings and evaluated expressions until the session expires.
Sessions expire due to lack of activity in a time period (currently 30 secs).
Currently, the plugin allows a maximum of 3 concurrent sessions.
%% send the julia plugin a command
nick> ,julia 1 + 1
%% bot create a session
bot> Working
bot> 2
%% bind a to 1:10
nick> ,julia a = 1:10
bot> 1:10
%% now use a
nick> ,julia sum(a)
bot> 55
%% multi-line output of squaring an vector
nick> ,julia map(i->i*i,[1,2,3,4])
nick> 4-element Array{Int64,1}:
nick> 1
nick> 4
nick> 9
nick> 16
%% create a function
nick> ,julia square(i) = i*i
bot> square (generic function with 1 method)
nick> ,julia map(square,[1,2,3,4])
nick> 4-element Array{Int64,1}:
nick> 1
nick> 4
nick> 9
nick> 16
Plugins are gen_servers that register with the bot. Plugins are required to implement a couple of messages in order to handle the input from the IRC client and forward the output back to the bot.
Each plugin can register itself with a prefix that is used to invoke it. For example:
%% register our timer service
useless_irc_services:register_service(timer,"timer",self(), node(),SERVER)
Registers plugin timer with a prefix of "timer" which can be invoked by ,timer
.
Each plugin uses the gen_server behaviour and receives requests from the bot using out-of-band messages. This means that the plugin needs to implement the handle_info
callback.
The bot sends a run request to the plugins in the following tuple format:
{run, Request, Chan, User, From}
- run : is an atom to indicate that the bot is requesting the plugin to run a command
- Request : the actual string for the command. For example in the
julia
plugin the Request could be something likefoldl(+,1:10)
- Chan : is the current channel if the request was made to the channel. When a private message was received for the request, Chan is an empty string.
- User : is the current nick that is making the request.
- From : is the pid of the bot, to which responses from the plugins are sent.