day09项目【课程列表和整合阿里云视频点播】

1 课程列表和课程删除

01-课程列表的显示

课程列表显示:

课程列表(参考讲师列表,完善条件查询和分页):

一、后端实现

1、定义搜索对象

CourseQuery:

@ApiModel(value = "Course查询对象", description = "课程查询对象封装")

@Data

public class CourseQuery {

    @ApiModelProperty(value = "课程名称")

    private String title;

    @ApiModelProperty(value = "讲师id")

    private String teacherId;

    @ApiModelProperty(value = "一级类别id")

    private String subjectParentId;

    @ApiModelProperty(value = "二级类别id")

    private String subjectId;

}

2、定义web层方法

    //条件查询带分页的方法

    @ApiOperation(value = "分页课程列表")

    @PostMapping("pageCourseCondition/{current}/{limit}")

    public R pageCourseCondition(@ApiParam(name = "current", value = "当前页码", required = true) @PathVariable long current,

                                @ApiParam(name = "limit", value = "每页记录数", required = true) @PathVariable long limit,

                                @RequestBody(required = false) CourseQuery courseQuery){

        //创建page对象

        Page<EduCourse> pageParam = new Page<>(current,limit);

        courseService.pageQuery(pageParam,courseQuery);

        List<EduCourse> records = pageParam.getRecords();

        long total = pageParam.getTotal();

        //调用方法实现条件查询分页

        return R.ok().data("rows",records).data("total",total);

    }

3、定义service方法

接口

void pageQuery(Page<EduCourse> pageParam, CourseQuery courseQuery);

实现

@Override

    public void pageQuery(Page<EduCourse> pageParam, CourseQuery courseQuery) {

        //构建条件

        QueryWrapper<EduCourse> courseQueryWrapper = new QueryWrapper<>();

        courseQueryWrapper.orderByDesc("gmt_create");

        //wrapper

        if (courseQuery == null){

            baseMapper.selectPage(pageParam,courseQueryWrapper);

            return;

        }

        String title = courseQuery.getTitle();

        String teacherId = courseQuery.getTeacherId();

        String subjectParentId = courseQuery.getSubjectParentId();

        String subjectId = courseQuery.getSubjectId();

        if (!StringUtils.isEmpty(title)){

            courseQueryWrapper.like("title",title);

        }

        if (!StringUtils.isEmpty(teacherId)){

            courseQueryWrapper.eq("teacher_id",teacherId);

        }

        if (!StringUtils.isEmpty(subjectParentId)){

            courseQueryWrapper.eq("subject_parent_id",subjectParentId);

        }

        if (!StringUtils.isEmpty(subjectId)){

            courseQueryWrapper.eq("subject_id",subjectId);

        }

        baseMapper.selectPage(pageParam,courseQueryWrapper);

    }

二、前端分页查询列表

1、定义api

course.js

 //课程列表(条件查询分页)

    //current当前页,limit每页记录数,teacherQuery条件对象

    getCoursePageList(current,limit,courseQuery){

        return request({

            url:`/eduservice/course/pageCourseCondition/${current}/${limit}`,

            method:'post',

            data:courseQuery

        })

    }

2、组件中的js

src/views/edu/list.vue

<script>

import course from '@/api/edu/course'

import teacher from '@/api/edu/teacher'

import subject from '@/api/edu/subject'

export default {

    data(){

        return {

            listLoading: true, // 是否显示loading信息

            list:null,

            page:1,//当前页

            limit:10,//每页记录数

            total:0,//总记录数

            courseQuery:{},//条件封装对象

            teacherList: [], // 讲师列表

            subjectNestedList: [], // 一级分类列表

            subSubjectList: [] // 二级分类列表,

        }

    },

    created(){

        this.getList()

        // 初始化分类列表

        this.initSubjectList()

        // 获取讲师列表

        this.initTeacherList()

    },

    methods:{

        getList(page=1) {

            this.page = page

            this.listLoading = true

            course.getCoursePageList(this.page,this.limit,this.courseQuery)

                .then(response=>{

                    if (response.success === true) {

                        this.list = response.data.rows

                        this.total = response.data.total

                    }   

                    this.listLoading = false

                })

                .catch(error =>{//请求失败

                    console.log(error)

                })

        },

        initTeacherList() {

            teacher.getTeacherList()

                .then(response => {

                    this.teacherList = response.data.items

                })

        },

        initSubjectList() {

            subject.getSubjectList()

                .then(response => {

                    this.subjectNestedList = response.data.list

                })

        },

        subjectLevelOneChanged(value){

            for (let i = 0; i < this.subjectNestedList.length; i++) {

                    if (this.subjectNestedList[i].id === value) {

                    this.subSubjectList = this.subjectNestedList[i].children

                    this.searchObj.subjectId = ''

                    }

                }

        },

        resetData(){//清空方法

            //表单输入项数据清空

            this.courseQuery = {}

            this.subSubjectList = [] // 二级分类列表

            //查询所有讲师数据

            this.getList()

        },

    }

}

