This is a client library to handle the API of the HIRO Graph.
This library also contains classes for handling the WebSockets event-ws
and action-ws
API.
Status
- Technical preview
For more information about HIRO Automation, look at https://www.arago.co/
For more information about the APIs this library covers, see https://developer.hiro.arago.co/7.0/api/ (Currently
implemented are app
, auth
, graph
, event-ws
and action-ws
)
You need at least Java 11.
https://github.com/arago/java-project and the respective packages under https://github.com/orgs/arago/packages or the Maven Central Library.
To use this library, you will need an account at https://id.arago.co/ and access to an OAuth Client-Id and Client-Secret to access the HIRO Graph. See also https://developer.hiro.arago.co.
Most of the documentation is done in the sourcecode.
import co.arago.hiro.client.connection.token.PasswordAuthTokenAPIHandler;
import co.arago.hiro.client.exceptions.HiroException;
import co.arago.hiro.client.rest.GraphAPI;
import co.arago.hiro.client.model.vertex.HiroVertexListMessage;
import java.io.IOException;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// Build an API handler which takes care of API paths via /api/versions and security tokens.
try (PasswordAuthTokenAPIHandler handler = PasswordAuthTokenAPIHandler
.newBuilder()
.setRootApiUri(API_URL)
.setCredentials(USERNAME, PASSWORD, CLIENT_ID, CLIENT_SECRET)
.build()) {
// Use the actual API you want with the handler
GraphAPI graphAPI = GraphAPI.newBuilder(handler)
.build();
// Execute a command from GraphAPI
HiroVertexListMessage queryResult = graphAPI
.queryVerticesCommand("ogit\\/_type:\"ogit/MARS/Machine\"")
.setLimit(4)
.execute();
System.out.println(queryResult.toPrettyJsonString());
}
}
}
Authorization against the HIRO Graph is done via tokens. These tokens are handled by classes of
type TokenAPIHandler
in this library. Each of the Hiro-Client-Objects (GraphAPI
, AuthAPI
, etc.) need to
have some kind of TokenApiHandler in their builder for construction.
This library supplies the following TokenApiHandlers:
A simple TokenApiHandler that is generated with a preset-token at construction. Cannot update its token.
A TokenApiHandler that reads an environment variable (default is HIRO_TOKEN
) from the runtime environment. Will only
update its token when the environment variable changes externally.
This TokenApiHandler logs into the HiroAuth backend and obtains a token from login credentials. This is also the only TokenApiHandler (so far) that automatically tries to renew a token from the backend when it has expired.
This TokenApiHandler can be used to handle OAuth2 authorization_code
flow. It automatically creates an
Authorization-URI for a web-browser (which usually leads to a login page of the Authorization Server) and handles the
incoming values code
and state
from the redirection of the Authorization Server. It then uses these values to obtain
the final access_token. This access_token will be refreshed automatically via refresh_token when the access_token
expires.
All code examples in this documentation can use these TokenApiHandlers interchangeably, depending on how such a token is provided.
The GraphAPI example from above with another customized TokenApiHandler:
import co.arago.hiro.client.connection.token.EnvironmentTokenAPIHandler;
import co.arago.hiro.client.exceptions.HiroException;
import co.arago.hiro.client.rest.GraphAPI;
import co.arago.hiro.client.model.vertex.HiroVertexListMessage;
import java.io.IOException;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// Build an API handler which takes care of API paths via /api/versions and security tokens.
try (EnvironmentTokenAPIHandler handler = EnvironmentTokenAPIHandler
.newBuilder()
.setRootApiUri(API_URL)
.build()) {
// Use the actual API you want with the handler
GraphAPI graphAPI = GraphAPI.newBuilder(handler)
.build();
// Execute a command from GraphAPI
HiroVertexListMessage queryResult = graphAPI
.queryVerticesCommand("ogit\\/_type:\"ogit/MARS/Machine\"")
.setLimit(4)
.execute();
System.out.println(queryResult.toPrettyJsonString());
}
}
}
This TokenApiHandler is also responsible to determine the most up-to-date paths for the API calls. If you want to have a specific path for your API, you have to set it up with the Builder of that API Client, like
AuthAPI.newBuilder(handler).setApiPath("/api/auth/6.1").build();
Releasing all resources assigned to a connection is done by closing the TokenAPIHandler
(unless the httpClient
is
provided externally, see External HTTP Client). You cannot close the API Clients (
i.e. GraphAPI
), because the handler carries the information about a connection to HIRO, and it can be shared between
multiple API Clients.
When you need to access multiple APIs, it is a good idea to share the TokenApiHandler between them. This avoids unnecessary API version requests and unnecessary token requests with the PasswordAuthTokenAPIHandler for instance.
import co.arago.hiro.client.connection.token.PasswordAuthTokenAPIHandler;
import co.arago.hiro.client.exceptions.HiroException;
import co.arago.hiro.client.rest.AuthAPI;
import co.arago.hiro.client.rest.GraphAPI;
import java.io.IOException;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// Build an API handler which takes care of API paths via /api/versions and security tokens.
try (PasswordAuthTokenAPIHandler handler = PasswordAuthTokenAPIHandler
.newBuilder()
.setRootApiUri(API_URL)
.setCredentials(USERNAME, PASSWORD, CLIENT_ID, CLIENT_SECRET)
.build()) {
// Use the actual APIs you want with the handler
GraphAPI graphAPI = GraphAPI.newBuilder(handler)
.build();
AuthAPI authAPI = AuthAPI.newBuilder(handler)
.build();
}
}
}
When you need to create a token externally, it is a good idea to share a GraphConnectionHandler between TokenAPIHandlers. The connection, cookies and API version information will be shared among them. Be aware, that you should close the connection via the GraphConnectionHandler here, not via the TokenAPIHandlers.
import co.arago.hiro.client.connection.token.FixedTokenAPIHandler;
import co.arago.hiro.client.rest.GraphAPI;
import co.arago.hiro.client.connection.GraphConnectionHandler;
import java.io.IOException;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// Build a connection without any token handling
try (GraphConnectionHandler connectionHandler = GraphConnectionHandler
.newBuilder(httpClientHandler)
.setRootApiUri(API_URL)
.build()) {
GraphAPI graphAPI_user1 = GraphAPI.newBuilder(
FixedTokenAPIHandler.newBuilder()
.setSharedConnectionHandler(connectionHandler)
.setToken("TOKEN_1"))
.build();
GraphAPI graphAPI_user2 = GraphAPI.newBuilder(
FixedTokenAPIHandler.newBuilder()
.setSharedConnectionHandler(connectionHandler)
.setToken("TOKEN_2"))
.build();
}
}
}
An external httpClient can be provided to the HttpClientHandler via its Builder using
setHttpClient(HttpClient client)
. If this is the case and httpClientAutoClose is not set to true
, a call
to close()
of such a handler will have no effect. Its removal then has to be handled externally.
Example with an external httpClient:
import co.arago.hiro.client.Config;
import co.arago.hiro.client.connection.httpclient.DefaultHttpClientHandler;
import co.arago.hiro.client.connection.token.PasswordAuthTokenAPIHandler;
import co.arago.hiro.client.exceptions.HiroException;
import co.arago.hiro.client.rest.GraphAPI;
import co.arago.hiro.client.model.vertex.HiroVertexListMessage;
import java.io.IOException;
import java.net.http.HttpClient;
import java.util.concurrent.Executors;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// The external httpClient
HttpClient httpClient = HttpClient.newBuilder()
.executor(Executors.newFixedThreadPool(1))
.build();
// Pass the external httpClient and set httpClientAutoClose, so it gets closed
// when the TokenAPIHandler closes.
try (PasswordAuthTokenAPIHandler handler = PasswordAuthTokenAPIHandler
.newBuilder()
.setHttpClient(httpClient)
.setHttpClientAutoClose(true) // If this is missing, the httpClient will not be closed.
.setRootApiUri(API_URL)
.setCredentials(USERNAME, PASSWORD, CLIENT_ID, CLIENT_SECRET)
.build()) {
// Use the actual API you want with the handler
GraphAPI graphAPI = GraphAPI.newBuilder(handler)
.build();
// Execute a command from GraphAPI
HiroVertexListMessage queryResult = graphAPI
.queryVerticesCommand("ogit\\/_type:\"ogit/MARS/Machine\"")
.setLimit(4)
.execute();
System.out.println(queryResult.toPrettyJsonString());
}
}
}
The API Clients are mostly straightforward to use, since all public methods of these classes represent an API call. Documentation is available in source code as well.
Currently, supported are
For usage, see the examples above.
There are two websocket clients supported:
Both of them have special client classes in this library.
When messages from the websockets are received, they are handled in sequential order in their own thread. This can prove to be a problem when you need to react to those messages swiftly. Isolating each handling of an incoming message in its own thread is not subject of this library and has to be done by the project using it.
The event websocket gets triggered when changes to vertices happen. These can be CREATE
, UPDATE
or DELETE
.
Example skeleton code:
import co.arago.hiro.client.Config;
import co.arago.hiro.client.connection.token.PasswordAuthTokenAPIHandler;
import co.arago.hiro.client.exceptions.HiroException;
import co.arago.hiro.client.model.websocket.events.impl.EventsMessage;
import co.arago.hiro.client.websocket.EventWebSocket;
import co.arago.hiro.client.websocket.listener.EventWebSocketListener;
import java.io.IOException;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// Build an API handler which takes care of API paths via /api/versions and security tokens.
try (PasswordAuthTokenAPIHandler handler = PasswordAuthTokenAPIHandler.newBuilder()
.setRootApiUri(API_URL)
.setCredentials(USERNAME, PASSWORD, CLIENT_ID, CLIENT_SECRET)
.build()
) {
try (EventWebSocket eventWebSocket = EventWebSocket.newBuilder(
handler,
new EventWebSocketListener() {
@Override
public void onCreate(EventsMessage eventsMessage) {
// React when a vertex has been created
}
@Override
public void onUpdate(EventsMessage eventsMessage) {
// React when a vertex has been updated
}
@Override
public void onDelete(EventsMessage eventsMessage) {
// React when a vertex has been deleted
}
})
.addEventsFilter(
"default",
"(element.ogit/_type = ogit/Automation/AutomationIssue)"
)
.build()
) {
eventWebSocket.start();
// Listen for 1 second for incoming events.
Thread.sleep(1000);
}
}
}
}
If you do not set a scope via .addScope(...)
, the default scope of your account will be used unless
setAllScopes(true)
is used. If you need to set a scope by hand, you can use the following:
import co.arago.hiro.client.Config;
import co.arago.hiro.client.connection.token.PasswordAuthTokenAPIHandler;
import co.arago.hiro.client.exceptions.HiroException;
import co.arago.hiro.client.model.websocket.events.impl.EventsMessage;
import co.arago.hiro.client.websocket.EventWebSocket;
import co.arago.hiro.client.websocket.listener.EventWebSocketListener;
import co.arago.hiro.client.model.token.DecodedToken;
import java.io.IOException;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// Build an API handler which takes care of API paths via /api/versions and security tokens.
try (PasswordAuthTokenAPIHandler handler = PasswordAuthTokenAPIHandler.newBuilder()
.setRootApiUri(API_URL)
.setCredentials(USERNAME, PASSWORD, CLIENT_ID, CLIENT_SECRET)
.build()
) {
// Obtain information from decoded token
DecodedToken decodedToken = handler.decodeToken();
try (EventWebSocket eventWebSocket = EventWebSocket.newBuilder(
handler,
new EventWebSocketListener() {
@Override
public void onCreate(EventsMessage eventsMessage) {
// React when a vertex has been created
}
@Override
public void onUpdate(EventsMessage eventsMessage) {
// React when a vertex has been updated
}
@Override
public void onDelete(EventsMessage eventsMessage) {
// React when a vertex has been deleted
}
})
// Set defaultScope by hand
.addScope(decodedToken.data.defaultScope)
.addEventsFilter(
"default",
"(element.ogit/_type = ogit/Automation/AutomationIssue)"
)
.build()
) {
eventWebSocket.start();
// Listen for 1 second for incoming events.
Thread.sleep(1000);
}
}
}
}
In contrast to the other API Client objects the websockets need to be closed explicitly after usage.
The action websocket receives messages from HIRO KIs. The project then has to handle those messages and send a response back.
This library handles the protocol overhead like ACK and NACK messages and persistent storage of messages for consistent communication in the background.
Receiving and sending of action messages and results should be done asynchronously - which is NOT subject of this library.
Example skeleton code:
import co.arago.hiro.client.Config;
import co.arago.hiro.client.connection.token.PasswordAuthTokenAPIHandler;
import co.arago.hiro.client.exceptions.HiroException;
import co.arago.hiro.client.model.websocket.action.impl.ActionHandlerSubmit;
import co.arago.hiro.client.websocket.ActionWebSocket;
import co.arago.hiro.client.websocket.listener.ActionWebSocketListener;
import java.io.IOException;
class Example {
public static void main(String[] args) throws HiroException, IOException, InterruptedException {
// Build an API handler which takes care of API paths via /api/versions and security tokens.
try (PasswordAuthTokenAPIHandler handler = PasswordAuthTokenAPIHandler.newBuilder()
.setRootApiUri(API_URL)
.setCredentials(USERNAME, PASSWORD, CLIENT_ID, CLIENT_SECRET)
.build()
) {
try (ActionWebSocket actionWebSocket = ActionWebSocket.newBuilder(
handler,
new ActionWebSocketListener() {
@Override
public void onActionSubmit(
ActionWebSocket actionWebSocket,
ActionHandlerSubmit submit
) throws Exception {
// Handle message
actionWebSocket.sendActionResult(
submit.getId(),
ActionWebSocket.newResultParams()
.setMessage("All is fine")
);
}
@Override
public void onConfigChanged(
ActionWebSocket actionWebSocket
) {
// Handle reload of configuration if necessary.
}
})
.build()
) {
actionWebSocket.start();
// Listen for 1 second for incoming actions.
Thread.sleep(1000);
}
}
}
}