CLIs bound to a REST API do simillar things for most of the resources. Typically there are CRUD actions that appear for nearly every resource. Actions differ with parameters accross resources but the operations remain the same.
Hammer is optimised for usage with ApiPie and generated API bindings and tries to reduce the effort neccessary for a command creation.
ApiPie is a documentation library for RESTful APIs. Unlike traditional tools ApiPie uses DSL for API descriptions. This brings many advantages. See its documentation for details.
Foreman comes with ruby bindings automatically generated from the information provided by ApiPie. Every resource (eg. Architecture, User) has it's own class with methods for each available action (e.g. create, show, index, destroy). Apart from that it contains also full API documentation with parameters for the actions. This allows to reuse the documentation on the client side for automatic option definition and to reduce the amount of custom code per CLI action.
Hammer provides HammerCLI::Apipie::Command
base class for apipie actions.
The command class is single resource related and expect the resource and an action to be defined.
There's a simple DSL for that:
class ListCommand < HammerCLI::Apipie::Command
# define resource and the action together
resource :architectures, :index
end
# or
class ListCommand2 < HammerCLI::Apipie::Command
# define them separately
resource :architectures
action :index
end
When the resource-action pair is defined we can take advantage of automatic option definition. There's an option builder for apipie parameters pre-set in the apipie command's base class. You can read details about the option builder principles here.
class ListCommand < HammerCLI::Apipie::Command
resource :architectures, :index
build_options
end
If we plug the command into an existing command tree and check the help we will see there are four parameters defined from the ApiPie docs. Compare the result with online API documentation.
$ hammer architecture list -h
Usage:
hammer architecture list [OPTIONS]
Options:
--search SEARCH filter results
--order ORDER sort results
--page PAGE paginate results
--per-page PER_PAGE number of entries per request
-h, --help print help
It is possible to combine apipie options with custom ones. If the generated options don't suit your needs for any reason, you can always redefine them by hand or just skip them.
See following example.
class ListCommand < HammerCLI::Apipie::Command
resource :architectures, :index
# first created option takes precedence
option '--search', 'QUERY', "search query"
# use option :without to define what options should be skipped
build_options :without => [:order]
end
hammer architecture list -h
Usage:
hammer architecture list [OPTIONS]
Options:
--page PAGE paginate results
--per-page PER_PAGE number of entries per request
--search QUERY search query
-h, --help print help
Note that the --search
description has changed and --order
disappeared.
Automaticaly generated options reflect:
- parameter names and descriptions
- required parameters
- parameter types - the only supported type is array, which is translated to option normalizer
List
Some commands are expected to print result of the API action. There are two class methods for setting success and failure messages. Messages are printed according to the HTTP status code the API returned.
success_message "The user has been created"
failure_message "Could not create the user"
class CreateCommand < HammerCLI::Apipie::Command
command_name "create"
resource :architectures, :create
success_message "Architecture created"
failure_message "Could not create the architecture"
build_options
end
$ hammer architecture create -h
Usage:
hammer architecture create [OPTIONS]
Options:
--name NAME
--operatingsystem-ids OPERATINGSYSTEM_IDS Operatingsystem ID’s
Comma separated list of values.
-h, --help print help
$ hammer architecture create
ERROR: option '--name' is required
See: 'hammer architecture create --help'
$ hammer architecture create --name test --operatingsystem-ids=1,2
Architecture created
$ hammer architecture create --name test
Could not create the architecture:
Name has already been taken
class InfoCommand < HammerCLI::Apipie::Command
command_name "info"
resource :architectures, :show
# It's a good practice to reuse output definition from list commands
# and add more details. This helps avoiding duplicities.
output ListCommand.output_definition do
from "architecture" do
field :operatingsystem_ids, "OS ids", Fields::List
field :created_at, "Created at", Fields::Date
field :updated_at, "Updated at", Fields::Date
end
end
build_options
end
$ hammer architecture info -h
Usage:
hammer architecture info [OPTIONS]
Options:
--id ID
-h, --help print help
$ hammer architecture info --id 1
Id: 1
Name: x86_64
OS ids: 1, 3
Created at: 2013/06/08 18:53:56
Updated at: 2013/06/08 19:17:43
When you define more commands like we've shown above you find yourself repeating
resource ...
in every one of them. As the commands are usually grouped by
the resource it is handy to extract the resource definition one level up to
the encapsulating command.
class Architecture < HammerCLI::Apipie::Command
resource :architectures
class ListCommand < HammerCLI::Apipie::Command
action :index
# ...
end
class InfoCommand < HammerCLI::Apipie::Command
action :show
# ...
end
# ...
end
ApiPie resources are being looked up in the encapsulating classes and modules when the definition is missing in the command class. If they are not found even there the resource of the parent command is used at runtime. This is useful for context-aware shared commands.
The following example shows a common subcommand that can be attached to
any parent which has a resource implementing the method add_tag
. Please note that this example
is fictitious. There are no tags in Foreman's architectures and users.
module Tags
class AddTag < HammerCLI::Apipie::WriteCommand
option '--id', 'ID', 'ID of the resource'
option '--tag', 'TAG', 'Name of the tag to add'
action :add_tag
command_name 'add-tag'
end
end
class Architecture < HammerCLI::Apipie::Command
resource :architectures
# ...
include Tags
autoload_subcommands
end
class User < HammerCLI::Apipie::Command
resource ForemanApi::Resources::User
# ...
include Tags
autoload_subcommands
end
$ hammer architecture add-tag -h
Usage:
hammer architecture add-tag [OPTIONS]
Options:
--id ID ID of the resource
--tag TAG Name of the tag to add
-h, --help print help
$ hammer user add-tag -h
Usage:
hammer user add-tag [OPTIONS]
Options:
--id ID ID of the resource
--tag TAG Name of the tag to add
-h, --help print help