HTTP client
Module provides a thin layer of abstraction over HTTP client libraries to create HTTP clients using declarative-style annotations or using client in imperative-style.
OkHttp¶
HTTP client implementation based on OkHttp library. Please note that the implementation is written in Kotlin and uses appropriate dependencies.
Dependency¶
Dependency build.gradle
:
Module:
Dependency build.gradle.kts
:
Module:
Configuration¶
Example of the complete configuration described in the OkHttpClientConfig
and HttpClientConfig
classes (default or example values are specified):
httpClient {
ok {
followRedirects = true //(1)!
httpVersion = "HTTP_1_1" //(2)!
}
connectTimeout = "5s" //(3)!
readTimeout = "2m" //(4)!
useEnvProxy = false //(5)!
proxy {
host = "localhost" //(6)!
port = 8090 //(7)!
user = "user" //(8)!
password = "password" //(9)!
nonProxyHosts = [ "host1", "host2" ] //(10)!
}
telemetry {
logging {
enabled = false //(11)!
mask = "***" //(12)!
maskQueries = [ ] //(13)!
maskHeaders = [ "authorization" ] //(14)!
pathTemplate = true //(15)!
}
metrics {
enabled = true //(16)!
slo = [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] //(17)!
}
tracing {
enabled = true //(18)!
}
}
}
- Whether to follow redirects in HTTP
- Maximum HTTP protocol version used (available values:
HTTP_1_1
/HTTP_2
/HTTP_3
) - Maximum time to establish a connection
- Maximum time to read a response
- Whether to use environment variables to configure the proxy
- Proxy address (optional)
- Proxy port (optional)
- User for the proxy (optional)
- Password for the proxy (optional)
- Hosts that should be excluded from proxying (optional)
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
httpClient:
ok:
followRedirects: true #(1)!
httpVersion: "HTTP_1_1" #(2)!
connectTimeout: "5s" #(3)!
readTimeout: "2m" #(4)!
useEnvProxy: false #(5)!
proxy:
host: "localhost" #(6)!
port: 8090 #(7)!
user: "user" #(8)!
password: "password" #(9)!
nonProxyHosts: [ "host1", "host2" ] #(10)!
telemetry:
logging:
enabled: false #(11)!
mask: "***" #(12)!
maskQueries: [ ] #(13)!
maskHeaders: [ "authorization" ] #(14)!
pathTemplate: true #(15)!
metrics:
enabled: true #(16)!
slo: [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] #(17)!
telemetry:
enabled: true #(18)!
- Whether to follow redirects in HTTP
- Maximum HTTP protocol version used (available values:
HTTP_1_1
/HTTP_2
/HTTP_3
) - Maximum time to establish a connection
- Maximum time to read a response
- Whether to use environment variables to configure the proxy
- Proxy address (optional)
- Proxy port (optional)
- User for the proxy (optional)
- Password for the proxy (optional)
- Hosts that should be excluded from proxying (optional)
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
Configurer¶
Example of how to configure OkHttp client builder, OkHttpConfigurer
must be available as component:
AsyncHttpClient¶
HTTP client implementation based on the Async HTTP Client library.
Dependency¶
Dependency build.gradle
:
Module:
Dependency build.gradle.kts
:
Module:
The HttpClient
interface implementation is AsyncHttpClient
and is available for manual implementation.
Configuration¶
Example of the complete configuration described in the AsyncHttpClientConfig
and HttpClientConfig
classes (default or example values are specified):
httpClient {
async {
followRedirects = true //(1)!
}
connectTimeout = "5s" //(2)!
readTimeout = "2m" //(3)!
useEnvProxy = false //(4)!
proxy {
host = "localhost" //(5)!
port = 8090 //(6)!
user = "user" //(7)!
password = "password" //(8)!
nonProxyHosts = [ "host1", "host2" ] //(9)!
}
telemetry {
logging {
enabled = false //(10)!
mask = "***" //(11)!
maskQueries = [ ] //(12)!
maskHeaders = [ "authorization" ] //(13)!
pathTemplate = true //(14)!
}
metrics {
enabled = true //(15)!
slo = [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] //(16)!
}
tracing {
enabled = true //(17)!
}
}
}
- Whether to follow redirects in HTTP
- Maximum time to establish a connection
- Maximum time to read a response
- Whether to use environment variables to configure the proxy
- Proxy address (optional)
- Proxy port (optional)
- User for the proxy (optional)
- Password for the proxy (optional)
- Hosts that should be excluded from proxying (optional)
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
httpClient:
async:
followRedirects: true #(1)!
connectTimeout: "5s" #(2)!
readTimeout: "2m" #(3)!
useEnvProxy: false #(4)!
proxy:
host: "localhost" #(5)!
port: 8090 #(6)!
user: "user" #(7)!
password: "password" #(8)!
nonProxyHosts: [ "host1", "host2" ] #(9)!
telemetry:
logging:
enabled: false #(10)!
mask: "***" #(11)!
maskQueries: [ ] #(12)!
maskHeaders: [ "authorization" ] #(13)!
pathTemplate: true #(14)!
metrics:
enabled: true #(15)!
slo: [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] #(16)!
telemetry:
enabled: true #(17)!
- Whether to follow redirects in HTTP
- Maximum time to establish a connection
- Maximum time to read a response
- Whether to use environment variables to configure the proxy
- Proxy address (optional)
- Proxy port (optional)
- User for the proxy (optional)
- Password for the proxy (optional)
- Hosts that should be excluded from proxying (optional)
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
You can also configure Netty transport.
Native client¶
Implementation of an HTTP client based on the native client provided in the JDK.
Dependency¶
Dependency build.gradle
:
Module:
Dependency build.gradle.kts
:
Module:
The HttpClient
interface implementation is JdkHttpClient
and is available for manual implementation.
Configuration¶
Example of the complete configuration described in the JdkHttpClientConfig
and HttpClientConfig
classes (default or example values are specified):
httpClient {
jdk {
threads = 2 //(1)!
httpVersion = "HTTP_1_1" //(2)!
}
connectTimeout = "5s" //(3)!
useEnvProxy = false //(4)!
proxy {
host = "localhost" //(5)!
port = 8090 //(6)!
user = "user" //(7)!
password = "password" //(8)!
nonProxyHosts = [ "host1", "host2" ] //(9)!
}
telemetry {
logging {
enabled = false //(10)!
mask = "***" //(11)!
maskQueries = [ ] //(12)!
maskHeaders = [ "authorization" ] //(13)!
pathTemplate = true //(14)!
}
metrics {
enabled = true //(15)!
slo = [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] //(16)!
}
tracing {
enabled = true //(17)!
}
}
}
- Number of threads for HTTP client
- Which version of HTTP protocol to use (available values:
HTTP_1_1
/HTTP_2
) - Maximum time to establish a connection
- Whether to use environment variables to configure the proxy
- Proxy address (optional)
- Proxy port (optional)
- User for the proxy (optional)
- Password for the proxy (optional)
- Hosts that should be excluded from proxying (optional)
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
httpClient:
jdk:
threads: 2 #(1)!
httpVersion: "HTTP_1_1" #(2)!
connectTimeout: "2s" #(3)!
useEnvProxy: false #(4)!
proxy:
host: "localhost" #(5)!
port: 8090 #(6)!
user: "user" #(7)!
password: "password" #(8)!
nonProxyHosts: [ "host1", "host2" ] #(9)!
telemetry:
logging:
enabled: false #(10)!
mask: "***" #(11)!
maskQueries: [ ] #(12)!
maskHeaders: [ "authorization" ] #(13)!
pathTemplate: true #(14)!
metrics:
enabled: true #(15)!
slo: [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] #(16)!
telemetry:
enabled: true #(17)!
- Number of threads for HTTP client
- Which version of HTTP protocol to use (available values:
HTTP_1_1
/HTTP_2
) - Maximum time to establish a connection
- Whether to use environment variables to configure the proxy
- Proxy address (optional)
- Proxy port (optional)
- User for the proxy (optional)
- Password for the proxy (optional)
- Hosts that should be excluded from proxying (optional)
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
Client declarative¶
It is suggested to use special annotations to create a declarative client:
@HttpClient
- indicates that the interface is a declarative HTTP client@HttpRoute
- specifies HTTP request type and request path
Client Configuration¶
The default configuration of a particular implementation of @HttpClient
uses the following path httpClient.{lower case class name}
for configuration lookup,
or specified in the configPath
parameter in the annotation:
Example configuration in the case of the httpClient.someClient
path described in the DeclarativeHttpClientConfig
class:
httpClient {
someClient {
url = "https://localhost:8090" //(1)!
requestTimeout = "10s" //(2)!
telemetry {
logging {
enabled = false //(3)!
mask = "***" //(4)!
maskQueries = [ ] //(5)!
maskHeaders = [ "authorization" ] //(6)!
pathTemplate = true //(7)!
}
metrics {
enabled = true //(8)!
slo = [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] //(9)!
}
tracing {
enabled = true //(10)!
}
}
}
}
- URL of the service where requests will be sent
- Maximum request timeout, may spans the entire call: resolving DNS, connecting, writing the request body, server processing, and reading the response body, if call requires redirects or retries all must complete within one timeout period
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
httpClient:
someClient:
url: "https://localhost:8090" #(1)!
requestTimeout: "10s" #(2)!
telemetry:
logging:
enabled: false #(3)!
mask: "***" #(4)!
maskQueries: [ ] #(5)!
maskHeaders: [ "authorization" ] #(6)!
pathTemplate: true #(7)!
metrics:
enabled: true #(8)!
slo: [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] #(9)!
telemetry:
enabled: true #(10)!
- URL of the service where requests will be sent
- Maximum request timeout, may spans the entire call: resolving DNS, connecting, writing the request body, server processing, and reading the response body, if call requires redirects or retries all must complete within one timeout period
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Enables module metrics (default
true
) - Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
Method Configuration¶
Using the above HTTP client example, it is possible to configure separately some of the parameters for a particular method, the configuration path
is determined by the path to the client and the method name, in the example above the configuration is httpClient.someClient
and method hello
the final path will be httpClient.someClient.hello
httpClient {
someClient {
hello {
requestTimeout = "10s" //(1)!
telemetry {
logging {
enabled = false //(2)!
mask = "***" //(3)!
maskQueries = [ ] //(4)!
maskHeaders = [ "authorization" ] //(5)!
pathTemplate = true //(6)!
}
metrics {
enabled = true //(7)!
slo = [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] //(8)!
}
tracing {
enabled = true //(9)!
}
}
}
}
}
- Maximum request timeout, may spans the entire call: resolving DNS, connecting, writing the request body, server processing, and reading the response body, if call requires redirects or retries all must complete within one timeout period
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Includes module metrics
- Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
httpClient:
someClient:
hello:
requestTimeout: "10s" #(1)!
telemetry:
logging:
enabled: false #(2)!
mask: "***" #(3)!
maskQueries: [ ] #(4)!
maskHeaders: [ "authorization" ] #(5)!
pathTemplate: true #(6)!
metrics:
enabled: true #(7)!
slo: [ 1, 10, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 20000, 30000, 60000, 90000 ] #(8)!
telemetry:
enabled: true #(9)!
- Maximum request timeout, may spans the entire call: resolving DNS, connecting, writing the request body, server processing, and reading the response body, if call requires redirects or retries all must complete within one timeout period
- Enables module logging (default
false
) - Mask that is used to hide specified headers and request/response parameters
- List of request parameters to be hidden
- List of request/response headers that should be hidden
- Whether to always use the request path template when logging. Default is to always use the path template, except for the
TRACE
logging level, which uses the full path. - Includes module metrics
- Configures SLO for DistributionSummary metrics
- Enables module tracing (default
true
)
Request¶
The section describes HTTP request transformations at a declarative HTTP client. It is suggested to use special annotations to specify request parameters.
Path parameter¶
@Path
- denotes the value of the request path part, the parameter itself is specified in {quote}
in the path
and the name of the parameter is specified in value
or is equal to the name of the method argument by default.
Query parameter¶
@Query
- value of the query parameter, the name of the parameter is specified in value
or is equal to the name of the method argument by default.
It is possible to send query parameters in key and value format, for this purpose it is assumed to use Map
type,
where the key is the parameter name and must be of type String
, and the parameter value can be of any type and will be processed through String.valueOf()
:
Header¶
@Header
- value of request header, parameter name is specified in value
or defaults to the method argument name.
It is possible to send request parameters in key and value format, for this purpose it is supposed to use HttpHeaders
type or Map
type,
where the key is the parameter name and must be of type String
, and the parameter value can be of any type and will be processed through String.valueOf()
:
Request body¶
Specifying the body of a request requires using a method argument without special annotations,
the default supported types are byte[]
, ByteBuffer
or String
.
Json¶
In order to indicate that the body is Json and needs to automatically create such a writer and embed it,
is required to use the special @Json
tag annotation:
Json module is required.
Text form¶
You can use FormUrlEncoded
as the body argument type and it will be processed as form data.
An example of a method call with this form would look like this:
Binary Form¶
You can use FormMultipart
as the body argument type and it will be treated as binary form.
An example of a method call with this form would look like this:
Custom body¶
If the body needs to be written in a way different from the standard mechanisms,
it is possible to use a special HttpClientRequestMapper
interface to implement your custom logic:
@HttpClient
public interface SomeClient {
record UserBody(String id) {}
final class UserRequestMapper implements HttpClientRequestMapper<UserBody> {
@Override
public HttpBodyOutput apply(Context ctx, UserBody value) {
return HttpBody.plaintext(value.id());
}
}
@HttpRoute(method = HttpMethod.POST, path = "/hello/world")
void hello(@Mapping(UserRequestMapper.class) UserBody body);
}
@HttpClient
interface SomeClient {
data class UserBody(val id: String)
class UserRequestMapper : HttpClientRequestMapper<UserBody> {
override fun apply(ctx: Context, value: UserBody): HttpBodyOutput {
return HttpBody.plaintext(value.id)
}
}
@HttpRoute(method = HttpMethod.POST, path = "/hello/world")
fun hello(@Mapping(UserRequestMapper::class) body: UserBody)
}
Cookie¶
@Cookie
- Cookie value, the parameter name is specified in value
or defaults to the method argument name.
Required parameters¶
By default, all arguments declared in a method are required (NotNull).
By default, all arguments declared in a method that do not use the Kotlin Nullability syntax are considered required (NotNull).
Optional parameters¶
If a method argument is optional, that is, it may not exist then,
@Nullable
annotation can be used:
@HttpClient
public interface SomeClient {
@HttpRoute(method = HttpMethod.GET, path = "/hello/world")
void hello(@Nullable @Query("queryValue") String queryValue); //(1)!
}
- Any
@Nullable
annotation will do, such asjavax.annotation.Nullable
/jakarta.annotation.Nullable
/org.jetbrains.annotations.Nullable
/ etc.
It is expected to use the Kotlin Nullability syntax and mark such a parameter as Nullable:
Response¶
The section describes the transformation of an HTTP response from a declarative HTTP client.
Response body¶
By default, you can use the standard response body return value types such as void
, byte[]
, ByteBuffer
or String
.
Json¶
If the body is to be read as Json, the @Json
annotation must be used over the method.
Json module is required.
Response Entity¶
If the intention is to read the body and also get the headers and status code of the response,
it is intended to use HttpResponseEntity
, which is a wrapper over the response body.
Below is an example similar to the Json example along with the HttpResponseEntity
wrapper:
Custom response¶
If you need to read the response in a different way, you can use the special HttpClientResponseMapper
interface:
@HttpClient
public interface SomeClient {
record MyResponse(String name) { }
final class ResponseMapper implements HttpClientResponseMapper<MyResponse> {
@Override
public MyResponse apply(HttpClientResponse response) throws IOException, HttpClientDecoderException {
try (var is = response.body().asInputStream()) {
final byte[] bytes = is.readAllBytes();
var body = new String(bytes, StandardCharsets.UTF_8);
return new MyResponse(body);
}
}
}
@Mapping(ResponseMapper.class)
@HttpRoute(method = HttpMethod.GET, path = "/hello/world")
MyResponse hello();
}
@HttpClient
interface SomeClient {
data class MyResponse(val name: String)
class ResponseMapper : HttpClientResponseMapper<MyResponse> {
@Throws(IOException::class, HttpClientDecoderException::class)
override fun apply(response: HttpClientResponse): MyResponse {
response.body().asInputStream().use {
val bytes: ByteArray = it.readAllBytes()
val body = String(bytes, StandardCharsets.UTF_8)
return MyResponse(body)
}
}
}
@Mapping(ResponseMapper::class)
@HttpRoute(method = HttpMethod.GET, path = "/hello/world")
fun hello(): MyResponse
}
Response Error¶
By default, when no converter tag or converter itself is specified, the conversion will be applied for 2xx
HTTP status codes,
for all others a HttpClientResponseException
exception will be thrown, which contains HTTP status code, response body and response headers.
Conversion by Code¶
If specific conversions are required depending on the HTTP status code of the response, you can use the @ResponseCodeMapper
annotation to specify a
correspondence between the HTTP status code and the HttpClientResponseMapper
resolver.
You can also use ResponseCodeMapper.DEFAULT
as an indication of the default behavior for all unlisted status codes.
@HttpClient
public interface SomeClient {
record UserResponse(UserResponse.Payload payload, UserResponse.Error error) {
public record Error(int code, String message) {}
public record Payload(String message) {}
}
@ResponseCodeMapper(code = ResponseCodeMapper.DEFAULT, mapper = ResponseErrorMapper.class)
@ResponseCodeMapper(code = 200, mapper = ResponseSuccessMapper.class)
@HttpRoute(method = HttpMethod.GET, path = "/hello/world")
UserResponse hello();
}
@HttpClient
interface SomeClient {
data class UserResponse(val payload: Payload, val error: Error) {
data class Error(val code: Int, val message: String)
data class Payload(val message: String)
}
@ResponseCodeMapper(code = ResponseCodeMapper.DEFAULT, mapper = ResponseErrorMapper::class)
@ResponseCodeMapper(code = 200, mapper = ResponseSuccessMapper::class)
@HttpRoute(method = HttpMethod.GET, path = "/hello/world")
fun hello(): UserResponse
}
In the example above, ResponseSuccessMapper
will be used for status code 200
,
and for all other status codes the ResponseErrorMapper
will be used.
Signatures¶
Available signatures for repository methods out of the box:
The T
refers to the type of the return value.
T myMethod()
CompletionStage<T> myMethod()
CompletionStageMono<T> myMethod()
Project Reactor (require dependency)
By T
we mean the type of the return value, either T?
, or Unit
.
myMethod(): T
suspend myMethod(): T
Kotlin Coroutine (require dependency asimplementation
)
Interceptors¶
You can create interceptors to change behavior or create additional behavior using the HttpClientInterceptor
class.
Interceptors can be applied to specific methods or to the entire @HttpClient
class:
@HttpClient
public interface SomeClient {
final class MethodInterceptor implements HttpClientInterceptor {
private final Component1 component1;
private MethodInterceptor(Component1 component1) {
this.component1 = component1;
}
@Override
public CompletionStage<HttpClientResponse> processRequest(Context ctx, InterceptChain chain, HttpClientRequest request) throws Exception {
component1.doSomething();
return chain.process(ctx, request);
}
}
@InterceptWith(MethodInterceptor.class)
@HttpRoute(method = HttpMethod.GET, path = "/hello/world")
void hello();
}
@HttpClient
interface SomeClient {
class MethodInterceptor(val component1: Component1) : HttpClientInterceptor {
@Throws(Exception::class)
override fun processRequest(
ctx: Context,
chain: HttpClientInterceptor.InterceptChain,
request: HttpClientRequest
): CompletionStage<HttpClientResponse> {
component1.doSomething()
return chain.process(ctx, request)
}
}
@InterceptWith(MethodInterceptor::class)
@HttpRoute(method = HttpMethod.GET, path = "/hello/world")
fun hello()
}
Authorization¶
Kora provides out-of-the-box interceptors that can be used for Basic/ApiKey/Bearer/OAuth authorization.
Basic¶
You need to configure an interceptor and configuration for Basic authorization:
@Module
public interface BasicAuthModule {
@ConfigSource("openapiAuth.basicAuth")
public interface BasicAuthConfig {
String username();
String password();
}
default BasicAuthHttpClientInterceptor basicAuther(BasicAuthConfig config) {
return new BasicAuthHttpClientInterceptor(config.username(), config.password());
}
}
@Module
interface BasicAuthModule {
@ConfigSource("openapiAuth.basicAuth")
interface BasicAuthConfig {
fun username(): String
fun password(): String
}
fun basicAuther(config: BasicAuthConfig): BasicAuthHttpClientInterceptor {
return BasicAuthHttpClientInterceptor(config.username(), config.password())
}
}
You can also provide your own HttpClientTokenProvider
implementation in the constructor if rules for getting secrets are different.
Then add interceptor for the entire HTTP client or specific methods.
ApiKey¶
You need to configure an interceptor and configuration for ApiKey authorization:
@Module
public interface ApiKeyAuthModule {
@ConfigSource("openapiAuth.apiKeyAuth")
interface ApiKeyAuthConfig {
String apiKey();
}
default ApiKeyHttpClientInterceptor apiKeyAuther(ApiKeyAuthConfig config) {
return new ApiKeyHttpClientInterceptor(ApiKeyLocation.HEADER, "X-API-KEY", config.apiKey());
}
}
Then add interceptor for the entire HTTP client or specific methods.
Bearer¶
You need to configure an interceptor for Bearer authorization:
You will need to implement the Bearer
token provisioning yourself using your custom HttpClientTokenProvider
implementation,
or use a constructor that accepts a static Bearer Token
.
public interface HttpClientTokenProvider {
CompletionStage<String> getToken(HttpClientRequest request);
}
Then add interceptor for the entire HTTP client or specific methods.
OAuth¶
Authorization by OAuth is similar to Bearer,
you need to implement HttpClientTokenProvider
yourself and put it in dependency container.
Client imperative¶
The base client represents the HttpClient
interface and is available for deployment:
public interface HttpClient {
CompletionStage<HttpClientResponse> execute(HttpClientRequest request); //(1)!
HttpClient with(HttpClientInterceptor interceptor); //(2)!
}
- Method of request execution
- A method that allows you to add various interceptors manually
You can use HttpClientRequestBuilder
to build requests manually: