/tascalate-javaflow

Continuations / CoRoutines for Java 1.6 - 23, build tools, CDI support. This project is based on completely re-worked Apache Jakarta Commons JavaFlow library.

Primary LanguageJavaApache License 2.0Apache-2.0

Maven Central GitHub release license

IMPORTANT NOTICE FOR RELEASE 2.7.0!!!

  • Agents & Tools projects were refactored to follow consisntent naming of classes and to avoid package name clashes
  • Separate provider was extracted from CdiProxy to allow reuse in mutliple scenarious (not only in Java Agent)
  • net.tascalate.javaflow.provider.asm[VERSION] artifacts are deprecated now and scheduled for removal in next release

IMPORTANT NOTICE FOR RELEASE 2.5.0!!!

Java 9+ is fully supported now, all artefacts are modular multi-release JAR-s that works correctly with Java versions 1.6 to 17. The library code was tested with JDK 17, and all features of the Java 17 bytecode (including nest of inner classes, sealed classes, records) works correctly.

IMPORTANT NOTICE FOR RELEASE 2.3.0!!!

Continuation support for Java

This project contains libary, tools and examples to develop Java applications using continuations.

According to Wikipedia "a continuation is an abstract representation of the control state of a computer program; a continuation reifies the program control state, i.e. the continuation is a data structure that represents the computational process at a given point in the process's execution; the created data structure can be accessed by the programming language, instead of being hidden in the runtime environment"

In other words, continuation allows you to capture execution state of the program (local variables, execution stack, program counters etc.) at the certain places and later resume execution from the saved point. Unfortunately, Java has no support for first-class continuations, but it can be added via bytecode instrumentation (like this library did)

