Skip to content

Latest commit

 

History

History
538 lines (391 loc) · 18.2 KB

README.md

File metadata and controls

538 lines (391 loc) · 18.2 KB

Extools plugin for Gradle

Introduction

Groovy and Gradle are very good at downloading Java based dependencies from jcenter or Maven central, but they come short when it comes to download external tools (like compilers or installer generators).

The "extools" plugin for Gradle provides a convenient way to do so. It is basically a portable app manager within Gradle. It automatically downloads the tools required by a Gradle build, extracts them and makes them available for execution on the fly, without any installation or changes to the host system.

Prerequisites

The extools plugin has no special dependencies apart from:

  • Gradle 4.6 or later (because of the CLI options)
  • Any Java runtime supported by Gradle. The Java 9 is not recommended because it displays warnings with Groovy code.

You also need a basic understanding of Gradle. More precisely, you need to be able to start a new Gradle project. The recommended way is to use the wrapper, which provides two major benefits to your users: automatic download of Gradle itself (no local installation required), and consistency of the Gradle version used.

Basic usage

Let's assume that you have a Gradle project and you want to call the program myclitool from your build, and that myclitool is provided as part of mytoolkit. Just check for the lastest version available, and add the following lines to your build.gradle:

// Apply the extools plugin (using the plugin syntax introduced in Gradle 2.1)
plugins {
    id 'com.github.ocroquette.extools' version '2.3'
}

// Configure the plugin, assuming a repo URL has already been set as a property (see below)
extools {
    // Define a dependency to the extool called "mytoolkit"
    tool "mytoolkit"
}

// Define a task similar to Gradle's standard "Exec" task, but that one uses some external tools
import com.github.ocroquette.extools.tasks.ExtoolExec
task execMyCliTool(type:ExtoolExec) {
    commandLine "myclitool"
}

// You can also execute an extool from any task, similarly to Gradle's standard "exec {}" statement,
// but in this case, you need to define a dependency to the task "extoolsLoad", which
// loads the required tools metadata
task doStuff {
    dependsOn "extoolsLoad"
    dolast {
        extoolexec {
            commandLine "myclitool", "CLI parameter for first run"
        }
        extoolexec {
            commandLine "myclitool", "CLI parameter for the second run"
        }
    }
}

For this to work, you will need to set the URL of the extools repository. You can do this either by setting an environment variable:

EXTOOLS_REPOSITORY_URL

or by setting a property, typically in gradle.properties:

extools.repositoryUrl=file:/...

or

extools.repositoryUrl=http://

