In this guide we are focusing on building applications with a microservices architecture. In this article, we are considering a look at how the services within a system communicate with one another.
This guide walks you through the process of describing implementing inter-process communication using Ballerina programming language.
The following are the sections available in this guide.
The old fashion monolithic application process, components interactions are designed in a way that invoke one another via language‑level method or function calls. On the other hand, a microservices‑based application fully focused on distributed system running on multiple singular containers. Each service instance is typically a process. Consequently, as the following diagram shows, services must interact using an inter‑process communication mechanism.
When selecting an Inter process communication mechanism for a service, it is always useful to think first about how services interact. There are a variety of client⇔service interaction styles. They can be categorized along two dimensions. The first dimension is whether the interaction is one‑to‑one or one‑to‑many:
-
One‑to‑one – Each client request is processed by exactly one service instance.
-
One‑to‑many – Each request is processed by multiple service instances.
-
The second dimension is whether the interaction is synchronous or asynchronous:
-
Synchronous – The client expects a timely response from the service and might even block while it waits.
-
Asynchronous – The client doesn’t block while waiting for a response, and the response, if any, isn’t necessarily sent immediately.
Each service typically uses a combination of these interaction styles. For some services, a single IPC mechanism is sufficient. Other services might need to use a combination of IPC mechanisms. The following diagram shows how services in a taxi-hailing application might interact when the user requests a trip.
In Microservices architecture, The services use a combination of notifications, request/response, and publish/subscribe. For example, the passenger’s smartphone sends a notification to the Trip Management service to request a pickup. The Trip Management service verifies that the passenger’s account is active by using request/response to invoke the Passenger Service. The Trip Management service then creates the trip and uses publish/subscribe to notify other services including the Dispatcher, which locates an available driver.
So, lets try to demonstrate the scenario,
In this example Apache ActiveMQ
has been used as the JMS broker for inter process communication through notification channel. Ballerina JMS Connector is used to connect Ballerina
and JMS Message Broker. With this JMS Connector, Ballerina can act as both JMS Message Consumer and JMS Message
Producer.
- Ballerina Distribution
- A Text Editor or an IDE
Tip: For a better development experience, install one of the following Ballerina IDE plugins: VSCode, IntelliJ IDEA
- A JMS Broker (Example: Apache ActiveMQ)
- After installing the JMS broker, copy its .jar files into the
<BALLERINA_HOME>/bre/lib
folder- For ActiveMQ 5.12.0: Copy
activemq-client-5.12.0.jar
andgeronimo-j2ee-management_1.1_spec-1.0.1.jar
- For ActiveMQ 5.12.0: Copy
- After installing the JMS broker, copy its .jar files into the
If you want to skip the basics, you can download the git repo and directly move to the "Testing" section by skipping "Implementation" section.
Ballerina is a complete programming language that supports custom project structures. Use the following package structure for this guide.
├── dispatcher
│ └── dispatcher.bal
├── driver-management
│ └── driver-management.bal
├── notification
│ └── notification.bal
├── passenger-management
│ └── passenger-management.bal
└── trip-management
└── trip-management.bal
-
Create the above directories in your local machine and also create empty
.bal
files. -
Then open the terminal and navigate to
inter-microservice-communicaiton/guide
and run Ballerina project initializing toolkit.
$ ballerina init
Let's get started with the implementation of the trip-management.bal
, which acts as the interface to coordinate interactions between passenger management passenger-management.bal
and dispatcher dispatcher.bal
. The dispatcher be act as an middleman which will coordinate Driver and Passenger. (The overall coordination seems to be very simple). However lets walk through each implementation
Refer to the code attached below. Inline comments added for better understanding.
import ballerina/log;
import ballerina/http;
import ballerina/jms;
import ballerinax/docker;
// Type definition for a Pickup order
type Pickup record {
string customerName;
string address;
string phonenumber;
};
// Initialize a JMS connection with the provider
// 'providerUrl' and 'initialContextFactory' vary based on the JMS provider you use
// 'Apache ActiveMQ' has been used as the message broker in this example
jms:Connection jmsConnection = new({
initialContextFactory: "org.apache.activemq.jndi.ActiveMQInitialContextFactory",
providerUrl: "tcp://localhost:61616"
});
// Initialize a JMS session on top of the created connection
jms:Session jmsSession = new(jmsConnection, {
acknowledgementMode: "AUTO_ACKNOWLEDGE"
});
// Initialize a queue sender using the created session
jms:QueueSender jmsTripDispatchOrder = new(jmsSession, queueName = "trip-dispatcher");
// Client endpoint to communicate with Airline reservation service
http:Client passengerMgtEP = new("http://localhost:9091/passenger-management");
//@doker:Config {
// registry:"ballerina.guides.io",
// name:"trip_management_service",
// tag:"v1.0"
//}
//@docker:CopyFiles {
// files:[{source:"/Users/dushan/workspace/wso2/ballerina/bbg/apache-activemq-5.13.0/lib/geronimo-j2ee-management_1.1_spec-1.0.1.jar",
// target:"/ballerina/runtime/bre/lib"},{source:"/Users/dushan/workspace/wso2/ballerina/bbg/apache-activemq-5.13.0/lib/activemq-client-5.13.0.jar",
// target:"/ballerina/runtime/bre/lib"}]
//}
//@docker:Expose{}
//listener http:Listener httpListener = new(9090);
// Service endpoint
listener http:Listener httpListener = new(9090);
// Trip manager service, which is managing trip requests received from the client
@http:ServiceConfig {
basePath:"/trip-manager"
}
service TripManagement on httpListener {
// Resource that allows users to place an order for a pickup
@http:ResourceConfig {
path : "/pickup",
methods: ["POST"],
consumes: ["application/json"],
produces: ["application/json"]
}
resource function pickup(http:Caller caller, http:Request request) returns error? {
http:Response response = new;
json reqPayload;
// Try parsing the JSON payload from the request
var payload = request.getJsonPayload();
if (payload is json) {
reqPayload = payload;
} else {
response.statusCode = 400;
response.setJsonPayload({"Message":"Invalid payload - Not a valid JSON payload"});
checkpanic caller->respond(response);
return;
}
json name = reqPayload.Name;
json address = reqPayload.pickupaddr;
json contact = reqPayload.ContactNumber;
// If payload parsing fails, send a "Bad Request" message as the response
if (name == null || address == null || contact == null) {
response.statusCode = 400;
response.setJsonPayload({"Message":"Bad Request - Invalid Trip Request payload"});
checkpanic caller->respond(response);
return;
}
// Order details
Pickup pickup = {
customerName: name.toString(),
address: address.toString(),
phonenumber: contact.toString()
};
log:printInfo("Calling passenger management service:");
// call passanger-management and get passegner orginization claims
json responseMessage;
http:Request passengerManagerReq = new;
json pickupjson = check json.convert(pickup);
passengerManagerReq.setJsonPayload(untaint pickupjson);
http:Response passengerResponse= check passengerMgtEP->post("/claims", passengerManagerReq);
json passengerResponseJSON = check passengerResponse.getJsonPayload();
// Dispatch to the dispatcher service
// Create a JMS message
jms:Message queueMessage = check jmsSession.createTextMessage(passengerResponseJSON.toString());
// Send the message to the JMS queue
log:printInfo("Hand over to the trip dispatcher to coordinate driver and passenger:");
checkpanic jmsTripDispatchOrder->send(queueMessage);
// Creating Trip
// call Dispatcher and contacting Driver and Passenger
log:printInfo("passanger-magement response:"+passengerResponseJSON.toString());
// Send response to the user
responseMessage = {"Message":"Trip information received"};
response.setJsonPayload(responseMessage);
checkpanic caller->respond(response);
return;
}
}
To see the complete implementation of the above, refer to the trip-management.bal.
import ballerina/log;
import ballerina/io;
import ballerina/jms;
import ballerina/http;
type Trip record {
string tripID;
Driver driver;
Person person;
string time;
};
type Driver record {
string driverID;
string drivername;
};
type Person record {
string name;
string address;
string phonenumber;
string registerID;
string email;
};
// Initialize a JMS connection with the provider
// 'Apache ActiveMQ' has been used as the message broker
jms:Connection conn = new({
initialContextFactory: "org.apache.activemq.jndi.ActiveMQInitialContextFactory",
providerUrl: "tcp://localhost:61616"
});
// Client endpoint to communicate with Airline reservation service
http:Client courierEP = new("http://localhost:9095/courier");
// Initialize a JMS session on top of the created connection
jms:Session jmsSession = new(conn, {
// Optional property. Defaults to AUTO_ACKNOWLEDGE
acknowledgementMode: "AUTO_ACKNOWLEDGE"
});
// Initialize a queue receiver using the created session
listener jms:QueueReceiver jmsConsumer = new (jmsSession, queueName = "trip-dispatcher");
// Initialize a queue sender using the created session
jms:QueueSender jmsPassengerMgtNotifer = new(jmsSession, queueName = "trip-passenger-notify");
// Initialize a queue sender using the created session
jms:QueueSender jmsDriverMgtNotifer = new(jmsSession, queueName = "trip-driver-notify");
// JMS service that consumes messages from the JMS queue
// Bind the created consumer to the listener service
service TripDispatcher on jmsConsumer {
// Triggered whenever an order is added to the 'OrderQueue'
resource function onMessage(jms:QueueReceiverCaller consumer, jms:Message message) returns error?{
log:printInfo("New Trip request ready to process from JMS Queue");
http:Request orderToDeliver = new;
// Retrieve the string payload using native function
string personDetail = check message.getTextMessageContent();
log:printInfo("person Details: " + personDetail);
json person = <json>personDetail;
orderToDeliver.setJsonPayload(untaint person);
string name = person.name.toString();
Trip trip = {
person: {
name: "dushan",
address: "1817",
phonenumber: "0014089881345",
email: "[email protected]",
registerID: "AB0001222"
},
driver: {
driverID: "driver001",
drivername: "Adeel Sign"
},
tripID: "0001",
time: "2018 Jan 6 10:10:20"
};
json tripjson = check json.convert(trip);
log:printInfo("Driver Contacted Trip notification dispatching " + tripjson.toString());
jms:Message queueMessage = check jmsSession.createTextMessage(tripjson.toString());
fork {
worker passengerNotification {
checkpanic jmsPassengerMgtNotifer->send(queueMessage);
}
worker driverNotification {
checkpanic jmsDriverMgtNotifer->send(queueMessage);
}
}
_ = wait {passengerNotification, driverNotification};
return;
}
}
To see the complete implementation of the above, refer to the dispatcher.bal.
import ballerina/http;
import ballerina/log;
import ballerina/jms;
type Person record {
string name;
string address;
string phonenumber;
string registerID;
string email;
};
listener http:Listener httpListener = new(9091);
// Initialize a JMS connection with the provider
// 'Apache ActiveMQ' has been used as the message broker
jms:Connection conn = new({
initialContextFactory: "org.apache.activemq.jndi.ActiveMQInitialContextFactory",
providerUrl: "tcp://localhost:61616"
});
// Initialize a JMS session on top of the created connection
jms:Session jmsSession = new(conn, {
// Optional property. Defaults to AUTO_ACKNOWLEDGE
acknowledgementMode: "AUTO_ACKNOWLEDGE"
});
// Initialize a queue receiver using the created session
listener jms:QueueReceiver jmsConsumer = new(jmsSession, queueName = "trip-passenger-notify");
@http:ServiceConfig {
basePath: "/passenger-management"
}
service PassengerManagement on httpListener {
@http:ResourceConfig {
path : "/claims",
methods : ["POST"]
}
resource function claims(http:Caller caller, http:Request request) returns error? {
// create an empty response object
http:Response res = new;
// check will cause the service to send back an error
// if the payload is not JSON
json responseMessage;
json passengerInfoJSON = check request.getJsonPayload();
log:printInfo("JSON :::" + passengerInfoJSON.toString());
string customerName = passengerInfoJSON.customerName.toString();
string address = passengerInfoJSON.address.toString();
string contact = passengerInfoJSON.phonenumber.toString();
Person person = {
name: customerName,
address: address,
phonenumber: contact,
email: "[email protected]",
registerID: "AB0001222"
};
log:printInfo("customerName:" + customerName);
log:printInfo("address:" + address);
log:printInfo("contact:" + contact);
json personjson = check json.convert(person);
responseMessage = personjson;
log:printInfo("Passanger claims included in the response:" + personjson.toString());
res.setJsonPayload(untaint personjson);
checkpanic caller->respond(res);
return;
}
}
// JMS service that consumes messages from the JMS queue
// Bind the created consumer to the listener service
service PassengerNotificationService on jmsConsumer {
// Triggered whenever an order is added to the 'OrderQueue'
resource function onMessage(jms:QueueReceiverCaller consumer, jms:Message message) returns error? {
log:printInfo("Trip information received passenger notification service notifying to the client");
http:Request orderToDeliver = new;
// Retrieve the string payload using native function
string personDetail = check message.getTextMessageContent();
log:printInfo("Trip Details:" + personDetail);
return;
}
}
To see the complete implementation of the above, refer to the passenger-management.bal.
import ballerina/http;
import ballerina/log;
import ballerina/jms;
type Person record {
string name;
string address;
string phonenumber;
string registerID;
string email;
};
listener http:Listener httpListener = new(9091);
// Initialize a JMS connection with the provider
// 'Apache ActiveMQ' has been used as the message broker
jms:Connection conn = new({
initialContextFactory: "org.apache.activemq.jndi.ActiveMQInitialContextFactory",
providerUrl: "tcp://localhost:61616"
});
// Initialize a JMS session on top of the created connection
jms:Session jmsSession = new(conn, {
// Optional property. Defaults to AUTO_ACKNOWLEDGE
acknowledgementMode: "AUTO_ACKNOWLEDGE"
});
// Initialize a queue receiver using the created session
listener jms:QueueReceiver jmsConsumer = new(jmsSession, queueName = "trip-driver-notify");
// JMS service that consumes messages from the JMS queue
// Bind the created consumer to the listener service
service DriverNotificationService on jmsConsumer {
// Triggered whenever an order is added to the 'OrderQueue'
resource function onMessage(jms:QueueReceiverCaller consumer, jms:Message message) returns error? {
log:printInfo("Trip information received for Driver notification service notifying coordinating with Driver the trip info");
http:Request orderToDeliver = new;
// Retrieve the string payload using native function
string personDetail = check message.getTextMessageContent();
log:printInfo("Trip Details: " + personDetail);
return;
}
}
To see the complete implementation of the above, refer to the driver-management.bal.
Similar to the JMS consumer, here also we require to provide JMS configuration details when defining the jms:QueueSender
endpoint. We need to provide the JMS session and the queue to which the producer pushes the messages.
Distributed system there is the ever-present risk of partial failure or complete system unavailability. Since clients and services are separate processes, a service might not be able to respond in a timely way to a client’s request. A service might be down because of a failure or for maintenance. Or the service might be overloaded and responding extremely slowly to requests. Ballerina language offered the components to implement 'The circuit breaker pattern' which is defined as the standard way to automatically degrade functionality when remote services fail. When you use the circuit breaker pattern, you can allow a web service to continue operating without waiting for unresponsive remote services.
Lets focus the trip-manager.bal
, the the integration required certain amount of resilience since the component suppose to communicate between multiple services. So you may have chance of introducing circuit breaker for passengerMgtEP
...
// Client endpoint to communicate with passager management service
endpoint http:Client passengerMgtEP {
// The 'circuitBreaker' term incorporate circuit breaker pattern to the client endpoint
// Circuit breaker will immediately drop remote calls if the endpoint exceeded the failure threshold
circuitBreaker: {
rollingWindow: {
timeWindowMillis: 10000,
bucketSizeMillis: 2000
},
// Failure threshold should be in between 0 and 1
failureThreshold: 0.2,
// Reset timeout for circuit breaker should be in milliseconds
resetTimeMillis: 10000,
// httpStatusCodes will have array of http error codes tracked by the circuit breaker
statusCodes: [400, 404, 500]
},
// HTTP client could be any HTTP endpoint that have risk of failure
url:"http://localhost:9091/passenger-management"
,
timeoutMillis: 2000
};
or introducing failover to connect multiple passenger-management endpoints to cope with unexpected failures
http:LoadBalanceClient passengerMgtEP = new({
targets: [
// Create an array of HTTP Clients that needs to be Load balanced across
{ url: "http://localhost:9011/passenger-management" },
{ url: "http://localhost:9012/passenger-management" },
{ url: "http://localhost:9013/passenger-management" }
]
});
- First, start the
Apache ActiveMQ
server by entering the following command in a terminal from<ActiveMQ_BIN_DIRECTORY>
.
$ ./activemq start
- Navigate to
inter-microservice-communicaiton/guide
and run the following commands in separate terminals to starttrip-management
,passenger-management
,dispatcher
,driver-management
microservices.
$ ballerina run trip-management
$ ballerina run dispatcher
$ ballerina run passenger-management
$ ballerina run driver-management
- Then you may simulate requesting pickup call to the
trip-management
.
curl -v -X POST -d \
'{"Name":"Dushan", "pickupaddr":"1817, Anchor Way, San Jose, US",
"ContactNumber":"0014089881345"}' \
"http://localhost:9090/trip-manager/pickup" -H "Content-Type:application/json"
The Trip management sends a response similar to the following which is similar to the acknowledged to the pickup request followed by he will receiving trip notification information through different channel
< HTTP/1.1 200 OK
< content-type: application/json
< content-length: 39
< server: ballerina/0.970.1
< date: Fri, 8 Jun 2018 21:25:29 -0700
<
* Connection #0 to host localhost left intact
{"Message":"Trip information received"}
Driver management service would receive trip notification, the notification service can will render relevant information in the customer handheld device through notification
2018-06-08 22:08:38,361 INFO [driver-management] - Trip Details: {"tripID":"0001","driver":{"driverID":"driver001","drivername":"Adeel Sign"},"person":{"name":"dushan","address":"1817","phonenumber":"0014089881345","registerID":"AB0001222","email":"[email protected]"},"time":"2018 Jan 6 10:10:20"}
Similarly passenger would get his trip notification
2018-06-08 22:08:38,346 INFO [passenger-management] - Trip Details:{"tripID":"0001","driver":{"driverID":"driver001","drivername":"Adeel Sign"},"person":{"name":"dushan","address":"1817","phonenumber":"0014089881345","registerID":"AB0001222","email":"[email protected]"},"time":"2018 Jan 6 10:10:20"}
In Ballerina, the unit test cases should be in the same package inside a folder named as 'tests'. When writing the test functions the below convention should be followed.
- Test functions should be annotated with
@test:Config
. See the below example.
@test:Config
function testResourcePickup() {
This guide contains unit test cases for each resource available in the 'trip-management' implemented above.
To run the unit tests, navigate to inter-microservice-communicaiton/guide/
and run the following command.
$ ballerina test
When running these unit tests, make sure that the ActiveMQ
is up and running.
To check the implementation of the test file, refer to the [trip-management.bal].
Once you are done with the development, you can deploy the services using any of the methods that we listed below.
As the first step, you can build Ballerina executable archives (.balx) of the services that we developed above. Navigate to inter-microservice-communicaiton/guide
and run the following command.
$ ballerina build
- Once the .balx files are created inside the target folder, you can run them using the following command.
$ ballerina run target/<Exec_Archive_File_Name>
- The successful execution of a service will show us something similar to the following output.
ballerina: initiating service(s) in 'target/trip-management.balx'
ballerina: initiating service(s) in 'target/dispatcher.balx'
ballerina: initiating service(s) in 'target/passenger-management.balx'
ballerina: initiating service(s) in 'target/driver-management.balx'
You can run the service that we developed above as a docker container.
As Ballerina platform includes Ballerina_Docker_Extension, which offers native support for running ballerina programs on containers,
you just need to put the corresponding docker annotations on your service code.
Since this guide requires ActiveMQ
as a prerequisite, you need a couple of more steps to configure it in docker container.
First let's see how to configure ActiveMQ
in docker container.
- Initially, you need to pull the
ActiveMQ
docker image using the below command.
$ docker pull webcenter/activemq
- Then launch the pulled image using the below command. This will start the
ActiveMQ
server in docker with default configurations.
$ docker run -d --name='activemq' -it --rm -P webcenter/activemq:latest
- Check whether the
ActiveMQ
container is up and running using the following command.
$ docker ps
Now let's see how we can deploy the trip-management
we developed above on docker. We need to import ballerinax/docker
and use the annotation @docker:Config
as shown below to enable docker image generation during the build time.
import ballerina/log;
import ballerina/http;
import ballerina/jms;
import ballerinax/docker;
@docker:Config {
registry:"ballerina.guides.io",
name:"trip-management.bal",
tag:"v1.0"
}
@docker:CopyFiles {
files:[{source:<path_to_JMS_broker_jars>,
target:"/ballerina/runtime/bre/lib"}]
}
@docker:Expose{}
listener http:Listener httpListener = new(9090);
// Type definition for a pickup order
type pickup {
string customerName;
string address;
string phonenumber;
};
// Initialize a JMS connection with the provider
// 'providerUrl' and 'initialContextFactory' vary based on the JMS provider you use
// 'Apache ActiveMQ' has been used as the message broker in this example
jms:Connection jmsConnection = new({
initialContextFactory: "org.apache.activemq.jndi.ActiveMQInitialContextFactory",
providerUrl: "tcp://localhost:61616"
});
// Initialize a JMS session on top of the created connection
jms:Session jmsSession = new(jmsConnection, {
acknowledgementMode: "AUTO_ACKNOWLEDGE"
});
...
-
You may configure other services the same way as above i.e
dispatcher.bal
,passenger-management.bal
,driver-management.bal
what you may need to change@docker:Config
names to the respective services -
@docker:Config
annotation is used to provide the basic docker image configurations for the sample.@docker:CopyFiles
is used to copy the JMS broker jar files into the ballerina bre/lib folder. You can provide multiple files as an array to fieldfiles
of CopyFiles docker annotation.@docker:Expose {}
is used to expose the port. -
Now you can build a Ballerina executable archive (.balx) of the service that we developed above, using the following command. This will also create the corresponding docker image using the docker annotations that you have configured above. Navigate to
inter-microservice-communicaiton/guide
and run the following command. -
Then start each services through docker container seperately
$ballerina build trip-management
Run following command to start docker container:
docker run -d -p 9090:9090 ballerina.guides.io/trip-management:v1.0
$ballerina build dispatcher
Run following command to start docker container:
docker run -d -p 9091:9090 ballerina.guides.io/dispatcher:v1.0
$ballerina build passenger-management
Run following command to start docker container:
docker run -d -p 9092:9090 ballerina.guides.io/passenger-management:v1.0
$ballerina build driver-management
Run following command to start docker container:
docker run -d -p 9093:9090 ballerina.guides.io/driver-management:v1.0
Here we run the docker image with flag -p <host_port>:<container_port>
so that we use the host port 9090 and the container port 9090. Therefore you can access the service through the host port. Please note when exposing the services we are using port offset to avoid port conflicts thus we use 9090, 9091,9092,9093 ports respectively.
-
Since services are inter communicating each other, e.g trip-manager <=> passenger-management which is happening through HTTP communication, thus, please you appropriate ports and configure the service layer
-
Verify docker container is running with the use of
$ docker ps
. The status of the docker container should be shown as 'Up'. -
You can access the service using the same curl commands that we've used above.
curl -v -X POST -d \
'{"Name":"Dushan", "pickupaddr":"1817, Anchor Way, San Jose, US",
"ContactNumber":"0014089881345"}' \
"http://localhost:9090/trip-manager/pickup" -H "Content-Type:application/json"
Ballerina is by default observable. Meaning you can easily observe your services, resources, etc.
However, observability is disabled by default via configuration. Observability can be enabled by adding following configurations to ballerina.conf
file in inter-microservice-communicaiton/guide/
.
[b7a.observability]
[b7a.observability.metrics]
# Flag to enable Metrics
enabled=true
[b7a.observability.tracing]
# Flag to enable Tracing
enabled=true
NOTE: The above configuration is the minimum configuration needed to enable tracing and metrics. With these configurations default values are load as the other configuration parameters of metrics and tracing.
You can monitor ballerina services using in built tracing capabilities of Ballerina. We'll use Jaeger as the distributed tracing system. Follow the following steps to use tracing with Ballerina.
- You can add the following configurations for tracing. Note that these configurations are optional if you already have the basic configuration in
ballerina.conf
as described above.
[b7a.observability]
[b7a.observability.tracing]
enabled=true
name="jaeger"
[b7a.observability.tracing.jaeger]
reporter.hostname="localhost"
reporter.port=5775
sampler.param=1.0
sampler.type="const"
reporter.flush.interval.ms=2000
reporter.log.spans=true
reporter.max.buffer.spans=1000
- Run Jaeger docker image using the following command
$ docker run -d -p5775:5775/udp -p6831:6831/udp -p6832:6832/udp -p5778:5778 \
-p16686:16686 p14268:14268 jaegertracing/all-in-one:latest
- Navigate to
inter-microservice-communicaiton/guide
and run thetrip-management
using following command
$ ballerina run trip-management/
- Observe the tracing using Jaeger UI using following URL
http://localhost:16686
Metrics and alerts are built-in with ballerina. We will use Prometheus as the monitoring tool. Follow the below steps to set up Prometheus and view metrics for trip-management service.
- You can add the following configurations for metrics. Note that these configurations are optional if you already have the basic configuration in
ballerina.conf
as described underObservability
section.
[b7a.observability.metrics]
enabled=true
provider="micrometer"
[b7a.observability.metrics.micrometer]
registry.name="prometheus"
[b7a.observability.metrics.prometheus]
port=9700
hostname="0.0.0.0"
descriptions=false
step="PT1M"
- Create a file
prometheus.yml
inside/tmp/
location. Add the below configurations to theprometheus.yml
file.
global:
scrape_interval: 15s
evaluation_interval: 15s
scrape_configs:
- job_name: prometheus
static_configs:
- targets: ['172.17.0.1:9797']
NOTE : Replace 172.17.0.1
if your local docker IP differs from 172.17.0.1
- Run the Prometheus docker image using the following command
$ docker run -p 19090:9090 -v /tmp/prometheus.yml:/etc/prometheus/prometheus.yml \
prom/prometheus
- You can access Prometheus at the following URL
http://localhost:19090/
NOTE: Ballerina will by default have following metrics for HTTP server connector. You can enter following expression in Prometheus UI
- http_requests_total
- http_response_time
Ballerina has a log package for logging to the console. You can import ballerina/log package and start logging. The following section will describe how to search, analyze, and visualize logs in real time using Elastic Stack.
- Start the Ballerina Service with the following command from
inter-microservice-communicaiton/guide
$ nohup ballerina run trip-management/ &>> ballerina.log&
NOTE: This will write the console log to the ballerina.log
file in the inter-microservice-communicaiton/guide
directory
-
Start Elasticsearch using the following command
-
Start Elasticsearch using the following command
$ docker run -p 9200:9200 -p 9300:9300 -it -h elasticsearch --name \
elasticsearch docker.elastic.co/elasticsearch/elasticsearch:6.2.2
NOTE: Linux users might need to run sudo sysctl -w vm.max_map_count=262144
to increase vm.max_map_count
- Start Kibana plugin for data visualization with Elasticsearch
$ docker run -p 5601:5601 -h kibana --name kibana --link \
elasticsearch:elasticsearch docker.elastic.co/kibana/kibana:6.2.2
- Configure logstash to format the ballerina logs
i) Create a file named logstash.conf
with the following content
input {
beats{
port => 5044
}
}
filter {
grok{
match => {
"message" => "%{TIMESTAMP_ISO8601:date}%{SPACE}%{WORD:logLevel}%{SPACE}
\[%{GREEDYDATA:package}\]%{SPACE}\-%{SPACE}%{GREEDYDATA:logMessage}"
}
}
}
output {
elasticsearch{
hosts => "elasticsearch:9200"
index => "store"
document_type => "store_logs"
}
}
ii) Save the above logstash.conf
inside a directory named as {SAMPLE_ROOT}\pipeline
iii) Start the logstash container, replace the {SAMPLE_ROOT} with your directory name
$ docker run -h logstash --name logstash --link elasticsearch:elasticsearch \
-it --rm -v ~/{SAMPLE_ROOT}/pipeline:/usr/share/logstash/pipeline/ \
-p 5044:5044 docker.elastic.co/logstash/logstash:6.2.2
- Configure filebeat to ship the ballerina logs
i) Create a file named filebeat.yml
with the following content
filebeat.prospectors:
- type: log
paths:
- /usr/share/filebeat/ballerina.log
output.logstash:
hosts: ["logstash:5044"]
NOTE : Modify the ownership of filebeat.yml file using $chmod go-w filebeat.yml
ii) Save the above filebeat.yml
inside a directory named as {SAMPLE_ROOT}\filebeat
iii) Start the logstash container, replace the {SAMPLE_ROOT} with your directory name
$ docker run -v {SAMPLE_ROOT}/filbeat/filebeat.yml:/usr/share/filebeat/filebeat.yml \
-v {SAMPLE_ROOT}/guide/trip-management/ballerina.log:/usr/share\
/filebeat/ballerina.log --link logstash:logstash docker.elastic.co/beats/filebeat:6.2.2
- Access Kibana to visualize the logs using following URL
http://localhost:5601