首页 > 编程语言 >Python调用星火认知大模型API(流式传输)

Python调用星火认知大模型API(流式传输)

时间:2024-12-21 15:31:53浏览次数:8  
标签:Python self 流式 url API import spark data

Python调用星火认知大模型API流式传输

前言

最近对大模型比较感兴趣,想要调用大模型的API,实现AI助手那种问答效果,正好讯飞星火有免费API额度,我就用Python来试试,还是很有趣的。接下来我就介绍一下使用方法,如果感兴趣可以直接看讯飞星火官方文档

1. 获取API认证信息

要想调用星火的API,首先在官网注册一下账号,再在选择需要的模型,购买免费的API,这里我看了下,免费的有一年期限,还是够用的。
链接:https://console.xfyun.cn/services/bm35
在这里插入图片描述
选择应用,没有就点 + 号创建一个,选择免费的,下单即可。
在这里插入图片描述
购买完后,记下前面看到的认证信息,后面填入代码参数中。

2. 快速调用集成星火认知大模型

先来个简单案例测试一下认证信息是否可用

from sparkai.llm.llm import ChatSparkLLM, ChunkPrintHandler
from sparkai.core.messages import ChatMessage

# 星火认知大模型Spark Max的URL值,其他版本大模型URL值请前往文档(https://www.xfyun.cn/doc/spark/Web.html)查看
SPARKAI_URL = 'wss://spark-api.xf-yun.com/v3.5/chat'
# 星火认知大模型调用秘钥信息,请前往讯飞开放平台控制台(https://console.xfyun.cn/services/bm35)查看
SPARKAI_APP_ID = ""  # 填写控制台中获取的 APPID 信息
SPARKAI_API_SECRET = ""  # 填写控制台中获取的 APISecret 信息
SPARKAI_API_KEY = ""  # 填写控制台中获取的 APIKey 信息
# 星火认知大模型Spark Max的domain值,其他版本大模型domain值请前往文档(https://www.xfyun.cn/doc/spark/Web.html)查看
SPARKAI_DOMAIN = 'generalv3.5'

if __name__ == '__main__':
    spark = ChatSparkLLM(
        spark_api_url=SPARKAI_URL,
        spark_app_id=SPARKAI_APP_ID,
        spark_api_key=SPARKAI_API_KEY,
        spark_api_secret=SPARKAI_API_SECRET,
        spark_llm_domain=SPARKAI_DOMAIN,
        streaming=False,
    )
    messages = [ChatMessage(
        role="user",
        content='你好呀'
    )]
    handler = ChunkPrintHandler()
    a = spark.generate([messages], callbacks=[handler])
    for chunk in a:
        print(chunk)

在这里插入图片描述
可以看到,API调用成功,但这个只是简单的案例,不太能够满足我们需要的场景。接下来我将介绍流式回复形式。

3. 实现AI流式回复

虽然官网也有流式回复代码,不过用的是websocket,我不太会,所以我查了下其他方法,通过spark.stream + yield 也可以实现流式返回,为了跟好的实现前后端联调的效果,我用flask搭建一个简易的服务,前端则用request模拟用户请求。

1)SparkApi.py

为了方便使用,我将调用API的方法单独创建了一个类

from sparkai.llm.llm import ChatSparkLLM, ChunkPrintHandler
from sparkai.core.messages import ChatMessage


class SparkApi(object):
    # 初始化
    def __init__(self, APP_ID, APIKey, APISecret, Spark_url, Spark_domain):
        self.APP_ID = APP_ID
        self.APIKey = APIKey
        self.APISecret = APISecret
        self.Spark_url = Spark_url
        self.Spark_domain = Spark_domain
        self.spark = None
        self.messages = []
        self.assistant_response = ""

    def init_spark(self, is_stream):
        self.spark = ChatSparkLLM(
            spark_api_url=self.Spark_url,
            spark_app_id=self.APP_ID,
            spark_api_key=self.APIKey,
            spark_api_secret=self.APISecret,
            spark_llm_domain=self.Spark_domain,
            streaming=is_stream,
        )

    # 获取ai回答 文生文
    def stream_generator(self, user_message):
        self.messages.append(ChatMessage(role="user", content=user_message))
        # 流式获取ai回复
        response = self.spark.stream(self.messages)
        for res in response:
            self.assistant_response += res.content
            # 回复结束
            if res.additional_kwargs:
                self.messages.append(ChatMessage(role="assistant", content=self.assistant_response))
                self.assistant_response = ""
                print(self.messages)
            yield res.content.encode('utf-8')

