IPC EventBus provides a simple EventBus API for intra-JVM and extra-JVM communication. You can use it within a JVM as a simple EventBus or between JVM for communication between process. Besides, this project comes with several builders to easily start process and bind them an event bus.
The goal of this project is to be:
-
self-contained (no external dependency)
-
simple
-
efficient
-
targets small usages, i.e. integration tests, system tests, which request several process synchronization and classpath isolation
-
Java 8 Compatible, so that you can use it in any TC project
Licensed under the Apache License, Version 2.0 © Terracotta, Inc., a Software AG company. Copyright IBM Corp. 2024, 2025
See also: http://www.github.com/ehcache/ehcache3
Fork, then:
git clone [email protected]:Terracotta-OSS/ipc-eventbus.git
mvn clean install
<dependency>
<groupId>org.terracotta</groupId>
<artifactId>ipc-eventbus</artifactId>
<version>X.Y</version>
</dependency>
This project contains several modules:
-
I/O Stuff
An EventBus API with a local and remote implementation:
-
Local EventBus
-
Remote EventBus
Process launching mechanisms:
-
Process Launching
-
Java Process Launching
And a way to start a Java process which can interact with its launcher through an event bus:
-
[Inter Java Process Communication](#ipc)
Pipe
Starts a thread that will copy input data into an output stream.
Pipe pipe = new Pipe("collect stdout", inputStream, outputStream);
pipe.waitFor(); // if you need, you can wait for the pipe to finish
pipe.close(); // close the pipe (does not close the stream!)
MultiplexOutputStream
Create an OutputStream
which will write into several output streams sequentially.
MultiplexOutputStream plex = new MultiplexOutputStream()
.addOutputStream(outputStream)
.addOutputStream(outputStream2)
.addOutputStream(System.out);
Usage:
plex.getOutputStreams(); // lists the streams
plex.isEmpty(); // true if no streams
plex.streamCount(); // number of streams
plex.write(data); // MultiplexOutputStream is an OutputStream
Create a local EventBus
by using a builder:
EventBus eventBus = new EventBus.Builder()
.onError(new PrintingErrorListener(System.err)) // OPTIONAL: print listener exceptions
.onError(new RethrowingErrorListener()) // OPTIONAL: rethrow listener exceptions immediately (by default)
.id("bus-id") // OPTIONAL: bus id (otherwise a UUID is generated)
.build();
eventBus.getId(); // returns the eventbus id
Implement EventListener
interface to listen to events
EventListener listener = new EventListener() {
@Override
public void onEvent(Event e) {
}
};
Bind your listener to events
eventBus.on(listener); // register a listener for all events
eventBus.on("my.event", listener); // register a listener for a specific event
You can dump (debug) what is going on
eventBus.on(new EventListenerSniffer()); // listener which dumps to the console all events (for debug)
Unbind events and listeners
eventBus.unbind(listener); // removes a listener from all events
eventBus.unbind("my.event"); // removes all listeners for this event
eventBus.unbind("my.event", listener); // removes a specific listener from an event
And, of course, trigger events!
eventBus.trigger("my.event"); // can trigger an event
eventBus.trigger("my.event", "some data"); // event with data (must be serializable)
Here is what you can do with the Event
object received by EventListener
implementations:
event.getData(); // the data
event.getData(String.class); // can cast the data in the wanted type
event.getName(); // the name of the event triggered
event.getSource(); // the ID of the eventbus
event.getTimestamp(); // time in ms of the event
event.isUserEvent(); // check if it is a user event. You might listen to system events such as eventbus.server.close, eventbus.client.connect, eventbus.client.disconnect
RemoteEventBus
have the same builder options that a local EventBus
but serves as inter-process communication through a socket. One EventBus
acts as a server and several clients can connect to it.
Clients cannot talks to each-other. This is only a client-server communication, so any events triggers from a client will arrive on the server and any events triggered from the server will then be propagated to all clients.
Server creation:
EventBusServer server = new EventBusServer.Builder()
.id("peer1") // OPTIONAL: bus id
.bind("0.0.0.0") // OPTIONAL: bind address
.listen(56789) // OPTIONAL: port to listen to. Default to 56789
.listenRandom() // OPTIONAL: choose a random port for listening
.build();
Client creation
EventBusClient client = new EventBusClient.Builder()
.id("peer2")
.connect(56789) // OPTIONAL: port to connect to
.connect("localhost", 56789) // OPTIONAL: port and host to connect to. Default is localhost:56789
.build();
If nothing is given in the builders, EventBus
will try to use the system property ipc.bus.host
for the host to connect to and ipc.bus.port
for the port to connect to (or listen).
If no system property is found, localhost
is used for the host and 56789
is used for the port.
Creates a Java process, similar to ProcessBuilder
but has several improvements to access stdout, stderr and stdin of the process, cache them, forward them, access the process PID, etc.
AnyProcess process = AnyProcess.newBuilder()
.command("bash", "-c", "sleep 3; echo $VAR")
.recordStdout() // OPTIONAL: save stdout from process for getStdout() (disabled by default). Disables getInputStream().
.recordStderr() // OPTIONAL: save stderr from process for getStderr() (disabled by default). Disabled getErrorStream().
.env("key", "value") // OPTIONAL: add a env. variable
.env(new HashMap<String, String>()) // OPTIONAL: se ta new env
.pipeStderr() // OPTIONAL: send stderr to the console
.pipeStderr(outputStream) // OPTIONAL: send stderr to a stream. You can both collect and pipe.
.pipeStdout() // OPTIONAL: send stdout to the console
.pipeStdout(outputStream) // OPTIONAL: send stdout to a stream. You can both collect and pipe.
.pipeStdin() // OPTIONAL: will bind process stdin to this process stding
.pipeStdin(inputStream) // OPTIONAL: will bind process stdin to this input stream
.redirectStderr() // OPTIONAL: treat stderr like stdout (both merged into stdout)
.workingDir(new File(".")) // OPTIONAL: change the working directly. Same as current process by default
.build();
Accessible methods:
process.destroy(); // destroy (kill with SIGTERM) the process
process.exitValue(); // the process exit value, when available
process.getCommand(); // the process command
process.getErrorStream();
process.getInputStream();
process.getOutputStream();
process.getPid(); // get the process PID
process.getStderr(); // if collected, get the stderr of the process
process.getStderrText(); // if collected, get the stderr of the process as a String
process.getStdout(); // if collected, get the stdout of the process
process.getStdoutText(); // if collected, get the stdout of the process as a String
process.getWorkingDirectory();
process.isDestroyed(); // check if process is destroyed
process.isRunning(); // check if process is still running
process.waitFor(); // wait and block while process finished
process.waitForTime(1, TimeUnit.MINUTES); // wait for the process to finish or timeout
You can also use a Java Future
:
Future future = process.getFuture(); // get a future representing the process execution. You can cancel (=destroy) the process or wait for its completion
Another builder allows you to quickly start a Java main class with specific env and system properties. You can access the same builder methods as above.
JavaProcess javaProcess = JavaProcess.newBuilder()
.mainClass("my.corp.Echo")
.addClasspath(Echo.class) // add a classpath entry from a class (find the enclosing jar or folder)
.arguments("one", "two") // add some program arguments
.env("VAR", "Hello") // add some env. variable
.addJvmProp("my.prop", "world") // add some jvm props
.addJvmArg("-Xmx512m") // add some jvm flags
.pipeStdout() // you can access all process builder methods seen above
.pipeStderr()
.recordStdout()
.recordStderr()
.build();
Java home and Java executable can be automatically discovered, but you can override them in the builder.
javaProcess.getJavaExecutable(); // automatically resolved from java home, but you can override it in the builder
javaProcess.getJavaHome(); // automatically resolved from java home, but you can override it in the builder
This builder allows you to start any main class linked to a remote EventBus
to be able to communicate with some other processes.
Special events
Each child process will listen to the event process.exit
so that you can force a child process to exit like this:
myProcess.trigger("process.exit");
// equivalent to
myProcess.trigger("process.exit", 0);
// or with a code:
myProcess.trigger("process.exit", 1);
The event process.exiting
is fired by the child process when exiting.
When the process has fully exited, an event process.exited
is fired.
But if the parent process calls process.destroy()
to kill the child process, then the event process.destroyed
will be fired after the process is destroyed by the SIGTERM signal.
Full Example
Create your main class. From there, you can access the EventBus
statically. The event bus is connected to the parent process. So each event you send will be propagated, and you can listen to events sent by the parent process also.
public class EchoEvent2 {
public static void main(String[] args) throws Exception {
Bus.get().on("ping", new EventListener() {
@Override
public void onEvent(Event e) {
Bus.get().trigger("pong", e.getData());
}
});
Thread.sleep(2000);
}
}
Then, just launch this main class by using the EventJavaProcess
builder. It extends all the JavaProcess
and AnyProcess
classes so you may want to also configure additional things.
EventJavaProcess process = EventJavaProcess.newBuilder()
.mainClass(EchoEvent2.class) // set main class to start and add it to classpath
.pipeStdout() // echo stdout
.pipeStderr() // echo stderr
.debug() // activate debug mode for ipc eventbus
.build();
assertTrue(process.isEventBusConnected());
And communicate with the child process like this:
process.on("process.exiting", new EventListener() {
@Override
public void onEvent(Event e) throws Throwable {
System.out.println("Exiting...");
}
});
process.on("process.exited", new EventListener() {
@Override
public void onEvent(Event e) throws Throwable {
System.out.println("Exited.");
}
});
process.on("pong", new EventListener() {
@Override
public void onEvent(Event e) throws Throwable {
System.out.println(e.getData());
}
});
process.trigger("ping", "hello");
process.trigger("process.exit");
process.waitFor();
You should see some output like this:
1440379569484 [11842] [main] ping@11842 at 1440379569484 - hello
1440379569485 [11844] [client-acceptor] eventbus.client.connect@11844 at 1440379569484 - localhost:62978
EchoEvent: Event{name='eventbus.client.connect', source=11844, data=localhost:62978}
1440379569489 [11842] [main] exit@11842 at 1440379569489
1440379569496 [11844] [reader@localhost:62978] pong@11844 at 1440379569496 - hello
EchoEvent: Event{name='pong', source=11844, data=hello}
1440379569499 [11844] [reader@localhost:62978] ping@11842 at 1440379569484 - hello
1440379569500 [11842] [reader@11842] pong@11844 at 1440379569496 - hello
EchoEvent: Event{name='ping', source=11842, data=hello}
1440379569842 [11842] [reader@11842] eventbus.client.disconnect@11842 at 1440379569842