Do you want to pick up from where you left of?
Take me there

Custom Mix Tasks

Creating custom Mix tasks for your Elixir projects.

Introduction

It’s not uncommon to want to extend your Elixir applications functionality by adding custom Mix tasks. Before we learn about how to create specific Mix tasks for our projects, let’s look at one that already exists:

$ mix phx.new my_phoenix_app

* creating my_phoenix_app/config/config.exs
* creating my_phoenix_app/config/dev.exs
* creating my_phoenix_app/config/prod.exs
* creating my_phoenix_app/config/prod.secret.exs
* creating my_phoenix_app/config/test.exs
* creating my_phoenix_app/lib/my_phoenix_app.ex
* creating my_phoenix_app/lib/my_phoenix_app/endpoint.ex
* creating my_phoenix_app/test/views/error_view_test.exs
...

As we can see from the shell command above, The Phoenix Framework has a custom Mix task to generate a new project. What if we could create something similar for our project? Well, the great news is we can, and Elixir provides excellent tools for this.

Setup

Let’s set up a basic Mix application.

$ mix new hello

* creating README.md
* creating .formatter.exs
* creating .gitignore
* creating mix.exs
* creating lib
* creating lib/hello.ex
* creating test
* creating test/test_helper.exs
* creating test/hello_test.exs

Your Mix project was created successfully.
You can use "mix" to compile it, test it, and more:

cd hello
mix test

Run "mix help" for more commands.

Now, in our lib/hello.ex file that Mix generated for us, let’s create a function that will output “Hello, World!”

defmodule Hello do
  @doc """
  Outputs `Hello, World!` every time.
  """
  def say do
    IO.puts("Hello, World!")
  end
end

Custom Mix Task

Let’s create our custom Mix task. Create a new directory and file hello/lib/mix/tasks/hello.ex. Within this file, let’s insert these 7 lines of Elixir.

defmodule Mix.Tasks.Hello do
  @moduledoc "The hello mix task: `mix help hello`"
  use Mix.Task

  @shortdoc "Calls the Hello.say/0 function."
  def run(_) do
    # calling our Hello.say() function from earlier
    Hello.say()
  end
end

Notice how we start the defmodule statement with Mix.Tasks and the name we want to call from the command line. On the second line, we introduce the use Mix.Task which brings the Mix.Task behaviour into the namespace. We then declare a run function which ignores any arguments for now. Within this function, we call our Hello module and the say function.

Loading your application

Mix does not automatically start our application or any of its dependencies which is fine for many Mix task use-cases but what if we need to use Ecto and interact with a database? In that case we need to make sure the app behind Ecto.Repo has started. There are 2 ways for us to handle this: explicitly starting an app or we can start our application which in turn will start the others.

Let’s look at how we can update our Mix task to start our application and dependencies:

defmodule Mix.Tasks.Hello do
  @moduledoc "The hello mix task: `mix help hello`"
  use Mix.Task

  @shortdoc "Calls the Hello.say/0 function."
  def run(_) do
    # This will start our application
    Mix.Task.run("app.start")

    Hello.say()
  end
end

Mix Tasks in Action

Let’s checkout our mix task. As long as we are in the directory it should work. From the command line, run mix hello, and we should see the following:

$ mix hello
Hello, World!

Mix is quite friendly by default. It knows that everyone can make a spelling error now and then, so it uses a technique called fuzzy string matching to make recommendations:

$ mix hell
** (Mix) The task "hell" could not be found. Did you mean "hello"?

Did you also notice that we introduced a new module attribute, @shortdoc? This comes in handy when shipping our application, such as when a user runs the mix help command from the terminal.

$ mix help

mix app.start         # Starts all registered apps
...
mix hello             # Simply calls the Hello.say/0 function.
...

Note: Our code must be compiled before new tasks will appear in the mix help output. We can do this either by running mix compile directly or by running our task as we did with mix hello, which will trigger the compilation for us.

It’s important to note that task names are derived from the module name, so Mix.Tasks.MyHelper.Utility will become my_helper.utility.

Caught a mistake or want to contribute to the lesson? Edit this lesson on GitHub!