/QCloudUpload

Primary LanguageJavaScript

QCloud 视频上传

介绍

  • QCloudUpload.js为用于向腾讯微视频云服务器上传视频的SDK,提供简单的方法、接口用于向微视频服务器上传视频,并获得上传的结果数据。
  • 视频文件采用分片上传的方式。

依赖

依赖sha1.js。其实只要在页面上正确实现了用于计算sha1值的window.hex_sha1方法即可。上传时组件会将整部视频读入并尝试用window.hex_sha1函数计算读入文件的sha1值。并在上传第一片时将其传给视频服务器,当所有分片上传完成之后,服务器会比对开始传过去的sha1值和服务器自己算出来的sha1值是否相同,若不相同会返回错误信息:-174 ERROR_CMD_COS_SHA_NOT_EQU 文件SHA不一致

  • 支持断点续传

简单的例子

  • 所有支持的回调方法见文末
  • 下面的例子中,[xxx.getUploadArgs]方法请在业务中自行实现!(为了权限管理,微视频服务器要求每个视频都需要动态请求回一个有效的上传地址,参见微视频api,所以在业务中需要一个返回上传地址的接口。)
// 初始化实例
var uploader = new QCloudUpload();

if ( !uploader.isSupportUpload() ) {
    return;
}

// 视频添加进来后立即上传
uploader.on("addFile", function(file) {
    this.upload();
});

// 通过 [uploader.add] 方法添加视频
$("#file").on("change", function(e) {
    var file = e.target.files[0];

    xxx.getUploadArgs(function(uploadUrl) {

        uploader.add(file, uploadUrl); // 添加视频,触发onaddFile -> 视频将会被上传到地址[@uploadUrl]

    });
});


// ** 这里给出一个[xxx.getUploadArgs]方法的实现例子(基于jQusery):
xxx.getUploadArgs = function(callback) {
    $.ajax({
        url : "http://upload.narutowyh.com/ajax/getqcloudargs",
        type : "get",
        dataType : "json",
        data : {},
        success : function(r) {
            if (r && r.data && r.data.sign) {
                callback && callback( "http://web.video.myqcloud.com/files/v1/[appid]/" + r.data.bucketName + r.data.path + "?sign=" + encodeURIComponent(r.data.sign) );
            } else {
                window.console && console.error(">> 获取上传参数失败");
            }
        },
        error : function(err) {
            window.console && console.log(">> 获取上传参数失败");
        }
    });
}

视频信息的设置(若需要)

  • 文件可以配置的信息参见:微视频api
  • 传递方法:在<input type="file">onchange事件中将同名参数填入file对象中,例:
var uploader = new QCloudUpload();
uploader.on("addFile", function(file) {
    this.upload(); // 添加进来文件之后立即上传
});

// 这里以设置第一个文件的信息为例
$("#file").on("change", function(e) {
    var file = e.target.files[0];
    file.video_cover  = "http://xxxx/some.png";     // 设置视频封面
    file.video_title  = "母猪的产后护理.avi";         // 设置视频标题
    file.video_desc   = "母猪产后...blah blah blah"; // 设置视频描述
    file.magicContext = "http://xxxx";              // 设置用于透传回调用者的业务后台的字段

    xxx.getUploadArgs(function(uploadUrl) {
        uploader.add(file, uploadUrl); // 添加视频
    });
});

获取文件信息

  • 可以在on方法订阅的回调函数中获得的所有信息如下:
  • 注 : pieceHashdurationhash两个字段是组件填进file对象中的,只有在回调中才存在,在<input type="file">onchange中是没有的!
// this.state [上传状态]:
{
    uploading    : false,       // 上传中
    succeed      : false,       // 上传完毕
    speed        : 256000000,   // 上传速度(B/s)
    remainTime   : 116,         // 剩余时间(s)
    progress     : 75,          // 已上传75%,[0-100]
    uploadedData : 521000       // 已上传的体积(B)
}

