uniapp开发小程序接入阿里云TTS语音合成(RESTful API)

news/2024/7/20 1:35:15 标签: uni-app, 小程序, tts, 阿里云tts, 语音合成
流程
  1. 首先小程序后台配置白名单
    1.1 路径:开发-开发管理-开发设置-服务器域名-request合法域名
    1.2 request合法域名参数:
          https://nls-meta.cn-shanghai.aliyuncs.com
          https://nls-gateway-cn-shanghai.aliyuncs.com
  2. 引入alitts.js
  3. 页面使用
    3.1 只需替换AccessKeyID、AccessKeySecret 、appkey三个参数即可直接使用
AccessKeyID、AccessKeySecret 、appkey获取地址:
  1. 阿里云RESTful API对接文档
  2. 阿里云TTS管理平台创建项目获取appkey
  3. 阿里云获取AccessKeyID和AccessKeySecret
ttsjs_token_16">static/js/alitts.js 用于获取阿里云动态token
// 这个东西我都没执行yarn add crypto竟然能用,可能另一个项目安装了全局共享了,如报错找不到,执行一下yarn add crypto
import crypto from 'crypto'

export class AccessToken {
    static encodeText(text) {
        let encodedText = encodeURIComponent(text);
        return encodedText.replace('+', '%20').replace('*', '%2A').replace('~', '%7E');
    }

    static encodeDict(dict) {
        let keys = Object.keys(dict).sort();
        return keys.map(key => `${this.encodeText(key)}=${this.encodeText(dict[key])}`).join('&');
    }

    static async createToken(accessKeyId, accessKeySecret) {
        const parameters = {
            AccessKeyId: accessKeyId,
            Action: 'CreateToken',
            Format: 'JSON',
            RegionId: 'cn-shanghai',
            SignatureMethod: 'HMAC-SHA1',
            SignatureNonce: uuidv4(),
            SignatureVersion: '1.0',
            Timestamp: new Date().toISOString(),
            Version: '2019-02-28'
        };

        const queryString = this.encodeDict(parameters);
        console.log('Normalized request string:', queryString);

        const stringToSign = `GET&${this.encodeText('/')}&${this.encodeText(queryString)}`;
        console.log('String to sign:', stringToSign);

        const hmac = crypto.createHmac('sha1', `${accessKeySecret}&`);
        hmac.update(stringToSign);
        const signature = hmac.digest('base64');
        console.log('Signature:', signature);

        const encodedSignature = this.encodeText(signature);
        console.log('URL-encoded signature:', encodedSignature);

        const fullUrl = `https://nls-meta.cn-shanghai.aliyuncs.com/?Signature=${encodedSignature}&${queryString}`;
        console.log('URL:', fullUrl);

        let resData =  await new Promise((resolve, reject) => {
            uni.request({
                url: fullUrl,
                method: 'GET',
                success: res => {
                    const data = res.data
                    resolve({
                        token: data.Token.Id,
                        expireTime: data.Token.ExpireTime
                    })
                },
                fail: error => {
                    console.log(error)
                    reject(error)
                }
            })
        })
         console.log('res',resData)
         if(resData){
            return resData
         }
        // Using fetch for HTTP request
        // const response = await fetch(fullUrl);
        // if (response.ok) {
        //     const jsonResponse = await response.json();
        //     if (jsonResponse.Token) {
        //         return {
        //             token: jsonResponse.Token.Id,
        //             expireTime: jsonResponse.Token.ExpireTime
        //         };
        //     }
        // }
        // console.error(await response.text());
        return {
            token: null,
            expireTime: null
        };
    }
}

// Sample UUIDv4 function, or you could use a library like `uuid`
function uuidv4() {
    return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
        var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
        return v.toString(16);
    });
}
使用
// 阿里云动态token获取函数
import { AccessToken } from "@/static/js/alitts"
// 阿里云动态token
const aliToken = null
// 需要在阿里云管理平台获取
const AccessKeyID = '你的AccessKeyID '
const AccessKeySecret = '你的AccessKeySecret '
// 需要在阿里云tts管理平台创建项目
const appkey = '你的阿里云后台创建项目的key'

