autorenew
Cláusula With en Elixir

Cláusula With en Elixir

Lo bueno de un lenguaje que está principalmente guiado por la comunidad, es que va evolucionando según las necesidades que se van encontrando sus usuarios. Un ejemplo es el que nos encontramos con la cláusula with de Elixir, que fue introducido en la versión 1.2 del lenguaje, y mejorado en la 1.3 para incluir la cláusula else.

El problema

Primero vamos a ver el problema que soluciona. Supongamos que tenemos una estructura del tipo %User{}. La estructura tendrá los campos típicos que tiene un usuario de una aplicación.

defmodule Example.With.User do
  defstruct name: "", age: 0, mail: "", accept_privacy: false, mail_sucriber: false
end

Ahora vamos a definir una serie de validaciones que realizará nuestro programa para gestionar este usuario. Algunas de las validaciones serán comprobar que es mayor de 18, que el nombre es válido, o que ha aceptado las políticas de privacidad.

defmodule Example.With.Program1 do

  alias Example.With.User

  def verify(user = %User{}) do
    users
    |> validate_name
    |> validate_age
    |> accepted_privacy
  end

  defp validate_name(user = %User{ name: n}) when length(n) == 0 do
    {:error, "Name is mandatory"}
  end

  defp validate_name(user = %User{}) do
    n = user.name

    case Regex.match?(~r/^[A-Z]\w{5,}/, n) do
      true -> user
      false -> {:error, "Name is not correct"}
    end
  end

  defp validate_age(user = %User{ age: a}) when a >= 18 do
    user
  end

  defp validate_age(%User{ age: a}) when a < 18 do
    {:error, "You are not 18 years old"}
  end

  defp accepted_privacy(%User{ accept_privacy: p}) when p == false do
    {:error, "You have to accept our privacy policy"}
  end

  defp accepted_privacy(user = %User{}) do
    user
  end
end

¿Veis cuál es el problema? El operador pipe |> es un instrumento genial para enlazar funciones. Pero claro, estas funciones tienen que ser consistentes y devolver siempre un mismo tipo de resultado. El problema es que por ejemplo validate_name puede devolver un :error, haciendo que recibamos un error en tiempo de ejecución, ya que una función envía una tupla, pero la siguiente espera recibir una estructura %User{}.

En este caso no nos sirve el pipe, así que en definitiva, necesitamos algo que sirva para concatenar funciones y que permita gestionar los errores. Y para eso entre otras cosas se utiliza with.

With

Como el movimiento se muestra andando, vamos a reescribir el ejemplo anterior, pero usando with. Ahí va:

 def verify(user = %User{}) do
    with %User{} <- validate_name(user),
         %User{} <- validate_age(user),
         %User{} <- accepted_privacy(user)
      do
        user 
      else
        {:error, e}->  IO.puts(e)
     end
  end

Como veis solo he cambiado la función verify, ya que el resto puede seguir igual. Con with iremos ejecutando las diferentes funciones, siempre que el pattern matching se vaya cumpliendo. En el caso de que no se cumpla en algún caso se pasará al bloque else. En este bloque también podemos tener pattern matching, y tener varias cláusulas. Yo solo he puesto un {:error, e}, pero podríamos ponero distintos tipos de error según su mensaje y para realizar distintas operaciones según el error devuelto.