cos-js-sdk-v5
Version:
JavaScript SDK for [腾讯云对象存储](https://cloud.tencent.com/product/cos)
1,557 lines (1,516 loc) • 197 kB
JavaScript
/**
* @jest-environment jsdom
*/
import { describe, expect, jest, test } from '@jest/globals';
import COS from '../index.js';
import Beacon from '../demo/common/beacon.min';
import ClsClient from '../demo/common/cls.min';
// config 替换成自己的桶信息
var config = {
SecretId: process.env.SecretId,
SecretKey: process.env.SecretKey,
Bucket: process.env.Bucket, // 需提前创建并设置跨域
Region: process.env.Region,
ReplicationBucket: process.env.ReplicationBucket, // 存储桶复制时用到的桶,需提前创建并设置跨域
ReplicationRegion: process.env.ReplicationRegion, // 存储桶复制时用到的桶的地域
Uin: process.env.Uin,
StsUrl: process.env.jssdkStsUrl,
};
// mock localStroage
var localStorageMock = (function () {
var store = {};
return {
getItem: function (key) {
return store[key];
},
setItem: function (key, value) {
store[key] = value.toString();
},
clear: function () {
store = {};
},
removeItem: function (key) {
delete store[key];
},
};
})();
Object.defineProperty(window, 'localStorage', {
writable: true,
configurable: true,
value: localStorageMock,
});
function checkEnvParams() {
if (!process.env.Bucket) {
console.warn('环境变量里未找到Bucket,请检查');
return;
}
if (!process.env.Region) {
console.warn('环境变量里未找到Region,请检查');
return;
}
if (!process.env.ReplicationBucket) {
console.warn('环境变量里未找到ReplicationBucket,请检查');
return;
}
if (!process.env.ReplicationRegion) {
console.warn('环境变量里未找到ReplicationRegion,请检查');
return;
}
if (!process.env.Uin) {
console.warn('环境变量里未找到Uin,请检查');
return;
}
return true;
}
var util = {
createFile: function (options) {
var buffer = new ArrayBuffer(options.size || 0);
var arr = new Uint8Array(buffer);
for (var i = 0; i < arr.length; i++) {
arr[i] = 0;
}
var opt = {};
options.type && (opt.type = options.type);
var blob = new Blob([buffer], options);
var file = new File([blob], `file-${Date.now()}`, {
type: options.type,
lastModified: Date.now(),
lastModifiedDate: new Date(),
});
file.lastModifiedDate = new Date();
return file;
},
str2blob: function (str) {
var size = str.length;
var buffer = new ArrayBuffer(size || 0);
var arr = new Uint8Array(buffer);
for (var i = 0; i < arr.length; i++) {
arr[i] = str[i];
}
var blob = new Blob([buffer]);
return blob;
},
};
function camSafeUrlEncode(str) {
return encodeURIComponent(str)
.replace(/!/g, '%21')
.replace(/'/g, '%27')
.replace(/\(/g, '%28')
.replace(/\)/g, '%29')
.replace(/\*/g, '%2A');
}
function comparePlainObject(a, b) {
if (Object.keys(a).length !== Object.keys(b).length) {
return false;
}
for (var key in a) {
if (typeof a[key] === 'object' && typeof b[key] === 'object') {
if (!comparePlainObject(a[key], b[key])) {
return false;
}
} else if (a[key] != b[key]) {
return false;
}
}
return true;
}
function prepareBigObject(needHeaders) {
return new Promise(function (resolve, reject) {
// 创建测试文件
var filename = name || 'bigger.zip';
var content = util.createFile({ size: 1024 * 1024 * 10 });
var put = function () {
// 调用方法
var params = {
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: content,
ContentLength: content.length,
};
if (needHeaders) {
params.ContentType = 'text/html';
params.CacheControl = 'max-age=7200';
params.ContentDisposition = 'inline;filename=hello.jpg';
params.Expires = new Date().toGMTString();
params.Headers = {
'x-cos-meta-test': 'xxx',
};
}
cos.putObject(params, function (err, data) {
err ? reject(err) : resolve();
});
};
put();
});
}
var createFileSync = function (size) {
return util.createFile({ size: size });
};
var dataURItoUploadBody = function (dataURI) {
var byteString = atob(dataURI.split(',')[1]);
var mimeString = dataURI.split(',')[0].split(':')[1].split(';')[0];
var ab = new ArrayBuffer(byteString.length);
var ia = new Uint8Array(ab);
for (var i = 0; i < byteString.length; i++) {
ia[i] = byteString.charCodeAt(i);
}
return new Blob([ab], { type: mimeString });
};
var getAuthorization = function (options, callback) {
var authorization = COS.getAuthorization({
SecretId: process.env.SecretId,
SecretKey: process.env.SecretKey,
Method: options.Method,
Pathname: options.Pathname,
Query: options.Query,
Headers: options.Headers,
Expires: 900,
});
callback({
Authorization: authorization,
});
};
var cos = new COS({
// 必选参数
SecretId: config.SecretId,
SecretKey: config.SecretKey,
UploadCheckContentMd5: true,
UploadAddMetaMd5: true,
Protocol: 'http:',
// getAuthorization: getAuthorization,
});
console.log('config.StsUrl========', config.StsUrl);
function getSts(options) {
return new Promise((resolve, reject) => {
var url = `${config.StsUrl}/sts`; // 如果是 npm run sts.js 起的 nodejs server,使用这个
var xhr = new XMLHttpRequest();
xhr.open('get', url, true);
xhr.setRequestHeader('Content-Type', 'application/json');
xhr.onload = function (e) {
var data,credentials;
try {
data = JSON.parse(e.target.responseText);
credentials = data.credentials;
} catch (e) {}
if (!data || !credentials) {
console.error('credentials invalid:\n' + JSON.stringify(data, null, 2));
reject();
}
resolve({
TmpSecretId: credentials.tmpSecretId,
TmpSecretKey: credentials.tmpSecretKey,
SecurityToken: credentials.sessionToken,
StartTime: data.startTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: data.expiredTime, // 时间戳,单位秒,如:1580000000
});
};
xhr.send(JSON.stringify(options?.Scope));
});
}
// todo 只能这么提前获取
const sts = {};
getSts().then(data => {
Object.assign(sts, data);
});
// 使用临时密钥
var tempCOS = new COS({
getAuthorization: async function (options, callback) {
try {
const res = await getSts(options);
callback({
TmpSecretId: res.TmpSecretId,
TmpSecretKey: res.TmpSecretKey,
SecurityToken: res.SecurityToken,
StartTime: res.StartTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: res.ExpiredTime, // 时间戳,单位秒,如:1580000000
ScopeLimit: true, // 细粒度控制权限需要设为 true,会限制密钥只在相同请求时重复使用
});
} catch (e) {
console.error('get sts error');
}
},
});
// 使用临时密钥(老版本使用的XCosSecurityToken)
var oldTempCOS = new COS({
// UseAccelerate: true,
getAuthorization: async function (options, callback) {
try {
const res = await getSts(options);
callback({
TmpSecretId: res.TmpSecretId,
TmpSecretKey: res.TmpSecretKey,
XCosSecurityToken: res.SecurityToken,
StartTime: res.StartTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: res.ExpiredTime, // 时间戳,单位秒,如:1580000000
ScopeLimit: true, // 细粒度控制权限需要设为 true,会限制密钥只在相同请求时重复使用
});
} catch (e) {
console.error('get sts error');
}
},
});
// 后端下发 putObject 前面 Key 为 1.txt
var getSignCOS = new COS({
// UseAccelerate: true,
getAuthorization: function (options, callback) {
var url = `${config.StsUrl}/uploadSign`; // 如果是 npm run sts.js 起的 nodejs server,使用这个
var xhr = new XMLHttpRequest();
xhr.open('get', url, true);
xhr.setRequestHeader('Content-Type', 'application/json');
xhr.onload = function (e) {
try {
var data = JSON.parse(e.target.responseText);
} catch (e) {}
if (!data) {
return console.error('credentials invalid:\n' + JSON.stringify(data, null, 2));
}
callback({
Authorization: data?.signMap?.PutObject,
});
};
xhr.send();
},
});
var getStsCOS = new COS({
// UseAccelerate: true,
getSTS: async function (options, callback) {
try {
const res = await getSts(options);
callback({
TmpSecretId: res.TmpSecretId,
TmpSecretKey: res.TmpSecretKey,
XCosSecurityToken: res.SecurityToken,
StartTime: res.StartTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: res.ExpiredTime, // 时间戳,单位秒,如:1580000000
ScopeLimit: true, // 细粒度控制权限需要设为 true,会限制密钥只在相同请求时重复使用
});
} catch (e) {
console.error('get sts error');
}
},
});
// 临时密钥允许的路径
var tempCOSPrefix = 'js-sdk/test/';
var Bucket = config.Bucket;
var BucketShortName = Bucket;
var TaskId;
var AppId;
var BucketLongName = Bucket + '-' + AppId;
var match = config.Bucket.match(/^(.+)-(\d+)$/);
if (match) {
BucketLongName = config.Bucket;
BucketShortName = match[1];
AppId = match[2];
}
var group = function (name, fn) {
if (!checkEnvParams()) return;
console.log(`${name}进行中....`);
describe(name, function () {
jest.setTimeout(1 * 60 * 1000);
fn.apply(this, arguments);
});
};
var assert = {
ok: function (val) {
expect(Boolean(val)).toBeTruthy();
},
};
var base64Url =
'';
var dataURLtoBlob = function (dataurl) {
var arr = dataurl.split(',');
var mime = arr[0].match(/:(.*?);/)[1];
var bstr = atob(arr[1]);
var n = bstr.length;
var u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
return new Blob([u8arr], { type: mime });
};
group('init cos', function () {
const putFile = function (cosIns, done, canSuccess = true) {
var key = '1.txt';
var content = Date.now().toString();
cosIns.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: key,
Body: content,
Headers: {
'x-cos-test': '1',
'x-cos-meta-test': 'meta',
},
'x-cos-test2': '2',
},
function (err, data) {
assert.ok(canSuccess ? !err : err);
done();
}
);
};
test('使用AppId', function (done) {
var initCos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
AppId: AppId,
});
putFile(initCos, done);
});
test('SecretId格式错误', function (done) {
var initCos = new COS({
SecretId: config.SecretId + ' ',
SecretKey: config.SecretKey,
});
putFile(initCos, done, false);
});
test('SecretKey格式错误', function (done) {
var initCos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey + ' ',
});
putFile(initCos, done, false);
});
test('Timeout=6000', function (done) {
var initCos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
Timeout: 6000,
});
putFile(initCos, done);
});
test('ForcePathStyle', function (done) {
try {
var initCos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
ForcePathStyle: true,
});
putFile(initCos, done, false);
} catch (e) {
assert.ok(e.message === 'ForcePathStyle is not supported');
done();
}
});
test('getAuthorization error tmpSecretId', function (done) {
var initCos = new COS({
getAuthorization: function (options, callback) {
callback({
tmpSecretId: config.SecretId,
TmpSecretKey: config.SecretKey,
});
},
});
putFile(initCos, done, false);
});
test('getAuthorization error tmpSecretKey', function (done) {
var initCos = new COS({
getAuthorization: function (options, callback) {
callback({
TmpSecretId: config.SecretId,
tmpSecretKey: config.SecretKey,
});
},
});
putFile(initCos, done, false);
});
test('getAuthorization error', function (done) {
var initCos = new COS({
getAuthorization: function (options, callback) {
callback({
TmpSecretId: config.SecretId,
TmpSecretKey: config.SecretKey,
});
},
});
putFile(initCos, done, false);
});
test('getAuthorization 使用临时密钥 putObject', function (done) {
tempCOS.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: tempCOSPrefix + Date.now().toString(36),
Body: '12345',
},
function (err, data) {
assert.ok(!err);
done();
}
);
});
test('getStsCOS 使用临时密钥 putObject', function (done) {
getStsCOS.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: tempCOSPrefix + Date.now().toString(36),
Body: '12345',
},
function (err, data) {
assert.ok(!err);
done();
}
);
});
test('getAuthorization 使用临时密钥 sliceUploadFile', function (done) {
const file = createFileSync(20 * 1024 * 1024);
oldTempCOS.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: tempCOSPrefix + Date.now().toString(36),
Body: file,
},
function (err, data) {
assert.ok(!err);
done();
}
);
});
test('getAuthorization 使用临时密钥 sliceUploadFile 没有权限', function (done) {
const file = createFileSync(20 * 1024 * 1024);
tempCOS.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: Date.now().toString(36),
Body: file,
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('getStsCOS 使用下发的签名 putObject', function (done) {
getSignCOS.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: '1.txt',
Body: '12345',
},
function (err, data) {
assert.ok(!err);
done();
}
);
});
test('getAuthorization 使用下发的签名 sliceUploadFile 没权限', function (done) {
const file = createFileSync(20 * 1024 * 1024);
getSignCOS.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: '1.txt',
Body: file,
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('推荐初始化方式 putObject 成功', function (done) {
const cos = new COS({
SecretId: sts.TmpSecretId,
SecretKey: sts.TmpSecretKey,
SecurityToken: sts.SecurityToken,
StartTime: sts.StartTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: sts.ExpiredTime, // 时间戳,单位秒,如:1580000000
});
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: tempCOSPrefix + '1.txt',
Body: '12345',
},
function (err, data) {
console.log(' putObject 成功', err || data);
assert.ok(!err);
done();
}
);
});
test('推荐初始化方式 putObject invalid StartTime', function (done) {
const cos = new COS({
SecretId: sts.TmpSecretId,
SecretKey: sts.TmpSecretKey,
SecurityToken: sts.SecurityToken,
StartTime: sts.StartTime * 1000, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: sts.ExpiredTime, // 时间戳,单位秒,如:1580000000
});
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: tempCOSPrefix + '1.txt',
Body: '12345',
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('推荐初始化方式 putObject invalid ExpiredTime', function (done) {
const cos = new COS({
SecretId: sts.TmpSecretId,
SecretKey: sts.TmpSecretKey,
SecurityToken: sts.SecurityToken,
StartTime: sts.StartTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: sts.ExpiredTime * 1000, // 时间戳,单位秒,如:1580000000
});
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: tempCOSPrefix + '1.txt',
Body: '12345',
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('推荐初始化方式 getObjectUrl invalid StartTime', function (done) {
const cos = new COS({
SecretId: sts.TmpSecretId,
SecretKey: sts.TmpSecretKey,
SecurityToken: sts.SecurityToken,
StartTime: sts.StartTime * 1000, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
ExpiredTime: sts.ExpiredTime, // 时间戳,单位秒,如:1580000000
});
cos.getObjectUrl(
{
Bucket: config.Bucket,
Region: config.Region,
Key: tempCOSPrefix + '1.txt',
Expires: Date.now() * 1000,
},
function (err, data) {
console.log('getObjectUrl invalid ExpiredTime', err||data);
assert.ok(err);
done();
}
);
});
});
group('task 队列', function () {
test('putObject() 批量上传', function (done) {
var cos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
UploadQueueSize: 100,
});
var upload = function () {
var filename = '5.txt';
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: '12345',
},
function (err, data) {}
);
};
for (var i = 0; i < 120; i++) {
upload();
}
var taskList = cos.getTaskList();
const isUploading = cos.isUploadRunning();
assert.ok(isUploading);
done();
});
test('putObject(),update-list()', function (done) {
var filename = '10m.zip';
const file = createFileSync(10 * 1024 * 1024);
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: file,
},
function (err, data) {
assert.ok(!err);
done();
}
);
cos.on('task-list-update', function () {});
});
});
group('兼容性测试', function () {
test('getBucketACL 老用法', function (done) {
cos.getBucketACL(
{
Bucket: config.Bucket,
Region: config.Region,
},
function (err, data) {
assert.ok(!err);
done();
}
);
});
});
group('getService()', function () {
var cos = new COS({
// 必选参数
SecretId: config.SecretId,
SecretKey: config.SecretKey,
UploadCheckContentMd5: true,
UploadAddMetaMd5: true,
Protocol: 'http:',
ServiceDomain: 'service.cos.myqcloud.com/',
});
test('getService 老用法', function (done) {
cos.getService(function (err, data) {
assert.ok(err);
done();
});
});
test('getService 传Region', function (done) {
var cos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
});
cos.getService(
{
Region: config.Region,
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('getService 不传Region和Domain', function (done) {
var cos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
});
cos.getService({}, function (err, data) {
assert.ok(err);
done();
});
});
test('不能正常列出 Bucket', function (done) {
cos.getService(
{
Region: config.Region,
},
function (err, data) {
assert.ok(err);
done();
}
);
});
});
group('putBucket()', function () {
var NewBucket = 'test' + Date.now().toString(36) + '-' + AppId;
test('创建 bucket js sdk因为跨域会失败', function (done) {
cos.putBucket(
{
Bucket: NewBucket,
Region: config.Region,
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('正常创建 bucket BucketAZConfig js sdk因为跨域会失败', function (done) {
cos.putBucket(
{
Bucket: NewBucket,
Region: config.Region,
BucketAZConfig: {},
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('deleteBucket() deleteBucket 不存在', function (done) {
cos.deleteBucket(
{
Bucket: Date.now().toString(36) + config.Bucket,
Region: config.Region,
},
function (err, data) {
assert.ok(err, 'deleteBucket 不存在');
done();
}
);
});
});
group('mock readAsBinaryString', function () {
test('mock readAsBinaryString', function (done) {
FileReader.prototype._readAsBinaryString = FileReader.prototype.readAsBinaryString;
FileReader.prototype.readAsBinaryString = false;
var filename = '10m.zip';
var blob = util.createFile({ size: 1024 * 1024 * 10 });
var paused = false;
cos.sliceUploadFile({
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
if (!paused && info.percent > 0.6) {
cos.cancelTask(TaskId);
var hasProgress = false;
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
if (info.percent === 0) return;
expect(info.percent).toBeGreaterThanOrEqual(0.3);
cos.cancelTask(TaskId);
FileReader.prototype.readAsBinaryString = FileReader.prototype._readAsBinaryString;
delete FileReader.prototype._readAsBinaryString;
done();
},
},
function (err) {
if (hasProgress) {
done();
}
}
);
}
},
});
});
});
group('getAuth()', function () {
test('getAuth()', function (done) {
var content = Date.now().toString();
var key = '1.txt';
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: key,
Body: content,
},
function (err, data) {
if (err) {
done();
return;
}
let AuthData = cos.getAuth({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
Method: 'get',
Key: key,
Scope: [
{
action: 'GetObject',
bucket: config.Bucket,
region: config.Region,
prefix: key,
},
],
});
if (typeof AuthData === 'string') {
AuthData = { Authorization: AuthData };
}
if (!AuthData.Authorization) {
AuthData.Authorization = COS.getAuthorization({
SecretId: AuthData.TmpSecretId,
SecretKey: AuthData.TmpSecretKey,
Method: 'get',
Key: key,
SystemClockOffset: cos.options.SystemClockOffset,
});
}
var link =
'http://' +
config.Bucket +
'.cos.' +
config.Region +
'.myqcloud.com' +
'/' +
camSafeUrlEncode(key).replace(/%2F/g, '/') +
'?' +
AuthData.Authorization +
(AuthData.SecurityToken ? '&x-cos-security-token=' + AuthData.SecurityToken : '');
done();
}
);
});
});
group('getObjectUrl()', function () {
test('getObjectUrl()', function (done) {
var content = Date.now().toString();
var key = '1.txt';
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: key,
Body: content,
},
function (err, data) {
cos.getObjectUrl(
{
Bucket: config.Bucket,
Region: config.Region,
Key: key,
Query: {
a: 1,
},
Sign: true,
Expires: 8000000,
},
function (err, data) {
expect(typeof data.Url).toBe('string');
err ? done(err) : done();
}
);
tempCOS.getObjectUrl(
{
Bucket: config.Bucket,
Region: config.Region,
Key: key,
Query: {
a: 1,
},
Sign: false,
},
function (err, data) {
expect(typeof data.Url).toBe('string');
err ? done(err) : done();
}
);
}
);
});
});
group('auth check', function () {
test('auth check', function (done) {
cos.getBucket(
{
Bucket: config.Bucket,
Region: config.Region,
Prefix: 'aksjhdlash sajlhj!@#$%^&*()_+=-[]{}\';:"/.<>?.,??sadasd#/.,/~`',
Headers: {
'x-cos-test': 'aksjhdlash sajlhj!@#$%^&*()_+=-[]{}\';:"/.<>?.,??sadasd#/.,/~`',
},
},
function (err, data) {
err ? done(err) : done();
}
);
});
});
group('getBucket(),listObjectVersions()', function () {
test('正常获取 bucket 里的文件列表', function (done) {
cos.getBucket(
{
Bucket: config.Bucket,
Region: config.Region,
},
function (err, data) {
expect(data.Name).toBe(BucketLongName);
expect(data.Contents).toBeInstanceOf(Array);
err ? done(err) : done();
}
);
});
test('正常获取 bucket 里的文件版本列表', function (done) {
cos.listObjectVersions(
{
Bucket: config.Bucket,
Region: config.Region,
},
function (err, data) {
expect(data.Name).toBe(config.Bucket);
expect(data.Versions).toBeInstanceOf(Array);
err ? done(err) : done();
}
);
});
});
group('putObject(),cancelTask()', function () {
test('putObject(),cancelTask()', function (done) {
var filename = '10m.zip';
var alive = false;
var canceled = false;
const file = createFileSync(10 * 1024 * 1024);
cos.putObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: file,
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
alive = true;
if (!canceled) {
cos.cancelTask(TaskId);
alive = false;
canceled = true;
setTimeout(function () {
expect(alive).toBe(false);
done();
}, 1200);
}
},
},
function (err, data) {
alive = true;
err ? done(err) : done();
}
);
});
});
group('putObject 测试老参数', function () {
test('putObject() options.AppId', function (done) {
var filename = '/1m.zip';
const file = createFileSync(1 * 1024 * 1024);
var cos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
AppId,
UseRawKey: true,
});
cos.putObject(
{
Bucket: BucketShortName,
Region: config.Region,
Key: filename,
Body: file,
},
function (err, data) {
assert.ok(!err);
done();
}
);
});
test('putObject() options.CompatibilityMode', function (done) {
var filename = '/1m.zip';
const file = createFileSync(1 * 1024 * 1024);
var cos = new COS({
SecretId: config.SecretId,
SecretKey: config.SecretKey,
CompatibilityMode: true,
});
cos.putObject(
{
Bucket: BucketShortName,
Region: config.Region,
Key: filename,
Body: file,
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('putObject() BucketShortName', function (done) {
var filename = '/1m.zip';
const file = createFileSync(1 * 1024 * 1024);
cos.putObject(
{
Bucket: BucketShortName,
AppId: AppId,
Region: config.Region,
Key: filename,
Body: file,
},
function (err, data) {
assert.ok(!err);
done();
}
);
});
test('putObject() error Body', function (done) {
cos.putObject(
{
Bucket: BucketShortName,
AppId: AppId,
Region: config.Region,
Key: COS.util.encodeBase64('转base64', true),
Body: { a: 1 },
},
function (err, data) {
assert.ok(err);
done();
}
);
});
test('putObject() missing Key', function (done) {
try {
cos.putObject(
{
Bucket: BucketShortName,
AppId: AppId,
Region: config.Region,
Body: file,
},
function (err, data) {}
);
} catch (e) {
assert.ok(e.message === 'file is not defined');
done();
}
});
});
group('sliceUploadFile() 完整上传文件', function () {
test('sliceUploadFile() 完整上传文件', function (done) {
var lastPercent;
var filename = '3m.zip';
var fileSize = 1024 * 1024 * 3;
var blob = createFileSync(fileSize);
cos.abortUploadTask(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Level: 'file',
},
function (err, data) {
if (err) {
done();
return;
}
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
Headers: {
'x-cos-traffic-limit': 8192000000,
},
onTaskReady: function (taskId) {
console.log(taskId);
},
onProgress: function (info) {
lastPercent = info.percent;
},
},
function (err, data) {
console.log('sliceUploadFile', err ? 'failed' : 'success');
if (err) {
done();
return;
}
cos.headObject(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
},
function (err, data) {
console.log('headObject', err ? 'failed' : 'success');
const success = data && data.headers && parseInt(data.headers['content-length'] || 0) === fileSize;
console.log(`data.headers['content-length']`, data.headers['content-length'], fileSize, success);
expect(success);
done();
}
);
}
);
}
);
});
test('sliceUploadFile(),pauseTask(),restartTask()', function (done) {
var filename = Date.now().toString() + '-10m.zip';
var blob = util.createFile({ size: 1024 * 1024 * 10 });
var paused = false;
var updateFn = function (info) {
var task = info.list.find((item) => item.Key === filename);
if (task && task.state === 'success') {
console.log('任务成功');
cos.off('list-update', updateFn);
assert.ok(1);
done();
}
};
cos.abortUploadTask(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Level: 'file',
},
function (err, data) {
if (err) {
done();
return;
}
var TaskId;
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
if (!paused && info.percent >= 0.3) {
cos.pauseTask(TaskId);
paused = true;
console.log('任务暂停');
cos.on('list-update', updateFn);
setTimeout(function () {
if (paused) {
console.log('任务重启');
cos.restartTask(TaskId);
}
}, 10);
}
},
},
function (err, data) {
assert.ok(1);
done();
}
);
}
);
});
test('sliceUploadFile(),cancelTask(),restartTask()', function (done) {
var filename = '10m.zip';
var blob = util.createFile({ size: 1024 * 1024 * 3 });
var paused = false;
cos.abortUploadTask(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Level: 'file',
},
function (err, data) {
var TaskId;
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
if (!paused && info.percent > 0.6) {
cos.cancelTask(TaskId);
setTimeout(function () {
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
},
function (err, data) {
err ? done(err) : done();
}
);
}, 10);
}
},
},
function (err, data) {}
);
}
);
});
test('sliceUploadFile(),cancelTask()', function (done) {
var filename = '3m.zip';
var alive = false;
var canceled = false;
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: util.createFile({ size: 1024 * 1024 * 3 }),
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
alive = true;
if (!canceled) {
cos.cancelTask(TaskId);
alive = false;
canceled = true;
setTimeout(function () {
done();
}, 1200);
}
},
},
function (err, data) {
alive = true;
}
);
});
});
group('sliceUploadFile() 同时上传2个文件', function () {
var filename = '30m.zip';
var blob = util.createFile({ size: 1024 * 1024 * 30 });
test('sliceUploadFile() 上传文件1', function (done) {
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
},
function (err, data) {
assert.ok(!err);
done();
}
);
setTimeout(() => {
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
},
function (err, data) {
assert.ok(!err);
done();
}
);
}, 2000);
});
});
group('sliceUploadFile() 续传', function () {
var filename = '30m.zip';
var blob = util.createFile({ size: 1024 * 1024 * 30 });
test('sliceUploadFile() 正常续传', function (done) {
var taskId;
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (id) {
taskId = id;
},
onProgress: function (progressData) {
if (progressData.percent >= 0.3) {
cos.pauseTask(taskId);
console.log('pause task');
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
TaskReady: function (id) {
console.log('TaskReady', id);
},
},
function (err, data) {
assert.ok(!err);
done();
}
);
}
},
},
function (err, data) {}
);
});
test('sliceUploadFile() cancelTask', function (done) {
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (id) {
setTimeout(() => {
cos.cancelTask(id);
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
},
function (err, data) {
assert.ok(!err);
done();
}
);
}, 2000);
},
},
function (err, data) {}
);
});
test('sliceUploadFile() 续传时远程为 0', function (done) {
var taskId;
var paused = false;
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (id) {
taskId = id;
},
onProgress: function (progressData) {
if (!paused && progressData.percent >= 0.3) {
cos.pauseTask(taskId);
paused = true;
setTimeout(() => {
if (paused) {
var blob = util.createFile({ size: 1024 * 1024 * 20 });
cos.sliceUploadFile(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
},
function (err, data) {
assert.ok(!err);
done();
}
);
}
}, 1000);
}
},
},
function (err, data) {}
);
});
});
group('abortUploadTask()', function () {
test('abortUploadTask(),Level=task', function (done) {
var filename = '1m.zip';
cos.multipartInit(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
},
function (err, data) {
cos.abortUploadTask(
{
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Level: 'task',
UploadId: data.UploadId,
},
function (err, data) {
var nameExist = false;
data.successList.forEach(function (item) {
if (filename === item.Key) {
nameExist = true;
}
});
assert.ok(data.successList.length >= 1);
assert.ok(nameExist);
err ? done(err) : done();
}
);
}
);
});
test('abortUploadTask(),Level=file', function (done) {
var filename = '1m.zip';
var blob = util.createFile({ size: 1024 * 1024 });
cos.sliceUploadFile({
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
cos.cancelTask(TaskId);
cos.abortUploadTask(
{
Bucket: config.Bucket,
Region: config.Region,
Level: 'file',
Key: filename,
},
function (err, data) {
assert.ok(data.successList.length >= 1);
assert.ok(data.successList[0] && data.successList[0].Key === filename);
err ? done(err) : done();
}
);
},
});
});
test('abortUploadTask(),Level=bucket', function (done) {
var filename = '1m.zip';
var blob = util.createFile({ size: 1024 * 1024 * 10 });
cos.sliceUploadFile({
Bucket: config.Bucket,
Region: config.Region,
Key: filename,
Body: blob,
Headers: {
'x-cos-traffic-limit': 838860800,
},
onTaskReady: function (taskId) {
TaskId = taskId;
},
onProgress: function (info) {
cos.cancelTask(TaskId);
cos.abortUploadTask(
{
Bucket: config.Bucket,
Region: config.Region,
Level: 'bucket',
},
function (err, data) {
var nameExist = false;
data.successList.forEach(function (item) {
if (filename === item.Key) {
nameExist = true;
}
});
assert.ok(data.successList.length >= 1);
assert.ok(nameExist);
err ? done(err) : done();
}
);
},
});
});
});
group('headBucket()', function () {
test('headBu