基于腾讯云的AI视频课程制作工具

news2025/1/13 7:43:44

1. 需求信息

1.1 需求背景

讲师们在制作视频的过程中,发现录制课程比较麻烦,要保证环境安静,保证录制过程不出错,很容易反复重复录制,为了解决重复录制的工作量,想通过 ai 课程制作工具,来解决这些问题。

2. 业务分析

2.1 视频生成过程

  1. 视频素材来源: 首先由产品研发团队的产品、架构师协助提供基础资料,根据产品材料来输出ppt文档;
  2. 素材上传:使用的授课课件材料,如:PPT 或 PDF(也有doc课件大纲、课程详细内容,现在我们基本上用不上),上传到AI视频制作工具平台;
  3. 素材文本提取:a. 提取素材文本内容(ppt内容+备注);b. 将素材截图成图片;
  4. 演讲稿制作:将第3步提取的文字,交由混元来生成讲课稿,这里可以人工校验句子合理性;
  5. 演讲稿合成语音:将第4步生成的讲课稿合成音频文件;
  6. 合成视频:将音频、图片合成视频。

其中,2~6可以合成一步,就是在有讲课稿的情况下,直接提取备注合成视频。

2.2 视频课程权限

  1. 数据隔离和安全:课程以空间的概念相互隔离, 用户使用OA登录系统后,默认不可见其他同事创建的空间,只能看到自己创建的空间。
  2. 数据共享协作:空间创建者默认拥有空间管理权限,可以邀请其他同事协作;
  3. 超级管理员:超级管理员课件所以课件,超级管理员不可配置,只能研发修改数据库角色。

2.4 发音修正

部分专业英文缩写发音不准的问题,可以通过SMAL标记语言修正发音,因此提供一个发音修正管理菜单,用户可以自定义发音部分单词。

2.5 用户登录限制

  1. 系统接入OA登录,只能在内网访问;
  2. 给用户添加权限,必须在用户登录过系统之后,才能添加(必须用户登录之后,系统才会记录用户信息)。

3. 技术设计

3.1 视频制作流程

目前我们的技术方案是腾讯云语音合成(TTS)+视频合成(云点播)+混元大模型来搭建的。支持2种方式来生成视频:

  1. 无讲课稿的情况下,通过解析读取ppt文档的内容和备注, 调用混元大模型来生成演讲稿, 演讲稿生成语音, 再截取PPT的图片,来合成视频;
  2. 有讲课稿的情况下, 支持一件生成讲课视频。

整体流程入下图所示:

在这里插入图片描述

3.2 发音纠正

由于课程内容主要是腾讯云的产品培训,因此有很多腾讯云相关的专有英文名词和缩写,腾讯云TTS对这些词的合成不够理想,不过提供了SSML 标记语言,用来自定义纠正发音,如上图发音纠正部分。

3.3 相关技术工具

  1. 腾讯云对象存储cos,制作课程过程中的各种素材,包括:文件、图片、音频、视频等都是存储在cos里面
  2. 数据万象: PPT转PDF使用的是cos 自带的数据万象能力,可以把ppt转换为pdf格式文件
  3. 开源工具pptx:可以读取ppt的演讲稿的文本和备注;
  4. 开源工具PyPDF2:可以把pdf文件截取成图;
  5. 混元大模型:提供AIGC能力,写入prompt讲提取的课件文本生成讲课稿;
  6. 语音合成(TTS):使用腾讯云TTS将文本生成生动的语音;
  7. 视频合成:腾讯云点播将音频和图片按时间线生成视频;
  8. 后端web框架:fastapi,一个python的http服务框架;
  9. 前端框架:内部开源的TDesign。

4 问题

4.1 语音合成(TTS)

目前业务方使用反馈最多的是腾讯云TTS的语言合成效果问题,例如:

  1. 专有名词发音不正确;
  2. 同一个语音类型发音过程中切换;
  3. 中、英切换过程中出现不同发音。

针对上述问题已经在尝试2个不同的解决方案:

  1. 推动腾讯云TTS优化:已经拉通TTS产品团队在支持,并逐步在收集发音的base case;
  2. 调研开源TTS语音合成大模型,目前已知的ChatTTS 和阿里开源的CosyVoice 都有非常流畅的效果;
  3. 第三方云平台的TTS 实现,目前国内讯飞的合成效果也不错。

4.2 讲课稿生成

讲课稿的生成过程比较耗时,一个课程小节经常长达100+页ppt,每一页都需要AIGC生成后,还需要人工精调。