</script>

3、组件模板

查询表单、表格和分页:表格添加了 row-class-name="myClassList" 样式定义

<template>

    <div class="app-container">

        <!--查询表单-->

        <el-form :inline="true" class="demo-form-inline">

        <!-- 所属分类:级联下拉列表 -->

        <!-- 一级分类 -->

        <el-form-item label="课程类别">

            <el-select

            v-model="courseQuery.subjectParentId"

            placeholder="请选择"

            @change="subjectLevelOneChanged">

            <el-option

                v-for="subject in subjectNestedList"

                :key="subject.id"

                :label="subject.title"

                :value="subject.id"/>

            </el-select>


            <!-- 二级分类 -->

            <el-select v-model="courseQuery.subjectId" placeholder="请选择">

            <el-option

                v-for="subject in subSubjectList"

                :key="subject.id"

                :label="subject.title"

                :value="subject.id"/>

            </el-select>

        </el-form-item>


        <!-- 标题 -->

        <el-form-item>

            <el-input v-model="courseQuery.title" placeholder="课程标题"/>

        </el-form-item>

        <!-- 讲师 -->

        <el-form-item>

            <el-select

            v-model="courseQuery.teacherId"

            placeholder="请选择讲师">

            <el-option

                v-for="teacher in teacherList"

                :key="teacher.id"

                :label="teacher.name"

                :value="teacher.id"/>

            </el-select>

        </el-form-item>


        <el-button type="primary" icon="el-icon-search" @click="getList()">查询</el-button>

        <el-button type="default" @click="resetData()">清空</el-button>

        </el-form>

        <!-- 表格 -->

        <el-table

            v-loading="listLoading"

            :data="list"

            element-loading-text="数据加载中"

            border

            fit

            highlight-current-row

            row-class-name="myClassList">


            <el-table-column

                label="序号"

                width="70"

                align="center">

                <template slot-scope="scope">

                {{ (page - 1) * limit + scope.$index + 1 }}

                </template>

            </el-table-column>

            <el-table-column label="课程信息" width="470" align="center">

                <template slot-scope="scope">

                <div class="info">

                    <div class="pic">

                        <img :src="scope.row.cover" alt="scope.row.title" width="150px">

                    </div>

                    <div class="title">

                    <a href="">{{ scope.row.title }}</a>

                    <p>{{ scope.row.lessonNum }}课时</p>

                    </div>

                </div>


                </template>

            </el-table-column>


            <el-table-column label="创建时间" align="center">

                <template slot-scope="scope">

                {{ scope.row.gmtCreate.substr(0, 10) }}

                </template>

            </el-table-column>

            <el-table-column label="发布时间" align="center">

                <template slot-scope="scope">

                {{ scope.row.gmtModified.substr(0, 10) }}

                </template>

            </el-table-column>

            <el-table-column label="价格" width="100" align="center" >

                <template slot-scope="scope">

                {{ Number(scope.row.price) === 0 ? '免费' :

                '¥' + scope.row.price.toFixed(2) }}

                </template>

            </el-table-column>

            <el-table-column prop="buyCount" label="付费学员" width="100" align="center" >

                <template slot-scope="scope">

                {{ scope.row.buyCount }}人

                </template>

            </el-table-column>


            <el-table-column label="操作" width="150" align="center">

                <template slot-scope="scope">

                <router-link :to="'/edu/course/info/'+scope.row.id">

                    <el-button type="text" size="mini" icon="el-icon-edit">编辑课程信息</el-button>

                </router-link>

                <router-link :to="'/edu/course/chapter/'+scope.row.id">

                    <el-button type="text" size="mini" icon="el-icon-edit">编辑课程大纲</el-button>

                </router-link>

                <el-button type="text" size="mini" icon="el-icon-delete">删除</el-button>

                </template>

            </el-table-column>

        </el-table>

        <!-- 分页 -->

        <el-pagination

        :current-page="page"

        :page-size="limit"

        :total="total"

        style="padding: 30px 0; text-align: center;"

        layout="total, prev, pager, next, jumper"

        @current-change="getList"

        />

    </div>

