An opinionated way of dealing with behaviours.
Opinionated means that it offers an easy way of defining a "facade" for a
behaviour. This facade then delegates calls to the real implementation, which
is either given directly to Knigge
or fetched from the configuration.
Knigge
can be use
d directly in a behaviour, or in a separate module by
passing the behaviour which should be "facaded" as an option.
See the documentation for more information.
Simply add knigge
to your list of dependencies in your mix.exs
:
def deps do
[
{:knigge, "~> 1.0"}
]
end
Knigge
was born out of a desire to standardize dealing with behaviours and
their implementations.
As great fans of mox
we longed for
an easy way to swap out implementations from the configuration which lead us
to introduce a facade pattern, where a module's sole responsibility was
loading the correct implementation and delegating calls.
This pattern turned out to be very flexible and useful but required a fair bit
of boilerplate code. Knigge
was born out of an attempt to reduce this
boilerplate to the absolute minimum.
You can read about our motivation in depth in our devblog.
Imagine a behaviour looking like this:
defmodule MyGreatBehaviour do
@callback my_great_callback(my_argument :: any()) :: any()
end
Now imagine you want to delegate calls to this behaviour like this:
defmodule MyGreatBehaviourFacade do
@behaviour MyGreatBehaviour
@implementation Application.fetch_env!(:my_application, __MODULE__)
defdelegate my_great_callback, to: @implementation
end
With this in place you can simply reference the "real implementation" by calling functions on your facade:
MyGreatBehaviourFacade.my_great_callback(:with_some_argument)
Knigge
allows you to reduce this boilerplate to the absolute minimum:
defmodule MyGreatBehaviourFacade do
use Knigge,
behaviour: MyGreatBehaviour,
otp_app: :my_application
end
Technically even passing the behaviour
is optional, it defaults to
the current __MODULE__
. This means that the example from above could
be shortened even more to:
defmodule MyGreatBehaviour do
use Knigge, otp_app: :my_application
@callback my_great_callback(my_argument :: any()) :: any()
end
Under the hood this compiles down to the explicit delegation visible on the top.
In case you don't want to fetch your implementation from the configuration,
Knigge
also allows you to explicitely pass the implementation of the
behaviour with the aptly named key implementation
:
defmodule MyGreatBehaviourFacade do
use Knigge,
behaviour: MyGreatBehaviour,
implementation: MyGreatImplementation
end
Now imagine you have a more sophisticated behaviour with some optional callbacks:
defmodule MySophisticatedBehaviour do
@callback an_optional_callback() :: any()
@callback a_required_callback() :: any()
@optional_callbacks an_optional_callback: 0
end
As you would expect Knigge
delegates calls to this callback as usual. But
since it's optional this delegation might fail. A common pattern is to check
if the implementation exports the function in question:
if function_exported?(MyImplementation, :an_optional_callback, 0) do
MyImplementation.an_optional_callback()
else
:my_fallback_implementation
end
Knigge
offers an easy way to specify these fallback implementations with
defdefault
:
defmodule MySophisticatedFacade do
use Knigge,
behaviour: MySophisticatedBehaviour,
otp_app: :my_application
defdefault an_optional_callback do
:my_fallback_implementation
end
end
Knigge
tries to determine at compile-time if the implementation exports
the function in question and only uses the default if this is not the case.
As such defdefault
incurs no runtime overhead and compiles to a simple def
.
Of course defdefault
s can accept arguments as any usual function:
defdefault my_optional_callback_with_arguments(first_argument, another_argument) do
case first_argument do
# ...
end
end
Knigge
expects either the otp_app
key or the implementation
key. If
neither is provided an error will be raised at compile time.
When using the otp_app
configuration you can also pass config_key
, which
results in a call looking like this: Application.fetch_env!(otp_app, config_key)
.
config_key
defaults to __MODULE__
.
By default Knigge
does as much work as possible at compile time. This will
be fine most of the time. In case you want to swap out the implementation at
runtime - by calling Application.put_env/2
- you can force Knigge
to do all
delegation at runtime. As you might expect this incurs runtime overhead,
since the implementing module will have to be loaded for each call.
If you want to do delegation at runtime simply pass delegate_at: :runtime
as
option.
For further information about options check the Knigge.Options
module.