针对这个问题可以考虑:

  1. 增强知识库,随着课程制作素材的累积,我们将形成一个优质的知识库,可以提供大模型非常好的知识增强;
  2. 个性化微调大模型:随着数据的积累,可以使用混元一站式训练平台,对针对性微调个性化模型专门用来优化演讲稿生成。

附件

主要python工具包

  • fastapi==0.111.0 // fastapi web框架
  • python-pptx==0.6.23 // pptx 内容读取
  • pdf2image==1.17.0 // pdf转图片
  • tencentcloud-sdk-python==3.0.1132 // 调用腾讯云api

文本提取

from pptx import Presentation
import PyPDF2
import tempfile,io,requests
from dependencies import FILE_NAME_MAX_LENGTH, new_file_name
from logger_config import logger
from repo import schemas
from typing import List
from fastapi import HTTPException
from starlette.status import HTTP_400_BAD_REQUEST, HTTP_200_OK

# 提取ppt中的正文和备注
def extract_info_from_ppt(ppt_url: str) -> List[schemas.PPtTextNode]:
        # 下载PDF文件
    logger.info('开始下载ppt文件: ' + ppt_url)
    response = requests.get(ppt_url)
    if response.status_code != HTTP_200_OK:
        logger.error("file download failed : " + ppt_url)
        raise HTTPException(
            status_code=HTTP_400_BAD_REQUEST,
            detail="文件下载失败",
        )
    pdf_bytes = response.content
     # 将字节流转换为文件对象
    file_obj = io.BytesIO(pdf_bytes)
    # 加载 PowerPoint 文档
    presentation = Presentation(file_obj)

    # 备注内容
    slide_infos : List[schemas.PPtTextNode] = []
    # 遍历幻灯片
    for slide in presentation.slides:
        # 获取幻灯片上的文本
        slide_text = ""
        for shape in slide.shapes:
            if hasattr(shape, "text"):
                slide_text += shape.text
        # 获取幻灯片的备注
        notes_slide = slide.notes_slide
        # 获取备注文本
        notes_text = notes_slide.notes_text_frame.text

        # 添加到备注列表
        slide_infos.append(schemas.PPtTextNode(text=slide_text, note=notes_text))
    return slide_infos
# 提取pdf中的内容
def extract_info_from_pdf(pdf_url: str):
    # 下载PDF文件
    logger.info('开始下载PDF文件: ' + pdf_url)
    response = requests.get(pdf_url)
    if response.status_code != HTTP_200_OK:
        raise HTTPException(
            status_code=HTTP_400_BAD_REQUEST,
            detail="文件下载失败",
        )
    pdf_bytes = response.content
    # 将字节流转换为文件对象
    file_obj = io.BytesIO(pdf_bytes)

    # 加载 PDF 文件
    pdf_reader = PyPDF2.PdfReader(file_obj)
    # 获取 PDF 文件的页数
    num_pages = len(pdf_reader.pages)
    # 遍历 PDF 文件的每一页
    slide_infos : List[schemas.PPtTextNode] = []
    for page_num in range(num_pages):
        # 获取当前页
        page = pdf_reader.pages[page_num]
        # 提取页面内容
        content = page.extract_text()
        # 添加到备注列表
        silde_info = schemas.PPtTextNode(text=content, notes="")
        slide_infos.append(silde_info)
    return slide_infos

pdf转图片

import os
from logic.qcloud import cosclient
import tempfile
from logger_config import logger
import requests,re
from pdf2image import convert_from_bytes
from typing import List
from repo import schemas
import tracemalloc
import config

from dependencies import FILE_NAME_MAX_LENGTH, new_file_name

# 定义一个函数来验证文件名是否安全
def is_safe_filename(filename):
    # 使用正则表达式来匹配合法的文件名
    return bool(re.match(r'^[\w\-.]+$', filename))