// this.file [视频文件]
{
    pieceHash : "2354d4fadf45sdfsdd", // 文件前128B的hash值,用来唯一标记一个文件
    hash : "4a5sd4f6sdf4..." // 文件真正的hash值,(考虑到性能,只在)第一片上传时读取整个文件后才会填入,可用于调试(上传出错时便于获取到本地生成的文件的hash)
    duration : 24.02,        // 视频时长(s),基于video计算,同样只存在于回调中
    readProgress : 72,    // 文件已读入的进度,只保留整数,(%)
    name : "江北房源.mp4",
    size : 58943441,         // B
    ...(同file API)
}
  • 注意,被上传的文件中都包含pieceHash字段,此为一个视频的唯一标识(通过计算视频文件的前128B的数据生成的hash码),请通过它建立上传对象和DOM节点的联系,不要通过视频的文件名建立联系,因为文件名可能会有重复或不符合关键字定义等情况
  • 可以在on订阅的回调函数中调用的所有方法如下:
this.upload() // 上传
this.pause()  // 暂停上传
this.resume() // 继续上传
this.cancle() // 取消上传
this.isExist() // 当前文件是否被add过

实例属性

  • fileList

为一个数组,数组元素为add进来的文件,文件包含pieceHash字段,无文件时为[]

  • uploaderList

为一个对象,对象元素为uploader对象的实例,通过pieceHash值与fileList中的文件一一对应,为空时为{}. 可以通过此属性配合文件的pieceHash属性控制一个文件的暂停、继续和取消等

实例方法

isSupportUpload

  • 判断当前环境能否启用视频上传
  • 返回值: true / false

add: 将选中的文件添加到上传列表

  • 参数必需是文件,例如<input type="file">元素onchange后的e.target.files[0]
  • add过后会触发onaddFile事件
var uploader = new QCloudUpload();

uploader.on("addFile", function(file) {

    // 文件小于300MB则立即上传
    if ( file.size <= 300 * 1024 * 1024 ) {
        this.upload();
    }
});

// 同时选择多部视频(<input type="file" multiple>)
$("#file").on("change", function(e) {
    var files = e.target.files;

    for (var i =0, len = files.length; i < len; ++i) {
        (function(index) {
            var file = files[index];

            xxx.getUploadArgs(function(uploadUrl) {
                uploader.add(file, uploadUrl);
            });
        })(i);
    }

});

cancle: 移除@pieceHash文件

  • 参数: 用于唯一标识视频的[pieceHash]字段
  • 如果被操作的文件正在上传,会停止上传并将其移除
  • 不传參数时移除所有上传
  • 移除后,文件及其上传进程都会被删掉,不可逆
  • 触发oncancle
var uploader = new QCloudUpload();
var file_piece_hash = "";

uploader.on("addFile", function(file) {
    file_piece_hash = file.pieceHash;
    this.upload(); // 将添加进来的文件立即上传
});

uploader.on("uploadCancle", function() {
    alert("已取消" + this.file.name + "文件的上传!");
});

$("#file").on("change", function(e) {
    var file = e.target.files[0];

    xxx.getUploadArgs(function(uploadUrl) {
        uploader.add(file, uploadUrl);
    });
});

// 10s未上传完成则取消文件的上传
setTimeout(function() {
    if ( !uploader.uploaderList.file_piece_hash.state.succeed ) {
        uploader.cancle();
    }
}, 10000);

isAllUploaded: 监测是否所有文件都已经上传完毕

  • 参数:
  • return true or false
  • 全部上传完毕的操作建议放在onallCompleted事件中进行

upload: 上传@pieceHash文件

  • 参数: 用于唯一标识视频的[pieceHash]字段
  • 不传參数时上传所有文件
  • 若文件未暂停状态则继续上传
  • 触发onuploadStart