</template>

4、css的定义

<style scoped>

.myClassList .info {

    width: 450px;

    overflow: hidden;

}

.myClassList .info .pic {

    width: 150px;

    height: 90px;

    overflow: hidden;

    float: left;

}

.myClassList .info .pic a {

    display: block;

    width: 100%;

    height: 100%;

    margin: 0;

    padding: 0;

}

.myClassList .info .pic img {

    display: block;

    width: 100%;

}

.myClassList td .info .title {

    width: 280px;

    float: right;

    height: 90px;

}

.myClassList td .info .title a {

    display: block;

    height: 48px;

    line-height: 24px;

    overflow: hidden;

    color: #00baf2;

    margin-bottom: 12px;

}

.myClassList td .info .title p {

    line-height: 20px;

    margin-top: 5px;

    color: #818181;

}

</style>

分页失败原因:

测试:

编辑:即可完成点击“编辑课程信息”和“编辑课程大纲”跳转到相应的页面进行编辑


02-删除课程

一、后端实现

1、web层

定义删除api方法:EduCourseController.java

    //根据ID删除课程

    @ApiOperation(value = "根据ID删除课程")

    @DeleteMapping("deleteCourse/{courseId}")

    public R deleteCourse(@ApiParam(name = "id", value = "课程ID", required = true)

                            @PathVariable String courseId){

        courseService.removeCourse(courseId);

        return R.ok();

    }

2、service层

如果用户确定删除,则首先删除video记录,然后删除chapter记录,最后删除Course记录

//当前在课程实现类EduCourseServiceImpl中,删除小节和章节需要在其实现类中删除,因此首先将相关的Service进行注入

    @Autowired

    private EduVideoService videoService;

    @Autowired

    private EduChapterService chapterService;

2.1、在VideoService中定义根据courseId删除video业务方法

接口:

void removeVideoByCourseId(String courseId);

实现:

    // 1 根据课程id删除小节

    //TODO 删除小节,删除对应视频文件

    @Override

    public void removeVideoByCourseId(String courseId) {

        //需要根据小节id进行删除,但是传递过来的是课程id,因此需要传递对象

        QueryWrapper<EduVideo> wrapper = new QueryWrapper<>();

        wrapper.eq("course_id",courseId);

        baseMapper.delete(wrapper);

    }

2.2、在ChapterService中定义根据courseId删除chapter业务方法

接口:

void removeChapterByCourseId(String courseId);

实现:

    //根据课程id删除章节

    @Override

    public void removeChapterByCourseId(String courseId) {

        QueryWrapper<EduChapter> wrapper = new QueryWrapper<>();

        wrapper.eq("course_id",courseId);

        baseMapper.delete(wrapper);

    }

2.3、删除当前course记录

接口:CourseService.java

    //根据ID删除课程

    void removeCourse(String courseId);

实现:CourseServiceImpl.java

    //根据ID删除课程

    @Override

    public void removeCourse(String courseId) {

        //1 根据课程id删除小节

        videoService.removeVideoByCourseId(courseId);

        //2 根据课程id删除章节

        chapterService.removeChapterByCourseId(courseId);

        //3 根据课程id删除描述

        courseDescriptionService.removeById(courseId);

        //4 根据课程id删除课程本身

        int result = baseMapper.deleteById(courseId);

        if(result == 0) { //失败返回

            throw new GuliException(20001,"删除失败");

        }

    }

测试:

测试删除失败:

成功:

二、前端实现

删除按钮(完善):

1、定义api

course.js中添加删除方法

     //根据ID删除课程

    removeById(courseId){

        return request({

            url:`/eduservice/course/deleteCourse/${courseId}`,

            method:'delete'

        })

    }

2、修改删除按钮

src/api/edu/course.js 删除按钮注册click事件

<el-button type="text" size="mini" icon="el-icon-delete" @click="removeDataById(scope.row.id)">删除</el-button>

