手搓一个ChatUI需要分几步

只关注项目代码的同学可以直接跳转目录中的正文部分,查看项目仓库和功能介绍。

引言

Chatbot的UI界面设计,是和传统软件不同的:都是当面一个简洁的对话框,框里预备着热乎的工具,可以随时更新。

像我这样做工的牛马,午休傍晚下了班,每每花几个token,问一两句弱智吧的问题,——这是一年多前的事,现在一个问题都得设计个COT,——不用登录,在主页改改prompt,测完了就发群里刷一下存在感。

倘肯多费点心思,便可以问问3.11和3.9谁大,或者让姥姥念CD-Key哄你睡,姑且当作谈资了。如果充点真金白金,那就能获得一个API-Key,但我们这些牛马,多是白嫖党,大抵没有这样阔绰。只有公司报销的领域专家,才能点进绑定支付方式的页面里,买这买那的,写个脚本直接调用API慢慢跑着测。

我从毕业起,就在IT通信行业当伙计,父母和掌柜都说,不是e人,怕侍候不了VIP客户,就在组里开发开发后端。

公司内的stackholder,虽然容易说话,但是唠唠叨叨缠夹不清的也很不少。他们往往要亲眼看着代码跑起来,看过消息交互的抓包格式对了没有,又亲看将服务器整到shutdown,还能定时恢复,然后放心:在这严重监督之下,出一点小问题也很为难。

所以过了几天,掌柜说这些活太简单了,让去看看AI。幸亏大学里有点底子,也算对口,便改为既要开发后端,又要设计前端,还要兼顾AI算法和应用场景的奇怪职务了。

我从此便每天挂在Github上,看看有什么可以拿来用的App做个Demo。虽然没什么失职,但是总觉有些单调。

有些App太臃肿,npm install跑完就带上了全家;

有些太简单,没法适配微信群里一天更新三回的模型和后端框架。

只有最后狠下心来自己悄悄写一个,取个名就叫Chat-UI,算是勉强满意,所以来说说。

https://github.com/AI-QL/chat-uiicon-default.png?t=O83Ahttps://github.com/AI-QL/chat-ui

Chat-UI是不需要npm install和run time就能运行的,只需唯一的html单文件。

它身材很小巧;蓝白配色,底部夹杂着一些mdi图标;只有简单的light主题。

用的虽然是VUE,可是只用CDN引入了JS库,简单快捷,而且都是latest版本,似乎不用更新,也不用安装。

    <script src="https://cdn.jsdelivr.org/npm/vue/dist/vue.global.prod.min.js"></script>
    <script src="https://cdn.jsdelivr.org/npm/vuetify/dist/vuetify.js"></script>
    <link href="https://cdn.jsdelivr.org/npm/vuetify/dist/vuetify.css" rel="stylesheet">


LLM对人说话,总是满口markdown,教人半懂不懂的。又因为我自己前端开发经验不过关,便直接引入了md-editor-v3,替我可视化markdown。

fetch回来的JSON body刚收到,所有的组件就都看着它笑,有的叫道,

“OpenAI,你又增加新的限制区域了!”它不回答,只是吐出一行,“Fail to fetch XXX”便不再多言。

他们又故意的高声嚷道,“Nidia NIM 都免费提供 Llama 405B 了!”

OpenAI睁大眼睛说,“哪有什么免费的...”

“就是免费的,只要注册时候额外申请一下免费token就行了,随便用。”

OpenAI便涨红了脸,额上的青筋条条绽出,争辩道,“那也是有数量上限的,而且不支持跨域,跨域你懂吗?”接连便是难懂的话,什么“CORS”, 什么“Allow Headers”之类,

引得众后端都哄笑起来:那我再开发一个Proxy不就搞定了吗,workflow内充满了开源的空气。

