273 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			273 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
const copy = require('copy-to');
 | 
						|
const callback = require('./callback');
 | 
						|
const { deepCopyWith } = require('./utils/deepCopy');
 | 
						|
const { isBuffer } = require('./utils/isBuffer');
 | 
						|
const { omit } = require('./utils/omit');
 | 
						|
 | 
						|
const proto = exports;
 | 
						|
 | 
						|
/**
 | 
						|
 * List the on-going multipart uploads
 | 
						|
 * https://help.aliyun.com/document_detail/31997.html
 | 
						|
 * @param {Object} options
 | 
						|
 * @return {Array} the multipart uploads
 | 
						|
 */
 | 
						|
proto.listUploads = async function listUploads(query, options) {
 | 
						|
  options = options || {};
 | 
						|
  const opt = {};
 | 
						|
  copy(options).to(opt);
 | 
						|
  opt.subres = 'uploads';
 | 
						|
  const params = this._objectRequestParams('GET', '', opt);
 | 
						|
  params.query = query;
 | 
						|
  params.xmlResponse = true;
 | 
						|
  params.successStatuses = [200];
 | 
						|
 | 
						|
  const result = await this.request(params);
 | 
						|
  let uploads = result.data.Upload || [];
 | 
						|
  if (!Array.isArray(uploads)) {
 | 
						|
    uploads = [uploads];
 | 
						|
  }
 | 
						|
  uploads = uploads.map(up => ({
 | 
						|
    name: up.Key,
 | 
						|
    uploadId: up.UploadId,
 | 
						|
    initiated: up.Initiated
 | 
						|
  }));
 | 
						|
 | 
						|
  return {
 | 
						|
    res: result.res,
 | 
						|
    uploads,
 | 
						|
    bucket: result.data.Bucket,
 | 
						|
    nextKeyMarker: result.data.NextKeyMarker,
 | 
						|
    nextUploadIdMarker: result.data.NextUploadIdMarker,
 | 
						|
    isTruncated: result.data.IsTruncated === 'true'
 | 
						|
  };
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * List the done uploadPart parts
 | 
						|
 * @param {String} name object name
 | 
						|
 * @param {String} uploadId multipart upload id
 | 
						|
 * @param {Object} query
 | 
						|
 * {Number} query.max-parts The maximum part number in the response of the OSS. Default value: 1000
 | 
						|
 * {Number} query.part-number-marker Starting position of a specific list.
 | 
						|
 * {String} query.encoding-type Specify the encoding of the returned content and the encoding type.
 | 
						|
 * @param {Object} options
 | 
						|
 * @return {Object} result
 | 
						|
 */
 | 
						|
proto.listParts = async function listParts(name, uploadId, query, options) {
 | 
						|
  options = options || {};
 | 
						|
  const opt = {};
 | 
						|
  copy(options).to(opt);
 | 
						|
  opt.subres = {
 | 
						|
    uploadId
 | 
						|
  };
 | 
						|
  const params = this._objectRequestParams('GET', name, opt);
 | 
						|
  params.query = query;
 | 
						|
  params.xmlResponse = true;
 | 
						|
  params.successStatuses = [200];
 | 
						|
 | 
						|
  const result = await this.request(params);
 | 
						|
 | 
						|
  return {
 | 
						|
    res: result.res,
 | 
						|
    uploadId: result.data.UploadId,
 | 
						|
    bucket: result.data.Bucket,
 | 
						|
    name: result.data.Key,
 | 
						|
    partNumberMarker: result.data.PartNumberMarker,
 | 
						|
    nextPartNumberMarker: result.data.NextPartNumberMarker,
 | 
						|
    maxParts: result.data.MaxParts,
 | 
						|
    isTruncated: result.data.IsTruncated,
 | 
						|
    parts: result.data.Part || []
 | 
						|
  };
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Abort a multipart upload transaction
 | 
						|
 * @param {String} name the object name
 | 
						|
 * @param {String} uploadId the upload id
 | 
						|
 * @param {Object} options
 | 
						|
 */
 | 
						|
proto.abortMultipartUpload = async function abortMultipartUpload(name, uploadId, options) {
 | 
						|
  this._stop();
 | 
						|
  options = options || {};
 | 
						|
  const opt = {};
 | 
						|
  copy(options).to(opt);
 | 
						|
  opt.subres = { uploadId };
 | 
						|
  const params = this._objectRequestParams('DELETE', name, opt);
 | 
						|
  params.successStatuses = [204];
 | 
						|
 | 
						|
  const result = await this.request(params);
 | 
						|
  return {
 | 
						|
    res: result.res
 | 
						|
  };
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Initiate a multipart upload transaction
 | 
						|
 * @param {String} name the object name
 | 
						|
 * @param {Object} options
 | 
						|
 * @return {String} upload id
 | 
						|
 */
 | 
						|
proto.initMultipartUpload = async function initMultipartUpload(name, options) {
 | 
						|
  options = options || {};
 | 
						|
  const opt = {};
 | 
						|
  copy(options).to(opt);
 | 
						|
  opt.headers = opt.headers || {};
 | 
						|
  this._convertMetaToHeaders(options.meta, opt.headers);
 | 
						|
 | 
						|
  opt.subres = 'uploads';
 | 
						|
  const params = this._objectRequestParams('POST', name, opt);
 | 
						|
  params.mime = options.mime;
 | 
						|
  params.xmlResponse = true;
 | 
						|
  params.successStatuses = [200];
 | 
						|
 | 
						|
  const result = await this.request(params);
 | 
						|
 | 
						|
  return {
 | 
						|
    res: result.res,
 | 
						|
    bucket: result.data.Bucket,
 | 
						|
    name: result.data.Key,
 | 
						|
    uploadId: result.data.UploadId
 | 
						|
  };
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Upload a part in a multipart upload transaction
 | 
						|
 * @param {String} name the object name
 | 
						|
 * @param {String} uploadId the upload id
 | 
						|
 * @param {Integer} partNo the part number
 | 
						|
 * @param {File} file upload File, whole File
 | 
						|
 * @param {Integer} start  part start bytes  e.g: 102400
 | 
						|
 * @param {Integer} end  part end bytes  e.g: 204800
 | 
						|
 * @param {Object} options
 | 
						|
 */
 | 
						|
proto.uploadPart = async function uploadPart(name, uploadId, partNo, file, start, end, options) {
 | 
						|
  const data = {
 | 
						|
    size: end - start
 | 
						|
  };
 | 
						|
  const isBrowserEnv = process && process.browser;
 | 
						|
  isBrowserEnv
 | 
						|
    ? (data.content = await this._createBuffer(file, start, end))
 | 
						|
    : (data.stream = await this._createStream(file, start, end));
 | 
						|
  return await this._uploadPart(name, uploadId, partNo, data, options);
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Complete a multipart upload transaction
 | 
						|
 * @param {String} name the object name
 | 
						|
 * @param {String} uploadId the upload id
 | 
						|
 * @param {Array} parts the uploaded parts, each in the structure:
 | 
						|
 *        {Integer} number partNo
 | 
						|
 *        {String} etag  part etag  uploadPartCopy result.res.header.etag
 | 
						|
 * @param {Object} options
 | 
						|
 *         {Object} [options.callback] The callback parameter is composed of a JSON string encoded in Base64
 | 
						|
 *         {String} options.callback.url  the OSS sends a callback request to this URL
 | 
						|
 *         {String} [options.callback.host]  The host header value for initiating callback requests
 | 
						|
 *         {String} options.callback.body  The value of the request body when a callback is initiated
 | 
						|
 *         {String} [options.callback.contentType]  The Content-Type of the callback requests initiated
 | 
						|
 *         {Boolean} [options.callback.callbackSNI] Whether OSS sends SNI to the origin address specified by callbackUrl when a callback request is initiated from the client
 | 
						|
 *         {Object} [options.callback.customValue]  Custom parameters are a map of key-values, e.g:
 | 
						|
 *                   customValue = {
 | 
						|
 *                     key1: 'value1',
 | 
						|
 *                     key2: 'value2'
 | 
						|
 *                   }
 | 
						|
 */
 | 
						|
proto.completeMultipartUpload = async function completeMultipartUpload(name, uploadId, parts, options) {
 | 
						|
  const completeParts = parts
 | 
						|
    .concat()
 | 
						|
    .sort((a, b) => a.number - b.number)
 | 
						|
    .filter((item, index, arr) => !index || item.number !== arr[index - 1].number);
 | 
						|
  let xml = '<?xml version="1.0" encoding="UTF-8"?>\n<CompleteMultipartUpload>\n';
 | 
						|
  for (let i = 0; i < completeParts.length; i++) {
 | 
						|
    const p = completeParts[i];
 | 
						|
    xml += '<Part>\n';
 | 
						|
    xml += `<PartNumber>${p.number}</PartNumber>\n`;
 | 
						|
    xml += `<ETag>${p.etag}</ETag>\n`;
 | 
						|
    xml += '</Part>\n';
 | 
						|
  }
 | 
						|
  xml += '</CompleteMultipartUpload>';
 | 
						|
 | 
						|
  options = options || {};
 | 
						|
  let opt = {};
 | 
						|
  opt = deepCopyWith(options, _ => {
 | 
						|
    if (isBuffer(_)) return null;
 | 
						|
  });
 | 
						|
  opt.subres = { uploadId };
 | 
						|
  opt.headers = omit(opt.headers, ['x-oss-server-side-encryption', 'x-oss-storage-class']);
 | 
						|
 | 
						|
  const params = this._objectRequestParams('POST', name, opt);
 | 
						|
  callback.encodeCallback(params, opt);
 | 
						|
  params.mime = 'xml';
 | 
						|
  params.content = xml;
 | 
						|
 | 
						|
  if (!(params.headers && params.headers['x-oss-callback'])) {
 | 
						|
    params.xmlResponse = true;
 | 
						|
  }
 | 
						|
  params.successStatuses = [200];
 | 
						|
  const result = await this.request(params);
 | 
						|
 | 
						|
  if (options.progress) {
 | 
						|
    await options.progress(1, null, result.res);
 | 
						|
  }
 | 
						|
 | 
						|
  const ret = {
 | 
						|
    res: result.res,
 | 
						|
    bucket: params.bucket,
 | 
						|
    name,
 | 
						|
    etag: result.res.headers.etag
 | 
						|
  };
 | 
						|
 | 
						|
  if (params.headers && params.headers['x-oss-callback']) {
 | 
						|
    ret.data = JSON.parse(result.data.toString());
 | 
						|
  }
 | 
						|
 | 
						|
  return ret;
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Upload a part in a multipart upload transaction
 | 
						|
 * @param {String} name the object name
 | 
						|
 * @param {String} uploadId the upload id
 | 
						|
 * @param {Integer} partNo the part number
 | 
						|
 * @param {Object} data the body data
 | 
						|
 * @param {Object} options
 | 
						|
 */
 | 
						|
proto._uploadPart = async function _uploadPart(name, uploadId, partNo, data, options) {
 | 
						|
  options = options || {};
 | 
						|
  const opt = {};
 | 
						|
  copy(options).to(opt);
 | 
						|
  opt.headers = opt.headers || {};
 | 
						|
  opt.headers['Content-Length'] = data.size;
 | 
						|
 | 
						|
  // Uploading shards does not require x-oss headers.
 | 
						|
  opt.headers = omit(opt.headers, ['x-oss-server-side-encryption', 'x-oss-storage-class']);
 | 
						|
  opt.subres = {
 | 
						|
    partNumber: partNo,
 | 
						|
    uploadId
 | 
						|
  };
 | 
						|
 | 
						|
  const params = this._objectRequestParams('PUT', name, opt);
 | 
						|
  params.mime = opt.mime;
 | 
						|
  const isBrowserEnv = process && process.browser;
 | 
						|
  isBrowserEnv ? (params.content = data.content) : (params.stream = data.stream);
 | 
						|
  params.successStatuses = [200];
 | 
						|
  params.disabledMD5 = options.disabledMD5;
 | 
						|
 | 
						|
  const result = await this.request(params);
 | 
						|
 | 
						|
  if (!result.res.headers.etag) {
 | 
						|
    throw new Error(
 | 
						|
      'Please set the etag of expose-headers in OSS \n https://help.aliyun.com/document_detail/32069.html'
 | 
						|
    );
 | 
						|
  }
 | 
						|
  if (data.stream) {
 | 
						|
    data.stream = null;
 | 
						|
    params.stream = null;
 | 
						|
  }
 | 
						|
  return {
 | 
						|
    name,
 | 
						|
    etag: result.res.headers.etag,
 | 
						|
    res: result.res
 | 
						|
  };
 | 
						|
};
 |