3、编写删除方法

removeDataById(id){

            // debugger

            this.$confirm('此操作将永久删除该课程,以及该课程下的章节和视频,是否继续?', '提示', {

                confirmButtonText: '确定',

                cancelButtonText: '取消',

                type: 'warning'

            }).then(() => {

                return course.removeById(id)

            }).then(() => {

                this.getList()

                this.$message({

                    type: 'success',

                    message: '删除成功!'

                })

            }).catch((response) => { // 失败

                if (response === 'cancel') {

                    this.$message({

                        type: 'info',

                        message: '已取消删除'

                    })

                }

            })

        },


2 阿里云视频点播

01-视频点播简介

一、阿里云视频点播技术能力盘点

参考文章:

https://blog.csdn.net/qq_33857573/article/details/79564255

视频点播(ApsaraVideo for VoD)是集音视频采集、编辑、上传、自动化转码处理、媒体资源管理、分发加速于一体的一站式音视频点播解决方案。

1、应用场景

    音视频网站:无论是初创视频服务企业,还是已拥有海量视频资源,可定制化的点播服务帮助客户快速搭建拥有极致观看体验、安全可靠的视频点播应用。

    短视频:集音视频拍摄、特效编辑、本地转码、高速上传、自动化云端转码、媒体资源管理、分发加速、播放于一体的完整短视频解决方案。目前已帮助1000+APP快速实现手机短视频功能。

    直播转点播:将直播流同步录制为点播视频,用于回看。并支持媒资管理、媒体处理(转码及内容审核/智能首图等AI处理)、内容制作(云剪辑)、CDN分发加速等一系列操作。

    在线教育:为在线教育客户提供简单易用、安全可靠的视频点播服务。可通过控制台/API等多种方式上传教学视频,强大的转码能力保证视频可以快速发布,覆盖全网的加速节点保证学生观看的流畅度。防盗链、视频加密等版权保护方案保护教学内容不被窃取。

    视频生产制作:提供在线可视化剪辑平台及丰富的OpenAPI,帮助客户高效处理、制作视频内容。除基础的剪切拼接、混音、遮标、特效、合成等一系列功能外,依托云剪辑及点播一体化服务还可实现标准化、智能化剪辑生产,大大降低视频制作的槛,缩短制作时间,提升内容生产效率。

    内容审核:应用于短视频平台、传媒行业审核等场景,帮助客户从从语音、文字、视觉等多维度精准识别视频、封面、标题或评论的违禁内容进行AI智能审核与人工审核。

2、功能介绍

二、开通视频点播云平台

1、选择视频点播服务

产品->企业应用->视频云->视频点播

2、开通视频点播

3、选择按使用流量计费

4、资费说明

https://www.aliyun.com/price/product?spm=a2c4g.11186623.2.12.7fbd59b9vmXVN6#/vod/detail

    后付费

    套餐包

    欠费说明

    计费案例:https://help.aliyun.com/document_detail/64032.html?spm=a2c4g.11186623.4.3.363db1bcfdvxB5

5、整体流程

使用视频点播实现音视频上传、存储、处理和播放的整体流程如下:

    用户获取上传授权。

    VoD下发 上传地址和凭证 及 VideoId。

    用户上传视频保存视频ID(VideoId)。

    用户服务端获取播放凭证。

    VoD下发带时效的播放凭证。

    用户服务端将播放凭证下发给客户端完成视频播放。

三、视频点播服务的基本使用

完整的参考文档

https://help.aliyun.com/product/29932.html?spm=a2c4g.11186623.6.540.3c356a58OEmVZJ

1、设置转码格式

选择全局设置 > 转码设置,单击添加转码模板组。

在视频转码模板组页面,根据业务需求选择封装格式和清晰度。

或直接将已有的模板设置为默认即可

2、分类管理

选择全局设置 > 分类管理

3、上传视频文件

选择媒资库 > 音视频,单击上传音视频

4、配置域名

音视频上传完成后,必须配一个已备案的域名,并完成CNAME绑定

得到CNAME

在购买域名的服务商处的管理控制台配置域名解析

5、在控制台查看视频

此时视频可以在阿里云控制台播放

6、获取web播放器代码