简单小巧的OpenAI API访问:一键部署,轻松上手-CSDN博客文章浏览阅读1.6k次,点赞62次,收藏16次。OpenAI Proxy Docker项目不仅简单小巧,而且易于上手。通过使用最新的Bun运行时,你可以在保持高性能的同时,得到可靠的代理服务解决方案。https://blog.csdn.net/aiqlcom/article/details/144684140

听人家背地里谈论,OpenAI原来也免费送过5刀的token,但没有境外手机号,又不会绑境外信用卡;于是愈用愈少,弄到将要过期了。幸而开源vllm跑得一手好模型,便换一个模型当后端用用。

可惜OpenAI又传染的一样坏脾气,就是API版本控制。坐不到几天,便连JSON和定义,一齐更新了。如是几次,没有办法,便免不了再开发一个API UI,偶然查看一下最新格式。

https://api-ui.aiql.com/ 第一次登陆比较慢,可以下载html修改CDNicon-default.png?t=O83Ahttps://api-ui.aiql.com/

但是OpenAI在我们这里,品行比别家都好,格式支持的最全,报错最详细;虽然间或没有在文档里更新,暂时记在Issue里,但不出一月,定然修复,从Issue里done掉了名字。


OpenAI融了几回资,涨红的股价渐渐走了高,旁人便又问道,“OpenAI,AGI当真有出路么”,OpenAI看着问他的人,显出不屑置辩的神气。

他们便接着说道,“你怎的连半个开源的模型也没有呢?”OpenAI立刻显出颓唐不安模样,脸上笼上了一层灰色,嘴里说些话;这回可是全是金鍂鑫𨰻之类,一下全懂了。在这时候,众人也都哄笑起来:workflow内充满了闭源的空气。

在这些时候,我可以附和着challenge,掌柜是决不责备的。而且掌柜见了那些用OpenAI模型的项目,也每每这样发难,引人发笑。

OpenAI自己知道不能开源,便只好向各方面发力。有一回对我发邮件,“你知道多模态和Function call么?”

我略略点一点头。

他说,“用过,……我便要考你一考。request里的messages,怎样写的?”

我想,API文档里都写着的,也配考我么? 便回过脸去,不再理会。

OpenAI等了许久,很恳切的说道,“不会用吧?……我教给你,记着!这些format应该记着。将来做掌柜的时候,装逼要用。”

我暗想我和掌柜的等级还很远呢,而且我们掌柜也从不看开发细节;又好笑,又不耐烦,懒懒的答他道,

“谁要你教,不就是content里面一个string么?”OpenAI显出极高兴的样子,将一个长长的schema敲着我的脑袋,点头说,“对呀对呀!……messages光是role就有六样写法,你知道么?”我愈不耐烦了,努着嘴走远。

OpenAI刚写了个oneOf,想在后面跟上[developer, system, user, assistent, tool, function],见我毫不热心,便又叹一口气,显出极惋惜的样子, 把function标为了deprecated,然后又把user和tool定义了不一样的content parts。

(user, assistent, tool 里面的支持格式其实不一样哟)

有几回,隔壁开发组见得着KPI,也赶热闹,围住了OpenAI。它便给他们模型吃,一周一更新。开发组从年初加班到年尾,仍然不散,眼睛都望着array里不断新增的 [o1, o1-2024-12-17, o1-preview, o1-preview-2024-09-12, o1-mini, o1-mini-2024-09-12, gpt-4o, gpt-4o-2024-11-20, gpt-4o-2024-08-06, gpt-4o-2024-05-13, gpt-4o-audio-preview, gpt-4o-audio-preview-2024-10-01, gpt-4o-audio-preview-2024-12-17, gpt-4o-mini-audio-preview, gpt-4o-mini-audio-preview-2024-12-17, chatgpt-4o-latest, gpt-4o-mini, gpt-4o-mini-2024-07-18, gpt-4-turbo, gpt-4-turbo-2024-04-09, gpt-4-0125-preview, gpt-4-turbo-preview, gpt-4-1106-preview, gpt-4-vision-preview, gpt-4, gpt-4-0314, gpt-4-0613, gpt-4-32k, gpt-4-32k-0314, gpt-4-32k-0613, gpt-3.5-turbo, gpt-3.5-turbo-16k, gpt-3.5-turbo-0301, gpt-3.5-turbo-0613, gpt-3.5-turbo-1106, gpt-3.5-turbo-0125, gpt-3.5-turbo-16k-0613]。