The URL can either be local (file:/local_path) or remote (http://, https://).

Creating extools packages and repositories

There is no central, public extools repository, and there will probably never be any, so you have to create and maintain your own.

Creating an extool package is pretty easy. Just put all the content you need in a directory dir, and add a text file called extools.conf. Here is an example:

dir/bin/myclitool
dir/extools.conf

The file extools.conf allows to extend the environment of the host when the extool is used. In this simple case, extools.conf should contain the following line:

prepend;env;relpath;PATH;bin

When this extool will be used, the bin subdirectory will be added at the beginning of the PATH variable, allowing to find "myclitools".

Warning: on Windows, you should be particularly careful when expanding the PATH, see "Best practice" below.

We now need to package the extool. It is very easy, since the ZIP format is used. Just make sure that extools.conf is at the root of the content of the ZIP file, and that the level "dir" is not used. For instance, when using zip on the command line on a Unix like system:

cd dir
zip -r ../mytoolkit.zip .

Whithin the ZIP file, the file structure should like this:

bin/myclitool
extools.conf

Creating a repository is very easy, just put the file mytoolkit.zip in a directory in the local file system or on an HTTP server, and set extools.repositoryUrl accordingly.

It is recommended to automate the generation of the packages. Gradle itself is the perfect tool since it provides all the required features like unzipping, zipping, file manipulation... See gradle-extools-recipes for sample recipes.

Additional features

Execution options

When executing an extool as ExtoolExec task or with the extoolexec statement, the following options are supported.

Options common to Gradle's standard exec mechanism:

  • commandLine: the full command line with parameters, as a list
  • executable: the name of the executable
  • args: a list of arguments to provide to the executable
  • environment: a map containing the environment variables to set in the child process in addition to System.getenv() and the variables set by the extools
  • standardOutput: the output stream to use for the error stream of the child process
  • errorOutput: the output stream to use for the error stream
  • standardInput: the input stream to use for the input stream of the child process
  • ignoreExitValue: a boolean that indicates if the non zero exit values from the child process must be ignored
  • workingDir: a file or path to set as working directory for the child process

Additional options:

  • usingExtools: the exact list of aliases of the extools to use for the execution
  • usingAdditionalExtools: additional list of aliases to use for the execution (see below)
  • runInBackground: a boolean to indicate if the tool should be run synchronously (false, default) or asynchronously in the background (true)

Specifying which tools to make available for execution

By default, all tools will be used for all tasks and extoolsexec executions.

If you need more control, you can reduce the list of extools available globally in the main configuration:

extools {
    tools "tool1", "tool2", "tool3"
    usingExtools "tool1" // "tool1" will be available in all extools executions by default, but not "tool2" nor "tool3"
}

task exec1(type:ExtoolExec) {
    // Here, only "tool1" is available, from the global configuration
}

task exec2(type:ExtoolExec) {
    usingAdditionalExtools "tool2"  // additional list
    // Here, both "tool1" and "tool2" are available
}

task exec3(type:ExtoolExec) {
    usingExtools "tool3" // explicit list
    // Here, only "tool3" is available
}

The options "usingAdditionalExtools" and "usingExtools" are available for the tasks and for extoolsexec.

This also to control the order in which the tools will be used. For example, if two extools provide the same command in their PATH, and you can specify which one will be found first at runtime using "usingExtools" or "usingAdditionalExtools":

task execTool1(type:ExtoolExec) {
    usingAdditionalExtools "tool1", "tool2"
    // Here "tool1" will have priority
}

task execTool2(type:ExtoolExec) {
    usingAdditionalExtools "tool2", "tool1"
    // Here "tool2" will have priority
}

Any "usingAdditionalExtools" or "usingExtools" have precedence over the global plugin configuration. The tools specified with "usingAdditionalExtools" can overlap with the global configuration, but the order will supersede the global one, so you can use "usingAdditionalExtools" just to control the priorities.

Within a given list, the tools at the left will have priority over the next tools (just like in the PATH environment variable).

Tool aliases

Usually, the real name of the extools will have a version number in it. When you update the tool, you will have to update the name everywhere it is used, in the global extools {} block and in the task definitions. To avoid this, you can define aliases in the main configuration:

extools {
    // "mytoolkit-v1.3" will be loaded from the repository, and made available
    // under the alias "mytoolkit" within the build
    tools "mytoolkit": "mytoolkit-v1.3",
          "othertoolkit" : "othertoolkit-v2.6-rc3"
}

task execMyCliTool(type:ExtoolExec) {
    usingExtools "mytoolkit"
    commandLine "myclitool"
}

Organizing and structuring packages

You can structure the extools in the repository into subfolders if you want. In the build script, they must be refered to with their complete relative path:

extools {
    tools "compiler/gcc-v7.1",
          "lang/perl-5.26",
          "lang/python-2.7.14"
}

You can use aliases as described above to remove or change the naming structure within a given build:

extools {
    tools "dev/gcc":     "compiler/gcc-v7.1",
          "misc/perl":   "lang/perl-5.26",
          "misc/python": "lang/python-2.7.14"
}

Customizing the execution environment

ExtoolExec and extoolexec extend the standard Exec Gradle task, so all the features of the latter are available. Additionally, you can prepend paths to environment variables using prependEnvPaths:

task execMyCliTool(type:Exec) {
	workingDir "../some/folder"

	environment "VAR": "VALUE"
	prependEnvPaths "PATH": new File(workingDir, "subdir")

	commandLine 'myclitool'

 	standardOutput = new ByteArrayOutputStream()

 	ext.output = {
		return standardOutput.toString()
	}
}

extools.conf: Comments

You can add comments by starting lines with the "#" sign:

# Comment
prepend;env;relpath;PATH;bin

extools.conf: Setting and extending environment variables

So far, we only extended the PATH variable in extools.conf, but it is possible to extend any environment variable with a path relative to the root directory of the extracted extool:

# Extend the PATH environment variable with the bin/ sub-directory
prepend;env;relpath;PATH;bin

# Extend the CMAKE_PREFIX_PATH environment variable with the lib/cmake sub-directory
prepend;env;relpath;CMAKE_PREFIX_PATH;lib/cmake

The separator inserted between the different paths in the variable value is the standard separator used by the operating system for the PATH variable, e.g. ';' on Windows and ':' on Linux, macOS and Unix.

Note: "path" is a deprecated synonym of "relpath".

You can also specify an absolute path:

# Extend the PATH environment variable with an absolute path
# on Windows:
prepend;env;abspath;PATH;C:\...
# on Unix like:
prepend;env;abspath;PATH;/dir/...

To set an environment variable to a fixed string:

# Set the environment variable called SOME_VAR to "Value of SOME_VAR"
set;env;string;SOME_VAR;Value of SOME_VAR

You can also prepend a string. The difference with prepending a path is that the standard separator will not be inserted:

prepend;env;string;VARNAME;prefix to add

extools.conf: Using internal variables, not exported to the environment

If you need the variable value only within Gradle and not as an environment variables in the child processes, use var instead of env in the extools.conf file:

# ExtoolExec will set the environment variable MY_VARIABLE to "Value of MY_VARIABLE"
set;env;string;MY_VARIABLE;Value of MY_VARIABLE

# The plugin will NOT set the environment variable MYVAR,
# but you can still access the value with getValue()
set;var;string;MY_VARIABLE;Value of MY_VARIABLE

Local cache and extract directory

By default, the plugin will store downloaded packages and extract them in the user directory .gradle/extools, so that they can be reused, saving time and space. This also allows for offline work. You can specify other directories if required as properties on the command line or in a gradle.properties file.

extools.localCache=<localpath>
extools.extractDir=<localpath>

Alternatively, you can specify them using the following environment variables:

EXTOOLS_LOCAL_CACHE
EXTOOLS_EXTRACT_DIR

Getting current configuration

The plugin adds a task called extoolsInfo that will dump on the console all global settings like the paths used and all referenced extools, with their variables, in YAML format:

$ gradlew extoolsInfo
...
globalconfig:
  repositoryUrl: http://...
  localCache: ...
  extractDir: ...
tools:
  -
    alias: alias_1
    realname: dummy_1
    variables:
      CMAKE_PREFIX_PATH: ...
      DUMMY1_DIR: ...
      DUMMY1_STRING: Value of DUMMY1_STRING
      DUMMY1_VAR: Value of DUMMY1_VAR
      DUMMY_STRING: Value of DUMMY_STRING from dummy_1
      PATH: ...
    variablesToSetInEnv:
      - DUMMY1_DIR
      - DUMMY1_STRING
      - DUMMY_STRING
    variablesToPrependInEnv:
      - CMAKE_PREFIX_PATH
      - PATH
  -
    alias: alias_2
    realname: dummy_1
    ...

Retrieving extool variable values from other tasks

You can retrieve the values set in the extools.conf files from Gradle tasks using getValue():

task accessVariable {
    dependsOn "extoolsLoad"

    doLast {
        println project.extensions.extools.getValue("toolalias", "MY_VARIABLE")
        println project.extensions.extools.getValueWithDefault("toolalias", "MY_VARIABLE", "Default value")
    }
}

Be aware of the build phases, the task definition will be executed by Gradle during the configuration phase. At this point, the task extoolsLoad task didn't run, and the extools configuration is not available. In this case, use Groovy's lazy string evaluation:

${->project.extensions.extools.getValue("toolalias", "MY_VARIABLE")}

Getting the list of loaded aliases

The getLoadedAliases() returns a sorted array containing the aliases of all loaded tools:

task getLoadedAliases {
    dependsOn "extoolsLoad"

    doLast {
        println project.extensions.extools.getLoadedAliases()
    }
}

Checking and resolving aliases

You can check if an alias is available by using the hasAlias() function, and resolve it using the resolvealias() function:

task demonstrateResolveAlias {
    dependsOn "extoolsLoad"

    doLast {
        if ( project.extensions.extools.hasAlias("toolalias") ) {
            println project.extensions.extools.resolveAlias("toolalias")
        }
    }
}

Overriding location of tools

For troubleshooting or development purposes, you might want to use your own copy of an extool instead of the normal one. For that, you can set a project property or an environment variable called "EXTOOL_OVERRIDE" to the corresponding local path. The name is the name of the extool or its alias in uppercase, with special characters replaced by underscores "". For instance, to override the location of "tool1", set:

EXTOOL_TOOL1_OVERRIDE=/some/local/path

If you are unsure of the variable to use, run Gradle with debug output enabled ("-d"), and search for the following line:

Extools: Checking if the location of "..." is overriden with "EXTOOL_..._OVERRIDE"

Overrides defeat the purpose of configuration management, therefore using them will always cause a Gradle warning to avoid them to be overlooked:

Extools: Location of "alias_3" overriden by project property "EXTOOL_ALIAS_3_OVERRIDE" to "/some/local/path"

Getting the local location of a tool

You can get the home directory of a specific tool based on its alias using getHomeDir(). It returns a standard Java File object.

Example:

println project.extensions.extools.getHomeDir("alias").canonicalPath

Generating scripts

You can generate scripts containing the environment variables of tool by using the generateEnvironmentScript() function. On Windows, it might return a string like:

set VAR1=value1
set PATH=...;%PATH%

Interactive use

It is possible to run commands using extools without modifying the build script. The task called "extoolsExec" can be used for that:

gradlew extoolsExec --commandLine=... --usingExtools=...

The command line parameters are separated by spaces. If you need spaces within the arguments, I can put the command line in a script and call the script.

The list of extools is comma separated.

The result is equivalent to the following task in the build script:

task someTaskName(type:ExtoolExec) {
    usingExtools ...
    commandLine ...
}

Creating launchers

If you don't want to start the tools directly from Gradle but rather interactively, you can generate launcher scripts:

task generateLauncher1 {
    dependsOn "extoolsLoad"
    doLast {
        extoollauncher {
            launcherFile "$buildDir/myscript"
        }
    }
}

This will generate a file called "myscript" which contains the commands to set up the environment for the extools. The extoollauncher command supports following parameters:

  • launcherFile: the launcher file to create. The usual extension will be added automatically if required (.bat on Windows, .sh on other platforms), and directories will be created automatically.
  • usingExtools (optional): the list of extools to use
  • additionalEnvironment (optional): additional environment variables, for instance "NAME": "VALUE"
  • textBefore (optional): Text to add before the generated part
  • textAfter (optional): Text to add after the generated part

Best practice

PATH and Windows

On Windows, you should be particularly careful when expanding the environment variable PATH, since the operating system uses it also to resolve the dynamic libraries (DLL). It can have unwanted side-effects on other applications. If you need only Gradle to find the executables, and not its subprocesses, then use a simple variable instead:

prepend;var;relpath;PATH;...

It will be used only internally to find the executables and will not affect the other tools or subprocesses.

Another option is to define the path under another name:

set;env;relpath;MYTOOL_PATH;bin

and adapt the subprocesses to use this environment variable, instead of the PATH.

Symbolic links

If symbolic links are required within an Extool ZIP file, which is the case for instance when using frameworks on macOS, create it with:

/usr/bin/zip --symlinks

It will be extracted with "/usr/bin/unzip", which retains links (and permissions).

(since v1.30)