def pdf_url_to_images(pdf_url: str, space_id: int)->List[schemas.PPtToImage]:
    # 下载PDF文件
    logger.info('开始下载PDF文件: ' + pdf_url)
    response = requests.get(pdf_url)
    pdf_bytes = response.content
    
     # 将PPT文件的每一页保存为图像
    output_folder = tempfile.mkdtemp()
    
    # 将PDF文件转换为图像
    tracemalloc.start()
    # 获取内存分配情况的快照
    if config.get_settings().is_tracemalloc == True:
        snapshot1 = tracemalloc.take_snapshot()
            
    images = convert_from_bytes(pdf_bytes)

    if config.get_settings().is_tracemalloc == True:
        snapshot2 = tracemalloc.take_snapshot()
        # 比较两个快照,找出内存分配差异
        top_stats = snapshot2.compare_to(snapshot1, "lineno")
        # 打印内存分配差异的统计信息
        for stat in top_stats[:10]:
            logger.info("读取文件后,内存分配差异: %s" % stat)

        total_size = sum(stat.size for stat in snapshot2.statistics("lineno"))

        # 将字节转换为合适的单位(如 MiB)
        total_size_mib = total_size / (1024 * 1024)

        logger.info(f"读取文件后,总内存分配: {total_size_mib:.2f} MiB")

    #image_urls = []
    file_infos : List[schemas.PPtToImage] = []
    for index, image in enumerate(images):
        image_path = os.path.join(output_folder, f"page_{index + 1}.png")
        image.save(image_path, "PNG")

        # 上传图像到COS
        url = ""
        file_name = os.path.basename(image_path)
        image_key = new_file_name(file_name, space_id)
        if len(image_key) > FILE_NAME_MAX_LENGTH:
            image_key = image_key[:FILE_NAME_MAX_LENGTH]
        
        cosclient.put_object_file(image_path, image_key)
        url = cosclient.get_presigned_url(image_key)
        file_infos.append(schemas.PPtToImage(image_name=image_key, image_url=url))
        logger.info(f"Uploaded {image_path} to {image_key}")

    # 删除临时文件夹
    for image_filename in os.listdir(output_folder):
            # 验证文件名是否安全
        if is_safe_filename(image_filename):
            # 如果文件名安全,则删除文件
            os.unlink(os.path.join(output_folder, image_filename))
        else:
            # 如果文件名不安全,记录日志并跳过删除操作
            logger.warning(f"Unsafe filename detected: {image_filename}. Skipping deletion.")

    os.rmdir(output_folder)

    return file_infos

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2215228.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

注册安全分析报告:北外网校

前言 由于网站注册入口容易被黑客攻击,存在如下安全问题: 暴力破解密码,造成用户信息泄露短信盗刷的安全问题,影响业务及导致用户投诉带来经济损失,尤其是后付费客户,风险巨大,造成亏损无底洞…

【软考】设计模式之中介者模式

