Init 1

Dogfight! - Elixir server

posted on 2025 Jan 01

Starting with the server side of the game, Elixir is pretty neat and allows for various ways to implement a distributed TCP server. Going for the most simple and straight forward approach, leaving refinements and polish for later. After more than 3 years of using Elixir full time, I believe it’s really one of the cleanest backend languages around, extremely productive and convenient (pattern matching in Elixir slaps), there are two points that make it stand-out and interesting to try for this little project:

  1. Concurrency and Scalability: Built on the Erlang VM (BEAM), which is known for its ability to handle a large number of concurrent processes with low latency, a real outstanding piece of engineering created by the major telecom experts on earth. This makes Elixir highly suitable for applications that require high concurrency and scalability, such as real-time systems and web applications.

  2. Fault Tolerance: Elixir inherits Erlang’s “let it crash” philosophy, which encourages developers to write systems that can recover from failures automatically, this results in highly resilient applications that can continue to operate even when parts of the system fail. Each process operates in isolation and supervision trees allow for automatically recovery; potentially, it’s possible to ship code really fast without the need of being too defensive and tune the system based on the evolution of the use-cases.

The main idea for the application is to write a highly concurrent and distributed TCP based system to handle players, for the first iteration:

  • A TCP entry point that acts as an acceptor process, its only purpose will be to accept new connecting clients and spawn process inside the cluster to handle the lifetime of the connection (i.e. the game)
  • GenServer is the main abstraction (kinda low level but easy to use for a first run) that will be used to represent
    • A game server, this will be the entity representing the main source of truth for connected clients, it’s basically an instantiated game. For the first iteration we will have a single running game server; subsequently, it will extended to be spawned on-demand as part of a hosting game logic from players requesting it
    • Each connecting player, once the acceptor complete the handshake, it will demand the managing of the client to its own process
  • The main structures exchanged between server and clients will be
    • A game state, will contain all the players, initially capped to a maximum to keep things simple, and each player will have a fixed amount of projectiles to shoot
    • Actions, all of which will be directional in nature, such as
      • MOVE UP
      • MOVE DOWN
      • MOVE LEFT
      • MOVE RIGHT
      • SHOOT (this will follow the direction of the projectile, which in turns will coincide with the player direction)
  • Each player process and game process will be spawned inside a cluster on multiple nodes, with location transparency and message passing this will be a breeze

A very basic TCP server acceptor

We’re gonna use :gen_tcp from the builtin Erlang library, starting the listen loop in a dedicated Task is enough for the time being, management of newly connected clients will be demanded to their own proper GenServer.

Let’s start with something really bare-bone, basically a slight variation of the official docs example is all we need to get started

defmodule Dogfight.Server do
  @moduledoc """
  Main entry point of the server, accepts connecting clients and defer their
  ownership to the game server. Each connected client is handled by a
  `GenServer`, the `Dogfight.Player`. The `Dogfight.Game.Server` governs the
  main logic of an instantiated game and broadcasts updates to each registered
  player.
  """
  require Logger

  alias Dogfight.ClusterServiceSupervisor

  def listen(port) do
    # The options below mean:
    #
    # 1. `:binary` - receives data as binaries (instead of lists)
    # 2. `packet: :raw` - receives data as it comes (stream of binary over the wire)
    # 3. `active: true` - non blocking on `:gen_tcp.recv/2`, doesn't wait for data to be available
    # 4. `reuseaddr: true` - allows us to reuse the address if the listener crashes
    #
    {:ok, socket} =
      :gen_tcp.listen(port, [:binary, packet: :raw, active: true, reuseaddr: true])

    Logger.info("Accepting connections on port #{port}")
    accept_loop(socket)
  end

  defp accept_loop(socket) do
    with {:ok, client_socket} <- :gen_tcp.accept(socket),
         player_id <- Dogfight.IndexAgent.next_id(),
         player_spec <- player_spec(client_socket, player_id),
         {:ok, pid} <- Horde.DynamicSupervisor.start_child(ClusterServiceSupervisor, player_spec) do
      Logger.info("Player #{player_id} connected")
      Dogfight.Game.Server.register_player(pid, player_id)
      :gen_tcp.controlling_process(client_socket, pid)
    else
      error -> Logger.error("Failed to accept connection, reason: #{inspect(error)}")
    end

    accept_loop(socket)
  end

  defp player_spec(socket, player_id) do
    %{
      id: Dogfight.Player,
      start: {Dogfight.Player, :start_link, [player_id, socket]},
      type: :worker,
      restart: :transient
    }
  end
end