阿里云视频点播概念:

    之前上传文件,保存的是上传文件之后的地址,但是在小节中保存视频的时候,而是保存的是视频的id,并不保存视频的地址,这是因为如果视频没有进行加密保存地址是可以的,但是如果视频加密,加密视频地址是不能播放的,在实际项目中,视频是需要加密的。

    如果视频不进行加密,那别人得到地址那视频就可以播放,可以不付费就可以观看。

    保存的是视频的id既可以得到视频播放的地址,也可以得到视频播放的凭证(既可以播放加密视频也可以播放不加密视频)。


02-使用服务端SDK

一、服务端SDK

1、简介

sdk的方式将api进行了进一步的封装,不用自己创建工具类。

我们可以基于服务端SDK编写代码来调用点播API,实现对点播产品和服务的快速操作

2、功能介绍

    SDK封装了对API的调用请求和响应,避免自行计算较为繁琐的 API签名。

    支持所有点播服务的API,并提供了相应的示例代码。

    支持7种开发语言,包括:Java、Python、PHP、.NET、Node.js、Go、C/C++。

    通常在发布新的API后,我们会及时同步更新SDK,所以即便您没有找到对应API的示例代码,也可以参考旧的示例自行实现调用。

二、使用SDK

在service创建子模块service_ vod ,引入相关依赖到3 整合阿里云视频点播 01-视频点播微服务的创建笔记进行复制

1、安装

参考文档:https://help.aliyun.com/document_detail/57756.html

添加maven仓库的配置和依赖到pom

<repositories>

        <repository>

            <id>sonatype-nexus-staging</id>

            <name>Sonatype Nexus Staging</name>

            <url>https://oss.sonatype.org/service/local/staging/deploy/maven2/</url>

            <releases>

                <enabled>true</enabled>

            </releases>

            <snapshots>

                <enabled>true</enabled>

            </snapshots>

        </repository>

    </repositories>

<dependencies>

        <dependency>

            <groupId>com.aliyun</groupId>

            <artifactId>aliyun-java-sdk-core</artifactId>

        </dependency>

        <dependency>

            <groupId>com.aliyun.oss</groupId>

            <artifactId>aliyun-sdk-oss</artifactId>

        </dependency>

        <dependency>

            <groupId>com.aliyun</groupId>

            <artifactId>aliyun-java-sdk-vod</artifactId>

        </dependency>

<!--        <dependency>-->

<!--            <groupId>com.aliyun</groupId>-->

<!--            <artifactId>aliyun-sdk-vod-upload</artifactId>-->

<!--        </dependency>-->

        <dependency>

            <groupId>com.alibaba</groupId>

            <artifactId>fastjson</artifactId>

        </dependency>

        <dependency>

            <groupId>org.json</groupId>

            <artifactId>json</artifactId>

        </dependency>

        <dependency>

            <groupId>com.google.code.gson</groupId>

            <artifactId>gson</artifactId>

        </dependency>

        <dependency>

            <groupId>joda-time</groupId>

            <artifactId>joda-time</artifactId>

        </dependency>

    </dependencies>

2、初始化

public class InitObject {

    public static DefaultAcsClient initVodClient(String accessKeyId, String accessKeySecret) throws ClientException {

        String regionId = "cn-shanghai";  // 点播服务接入区域

        DefaultProfile profile = DefaultProfile.getProfile(regionId, accessKeyId, accessKeySecret);

        DefaultAcsClient client = new DefaultAcsClient(profile);

        return client;

    }

}

3、获取视频播放地址

public class VodSdkTest {

    public static void main(String[] args) throws Exception {

        //1 根据视频id获取视频播放地址

        //获取初始化对象

        DefaultAcsClient client = InitObject.initVodClient("LTAI4GKJXUR4VuFMrtnzzWBG","yiEPUWzpivYpYTG8o0tVW4UFqE2EQh");

        //创建获取视频地址的request和response

        GetPlayInfoResponse response = new GetPlayInfoResponse();

        GetPlayInfoRequest request = new GetPlayInfoRequest();

        //向request对象里面设置视频id

        request.setVideoId("7d92de4882d44b24985862e5c368b195");

        //调用初始化对象里面的方法,传递request,获取数据

        response = client.getAcsResponse(request);

        List<GetPlayInfoResponse.PlayInfo> playInfoList = response.getPlayInfoList();

        //播放地址

        for (GetPlayInfoResponse.PlayInfo playInfo : playInfoList) {

            System.out.print("PlayInfo.PlayURL = " + playInfo.getPlayURL() + "\n");

        }

        //Base信息

        System.out.print("VideoBase.Title = " + response.getVideoBase().getTitle() + "\n");

    }

}