The project is based on the completely reworked Apache Jakarta-Commons JavaFlow project (http://commons.apache.org/sandbox/commons-javaflow/). Below is a list of major changes:

  1. Original JavaFlow instruments bytecode of each and every method available for tooling to add Continuations support. This adds significant overhead to both code size and execution time. The reworked version requires developer to mark Continuation-aware classes/methods explicitly with annotations.
  2. Code is updated to use new bytecode instrumentation tools (ObjectWeb ASM 5.x); support for BCEL is discontinued
  3. Codebase is split to separate modules: run-time API, instrumentation provider SPI, offline instrumentation tools (Maven, Ant, command-line), run-time instrumentation JavaAgent, utilities and examples
  4. The library adds support for recent Java features like lambdas and dynamic dispatch

Maven

You have to add the following configuration to enable build-time instrumentation of classes during Maven build:

<dependencies>
	<dependency>
		<groupId>net.tascalate.javaflow</groupId>
		<artifactId>net.tascalate.javaflow.api</artifactId>
		<version>2.7.6</version>
	</dependency>
	<!-- Add-on for Java 8 and above -->
	<dependency>
		<groupId>net.tascalate.javaflow</groupId>
		<artifactId>net.tascalate.javaflow.extras</artifactId>
		<version>2.4.3</version>
	</dependency>	
	...
</dependencies>

<build>
	<plugins>
		<plugin>
			<groupId>org.apache.maven.plugins</groupId>
			<artifactId>maven-compiler-plugin</artifactId>
			<configuration>
				<!-- versions 1.6 -- 23 are supported -->
				<source>1.6</source>
				<target>1.6</target>
			</configuration>
		</plugin>
		<plugin>
			<groupId>net.tascalate.javaflow</groupId>
			<artifactId>net.tascalate.javaflow.tools.maven</artifactId>
			<version>2.7.6</version>
			<executions>
				<execution>
					<id>javaflow-enhance-main-classes</id> 
					<phase>process-classes</phase>
					<goals>
						<goal>javaflow-enhance</goal>
					</goals>
				</execution>
				<!-- Only if you need to enhance test classes -->
				<execution>
					<id>javaflow-enhance-test-classes</id> 
					<phase>process-test-classes</phase>
					<goals>
						<goal>javaflow-enhance</goal>
					</goals>
				</execution>				
			</executions>
		</plugin>
	</plugins>
</build>

Note that if you are using continuations with Java 1.8 lambdas then you need to add Tascalate JavaFlow instrumentation agent at run-time as command-line option, while lambda-related classes are generated by JVM on the fly and there is no other way to instrument them. If this is not an option, then you can de-sugar all lambdas with RetroLambda Maven plugin at build-time (RetroLambda output is supported by Tascalate JavaFlow 2.3.2 or higher).

Please refer to pom.xml in examples project for typical Maven configuration.

Gradle

To use continuations bytecode enhancer with Gradle you need to do the following. First, specify JavaFlow gradle plugin dependency in build.gradle:

buildscript {
    repositories {
        mavenCentral()
    }

    dependencies {
      classpath 'net.tascalate.javaflow:net.tascalate.javaflow.tools.gradle:2.7.6'
    }
}

Next, apply the continuations plugin and specify the dependency to the net.tascalate.javaflow.api:

apply plugin: "java"
apply plugin: "continuations"

repositories {
    mavenCentral()
}

dependencies {
    implementation 'net.tascalate.javaflow:net.tascalate.javaflow.api:2.7.6'
}

Ant

There is a separate Ant task for applying JavaFlow instrumentation at build-time.

It's possibe to instrument compiled Java classes as below:

<target name="instrument-classes" description="JavaFlow Instrumentation" depends="compile">
    <taskdef name="javaflow" 
    classname="org.apache.commons.javaflow.tools.ant.ContinuableClassesInstrumentationTask" 
    classpathref="ant-lib-classpath"/>
    <echo message="JavaFlow instrumentation of compiled classes in ${classes.dir}" />
    <javaflow srcdir="${classes.dir}" destdir="${i-classes.dir}" classpathref="classpath"/>
</target>

... as well as re-write packaged JAR file:

<target name="instrument-jar" description="JavaFlow Instrumentation" depends="jar">
    <taskdef name="javaflow" 
    classname="org.apache.commons.javaflow.tools.ant.ContinuableClassesInstrumentationTask" 
    classpathref="ant-lib-classpath"/>
    <echo message="JavaFlow instrumentation of compiled classes in ${jar.dir}/${ant.project.name}.jar" />
    <javaflow srcdir="${jar.dir}" destdir="${i-jar.dir}" classpathref="classpath"/>
</target>

You may download a complete examples project setup from the latest release for complete configuration template. Please pay attention to ant-lib folder with Ant TaskDef and lib folders with compile-/runtime-dependencies.

Java Instrumentation Agent (Runt-time Instrumentation)

As an alternative to compile-time bytecode instrumentation, you MAY use Tascalate JavaFlow Instrumentation Agent from the latest release to enable continuations support at class-loading time. Please note, that if you are using Java 8 and creating continuable lambda functions (either anonymous or/and as method references), and you don't replace them with tools like RetroLambda as mentioned above, then you SHOULD use this instrumentation agent always: as long as Java run-time generates implementation of functional interfaces on the fly there is no other option to instrument them. To enable Tascalate JavaFlow Instrumentation Agent please add the following arguments to Java command line:

java -javaagent:<path-to-jar>/javaflow.instrument-continuations.jar <rest-of arguments>

The agent JAR file includes all necessary dependencies and requires no additional CLASSPATH settings. It's recommended to use this agent in conjunction with either Maven or Ant build tools supplied to minimize the associated overhead of the instrumentation during class-loading process at run-time.

Another useful application of the instrumentation agent is to apply it for debugging code within your IDE of choice. Just specify the "-javaagent" option listed above in your IDE debug/run configuration and you will be able to perform quick "debug-fix" loops without executing full project rebuild.

Command-line tools

It's possible to use a stand-alone command-line utility JavaFlowRewriteJar.jar to instrument JAR archives containing continuable classes. Please use the following command:

java -jar JavaFlowRewriteJar.jar src1.jar dst1.jar src2.jar dst2.jar...

Note, that the source and the destination should be different files.

CDI Support

To work correctly in CDI environment continuable methods should be advised only by continuation-aware CDI proxies (interceptors, scope proxies, etc). Obviously, generation of these proxies is out of our control. Plus, major CDI containers (JBoss Weld and Apache OpenWebBeans) generates such proxies dynamically at run-time. Therefore if you plan to use Tascalate JavaFlow continuations with managed beans' methods then it's necessary to instrument CDI-specific proxies with javaflow.instrument-proxies.jar Java Agent:

java -javaagent:<path-to-jar>/javaflow.instrument-proxies.jar <rest-of arguments>

Please note, that CDI-specific agent neither requires javaflow.instrument-continuations.jar to operate correctly nor provides class file transformers for continuable methods. So if your project runs with CDI environment AND uses Java 8 lambdas then you have to add 2 Java agents, every serving different purpose:

java -javaagent:<path-to-jar>/javaflow.instrument-continuations.jar \
     -javaagent:<path-to-jar>/javaflow.instrument-proxies.jar \
     <rest-of arguments>

CDI functionality is tested with JBoss Weld 2.x - 3.1.7, 4.0.x and Apache OpenWebBeans 1.6.x - 2.0.23. Contribution for other CDI/CDI-like containers (Spring, Google Guice, etc) is welcome.

More documentation & exmaples

Examples of the library usage may be found in the Tascalate JavaFlow Examples project. The covered topics are common tasks, inheritance, lambdas support, proxies, usage with CDI containers like JBoss Weld and Apache OpenWebBeans.

For additional documentation, tutorials and guidelines please visit my blog