Actuators and HTTP client
Actuator endpoints
The following admin endpoints are available.
GET /info
GET /health
GET /livenessprobe
Endpoint | Purpose |
---|---|
/info | Describe the application |
/health | Application health check endpoint |
/livenessprobe | Check if application is running normally |
Custom health services
You can extend the "/health" endpoint by implementing a composable functions to be added to the "health check" dependencies.
health.dependencies=database.health, cache.health
Your custom health service must respond to the following requests:
- Info request (type=info) - it should return a map that includes service name and href (protocol, hostname and port)
- Health check (type=health) - it should return a text string of the health check. e.g. read/write test result. It can throw AppException with status code and error message if health check fails.
A sample health service is available in the health-check.ts
class of the hello world project as follows:
import { preload, Composable, EventEnvelope, AppException } from 'mercury';
const TYPE = 'type';
const INFO = 'info';
const HEALTH = 'health';
export class DemoHealthCheck implements Composable {
@preload('demo.health')
initialize(): DemoHealthCheck {
return this;
}
// Your service should be declared as an async function with input as EventEnvelope
async handleEvent(evt: EventEnvelope) {
const command = evt.getHeader(TYPE);
if (command == INFO) {
return {'service': 'demo.service', 'href': 'http://127.0.0.1'};
}
if (command == HEALTH) {
// this is a dummy health check
return {'status': 'demo.service is running fine'};
}
throw new AppException(400, 'Request type must be info or health');
}
}
AsyncHttpClient API
The "async.http.request" function can be used as a non-blocking HTTP client.
To make an HTTP request to an external REST endpoint, you can create an HTTP request object using the
AsyncHttpRequest
class and make an async RPC call to the "async.http.request" function like this:
const po = new PostOffice(evt.getHeaders());
const req = new AsyncHttpRequest();
req.setMethod("GET");
req.setHeader("accept", "application/json");
req.setUrl("/api/hello/world?hello world=abc");
req.setQueryParameter("x1", "y");
const list = new Array<string>();
list.push("a");
list.push("b");
req.setQueryParameter("x2", list);
req.setTargetHost("http://127.0.0.1:8083");
const event = new EventEnvelope().setTo("async.http.request").setBody(req);
const result = po.request(event, 5000);
// the result is an EventEnvelope
Send HTTP request body for HTTP PUT, POST and PATCH methods
For most cases, you can just set a JSON object into the request body and specify content-type as JSON.
Example code may look like this:
const req = new AsyncHttpRequest();
req.setMethod("POST");
req.setHeader("accept", "application/json");
req.setHeader("content-type", "application/json");
req.setUrl("/api/book");
req.setTargetHost("https://service_provider_host");
req.setBody(jsonKeyValues);
Send HTTP request body as a stream
For larger payload, you may use the streaming method. See sample code below:
const stream = new ObjectStreamIO(timeoutInSeconds);
const out = stream.getOutputStream();
out.write(blockOne);
out.write(blockTwo);
// closing the output stream would send a EOF signal to the stream
out.close();
// tell the HTTP client to read the input stream
req.setStreamRoute(stream.getInputStreamId());
The AsyncHttpClient service (route name async.http.request
) uses native Node.js streams to integrate
with the underlying Axios HTTP client. It uses the temporary local file system (folder /tmp/node/streams
)
to reduce memory footprint. This makes the producer and consumer of a stream asynchronous. i.e. The producer
can write data blocks into a stream before a consumer is available.
Read HTTP response body stream
If content length is not given, the response body will be received as a stream.
Your application should check if the HTTP response header "stream" exists. Its value is the input "stream ID".
Sample code to read a stream may look like this:
static async downloadFile(streamId: string, filename: string) {
let n = 0;
let len = 0;
const stream = new ObjectStreamReader(streamId, 5000);
while (true) {
try {
const block = await stream.read();
if (block) {
n++;
if (block instanceof Buffer) {
len += block.length;
log.info(`Received ${filename}, block-${n} - ${block.length} bytes`)
}
} else {
log.info("EOF reached");
break;
}
} catch (e) {
const status = e instanceof AppException? e.getStatus() : 500;
log.error(`Exception - rc=${status}, message=${e.message}`);
break;
}
}
return len;
}
Content length for HTTP request
IMPORTANT: Do not set the "content-length" HTTP header because the system will automatically compute the correct content-length for small payload. For large payload, it will use the chunking method.
Appendix-I | Home |
---|---|
Application config | Table of Contents |