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.

diagram
Figure 1. Execution Specifications

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 jvm instead. It is of type CmdlineArgumentSpec and similar in nature to the args kind of methods in an ExecSpec.
3 Configures the main class and the classpath for the execution. It is of type JvmEntryPoint and have similar methods to mainClass and classpath that 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 ExecOutput instance.

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 main exits with System.exit call 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 WorkerExecutor somewhere.
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.