https://help.aliyun.com/document_detail/61064.html?spm=a2c4g.11186623.6.910.65487bdcUJ2Djx

4、获取视频播放凭证

public class VodSdkTest {

    public static void main(String[] args) throws Exception {

        //根据视频id获取视频播放凭证

        //初始化客户端、请求对象和相应对象

        DefaultAcsClient client = InitObject.initVodClient("LTAI4GKJXUR4VuFMrtnzzWBG","yiEPUWzpivYpYTG8o0tVW4UFqE2EQh");

        GetVideoPlayAuthResponse response = new GetVideoPlayAuthResponse();

        GetVideoPlayAuthRequest request = new GetVideoPlayAuthRequest();

        request.setVideoId("7d92de4882d44b24985862e5c368b195");

        response = client.getAcsResponse(request);

        System.out.print("PlayAuth = " + response.getPlayAuth() + "\n");

    }

}


03-文件上传测试

参考文档:https://help.aliyun.com/document_detail/53406.html

一、安装SDK

1、配置pom

2、安装非开源jar包

在本地Maven仓库中安装jar包:

下载视频上传SDK,解压,命令行进入lib目录,执行以下代码

mvn install:install-file-DgroupId=com.aliyun-DartifactId=aliyun-sdk-vod-upload-Dversion=1.4.11-Dpackaging=jar-Dfile=aliyun-java-vod-upload-1.4.11.jar

把该文件下的生成的包复制到工作目录下:

然后在pom中引入jar包,版本号已经在整个项目的pom中指定

        <dependency>

            <groupId>com.aliyun</groupId>

            <artifactId>aliyun-sdk-vod-upload</artifactId>

        </dependency>

二、测试

1、创建测试文件

2、测试本地文件上传

public class VodSdkTest {

    public static void main(String[] args) throws Exception {

        //本地文件上传接口

        String accessKeyId = "LTAI4GKJXUR4VuFMrtnzzWBG";

        String accessKeySecret = "yiEPUWzpivYpYTG8o0tVW4UFqE2EQh";

        String title = "6 - What If I Want to Move Faster-upload by sdk";//上传之后文件名称

        String fileName = "E:/6 - What If I Want to Move Faster.mp4";//本地文件路径和名称

        UploadVideoRequest request = new UploadVideoRequest(accessKeyId, accessKeySecret, title, fileName);

        /* 可指定分片上传时每个分片的大小,默认为2M字节 */

        request.setPartSize(2 * 1024 * 1024L);

        /* 可指定分片上传时的并发线程数,默认为1,(注:该配置会占用服务器CPU资源,需根据服务器情况指定)*/

        request.setTaskNum(1);

        UploadVideoImpl uploader = new UploadVideoImpl();

        UploadVideoResponse response = uploader.uploadVideo(request);

        if (response.isSuccess()) {

            System.out.print("VideoId=" + response.getVideoId() + "\n");

        } else {

            /* 如果设置回调URL无效,不影响视频上传,可以返回VideoId同时会返回错误码。其他情况上传失败时,VideoId为空,此时需要根据返回错误码分析具体错误原因 */

            System.out.print("VideoId=" + response.getVideoId() + "\n");

            System.out.print("ErrorCode=" + response.getCode() + "\n");

            System.out.print("ErrorMessage=" + response.getMessage() + "\n");

        }

    }



3 整合阿里云视频点播

01-视频点播微服务的创建

一、创建视频点播微服务

1、创建微服务模块

Artifact:service-vod

2、引入依赖

3、application.properties

# 服务端口

server.port=8003

# 服务名

spring.application.name=service-vod

# 环境设置:dev、test、prod

spring.profiles.active=dev

#阿里云 vod

#不同的服务器,地址不同

aliyun.vod.file.keyid=LTAI4GKJXUR4VuFMrtnzzWBG

aliyun.vod.file.keysecret=yiEPUWzpivYpYTG8o0tVW4UFqE2EQh

4、启动类

VodApplication.java

@SpringBootApplication(exclude = DataSourceAutoConfiguration.class)//在vod中,我们只是做视频的相关操作,并没有操作数据库

@ComponentScan(basePackages={"com.atguigu"})//包的扫描规则

public class VodApplication {