The main interesting points are

  • accept_loop/1 is a recursive call, once a new client connect it dispatches it to a process and call itself again
  • if the accept call succeed, the first thing we do is to generate a basic ID (using a monotonic integer for the time being, this to keep compatibility with the original implementation of battletank). This is achieved with a very simple Agent (a builtin abstraction on top of the GenServer for handy state management).

    Something similar is more than enough to start

      defmodule Dogfight.IndexAgent do
        @moduledoc """
        Simple agent to generate monotonic increasing indexes for connecting players,
        they will be used as player ids as a first extremely basic iteration, later we
        may prefer to adopt some proper UUID.
        """
        use Agent
    
        def start_link(_initial_value) do
          Agent.start_link(fn -> 0 end, name: __MODULE__)
        end
    
        def next_id do
          Agent.get_and_update(__MODULE__, fn index ->
            {index, index + 1}
          end)
        end
      end
    
  • After the ID generation, we’re gonna use Horde dynamic supervisor to spawn processes that are automatically cluster-aware, the node where they will be spawned is completely abstracted away and doesn’t matter. (For a simpler solution we could even manually start the GenSever with a call to start_link/1, it’s really that simple).
  • The Dogfight.Game.Server.register_player/2 call basically add the pid of the newly instantiated process to the existing game server (as explained above, there will be a single global game server for the first version). This assumes that the game server is already running here, started at application level already.
  • Finally, we transfer the ownership of the connected socket to the newly spawned process through :gen_tcp.controlling_process/2

The player process

After the accept has been successfully performed, the player_spec/2 call provides all that’s required for the Horde.DynamicSupervisor to spawn a Dogfight.Player process somewhere in the cluster. Conceptually it’s nothing more than a connection state, and thus it’s pretty trivial. The registration of each PID to the game server can be a little cumbersome later on, and it may be a good opportunity to explore some pub-sub based solution such as the one shipped by Phoenix; for the time being it’s ok to keep things simple.

defmodule Dogfight.Player do
  @moduledoc """
  This module represents a player in the Dogfight game. It handles the player's
  connection, actions, and communication with the game server.
  """

  require Logger
  use GenServer

  alias Dogfight.Game.State, as: GameState
  alias Dogfight.Game.Action, as: GameAction

  def start_link(player_id, socket) do
    GenServer.start_link(__MODULE__, {player_id, socket})
  end

  def init({player_id, socket}) do
    Logger.info("Player connected, registering to game server")
    {:ok, %{player_id: player_id, socket: socket}}
  end

  def handle_info({:tcp, _socket, data}, state) do
    action = GameAction.decode!(data)
    Dogfight.Game.Server.apply_action(self(), action, state.player_id)
    {:noreply, state}
  end

  def handle_info({:tcp_closed, _socket}, state) do
    Logger.info("Player disconnected")
    {:noreply, state}
  end

  def handle_info({:tcp_error, _socket, reason}, state) do
    Logger.error("Player transmission error #{inspect(reason)}")
    {:noreply, state}
  end

  def handle_info({:update, game_state}, state) do
    send_game_update(state.socket, game_state)
    {:noreply, state}
  end

  defp send_game_update(socket, game_state) do
    :gen_tcp.send(socket, GameState.encode(game_state))
  end
end

Most of the handlers are the typical ones for a bare bone TCP connection, the only real addition is the send_game_update/2 call which forward a message to the connected client, in this case the binary encoded game state as the payload.

The game server

This is the main process that govern the state of a game, it is in essence a match which can be hosted by one of the player to invite others to join. At the moment this logic is yet to be implemented and it’s essentially skipped, there is a single game server started as part of the application startup.

The main responsibilities of the GenServer is to provide a single source of truth to all the connected clients:

  • Each connecting player, after having received an identifier, are registered to the game server (this happens in the acceptor TCP server via the register_player/2 call)
  • Once a player is registered, a ship is spawned in a random position for them via the GameState.spawn_ship/2 for the given ID.
  • Broadcast periodic updates of the game state to all connected clients to keep updating the GUI of each of them (currently set at 50 ms by @tick_rate_ms, but it’s not really important now)
