[](https://hexdocs.pm/phoenix_gen_api)
[](https://hex.pm/packages/phoenix_gen_api)
# PhoenixGenApi
The library helps quickly develop APIs for client, the library is based on Phoenix Channel.
Developers can add or update APIs in runtime from other nodes in the cluster without restarting or reconfiguring the Phoenix app.
In this case, the Phoenix app will take on the role of an API gateway.
The library can use with [EasyRpc](https://hex.pm/packages/easy_rpc) and [ClusterHelper](https://hex.pm/packages/cluster_helper) for fast and easy to develop a dynamic Elixir cluster.
## Concept
After received an event from client(in handle_in callback of Phoenix Channel), the event will be passed to PhoenixGenApi to find target API & target node to execute then get result for response to client.
For service nodes (target node), the libray support some basic strategy for selecting node (:choose_node_mode) like: :random, :hash, :round_robin.
Supported :sync, :async, :stream for request/response to client.
Supported basic check type & permission.
## Installation
The package can be installed
by adding `phoenix_gen_api` to your list of dependencies in `mix.exs`:
```Elixir
def deps do
  [
    {:phoenix_gen_api, "~> 0.2.1"}
  ]
end
```
Note: You can use [`:libcluster`](https://hex.pm/packages/libcluster) to build a Elixir cluster.
## Usage
### Remote Node (optional)
Add config to your `config.exs` file to mark this is remote node.
```Elixir
config :phoenix_gen_api, :client_mode, true
```
Declare a module for support PhoenixGenApi can pull config.
Example:
```Elixir
defmodule MyApp.GenApi.Supporter do
  alias PhoenixGenApi.Structs.FunConfig
  @doc """
  Support for remote pull general api config.
  """
  def get_config() do
    {:ok, my_fun_configs()}
  end
  @doc """
  Return list of %FunConfig{}.
  """
  def my_fun_configs() do
    [
      %FunConfig{
        request_type: "get_data",
        service: "my_service",
        nodes: [Node.self()],
        choose_node_mode: :random,
        timeout: 5_000,
        mfa: {MyApp.Interface.Api, :get_data, []},
        arg_types: %{"id" => :string},
        response_type: :async
      }
    ]
  end
end
```
Note: You can add directly in runtime in gateway node without using client mode.
### Phoenix Node (Gateway node)
Add config for Phoenix can pull config from remote nodes(above) like:
```Elixir
# Config for general api, lib made by team.
config :phoenix_gen_api, :gen_api,
  service_configs: [
    # service config for pulling general api config.
    %{
      # service type
      service: :my_service,
      # nodes of service in cluster, need to connecto to get config
      nodes: [:"remote_service@test.local"], # or using MFA like: {ClusterHelper, get_nodes, [:my_api]}
      # module to get config
      module: MyApp.GenApi.Supporter,
      # function to get config
      function: :get_config,
      # args to get config, using for identity or check security.
      args: [:gateway_1],
    }
  ]
```
In Phoenix Channel you can add a lit of bit code like:
```Elixir
@impl true
def handle_in("phoenix_gen_api", payload, socket) do
  result =
    payload
    |> Map.put("user_id", socket.assigns.user_id) # avoid security issue.
    |> PhoenixGenApi.Executor.execute_params()
    case result do
      result = %Response{} ->
      # not a final result for async/stream call.
      push(socket, "gen_api_result", result)
    # request type is :none, no response.
    {:ok, :none} ->
      :ok
    end
  {:noreply, socket}
end
@impl true
def handle_info({:push, result}, socket) do
  push(socket, "phoenix_gen_api_result", result)
  {:noreply, socket}
end
def handle_info({:async_call, result = %Response{}}, socket) do
  push(socket, "phoenix_gen_api_result", result)
  {:noreply, socket}
end
# For receiving data from stream.
def handle_info({:stream_response, result}, socket) do
  push(socket, "gen_api_result", result)
  {:noreply, socket}
end
```
In this case, if need you can authenticate by using Phoenix framework.
Now you can start your cluster and test!
After start Phoenix app, PhoenixGenApi will auto pull config from remote node to serve client.
For test in Elixir you can use  [`phoenix_client`](https://hex.pm/packages/phoenix_client) to create a connection to Phoenix Channel.
You can push a event with content like:
```json
{
  "user_id": "user_1",
  "device_id": "device_1",
  "service": "my_service",
  "request_type": "get_data",
  "request_id": "test_request_1",
  "args": {
    "id": "test_data_id"
  }
}
```
Result like:
If is async/stream call you will receive a message like this:
```json
{
  "async": true,
  "error": "",
  "has_more": false,
  "request_id": "test_request_1",
  "result": null,
  "success": true
}
```
After that is a another message with result:
```json
{
  "async": false,
  "error": "",
  "has_more": false,
  "request_id": "test_request_1",
  "result": [
    {
      "id": "14e99227-512a-47b6-b6b1-2d4bc29ca13e",
      "name": "Hello World!"
    }
  ]
}
```
For better security you can overwrite user_id in server, using basic check permission or passing request info (user_id, device_id, request_id).
## Full Example
We will add a full example in the future.
## Planned Features
- [DONE]Add pool processes for save/limit resource.
- Sticky node.
- Rate limiter.
## Support AI agents & MCP
Run this command for update guide & rules from deps to repo for supporting ai agents.
```bash
mix usage_rules.sync AGENTS.md --all \
  --link-to-folder deps \
  --inline usage_rules:all
```
Run this command for enable MCP server
```bash
mix tidewave
```
Config MCP for agent `http://localhost:4114/tidewave/mcp`, changes port in `mix.exs` file if needed. Go to [Tidewave](https://hexdocs.pm/tidewave/) for more informations.