Thoughts About Elixir
I’ve been programming Erlang for several years now and have always liked it. Erlang syntax has it’s quirks, but I’ve always considered OTP so cool that I don’t mind all the braces, colons, semicolons and stuff. Then I heard about Elixir that runs on Erlang VM, has a Rubyish syntax and can take advantage of all the cool stuff that the Erlang OTP offers. It took a while for Elixir to reach version 1.0 and become really a viable option for developing server software, but I really think it’s now there and I though that I should write about why I think so, and why I’m probably going to do more Elixir programming than Erlang programming in the near future.
Syntax
Elixir syntax is pretty close to Ruby. You don’t have to add braces to function calls if you don’t feel like it and all that stuff. Personally, this is what I don’t like about Ruby. I’m more of a Python guy anyways, forgive me. The Zen of Python states that…
There should be one-- and preferably only one --obvious way to do it.
I kind of agree with that. Although, shamed to admit, but Elixir’s Rubyish syntax is growing on me and I’m liking it more and more.
The one thing where Erlang’s syntax is at it’s best is writing recursive code using pattern matching. It combines the different parts of the recursive function nicely using semicolon. Elixir doesn’t make this distinction basically at all. Pattern matched functions all look the same and are somewhat more difficult to spot from the code. For example in Erlang a simple recursive function looks like this…
And in Elixir…
Note that both examples only have one function, which is maybe not
that visible from Elixir code. Not a big deal, but makes code somewhat
less readable when, for example, you have gen_fsm
module with a lot
of callbacks. The Elixir syntax does not group the def clauses in any
way, but the compiler politely warns you if you happen to have your
function definitions spread all over.
Handling Nested Data Structures
One area where Elixir particularly shines over Erlang is handling nested data structures. It is quite common to have some entity’s state represented by a nested map data structure and you usually all your application does is modify that state. Due to the fact that Erlang (and Elixir) data structures are immutable, modifying these nested data structures gets rather cumbersome. Elixir has some neat utilities to help you deal with these.
Keyword Lists vs Proplists
Proplists are common way of passing on configuration data in Erlang
applications. Proplist is a list of two element tuples, where the
first element is an key atom and second element is the value which can
be any Erlang term. Data in proplists is accessed using the
proplists
module. So you’ll end up doing a lot of this…
Or if you happen to have nested proplist…
Elixir has a concept of keyword lists. Which are essentially the same
as proplists in Erlang. Well, proplists can contain also atoms which
are just a shorthand for {Atom, true}
. But basically they are the
same. Elixir syntax just has a lot more convenient way of accessing
the data. In Elixir you can…
Or with nested…
Updating Nested Maps
Due to the fact that the data structures are immutable you will end up doing something like this when you need to modify a map. Let’s say we have a player state where we have player’s “wallet” containing all the soft/hard currency the player has and we want to reduce the amount of diamonds by ten. We’d do it like this…
Looks unnecessarily complex doesn’t it? Luckily Elixir is all about
making things easier for you so it has Kernel.put_in
and
Kernel.get_in
just for this. So here’s how we would use those…
Pipe Operator
Simple, yet one of the most useful syntax feature in my opinion is the Elixir pipe operator. Similar to that found from F# language, pipe operator passes on the output of the previous function as the first parameter of the next function. Let’s compare to Erlang once again. Coding in Erlang you easily end up doing plenty of this…
And the same using Elixir pipe operator…
I just gathered here pretty randomly the syntax related stuff that came to my mind. I really like how it is very clear that Elixir designers have really focused on making a functional language that is easy and pleasant to write and produces easily readable code.
Tooling
It’s common knowledge that Erlang’s build/release tools are quite horrible. There are 3rd party tools like rebar, Erlang.mk and relx, which are pretty neat but are not included in Erlang by default and don’t help the person writing his first Erlang software at all.
Luckily Elixir has really nice tools included. The standard
distribution contains a tool called mix
that is one of the best I’ve
seen in any language. It lets you bootstrap your project, package it,
run tests, install dependencies etc. Really nice. One thing it does
not do is create a release like Erlang reltool
or relx
. There is a
plugin for mix
that handles this called
[exrm](https://github.com/bitwalker/exrm)
.
Documentation
One more thing where Elixir beats Erlang is documentation. Everything
is well documented and documentation is easily accessible either
through the website or even more easily through the Elixir shell. For
example, try this in iex
.
Interfacing Erlang Code
The thing that finally got me convinced that Elixir is to be taken seriously is the way it interacts with Erlang code. Well, since Elixir compiles to Erlang it is quite straightforward. You just call Erlang modules and for the most part it just works. I can call Erlang like this…
Pretty much the only thing that can lead to problems is that since Elixir strings are Erlang binaries it is good to keep in mind that if some Erlang module takes string as parameter you need to give it Elixir’s character list which is the single quoted string in Elixir syntax.
All in all
I’ve been extremely happy working with Elixir. I quite recently started as a Lead Server Programmer in a mobile gaming company called Ministry of Games, based in Helsinki, Finland. We’re building our entire application server using Elixir. What is cool is that, while we’re using Elixir, we have the whole Erlang ecosystem at our disposal as well. E.g., we’re using ranch and locker which both are awesome pieces or Erlang software that do some of the most difficult parts of writing distributed servers extremely well, and we can write our application code on top of those using Elixir. A match made in heaven. :)
Ok, now… go give Elixir a try!