README.md

# BotEx

Bot development core for `Elixir`

# How it works
The core is built using three key concepts:
- updaters - the main task is to receive a message and send it to the handler
- middleware - receive a message and transform it in some way. The first in the chain should implement the behavior `BotEx.Behaviours.MiddlewareParser`, all next - `BotEx.Behaviours.Middleware`
- handlers - process the message and interact with the user. Each handler must implement a behavior `BotEx.Behaviours.Handler`

# Existing libs:
- [telegram](https://github.com/bot-ex/botex-telegram)

# How to start:
  
  ```elixir
  #mix.exs
  def deps do
    [
      {:botex, "~> 0.1"}
    ]
  end

 #full available config reference
 #this values set to default
 config :bot_ex,
    menu_path: "config/menu.exs",
    routes_path: "config/routes.exs",
    short_map_path: "config/short_map.exs",
    after_start: [],
    show_msg_log: true,
    analytic_key: nil,
    middlware: [],
    handlers: [],
    bots: []
  ```

```bash
touch config/menu.exs
```
## Example `config`
```elixir
 config :bot_ex,
    middlware: [
      my_bot: [
        MyBot.Middleware.MessageTransformer,
        MyBot.Middleware.Auth
      ]
    ],
    handlers: [
      my_bot: [
        {MyBot.Handlers.Start, 1} # {module, count worker processes in pool}
      ]
    ],
    bots: [:my_bot]
  ```

## Example `menu.exs`
```elixir
%{
   "main_menu" => %BotEx.Models.Menu{
     buttons: [
       [
         %BotEx.Models.Button{
           action: "some",
           data: "data",
           module: MyBot.Handlers.Start.get_cmd_name(),
           text: "This is button"
         }
       ]
     ]
   }
 }
```
# Routing
Routes create from defined in config handlers. Each handler have function `get_cmd_name/0` that return command name for this handler. When user call `/start` command, router find module for handle this message by answer `get_cmd_name/0` value.

Optionally you can create file `routes.exs` and redefine or add aliases for your commands

### Example `routes.exs`
```elixir
%{
  :my_bot:
    %{"s" => MyBot.Handlers.Start}
}
```
Also you can create file `short_map.exs` that contains text aliases for command

### Example `short_map.exs`
```elixir
%{
  :my_bot:
  %{
    "i" => {MyBot.Handlers.Start.get_cmd_name(), "some action"}
  }
}
```

## Example `Updater`

```elixir
defmodule MyBot.Updaters.MySource do
  @moduledoc false

  use GenServer

  alias BotEx.Routing.Handler

  def child_spec(opts) do
    %{
      id: __MODULE__,
      start: {__MODULE__, :start_link, [opts]},
      type: :worker
    }
  end

  @spec start_link(any) :: :ignore | {:error, any} | {:ok, pid}
  def start_link(state \\ []) do
    GenServer.start_link(__MODULE__, state, name: __MODULE__)
  end

  @spec init(any) :: {:ok, :no_state}
  def init(_opts) do
    cycle()
    {:ok, :no_state}
  end

  defp cycle() do
    Process.send_after(self(), :get_updates, 1000)
  end

  @doc """
  Fetch any messages from your source
  """
  @spec handle_info(:get_updates, map()) :: {:noreply, map()}
  def handle_info(:get_updates, state) do
    # fetch any messages from your source
    msgs = []
    Handler.handle(msgs, :my_bot)
    cycle()
    {:noreply, state}
  end
end
```

## Example `MessageTransformer`

```elixir
defmodule MyBot.Middleware.MessageTransformer do
  @behaviour BotEx.Behaviours.MiddlewareParser

  alias BotEx.Models.Message

  @spec transform({binary(), binary(), binary(), map()}) ::
          Message.t()
  def transform({command, action, text, _user} = msg) do
    %Message{
      msg: msg,
      text: text,
      date_time: Timex.local(),
      module: command,
      action: action,
      data: nil,
      from: :my_bot
    }
  end
end
```
## Example `Middleware`

```elixir
defmodule MyBot.Middleware.Auth do
  @behaviour BotEx.Behaviours.Middleware

  alias BotEx.Models.Message

  @spec transform(Message.t()) :: Message.t()
  def transform(%Message{msg: {__, _, _, %{"id" => id} = user}} = msg) do
    %Message{msg | user: user, user_id: id}
  end
end
```

## Example `Handler`
```elixir
defmodule MyBot.Handlers.Start do
  @moduledoc false

  use BotEx.Handlers.ModuleHandler
  use BotEx.Handlers.ModuleInit

  alias BotEx.Models.Message

  def get_cmd_name, do: "start"

  @doc """
  Message handler
  ## Parameters
  - msg: incoming `BotEx.Models.Message` message.
  - state: current state
  return new state
  """
  @spec handle_message(Message.t(), State.t()) :: {:noreply, State.t()}
  def handle_message(%Message{chat_id: ch_id}, state) do
    MyBotApi.send_message(ch_id, "Hello")

    {:noreply, state}
  end
end

```