使用百度的长文本转语音API时无法下载.MP3文件

今天是学生们交作业的时候,结果是我最忙碌的一天,各种改bug。

有个学生来问:

我在百度提供的API代码(长文本转语音)的基础上添加了下载生成的.MP3文件的代码,运行之后成功建成了.MP3文件,但是文件的内容确实以下的报错信息,根据这一信息采用了新的accesstoken之后依旧无法解决。

看了输出,发现获取链接的部分已经完成,是后面下载这一链接里MP3文件的代码中存在的问题,但是经过查询都没有指出这一部分代码的问题

从这个错误信息看,似乎是下载MP3文件时,百度的服务器认为链接非法,所以拒绝了。但是如果单独用curl或者wget去下载这个MP3的链接,是可以得到正确的文件的。看来这个问题是在程序里面隐藏的。

要来了学生的代码,大致是这样的:

cpp 复制代码
int main() {
    std::cout << "Current working directory: " << getcwd(NULL, 0) << std::endl;
    CURL *curl;
    FILE *fp;

    CURLcode res;
    std::string result;
    std::string downloadUrl;
    std::string outputFilename = "output.mp3";
    curl_global_init(CURL_GLOBAL_ALL);
    curl = curl_easy_init();
    if(curl) {
        // 发送 POST 请求获取语音合成任务结果
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L); // 禁用证书验证
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L); // 禁用证书验证
        curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, "POST");
        // URL和access_token
        curl_easy_setopt(curl, CURLOPT_URL, "https://aip.baidubce.com/rpc/2.0/tts/v1/query?access_token=xxxxxxxx");
        curl_easy_setopt(curl, CURLOPT_POSTFIELDS,"{\"task_ids\":[\"yyyyyyy\"]}");
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, &result);
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, onWriteData);
        res = curl_easy_perform(curl);
        if(res != CURLE_OK) {
            std::cerr << "curl_easy_perform() failed: " << curl_easy_strerror(res) << std::endl;
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }

        // 解析 JSON 响应,获取 MP3 文件的下载链接
        cJSON *json = cJSON_Parse(result.c_str());
        if (json == NULL) {
            std::cerr << "Error parsing JSON." << std::endl;
            // 清理工作
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }

......

        cJSON *task_result = cJSON_GetObjectItem(task_info, "task_result");
        if (task_result != NULL) {
            // "speech_url",字段名
            cJSON *download_url_json = cJSON_GetObjectItem(task_result, "speech_url");
	    if (download_url_json != NULL && download_url_json->type == cJSON_String) {
                downloadUrl = download_url_json->valuestring; // 获取下载链接
                //fprintf(stderr, "speech-url: %s\n", downloadUrl.c_str());
	    } else {
                std::cerr << "No download URL found in task result." << std::endl;
                // 清理
                cJSON_Delete(json);
                curl_easy_cleanup(curl);
                curl_global_cleanup();
                return 1;
            }
        } else {
            std::cerr << "No task result found." << std::endl;
            // 清理
            cJSON_Delete(json);
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }
        // 下载 MP3 文件
	fp = fopen(outputFilename.c_str(), "wb");
        if (fp==nullptr) {
            std::cerr << "Failed to open file for writing." << std::endl;
            // 清理
            cJSON_Delete(json);
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }
        curl_easy_setopt(curl, CURLOPT_URL, downloadUrl.c_str()); // 下载链接
        fprintf(stderr, "speech-url: %s\n", downloadUrl.c_str()); // 打印查看
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback);
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, fp);
	res = curl_easy_perform(curl); // 下载
        if (res != CURLE_OK) {
            std::cerr << "curl_easy_perform() failed: " << curl_easy_strerror(res) << std::endl;
            fclose(fp);
            // 清理
            cJSON_Delete(json);
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }

从这段代码可以看出,curl变量前面被用来做POST请求,后面又被用来做GET请求,第一次操作的状态影响了第二次,导致第二次发给服务器的数据错了。让学生改成下面这样就对了:

当然更好的办法是用两个curl变量来完成不同的任务。

一定要记住,不要重复使用一个变量完成两个不同的任务!

相关推荐
1G36 分钟前
openclaw控制浏览器/自动化的playwright MCP + Mcporter方案实现
人工智能
踩着两条虫42 分钟前
VTJ.PRO 双向代码转换原理揭秘
前端·vue.js·人工智能
扉川川42 分钟前
OpenClaw 架构解析:一个生产级 AI Agent 是如何设计的
前端·人工智能
星浩AI1 小时前
让模型自己写 Skills——从素材到自动生成工作流
人工智能·后端·agent
千寻girling5 小时前
Python 是用来做 AI 人工智能 的 , 不适合开发 Web 网站 | 《Web框架》
人工智能·后端·算法
AI攻城狮5 小时前
OpenClaw 里 TAVILY_API_KEY 明明写在 ~/.bashrc,为什么还是失效?一次完整排查与修复
人工智能·云原生·aigc
stark张宇5 小时前
构建第一个AI聊天机器人:Flask+DeepSeek+Postgres实战
人工智能·postgresql·flask
yiyu07167 小时前
3分钟搞懂深度学习AI:自我进化的最简五步法
人工智能·深度学习
浪浪山_大橙子8 小时前
OpenClaw 十分钟快速,安装与接入完全指南 - 推荐使用trae 官方 skills 安装
前端·人工智能
火山引擎开发者社区9 小时前
OpenClaw 快速上手:把云手机变成你的 7×24 小时 AI 手机助手
人工智能