聊聊langchain4j的HTTP Client
csdh11 2025-05-16 12:37 2 浏览
序
本文主要研究一下langchain4j的HTTP Client
langchain4j-http-client
langchain4j提供了langchain4j-http-client模块,它实现了一个HttpClient SPI(服务提供者接口),其他模块通过该接口调用LLM提供商的REST API。这意味着底层HTTP客户端可以被自定义,通过实现HttpClient SPI,还可以集成任何其他HTTP客户端。目前,有两个现成的实现:
- langchain4j-http-client-jdk模块中的JdkHttpClient。在例如langchain4j-open-ai等有引用的模块中默认使用
- langchain4j-http-client-spring-restclient模块中的SpringRestClient。在例如langchain4j-open-ai-spring-boot-starter等有引用的spring boot starter中默认使用
HttpClient
langchain4j-http-client/src/main/java/dev/langchain4j/http/client/HttpClient.java
@Experimental
public interface HttpClient {
/**
* Executes a given HTTP request synchronously and returns the response.
* This method blocks until the entire response is received.
*
* @param request the HTTP request to be executed.
* @return a {@link SuccessfulHttpResponse} containing the response data for successful HTTP requests (2XX status codes)
* @throws HttpException if the server returns a client (4XX) or server (5XX) error response
* @throws RuntimeException if an unexpected error occurs during request execution (e.g., network issues, timeouts)
*/
SuccessfulHttpResponse execute(HttpRequest request) throws HttpException, RuntimeException;
/**
* Executes a given HTTP request asynchronously with server-sent events (SSE) handling.
* This method returns immediately while processing continues on a separate thread.
* Events are processed through the provided {@link ServerSentEventListener}.
* <p>
* The execution flow is as follows:
* <ol>
* <li>The request is initiated asynchronously</li>
* <li>Received SSE data is parsed using the {@link DefaultServerSentEventParser}</li>
* <li>Parsed events are delivered to the listener's appropriate methods</li>
* <li>If an error occurs, {@link ServerSentEventListener#onError(Throwable)} is called</li>
* </ol>
* <p>
* If any exception is thrown from the listener's methods, the stream processing
* will be terminated and no further events will be processed.
*
* @param request the HTTP request to be executed.
* @param listener the listener to receive parsed events and error notifications.
*/
default void execute(HttpRequest request, ServerSentEventListener listener) {
execute(request, new DefaultServerSentEventParser(), listener);
}
/**
* Executes a given HTTP request asynchronously with server-sent events (SSE) handling.
* This method returns immediately while processing continues on a separate thread.
* Events are processed through the provided {@link ServerSentEventListener}.
* <p>
* The execution flow is as follows:
* <ol>
* <li>The request is initiated asynchronously</li>
* <li>Received SSE data is parsed using the provided parser</li>
* <li>Parsed events are delivered to the listener's appropriate methods</li>
* <li>If an error occurs, {@link ServerSentEventListener#onError(Throwable)} is called</li>
* </ol>
* <p>
* If any exception is thrown from the listener's methods, the stream processing
* will be terminated and no further events will be processed.
*
* @param request the HTTP request to be executed.
* @param parser the parser to process incoming server-sent events.
* @param listener the listener to receive parsed events and error notifications.
*/
void execute(HttpRequest request, ServerSentEventParser parser, ServerSentEventListener listener);
}
HttpClient定义了execute方法,其中有一个支持ServerSentEventListener
HttpClientBuilderFactory
langchain4j-http-client/src/main/java/dev/langchain4j/http/client/HttpClientBuilderFactory.java
@Experimental
public interface HttpClientBuilderFactory {
HttpClientBuilder create();
}
HttpClientBuilderFactory定义了create方法返回HttpClientBuilder
HttpClientBuilder
langchain4j-http-client/src/main/java/dev/langchain4j/http/client/HttpClientBuilder.java
@Experimental
public interface HttpClientBuilder {
Duration connectTimeout();
HttpClientBuilder connectTimeout(Duration timeout);
Duration readTimeout();
HttpClientBuilder readTimeout(Duration timeout);
HttpClient build();
}
HttpClientBuilder定义了connectTimeout、readTimeout、build方法
HttpClientBuilderLoader
langchain4j-http-client/src/main/java/dev/langchain4j/http/client/HttpClientBuilderLoader.java
@Experimental
public class HttpClientBuilderLoader {
public static HttpClientBuilder loadHttpClientBuilder() {
Collection<HttpClientBuilderFactory> factories = loadFactories(HttpClientBuilderFactory.class);
if (factories.size() > 1) {
List<String> factoryNames = factories.stream()
.map(factory -> factory.getClass().getName())
.toList();
throw new IllegalStateException(String.format("Conflict: multiple HTTP clients have been found " +
"in the classpath: %s. Please explicitly specify the one you wish to use.", factoryNames));
}
for (HttpClientBuilderFactory factory : factories) {
return factory.create();
}
throw new IllegalStateException("No HTTP client has been found in the classpath");
}
}
HttpClientBuilderLoader提供了loadHttpClientBuilder方法,它通过
dev.langchain4j.spi.ServiceHelper.loadFactories方法来加载HttpClientBuilderFactory的实现类
ServiceHelper
langchain4j-core/src/main/java/dev/langchain4j/spi/ServiceHelper.java
/**
* Load all the services of a given type.
*
* @param clazz the type of service
* @param <T> the type of service
* @return the list of services, empty if none
*/
public static <T> Collection<T> loadFactories(Class<T> clazz) {
return loadFactories(clazz, null);
}
/**
* Load all the services of a given type.
*
* <p>Utility mechanism around {@code ServiceLoader.load()}</p>
*
* <ul>
* <li>If classloader is {@code null}, will try {@code ServiceLoader.load(clazz)}</li>
* <li>If classloader is not {@code null}, will try {@code ServiceLoader.load(clazz, classloader)}</li>
* </ul>
*
* <p>If the above return nothing, will fall back to {@code ServiceLoader.load(clazz, $this class loader$)}</p>
*
* @param clazz the type of service
* @param classLoader the classloader to use, may be null
* @param <T> the type of service
* @return the list of services, empty if none
*/
public static <T> Collection<T> loadFactories(Class<T> clazz, /* @Nullable */ ClassLoader classLoader) {
List<T> result;
if (classLoader != null) {
result = loadAll(ServiceLoader.load(clazz, classLoader));
} else {
// this is equivalent to:
// ServiceLoader.load(clazz, TCCL);
result = loadAll(ServiceLoader.load(clazz));
}
if (result.isEmpty()) {
// By default, ServiceLoader.load uses the TCCL, this may not be enough in environment dealing with
// classloaders differently such as OSGi. So we should try to use the classloader having loaded this
// class. In OSGi it would be the bundle exposing vert.x and so have access to all its classes.
result = loadAll(ServiceLoader.load(clazz, ServiceHelper.class.getClassLoader()));
}
return result;
}
ServiceHelper是对JDK自带的ServiceLoader.load()进行了封装
langchain4j-http-client-jdk
JdkHttpClientBuilderFactory
dev/langchain4j/http/client/jdk/JdkHttpClientBuilderFactory.java
public class JdkHttpClientBuilderFactory implements HttpClientBuilderFactory {
@Override
public JdkHttpClientBuilder create() {
return JdkHttpClient.builder();
}
}
JdkHttpClientBuilderFactory的create返回的是JdkHttpClientBuilder
JdkHttpClientBuilder
dev/langchain4j/http/client/jdk/JdkHttpClientBuilder.java
public class JdkHttpClientBuilder implements HttpClientBuilder {
private java.net.http.HttpClient.Builder httpClientBuilder;
private Duration connectTimeout;
private Duration readTimeout;
public java.net.http.HttpClient.Builder httpClientBuilder() {
return httpClientBuilder;
}
public JdkHttpClientBuilder httpClientBuilder(java.net.http.HttpClient.Builder httpClientBuilder) {
this.httpClientBuilder = httpClientBuilder;
return this;
}
@Override
public Duration connectTimeout() {
return connectTimeout;
}
@Override
public JdkHttpClientBuilder connectTimeout(Duration connectTimeout) {
this.connectTimeout = connectTimeout;
return this;
}
@Override
public Duration readTimeout() {
return readTimeout;
}
@Override
public JdkHttpClientBuilder readTimeout(Duration readTimeout) {
this.readTimeout = readTimeout;
return this;
}
@Override
public JdkHttpClient build() {
return new JdkHttpClient(this);
}
}
JdkHttpClientBuilder的build构建的是JdkHttpClient
JdkHttpClient
dev/langchain4j/http/client/jdk/JdkHttpClient.java
public class JdkHttpClient implements HttpClient {
private final java.net.http.HttpClient delegate;
private final Duration readTimeout;
public JdkHttpClient(JdkHttpClientBuilder builder) {
java.net.http.HttpClient.Builder httpClientBuilder =
getOrDefault(builder.httpClientBuilder(), java.net.http.HttpClient::newBuilder);
if (builder.connectTimeout() != null) {
httpClientBuilder.connectTimeout(builder.connectTimeout());
}
this.delegate = httpClientBuilder.build();
this.readTimeout = builder.readTimeout();
}
public static JdkHttpClientBuilder builder() {
return new JdkHttpClientBuilder();
}
@Override
public SuccessfulHttpResponse execute(HttpRequest request) throws HttpException {
try {
java.net.http.HttpRequest jdkRequest = toJdkRequest(request);
java.net.http.HttpResponse<String> jdkResponse = delegate.send(jdkRequest, BodyHandlers.ofString());
if (!isSuccessful(jdkResponse)) {
throw new HttpException(jdkResponse.statusCode(), jdkResponse.body());
}
return fromJdkResponse(jdkResponse, jdkResponse.body());
} catch (IOException | InterruptedException e) {
throw new RuntimeException(e);
}
}
@Override
public void execute(HttpRequest request, ServerSentEventParser parser, ServerSentEventListener listener) {
java.net.http.HttpRequest jdkRequest = toJdkRequest(request);
delegate.sendAsync(jdkRequest, BodyHandlers.ofInputStream())
.thenAccept(jdkResponse -> {
if (!isSuccessful(jdkResponse)) {
listener.onError(new HttpException(jdkResponse.statusCode(), readBody(jdkResponse)));
return;
}
SuccessfulHttpResponse response = fromJdkResponse(jdkResponse, null);
listener.onOpen(response);
try (InputStream inputStream = jdkResponse.body()) {
parser.parse(inputStream, listener);
listener.onClose();
} catch (IOException e) {
throw new RuntimeException(e);
}
})
.exceptionally(throwable -> {
if (throwable.getCause() instanceof HttpTimeoutException) {
listener.onError(throwable);
}
return null;
});
}
//......
}
JdkHttpClient的构造器创建了java.net.http.HttpClient作为delegate
langchain4j-http-client-spring-restclient
SpringRestClientBuilderFactory
dev/langchain4j/http/client/spring/restclient/SpringRestClientBuilderFactory.java
public class SpringRestClientBuilderFactory implements HttpClientBuilderFactory {
@Override
public SpringRestClientBuilder create() {
return SpringRestClient.builder();
}
}
SpringRestClientBuilderFactory的create返回的是SpringRestClientBuilder
SpringRestClientBuilder
dev/langchain4j/http/client/spring/restclient/SpringRestClientBuilder.java
public class SpringRestClientBuilder implements HttpClientBuilder {
private RestClient.Builder restClientBuilder;
private AsyncTaskExecutor streamingRequestExecutor;
private Boolean createDefaultStreamingRequestExecutor = true;
private Duration connectTimeout;
private Duration readTimeout;
public RestClient.Builder restClientBuilder() {
return restClientBuilder;
}
public SpringRestClientBuilder restClientBuilder(RestClient.Builder restClientBuilder) {
this.restClientBuilder = restClientBuilder;
return this;
}
public AsyncTaskExecutor streamingRequestExecutor() {
return streamingRequestExecutor;
}
public SpringRestClientBuilder streamingRequestExecutor(AsyncTaskExecutor streamingRequestExecutor) {
this.streamingRequestExecutor = streamingRequestExecutor;
return this;
}
public Boolean createDefaultStreamingRequestExecutor() {
return createDefaultStreamingRequestExecutor;
}
public SpringRestClientBuilder createDefaultStreamingRequestExecutor(Boolean createDefaultStreamingRequestExecutor) {
this.createDefaultStreamingRequestExecutor = createDefaultStreamingRequestExecutor;
return this;
}
@Override
public Duration connectTimeout() {
return connectTimeout;
}
@Override
public SpringRestClientBuilder connectTimeout(Duration connectTimeout) {
this.connectTimeout = connectTimeout;
return this;
}
@Override
public Duration readTimeout() {
return readTimeout;
}
@Override
public SpringRestClientBuilder readTimeout(Duration readTimeout) {
this.readTimeout = readTimeout;
return this;
}
@Override
public SpringRestClient build() {
return new SpringRestClient(this);
}
}
SpringRestClientBuilder的build创建的是SpringRestClient
SpringRestClient
dev/langchain4j/http/client/spring/restclient/SpringRestClient.java
public class SpringRestClient implements HttpClient {
private final RestClient delegate;
private final AsyncTaskExecutor streamingRequestExecutor;
public SpringRestClient(SpringRestClientBuilder builder) {
RestClient.Builder restClientBuilder = getOrDefault(builder.restClientBuilder(), RestClient::builder);
ClientHttpRequestFactorySettings settings = ClientHttpRequestFactorySettings.DEFAULTS;
if (builder.connectTimeout() != null) {
settings = settings.withConnectTimeout(builder.connectTimeout());
}
if (builder.readTimeout() != null) {
settings = settings.withReadTimeout(builder.readTimeout());
}
ClientHttpRequestFactory clientHttpRequestFactory = ClientHttpRequestFactories.get(settings);
this.delegate = restClientBuilder
.requestFactory(clientHttpRequestFactory)
.build();
this.streamingRequestExecutor = getOrDefault(builder.streamingRequestExecutor(), () -> {
if (builder.createDefaultStreamingRequestExecutor()) {
return createDefaultStreamingRequestExecutor();
} else {
return null;
}
});
}
//......
@Override
public SuccessfulHttpResponse execute(HttpRequest request) throws HttpException {
try {
ResponseEntity<String> responseEntity = toSpringRestClientRequest(request)
.retrieve()
.toEntity(String.class);
return SuccessfulHttpResponse.builder()
.statusCode(responseEntity.getStatusCode().value())
.headers(responseEntity.getHeaders())
.body(responseEntity.getBody())
.build();
} catch (RestClientResponseException e) {
throw new HttpException(e.getStatusCode().value(), e.getMessage());
}
}
@Override
public void execute(HttpRequest request, ServerSentEventParser parser, ServerSentEventListener listener) {
streamingRequestExecutor.execute(() -> {
try {
toSpringRestClientRequest(request)
.exchange((springRequest, springResponse) -> {
int statusCode = springResponse.getStatusCode().value();
if (!springResponse.getStatusCode().is2xxSuccessful()) {
String body = springResponse.bodyTo(String.class);
listener.onError(new HttpException(statusCode, body));
return null;
}
SuccessfulHttpResponse response = SuccessfulHttpResponse.builder()
.statusCode(statusCode)
.headers(springResponse.getHeaders())
.build();
listener.onOpen(response);
try (InputStream inputStream = springResponse.getBody()) {
parser.parse(inputStream, listener);
listener.onClose();
}
return null;
});
} catch (Exception e) {
if (e.getCause() instanceof SocketTimeoutException) {
listener.onError(e);
}
}
});
}
}
SpringRestClient的构造器构建了
org.springframework.web.client.RestClient为delegate,同时要构建了streamingRequestExecutor用于ServerSentEventListener
小结
langchain4j的langchain4j-http-client模块定义了HttpClient SPI(服务提供者接口),目前有两个现成的实现:
- langchain4j-http-client-jdk模块中的JdkHttpClient,使用的是java.net.http.HttpClient
- langchain4j-http-client-spring-restclient模块中的SpringRestClient,使用的是org.springframework.web.client.RestClient
doc
- customizable-http-client
相关推荐
- 完美解决华硕主板机箱前置面板连接音箱无声音问题
-
#精品长文创作季#最近新组装了一台电脑主机,听歌时发现,音箱音频输入线插入主板自带的音频输出端口LINEOUT声音正常。但当将音箱音频输入线插入机箱自带的音频输出端口却没有声音。...
- Java 监控直播流rtsp协议转rtmp、hls、httpflv协议返回浏览器
-
Java监控直播流rtsp协议转rtmp、hls、httpflv协议返回浏览器...
- 编译安装OpenCV全攻略,附本地源码包
-
本文以Ubuntu(适用于18,20,22版本,其他的未进行测试)为例,介绍了编译安装OpenCV的最佳实践。其他平台比如windows,只需要参考本文的思路做稍微调整即可。一、为什么需要编译安装Op...
- 音视频录制+RTMP直播推拉流
-
音视频录制:1,录音通过条件编译识别pc或者mac://条件编译技术识别pc或者mac#ifdefQ_OS_WIN#defineFMT_NAME"dshow"...
- Python Socket.IO 简介
-
Socket.IO是一种传输协议,支持客户端与服务器之间基于事件的实时双向通信。客户端通常但不总是web浏览器,一般情况下,客户端是采用js编写的,当然,也可以用python、Java等。...
- MyCat系列二--配置文件之server.xml
-
MyCat系列二--配置文件之server.xml从【安装与基本使用】一文中,可以看到MyCat运行,需要进行一些配置文件的相关配置,本文将重点介绍配置文件server.xml的基本元素信息,尽量了解...
- 3分钟搞懂Socket.IO:接口调试技巧
-
在数字化协作需求井喷的今天,即时通讯、协同文档、MMO游戏等场景对实时交互提出了严苛要求。传统HTTP协议"请求-响应"的离散式通信机制已难以满足需求,这正是Socket.IO这类实时...
- 「云原生」Containerd ctr,crictl 和 nerdctl 命令介绍与实战操作
-
一、概述作为接替Docker运行时的Containerd在早在Kubernetes1.7时就能直接与Kubelet集成使用,只是大部分时候我们因熟悉Docker,在部署集群时采用了默认的dockers...
- 搜狗开源srpc:自研高性能通用RPC框架
-
今年7月底,搜狗公司开源了内部的工业级C++服务器引擎Workflow,一路收获业内许多认可和关注。9月15日,作为Workflow最重要的生态项目——srpc,一个基于其打造的轻量级RPC框架,也在...
- socket.io 权限认证
-
socket.io是一款流行的WebSocket库,提供了实时双向通信的能力。它支持多种身份验证方式,可用于保护某些敏感信息。接下来,我们将介绍如何使用socket.io权限认证1.安装依...
- Linux高性能服务器设计
-
C10K和C10M计算机领域的很多技术都是需求推动的,上世纪90年代,由于互联网的飞速发展,网络服务器无法支撑快速增长的用户规模。1999年,DanKegel提出了著名的C10问题:一台服务器上同时...
- 19K 标星!开源局域网文件传输神器,AirDrop 跨平台替代品!
-
传统文件传输方式(如USB、蓝牙、邮件)要么速度慢,要么操作繁琐,而AirDrop仅限苹果生态。如果你还在为手机和电脑之间传文件来回折腾?还在装各种臃肿的客户端、扫码连Wi-Fi、用数据线拖来...
- WebRTC 入门教程:搭建WebRTC信令服务器
-
前言我们在学习WebRTC时,首先要把实验环境搭建好,这样我们就可以在上面做各种实验了。对于WebRTC来说,它有一整套规范,如怎样使用它的接口、使用SDP进行媒体协商、通过ICE收集地址并进...
- 在 Go 中使用 Websockets 和 Socket.IO - 教程
-
注意-本教程是使用Go1.9版和googollee/go-socket.io编写的Websockets是我觉得有趣的东西,因为它们为我们提供了应用程序之间通信的替代选项,而不是标准的R...
- 2021 年 Node.js 开发人员学习路线图
-
Node.js自发布以来,已成为业界重要破局者之一。Uber、Medium、PayPal和沃尔玛等大型企业,纷纷将技术栈转向Node.js。Node.js支持开发功能强大的应用,例如实时追踪...
- 一周热门
- 最近发表
- 标签列表
-
- mydisktest_v298 (34)
- document.appendchild (35)
- 头像打包下载 (61)
- acmecadconverter_8.52绿色版 (39)
- word文档批量处理大师破解版 (36)
- server2016安装密钥 (33)
- mysql 昨天的日期 (37)
- parsevideo (33)
- 个人网站源码 (37)
- centos7.4下载 (33)
- mysql 查询今天的数据 (34)
- intouch2014r2sp1永久授权 (36)
- 先锋影音源资2019 (35)
- jdk1.8.0_191下载 (33)
- axure9注册码 (33)
- pts/1 (33)
- spire.pdf 破解版 (35)
- shiro jwt (35)
- sklearn中文手册pdf (35)
- itextsharp使用手册 (33)
- 凯立德2012夏季版懒人包 (34)
- 冒险岛代码查询器 (34)
- 128*128png图片 (34)
- jdk1.8.0_131下载 (34)
- dos 删除目录下所有子目录及文件 (36)