export default {
    name: "tts",
    data() {
        return {
        	isPlay:false,
        	// tts播放实例
            ttsAudio: null,
        }
    },
    onUnload() {
        if(this.ttsAudio){
            this.ttsAudio.stop()
            this.ttsAudio.destroy()
        }
    },
    async onLoad(val) {
		// 获取阿里云动态token,tts需要此参数
		AccessToken.createToken(AccessKeyID, AccessKeySecret).then(({ token, expireTime }) => {
		   console.log('阿里云token:', token, 'Expire Time:', expireTime);
		   aliToken = token
		});

		// 模拟调用
		setTimeout(() => {
			this.tts('刘斩仙明天要去江苏,晚上回来又约了朋友撸串,忙死了')
		},5000)
	},
	methods: {
		/**
		* 文字转语音
		* @param {string} text 
		*/
		tts(text) {
			uni.request({
			      url:'https://nls-gateway-cn-shanghai.aliyuncs.com/stream/v1/tts',
			      method:'POST',
			      // header:{
			      //     "Content-Type": "application/json"
			      // },
			      data:{
			          appkey: appkey,
			          token: aliToken,
			          text: text,
			          format: 'mp3',
			          sample_rate: 16000,
			          volume: 100
			      },
			      // dataType:'tts',
			      responseType:'arraybuffer',
			      success:ttsRes => {
			           console.log('阿里云:')
			          // 语音
			          const audio = uni.createInnerAudioContext();
			          // 设置不遵循静音开关播放,否则ios无法外音播放
			          audio.obeyMuteSwitch = false
			          uni.setInnerAudioOption({
			              obeyMuteSwitch: false
			          })
			          // 临时路径-此处必须加时间戳或者随机数,否则同样临时路径无法覆盖,小程序bug
			          const ttsPath = `${wx.env.USER_DATA_PATH}/tts${new Date().getTime()}.mp3`
			          if(this.ttsAudio){
			              this.ttsAudio.stop()
			              this.ttsAudio.destroy()
			          }
			          this.ttsAudio = audio
			          // 将 arrayBuffer 写入临时文件
			          const fs = uni.getFileSystemManager()
			          try {
			              const writeRes = fs.writeFileSync(ttsPath, ttsRes.data, "binary")
			              console.log('writeRes',writeRes)
			          } catch(e) {
			              console.error(e)
			          }
			          
			          audio.src = ttsPath
			          audio.autoplay = false;
			          audio.onError((res) => {
			              console.error('音频播放出错', res);
			          });
			          // 监听播放完成
			          audio.onEnded(() => {
			              console.log('音频播放结束');
			              this.isPlay = false
			              // 播放完成后删除临时文件,此处虽然设置同步删除即使执行成功,文件也不会立即删除,还是能访问到,实际删除为异步操作
			              try {
			                  const unlinkRes = fs.unlinkSync(ttsPath)
			                  console.log('unlinkRes',unlinkRes)
			              } catch(e) {
			                  console.error(e)
			              }
			          });
			          // 播放音频
			          audio.onCanplay(() => {
			              console.log('音频开始播放');
			              this.ttsAudio.play();
			              this.isPlay = true
			          })   
			      }
			  })
		}
	},
}
备注:

就想到这些,如果还有需要注意的后续再补充;最近骑电车要带头盔,飘逸的发型压得趴在头上,影响刘斩仙风度翩翩谦谦君子形象,可恶啊!


http://www.niftyadmin.cn/n/5149416.html

相关文章

TensorFlow案例学习:使用 YAMNet 进行迁移学习,对音频进行识别

前言 上一篇文章 TensorFlow案例学习:简单的音频识别 我们简单学习了音频识别。这次我们继续学习如何使用成熟的语音分类模型来进行迁移学习 官方教程: 使用 YAMNet 进行迁移学习,用于环境声音分类 模型下载地址(需要科学上网&…

python第一课 变量

1.离线的情况下首选txt文档 2.有道云笔记 3.思维导图 xmind mindmaster 4.博客 5.wps流程图 # 变量的命名规则 1.变量名只能由数字字母下划线组成 2.变量名不能以数字开头 3.变量名不能与关键字重名 快捷键 撤销:Ctrl/Command Z 新建:Ctrl/Com…

Android标题栏(TitleBar)绝佳解决方案

标题栏框架 项目地址:Github博客地址:Android标题栏(TitleBar)绝佳解决方案 想了解实现原理的可以参考文章:纯手工打造一个通用的标题栏 TitleBar 集成步骤 如果你的项目 Gradle 配置是在 7.0 以下,需要…

uinapp微信小程序隐私政策授权

&#x1f680; 隐私弹窗效果图&#xff1a; 1、启用隐私相关功能在manifest.json文件中配置 usePrivacyCheck: true "mp-weixin" : {"__usePrivacyCheck__" : true, },2、创建组件 <template><view><!-- 隐私政策弹窗 --><uni-popu…

C#开发的OpenRA游戏之步兵射击(2)

C#开发的OpenRA游戏之步兵射击(2) 前面已经分析士兵射击的整个过程,理解它是怎么样根据武器来创建弹盒,然后加载子弹。现在来分析子弹是怎么伤害到对方的过程。 继续前面的分析,它创建了子弹类Bullet,在这个类里实现爆炸效果和伤害转化。类Bullet也是由它的信息类Bulle…

Linux命令(113)之rev

linux命令之rev 1.rev介绍 linux命令rev是将文件中的每行内容已字符为单位反向输出&#xff0c;即第一个字符最后输出&#xff0c;最后一个字符最先输出 2.rev用法 rev [参数] filename rev参数 参数说明-V显示版本信息-h显示帮助信息 3.实例 3.1.显示rev的版本信息 命令…

(论文阅读13/100)R-CNN minus R

文献阅读笔记 简介 题目 R-CNN minus R 作者 Karel Lenc Andrea Vedaldi 原文链接 https://arxiv.org/pdf/1506.06981.pdf 关键词 Null 研究问题 proposal generation在基于CNN的探测器中的作用&#xff0c;以确定它是否是一个必要的建模组件。 R-CNN留下的几个有趣…

默认路由配置

默认路由&#xff1a; 在末节路由器上使用。&#xff08;末节路由器是前往其他网络只有一条路可以走的路由器&#xff09; 默认路由被称为最后的关卡&#xff0c;也就是静态路由不可用并且动态路由也不可用&#xff0c;最后就会选择默认路由。有时在末节路由器上写静态路由时…