OpenAI着了慌,伸开五指将文档罩住,弯腰下去说道,“不多了,我已经不多了。”直起身又看一看股价,自己摇头说,“不多不多!多乎哉?不多也。”于是这一群Prompt Engineer都在哭声里走散了。

OpenAI是这样的使人快活,可是没有他,别人也便这么过。vllm,TGI,部署个小模型当后端也不是不能用。

https://github.com/huggingface/text-generation-inferenceicon-default.png?t=O83Ahttps://github.com/huggingface/text-generation-inferencehttps://github.com/vllm-project/vllmicon-default.png?t=O83Ahttps://github.com/vllm-project/vllm

谁曾想TGI基于conda,而这玩意license不能给大组织商用:

我知道AI多半是个杀猪盘,但是没曾想,屠夫和刀子都没动手,甚至猪还没哼哼,砧板第一个跳起来给了我一巴掌:

Anaconda 许可变更解读icon-default.png?t=O83Ahttps://www.datacamp.com/blog/navigating-anaconda-licensing

有一天,大约是2024年圣诞前的两三天,掌柜正在慢慢的定OKR,取下粉板,忽然说,“OpenAI长久没有新东西了。”我才也觉得它的确长久没啥新闻了。

一个带着黑眼圈的人说道,“它怎么没更新? ……他o1-pro了。”

掌柜说,“哦!”

“它总仍旧是更新。这一回,是自己卷,竟连更12天。它家的新东西,200刀一个月,还只能Web访问,用得起么?”

“后来怎么样?”“怎么样?先充钱,后来是封号,信用卡记录被老婆看到了,再打折了腿。”

“后来呢?”“后来打折了腿了。”

“打折了怎样呢?”“怎样?……谁晓得?许是换用Llama了。”掌柜也不再问,仍然慢慢的算他的帐。

圣诞过后,东北风是一天凉比一天,看看将近元旦;

我整天的靠着AI的火,也须换开源模型了。

clone了一整天的模型,没有一个能用的,我正合了眼坐着。忽然间听得一个声音,“试试这MOE吧。”

这声音虽然极低,却很耳熟。看时又全没有人。站起来向外一望,竟是Deepseek在微信群里悄悄发着自己的照片:

它脸上全是benchmark,已经看不出原本的样子;穿一件自研的大衣,盘着两腿,下面垫一个MOE,用草绳在肩上挂住;见了我,又说道,“试一试MOE吧。”

掌柜也伸出头去,一面说,“新的开源模型么?训练成本才557.6万美元呢!”

Deepseek很疲惫的仰面答道,“这,v3不算新模型了……已经超过Llama 405B了。也赶上GPT-4o了,性能算好的了。”

掌柜仍然同平常一样,笑着对我说,“你看,又多了一个连部署都部署不起的模型”。但Deepseek这回却不十分分辩,单说了一句“我在Huggingface上开源了!”

“开源? 那要不我们试试?支持Function call吗” Deepseek低声说道,“支持,支,早就支持了……”他的眼色,很像恳求我,不要再细问(问就是官方文档更新慢)。

此时已经聚集了几个开发,便去查看文档了。我重新配置了Chat-UI,清空对话记录,重置了MCP server。他从新人注册里面摸出了一个月的免费token,放在我手里,见他满手都是卡,也不知是哪个哥布林洞穴里来的,想必模型便是用这2000张卡炼出来。不一会,我们调通了代码,他便又在旁人的说笑声中,去下一个交流群里了。