目录 1. 说明2. 应用场景3. 结构图4. 构成5. 适用性6. 优点 1. 说明 1.用一个中介对象来封装一系列的对象交互。2.中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。3.中介者模式(Mediator Pattern&…

Qt-自定义控件鼠标事件键盘事件定时器绘图

1. 自定义控件 1.1 创建自定义控件 1.在项目目录上右键, 选择 "Add New" 2.选择 "Qt" --> "Qt 设计师界面类" 3.根据需求选择模板,此处选择空窗口 4.设置类名 和 相关文件名 使用设计师界面类会产生三个文件&…

媒界:插混VS增程:魏牌蓝山用天花板Hi4诠释都市家庭用车最优解

在新能源混动领域,关于插混、增程谁才是混动最优解,一直业内争论的焦点。正如路遥知马力、日久见人心。对于新能源动力系统的评判标准来说,最好的答案就是路上见。 近日,一位媒体博主驾驶着魏牌全新蓝山从阿拉善到武汉往返狂飙30…

Python面向对象编程:封装和私有属性④

文章目录 1. 引言2. 什么是封装?3. 公有属性和方法4. 私有属性和方法5. 属性访问器(Getters 和 Setters)6. 使用 property 函数7. 综合示例7.1 项目结构7.2 模块代码__init__.pystudent.pycourse.pymanager.py 7.3 主程序代码main.py 7.4 运行…

cmake模板-支持编译动态/静态文件

代码链接:代码仓库 git clone https://gitee.com/etsuyou/cmake-template.git模板 模板截图 如何使用 在src和inc中写代码 此处用我默认提供的代码 ./go.sh cmake 生成Makefile ./go.sh make 生成bin文件和.a以及.so ./go.sh run app 运行 ./go.sh clean 以…

Tomcat服务部署及优化

一、Tomcat的基本介绍 1. tomcat是什么? Tomcat服务器是一个免费的开放源代码的Web应用服务器,属于轻量级应用服务器,在中小型系统和并发访问用户不是很多的场合下被普遍使用,是开发和调试JSP程序的首选。一般来说,T…

QT QML 练习8-Simple Transformations

简单的转换(Simple Transformations) 转换操作改变了一个对象的几何状态。QML元素对象通常能够被平移,旋转,缩放。下面我们将讲解这些简单的操作和一些更高级的用法。 我们先从一个简单的转换开始。用下面的场景作为我们学习的开始…

Python学习100天第9天之面向对象进阶

1 前言 在前面的章节我们已经了解了面向对象的入门知识,知道了如何定义类,如何创建对象以及如何给对象发消息。为了能够更好的使用面向对象编程思想进行程序开发,我们还需要对Python中的面向对象编程进行更为深入的了解。 2 property装饰器…

AVLTree 旋转笔记(根据平衡因子插入的公式,贼好理解)

平衡因子 avltree是一棵每个节点的左右子树的高度差不超过1的二叉树搜索树,对于avltree最重要的就是对平衡因子的控制。 对于旋转我们重点要注意的是三个节点,以左旋举例,需要注意的就是parent,subr,subrl。而旋转的方…

MYSQL架构、执行过程和顺序

MYSQL架构、执行过程和顺序 一、前言 1.1、说明 就MySQL的架构,以及执行过程、sql执行顺序,以及一些相关学习分享内容。 在参考文章的基础上,会增加自己的理解、看法,希望本文章能够在您的学习中提供帮助。 如有错误的地方&a…

Dokcer如何容器部署及常见问题

本文讲解通过Docker部署Jenkins过程及遇到的问题。 通过 Docker 部署 Jenkins 使用 Docker 来部署 Jenkins 是一个快速且高效的方式。以下是使用 Docker 部署 Jenkins 的分步骤指南: 1. 安装 Docker 如果你的系统上还没有安装 Docker,请根据操作系统…

春日技术问答:Spring Boot课程答疑

3系统分析 3.1可行性分析 通过对本课程答疑系统实行的目的初步调查和分析,提出可行性方案并对其一一进行论证。我们在这里主要从技术可行性、经济可行性、操作可行性等方面进行分析。 3.1.1技术可行性 本课程答疑系统采用JAVA作为开发语言,Spring Boot框…

【2D/3D-Lidar-SLAM】 Cartographer详细解读

【2D/3D-Lidar-SLAM】 Cartographer详细解读 1. 摘要 2. Cartographer系统数据处理流程2.1. 数据获取(Input Sensor Data)2.2 姿态外推器(PoseExtrapolator)2.3 局部建图(Local SLAM) 3. 关键模块实现 3.1 …

5、springboot-基础入门

1、系统要求 Java 8 & 兼容java14 .Maven 3.3idea 2019.1.2 1.1、maven设置 修改maven的settings.xml文件中的镜像&#xff0c;如下 <mirrors><mirror><id>nexus-aliyun</id><mirrorOf>central</mirrorOf><name>Nexus aliyu…

vue3 在store的index.js

导入vuex&#xff0c;在store的index.js创建store对象 在main.js挂载store import store from ./storenew Vue ({/* 将store对象实例挂载到vue实例中 所有组件就可以直接从store中获取全局数据了*/ store, render: h > h(App) }).$mount(#app) 在store中的index.js进行声明…

【IPv6】IPv6 NAT66介绍

参考链接 IPv6-to-IPv6 Network Address Translation (NAT66) (ietf.org)https://datatracker.ietf.org/doc/id/draft-mrw-nat66-00.html IPv6 NAT66 NAT66&#xff0c;全称为Network Address Translation for IPv6 to IPv6&#xff0c;是一种用于IPv6网络的地址转换技术。在…

FPGA基于SRIO Auraro 三速以太网 IIC SPI等多协议的高速传输处理项目

高速传输处理项目 此项目涉及较多协议和接口&#xff0c;有较复杂的系统顶层框图设计。在涉及设备较多的应用场景中&#xff0c;需要涉及一款PCI-E板卡&#xff0c;将多个子系统的数据汇总上传到PC或服务器上。在此项目中有3路数据源&#xff0c;分别是:srio数据&#xff0c; …

数据结构与算法:动态规划的深度探讨

目录 12.1 动态规划的核心思想 12.2 经典动态规划问题 12.3 动态规划在图中的应用 12.4 高级动态规划技术 总结 数据结构与算法&#xff1a;动态规划的深度探讨 动态规划&#xff08;Dynamic Programming, DP&#xff09;是一种解决复杂问题的有效方法&#xff0c;特别适…

Nuxt3部署-Ubuntu系统(Node 服务 + pm2 + Nginx 反向代理)

Nuxt3部署-Ubuntu系统&#xff08;Node 服务 pm2 Nginx 反向代理&#xff09; 文章目录 Nuxt3部署-Ubuntu系统&#xff08;Node 服务 pm2 Nginx 反向代理&#xff09;一、安装 Nodejs 环境二、安装 Nginx三、安装 pm2四、本地项目打包1️⃣&#xff1a;打包2️⃣&#xff1…