v0.0.1
spring ai的基本使用
This commit is contained in:
commit
3208efe280
38
.gitignore
vendored
Normal file
38
.gitignore
vendored
Normal file
@ -0,0 +1,38 @@
|
||||
target/
|
||||
!.mvn/wrapper/maven-wrapper.jar
|
||||
!**/src/main/**/target/
|
||||
!**/src/test/**/target/
|
||||
|
||||
### IntelliJ IDEA ###
|
||||
.idea/modules.xml
|
||||
.idea/jarRepositories.xml
|
||||
.idea/compiler.xml
|
||||
.idea/libraries/
|
||||
*.iws
|
||||
*.iml
|
||||
*.ipr
|
||||
|
||||
### Eclipse ###
|
||||
.apt_generated
|
||||
.classpath
|
||||
.factorypath
|
||||
.project
|
||||
.settings
|
||||
.springBeans
|
||||
.sts4-cache
|
||||
|
||||
### NetBeans ###
|
||||
/nbproject/private/
|
||||
/nbbuild/
|
||||
/dist/
|
||||
/nbdist/
|
||||
/.nb-gradle/
|
||||
build/
|
||||
!**/src/main/**/build/
|
||||
!**/src/test/**/build/
|
||||
|
||||
### VS Code ###
|
||||
.vscode/
|
||||
|
||||
### Mac OS ###
|
||||
.DS_Store
|
||||
206
README.md
Normal file
206
README.md
Normal file
@ -0,0 +1,206 @@
|
||||
# Spring AI Demo
|
||||
|
||||
> 🤖 一个简洁优雅的 Spring AI 对话演示项目,基于 Spring Boot 3.2.0 与 Spring AI 1.0.0-M3 构建,支持流式响应与 Markdown 渲染。
|
||||
|
||||
[](https://spring.io/projects/spring-boot)
|
||||
[](https://spring.io/projects/spring-ai)
|
||||
[](https://openjdk.org/)
|
||||
[](LICENSE)
|
||||
|
||||
## ✨ 特性
|
||||
|
||||
- 🌐 **智能对话** - 基于 OpenAI/Ollama 的自然语言交互
|
||||
- 📡 **流式响应** - 实时逐字输出,体验流畅
|
||||
- 📝 **Markdown 支持** - 完整渲染代码块、表格、列表等格式
|
||||
- 👨💻 **代码高亮** - Highlight.js 自动语言检测与语法着色
|
||||
- 🖼️ **向量数据库** - Milvus 集成(预留 RAG 扩展能力)
|
||||
- 📔 **PDF 解析** - Apache PDFBox 文档处理支持
|
||||
- 🎨 **精美界面** - 深色主题响应式设计
|
||||
|
||||
## 🚀 快速开始
|
||||
|
||||
### 环境要求
|
||||
|
||||
- JDK 17+
|
||||
- Maven 3.8+
|
||||
- [Ollama](https://ollama.ai/) 本地服务
|
||||
|
||||
### 1. 安装 Ollama
|
||||
|
||||
```bash
|
||||
# macOS/Linux
|
||||
curl -fsSL https://ollama.ai/install.sh | sh
|
||||
|
||||
# 拉取模型
|
||||
ollama pull gpt-oss:120b-cloud
|
||||
|
||||
# 启动服务 (默认端口 11434)
|
||||
ollama serve
|
||||
```
|
||||
|
||||
### 2. 启动项目
|
||||
|
||||
```bash
|
||||
mvn spring-boot:run
|
||||
```
|
||||
|
||||
### 3. 访问应用
|
||||
|
||||
- **Web 界面**: http://localhost:8080
|
||||
- **健康检查**: http://localhost:8080/api/chat/test
|
||||
|
||||
## 🔧 技术栈
|
||||
|
||||
### 后端
|
||||
|
||||
| 组件 | 技术 | 版本 |
|
||||
|:---|:---|:---|
|
||||
| 基础框架 | Spring Boot | 3.2.0 |
|
||||
| AI 框架 | Spring AI | 1.0.0-M3 |
|
||||
| AI 模型 | OpenAI/Ollama | - |
|
||||
| 响应式编程 | Spring WebFlux | 3.2.0 |
|
||||
| 向量数据库 | Milvus Store | 1.0.0-M3 |
|
||||
| 文档处理 | Apache PDFBox | 2.0.29 |
|
||||
|
||||
### 前端
|
||||
|
||||
| 技术 | 用途 |
|
||||
|:---|:---|
|
||||
| HTML5 + CSS3 | 页面结构与样式 |
|
||||
| Marked.js | Markdown 解析渲染 |
|
||||
| Highlight.js | 代码语法高亮 |
|
||||
|
||||
## 📁 项目结构
|
||||
|
||||
```
|
||||
springAiDemo/
|
||||
├── src/
|
||||
│ └── main/
|
||||
│ ├── java/com/demo/
|
||||
│ │ ├── MyApplication.java # Spring Boot 启动入口
|
||||
│ │ └── controller/
|
||||
│ │ └── ChatController.java # AI 聊天 REST API
|
||||
│ └── resources/
|
||||
│ ├── application.yaml # 应用配置
|
||||
│ └── static/ # 前端静态资源
|
||||
│ ├── index.html # 主页面
|
||||
│ ├── css/style.css # 深色主题样式
|
||||
│ └── js/app.js # 流式响应逻辑
|
||||
├── pom.xml
|
||||
└── README.md
|
||||
```
|
||||
|
||||
## 💬 API 文档
|
||||
|
||||
### 端点列表
|
||||
|
||||
| 方法 | 端点 | 描述 | 参数 |
|
||||
|:---|:---|:---|:---|
|
||||
| GET | `/api/chat/test` | 健康检查 | - |
|
||||
| GET | `/api/chat/ai` | 同步 AI 对话 | `msg` |
|
||||
| GET | `/api/chat/ai/stream` | 流式 AI 对话 | `msg` |
|
||||
|
||||
### 请求示例
|
||||
|
||||
**同步对话:**
|
||||
```bash
|
||||
curl "http://localhost:8080/api/chat/ai?msg=什么是Spring AI"
|
||||
```
|
||||
|
||||
**流式对话:**
|
||||
```bash
|
||||
curl "http://localhost:8080/api/chat/ai/stream?msg=讲一个故事"
|
||||
```
|
||||
|
||||
### 响应格式
|
||||
|
||||
- **同步**: `text/plain` - 完整回复文本
|
||||
- **流式**: `text/html;charset=UTF-8` - Server-Sent Events 分块传输
|
||||
|
||||
## 🎨 界面预览
|
||||
|
||||
🧩 **现代化深色主题**
|
||||
- 渐变标题与毛玻璃效果
|
||||
- 流畅的消息气泡动画
|
||||
- Markdown 代码块高亮
|
||||
- 流式响应动画指示器
|
||||
|
||||
📝 **Markdown 支持**
|
||||
- 代码块与行内代码
|
||||
- 加粗、斜体、删除线
|
||||
- 有序/无序列表
|
||||
- 引用块、表格、链接
|
||||
|
||||
## 🛠️ 配置说明
|
||||
|
||||
### 模型配置 (`application.yaml`)
|
||||
|
||||
```yaml
|
||||
spring:
|
||||
ai:
|
||||
openai:
|
||||
base-url: http://localhost:11434 # Ollama 服务地址
|
||||
chat:
|
||||
options:
|
||||
model: gpt-oss:120b-cloud # 当前模型
|
||||
temperature: 0.7 # 生成温度
|
||||
```
|
||||
|
||||
**支持模型**(需在 Ollama 中预先拉取):
|
||||
- `gpt-oss:120b-cloud` - 默认模型
|
||||
- `kimi-k2.5:cloud` - Kimi 模型
|
||||
- `gemma4:e2b` - Google Gemma 模型
|
||||
|
||||
## 🎬 架构图
|
||||
|
||||
```
|
||||
┌─────────────┐ ┌──────────────────┐ ┌─────────────────┐
|
||||
│ Client │────▶│ ChatController │────▶│ ChatClient │
|
||||
│ (HTTP) │ │ (Spring MVC) │ │ (Spring AI) │
|
||||
└─────────────┘ └──────────────────┘ └────────┬────────┘
|
||||
│
|
||||
▼
|
||||
┌─────────────────┐
|
||||
│ Ollama/OpenAI │
|
||||
│ (LLM Provider) │
|
||||
└─────────────────┘
|
||||
```
|
||||
|
||||
## 📄 预留功能
|
||||
|
||||
以下依赖已集成,可按需启用:
|
||||
|
||||
- 🔍 **Milvus 向量数据库** - RAG 语义搜索
|
||||
- 📔 **PDF 文档解析** - 文档上传与处理
|
||||
- 🗨️ **Embedding 服务** - 文本向量化
|
||||
|
||||
## 👷 已知限制
|
||||
|
||||
1. 暂无对话历史持久化
|
||||
2. 暂未实现请求频率限制
|
||||
3. JUnit 版本较旧(3.8.1),建议升级至 JUnit 5
|
||||
|
||||
## 📖 待办事项
|
||||
|
||||
- [ ] 添加全局异常处理
|
||||
- [ ] 实现 RAG 文档问答
|
||||
- [ ] 对话历史存储
|
||||
- [ ] 动态模型切换 API
|
||||
- [ ] Docker 部署支持
|
||||
- [ ] 单元测试完善
|
||||
|
||||
## 📤 更新日志
|
||||
|
||||
### v1.0.0 (2026-04-17)
|
||||
- 🎉 初始版本发布
|
||||
- 支持流式 AI 对话
|
||||
- Markdown 渲染与代码高亮
|
||||
- 深色主题 Web 界面
|
||||
|
||||
## 📗 License
|
||||
|
||||
MIT © 2026 Spring AI Demo
|
||||
|
||||
---
|
||||
|
||||
🚀 **Made with Spring AI**
|
||||
118
pom.xml
Normal file
118
pom.xml
Normal file
@ -0,0 +1,118 @@
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<groupId>org.example</groupId>
|
||||
<artifactId>springAiDemo</artifactId>
|
||||
<version>1.0-SNAPSHOT</version>
|
||||
<packaging>jar</packaging>
|
||||
|
||||
<name>springAiDemo</name>
|
||||
<url>http://maven.apache.org</url>
|
||||
|
||||
<parent>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-parent</artifactId>
|
||||
<version>3.2.0</version>
|
||||
<relativePath/>
|
||||
</parent>
|
||||
<properties>
|
||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||
</properties>
|
||||
|
||||
<dependencies>
|
||||
<!-- spring web 依赖 -->
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-web</artifactId>
|
||||
</dependency>
|
||||
|
||||
<!--Spring AI Core -->
|
||||
<dependency>
|
||||
<groupId>org.springframework.ai</groupId>
|
||||
<artifactId>spring-ai-openai-spring-boot-starter</artifactId>
|
||||
</dependency>
|
||||
<!-- <dependency>-->
|
||||
<!-- <groupId>org.springframework.ai</groupId>-->
|
||||
<!-- <artifactId>spring-ai-ollama-spring-boot-starter</artifactId>-->
|
||||
<!-- </dependency>-->
|
||||
<!--spring web 流式接口依赖 -->
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-webflux</artifactId>
|
||||
</dependency>
|
||||
|
||||
<!--引入milvus和spring ai的集成包–>-->
|
||||
<dependency>
|
||||
<groupId>org.springframework.ai</groupId>
|
||||
<artifactId>spring-ai-milvus-store-spring-boot-starter</artifactId>
|
||||
</dependency>
|
||||
|
||||
<!-- <!– milvus客户端 –>-->
|
||||
<!-- <dependency>-->
|
||||
<!-- <groupId>io.milvus</groupId>-->
|
||||
<!-- <artifactId>milvus-sdk-java</artifactId>-->
|
||||
<!-- <version>2.3.4</version>-->
|
||||
<!-- </dependency>-->
|
||||
|
||||
|
||||
<!-- spring中用于校验的包 -->
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-validation</artifactId>
|
||||
</dependency>
|
||||
|
||||
|
||||
<!-- 增加处理pdf的相关依赖 -->
|
||||
<dependency>
|
||||
<groupId>org.apache.pdfbox</groupId>
|
||||
<artifactId>pdfbox</artifactId>
|
||||
<version>2.0.29</version>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>commons-fileupload</groupId>
|
||||
<artifactId>commons-fileupload</artifactId>
|
||||
<version>1.5</version>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.projectlombok</groupId>
|
||||
<artifactId>lombok</artifactId>
|
||||
<optional>true</optional>
|
||||
</dependency>
|
||||
|
||||
|
||||
<dependency>
|
||||
<groupId>junit</groupId>
|
||||
<artifactId>junit</artifactId>
|
||||
<version>3.8.1</version>
|
||||
<scope>test</scope>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
|
||||
|
||||
<dependencyManagement>
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>org.springframework.ai</groupId>
|
||||
<artifactId>spring-ai-bom</artifactId>
|
||||
<version>1.0.0-M3</version>
|
||||
<type>pom</type>
|
||||
<scope>import</scope>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
</dependencyManagement>
|
||||
|
||||
<repositories>
|
||||
<repository>
|
||||
<id>spring-milestones</id>
|
||||
<name>Spring Milestones</name>
|
||||
<url>https://repo.spring.io/milestone</url>
|
||||
<snapshots>
|
||||
<enabled>false</enabled>
|
||||
</snapshots>
|
||||
</repository>
|
||||
</repositories>
|
||||
|
||||
</project>
|
||||
|
||||
10
src/main/java/com/demo/MyApplication.java
Normal file
10
src/main/java/com/demo/MyApplication.java
Normal file
@ -0,0 +1,10 @@
|
||||
package com.demo;
|
||||
|
||||
import org.springframework.boot.SpringApplication;
|
||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||
@SpringBootApplication
|
||||
public class MyApplication {
|
||||
public static void main(String[] args) {
|
||||
SpringApplication.run(MyApplication.class, args);
|
||||
}
|
||||
}
|
||||
49
src/main/java/com/demo/controller/ChatController.java
Normal file
49
src/main/java/com/demo/controller/ChatController.java
Normal file
@ -0,0 +1,49 @@
|
||||
package com.demo.controller;
|
||||
|
||||
import org.springframework.ai.chat.client.ChatClient;
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
import org.springframework.web.bind.annotation.RequestMapping;
|
||||
import org.springframework.web.bind.annotation.RequestParam;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
import reactor.core.publisher.Flux;
|
||||
|
||||
@RestController
|
||||
@RequestMapping("/api/chat")
|
||||
public class ChatController {
|
||||
|
||||
private final ChatClient chatClient;
|
||||
|
||||
public ChatController(ChatClient.Builder chatClientBuilder) {
|
||||
this.chatClient = chatClientBuilder.build();
|
||||
}
|
||||
|
||||
@GetMapping("/test")
|
||||
public String test(){
|
||||
return "hello world";
|
||||
}
|
||||
|
||||
@GetMapping("/ai")
|
||||
public String ai(@RequestParam(value = "msg",defaultValue = "你是谁")String msg) {
|
||||
return this.chatClient
|
||||
.prompt()
|
||||
.user(msg)
|
||||
.call()
|
||||
.content();
|
||||
}
|
||||
|
||||
/**
|
||||
* 流式的聊天接口,要注意如果中文有乱码,就是编码得问题,需要添加produces = "text/html;charset=UTF-8
|
||||
* @param msg
|
||||
* @return
|
||||
*/
|
||||
@GetMapping(value = "/ai/stream",produces = "text/html;charset=UTF-8")
|
||||
public Flux<String> aistream(@RequestParam(value = "msg",defaultValue = "你是谁")String msg) {
|
||||
return this.chatClient
|
||||
.prompt()
|
||||
.user(msg)
|
||||
.stream()
|
||||
.content();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
29
src/main/resources/application.yaml
Normal file
29
src/main/resources/application.yaml
Normal file
@ -0,0 +1,29 @@
|
||||
server:
|
||||
port: 8080
|
||||
|
||||
spring:
|
||||
ai:
|
||||
openai:
|
||||
api-key: ollama
|
||||
base-url: http://localhost:11434
|
||||
chat:
|
||||
api-key: ollama
|
||||
base-url: http://localhost:11434
|
||||
options:
|
||||
model: gpt-oss:120b-cloud
|
||||
# model: kimi-k2.5:cloud
|
||||
# model: gemma4:e2b
|
||||
# max-tokens: 10000
|
||||
temperature: 0.7
|
||||
# embedding:
|
||||
# api-key: ollama
|
||||
# base-url: http://localhost:11434
|
||||
|
||||
#spring:
|
||||
# ai:
|
||||
# ollama:
|
||||
# base-url: http://localhost:11434
|
||||
# chat:
|
||||
# options:
|
||||
# model: gemma4:e2b
|
||||
# temperature: 0.7
|
||||
320
src/main/resources/static/css/style.css
Normal file
320
src/main/resources/static/css/style.css
Normal file
@ -0,0 +1,320 @@
|
||||
* {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
:root {
|
||||
--primary: #6366f1;
|
||||
--primary-hover: #4f46e5;
|
||||
--bg-dark: #0f172a;
|
||||
--bg-card: #1e293b;
|
||||
--bg-input: #334155;
|
||||
--text-primary: #f1f5f9;
|
||||
--text-secondary: #94a3b8;
|
||||
--border: #334155;
|
||||
--user-bubble: #6366f1;
|
||||
--ai-bubble: #334155;
|
||||
}
|
||||
|
||||
body {
|
||||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||
background: var(--bg-dark);
|
||||
color: var(--text-primary);
|
||||
min-height: 100vh;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.container {
|
||||
width: 100%;
|
||||
max-width: 800px;
|
||||
height: 100vh;
|
||||
max-height: 900px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
.header {
|
||||
text-align: center;
|
||||
padding: 20px 0;
|
||||
}
|
||||
|
||||
.header h1 {
|
||||
font-size: 28px;
|
||||
font-weight: 700;
|
||||
background: linear-gradient(135deg, #6366f1, #a855f7);
|
||||
-webkit-background-clip: text;
|
||||
-webkit-text-fill-color: transparent;
|
||||
background-clip: text;
|
||||
}
|
||||
|
||||
.subtitle {
|
||||
color: var(--text-secondary);
|
||||
font-size: 14px;
|
||||
margin-top: 4px;
|
||||
}
|
||||
|
||||
.chat-container {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
padding: 20px 0;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 16px;
|
||||
}
|
||||
|
||||
.welcome-message, .message {
|
||||
display: flex;
|
||||
gap: 12px;
|
||||
animation: fadeIn 0.3s ease;
|
||||
}
|
||||
|
||||
@keyframes fadeIn {
|
||||
from { opacity: 0; transform: translateY(10px); }
|
||||
to { opacity: 1; transform: translateY(0); }
|
||||
}
|
||||
|
||||
.message.user {
|
||||
flex-direction: row-reverse;
|
||||
}
|
||||
|
||||
.message-content {
|
||||
max-width: 70%;
|
||||
padding: 12px 16px;
|
||||
border-radius: 16px;
|
||||
line-height: 1.5;
|
||||
font-size: 15px;
|
||||
}
|
||||
|
||||
.ai-avatar {
|
||||
width: 36px;
|
||||
height: 36px;
|
||||
background: linear-gradient(135deg, #6366f1, #a855f7);
|
||||
border-radius: 50%;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.ai-avatar svg {
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.user-avatar {
|
||||
width: 36px;
|
||||
height: 36px;
|
||||
background: var(--bg-input);
|
||||
border-radius: 50%;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.user-avatar svg {
|
||||
width: 18px;
|
||||
height: 18px;
|
||||
color: var(--text-secondary);
|
||||
}
|
||||
|
||||
.message.user .message-content {
|
||||
background: var(--user-bubble);
|
||||
border-bottom-right-radius: 4px;
|
||||
}
|
||||
|
||||
.message.ai .message-content {
|
||||
background: var(--ai-bubble);
|
||||
border-bottom-left-radius: 4px;
|
||||
}
|
||||
|
||||
.input-container {
|
||||
padding: 16px 0;
|
||||
border-top: 1px solid var(--border);
|
||||
}
|
||||
|
||||
.input-wrapper {
|
||||
display: flex;
|
||||
gap: 12px;
|
||||
background: var(--bg-card);
|
||||
border-radius: 24px;
|
||||
padding: 6px;
|
||||
border: 1px solid var(--border);
|
||||
transition: border-color 0.2s;
|
||||
}
|
||||
|
||||
.input-wrapper:focus-within {
|
||||
border-color: var(--primary);
|
||||
}
|
||||
|
||||
#messageInput {
|
||||
flex: 1;
|
||||
background: transparent;
|
||||
border: none;
|
||||
padding: 12px 16px;
|
||||
font-size: 15px;
|
||||
color: var(--text-primary);
|
||||
outline: none;
|
||||
}
|
||||
|
||||
#messageInput::placeholder {
|
||||
color: var(--text-secondary);
|
||||
}
|
||||
|
||||
.send-btn {
|
||||
width: 44px;
|
||||
height: 44px;
|
||||
background: var(--primary);
|
||||
border: none;
|
||||
border-radius: 50%;
|
||||
cursor: pointer;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
transition: background 0.2s, transform 0.15s;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.send-btn:hover {
|
||||
background: var(--primary-hover);
|
||||
}
|
||||
|
||||
.send-btn:active {
|
||||
transform: scale(0.95);
|
||||
}
|
||||
|
||||
.send-btn:disabled {
|
||||
background: var(--bg-input);
|
||||
cursor: not-allowed;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
.send-btn svg {
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.stream-indicator {
|
||||
display: none;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 12px 16px;
|
||||
color: var(--text-secondary);
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.stream-indicator.active {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.dot {
|
||||
width: 6px;
|
||||
height: 6px;
|
||||
background: var(--primary);
|
||||
border-radius: 50%;
|
||||
animation: bounce 1.4s infinite ease-in-out both;
|
||||
}
|
||||
|
||||
.dot:nth-child(1) { animation-delay: -0.32s; }
|
||||
.dot:nth-child(2) { animation-delay: -0.16s; }
|
||||
|
||||
@keyframes bounce {
|
||||
0%, 80%, 100% { transform: scale(0); }
|
||||
40% { transform: scale(1); }
|
||||
}
|
||||
|
||||
::-webkit-scrollbar {
|
||||
width: 6px;
|
||||
}
|
||||
|
||||
::-webkit-scrollbar-track {
|
||||
background: transparent;
|
||||
}
|
||||
|
||||
::-webkit-scrollbar-thumb {
|
||||
background: var(--border);
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
.message-content {
|
||||
word-wrap: break-word;
|
||||
}
|
||||
|
||||
.message-content pre {
|
||||
background: #0d1117;
|
||||
border-radius: 8px;
|
||||
padding: 12px;
|
||||
margin: 8px 0;
|
||||
overflow-x: auto;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.message-content code {
|
||||
background: #0d1117;
|
||||
padding: 2px 6px;
|
||||
border-radius: 4px;
|
||||
font-family: 'Fira Code', Consolas, monospace;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.message-content pre code {
|
||||
background: transparent;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.message-content p {
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
|
||||
.message-content p:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.message-content ul, .message-content ol {
|
||||
padding-left: 20px;
|
||||
margin: 8px 0;
|
||||
}
|
||||
|
||||
.message-content blockquote {
|
||||
border-left: 3px solid var(--primary);
|
||||
padding-left: 12px;
|
||||
margin: 8px 0;
|
||||
color: var(--text-secondary);
|
||||
}
|
||||
|
||||
.message-content a {
|
||||
color: #818cf8;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.message-content a:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
.message-content h1, .message-content h2, .message-content h3,
|
||||
.message-content h4, .message-content h5, .message-content h6 {
|
||||
margin: 12px 0 8px;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.message-content table {
|
||||
border-collapse: collapse;
|
||||
margin: 8px 0;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.message-content th, .message-content td {
|
||||
border: 1px solid var(--border);
|
||||
padding: 6px 10px;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.message-content th {
|
||||
background: var(--bg-input);
|
||||
}
|
||||
57
src/main/resources/static/index.html
Normal file
57
src/main/resources/static/index.html
Normal file
@ -0,0 +1,57 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh-CN">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>Spring AI Chat</title>
|
||||
<link rel="stylesheet" href="css/style.css">
|
||||
<script src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
|
||||
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/styles/github-dark.min.css">
|
||||
<script src="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.9.0/build/highlight.min.js"></script>
|
||||
</head>
|
||||
<script>
|
||||
marked.setOptions({
|
||||
highlight: function(code, lang) {
|
||||
if (lang && hljs.getLanguage(lang)) {
|
||||
return hljs.highlight(code, { language: lang }).value;
|
||||
}
|
||||
return hljs.highlightAuto(code).value;
|
||||
}
|
||||
});
|
||||
</script>
|
||||
<body>
|
||||
<div class="container">
|
||||
<header class="header">
|
||||
<h1>Spring AI Chat</h1>
|
||||
<p class="subtitle">基于 Spring AI 的智能对话助手</p>
|
||||
</header>
|
||||
|
||||
<div class="chat-container" id="chatContainer">
|
||||
</div>
|
||||
|
||||
<div class="input-container">
|
||||
<div class="input-wrapper">
|
||||
<input
|
||||
type="text"
|
||||
id="messageInput"
|
||||
placeholder="输入您的问题..."
|
||||
autocomplete="off"
|
||||
>
|
||||
<button id="sendBtn" class="send-btn">
|
||||
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||
<path d="M22 2L11 13M22 2l-7 20-4-9-9-4 20-7z"/>
|
||||
</svg>
|
||||
</button>
|
||||
</div>
|
||||
<div class="stream-indicator" id="streamIndicator">
|
||||
<span class="dot"></span>
|
||||
<span class="dot"></span>
|
||||
<span class="dot"></span>
|
||||
AI 正在思考中...
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script src="js/app.js"></script>
|
||||
</body>
|
||||
</html>
|
||||
86
src/main/resources/static/js/app.js
Normal file
86
src/main/resources/static/js/app.js
Normal file
@ -0,0 +1,86 @@
|
||||
const chatContainer = document.getElementById('chatContainer');
|
||||
const messageInput = document.getElementById('messageInput');
|
||||
const sendBtn = document.getElementById('sendBtn');
|
||||
const streamIndicator = document.getElementById('streamIndicator');
|
||||
|
||||
let isStreaming = false;
|
||||
|
||||
messageInput.addEventListener('keypress', (e) => {
|
||||
if (e.key === 'Enter' && !e.shiftKey) {
|
||||
e.preventDefault();
|
||||
sendMessage();
|
||||
}
|
||||
});
|
||||
|
||||
sendBtn.addEventListener('click', sendMessage);
|
||||
|
||||
async function sendMessage() {
|
||||
const message = messageInput.value.trim();
|
||||
if (!message || isStreaming) return;
|
||||
|
||||
addMessage(message, 'user');
|
||||
messageInput.value = '';
|
||||
isStreaming = true;
|
||||
sendBtn.disabled = true;
|
||||
streamIndicator.classList.add('active');
|
||||
|
||||
const aiMessageEl = addMessage('', 'ai');
|
||||
|
||||
try {
|
||||
const response = await fetch(`/api/chat/ai/stream?msg=${encodeURIComponent(message)}`);
|
||||
|
||||
const reader = response.body.getReader();
|
||||
const decoder = new TextDecoder();
|
||||
let result = '';
|
||||
|
||||
while (true) {
|
||||
const { done, value } = await reader.read();
|
||||
if (done) break;
|
||||
const chunk = decoder.decode(value);
|
||||
result += chunk;
|
||||
aiMessageEl.querySelector('.message-content').innerHTML = marked.parse(result);
|
||||
chatContainer.scrollTop = chatContainer.scrollHeight;
|
||||
}
|
||||
} catch (error) {
|
||||
aiMessageEl.querySelector('.message-content').textContent = '抱歉,发生了错误: ' + error.message;
|
||||
} finally {
|
||||
isStreaming = false;
|
||||
sendBtn.disabled = false;
|
||||
streamIndicator.classList.remove('active');
|
||||
}
|
||||
}
|
||||
|
||||
function addMessage(text, type) {
|
||||
const messageDiv = document.createElement('div');
|
||||
messageDiv.className = `message ${type}`;
|
||||
|
||||
const avatar = type === 'ai'
|
||||
? `<div class="ai-avatar">
|
||||
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||
<path d="M12 2a10 10 0 1 0 10 10A10 10 0 0 0 12 2zm0 18a8 8 0 1 1 8-8 8 8 0 0 1-8 8z"/>
|
||||
<circle cx="12" cy="12" r="3"/>
|
||||
<path d="M12 2v4m0 12v4M2 12h4m12 0h4"/>
|
||||
</svg>
|
||||
</div>`
|
||||
: `<div class="user-avatar">
|
||||
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||
<path d="M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2"/>
|
||||
<circle cx="12" cy="7" r="4"/>
|
||||
</svg>
|
||||
</div>`;
|
||||
|
||||
messageDiv.innerHTML = `
|
||||
${avatar}
|
||||
<div class="message-content"><p>${escapeHtml(text)}</p></div>
|
||||
`;
|
||||
|
||||
chatContainer.appendChild(messageDiv);
|
||||
chatContainer.scrollTop = chatContainer.scrollHeight;
|
||||
return messageDiv;
|
||||
}
|
||||
|
||||
function escapeHtml(text) {
|
||||
const div = document.createElement('div');
|
||||
div.textContent = text;
|
||||
return div.innerHTML;
|
||||
}
|
||||
Loading…
Reference in New Issue
Block a user