自此以后,就没有再看其他模型了。到了年底,掌柜取下粉板说,“还有Llama 3.3呢!”到了第二天的中午,又说,“Mistral最近有更新么?”到第三天又问,“Qwen是不是也在推新模型”,于是我狠狠心,把这些都用vllm测了一遍。


我到现在终于差不多可以确定Chat-UI大概可以拿出来给大家用一用了。

https://github.com/AI-QL/chat-ui 帮忙点个赞吧icon-default.png?t=O83Ahttps://github.com/AI-QL/chat-ui

正文

其实要把一个项目说清楚,一篇短文或者回答是不够的。

然而我这人又不善言辞,只好自欺欺人:“一个纯粹的前端项目,不需要用复杂的介绍来丰富的,交互逻辑清晰,满足基本功能需求,就可以了。“

所以这里只是估摸着说一下基本的功能,具体的代码解读,等有时间再做吧。

配置

我开发部署后端的时候,经常头疼没有一个前端可以用,主要是后端的URL和model名称一直在变,有时候又需要给别人展示,直接curl太粗暴了,所以Chat-UI里可以随意配置URL,还可以通过Json文件快速导入写好的配置:

Prompts

支持插入多个prompts,会在messages里面帮你合并一起放到system role对应的text content里。这样可以测试不同提示词以及顺序对模型的影响。暂时用不上的,但是已经写好的prompt,可以放到backup里,随时拖拽交换对比。

MD格式支持

可以支持Markdown语法显示,比如Mermaid图和代码块:

同时也可以一键复制文本,或者显示原始文本。

历史记录

记录聊天历史,可以点击载入,或者下载并保存为Json文件,也可以单独或批量删除。

多模态

可以支持图片多模态,记得选择后端模型的时候要选择vision模型,比如Llama vision或者Pixtral。

i18n

提供基本的多语言支持,选了5种特殊字符比较多的语言,有需要的同学可以自己加

MCP

可以作为渲染端对接 MCP(Model Context Protocol)Servers,具体参考另一个项目:

MCP(Model Context Protocol)模型上下文协议 实战篇1_mcp协议内容-CSDN博客文章浏览阅读2.6k次,点赞70次,收藏10次。MCP 协议(Model Context Protocol)模型上下文协议,简单实战,快速开发一个AI Agent APP_mcp协议内容https://ai-ql.blog.csdn.net/article/details/144324496

这样可以扩展更多agent功能以及function call

Response兼容

对于一些正常OpenAI SDK兼容的后端都可以支持。对于一些特殊后端,比如Cloudflare的 Workers AI (如下这种Rest API定义):

{
  "result": {
    "response": "Hello, World first appeared in 1974 at Bell Labs when Brian Kernighan included it in the C programming language example. It became widely used as a basic test program due to simplicity and clarity. It represents an inviting greeting from a program to the world."
  },
  "success": true,
  "errors": [],
  "messages": []
}

或者python写的直接plain text返回的response,也都可以兼容(是的,真的有这样的后端,除非自己写一个不然真的找不到能支持这种离谱格式的玩意)。

也可以勾选支持流式输出(stream)。

Cloudflare Pages

项目比较简单,直接fork之后可以部署在自己的域名下,如果不熟悉Cloudflare pages请查看链接:

https://developers.cloudflare.com/pagesicon-default.png?t=O83Ahttps://developers.cloudflare.com/pages

Docker

也提供直接Docker部署的镜像:

Docker Image - AIQL/Chat-UIicon-default.png?t=O83Ahttps://hub.docker.com/repository/docker/aiql/chat-ui/tags?page=1&ordering=last_updated

既然有了镜像,所以默认可以支持其他容器化部署,比如K8s,或者在Huggingface上直接一行代码搞定:

From aiql/chat-ui

https://huggingface.co/spaces/AI-QL/chat-uiicon-default.png?t=O83Ahttps://huggingface.co/spaces/AI-QL/chat-ui

