mirror of https://github.com/jkjoy/sunpeiwen.git
776 lines
23 KiB
JavaScript
776 lines
23 KiB
JavaScript
'use strict';
|
||
|
||
var fs = require('fs');
|
||
var crypto = require('crypto');
|
||
var { XMLParser, XMLBuilder } = require('fast-xml-parser');
|
||
var xmlParser = new XMLParser({
|
||
ignoreDeclaration: true, // 忽略 XML 声明
|
||
ignoreAttributes: true, // 忽略属性
|
||
parseTagValue: false, // 关闭自动解析
|
||
});
|
||
var xmlBuilder = new XMLBuilder();
|
||
|
||
function camSafeUrlEncode(str) {
|
||
return encodeURIComponent(str)
|
||
.replace(/!/g, '%21')
|
||
.replace(/'/g, '%27')
|
||
.replace(/\(/g, '%28')
|
||
.replace(/\)/g, '%29')
|
||
.replace(/\*/g, '%2A');
|
||
}
|
||
|
||
var getObjectKeys = function (obj, forKey) {
|
||
var list = [];
|
||
for (var key in obj) {
|
||
if (obj.hasOwnProperty(key)) {
|
||
list.push(forKey ? camSafeUrlEncode(key).toLowerCase() : key);
|
||
}
|
||
}
|
||
return list.sort(function (a, b) {
|
||
a = a.toLowerCase();
|
||
b = b.toLowerCase();
|
||
return a === b ? 0 : a > b ? 1 : -1;
|
||
});
|
||
};
|
||
|
||
/**
|
||
* obj转为string
|
||
* @param {Object} obj 需要转的对象,必须
|
||
* @param {Boolean} lowerCaseKey key是否转为小写,默认false,非必须
|
||
* @return {String} data 返回字符串
|
||
*/
|
||
var obj2str = function (obj, lowerCaseKey) {
|
||
var i, key, val;
|
||
var list = [];
|
||
var keyList = getObjectKeys(obj);
|
||
for (i = 0; i < keyList.length; i++) {
|
||
key = keyList[i];
|
||
val = obj[key] === undefined || obj[key] === null ? '' : '' + obj[key];
|
||
key = lowerCaseKey ? camSafeUrlEncode(key).toLowerCase() : camSafeUrlEncode(key);
|
||
val = camSafeUrlEncode(val) || '';
|
||
list.push(key + '=' + val);
|
||
}
|
||
return list.join('&');
|
||
};
|
||
|
||
// 可以签入签名的headers
|
||
var signHeaders = [
|
||
'content-disposition',
|
||
'content-encoding',
|
||
'content-length',
|
||
'content-md5',
|
||
'expect',
|
||
'expires',
|
||
'host',
|
||
'if-match',
|
||
'if-modified-since',
|
||
'if-none-match',
|
||
'if-unmodified-since',
|
||
'origin',
|
||
'range',
|
||
'transfer-encoding',
|
||
];
|
||
|
||
var getSignHeaderObj = function (headers) {
|
||
var signHeaderObj = {};
|
||
for (var i in headers) {
|
||
var key = i.toLowerCase();
|
||
if (key.indexOf('x-cos-') > -1 || signHeaders.indexOf(key) > -1) {
|
||
signHeaderObj[i] = headers[i];
|
||
}
|
||
}
|
||
return signHeaderObj;
|
||
};
|
||
|
||
//测试用的key后面可以去掉
|
||
var getAuth = function (opt) {
|
||
opt = opt || {};
|
||
|
||
var SecretId = opt.SecretId;
|
||
var SecretKey = opt.SecretKey;
|
||
var KeyTime = opt.KeyTime;
|
||
var method = (opt.method || opt.Method || 'get').toLowerCase();
|
||
var queryParams = clone(opt.Query || opt.params || {});
|
||
var headers = getSignHeaderObj(clone(opt.Headers || opt.headers || {}));
|
||
|
||
var Key = opt.Key || '';
|
||
var pathname;
|
||
if (opt.UseRawKey) {
|
||
pathname = opt.Pathname || opt.pathname || '/' + Key;
|
||
} else {
|
||
pathname = opt.Pathname || opt.pathname || Key;
|
||
pathname.indexOf('/') !== 0 && (pathname = '/' + pathname);
|
||
}
|
||
|
||
// ForceSignHost明确传入false才不加入host签名
|
||
var forceSignHost = opt.ForceSignHost === false ? false : true;
|
||
|
||
// 如果有传入存储桶,那么签名默认加 Host 参与计算,避免跨桶访问
|
||
if (!headers.Host && !headers.host && opt.Bucket && opt.Region && forceSignHost)
|
||
headers.Host = opt.Bucket + '.cos.' + opt.Region + '.myqcloud.com';
|
||
|
||
if (!SecretId) throw new Error('missing param SecretId');
|
||
if (!SecretKey) throw new Error('missing param SecretKey');
|
||
|
||
// 签名有效起止时间
|
||
var now = Math.round(getSkewTime(opt.SystemClockOffset) / 1000) - 1;
|
||
var exp = now;
|
||
|
||
var Expires = opt.Expires || opt.expires;
|
||
if (Expires === undefined) {
|
||
exp += 900; // 签名过期时间为当前 + 900s
|
||
} else {
|
||
exp += Expires * 1 || 0;
|
||
}
|
||
|
||
// 要用到的 Authorization 参数列表
|
||
var qSignAlgorithm = 'sha1';
|
||
var qAk = SecretId;
|
||
var qSignTime = KeyTime || now + ';' + exp;
|
||
var qKeyTime = KeyTime || now + ';' + exp;
|
||
var qHeaderList = getObjectKeys(headers, true).join(';').toLowerCase();
|
||
var qUrlParamList = getObjectKeys(queryParams, true).join(';').toLowerCase();
|
||
|
||
// 签名算法说明文档:https://www.qcloud.com/document/product/436/7778
|
||
// 步骤一:计算 SignKey
|
||
var signKey = crypto.createHmac('sha1', SecretKey).update(qKeyTime).digest('hex');
|
||
|
||
// 步骤二:构成 FormatString
|
||
var formatString = [method, pathname, obj2str(queryParams, true), obj2str(headers, true), ''].join('\n');
|
||
formatString = Buffer.from(formatString, 'utf8');
|
||
|
||
// 步骤三:计算 StringToSign
|
||
var res = crypto.createHash('sha1').update(formatString).digest('hex');
|
||
var stringToSign = ['sha1', qSignTime, res, ''].join('\n');
|
||
|
||
// 步骤四:计算 Signature
|
||
var qSignature = crypto.createHmac('sha1', signKey).update(stringToSign).digest('hex');
|
||
|
||
// 步骤五:构造 Authorization
|
||
var authorization = [
|
||
'q-sign-algorithm=' + qSignAlgorithm,
|
||
'q-ak=' + qAk,
|
||
'q-sign-time=' + qSignTime,
|
||
'q-key-time=' + qKeyTime,
|
||
'q-header-list=' + qHeaderList,
|
||
'q-url-param-list=' + qUrlParamList,
|
||
'q-signature=' + qSignature,
|
||
].join('&');
|
||
|
||
return authorization;
|
||
};
|
||
|
||
var getV4Auth = function (opt) {
|
||
if (!opt.SecretId) return console.error('missing param SecretId');
|
||
if (!opt.SecretKey) return console.error('missing param SecretKey');
|
||
if (!opt.Bucket) return console.error('missing param Bucket');
|
||
|
||
var longBucket = opt.Bucket;
|
||
var ShortBucket = longBucket.substr(0, longBucket.lastIndexOf('-'));
|
||
var AppId = longBucket.substr(longBucket.lastIndexOf('-') + 1);
|
||
var random = Math.round(Math.random() * Math.pow(2, 32));
|
||
var now = Math.round(Date.now() / 1000);
|
||
var e = now + (opt.Expires === undefined ? 900 : opt.Expires);
|
||
var path =
|
||
'/' +
|
||
AppId +
|
||
'/' +
|
||
ShortBucket +
|
||
'/' +
|
||
encodeURIComponent((opt.Key || '').replace(/(^\/*)/g, '')).replace(/%2F/g, '/');
|
||
var plainText =
|
||
'a=' + AppId + '&b=' + ShortBucket + '&k=' + opt.SecretId + '&t=' + now + '&e=' + e + '&r=' + random + '&f=' + path;
|
||
var signKey = crypto.createHmac('sha1', opt.SecretKey).update(plainText).digest();
|
||
var sign = Buffer.concat([signKey, Buffer.from(plainText)]).toString('base64');
|
||
return sign;
|
||
};
|
||
|
||
var getSourceParams = function (source) {
|
||
var parser = this.options.CopySourceParser;
|
||
if (parser) return parser(source);
|
||
var m = source.match(/^([^.]+-\d+)\.cos(v6|-cdc|-internal)?\.([^.]+)\.((myqcloud\.com)|(tencentcos\.cn))\/(.+)$/);
|
||
if (!m) return null;
|
||
return { Bucket: m[1], Region: m[3], Key: m[7] };
|
||
};
|
||
|
||
var noop = function () {};
|
||
|
||
// 清除对象里值为的 undefined 或 null 的属性
|
||
var clearKey = function (obj) {
|
||
var retObj = {};
|
||
for (var key in obj) {
|
||
if (obj.hasOwnProperty(key) && obj[key] !== undefined && obj[key] !== null) {
|
||
retObj[key] = obj[key];
|
||
}
|
||
}
|
||
return retObj;
|
||
};
|
||
|
||
// XML 对象转 JSON 对象
|
||
var xml2json = function (bodyStr) {
|
||
var d = xmlParser.parse(bodyStr);
|
||
|
||
return d;
|
||
};
|
||
|
||
// JSON 对象转 XML 对象
|
||
var json2xml = function (json) {
|
||
var xml = xmlBuilder.build(json);
|
||
return xml;
|
||
};
|
||
|
||
// 计算 MD5
|
||
var md5 = function (str, encoding) {
|
||
return crypto
|
||
.createHash('md5')
|
||
.update(str)
|
||
.digest(encoding || 'hex');
|
||
};
|
||
|
||
// 获取文件分片
|
||
var fileSlice = function (FilePath, start, end, callback) {
|
||
if (FilePath) {
|
||
try {
|
||
var readStream = fs.createReadStream(FilePath, { start: start, end: end - 1 });
|
||
readStream.isSdkCreated = true;
|
||
callback(readStream);
|
||
} catch (e) {}
|
||
} else {
|
||
callback(null);
|
||
}
|
||
};
|
||
|
||
// 获取文件内容的 MD5
|
||
var getBodyMd5 = function (UploadCheckContentMd5, Body, callback) {
|
||
callback = callback || noop;
|
||
if (UploadCheckContentMd5) {
|
||
if (Body instanceof Buffer || typeof Body === 'string') {
|
||
callback(util.md5(Body));
|
||
} else {
|
||
callback();
|
||
}
|
||
} else {
|
||
callback();
|
||
}
|
||
};
|
||
|
||
// 获取文件 md5 值
|
||
var getFileMd5 = function (readStream, callback) {
|
||
var md5 = crypto.createHash('md5');
|
||
readStream.on('data', function (chunk) {
|
||
md5.update(chunk);
|
||
});
|
||
readStream.on('error', function (err) {
|
||
callback(util.error(err));
|
||
});
|
||
readStream.on('end', function () {
|
||
var hash = md5.digest('hex');
|
||
callback(null, hash);
|
||
});
|
||
};
|
||
|
||
function clone(obj) {
|
||
return map(obj, function (v) {
|
||
return typeof v === 'object' && v !== null ? clone(v) : v;
|
||
});
|
||
}
|
||
|
||
function attr(obj, name, defaultValue) {
|
||
return obj && name in obj ? obj[name] : defaultValue;
|
||
}
|
||
|
||
function extend(target, source) {
|
||
each(source, function (val, key) {
|
||
target[key] = source[key];
|
||
});
|
||
return target;
|
||
}
|
||
|
||
function isArray(arr) {
|
||
return arr instanceof Array;
|
||
}
|
||
|
||
function isInArray(arr, item) {
|
||
var flag = false;
|
||
for (var i = 0; i < arr.length; i++) {
|
||
if (item === arr[i]) {
|
||
flag = true;
|
||
break;
|
||
}
|
||
}
|
||
return flag;
|
||
}
|
||
|
||
function makeArray(arr) {
|
||
return isArray(arr) ? arr : [arr];
|
||
}
|
||
|
||
function each(obj, fn) {
|
||
for (var i in obj) {
|
||
if (obj.hasOwnProperty(i)) {
|
||
fn(obj[i], i);
|
||
}
|
||
}
|
||
}
|
||
|
||
function map(obj, fn) {
|
||
var o = isArray(obj) ? [] : {};
|
||
for (var i in obj) {
|
||
if (obj.hasOwnProperty(i)) {
|
||
o[i] = fn(obj[i], i);
|
||
}
|
||
}
|
||
return o;
|
||
}
|
||
|
||
function filter(obj, fn) {
|
||
var iaArr = isArray(obj);
|
||
var o = iaArr ? [] : {};
|
||
for (var i in obj) {
|
||
if (obj.hasOwnProperty(i)) {
|
||
if (fn(obj[i], i)) {
|
||
if (iaArr) {
|
||
o.push(obj[i]);
|
||
} else {
|
||
o[i] = obj[i];
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return o;
|
||
}
|
||
|
||
var binaryBase64 = function (str) {
|
||
var i,
|
||
len,
|
||
char,
|
||
arr = [];
|
||
for (i = 0, len = str.length / 2; i < len; i++) {
|
||
char = parseInt(str[i * 2] + str[i * 2 + 1], 16);
|
||
arr.push(char);
|
||
}
|
||
return Buffer.from(arr).toString('base64');
|
||
};
|
||
var uuid = function () {
|
||
var S4 = function () {
|
||
return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
|
||
};
|
||
return S4() + S4() + '-' + S4() + '-' + S4() + '-' + S4() + '-' + S4() + S4() + S4();
|
||
};
|
||
|
||
var hasMissingParams = function (apiName, params) {
|
||
var Bucket = params.Bucket;
|
||
var Region = params.Region;
|
||
var Key = params.Key;
|
||
if (
|
||
apiName.indexOf('Bucket') > -1 ||
|
||
apiName === 'deleteMultipleObject' ||
|
||
apiName === 'multipartList' ||
|
||
apiName === 'listObjectVersions'
|
||
) {
|
||
if (!Bucket) return 'Bucket';
|
||
if (!Region) return 'Region';
|
||
} else if (
|
||
apiName.indexOf('Object') > -1 ||
|
||
apiName.indexOf('multipart') > -1 ||
|
||
apiName === 'sliceUploadFile' ||
|
||
apiName === 'abortUploadTask'
|
||
) {
|
||
if (!Bucket) return 'Bucket';
|
||
if (!Region) return 'Region';
|
||
if (!Key) return 'Key';
|
||
}
|
||
return false;
|
||
};
|
||
|
||
var formatParams = function (apiName, params) {
|
||
// 复制参数对象
|
||
params = extend({}, params);
|
||
|
||
// 统一处理 Headers
|
||
if (apiName !== 'getAuth' && apiName !== 'getV4Auth' && apiName !== 'getObjectUrl') {
|
||
var Headers = params.Headers || {};
|
||
if (params && typeof params === 'object') {
|
||
(function () {
|
||
for (var key in params) {
|
||
if (params.hasOwnProperty(key) && key.indexOf('x-cos-') > -1) {
|
||
Headers[key] = params[key];
|
||
}
|
||
}
|
||
})();
|
||
|
||
var headerMap = {
|
||
// params headers
|
||
'x-cos-mfa': 'MFA',
|
||
'Content-MD5': 'ContentMD5',
|
||
'Content-Length': 'ContentLength',
|
||
'Content-Type': 'ContentType',
|
||
Expect: 'Expect',
|
||
Expires: 'Expires',
|
||
'Cache-Control': 'CacheControl',
|
||
'Content-Disposition': 'ContentDisposition',
|
||
'Content-Encoding': 'ContentEncoding',
|
||
Range: 'Range',
|
||
'If-Modified-Since': 'IfModifiedSince',
|
||
'If-Unmodified-Since': 'IfUnmodifiedSince',
|
||
'If-Match': 'IfMatch',
|
||
'If-None-Match': 'IfNoneMatch',
|
||
'x-cos-copy-source': 'CopySource',
|
||
'x-cos-copy-source-Range': 'CopySourceRange',
|
||
'x-cos-metadata-directive': 'MetadataDirective',
|
||
'x-cos-copy-source-If-Modified-Since': 'CopySourceIfModifiedSince',
|
||
'x-cos-copy-source-If-Unmodified-Since': 'CopySourceIfUnmodifiedSince',
|
||
'x-cos-copy-source-If-Match': 'CopySourceIfMatch',
|
||
'x-cos-copy-source-If-None-Match': 'CopySourceIfNoneMatch',
|
||
'x-cos-acl': 'ACL',
|
||
'x-cos-grant-read': 'GrantRead',
|
||
'x-cos-grant-write': 'GrantWrite',
|
||
'x-cos-grant-full-control': 'GrantFullControl',
|
||
'x-cos-grant-read-acp': 'GrantReadAcp',
|
||
'x-cos-grant-write-acp': 'GrantWriteAcp',
|
||
'x-cos-storage-class': 'StorageClass',
|
||
'x-cos-traffic-limit': 'TrafficLimit',
|
||
'x-cos-mime-limit': 'MimeLimit',
|
||
// SSE-C
|
||
'x-cos-server-side-encryption-customer-algorithm': 'SSECustomerAlgorithm',
|
||
'x-cos-server-side-encryption-customer-key': 'SSECustomerKey',
|
||
'x-cos-server-side-encryption-customer-key-MD5': 'SSECustomerKeyMD5',
|
||
// SSE-COS、SSE-KMS
|
||
'x-cos-server-side-encryption': 'ServerSideEncryption',
|
||
'x-cos-server-side-encryption-cos-kms-key-id': 'SSEKMSKeyId',
|
||
'x-cos-server-side-encryption-context': 'SSEContext',
|
||
};
|
||
util.each(headerMap, function (paramKey, headerKey) {
|
||
if (params[paramKey] !== undefined) {
|
||
Headers[headerKey] = params[paramKey];
|
||
}
|
||
});
|
||
|
||
params.Headers = clearKey(Headers);
|
||
}
|
||
}
|
||
|
||
return params;
|
||
};
|
||
|
||
var apiWrapper = function (apiName, apiFn) {
|
||
return function (params, callback) {
|
||
var self = this;
|
||
|
||
// 处理参数
|
||
if (typeof params === 'function') {
|
||
callback = params;
|
||
params = {};
|
||
}
|
||
|
||
// 整理参数格式
|
||
params = formatParams(apiName, params);
|
||
|
||
// 代理回调函数
|
||
var formatResult = function (result) {
|
||
if (result && result.headers) {
|
||
result.headers['x-cos-request-id'] && (result.RequestId = result.headers['x-cos-request-id']);
|
||
result.headers['x-ci-request-id'] && (result.RequestId = result.headers['x-ci-request-id']);
|
||
result.headers['x-cos-version-id'] && (result.VersionId = result.headers['x-cos-version-id']);
|
||
result.headers['x-cos-delete-marker'] && (result.DeleteMarker = result.headers['x-cos-delete-marker']);
|
||
}
|
||
return result;
|
||
};
|
||
var _callback = function (err, data) {
|
||
callback && callback(formatResult(err), formatResult(data));
|
||
};
|
||
|
||
var checkParams = function () {
|
||
if (apiName !== 'getService' && apiName !== 'abortUploadTask') {
|
||
// 判断参数是否完整
|
||
var missingResult = hasMissingParams(apiName, params);
|
||
if (missingResult) {
|
||
return 'missing param ' + missingResult;
|
||
}
|
||
// 判断 region 格式
|
||
if (params.Region) {
|
||
if (params.Region.indexOf('cos.') > -1) {
|
||
return 'param Region should not be start with "cos."';
|
||
} else if (!/^([a-z\d-]+)$/.test(params.Region)) {
|
||
return 'Region format error.';
|
||
}
|
||
// 判断 region 格式
|
||
if (
|
||
!self.options.CompatibilityMode &&
|
||
params.Region.indexOf('-') === -1 &&
|
||
params.Region !== 'yfb' &&
|
||
params.Region !== 'default' &&
|
||
params.Region !== 'accelerate'
|
||
) {
|
||
console.warn(
|
||
'warning: param Region format error, find help here: https://cloud.tencent.com/document/product/436/6224',
|
||
);
|
||
}
|
||
}
|
||
// 兼容不带 AppId 的 Bucket
|
||
if (params.Bucket) {
|
||
if (!/^([a-z\d-]+)-(\d+)$/.test(params.Bucket)) {
|
||
if (params.AppId) {
|
||
params.Bucket = params.Bucket + '-' + params.AppId;
|
||
} else if (self.options.AppId) {
|
||
params.Bucket = params.Bucket + '-' + self.options.AppId;
|
||
} else {
|
||
return 'Bucket should format as "test-1250000000".';
|
||
}
|
||
}
|
||
if (params.AppId) {
|
||
console.warn(
|
||
'warning: AppId has been deprecated, Please put it at the end of parameter Bucket(E.g Bucket:"test-1250000000" ).',
|
||
);
|
||
delete params.AppId;
|
||
}
|
||
}
|
||
// 如果 Key 是 / 开头,强制去掉第一个 /
|
||
if (!self.options.UseRawKey && params.Key && params.Key.substr(0, 1) === '/') {
|
||
params.Key = params.Key.substr(1);
|
||
}
|
||
}
|
||
};
|
||
|
||
var errMsg = checkParams();
|
||
var isSync = ['getAuth', 'getV4Auth', 'getObjectUrl'].includes(apiName) || apiName.indexOf('Stream') > -1;
|
||
if (Promise && !isSync && !callback) {
|
||
return new Promise(function (resolve, reject) {
|
||
callback = function (err, data) {
|
||
err ? reject(err) : resolve(data);
|
||
};
|
||
if (errMsg) return _callback(util.error(new Error(errMsg)));
|
||
apiFn.call(self, params, _callback);
|
||
});
|
||
} else {
|
||
if (errMsg) return _callback(util.error(new Error(errMsg)));
|
||
var res = apiFn.call(self, params, _callback);
|
||
if (isSync) return res;
|
||
}
|
||
};
|
||
};
|
||
|
||
var throttleOnProgress = function (total, onProgress) {
|
||
var self = this;
|
||
var size0 = 0;
|
||
var size1 = 0;
|
||
var time0 = Date.now();
|
||
var time1;
|
||
var timer;
|
||
|
||
function update() {
|
||
timer = 0;
|
||
if (onProgress && typeof onProgress === 'function') {
|
||
time1 = Date.now();
|
||
var speed = Math.max(0, Math.round(((size1 - size0) / ((time1 - time0) / 1000)) * 100) / 100) || 0;
|
||
var percent;
|
||
if (size1 === 0 && total === 0) {
|
||
percent = 1;
|
||
} else {
|
||
percent = Math.floor((size1 / total) * 100) / 100 || 0;
|
||
}
|
||
time0 = time1;
|
||
size0 = size1;
|
||
try {
|
||
onProgress({ loaded: size1, total: total, speed: speed, percent: percent });
|
||
} catch (e) {}
|
||
}
|
||
}
|
||
|
||
return function (info, immediately) {
|
||
if (info) {
|
||
size1 = info.loaded;
|
||
total = info.total;
|
||
}
|
||
if (immediately) {
|
||
clearTimeout(timer);
|
||
update();
|
||
} else {
|
||
if (timer) return;
|
||
timer = setTimeout(update, self.options.ProgressInterval);
|
||
}
|
||
};
|
||
};
|
||
|
||
var getFileSize = function (api, params, callback) {
|
||
var size;
|
||
if (api === 'sliceUploadFile') {
|
||
if (params.FilePath) {
|
||
fs.stat(params.FilePath, function (err, fileStats) {
|
||
if (err) {
|
||
if (params.ContentLength !== undefined) {
|
||
size = params.ContentLength;
|
||
} else {
|
||
return callback(err);
|
||
}
|
||
} else {
|
||
params.FileStat = fileStats;
|
||
params.FileStat.FilePath = params.FilePath;
|
||
size = fileStats.isDirectory() ? 0 : fileStats.size;
|
||
}
|
||
params.ContentLength = size = size || 0;
|
||
callback(null, size);
|
||
});
|
||
return;
|
||
} else {
|
||
callback(util.error(new Error('missing param FilePath')));
|
||
return;
|
||
}
|
||
} else {
|
||
if (params.Body !== undefined) {
|
||
if (typeof params.Body === 'string') {
|
||
params.Body = global.Buffer.from(params.Body);
|
||
}
|
||
if (params.Body instanceof global.Buffer) {
|
||
size = params.Body.length;
|
||
} else if (typeof params.Body.pipe === 'function') {
|
||
if (params.ContentLength === undefined) {
|
||
size = undefined;
|
||
} else {
|
||
size = params.ContentLength;
|
||
}
|
||
} else {
|
||
callback(util.error(new Error('params Body format error, Only allow Buffer|Stream|String.')));
|
||
return;
|
||
}
|
||
} else {
|
||
callback(util.error(new Error('missing param Body')));
|
||
return;
|
||
}
|
||
}
|
||
params.ContentLength = size;
|
||
callback(null, size);
|
||
};
|
||
|
||
// 获取调正的时间戳
|
||
var getSkewTime = function (offset) {
|
||
return Date.now() + (offset || 0);
|
||
};
|
||
|
||
// 重写 callback,等待流结束后才 callback
|
||
var callbackAfterStreamFinish = function (stream, callback) {
|
||
if (!stream) return callback;
|
||
var err,
|
||
data,
|
||
count = 2,
|
||
loaded = false;
|
||
var cb = function (e, d) {
|
||
if (loaded) return;
|
||
// 如果有数据,且没有错误,清理 设置错误
|
||
if ((d && !data) || e || err) {
|
||
data = d;
|
||
}
|
||
if (e && !err) {
|
||
err = e;
|
||
data = null;
|
||
}
|
||
if (err || --count === 0) {
|
||
loaded = true;
|
||
callback(err, data);
|
||
}
|
||
};
|
||
stream.on('error', function (err) {
|
||
cb(err);
|
||
});
|
||
stream.on('finish', function () {
|
||
cb();
|
||
});
|
||
return cb;
|
||
};
|
||
|
||
var error = function (err, opt) {
|
||
var sourceErr = err;
|
||
err.message = err.message || null;
|
||
|
||
if (typeof opt === 'string') {
|
||
err.error = opt;
|
||
err.message = opt;
|
||
} else if (typeof opt === 'object' && opt !== null) {
|
||
extend(err, opt);
|
||
if (opt.code || opt.name) err.code = opt.code || opt.name;
|
||
if (opt.message) err.message = opt.message;
|
||
if (opt.stack) err.stack = opt.stack;
|
||
}
|
||
|
||
if (typeof Object.defineProperty === 'function') {
|
||
Object.defineProperty(err, 'name', { writable: true, enumerable: false });
|
||
Object.defineProperty(err, 'message', { enumerable: true });
|
||
}
|
||
|
||
err.name = (opt && opt.name) || err.name || err.code || 'Error';
|
||
if (!err.code) err.code = err.name;
|
||
|
||
if (!err.error) {
|
||
var objectType = Object.prototype.toString.call(err);
|
||
if (objectType === '[object Object]') {
|
||
// 兼容老的错误格式
|
||
err.error = clone(sourceErr);
|
||
} else if (objectType === '[object Error]') {
|
||
// 有环境报出[object Error]对象的情况,兼容处理一下
|
||
err = {
|
||
code: err.code || err.name || 'Error',
|
||
name: err.name || err.code || 'Error',
|
||
message: err.reason || err.message || 'Error',
|
||
};
|
||
}
|
||
}
|
||
return err;
|
||
};
|
||
|
||
var isWeb = function () {
|
||
return typeof window === 'object';
|
||
};
|
||
|
||
var isCIHost = function (url) {
|
||
return /^https?:\/\/([^/]+\.)?ci\.[^/]+/.test(url);
|
||
};
|
||
|
||
var encodeBase64 = function (str, safe) {
|
||
let base64Str = Buffer.from(str).toString('base64');
|
||
// 万象使用的安全base64格式需要特殊处理
|
||
if (safe) {
|
||
base64Str = base64Str.replaceAll('+', '-').replaceAll('/', '_').replaceAll('=', '');
|
||
}
|
||
return base64Str;
|
||
};
|
||
|
||
var util = {
|
||
noop: noop,
|
||
formatParams: formatParams,
|
||
apiWrapper: apiWrapper,
|
||
xml2json: xml2json,
|
||
json2xml: json2xml,
|
||
md5: md5,
|
||
clearKey: clearKey,
|
||
fileSlice: fileSlice,
|
||
getBodyMd5: getBodyMd5,
|
||
getFileMd5: getFileMd5,
|
||
binaryBase64: binaryBase64,
|
||
extend: extend,
|
||
isArray: isArray,
|
||
isInArray: isInArray,
|
||
makeArray: makeArray,
|
||
each: each,
|
||
map: map,
|
||
filter: filter,
|
||
clone: clone,
|
||
attr: attr,
|
||
uuid: uuid,
|
||
camSafeUrlEncode: camSafeUrlEncode,
|
||
throttleOnProgress: throttleOnProgress,
|
||
getFileSize: getFileSize,
|
||
getSkewTime: getSkewTime,
|
||
error: error,
|
||
getAuth: getAuth,
|
||
callbackAfterStreamFinish: callbackAfterStreamFinish,
|
||
getV4Auth: getV4Auth,
|
||
isBrowser: false,
|
||
obj2str: obj2str,
|
||
isWeb: isWeb,
|
||
isCIHost: isCIHost,
|
||
getSourceParams: getSourceParams,
|
||
encodeBase64: encodeBase64,
|
||
};
|
||
|
||
module.exports = util;
|