defmodule Dogfight.Game.Server do
  @moduledoc """
  Represents a game server for the Dogfight game. It handles the game state and
  player actions. It is intended as the main source of truth for each instantiated game,
  broadcasting the game state to each connected player every `@tick` milliseconds.
  """
  require Logger
  use GenServer

  alias Dogfight.Game.State, as: GameState

  @tick_rate_ms 50

  def start_link(_) do
    GenServer.start_link(__MODULE__, %{}, name: __MODULE__)
  end

  @impl true
  def init(_) do
    game_state = GameState.new()
    schedule_tick()
    {:ok, %{players: [], game_state: game_state}}
  end

  def register_player(pid, player_id) do
    GenServer.call(__MODULE__, {:register_player, pid})
    GenServer.cast(__MODULE__, {:spawn_new_ship, player_id})
  end

  def apply_action(pid, action, player_index) do
    GenServer.cast(__MODULE__, {:apply_action, pid, action, player_index})
  end

  defp schedule_tick() do
    Process.send_after(self(), :tick, @tick_rate_ms)
  end

  @impl true
  def handle_info(:tick, state) do
    new_game_state = GameState.update(state.game_state)
    updated_state = %{state | game_state: new_game_state}
    broadcast_game_state(updated_state)
    schedule_tick()
    {:noreply, updated_state}
  end

  @impl true
  def handle_call({:register_player, pid}, from, state) do
    new_state =
      Map.update!(state, :players, fn players -> [{:player_id, %{pid: pid}} | players] end)

    {:reply, from, new_state}
  end

  @impl true
  def handle_cast({:spawn_new_ship, player_id}, state) do
    game_state =
      case GameState.spawn_ship(state.game_state, player_id) do
        {:ok, game_state} ->
          broadcast_game_state(%{state | game_state: game_state})
          game_state

        {:error, error} ->
          Logger.error("Failed spawning ship, reason: #{inspect(error)}")
          state.game_state
      end

    {:noreply, %{state | game_state: game_state}}
  end

  @impl true
  def handle_cast({:apply_action, _pid, action, player_index}, state) do
    game_state = GameState.apply_action(state.game_state, action, player_index)
    updated_state = %{state | game_state: game_state}
    broadcast_game_state(updated_state)
    {:noreply, updated_state}
  end

  defp broadcast_game_state(state) do
    Enum.each(state.players, fn {_pid, player} ->
      send(player.pid, state.game_state)
    end)
  end
end

The init/1 call is pretty cheap, no need to add an handle_continue/2 as it won’t really block. To be noted that the registration of a new player, performs a call and a cast. That because we want to make sure the state is updated before spawning the ship for the new player. Also there is no check yet for already existing ships, disconnections and reconnection etc. For all the unexpected cases we just log error or crash, it will be for a second pass to actually clean up and harden the code. It’s pretty simple and straight-forward.

The game state

The game state represents the main entities present in a game, it’s the main payload of information exchanged between clients and server to keep all the players in sync.

it is indeed pretty simple and definitely not optimized at the moment, there’s a number of things that can be improved and done differently, some of the fields are a carry over from the original C implementation which I’m gonna use as the main test-bed, but the aim is to then update the state for a better v2 representation. For the sake of simplicity, the first version will carry the following entities:

  • A list of players active in the game, for simplicity a maximum of 5 units is initially set
    • Each player is represented by
      • a vector 2D coordinate x and y representing the position in the screen
      • health points, initially set at 5
      • a direction, this can be (not exactly precise but work for a prototype)
        • :idle | :up | :down | :left | :right
      • an alive boolean flag
      • bullets, set to a maximum of 5, composed by
        • a vector 2D coordinate x and y representing the position in the screen
        • a direction, this will be aligned with the player direction
        • an active boolean flag
  • Power-ups, this will be spawned randomly on a time-basis, made of
    • a vector 2D coordinate x and y representing the position in the screen
    • a kind atom representing the type of effect it will have on the player that captures it, initially can be
      • :hp_plus_one | :hp_plus_three | :ammo_plus_one | nil

There are two more fields that are probably going to be deprecated pretty soon:

  • active_players
  • player_index, this last one basically represents which player to update on each client update. Technically not necessary as ideally the client should be able to just update the entire delta of each update without knowing this, it exists due to legacy reasons.

At the moment we’re not gonna handle scaling and screen size for each player, it is initially assumed that each player will play in a small 800x600 window.

