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].