2)SparkApiMain.py

from Tools.scripts.make_ctype import method
from aiohttp.web_routedef import route
from flask import Flask, request, Response, jsonify
from SparkApi import SparkApi
# from SparkPicture import SparkPicture

app = Flask(__name__)

# 星火认知大模型Spark Max的URL值,其他版本大模型URL值请前往文档(https://www.xfyun.cn/doc/spark/Web.html)查看
SPARKAI_URL = 'wss://spark-api.xf-yun.com/v3.5/chat'
# SPARKAI_URL = 'wss://spark-api.xf-yun.com/v4.0/chat'
# 星火认知大模型调用秘钥信息,请前往讯飞开放平台控制台(https://console.xfyun.cn/services/bm35)查看
SPARKAI_APP_ID = ''
SPARKAI_API_SECRET = ''
SPARKAI_API_KEY = ''
# 星火认知大模型Spark Max的domain值,其他版本大模型domain值请前往文档(https://www.xfyun.cn/doc/spark/Web.html)查看
SPARKAI_DOMAIN = 'generalv3.5'
# SPARKAI_DOMAIN = '4.0Ultra'

# 注释的这块是绘图的API,由于我绘图和ai聊天创建了两个应用,所以得用两套认证信息,如果你创建在一个应用就不用向我这样
# ai 绘画
# APP_ID = ''
# API_Secret = ''
# API_KEY = ''

CHAT = {
    "domain": 'general',
    "temperature": 0.5,
    "max_tokens": 4096,
    "width": 512,
    "height": 512
}

@app.route('/stream', methods=['POST'])
def main():
    user_data = request.json.get('question', '')  # 假设前端发送的 JSON 数据中包含 'message' 键
    spark.init_spark(True)
    return Response(spark.stream_generator(user_data), mimetype='text/plain')

# 这块是绘图的接口
# @app.route('/picture', methods=['POST'])
# def gener_picture():
#     desc = request.json.get('desc', '')  # 假设前端发送的 JSON 数据中包含 'message' 键
#     res = spark_picture.generator_picture(desc)
#     mock_image_url = spark_picture.parser_Message(res)
#     return jsonify({"image_url": mock_image_url})


if __name__ == '__main__':
    spark = SparkApi(SPARKAI_APP_ID, SPARKAI_API_KEY, SPARKAI_API_SECRET, SPARKAI_URL, SPARKAI_DOMAIN)
    # 下面是绘图的API,如果需要可以把下面绘图的工具类加进去,SparkPicture类代码我放后面了
    # spark_picture = SparkPicture(app_id=APP_ID, api_key=API_KEY, api_secret=API_Secret, chat=CHAT)
    app.run(host='0.0.0.0', port=5000, debug=True)

3)SparkPicture.py

因为我将聊天和绘图放到一起了,所以这里也把绘图的代码放上来,不过我把有关绘图的都注释了,可以先调试ai聊天的,有兴趣可以试一下绘图。

# encoding: UTF-8
import time
import requests
from datetime import datetime
from wsgiref.handlers import format_date_time
from time import mktime
import hashlib
import base64
import hmac
from urllib.parse import urlencode
import json
from PIL import Image
from io import BytesIO


class AssembleHeaderException(Exception):
    def __init__(self, msg):
        self.message = msg


class Url:
    def __init__(self, host, path, schema):
        self.host = host
        self.path = path
        self.schema = schema

