<properties>
<easy.version>0.0.72</adrastea.version>
</properties>
<!-- Get the latest version from https://mvnrepository.com/artifact/io.github.devlibx.easy/http -->
The link to helper library which has many common useful utilities. Helper Library
The link to Kafka library which has a easy way to produce or consume on Kafka. Kafka Library
Http Module provides API to make HTTP calls. It ensures that APIs are called with circuit-breaker, time limit.
Maven Dependency
<dependency>
<groupId>io.github.devlibx.easy</groupId>
<artifactId>http</artifactId>
<version>${easy.version}</version>
</dependency>
Calling http in Sync
// Example 1 - Make a call and get response in a Map
Map result = EasyHttp.callSync(
Call.builder(Map.class)
.withServerAndApi("jsonplaceholder", "getPosts")
.addPathParam("id", 1)
.withBody("Any object - it will be converted to json string internally")
.build()
);
// Example 2 - Make a call and get response in a Pojo
@Data
private static class ResponsePojo {
@JsonProperty("userId")
private Integer userId;
@JsonProperty("id")
private Integer id;
@JsonProperty("title")
private String title;
@JsonProperty("completed")
private boolean completed;
}
ResponsePojo resultWithPojo = EasyHttp.callSync(
Call.builder(ResponsePojo.class)
.withServerAndApi("jsonplaceholder", "getPosts")
.addPathParam("id", 1)
.build()
);
String jsonString = JsonUtils.asJson(resultWithPojo);
log.info("Print Result as Json String = " + jsonString);
// Print Result as Json String = {"userId":1,"id":1,"title":"sunt aut facere repellat provident occaecati excepturi optio reprehenderit","completed":false}
// Example 3 - Make a call and get process error
// EasyHttpExceptions.EasyHttpRequestException - this is the super class to catch all error (or you can use specific sub-classes)
try {
ResponsePojo resultWithPojoError = EasyHttp.callSync(
Call.builder(ResponsePojo.class)
.withServerAndApi("jsonplaceholder", "getPosts")
.addPathParam("id_make_it_fail", 1)
.build()
);
// You can catch
// EasyHttpExceptions.Easy4xxException e1;
// EasyHttpExceptions.EasyUnauthorizedRequestException e;
// EasyHttpExceptions.EasyRequestTimeOutException e;
} catch (EasyHttpExceptions.Easy5xxException e) {
// You can cache specific errors
log.error("Api failed (5xx error): status=" + e.getStatusCode() + " byteBody=" + e.getBody());
} catch (EasyHttpExceptions.EasyHttpRequestException e) {
log.error("Api failed: status=" + e.getStatusCode() + " byteBody=" + e.getBody());
}
Calling http in Async
EasyHttp.callAsync(
Call.builder(Map.class)
.withServerAndApi("jsonplaceholder", "getPostsAsync")
.addPathParam("id", 1)
.withBody("Any object - it will be converted to json string internally")
.build()
)
.subscribeOn(Schedulers.io())
.subscribe(
result -> {
log.info("Print Result as Json String = " + JsonUtils.asJson(result));
// Result = {"userId":1,"id":1,"title":"some text ..."}
},
throwable -> {
// throwable is a EasyHttpRequestException
// You can visit sub-classes EasyHttpRequestException to get catch exact issue
// e.g. EasyNotFoundException - for Http 404
});
Custom request and response body function. e.g. proto-buf API (over HTTP)
AddUserRequest request = AddUserRequest.newBuilder()
.setNameProvided(name)
.build();
AddUserResponse response = EasyHttp.callSync(
Call.builder(AddUserResponse.class)
.withServerAndApi("someService", "someApi")
.asContentTypeProtoBuffer()
.withResponseBuilder(AddUserResponse::parseFrom)
.withRequestBodyFunc(request::toByteArray)
.build()
);
package io.github.devlibx.easy.http;
import com.google.inject.Guice;
import com.google.inject.Injector;
import io.gitbub.devlibx.easy.helper.ApplicationContext;
import io.gitbub.devlibx.easy.helper.LoggingHelper;
import io.gitbub.devlibx.easy.helper.json.JsonUtils;
import io.gitbub.devlibx.easy.helper.yaml.YamlUtils;
import io.github.devlibx.easy.http.config.Config;
import io.github.devlibx.easy.http.module.EasyHttpModule;
import io.github.devlibx.easy.http.sync.SyncRequestProcessor;
import io.github.devlibx.easy.http.util.Call;
import io.github.devlibx.easy.http.util.EasyHttp;
import junit.framework.TestCase;
import lombok.extern.slf4j.Slf4j;
import org.apache.log4j.Logger;
import java.util.Map;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import static org.apache.log4j.Level.TRACE;
@Slf4j
public class DemoApplication extends TestCase {
private Injector injector;
@Override
protected void setUp() throws Exception {
super.setUp();
LoggingHelper.setupLogging();
Logger.getLogger(SyncRequestProcessor.class).setLevel(TRACE);
// Setup injector (Onetime MUST setup before we call EasyHttp.setup())
injector = Guice.createInjector(new EasyHttpModule());
ApplicationContext.setInjector(injector);
// Read config and setup EasyHttp
Config config = YamlUtils.readYamlCamelCase("demo_app_config.yaml", Config.class);
EasyHttp.setup(config);
}
public void testSyncApiCall() {
Map result = EasyHttp.callSync(
Call.builder(Map.class)
.withServerAndApi("jsonplaceholder", "getPosts")
.addPathParam("id", 1)
.build()
);
log.info("Print Result as Json String = " + JsonUtils.asJson(result));
// Result = {"userId":1,"id":1,"title":"some text ..."}
}
public void testAsyncApiCall() throws Exception {
CountDownLatch waitForComplete = new CountDownLatch(1);
EasyHttp.callAsync(
Call.builder(Map.class)
.withServerAndApi("jsonplaceholder", "getPostsAsync")
.addPathParam("id", 1)
.build()
).subscribe(
result -> {
log.info("Print Result as Json String = " + JsonUtils.asJson(result));
// Result = {"userId":1,"id":1,"title":"some text ..."}
},
throwable -> {
});
waitForComplete.await(5, TimeUnit.SECONDS);
// Or you can use blockingSubscribe();
}
}
demo_app_config.yaml
servers:
jsonplaceholder:
host: jsonplaceholder.typicode.com
port: 443
https: true
connectTimeout: 1000
connectionRequestTimeout: 1000
apis:
getPosts:
method: GET
path: /posts/${id}
server: jsonplaceholder
timeout: 10000
concurrency: 3
waitBeforeClosingCircuitAfterError: 5000
getPostsAsync:
method: GET
path: /posts/${id}
server: jsonplaceholder
timeout: 1000
concurrency: 3
async: true
"use port = -1": this will remove the port from url "http://something:/abcd". With -1 it will become "http://something/abcd"
- timeout - timeout for the API. Your EasyHttp.call**() Api will timeout after the given time
- concurrency - how many parallel calls can be made to this API.
- rps - if you know "rps" of API call, then you should set
rps
e.g. rps: 100. The EasyHttp will automatically setup required threads to support concurrent calls. You don't need to setconcurrency
manually. For example, if timeout=20 and rps=100 then EasyHttp will setconcurrency=2
- waitBeforeClosingCircuitAfterError = when circuit is opened due to error, all the calls to external service will
not be done (circuit breaker will avoid calling external service).
However, circuit breaker has to check after some time e.g. 10sec to see if external service is up or not. To do this it allows few calls to go to external service to see if external service is up.
This time is configured usingwaitBeforeClosingCircuitAfterError (default 10sec)
.
If you keep it too small e.g. 10-50ms; your circuit breaker will call external service frequently after error.
If you keep it large e.g. 30sec then; your circuit breaker will wait for 30 sec before calling external service. And you can see many requests are failing.
When you set rps
then you have to consider rps
from the single node i.e. how many requests this single node is going
to call. For example, if you call an external API with 1000 rps
; and you run 10 nodes, then a single node has rps=100
database-mysql module provides support for easy MySQL helper.
See "io.github.devlibx.easy.database.mysql.ExampleApp" example
<!-- POM Dependency -->
<dependency>
<groupId>io.github.devlibx.easy</groupId>
<artifactId>database-mysql</artifactId>
<version>${easy.version}</version>
</dependency>
You must setup IMysqlHelper before using it. A sample setup is als given below.
// Insert to DB
IMysqlHelper mysqlHelper = injector.getInstance(IMysqlHelper.class);
Long id = mysqlHelper.persist(
"",
"INSERT INTO my_table(col) VALUES(?)",
preparedStatement -> {
preparedStatement.setString(1, "some value");
}
);
// Find a row
String result = mysqlHelper.findOne(
"",
"SELECT col from my_table",
statement -> {
},
rs -> rs.getString(1),
String.class
).orElse("");
Setup to use this MySQL helper:
Create test database for tests:
create database users;
create database test_me;
// Setup DB - datasource
DbConfig dbConfig = new DbConfig();
dbConfig.setDriverClassName("com.mysql.jdbc.Driver");
dbConfig.setJdbcUrl("YOUR JDBC URL");
dbConfig.setUsername("username");
dbConfig.setPassword("password");
MySqlConfigs mySqlConfigs = new MySqlConfigs();
mySqlConfigs.addConfig(dbConfig);
// Setup module
injector = Guice.createInjector(new AbstractModule() {
@Override
protected void configure() {
bind(IMetrics.class).to(IMetrics.NoOpMetrics.class);
bind(MySqlConfigs.class).toInstance(mySqlConfigs);
}
}, new DatabaseMySQLModule());
ApplicationContext.setInjector(injector);
// Start DB
IDatabaseService databaseService = injector.getInstance(IDatabaseService.class);
databaseService.startDatabase();
This module provides a distributed lock e.g. a MySQL based distributed lock is implemented by easy libs. This example
shows a class ResourceWithLocking
with a method which should take a lock before it is called.
Note - you will see MySQL and database dependency in the example code.
package com.devlibx.pack.resources.lock;
import io.github.devlibx.easy.lock.DistributedLock;
import io.github.devlibx.easy.lock.IDistributedLock;
import io.github.devlibx.easy.lock.IDistributedLockIdResolver;
import lombok.extern.slf4j.Slf4j;
import org.aopalliance.intercept.MethodInvocation;
import javax.inject.Singleton;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.atomic.AtomicLong;
@Slf4j
public class ResourceWithLocking {
private static final AtomicLong COUNTER = new AtomicLong();
// When this method is called - it will first take a distributed lock
//
// MySQL based lock:
// ================
// For example if we use MySQL lock provider then we take a lock using the "lockId" in MySQL.
//
// A implementation of IDistributedLockIdResolver class (InternalDistributedLockIdResolver in this case)
// will be called to get the value of "lockId".
// "createLockRequest()" method os called with the method arguments. You can extract the ID to lock against.
//
@DistributedLock(lockIdResolver = InternalDistributedLockIdResolver.class)
public Map<String, Object> methodWhichShouldBeLocked(String someId) {
try {
Thread.sleep(1);
} catch (InterruptedException ignored) {
}
log.trace("Called method methodWhichShouldBeLocked - id={}", someId);
Map<String, Object> result = new HashMap<>();
result.put("counter", COUNTER.incrementAndGet());
result.put("id", someId);
return result;
}
@Singleton
public static class InternalDistributedLockIdResolver implements IDistributedLockIdResolver {
@Override
public IDistributedLock.LockRequest createLockRequest(MethodInvocation invocation, Object[] arguments) {
return IDistributedLock.LockRequest.builder()
.lockId(arguments[0].toString())
.build();
}
}
}
public class Application {
public static void main(String[] args) {
// Setup DB - datasource
DbConfig dbConfig = new DbConfig();
dbConfig.setDriverClassName("com.mysql.jdbc.Driver");
dbConfig.setJdbcUrl("YOUR JDBC URL");
dbConfig.setUsername("username");
dbConfig.setPassword("password");
MySqlConfigs mySqlConfigs = new MySqlConfigs();
mySqlConfigs.addConfig(dbConfig);
// Setup module
injector = Guice.createInjector(new AbstractModule() {
@Override
protected void configure() {
bind(IMetrics.class).to(IMetrics.NoOpMetrics.class);
bind(MySqlConfigs.class).toInstance(mySqlConfigs);
}
}, new DatabaseMySQLModule());
ApplicationContext.setInjector(injector);
// Start DB
IDatabaseService databaseService = injector.getInstance(IDatabaseService.class);
databaseService.startDatabase();
// Setup lock service
IDistributedLockService distributedLockService = injector.getInstance(IDistributedLockService.class);
distributedLockService.initialize();
// Example ResourceWithLocking
ResourceWithLocking resourceWithLocking = injector.getInstance(ResourceWithLocking.class);
// This API call will lock before running
// For example if you run this method concurrently in many threads, then all execution
// will be sequential (for same lock id)
Map<String, Object> response = resourceWithLocking.methodWhichShouldBeLocked("1234");
System.out.println(response);
}
}
Convert Java object to JSON string
<!-- POM Dependency -->
<dependency>
<groupId>io.github.devlibx.easy</groupId>
<artifactId>helper</artifactId>
<version>${easy.version}</version>
</dependency>
// A pojo object to stringify
@Data
public class PojoClass {
private String str;
private int anInt;
}
PojoClass testClass = new PojoClass();
testClass.setStr("some string");
testClass.setAnInt(11);
StringHelper stringHelper = new StringHelper();
stringHelper.stringify(testClass);
// Output - {"str":"some string","an_int":11}
Testing module is create to help testing with MySQL, DynamoDB, Kafaka
Following setup is needed to ues testing module
# Run this command if you are using DynamoDB testing
====================================================
docker pull testcontainers/ryuk:0.3.0
# Only If you are buulding "easy" sourcecode
==========================================
Create following databases in MySQL
create database users;
create database test_me;