这里也提供一个例子。

HTML

最后,也是最推荐的用法,可以直接在Github上下载 index.html,双击打开即可,不用安装也不用run。

https://github.com/AI-QL/chat-uiicon-default.png?t=O83Ahttps://github.com/AI-QL/chat-ui所有代码和部署教程已经同步至 GitHub 仓库,如果觉得项目对你有帮助,欢迎点个 ⭐️ 支持,非常感谢!

### 如何动实现或定制化 XXL-Job 调度框架 #### 一、XXL-JOB 的基本概念与架构 XXL-JOB 是一种高效的布式任务调度框架,其核心组件为两部:**调度中心 (Admin)** 和 **执行器 (Executor)**。调度中心负责管理任务的创建、修改和触发逻辑,而执行器则专注于具体任务的实际运行。 在实际应用中,如果需要动实现或定制化 XXL-JOB,则需深入了解以下几个关键点: 1. **调度中心的功能扩展** - 调度中心的核心职责是对任务进行统一管理,包括任务的状态监控、日志记录以及动态调整任务参数等功能。 - 可通过继承 `IJobHandler` 接口来自定义任务处理器[^1]。 2. **执行器的角色与配置** - 执行器作为任务的具体执行单元,在 XXL-JOB 中扮演着至关重要的角色。每个执行器都具有唯一的 `AppName` 标识,并能够周期性向调度中心注册自身的状态信息。 - 开发者可以通过设置不同的属性(如名称、排序、注册方式等),灵活控制执行器的行为模式[^2]。 --- #### 二、动实现自定义 XXL-JOB 教程 以下是基于 Spring Boot 集成的动实现流程: ##### 1. 添加依赖项 首先,在项目的 `pom.xml` 文件中引入必要的 Maven 依赖库: ```xml <dependency> <groupId>com.xuxueli</groupId> <artifactId>xxl-job-core</artifactId> <version>2.3.0</version> </dependency> ``` ##### 2. 创建自定义 Job 处理类 开发人员可以根据业务需求编写具体的任务逻辑,如下所示: ```java import com.xxl.job.core.handler.annotation.XXLJob; import org.springframework.stereotype.Component; @Component public class CustomJob { @XXLJob("customTask") // 定义任务名 public void execute() { System.out.println("Custom task is running..."); // 自定义任务逻辑写在此处 } } ``` 此处需要注意的是,方法上的注解 `@XXLJob` 提供了对任务基本信息的支持,例如指定任务名称以便于后续调用[^1]。 ##### 3. 配置执行器信息 编辑 `application.properties` 或 `application.yml` 文件来完成执行器的基础设定工作: ```properties xxl.job.admin.addresses=http://localhost:8080/xxl-job-admin # 调度中心地址 xxl.job.executor.appname=MyCustomExecutor # 设置执行器 AppName xxl.job.executor.ip= # 动态获取本地 IP 地址 xxl.job.executor.port=9999 # 指定端口号用于接收回调请求 xxl.job.executor.logpath=/data/applogs/xxl-job/jobhandler # 日志存储路径 xxl.job.executor.logretentiondays=30 # 日志保留天数 ``` 以上配置允许开发者精确地定义执行器的工作环境及其与其他服务交互的方式。 ##### 4. 启动并测试 当所有准备工作完成后,启动应用程序即可使新构建的任务生效。此时访问调度中心页面验证是否成功加载新的执行器实例及相关任务条目。 --- #### 三、高级定制选项 对于更复杂的场景,还可以考虑以下几种增强段: - **动态调整任务优先级**: 修改数据库表结构中的字段值影响任务队列顺序; - **跨平台兼容性优化**: 改善 RESTful API 设计满足不同语言客户端接入需求; - **性能瓶颈析工具集成**: 借助 APM 平台跟踪慢查询或者高耗资源操作现象。 这些措施有助于进一步提升系统的稳定性和适应能力。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值