File size: 4,528 Bytes
2d029b1
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
import os
import json
import gradio as gr
import requests
from dotenv import load_dotenv

# 加载环境变量
load_dotenv()
API_KEY = os.getenv("DEEPSEEK_API_KEY")
API_KEY = "sk-0d040569ee2b4932a807d56a5b12ac05"
API_URL = "https://api.deepseek.com/v1/chat/completions"

def build_messages(query: str, history: list):
    """构造对话历史"""
    messages = [{"role": "system", "content": "你是一个有帮助的助手"}]
    for user_msg, bot_msg in history:
        messages.extend([
            {"role": "user", "content": user_msg},
            {"role": "assistant", "content": bot_msg}
        ])
    messages.append({"role": "user", "content": query})
    return messages

def stream_response(query: str, history: list):
    """流式响应生成器"""
    headers = {
        "Authorization": f"Bearer {API_KEY}",
        "Content-Type": "application/json"
    }

    try:
        response = requests.post(
            API_URL,
            headers=headers,
            json={
                "model": "deepseek-chat",
                "messages": build_messages(query, history),
                "temperature": 0.7,
                "stream": True
            },
            stream=True,
            timeout=60
        )
        
        # 先检查响应状态再处理内容
        if response.status_code != 200:
            error_msg = f"API 返回错误状态码: {response.status_code} - {response.text}"
            print(error_msg)
            yield error_msg
            return

        partial_message = ""
        for chunk in response.iter_lines():
            # 过滤空行和 keep-alive 消息
            if not chunk or b'[DONE]' in chunk:
                continue

            try:
                # 调试输出原始数据
                print("原始 chunk:", chunk)
                
                decoded = chunk.decode('utf-8').strip()
                # 处理可能的多个 data: 前缀
                if decoded.startswith('data:'):
                    json_str = decoded[5:].strip()
                else:
                    json_str = decoded
                
                # 验证 JSON 有效性
                if not json_str.startswith('{'):
                    continue
                
                data = json.loads(json_str)
                
                if content := data['choices'][0]['delta'].get('content'):
                    partial_message += content
                    yield partial_message

            except json.JSONDecodeError as e:
                print(f"JSON 解析失败: {e} | 原始数据: {decoded}")
                continue
            except Exception as e:
                print(f"处理 chunk 时发生错误: {str(e)}")
                continue

    except Exception as e:
        yield f"⚠️ 请求失败:{str(e)}"

def stream_response_back(query: str, history: list):
    """流式响应生成器"""
    headers = {
        "Authorization": f"Bearer {API_KEY}",
        "Content-Type": "application/json"
    }

    print(query)
    print(history)

    try:
        response = requests.post(
            API_URL,
            headers=headers,
            json={
                "model": "deepseek-chat",
                "messages": build_messages(query, history),
                "temperature": 0.7,
                "stream": True
            },
            stream=True,
            timeout=60
        )
        response.raise_for_status()
        
        print(response)

        partial_message = ""
        for chunk in response.iter_lines():
            if chunk:
                # 处理流式数据格式
                decoded = chunk.decode('utf-8').strip()
                if decoded.startswith('data:'):
                    data = json.loads(decoded[5:])
                    if content := data['choices'][0]['delta'].get('content'):
                        partial_message += content
                        yield partial_message

    except Exception as e:
        yield f"⚠️ 请求失败:{str(e)}"

# 创建带打字机效果的聊天界面
demo = gr.ChatInterface(
    fn=stream_response,
    title="DeepSeek 智能助手",
    description="输入消息开始对话(支持流式打字效果)",
    theme="soft",
    examples=["你好!", "如何学习AI?", "写一首关于春天的诗"],
    cache_examples=False,
    # retry_btn=None,
    # undo_btn=None,
    # clear_btn="清空历史",
    stop_btn="停止生成",
)

if __name__ == "__main__":
    demo.launch(server_name="0.0.0.0", server_port=7860)