Protoss
Evil and Powerful Protocols for Elixir
Protocols are just modules! And sometimes you just want to do module things in protocols. And when you're in control of both the protocol and the module you're implementing for, colocating the code in the module you're writing just makes sense. This library does this automatically for you.
Usage
Protocol definition
use Protoss
defprotocol MyProtocol do
# this works like a normal protocol forward definition.
def fun_must_be_implemented(argument)
# delegations forward to the passed module. Currently
# this is not currently runtime-checked, but it will be
# in the future under a flag.
#
# call as:
# `MyProtocol.from_json(MyStruct, ...)`
#
defdelegate from_json(module, argument)
# after block separates protocol code from module code
after
# struct protocol implementations MUST implement these
# extra callbacks:
@callback extra_callback() :: term
# this can be called directly as: `MyProtocol.root_function()`
def root_function do
:ok
end
end
Protocol implementation
In your (struct) protocols, the implementation of MyProtocol
from above
would look like this.
defmodule MyStruct do
defstruct [:value]
use MyProtocol
@impl MyProtocol
def fun_must_be_implemented(%__MODULE__{value: value}), do: value
# note the arity change!
@impl MyProtocol
def from_json(%{"value" => value}), do: %__MODULE__{value: value}
@impl MyProtocol
def extra_callback, do: "I was required to implement this"
end
Formatter changes
In order to get proper formatting, you should add the following option to your .formatter.exs
locals_without_parens: [defdelegate: 1]
Installation
The package is available on hex and can be installed by adding protoss
to your
list of dependencies in mix.exs
:
def deps do
[
{:protoss, "~> 0.2"}
]
end
Documentation can be found at https://hexdocs.pm/protoss.