Http流式响应接口封装

1. 背景

需要对接某个大模型的流式http对话接口,因为该项目中还需要对接另外的大模型服务但是另一个请求方式为websocket,为了实现模型服务切换以及对客户端做统一的流式响应,需要对大模型服务的调用做统一的流式响应封装。

环境:JDK11

2. 代码

2.1 回调函数

java 复制代码
@FunctionalInterface
public interface LLMChatCallback {

    void callback(String message);

}

2.2 统一对话接口

java 复制代码
public interface LLMService {

    /**
     * 模型对话
     *
     * @param payload 用于请求大模型的参数
     * @param callback 响应回调 调用者通过实现callback函数接收消息
     **/
    void dialogue(String payload, LLMChatCallback callback);

}

2.3 StringSubscriber

java 复制代码
import lombok.extern.slf4j.Slf4j;
import java.util.concurrent.Flow;

/**
 * http响应式流订阅
 **/
@Slf4j
public class StringSubscriber implements Flow.Subscriber<String> {

    Flow.Subscription subscription;

    /**
     * 消息回调接口
     */
    private final LLMChatCallback callback;

    public StringSubscriber(LLMChatCallback callback) {
        this.callback = callback;
    }

    @Override
    public void onSubscribe(Flow.Subscription subscription) {
        this.subscription = subscription;
        subscription.request(1);
    }

    @Override
    public void onNext(String item) {
        log.debug("onNext: {}", item);
        if (callback != null) {
            // 可做业务处理
            callback.callback(item);
        }
        subscription.request(1);
    }

    @Override
    public void onError(Throwable throwable) {
        log.error("StringSubscriber error", throwable);
    }

    @Override
    public void onComplete() {
        log.debug("StringSubscriber complete");
    }

}

2.4 大模型Service实现

java 复制代码
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;

import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.time.Duration;

/**
 * B大模型服务
 **/
@Slf4j
@Service
public class BLMService implements LLMService {

    @Override
    public void dialogue(String payload, LLMChatCallback callback) {
        HttpRequest httpRequest = HttpRequest.newBuilder(URI.create("this is url")).POST(HttpRequest.BodyPublishers.ofString("this is payload")).build();
        HttpClient httpClient = HttpClient.newBuilder().connectTimeout(Duration.ofMinutes(5)).version(HttpClient.Version.HTTP_2).build();
        HttpResponse.BodySubscriber<Void> bodySubscriber = HttpResponse.BodySubscribers.fromLineSubscriber(new StringSubscriber(callback));
        httpClient.sendAsync(httpRequest, responseInfo -> bodySubscriber).join();
    }

}

2.5 使用

java 复制代码
@Resource
private BLMService bLMService;

public void void dialogue() {
    // 构建请求负载
    String payload = "this is payload";
    // 可以扩展使用工厂及策略模式获取对应的模型对话service
    bLMService.dialogue(payload, message -> {
        // 处理业务接收返回的信息 比如通过grpc的流式响应 onNext(message) 同步返回
        System.out.println(message);
    });
}
相关推荐
BUG?不,是彩蛋!几秒前
Maven-Java 项目到底解决了什么痛点?
java·servlet·maven
小池先生1 分钟前
idea配置代码注释模板
java·ide·intellij-idea
inferno2 分钟前
Maven基础(一)
java·开发语言·maven
摇滚侠6 分钟前
Spring Boot3零基础教程,Reactive-Stream 规范核心接口,笔记103
java·spring boot·笔记
我也要当昏君12 分钟前
4.4 【2014统考真题】
网络·智能路由器
憧憬成为原神糕手13 分钟前
传输层协议 TCP 三次握手/四次挥手 TIME_WAIT
网络·tcp/ip·安全
花阴偷移13 分钟前
逆向基础--汇编基础(CS与IP) (05)
网络·汇编·网络协议·tcp/ip
捷米研发三部37 分钟前
全自动化立体仓库巷道堆垛机使用西门子1500PLC通过EtherCAT主站转Profinet实现与EtherCAT协议的伺服进行通讯方案案例
网络
天玺-vains37 分钟前
借助Github Action实现通过 HTTP 请求触发邮件通知
网络协议·http·github
王道长服务器 | 亚马逊云1 小时前
帝国CMS + AWS:老牌内容系统的新生之路
服务器·网络·数据库·云计算·aws