pause: 暂停@pieceHash文件上传

  • 参数: 用于唯一标识视频的[pieceHash]字段
  • 不传參数时暂停所有上传
  • 触发onpause

resume: 让被暂停的@pieceHash文件继续上传

  • 参数: 用于唯一标识视频的[pieceHash]字段
  • 不传參数时全部继续上传
  • 正在上传中或已上传完毕的文件不受控制
  • 触发onresume

所有支持订阅的方法名

  • 考虑到上传逻辑,每个方法名只能订阅一次!,重复订阅会抛出错误
  • 通过执行实例的on方法可以绑定以下回调函数,绑定的方法会在相应的时机执行,并带入有用的参数
  • 通过调用off方法将绑定的回调解绑
  • 通过trigger方法手动触发绑定过的回调,此方法一般为类内部使用,不推荐手动调用
  • fileReadStartfileReadProgressfileReadEndfileReadError四个跟文件读取相关的事件会在实例的upload方法之后自动依次触发。因为考虑到读文件耗费较大性能,所以只有在upload方法被调用时,组件才会将文件读入,四个fileReader才会随即依次触发。
方法名 回调参数 执行时机 return false时
addFile ( file ) 被添加的文件 文件被添加到上传列表时执行,
pieceHash 字段被填入
此文件将会被忽略
fileReadError ( reader ) reader对象的引用 文件读取失败时执行,
文件不会被添加到上传列表
--
fileReadStart ( reader ) reader对象的引用 文件开始读取时执行 不会继续读取整个文件。
后续仍然可以对其执行upload
fileReadProgress ( progress ) progress对象的引用 文件读取中持续执行,
progress.lengthComputabletrue
可以拿到progress.loaded, progress.total
取消本次读取,后续仍然可以对其执行upload
fileReadEnd ( reader ) reader对象的引用 文件读取完毕时执行 此文件不会上传,
后续仍然可以对其执行upload
uploadStart ( ) 第一片的数据成功返回后,
文件开始上传时执行
暂停上传,后续仍然可以对其执行upload
uploadPause ( ) 文件暂停上传时执行 --
uploadResume ( ) 文件恢复上传时执行 --
uploadCancle ( ) 文件被移除时执行 取消移除操作
uploadProgress ( ) 文件上传过程中持续执行,
可用来动态显示上传进度、网速、剩余时间等
--
uploadEnd ( result ) 上传成功后服务器返回的结果 文件上传成功时执行 --
uploadError ( ) 文件上传过程中出错时执行,
可能发生在uploadStart未成功时
uploadProgress过程中
--
allCompleted () 所有文件上传成功时执行 --

其它

文档

断点续传

  • 断点续传的功能组件已封装好,用户再次上传曾上传过的视频时会自动从相应进度开始上传
  • 原理:微云服务器会自动判断要上传的视频是不是可以断点续传的视频,并在第一次分片请求回的offset字段中自动将offset设置好,后续分片从这个offset处开始取即可
  • 所以:刚进页面时无法获取上传过的视频和关闭页面时正在上传的视频及其进度,只有用户选择了某个视频并开始上传之后才能从服务器获取到这些信息
  • 已上传过的视频会被秒传

兼容性

兼容浏览器 ?

  • chrome
  • 360极速模式
  • 火狐版本
  • IE版本 IE10 PR2不支持readAsBinaryString() 和 readAsArrayBuffer(),所以不支持上传视频 ????
  • safari版本
  • opera
  • 搜狗
  • ...

不兼容的处理方法 ?

不兼容浏览器出现的问题 ?

限制

  • 将视频时长的计算方式兼容性待查文档
  • 文件过大时服务器会返回错误: -181 ERROR_CMD_COS_ERROR 存储后端错误(上限:300MB,iphone6p录制3分50秒)
  • 多个窗口/tab同时上传统一部视频时:其中一个会引发错误(分片索引错误),并且不可以续传。另一个窗口中的上传可以一直正常执行。

who use?