Reactors
Reactors are stateless collections of methods that can perform actions inside Hantera based on input. Reactors can be used to create custom APIs and automate background work.
A Reactors starts with a Component that is then configured as a Reactor instance. This makes Reactor components re-usable for many different use-cases.
Reactors can be scheduled in Jobs or invoked directly through it’s public API, making it possible to essentially create custom APIs in your Hantera installation.
Reactors methods can take arguments, and return values. Given the nature of Reactors, besides being a bridge between Hantera’s Actors and the outside world, they can often replace serverless functions provided by my of today’s cloud providers.
A Simple Example
To get a better idea of what a Reactor looks like, take a look at the below Reactor Component:
param uri: text
from { post = args => web.http( uri, { method = 'POST' body = args.body } )}
This very simple Reactor allows Rules to make external HTTP requests towards a predetermined URL. Let’s break it down.
param uri: text
This line defines a Reactor Parameter called uri
or type text
. This is used later to determine the endpoint of the requests. Reactor paramters are set when the Reactor is created. The same component can be re-used in multiple reactors if we need to make requests to many different endpoints.
from
The from
keyword makes up the main output of the Reactor. Reactors must only have a single return value, and it must be a record of method handlers.
post = args => web.http( uri, { method = 'POST' body = args.body })
This part defines a record field called post
. This will be the name of the method when we call it later. The part that comes next is a function definition which is the handler that runs when the reactor is invoked. In this case, we’re performing an HTTP POST request.
The web.http
function takes two arguments. The first is the URL, and the second is a record of options. In this case, we’re setting the method to POST
and adding a body
which will be included in the request. The args.body
means that we’re expecting to get the body as a method argument. This means that any Rule that calls this method, can provide the body to be used for the request.
For more information on the http function, refer to the http article
Testing the Reactor
In order to test this reactor, we must first install the component in Hantera. For the sake of this guide, we will use a manifest file and push the component to Hantera using hantera-cli. We will then create the Reactor instance itself using another manifest file referencing the newly created component.
-
Start by creating the manifest file for Reactor component:
component.myReactors.httpPoster.h_manifest.yaml uri: /resources/components/myReactors/httpPosterspec:displayName: 'Http Poster'description: 'Makes a POST request to a configured endpoint, allowing the caller to set request body'code: |-param uri: textfrom {post = args => web.http(uri,{method = 'POST'body = args.body})} -
Next, let’s define the manifest that will instantiate the Reactor resource:
reactor.my-api.h_manifest.yaml uri: /resources/reactors/my-apispec:componentId: myReactors/httpPosterparameters:uri: https://my-api.example.com/v1/endpoint- Now let’s apply these manifests to our Hantera instance:
Terminal window h_ manage apply component.myReactor.httpPoster.h_manifest.yamlh_ manage apply reactor.my-api.h_manifest.yaml -
You should now have a new Reactor resource live in your Hantera. You can verify it by querying the Reactors API. This should return metadata about the Reactor and it’s available methods:
GET https://<hantera-hostname>/resources/reactors/my-apiAuthorization: Bearer <YOUR TOKEN>Terminal window curl -i -X GET \https://<hantera-hostname>/resources/reactors/my-api \-H 'Authorization: Bearer <YOUR TOKEN>'Terminal window Invoke-WebRequest `-Uri "https://<hantera-hostname>/resources/reactors/my-api" `-Method GET ` `-Headers @{Authorization="Bearer <YOUR TOKEN>"} -
Once you have verified that the Reactor is there, you can now call the method using the API:
POST https://<hantera-hostname>/resources/reactors/my-api/postAuthorization: Bearer <YOUR TOKEN>Content-Type: application/json{"body": "This will be the body of the request"}Terminal window curl -i -X POST \https://<hantera-hostname>/resources/reactors/my-api/post \-H 'Authorization: Bearer <YOUR TOKEN>' \-H 'Content-Type: application/json' \-d '{"body": "This will be the body of the request"}'Terminal window Invoke-WebRequest `-Uri "https://<hantera-hostname>/resources/reactors/my-api/post" `-Method POST ` `-Headers @{Authorization="Bearer <YOUR TOKEN>"; 'Content-Type'="application/json"}-Body '{"body": "This will be the body of the request"}'
That’s it. Notice how the method name defined in the Reactor component code is manifested in it’s public API, while we can name the Reactor anything we want in the actual Reactor resource.
Access Control
You may wonder, how is it safe to make arbitrary external requests available through the public API. This sounds like a recipe for all kinds of denial-of-service problems.
While it’s true that Reactors are powerful and used wrong it can be misused. Reactors are tightly integrated with Hantera’s access control layer, which means that calling Reactor Methods require explicit permission for the session. Additinally, as opposed to many other similar platforms, Hantera makes all effects of Reactors very easy to monitor through the Reactor Logs.
The ACE pattern for reactors looks like this:
reactor[/<id>][/<method>]:read|write|invoke