class SparkPicture:
    def __init__(self,app_id,api_key,api_secret, chat):
        self.app_id = app_id
        self.api_key = api_key
        self.api_secret = api_secret
        self.chat = chat

    # calculate sha256 and encode to base64
    def sha256base64(self, data):
        sha256 = hashlib.sha256()
        sha256.update(data)
        digest = base64.b64encode(sha256.digest()).decode(encoding='utf-8')
        return digest


    def parse_url(self,requset_url):
        stidx = requset_url.index("://")
        host = requset_url[stidx + 3:]
        schema = requset_url[:stidx + 3]
        edidx = host.index("/")
        if edidx <= 0:
            raise AssembleHeaderException("invalid request url:" + requset_url)
        path = host[edidx:]
        host = host[:edidx]
        u = Url(host, path, schema)
        return u


    # 生成鉴权url
    def assemble_ws_auth_url(self,requset_url, method="GET"):
        u = self.parse_url(requset_url)
        host = u.host
        path = u.path
        now = datetime.now()
        date = format_date_time(mktime(now.timetuple()))
        # print(date)
        # date = "Thu, 12 Dec 2019 01:57:27 GMT"
        signature_origin = "host: {}\ndate: {}\n{} {} HTTP/1.1".format(host, date, method, path)
        # print(signature_origin)
        signature_sha = hmac.new(self.api_secret.encode('utf-8'), signature_origin.encode('utf-8'),
                                 digestmod=hashlib.sha256).digest()
        signature_sha = base64.b64encode(signature_sha).decode(encoding='utf-8')
        authorization_origin = "api_key=\"%s\", algorithm=\"%s\", headers=\"%s\", signature=\"%s\"" % (
            self.api_key, "hmac-sha256", "host date request-line", signature_sha)
        authorization = base64.b64encode(authorization_origin.encode('utf-8')).decode(encoding='utf-8')
        # print(authorization_origin)
        values = {
            "host": host,
            "date": date,
            "authorization": authorization
        }

        return requset_url + "?" + urlencode(values)

    # 生成请求body体
    def get_body(self,text):
        body= {
            "header": {
                "app_id": self.app_id,
                "uid":"123456789"
            },
            "parameter": {
                "chat": self.chat
            },
            "payload": {
                "message":{
                    "text":[
                        {
                            "role":"user",
                            "content":text
                        }
                    ]
                }
            }
        }
        return body

    # 发起请求并返回结果
    def generator_picture(self,text):
        host = 'http://spark-api.cn-huabei-1.xf-yun.com/v2.1/tti'
        url = self.assemble_ws_auth_url(host,method='POST')
        content = self.get_body(text)
        print(time.time())
        response = requests.post(url,json=content,headers={'content-type': "application/json"}).text
        print(time.time())
        return response

    #将base64 的图片数据存在本地
    def base64_to_image(self,base64_data, save_path):
        # 解码base64数据
        img_data = base64.b64decode(base64_data)
        # 将解码后的数据转换为图片
        img = Image.open(BytesIO(img_data))
        # 保存图片到本地
        img.save(save_path)


    # 解析并保存到指定位置
    def parser_Message(self,message):
        data = json.loads(message)
        # print("data" + str(message))
        code = data['header']['code']
        if code != 0:
            print(f'请求错误: {code}, {data}')
        else:
            text = data["payload"]["choices"]["text"]
            imageContent = text[0]
            # if('image' == imageContent["content_type"]):
            imageBase = imageContent["content"]
            imageName = data['header']['sid']
            savePath = f"D://chen2024//Pycharm//AIGC_OPENAI//讯飞薪火//image//{imageName}.jpg"
            # print("图片的imageBase:" + imageBase)
            self.base64_to_image(imageBase,savePath)
            print("图片保存路径:" + savePath)
            return savePath

4)spider.py

这块是模拟前端向后端发送请求的,可以更好观察流式回复的效果。

import json
import requests
from flask import jsonify


def get_stream_response(response):
    # 流式接收response
    temp_data = ''
    print("AI:", end="")
    for chunk in response.iter_content(chunk_size=1024):
        temp_data += chunk.decode('utf-8')
        print(chunk.decode('utf-8'), end="")
    # print(rec_data_list)
    # print("----------------------------")
    # print(temp_data)
    print()
    return temp_data

# 获取文生文接口
def stream_upload(url, question):
    headers = {'Content-Type': 'application/json'}
    data = json.dumps({"question": question})  # 使用固定的键名"question"
    # 流式接收response
    response = requests.post(url,headers=headers , data=data, stream=True)
    final_response = get_stream_response(response)
    return final_response

# 获取文生图接口
def picture_upload(url ,desc):
    headers = {'Content-Type': 'application/json'}
    data = json.dumps({"desc": desc})  # 使用固定的键名"question"
    # 流式接收response
    image_path = requests.post(url, headers=headers, data=data).json()
    print(f"图片路径为:{image_path['image_url']}")
    return image_path


def back_to(is_back):
    if is_back == "back":
        return True
    return False


if __name__ == '__main__':
    url = 'http://127.0.0.1:5000/'
    while True:
        flag = input("1.文生文 2.文生图 请选择:")
        # back 返回上一级
        while flag != "back":
            if int(flag) == 1:
                question = input("我:")
                if back_to(question):
                    flag = "back"
                    continue
                response = stream_upload(url+'stream', question)
            elif int(flag) == 2:
                desc = input("请描绘图片:")
                if back_to(desc):
                    flag = "back"
                    continue
                resp = picture_upload(url+'picture', desc)

5)开始测试

