Forked JVM-based Execution
It is possibly to simplify the execution of applications on forked JVMs. Typical cases are running specific JVM-based tools or scripts of a JVM-based scripting language such as JRuby.
Execution specifications
Execution specifications are used for configuring the necessary details for running an external process.
- 
AbstractJvmExecSpec is the core class to extend in order to implement JVM-based execution specifications. 
- 
AbstractJvmExecTask is the core class to extend in order to implement JVM-based execution tasks. 
- 
AbstractJvmScriptExecSpec is the core class to extend in order to implement JVM-based script specifications. 
- 
AbstractJvmScriptExecTask is the core class to extend in order to implement JVM-based script tasks. 
Configuration
Assuming that you have a task type myJvmTask that is of a task type that extends AbstractJvmExecTask you can configure all the relevant settings as follows:
myJvmTask {
  jvm {  (1)
  }
  runnerSpec { (2)
  }
  entrypoint { (3)
  }
  process { (4)
  }
}| 1 | Configures all of the JVM fork options.
It is of type JavaForkOptionsWithEnvProvider and similar to JavaForkOptions, but with the addition of environment variable providers. | 
| 2 | Configures the command-line arguments for executing the class.
This is not the JVM arguments, which need to be configured in jvminstead.
It is of type CmdlineArgumentSpec and similar in nature to theargskind of methods in anExecSpec. | 
| 3 | Configures the main class and the classpath for the execution.
It is of type JvmEntryPoint and have similar methods to mainClassandclasspaththat are in `JavaExecSpec. | 
| 4 | Configures the actual execution of the process such as setting the streams. It is of type ProcessExecutionSpec | 
In addition to the above, a script task includes a script configuration block
myJvmTask {
  script { (1)
    name = 'install' (2)
    path = '/path/to/script' (3)
  }
}| 1 | Configures the script name or path as well as any arguments. It is of type JvmScript. | 
| 2 | It is possible to just specify the name of the script. In this case is up to the specific implementation on how to resolve its location. | 
| 3 | It is also possible to specify a path instead of a script name. If both a name and a path is specified, the name takes preference. | 
Configuring the post-process
When the execution method is JAVA_EXEC, additional configuration can be done to handle the post process actions.
process {
  ignoreExitValue = true (1)
  output { (2)
    capture() (3)
    captureAndForward() (4)
    captureTo( project.provider { -> file('build/this-output.txt') }) (5)
    forward() (6)
    noOutput() (7)
  }
  errorOutput { (8)
    capture()
    captureAndForward()
    captureTo( project.provider { -> file('build/this-output.txt') })
    forward()
    noOutput()
  }
  afterExecute { (9)
  }
}| 1 | Whether to ignore the exit value. | 
| 2 | Configure the standard output. | 
| 3 | Capture the output. | 
| 4 | Capture the output and forward the output to console. | 
| 5 | Write the output to the file. | 
| 6 | Forward the output to console. | 
| 7 | Do not produce any output. | 
| 8 | Configure the error output. | 
| 9 | Add one of more closures/actions to process output.
This is called when the output or error output is captured.
It is passed a ExecOutputinstance. | 
Use execution specification for workers
You can use an execution specification that is derived from AbstractJvmExecSpec to run a worker.
This helper is specifically for targeting classes which take a command-line set of parameters as a String[].
For this you will need WorkerAppCmdline.
Here is an example of how this can be used to execute JRuby.
@CompileStatic
abstract class JRubyStandardExecutor implements WorkerAppCmdline {
    @Override
    void runWith(String[] args) {
        final runnable = new org.jruby.Main() (1)
        final status = runnable.run(parameters.applicationArguments.get() as String[]).status
        if(status) {
            throw new RuntimeException("Process exited with code ${status}")
        }
    }
}| 1 | Deal with the correct class instantiation.
Sometimes the main class entrypoint mainexits withSystem.exitcall which is a no-no for running side a worker.
Also note that the main class that is specified in the execution specification will be ignored. | 
Once you have the worker class setup, you can now invoke it directly from the execution specification
AbstractJvmExecSpec execSpec (1)
WorkerExecutor worker (2)
final promise = execSpec.submitToWorkQueue(
    WorkerIsolation.CLASSPATH, (3)
    worker,
    JRubyStandardExecutor (4)
)
promise.await() (5)| 1 | Assuming an execution specification derived from AbstractJvmExecSpec. | 
| 2 | Assuming that you have injected a Gradle WorkerExecutorsomewhere. | 
| 3 | You can also pass WorkerIsolation.OUT_OF_PROCESS.
These are the only two supported modes. | 
| 4 | The worker class that implements WorkerAppCmdline. | 
| 5 | If needed, you can wait on the work queue to complete. |