Rake tasks for managing binary dependencies used within a build.
Add this line to your application's Gemfile:
gem 'rake_dependencies'
And then execute:
$ bundle
Or install it yourself as:
$ gem install rake_dependencies
RakeDependencies provides a suite of tasklibs for downloading and extracting a
distribution of some dependency. The simplest way to configure all of these
tasks is via the RakeDependencies::Tasks::All
tasklib. The following provides
an example usage with terraform as the target dependency:
RakeDependencies::Tasks::All.new do |t|
t.namespace = :terraform
t.dependency = 'terraform'
t.version = '0.9.0'
t.path = File.join('vendor', 'terraform')
t.type = :zip
t.platform_os_names = { darwin: 'darwin', linux: 'linux' }
t.platform_cpu_names = { x86_64: 'amd64', arm64: 'arm64' }
t.uri_template =
'https://releases.hashicorp.com/terraform/<%= @version %>/' +
'terraform_<%= @version %>_' +
'<%= @platform_os_name %>_<%= @platform_cpu_name %><%= @ext %>'
t.file_name_template =
'terraform_<%= @version %>_' +
'<%= @platform_os_name %>_<%= @platform_cpu_name %><%= @ext %>'
t.needs_fetch = lambda do |parameters|
terraform_binary = File.join(
parameters[:path], parameters[:binary_directory], 'terraform')
!(File.exist?(terraform_binary) &&
`#{terraform_binary} -version`.lines.first =~ /#{parameters[:version]}/)
end
end
With this in place, a number of tasks will be defined:
> rake -T
rake terraform:clean # Clean vendored terraform
rake terraform:download # Download terraform distribution
rake terraform:ensure # Ensure terraform present
rake terraform:extract # Extract terraform archive
rake terraform:fetch # Fetch terraform
The tasks perform the following:
<ns>:clean
- recursively deletes the directory containing the dependency<ns>:download
- downloads the distribution from the provided path into the dependency directory<ns>:extract
- extracts, in the case of a compressed archive, or copies, in the case of an uncompressed distribution, the binaries into the binary directory under the dependency directory<ns>:fetch
- downloads then extracts<ns>:ensure
- checks whether the dependency needs to be fetched and cleans and fetches if necessary
With these tasks defined, any task that requires the dependency to be present
should depend on <ns>:ensure
. Continuing the terraform example:
task :provision_database => ['terraform:ensure'] do
sh('vendor/terraform/bin/terraform apply infra/database')
end
If the installation_directory
attribute is supplied, an additional install
task will be defined:
RakeDependencies::Tasks::All.new do |t|
t.namespace = :terraform
t.dependency = 'terraform'
# ...
t.installation_directory = "#{ENV['HOME']}/bin"
# ...
end
Then:
> rake -T
rake terraform:clean # Clean vendored terraform
rake terraform:download # Download terraform distribution
rake terraform:ensure # Ensure terraform present
rake terraform:extract # Extract terraform archive
rake terraform:fetch # Fetch terraform
rake terraform:install # Install terraform
The <ns>:install
task copies the binary into the defined installation
directory which can be anywhere on the filesystem.
The RakeDependencies::Tasks::All
tasklib supports the following configuration
parameters:
Name | Description | Default | Required |
---|---|---|---|
namespace |
The namespace in which to define the tasks | - | no |
dependency |
The name of the dependency, used in status reporting and as the default binary name | - | yes |
version |
The version of the dependency to manage, only required if used in templates or needs_fetch |
- | no |
path |
The path in which to install the dependency | - | yes |
type |
The archive type of the distribution, one of :zip , :tar_gz , :tgz or :uncompressed |
:zip |
yes |
platform_os_names |
A map of platform OS identifiers to OS names to use in templates | RakeDependencies::PlatformNames::OS |
yes |
platform_cpu_names |
A map of platform CPU identifiers to CPU names to use in templates | RakeDependencies::PlatformNames::CPU |
yes |
distribution_directory |
The name of the directory under the supplied path into which to download the distribution | 'dist' |
yes |
binary_directory |
The name of the directory under the supplied path into which to extract/copy the binaries | 'bin' |
yes |
installation_directory |
The name of the directory into which to install the binaries, anywhere on the file system | - | no |
uri_template |
A template for the URI of the distribution | - | yes |
file_name_template |
A template for the name of the downloaded file | - | yes |
source_binary_name_template |
A template for the name of the binary before rename after extraction/copying | - | no |
target_binary_name_template |
A template for the name to rename the binary to after extraction/copying | - | no |
strip_path_template |
A template for the path to strip within an archive before extracting | - | no |
needs_fetch |
A lambda taking a parameter map that should return true if the dependency needs to be fetched, false otherwise |
Will always return true |
no |
clean_task_name |
The name of the clean task, required if it should be different from the default | :clean |
yes |
download_task_name |
The name of the download task, required if it should be different from the default | :download |
yes |
extract_task_name |
The name of the extract task, required if it should be different from the default | :extract |
yes |
install_task_name |
The name of the install task, required if it should be different from the default | :install |
no |
fetch_task_name |
The name of the fetch task, required if it should be different from the default | :fetch |
yes |
ensure_task_name |
The name of the ensure task, required if it should be different from the default | :ensure |
yes |
Notes:
- Each of the templates will have the following instance variables in scope when
rendered:
@version
: the supplied version string@platform
: theGem::Platform
on which the task is executing@platform_os_name
: the OS name derived from the platform on which the task is executing and the providedplatform_os_names
map@platform_cpu_name
: the CPU name derived from the platform on which the task is executing and the providedplatform_cpu_names
map@ext
: the file extension corresponding to the providedtype
, one of.zip
,.tar.gz
,.tgz
or empty string for uncompressed files
- The
needs_fetch
lambda will receive a map with the following entries:path
: the supplied pathversion
: the supplied version stringbinary_directory
: the supplied or default binary directory
The RakeDependencies::Tasks::All
tasklib uses each of the following tasklibs
in its definition:
RakeDependencies::Tasks::Clean
RakeDependencies::Tasks::Download
RakeDependencies::Tasks::Extract
RakeDependencies::Tasks::Install
RakeDependencies::Tasks::Fetch
RakeDependencies::Tasks::Ensure
The RakeDependencies::Tasks::Clean
tasklib supports the following
configuration parameters:
Name | Description | Default | Required |
---|---|---|---|
name |
The name of the task, required if it should be different from the default | :clean |
yes |
path |
The path in which the dependency is installed | - | yes |
dependency |
The name of the dependency, used in status reporting | - | yes |
The RakeDependencies::Tasks::Download
tasklib supports the following
configuration parameters:
Name | Description | Default | Required |
---|---|---|---|
name |
The name of the task, required if it should be different from the default | :download |
yes |
dependency |
The name of the dependency, used in status reporting | - | yes |
version |
The version of the dependency to manage, only required if used in templates | - | no |
path |
The path in which to install the dependency | - | yes |
type |
The archive type of the distribution, one of :zip , :tar_gz , :tgz or :uncompressed |
:zip |
yes |
platform_os_names |
A map of platform OS identifiers to OS names to use in templates | RakeDependencies::PlatformNames::OS |
yes |
platform_cpu_names |
A map of platform CPU identifiers to CPU names to use in templates | RakeDependencies::PlatformNames::CPU |
yes |
distribution_directory |
The name of the directory under the supplied path into which to download the distribution | 'dist' |
yes |
uri_template |
A template for the URI of the distribution | - | yes |
file_name_template |
A template for the name of the downloaded file | - | yes |
Notes:
- The templates have the same instance variables in scope when rendered as mentioned above.
The RakeDependencies::Tasks::Extract
tasklib supports the following
configuration parameters:
Name | Description | Default | Required |
---|---|---|---|
name |
The name of the task, required if it should be different from the default | :extract |
yes |
dependency |
The name of the dependency, used in status reporting | - | yes |
version |
The version of the dependency to manage, only required if used in templates | - | no |
path |
The path in which to install the dependency | - | yes |
type |
The archive type of the distribution, one of :zip , :tar_gz , :tgz or :uncompressed |
:zip |
yes |
platform_os_names |
A map of platform OS identifiers to OS names to use in templates | RakeDependencies::PlatformNames::OS |
yes |
platform_cpu_names |
A map of platform CPU identifiers to CPU names to use in templates | RakeDependencies::PlatformNames::CPU |
yes |
extractors |
A map of archive types to extractor classes, see notes for further details | Extractors for all supported types | yes |
distribution_directory |
The name of the directory under the supplied path into which the distribution was downloaded | 'dist' |
yes |
binary_directory |
The name of the directory under the supplied path into which to extract/copy the binaries | 'bin' |
yes |
file_name_template |
A template for the name of the downloaded file | - | yes |
source_binary_name_template |
A template for the name of the binary before rename after extraction/copying | - | no |
target_binary_name_template |
A template for the name to rename the binary to after extraction/copying | - | no |
strip_path_template |
A template for the path to strip within an archive before extracting | - | no |
Notes:
- The templates have the same instance variables in scope when rendered as mentioned above.
- The extractors map has entries for the following keys:
:zip
: An extractor class for zip files:tar_gz
: An extractor class for tar.gz files:tgz
: An alias for:tar_gz
using the same extractor class:uncompressed
: An extractor class that copies the source to the destination
- The extractor map can be overridden but should include entries for all of the above.
The RakeDependencies::Tasks::Install
tasklib supports the following
configuration parameters:
Name | Description | Default | Required |
---|---|---|---|
name |
The name of the task, required if it should be different from the default | :install |
yes |
dependency |
The name of the dependency, used in status reporting | - | yes |
version |
The version of the dependency to manage, only required if used in templates | - | no |
path |
The path in which to install the dependency | - | yes |
type |
The archive type of the original distribution, one of :zip , :tar_gz , :tgz or :uncompressed |
:zip |
yes |
platform_os_names |
A map of platform OS identifiers to OS names to use in templates | RakeDependencies::PlatformNames::OS |
yes |
platform_cpu_names |
A map of platform CPU identifiers to CPU names to use in templates | RakeDependencies::PlatformNames::CPU |
yes |
binary_directory |
The name of the directory under the supplied path into which to extract/copy the binaries | 'bin' |
yes |
installation_directory |
The name of the directory into which the binary should be installed | - | yes |
binary_name_template |
A template for the name of the binary | - | yes |
Notes:
- The templates have the same instance variables in scope when rendered as mentioned above.
The RakeDependencies::Tasks::Fetch
tasklib supports the following
configuration parameters:
Name | Description | Default | Required |
---|---|---|---|
name |
The name of the task, required if it should be different from the default | :fetch |
yes |
dependency |
The name of the dependency, used in status reporting | - | yes |
download_task |
The full name including namespaces of the download task | <current-namespace>:download |
yes |
extract_task |
The full name including namespaces of the extract task | <current-namespace>:extract |
yes |
The RakeDependencies::Tasks::Fetch
tasklib supports the following
configuration parameters:
Name | Description | Default | Required |
---|---|---|---|
name |
The name of the task, required if it should be different from the default | :fetch |
yes |
dependency |
The name of the dependency, used in status reporting | - | yes |
version |
The version of the dependency to manage, only required if used in templates | - | no |
path |
The path in which to install the dependency | - | yes |
binary_directory |
The name of the directory under the supplied path into which to extract/copy the binaries | 'bin' |
yes |
needs_fetch |
A lambda taking a parameter map that should return true if the dependency needs to be fetched, false otherwise |
Will always return true |
no |
clean_task |
The full name including namespaces of the clean task | <current-namespace>:clean |
yes |
download_task |
The full name including namespaces of the download task | <current-namespace>:download |
yes |
extract_task |
The full name including namespaces of the extract task | <current-namespace>:extract |
yes |
install_task |
The full name including namespaces of the install task | <current-namespace>:install |
yes |
Notes:
- The templates have the same instance variables in scope when rendered as mentioned above.
- The needs_fetch method receives the same parameter map as mentioned above.
After checking out the repo, run bin/setup
to install dependencies. Then,
run rake spec
to run the tests. You can also run bin/console
for an
interactive prompt that will allow you to experiment.
To install this gem onto your local machine, run bundle exec rake install
. To
release a new version, update the version number in version.rb
, and then run
bundle exec rake release
, which will create a git tag for the version, push
git commits and tags, and push the .gem
file to
rubygems.org.
To encrypt a GPG key for use by CircleCI:
openssl aes-256-cbc \
-e \
-md sha1 \
-in ./config/secrets/ci/gpg.private \
-out ./.circleci/gpg.private.enc \
-k "<passphrase>"
To check decryption is working correctly:
openssl aes-256-cbc \
-d \
-md sha1 \
-in ./.circleci/gpg.private.enc \
-k "<passphrase>"
Bug reports and pull requests are welcome on GitHub at https://github.com/infrablocks/rake_dependencies. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the Contributor Covenant code of conduct.
The gem is available as open source under the terms of the MIT License.