先执行 SparkApiMain.py ,有active就应该是成功了,往上滑还可以看到 Running on http://127.0.0.1:5000
在这里插入图片描述
再执行 spider.py 程序
在这里插入图片描述
现在可以看到两个程序都执行了,我们可以在spider的运行框中输入问题,同时可以看到ai的回复,注意,先输入1,选择文生文
视频效果链接:https://live.csdn.net/v/439539
在这里插入图片描述
到这里就已经实现了前后端流式传输得效果了,讯飞星火除了绘图和聊天,还有其他的API,有兴趣都可以试试。

参考文档

标签:Python,self,流式,url,API,import,spark,data
From: https://blog.csdn.net/m0_70405779/article/details/144570214

相关文章

  • 如何利用Python爬虫获取商品历史价格信息
    在电商的世界里,商品价格的波动对消费者和商家都具有重要意义。了解商品的历史价格可以帮助消费者做出更明智的购买决策,同时也为商家提供了市场趋势的重要信息。本文将详细介绍如何使用Python爬虫技术来获取商品的历史价格信息,并提供一个详细的代码示例。一、项目准备在开始......
  • 速通Python 第一节
    一、引言1.Python的来源?吉多·范罗苏姆(GuidovanRossum)是一个荷兰程序员(人称龟叔,名字前三个字母是Gui),龟叔在1989年圣诞节的时候(当时33岁),因为在家里待着无聊,为了打发时间,开始了Python的开发, 第一个正式版本发布于1991......
  • selenium+python学习笔记-1(未完待续)
    1.使用selenium前首先需要配置好版本一致的谷歌浏览器webdriver,下载地址旧版本:https://chromedriver.storage.googleapis.com/index.html;新版本:https://googlechromelabs.github.io/chrome-for-testing/;(1)首先win+r,打开cmd,输入chromedriver--version,查看当前chomedriver版本,检......
  • django基于Python的校园个人闲置物品换购平台
    文章目录程序资料获取一、项目技术二、项目内容和项目介绍三、核心代码四、效果图五、资料获取程序资料获取......
  • 计算机毕设项目96xeny5v+python基于python+django的医疗预约与诊断系统,计算机毕业生可
    python基于python+django的医疗预约与诊断系统摘要随着医疗资源的日益紧张,医疗预约与诊断系统的出现成为解决这一问题的关键。该系统旨在提高医疗服务效率,减少患者等待时间,使医疗资源得到更合理的分配。系统利用先进的信息技术,实现了患者在线预约、医生诊断、检查结果的实......
  • BBU-Python期末考试复习题目总结
    临近期末,抽个时间把BBU-python期末考试会考的题型(原题?)哈哈总结一下,python考试是比较简单的,题型分为选择题,判断题,填空题,程序阅读题,编程题,我在最后还加了一点重要的书上知识点,放到我的个人bolg上供大家参考,祝考试高分通过————一.选择题选择题大概率为三次学习通作业上的......
  • 【恶意软件检测-CCFA文章】SDAC:使用基于语义距离的 API 集群进行 Android 恶意软件检
    ​SDAC:使用基于语义距离的API集群进行Android恶意软件检测的慢老化解决方案摘要提出了一种名为SDAC的新型缓慢老化解决方案,用于解决Android恶意软件检测中的模型老化问题,该问题是由于在恶意软件检测过程中未能适应Android规范的变化所致。与现有解决方案中的检测模......
  • python数据分析:介绍pandas库的数据类型Series和DataFrame
    安装pandaspipinstallpandas-ihttps://mirrors.aliyun.com/pypi/simple/使用pandas直接导入即可importpandasaspdpandas的数据结构pandas提供了两种主要的数据结构:Series和DataFrame,类似于python提供list列表,dict字典,tuple元组等数据类型用于存储数据。1.Se......
  • 用Python实现经典游戏——飞机大战(代码附在文末)
    引言在Python中,Pygame是一个非常流行的游戏开发库,适合初学者入门。本文将带你一步步实现一个经典的“飞机大战”游戏,并逐步优化代码,增加关卡系统、难度递增等功能。通过本文,你将学习到如何使用Pygame创建游戏窗口、处理用户输入、实现精灵类、碰撞检测以及游戏逻辑的优化。1......
  • python毕设 家政服务管理系统程序+论文
    本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。系统程序文件列表开题报告内容一、选题背景关于家政服务管理系统的研究,现有研究多以传统的管理模式或特定技术框架下(如Javaweb)的实现为主 [1] 。专门针对Python语言来构建家政......