ExGram alternatives and similar packages
Based on the "Chatting" category.
Alternatively, view ExGram alternatives based on common mentions on social networks and blogs.
InfluxDB - Purpose built for real-time analytics at any scale.
Do you think we are missing an alternative of ExGram or a related project?
Popular Comparisons
README
ExGram
ExGram is a library to build Telegram Bots, you can use the low-level methods and models, or use the really opinionated framework included.
Installation
Add ex_gram
as dependency in mix.exs
def deps do
[
{:ex_gram, "~> 0.24"},
{:tesla, "~> 1.2"},
{:hackney, "~> 1.12"},
{:jason, ">= 1.0.0"}
]
end
See the next sections to select a different HTTP adapter or JSON engine.
HTTP Adapter
You should add Tesla or Maxwell http adapter, by default it will try to use the Tesla adapter, this are the defaults:
On deps:
{:tesla, "~> 1.2"},
{:hackney, "~> 1.12"}
- If you want to use Gun:
On deps:
{:tesla, "~> 1.2"},
{:gun, "~> 1.3"}
On config:
config :tesla, adapter: Tesla.Adapter.Gun
- If you prefer maxwell instead of tesla:
On deps:
{:maxwell, "~> 2.3.1"},
{:hackney, "~> 1.12"},
On config:
config :ex_gram, adapter: ExGram.Adapter.Maxwell
config :maxwell, default_adapter: Maxwell.Adapter.Hackney
JSON Engine
By default ExGram will use Jason
engine, but you can change it to your prefered JSON engine, the module just have to expose encode/2, encode!/2, decode/2, decode!/2
You can change the engine in the configuration:
config :ex_gram, json_engine: Poison
Configuration
There are some optional configuration that you can add to your config.exs
:
Token
config :ex_gram, token: "TOKEN"
This configuration will be used by default, but you can specify on every call a token or a bot to use.
If you use the framework, you will need to add ExGram
and your bot (let's say it's MyBot
) to your application:
children = [
ExGram, # This will setup the Registry.ExGram
{MyBot, [method: :polling, token: "TOKEN"]}
]
Configure Tesla middlewares
If you are using the Tesla
adapter, you can add Tesla
middlewares to ExGram
via config file. Add to your config:
config :ex_gram, ExGram.Adapter.Tesla,
middlewares: [
{Tesla.Middleware.BaseUrl, "https://example.com/foo"}
]
The middlewares
list will be loaded in the ExGram.Adapter.Tesla
module.
In case you want to use a middleware that requires a function or any
invalid element for a configuration file, you can define a function in
any module that returns the Tesla configuration. Then put the {m, f,
a}
in the configuration file, for example:
# lib/tesla_middlewares.ex
defmodule TeslaMiddlewares do
def retry() do
{Tesla.Middleware.Retry,
delay: 500,
max_retries: 10,
max_delay: 4_000,
should_retry: fn
{:ok, %{status: status}} when status in [400, 500] -> true
{:ok, _} -> false
{:error, _} -> true
end}
end
end
And in the config file:
# config/config.exs
config :ex_gram, ExGram.Adapter.Tesla,
middlewares: [
{TeslaMiddlewares, :retry, []}
]
Take into account that the defined function has to return a two-tuple as the Tesla config requires.
Framework Usage
This section will show how to use the opinionated framework ex_gram
for telegram bots!
Creating a bot!
Creating a bot is pretty simple, you can use the mix bot.new
task to setup your bot. For example:
$ mix new my_bot --sup
$ cd my_bot
Add and setup ExGram
and it's adapters in your project as shown in the Installation section. After that, get the project deps and run the bot new task:
$ mix deps.get
$ mix bot.new
You will get a message like this:
You should also add ExGram and MyBot.Bot as children of the application Supervisor,
here is an example using polling:
children = [
ExGram,
{MyBot.Bot, [method: :polling, token: token]}
]
This is basically telling you to configure the project as shown in the Configuration section. Get your token and put ExGram
and MyBot.Bot
under the Application
.
Now you are ready to run the bot with mix run --no-halt
and go to Telegram and send your bot the command /start
.
How to handle messages
If you followed the Creating a bot! section you should see a handle/2
function in your MyBot.Bot
module that looks like this:
def handle({:command, "start", _msg}, context) do
answer(context, "Hi!")
end
The handle/2
function receives two arguments:
The first argument is a tuple that changes depending on the update. In this case we are expecting a command called
start
in Telegram, this means a/start
message. This type of commands can be sent next to a message, for example/start Well hello
, in this cases theWell hello
text will arrive to the third element of the tuple named_msg
(because we are ignoring it right now). In case no text is given an empty string will arrive in the third element.The second argument is a map called Context (
%ExGram.Cnt{}
) with information about the update that just arrived, with information like the message object and internal data thatExGram
will use to answer the message. You can also save your own information from your own middlewares in the:extra
key using theadd_extra
method.
This are the type of tuples that handle/2
can receive as first parameter:
{:command, key, message}
→ This tuple will match when a command is received{:text, text, message}
→ This tuple will match when plain text is sent to the bot (check privacy mode){:regex, key, message}
→ This tuple will match if a regex is defined at the beginning of the module{:location, location}
→ This tuple will match when a location message is received{:callback_query, callback_query}
→ This tuple will match when a Callback Query is received{:inline_query, inline_query}
→ This tuple will match when an Inline Query is received{:edited_message, edited_message}
→ This tuple will match when a message is edited{:message, message}
→ This will match any message that does not fit with the ones described above{:update, update}
→ This tuple will match as a default handle
Sending files
ExGram
lets you send files by id (this means using files already uploaded to Telegram servers), providing a local path, or with the content directly. Some examples of this methods for sending files:
ExGram.send_document(chat_id, document_id) # By document ID
ExGram.send_document(chat_id, {:file, "path/to/file"}) # By local path
ExGram.send_document(chat_id, {:file_content, "FILE CONTENT", "filename.txt"}) # By content
This three ways of sending files works when the API has a file field, for example send_photo
, send_audio
, send_video
, ...
Library Usage
Sometimes you just want to be able to send messages to some channel, or you don't like the way the framework works and want to be your own manager of the messages flows. For that cases, the low level API allows you to use the ex_gram
library as raw as possible.
You can configure ex_gram
in config.exs
as explained in the Configuration section (you don't need to add anything to the application if you don't want to use the framework) and just use the low level API, for example:
ExGram.send_message("@my_channel", "Sending messages!!!")
Alternatively, you can not configure ex_gram at all (or use this to use different bots, having one configured or not), and use the extra parameter token
:
ExGram.send_message("@my_channel", "Sending messages!!!", token: "BOT_TOKEN")
If you want to know how the low level API is designed and works, you can read the next section.
Low level API
All the models and methods are equal one to one with the models and methods defined on the Telegram Bot API Documentation!
Models
All the models are inside of the ExGram.Model
module. You can see all the models in lib/ex_gram.ex
file, for example User
:
model(User, [
{:id, :integer},
{:is_bot, :boolean},
{:first_name, :string},
{:last_name, :string},
{:username, :string},
{:language_code, :string}
])
Also, all the models have the type t
defined, so you can use it on your typespecs or see their types inside of an IEx console:
>>> t ExGram.Model.User
@type t() :: %ExGram.Model.User{
first_name: String.t(),
id: integer(),
is_bot: boolean(),
language_code: String.t(),
last_name: String.t(),
username: String.t()
}
Methods
All the methods are inside of the ExGram
module, they are like the documentation ones but in snake_case instead of camelCase.
If a method have mandatory arguments will be the arguments (in order that are defined on the documentation) to the method, all the optional values will go in the last argument as keyword list.
Also, the parameters must be of the types defined on the documentation (if multiple types, it must be one of them), and the method will return the model assigned of the one in the documentation. If you want to see the parameters and types that gets and returns a method, you can use the h
method in an Iex
instance:
>>> h ExGram.send_message
def send_message(chat_id, text, ops \\ [])
@spec send_message(
chat_id :: integer() | String.t(),
text :: String.t(),
ops :: [
parse_mode: String.t(),
disable_web_page_preview: boolean(),
disable_notification: boolean(),
reply_to_message_id: integer(),
reply_markup:
ExGram.Model.InlineKeyboardMarkup.t()
| ExGram.Model.ReplyKeyboardMarkup.t()
| ExGram.Model.ReplyKeyboardRemove.t()
| ExGram.Model.ForceReply.t()
]
) :: {:ok, ExGram.Model.Message.t()} | {:error, ExGram.Error.t()}
All the methods have their unsafe brother with the name banged(!) (get_me!
for the get_me
method) that instead of returning {:ok, model} | {:error, ExGram.Error}
will return model
and raise if there is some error.
For example, the method "getUpdates" from the documentation will be get_updates
, and this one takes 4 optional parameters, we'll use on the example the parameters offset
and limit
:
ExGram.get_updates(offset: 123, limit: 100)
Another example, the method "sendMessage" it's send_message
, this one have two mandatory parameters, chat_id
(either an integer or a string) and text
(a string), and 5 optional parameters:
ExGram.send_message("@rockneurotiko", "Hey bro! Checkout the ExGram library!", disable_notification: true)
Extra options
All the methods have three extra options:
debug
: When true it will print the HTTP request response.token
: It will use this token for the requestbot
: It will search onRegistry.ExGram
thebot
name to extract the token. This registry is setted up byExGram
and all the bots made by the framework will register on it.
Note: Only one of token
and bot
must be used.
How it's made?
There is a python script called extractor.py
, it uses the telegram_api_json
project that scrapes the Telegram Bot API documentation and provides a JSON with all the information, check the project description if you want to create your own projects that uses an standarized file to auto-generate the API.
This scripts uses the JSON description and prints to the stdout the lines needed to create all the methods and models, this auto generated lines uses two macros defined on lib/ex_gram/macros.ex
: method
and model
.
Custom types defined
:string
->String.t()
:int
or:integer
->integer
:bool
or:boolean
->boolean
:file
->{:file, String.t()}
{:array, t}
->[t]
- Any ExGram.Model
Model macro
Parameters:
- Name of the model
- Properties of the model, it's a list of tuples, where the first parameter is the name of the property, and the second one is the type.
This macro is the simple one, just create a module with the first name passed and use the params to create the struct and typespecs.
Method macro
Parameters:
- Verb of the method (
:get
or:post
) - Name of the method as string, it will be underscored.
- The parameters of the method, this is a list of tuples, the tuples contains:
- Name of the parameters
- Type(s) of the parameter, it is a list of types, if there are more than one type on the list, it is expected one of them.
- An optional third parameter (always
:optional
) to set that parameter as optional - The type to be returned, it can be a model.
The macro will create two methods, one that will return tuple ok|error, and a banged(!) version that will raise if there are some error.
This methods do some stuff, like retrieving the token, check the parameters types, set up the body of some methods/verbs (specially the ones with files), call the method and parse the result.
Creating your own updates worker
The ExGram framework use updates worker to "receive" the updates and send them to the dispatcher, this is the first parameter that you provide to your bot, the ones currently are :polling
that goes to the module ExGram.Updates.Polling
for polling updates and :noup
that uses ExGram.Updates.NoUp
that do nothing (great for some offline testing). Sadly the webhook and test worker are on the way.
But you can implement your own worker to retrieve the updates as you want!
The only specs are that start_link
will receive {:bot, <pid>, :token, <token>}
, the PID is where you should send the updates, and the token is that specific token so your worker will be able to use it to retrieve the updates.
Whenever you have and update ExGram.Model.Update
, send it to the bot's PID like: {:update, <update>}
with Genserver.call
.
You can see the code of ExGram.Updates.Polling
.