Croma
Elixir macro utilities.
Usage
-
Add
:cromaas a mix dependency. $ mix deps.get-
Add
use Cromato import macros defined in this package. - Hack!
Defining functions
Croma.Defpt.defpt
-
Unit-testable
defpthat is simply converted todefifMix.env == :test,defpotherwise.
Croma.Defun
Type specification oriented function definition
Example 1
import Croma.Defun defun f(a: integer, b: String.t) :: String.t do "#{a} #{b}" end
is expanded to
```ex @spec f(integer, String.t) :: String.t def f(a, b) do "#{a} #{b}" end ```Example 2
import Croma.Defun defun dumbmap(as: [a], f: (a -> b)) :: [b] when a: term, b: term do ([] , _) -> [] ([h | t], f) -> [f.(h) | dumbmap(t, f)] end
is expanded to
```ex @spec dumbmap([a], (a -> b)) :: [b] when a: term, b: term def dumbmap(as, f) def dumbmap([], _) do [] end def dumbmap([h | t], f) do [f.(h) | dumbmap(t, f)] end ```-
There are also
defunpanddefunptmacros for private functions. -
Known limitations:
-
Pattern matching against function parameters should use
(param1, param2) when guards -> blockstyle. - Overloaded typespecs are not supported.
-
Pattern matching against function parameters should use
Croma.Monad
An interface definition of the monad typeclass.
Modules that
use Croma.Monadmust implement the following interface:@type t(a)with a type parametera.@spec pure(a: a) :: t(a) when a: any@spec bind(t(a), (a -> t(b))) :: t(b) when a: any, b: any
By using the concrete implementations of the above interface,
Croma.Monadprovides the default implementations of the following functions:-
As Functor:
@spec map(t(a), (a -> b)) :: t(b) when a: any, b: any
-
As Applicative:
@spec ap(t(a), t((a -> b))) :: t(b) when a: any, b: any@spec sequence([t(a)]) :: t([a]) when a: any
-
As Functor:
Note that the order of parameters in
map/apis different from that of Haskell counterparts, in order to leverage Elixir’s pipe operator|>.Croma.Monadalso providesbind-less syntax similar to Haskell’s do-notation. For example,MonadImpl.m do x <- mx y <- my pure f(x, y) end
is converted to
```ex
MonadImpl.bind(mx, fn x ->
MonadImpl.bind(my, fn y ->
MonadImpl.pure f(x, y)
end)
end)
```Croma.Result
Corma.Result.t(a)is defined as@type t(a) :: {:ok, a} | {:error, any}. This module implementsCroma.Monadinterface.
Croma.List
-
Implementation of
Croma.Monadfor lists.Croma.List.t(a)is just an alias to[a].
Working with structs
Croma.Struct
Utility module to define structs with type specification and validation functions.
iex> defmodule F do ...> @type t :: integer ...> def validate(i) when is_integer(i), do: {:ok, i} ...> def validate(_), do: {:error, :invalid_f} ...> def default, do: 0 ...> end ...> defmodule S do ...> use Croma.Struct, f: F ...> end ...> S.validate([f: 5]) {:ok, %S{f: 5}} ...> S.validate(%{f: "not_an_integer"}) {:error, :invalid_f} ...> s = S.new([]) %S{f: 0} ...> S.update(s, [f: 2]) {:ok, %S{f: 2}} ...> S.update(s, %{"f" => "not_an_integer"}) {:error, :invalid_f}Some helper modules for “per-field module”s that are passed as options to
use Croma.Struct(e.g.Fin the above example) are available.Croma.SubtypeOf*Croma.TypeGenCroma.Boolean
Croma.StructCallSyntax
A new syntax (which uses
~>operator) for calls to functions that take structs as 1st argument.iex> import Croma.StructCallSyntax ...> defmodule S do ...> defstruct [:a, :b] ...> def f(s, i) do ...> s.a + s.b + i ...> end ...> end ...> s = %S{a: 1, b: 2} ...> s~>f(3) # => S.f(s, 3) 6