defmodule Dogfight.Game.State do
  @moduledoc """
  Game state management, Ships and bullet logics, including collision.
  Represents the source of truth for each connecting player, and its updated
  based on the input of each one of the connected active players
  """

  alias Dogfight.Encoding.Helpers, as: Encoding
  alias Dogfight.Game.Action

  @type t :: %__MODULE__{
          players: [ship()],
          active_players: non_neg_integer(),
          player_index: non_neg_integer(),
          powerup: powerup()
        }

  @typep ship :: %{
           coord: vec2(),
           hp: integer(),
           direction: integer(),
           alive: boolean(),
           bullets: [bullet()]
         }

  @typep powerup :: %{
           coord: vec2(),
           kind: :hp_plus_one | :hp_plus_three | :ammo_plus_one | nil
         }

  @typep vec2 :: %{
           x: integer(),
           y: integer()
         }

  @typep bullet :: %{
           coord: vec2(),
           direction: integer(),
           active: boolean()
         }

  defstruct [:players, :active_players, :player_index, :powerup]

  @max_players 5
  @max_bullets 5
  @base_hp 5
  @screen_width 800
  @screen_height 600

  def new do
    %__MODULE__{
      player_index: 0,
      active_players: 0,
      powerup: %{coord: %{x: 0, y: 0}, kind: nil},
      players: Stream.repeatedly(&new_ship/0) |> Enum.take(@max_players)
    }
  end

  defp new_ship do
    %{
      coord: %{
        x: 0,
        y: 0
      },
      hp: 0,
      direction: :idle,
      alive: false,
      bullets: Stream.repeatedly(&new_bullet/0) |> Enum.take(@max_bullets)
    }
  end

  defp new_bullet do
    %{
      active: false,
      coord: %{
        x: 0,
        y: 0
      },
      direction: :idle
    }
  end

  # TODO move to a map instead of the array, keepeing as-is for the first
  # translation pass
  @spec spawn_ship(t(), integer()) :: {:ok, t()} | {:error, :dismissed_ship}
  def spawn_ship(game_state, index) do
    if Enum.at(game_state.players, index).alive do
      {:ok, game_state}
    else
      # TODO fix this monstrosity
      new_state = %{
        game_state
        | active_players: game_state.active_players + 1,
          players:
            Enum.with_index(game_state.players, fn
              player, ^index ->
                %{
                  player
                  | alive: true,
                    hp: @base_hp,
                    coord: %{x: :rand.uniform(@screen_width), y: :rand.uniform(@screen_height)},
                    direction: :up
                }

              other, _i ->
                other
            end)
      }

      {:ok, new_state}
    end
  end

  @spec update(t()) :: t()
  def update(game_state) do
    %{game_state | players: update_ships(game_state.players)}
  end

  defp update_ships(players) do
    Enum.map(players, &update_ship/1)
  end

  defp update_ship(%{alive: false} = ship), do: ship

  defp update_ship(%{alive: true} = ship) do
    bullets = Enum.map(ship.bullets, &update_bullet/1)

    %{
      ship
      | bullets: bullets
    }
  end

  defp update_bullet(%{active: false} = bullet), do: bullet

  defp update_bullet(%{active: true} = bullet) do
    case bullet.direction do
      :up ->
        %{
          bullet
          | coord: %{x: bullet.coord.x, y: bullet.coord.y - 6}
        }

      :down ->
        %{
          bullet
          | coord: %{x: bullet.coord.x, y: bullet.coord.y + 6}
        }

      :left ->
        %{
          bullet
          | coord: %{x: bullet.coord.x - 6, y: bullet.coord.y}
        }

      :right ->
        %{
          bullet
          | coord: %{x: bullet.coord.x + 6, y: bullet.coord.y}
        }

      _ ->
        bullet
    end
  end

  @spec apply_action(t(), Game.Action.t(), non_neg_integer()) :: t()
  def apply_action(game_state, action, player_index) do
    case action do
      direction when direction in [:up, :down, :left, :right] ->
        move_ship(game_state, player_index, direction)

      :shoot ->
        shoot(game_state, player_index)

      _ ->
        game_state
    end
  end

  defp move_ship(game_state, player_index, direction) do
    player_coord = Enum.at(game_state.players, player_index).coord

    player_coord = move_ship_coord(player_coord, direction)

    ships =
      Enum.with_index(game_state.players, fn
        player, ^player_index ->
          %{
            player
            | direction: direction,
              coord: player_coord
          }

        other, _i ->
          other
      end)

    %{
      game_state
      | players: ships
    }
  end

  defp move_ship_coord(%{x: x, y: y}, direction) do
    case direction do
      :up ->
        %{x: x, y: y - 3}

      :down ->
        %{x: x, y: y + 3}

      :left ->
        %{x: x - 3, y: y}

      :right ->
        %{x: x + 3, y: y}
    end
  end

  defp shoot(game_state, player_index) do
    %{
      game_state
      | players:
          Enum.with_index(game_state.players, fn
            player, ^player_index ->
              bullets = update_bullets(player.bullets, player)

              %{player | bullets: bullets}

            other, _i ->
              other
          end)
    }
  end

  defp update_bullets(bullets, player) do
    Enum.map_reduce(bullets, false, fn
      bullet, false when bullet.active == false ->
        {
          %{
           bullet
           | active: true,
             direction: player.direction,
             coord: %{x: player.coord.x, y: player.coord.y}
         }, true}

      bullet, updated ->
        {bullet, updated}
    end)
    |> elem(0)
  end
end

Binary encoding and decoding

TBD

References