| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117 | import websocketimport datetimeimport hashlibimport base64import hmacimport jsonfrom urllib.parse import urlencodeimport sslfrom wsgiref.handlers import format_date_timefrom datetime import datetimefrom time import mktimeimport threading"""调用讯飞模型 文字转语音 """class WsParamSynthesize:    def __init__(self, APPID, APIKey, APISecret, Text, AudioName="demo"):        self.APPID = APPID        self.APIKey = APIKey        self.APISecret = APISecret        self.Text = Text        self.AudioType = "mp3"        self.AudioName = AudioName        # 初始化其他需要的属性        self.CommonArgs = {"app_id": self.APPID}        if self.AudioType == "mp3":            self.BusinessArgs = {"aue": "lame", "auf": "audio/L16;rate=8000", "vcn": "xiaoyan", "tte": "utf8",                                 "sfl": 1}        else:            self.BusinessArgs = {"aue": "raw", "auf": "audio/L16;rate=8000", "vcn": "xiaoyan", "tte": "utf8"}        self.Data = {"status": 2, "text": str(base64.b64encode(self.Text.encode('utf-8')), "UTF8")}    def create_url(self):        url = 'wss://tts-api.xfyun.cn/v2/tts'        # 生成RFC1123格式的时间戳        now = datetime.now()        date = format_date_time(mktime(now.timetuple()))        # 拼接字符串        signature_origin = "host: " + "ws-api.xfyun.cn" + "\n"        signature_origin += "date: " + date + "\n"        signature_origin += "GET " + "/v2/tts " + "HTTP/1.1"        # 进行hmac-sha256进行加密        signature_sha = hmac.new(self.APISecret.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.APIKey, "hmac-sha256", "host date request-line", signature_sha)        authorization = base64.b64encode(authorization_origin.encode('utf-8')).decode(encoding='utf-8')        # 将请求的鉴权参数组合为字典        v = {            "authorization": authorization,            "date": date,            "host": "ws-api.xfyun.cn"        }        # 拼接鉴权参数,生成url        url = url + '?' + urlencode(v)        return url    def on_message(self, ws, message):        try:            message = json.loads(message)            code = message["code"]            sid = message["sid"]            audio = message["data"]["audio"]            if code != 0:                errMsg = message["message"]                print(f"Error: {errMsg}, code: {code}")                return None            audio = base64.b64decode(audio)            status = message["data"]["status"]            if status == 2:                print("WebSocket is closed")                ws.close()            if code != 0:                errMsg = message["message"]                print("sid:%s call error:%s code is:%s" % (sid, errMsg, code))            else:                with open(f'static/demo_files/{self.AudioName}.{self.AudioType}', 'ab') as f:                    f.write(audio)        except Exception as e:            print("Exception while parsing message:", e)            return None    # on_error和on_close方法类似地修改,可以访问类实例的属性    def on_error(self, error):        print("### error:", error)    # 收到websocket关闭的处理    def on_close(self):        print("### closed ###")    def on_open(self, ws):        def run(*args):            d = {"common": self.CommonArgs, "business": self.BusinessArgs, "data": self.Data}            d = json.dumps(d)            ws.send(d)        threading.Thread(target=run).start()    def start(self):        websocket.enableTrace(False)        self.ws = websocket.WebSocketApp(self.create_url(),                                         on_message=lambda ws, msg: self.on_message(ws, msg),                                         on_error=lambda msg: self.on_error(msg),                                         on_close=self.on_close,                                         on_open=lambda ws: self.on_open(ws))  # 使用 lambda 来确保 ws 参数传递        self.ws.run_forever(sslopt={"cert_reqs": ssl.CERT_NONE})
 |