Building Plugins
Rebar3’s system is based on the concept of providers. A provider has three callbacks:
init(State) -> {ok, NewState}
, which helps set up the state required, state dependencies, etc.do(State) -> {ok, NewState} | {error, Error}
, which does the actual work.format_error(Error) -> String
, which prints errors when they happen, and to filter out sensitive elements from the state.
A provider should also be an OTP Library application, which can be fetched as any other Erlang dependency, except for Rebar3 rather than your own system or application.
This document contains the following elements:
Using a Plugin
To use the a plugin, add it to the rebar.config:
Then you can just call it directly:
Reference
Provider Interface
Each provider has the following options available:
- name: The ‘user friendly’ name of the task.
- module: The module implementation of the task.
- hooks: A two-tuple of provider names for pre and post-hooks (
{Pre, Post}
). - bare: Indicates whether task can be run by users or not. Should be
true
. - deps: The list of dependencies, providers that need to run before this one. You do not need to include the dependencies of your dependencies.
- desc: The description for the task, used by
rebar3 help
- short_desc: A one line short description of the task, used in lists of providers
- example: An example of the task usage, such as
"rebar3 my-provider args"
- opts: The list of options that the task requires/understands. The form of each option is
{Key, $Character, "StringName", Spec, HelpText}
, where:Key
is an atom, to be used to fetch the value later;$Character
is the short form of the option. So if the command is to be entered as a-c Arg
,$c
is the value of this fieldSpec
is either a type (atom
,binary
,boolean
,float
,integer
, orstring
), a type with a default value ({Type, Val}
), or the atomundefined
.
- profiles: Profiles to use for provider. Default to
[default]
. - namespace: namespace the provider is registered in. Defaults to
default
, which is the main namespace.
These options are to be added to the provider when creating it.
A provider has the following implementation:
List of Possible Dependencies
All dependencies are in the default namespace until indicated otherwise
Name | Function | Profile | Also depends on |
---|---|---|---|
app_discovery | Explore user applications and loads their configuration. | default | |
clean | Remove compiled beam files from apps. | default | app_discovery |
compile | Compile apps .app.src and .erl files. | default | lock |
cover | Analyze cover-compiled files | default | lock |
ct | Run common test suites. | test | compile |
deps | List dependencies. | default | app_discovery |
dialyzer | Run the Dialyzer analyzer on the project. | default | compile |
edoc | Generate documentation using edoc. | default | app_discovery |
eunit | Run EUnit Tests. | test | compile |
help | Display a list of tasks or help for a given task or subtask. | default | |
install_deps | Download dependencies | default | app_discovery |
lock | Lock dependencies and add rebar.lock | default | install_deps |
new | Create new project from templates. | default | |
pkgs | List available packages. | default | |
release | Build release of project. | default | compile |
report | Provide a crash report to be sent to the rebar3 issues page | default | |
shell | Run shell with project apps and deps in path. | default | compile |
tar | Tar archive of release built of project. | default | compile |
update | Update package index. | default | |
upgrade | Upgrade dependencies. | default | |
version | Print version for rebar and current Erlang. | default | |
xref | Run cross reference analysis | default | compile |
Note that you can depend on more than one provider, but they must be in the same namespace
Rebar API
Rebar comes with a module called rebar_api
exporting commonly needed functions when writing providers. Functions include:
Function | Usage |
---|---|
abort() | Interrupts program flow |
abort(FormatString, Args) | Interrupts program flow; displays an ERROR message along with it. Equivalent to calling rebar_api:error(FormatString, Args) followed by rebar_api:abort() |
console(FormatString, Args) | Prints to the console. |
info(FormatString, Args) | Logs with the severity INFO |
warn(FormatString, Args) | Logs with the severity WARNING |
error(FormatString, Args) | Logs with the severity ERROR |
debug(FormatString, Args) | Logs with the severity DEBUG |
expand_env_variable(InStr, VarName, RawVarValue) | Given the env variable FOO, we want to expand all references to it in InStr. References can have two forms: $FOO and ${FOO}. The form $FOO is delimited by whitespace characters or the end of a line (eol). |
get_arch() | Returns the ‘architecture’ as a string of the form “$OTP_VSN-$SYSTEM_$ARCH-WORDSIZE. Final strings will look like “17-x86_64-apple-darwin13.4.0-8” or “17-x86_64-unknown-linux-gnu-8” |
wordsize() | Returns the true wordsize of the emulator, i.e. the size of a pointer, in bytes as an string. |
add_deps_to_path(RebarState) | The project’s dependencies are added to the code path. Useful when a tool is invoked and needs to have global stateful access to libraries. |
restore_code_path(RebarState) | Revert the code path to only include the libraries required to run Rebar3 and its plugins. This is the desired state for Rebar3 to avoid conflicts with user-provided tools. |
ssl_opts(Url) | Returns the ssl options to use with httpc to make a secure and verified HTTP request. |
Do note that all logging functions automatically add a new line (~n
) to every expression logged.
Rebar State Manipulation
The State
argument passed to the plugin provider can be operated on with the rebar_state
module through the following interface:
Function | Usage |
---|---|
get(State, Key, [DefaultValue]) -> Value | When a rebar.config element is of the form {Key, Value}., fetches the value for it |
set(State, Key, Value) -> NewState | Adds a configuration value to the rebar state. |
lock(State) -> ListOfLocks | Returns a list of locked dependencies |
escript_path(State) -> Path | Returns the Rebar3 escript location |
command_args(State) -> RawArgs | Returns the arguments passed to rebar3 |
command_parsed_args(State) -> Args | Returns the arguments passed to rebar3, parsed. |
deps_names(State) -> DepsNameList | Returns a list of dependencies’ names |
project_apps(State) -> AppList | Returns a list of applications. These can be handled using rebar_app_info. |
all_deps(State) -> DepsList | Returns a list of dependencies. These can be handled using rebar_app_info. |
add_provider(State, Provider) -> NewState | Registers a new provider, where Provider is the result of calling providers:create(Options). To be effective, this function must be called as part of a provider’s init/1 function. It can be called multiple times, allowing a plugin to register multiple commands. |
add_resource(State, {Key, Module}) -> NewState | Registers a new resource type (such as git, hg, and so on) with the module used to handle it. The resource must implement the rebar_resource behaviour. To be effective, this function must be called as part of a provider’s init/1 function. |
Manipulate Application State
Each application being built (project applications and dependencies). All AppInfo records can be found in the State and accessed through project_apps/1
and all_deps/1
Function | Usage |
---|---|
get(AppInfo, Key, [DefaultValue]) -> Value | Fetch value of Key as defined for the application AppInfo |
set(AppInfo, Key, Value) -> NewState | Adds a configuration value to the application’s record |
Namespaces
For plugins that might require multiple commands all adapted to a single type of task (such as implementing a suite of tools for a BEAM language other than Erlang), rather than having multiple commands polluting the command space or requiring prefixes such as rebar3 mylang_compile
, rebar3 introduces support for namespaces.
A plugin can be declared to belong to a given namespace. For example, the ErlyDTL compiler plugin introduces the compile
command under the erlydtl
namespace. It can therefore be invoked as rebar3 erlydtl compile
. If the erlydtl
namespace had other commands such as clean
, they could be chained as rebar3 erlydtl clean, compile
.
In other ways, a namespace acts like do
(rebar3 do compile, edoc
), but operating on a non-default set of commands.
To declare a namespace, an provider needs only to use the {namespace, Namespace}
option in its configuration list. The provider will automatically register the new namespace and be available under this term.
`🚧
Namespaces also apply to provider dependencies and hooks
If a provider is part of a given namespace, its dependencies will be searched within that same namespace. Therefore if
rebar3 mytool rebuild
depends oncompile
, thecompile
command will be looked for in themytool
namespace.To use the default
compile
command, the dependency must be declared as{default, compile}
, or more generally{NameSpace, Command}
.The same mechanism is applied for hooks.
Tutorial
First version
In this tutorial, we’ll show how to start from scratch, and get a basic plugin written. The plugin will be quite simple: it will look for instances of TODO:
lines in comments and report them as warnings. The final code for the plugin can be found on bitbucket.
The first step is to create a new OTP Application that will contain the plugin:
The src/todo.erl
file will be used to call the initialization of all commands. For now we’ll only have one todo
command. Open up the src/todo_prv.erl
file that will contain the command implementation, and make sure you have the following skeleton in place:
This shows all the basic content needed. Note that we leave the DEPS
macro to the value app_discovery
, used to mean that the plugin should at least find the project’s source code (excluding dependencies).
In this case, we need to change very little in init/1
. Here’s the new provider description:
Instead, most of the work will need to be done directly in do/1
. We’ll use the rebar_state
module to fetch all the applications we need. This can be done by calling the project_apps/1
function, which returns the list of the project’s top-level applications.
This, on a high level, means that we’ll check each top-level app one at a time (there may often be more than one top-level application when working with releases)
The rest is filler code specific to the plugin, in charge of reading each app path, go read code in there, and find instances of ‘TODO:’ in comments in the code:
Just using io:format/2
to output is going to be fine.
To test the plugin, push it to a source repository somewhere. Pick one of your projects, and add something to the rebar.config
:
Then you can just call it directly:
Rebar3 will download and install the plugin, and figure out when to run it. Once compiled, it can be run at any time again.
Optionally Search Deps
Let’s extend things a bit. Maybe from time to time (when cutting a release), we’d like to make sure none of our dependencies contain ‘TODO:’s either.
To do this, we’ll need to go parse command line arguments a bit, and change our execution model. The ?DEPS
macro will now need to specify that the todo
provider can only run after dependencies have been installed:
We can add the option to the list we use to configure the provider in init/1
:
And then we can implement the switch to figure out what to search:
The deps
option is found using rebar_state:command_parsed_args(State)
, which will return a proplist of terms on the command-line after ‘todo’, and will take care of validating whether the flags are accepted or not. The rest can remain the same.
Push the new code for the plugin, and try it again on a project with dependencies:
Rebar3 will now go pick dependencies before running the plugin on there.
you can also see that the help will be completed for you:
That’s it, the todo plugin is now complete! It’s ready to ship and be included in other repositories.
Adding More Commands
To add more commands to the same plugin, simply add entries to the init
function in the main module:
And Rebar3 will pick it up from there.