Note: This is a very basic example of mix tool and escript utility. One can skip this post if already have an idea about these tools.
Lets write a simple command line program which would take a String as a command line argument and display the number of vowels in it. We’ll use Elixir’s mix build tool and OptionParser to parse command line arguments.
Lets start by setting up the application:
$ mix new vowel_counter
Open mix.exs
, you will see something like:
defmodule VowelCounter.Mixfile do use Mix.Project def project do [app: :vowel_counter, version: "0.0.1", elixir: "~> 1.1", build_embedded: Mix.env == :prod, start_permanent: Mix.env == :prod, deps: deps] end # Configuration for the OTP application # # Type "mix help compile.app" for more information def application do [applications: [:logger]] end # Dependencies can be Hex packages: # # {:mydep, "~> 0.3.0"} # # Or git/path repositories: # # {:mydep, git: "https://github.com/elixir-lang/mydep.git", tag: "0.1.0"} # # Type "mix help deps" for more examples and options defp deps do [] end end
We’ll be using Erlang’s escript utility to precompile our app and package our application’s code along with its dependencies(no dependencies in this example app) in one single executable. So lets add escript: escript_config
to project
method in our mix.exs
to define our escript configuration, also define a private method escript_config
to specify the the main_module. In this case, after these changes our mix.exs
would look like:
defmodule VowelCounter.Mixfile do use Mix.Project def project do [app: :vowel_counter, version: "0.0.1", elixir: "~> 1.1", escript: escript_config, build_embedded: Mix.env == :prod, start_permanent: Mix.env == :prod, deps: deps] end # Configuration for the OTP application # # Type "mix help compile.app" for more information def application do [applications: [:logger]] end # Dependencies can be Hex packages: # # {:mydep, "~> 0.3.0"} # # Or git/path repositories: # # {:mydep, git: "https://github.com/elixir-lang/mydep.git", tag: "0.1.0"} # # Type "mix help deps" for more examples and options defp deps do [] end defp escript_config do [main_module: VowelCounter] end end
Lets compile and run our application by running:
$ mix escript.build
$ ./vowel_counter "my test string"
Since we have specified VovelCounter
module as our main_module in elixir_config, our app expects and call the main
method from this module and pass command line arguments to it. We haven’t defined this main
method yet, so we should see something similar when we execute our app:
** (UndefinedFunctionError) undefined function: VowelCounter.main/1 (vowel_counter) VowelCounter.main(["my test string"]) (elixir) lib/kernel/cli.ex:76: anonymous fn/3 in Kernel.CLI.exec_fun/2
Lets open lib/vowel_counter.ex
and define main
method as:
defmodule VowelCounter do # Take command line arguments # Then parse them # Then count number of vowel # Print the outpu def main(args) do args |> parse_args |> count_vowel |> IO.puts end defp parse_args(args) do # Covering only expected arguments. We can add code to handle cases when unexpected arguments are passed or help text etc. # Checkout OptionParser lib for details. {_, [str], _} = OptionParser.parse(args) str end defp count_vowel(str) do # Scan for vowels and take the list's length len = length(Regex.scan(~r/[aeiou]/i, str)) "There are #{len} vowels" end end
Lets re-compile and re-run our application by:
$ mix escript.build
$ ./vowel_counter "This string has 5 vowels."
We should see: “There are 5 vowels”
Your comments/suggestions are most welcome.