    public static void main(String[] args) {

        SpringApplication.run(VodApplication.class,args);

    }

}

二、整合阿里云vod实现视频上传

1、创建常量类

ConstantPropertiesUtil.java

@Component

public class ConstantPropertiesUtil implements InitializingBean {

    @Value("${aliyun.vod.file.keyid}")

    private String keyId;

    @Value("${aliyun.vod.file.keysecret}")

    private String keySecret;


    //定义公开静态常量

    public static String ACCESS_KEY_ID;

    public static String ACCESS_KEY_SECRET;

    @Override

    public void afterPropertiesSet() throws Exception {

        ACCESS_KEY_ID = keyId;

        ACCESS_KEY_SECRET = keySecret;

    }

}

2、创建controller、service

VodController.java

@Api(description="阿里云视频点播微服务")

@RestController

@RequestMapping("/eduvod/video")

@CrossOrigin

public class VodController {

    @Autowired

    private VodService vodService;

    //上传视频到阿里云

    @PostMapping("uploadAlyiVideo")

    public R uploadAlyiVideo(MultipartFile file){ //MultipartFile对象获取上传的文件

        //上传视频返回视频的id

        String videoId = vodService.uploadVideoAly(file);

        return R.ok().data("videoId",videoId);

    }

}

VodService.java

package com.atguigu.vod.service;

import org.springframework.web.multipart.MultipartFile;

public interface VodService {

    String uploadVideoAly(MultipartFile file);

}

VodServiceImpl.java

@Service

public class VodServiceImpl implements VodService {

    @Override

    public String uploadVideoAly(MultipartFile file) {

        try {

            //如01.MP4

            String fileName = file.getOriginalFilename();

            //01

            String title = fileName.substring(0,fileName.lastIndexOf("."));

            InputStream inputStream = file.getInputStream();

            UploadStreamRequest request = new UploadStreamRequest(ConstantPropertiesUtil.ACCESS_KEY_ID, ConstantPropertiesUtil.ACCESS_KEY_SECRET, title, fileName, inputStream);

            UploadVideoImpl uploader = new UploadVideoImpl();

            UploadStreamResponse response = uploader.uploadStream(request);

            String videoId = response.getVideoId(); //请求视频点播服务的请求ID

            if (response.isSuccess()) {

                videoId = response.getVideoId();

            } else { //如果设置回调URL无效,不影响视频上传,可以返回VideoId同时会返回错误码。其他情况上传失败时,VideoId为空,此时需要根据返回错误码分析具体错误原因

                videoId = response.getVideoId();

            }

            return videoId;

        } catch (IOException e) {

            e.printStackTrace();

            return null;

        }

    }

}

3、启动后端vod微服务

4、swagger测试 

org.springframework.web.multipart.MaxUploadSizeExceededException: Maximum upload size exceeded; nested exception is java.lang.IllegalStateException: org.apache.tomcat.util.http.fileupload.FileUploadBase$FileSizeLimitExceededException: The field file exceeds its maximum permitted size of 1048576 bytes.

解决办法:在配置文件中添加如下配置

# 最大上传单个文件大小:默认1M

spring.servlet.multipart.max-file-size=1024MB

# 最大置总上传的数据大小 :默认10M

spring.servlet.multipart.max-request-size=1024MB

测试:


02-前端整合视频上传

一、配置nginx反向代理

将接口地址加入nginx配置:

       location ~ /eduvod/ {

            proxy_pass http://localhost:8003;

        }

配置nginx上传文件大小,否则上传时会有 413 (Request Entity Too Large) 异常:

打开nginx主配置文件nginx.conf,找到http{},添加:

client_max_body_size 1024m;

重启nginx:

nginx.exe -s stop

二、前端实现

1、数据定义

fileList: [],//上传文件列表

BASE_API: process.env.BASE_API // 接口API地址

2、整合上传组件

         <el-form-item label="上传视频">

                <el-upload

                    :on-success="handleVodUploadSuccess"

                    :on-remove="handleVodRemove"

                    :before-remove="beforeVodRemove"

                    :on-exceed="handleUploadExceed"

                    :file-list="fileList"

