DEV Community

Nick Vernij
Nick Vernij

Posted on • Updated on • Originally published at nickvernij.nl

Testing your Elixir + Phoenix + Postgres app with Github CI

Originally posted on nickvernij.nl

GitHub released their own CI recently, its UI and marketing look very fancy. I had to set up a new Elixir + Phoenix project and decided to give it a shot.

It took me a full morning to get a green check, because the default elixir template is very minimal and the documentation for creating workflows is quite a lot of reading to get through.

There are some nooks and crannies to get through, but when you get it to work it's very powerful. Essentially, the default Elixir workflow is fine for testing your average Elixir library, however a Phoenix app will need some additional setup and services. In my example I needed to set-up a PostgreSQL service.

First of all we want to set the MIX_ENV environment variable, so when we use ecto, it knows it should setup the test database. We can define global environment variables when defining the container in which our tests will run.

    container:
      image: elixir:1.9.1-slim
      env:
        MIX_ENV: test
Enter fullscreen mode Exit fullscreen mode

Setting up a database

But to use ecto we actually need a database. In the workflow definition we can define a list of services. These are docker images that are prerequisites for our our test steps to run.

This is what my services configuration looks like:

    services:
      postgres:
        image: postgres
        ports:
          - 5432:5432
        env:
          POSTGRES_PASSWORD: postgres
          POSTGRES_USER: postgres
        options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 3
Enter fullscreen mode Exit fullscreen mode

Let's go through this service definition rule-by-rule:

image: postgres defines which docker image to pull in, in my case I am using the official docker image for Postgres: https://hub.docker.com/_/postgres

ports:
    - 5432:5432
Enter fullscreen mode Exit fullscreen mode

We always have to define which ports we want to expose for our docker container. In this case I am just exposing the default Postgres port

env:
  POSTGRES_PASSWORD: postgres
  POSTGRES_USER: postgres
Enter fullscreen mode Exit fullscreen mode

The docker container we use has some environment variables defined, which it will use to configure the Postgres server. A full list of those variables can be found in the Postgres Docker Hub description.

options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5
Enter fullscreen mode Exit fullscreen mode

Lastly, we have to pass some additional options. These options will directly map to the docker create command. The reason we need to add this health check is because docker create exits before the container is actually ready to be used.

Luckily for us, there is a simple command pg_isready available, which we can check. As soon as the Postgres container is ready, your test run will continue.

Connecting with the database

The default Elixir workflow should already have some steps defined. We are modifying the Run Tests step a bit so it will create a database and pass some additional environment variables.

All services defined will create a virtual host on the network, named after the service. This means you cannot simply reach the Postgres on localhost.

I have added an environment variable in my run step that exposes the name of the services.

    - name: Run Tests
      run: mix test
      env:
        DB_HOST: postgres  
Enter fullscreen mode Exit fullscreen mode

Additionally I modified my Mix test configuration to read this variable and fall back on localhost by default

# Configure your database
config :myapp, MyApp.Repo,
  username: "postgres",
  password: "postgres",
  database: "myapp_test",
  hostname: System.get_env("DB_HOST", "localhost"),
  pool: Ecto.Adapters.SQL.Sandbox
Enter fullscreen mode Exit fullscreen mode

Running your tests

By default phoenix apps should have an alias for mix run test that runs ecto.setup, ecto.migrate and test.

When pushing new commits to your repository, the workflow should trigger, do all kinds of magic and run this command for you.

You can find the full file in this gist: https://gist.github.com/Nickforall/e49f5f3c37414e05f9a6c604accf2c3e

If you have any questions feel free to hit me up on twitter

References

Top comments (0)