                    :action="BASE_API+'/eduvod/video/uploadAlyiVideo'"

                    :limit="1"

                    class="upload-demo">

                <el-button size="small" type="primary">上传视频</el-button>

                <el-tooltip placement="right-end">

                    <div slot="content">最大支持1G,<br>

                        支持3GP、ASF、AVI、DAT、DV、FLV、F4V、<br>

                        GIF、M2T、M4V、MJ2、MJPEG、MKV、MOV、MP4、<br>

                        MPE、MPG、MPEG、MTS、OGG、QT、RM、RMVB、<br>

                        SWF、TS、VOB、WMV、WEBM 等视频格式上传</div>

                    <i class="el-icon-question"/>

                </el-tooltip>

                </el-upload>

            </el-form-item>

3、方法定义

//在上传接口中,上传成功后返回视频id(videoId)

//上传成功之后得到视频id,最终把视频id加到数据库中

 handleVodUploadSuccess(response, file, fileList){

            //上传id名称赋值

            this.video.videoSourceId = response.data.videoId

        },

//上传之前:设置一个提示

handleUploadExceed(files, fileList){

            this.$message.warning('想要重新上传视频,请先删除已上传的视频')

},

测试:

将视频名称存储到数据库表:

定义:

videoOriginalName:'', //视频名称

 //上传视频成功调用的方法,返回视频id因此需要response

        handleVodUploadSuccess(response, file, fileList){

            //上传id名称赋值

            this.video.videoSourceId = response.data.videoId

            //上传视频名称赋值

            this.video.videoOriginalName = file.name

        },

测试:


03-删除云端视频

文档:服务端SDK->Java SDK->媒资管理

https://help.aliyun.com/document_detail/61065.html?spm=a2c4g.11186623.6.831.654b3815cIxvma#h2--div-id-deletevideo-div-7

一、后端

VodController.java

//根据视频id删除阿里云视频

    @DeleteMapping("removeAlyVideo/{id}")

    public R removeAlyVideo(@PathVariable String id){

        try {

            //初始化对象

            DefaultAcsClient client = InitObjectV.initVodClient(ConstantPropertiesUtil.ACCESS_KEY_ID,ConstantPropertiesUtil.ACCESS_KEY_SECRET);

            //创建删除视频request对象

            DeleteVideoRequest request = new DeleteVideoRequest();

            //向request设置视频id

            request.setVideoIds(id);

            //调用初始化对象的方法实现删除

            client.getAcsResponse(request);

            return R.ok();

        } catch (ClientException e) {

            e.printStackTrace();

            throw new GuliException(20001,"删除视频失败");

        }

    }

二、前端

1、定义api

api/edu/video.js

    //根据视频id删除阿里云视频

    deleteAliyunvod(id){

        return request({

            url:`/eduvod/video/removeAlyVideo/${id}`,

            method:'delete'

        })

    }

2、组件方法

views/edu/course/chapter.vue

    handleVodRemove(file, fileList){

            video.deleteAliyunvod(this.video.videoSourceId)

                 .then(response=>{

                      this.$message({

                        type: 'success',  

                        message: "删除成功"

                    })

                 }) 

        },

        beforeVodRemove(file, fileList){

             return this.$confirm(`确定移除 ${file.name}?`)

        },

        视频点击删除,但是数据库中还是把视频名称和id加入进来,这是因为在过程中先做了上传,上传之后视频名称和id赋值到video对象中了,我们做的删除只是删除了阿里云中的视频,但是video中的数据还存在。

        解决:

  handleVodRemove(file, fileList){

            video.deleteAliyunvod(this.video.videoSourceId)

                 .then(response=>{

                      this.$message({

                        type: 'success',  

                        message: "删除成功"

                    })

                    //把文件列表清空

                    this.fileList = []

                    //把video视频id和视频名称值清空

                    //上传视频id赋值

                    this.video.videoSourceId = ''

                    //上传视频名称赋值

                    this.video.videoOriginalName = ''

                 }) 

        },


04-视频文件回显


       //方法中需要小节id

        openEditVideo(videoId){

            //弹框

            this.dialogVideoFormVisible = true

            //调用接口

            video.getVideo(videoId)

                .then(response=>{

                    this.video = response.data.video

                    // this.fileList = [{'name':this.video.videoOriginalName}]

                })

        },


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352