租房小程序前端代码
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

240 lines
7.3 KiB

3 months ago
  1. /* eslint-disable no-async-promise-executor */
  2. const debug = require('debug')('ali-oss:multipart-copy');
  3. const copy = require('copy-to');
  4. const proto = exports;
  5. /**
  6. * Upload a part copy in a multipart from the source bucket/object
  7. * used with initMultipartUpload and completeMultipartUpload.
  8. * @param {String} name copy object name
  9. * @param {String} uploadId the upload id
  10. * @param {Number} partNo the part number
  11. * @param {String} range like 0-102400 part size need to copy
  12. * @param {Object} sourceData
  13. * {String} sourceData.sourceKey the source object name
  14. * {String} sourceData.sourceBucketName the source bucket name
  15. * @param {Object} options
  16. */
  17. /* eslint max-len: [0] */
  18. proto.uploadPartCopy = async function uploadPartCopy(name, uploadId, partNo, range, sourceData, options = {}) {
  19. options.headers = options.headers || {};
  20. const versionId = options.versionId || (options.subres && options.subres.versionId) || null;
  21. let copySource;
  22. if (versionId) {
  23. copySource = `/${sourceData.sourceBucketName}/${encodeURIComponent(sourceData.sourceKey)}?versionId=${versionId}`;
  24. } else {
  25. copySource = `/${sourceData.sourceBucketName}/${encodeURIComponent(sourceData.sourceKey)}`;
  26. }
  27. options.headers['x-oss-copy-source'] = copySource;
  28. if (range) {
  29. options.headers['x-oss-copy-source-range'] = `bytes=${range}`;
  30. }
  31. options.subres = {
  32. partNumber: partNo,
  33. uploadId
  34. };
  35. const params = this._objectRequestParams('PUT', name, options);
  36. params.mime = options.mime;
  37. params.successStatuses = [200];
  38. const result = await this.request(params);
  39. return {
  40. name,
  41. etag: result.res.headers.etag,
  42. res: result.res
  43. };
  44. };
  45. /**
  46. * @param {String} name copy object name
  47. * @param {Object} sourceData
  48. * {String} sourceData.sourceKey the source object name
  49. * {String} sourceData.sourceBucketName the source bucket name
  50. * {Number} sourceData.startOffset data copy start byte offset, e.g: 0
  51. * {Number} sourceData.endOffset data copy end byte offset, e.g: 102400
  52. * @param {Object} options
  53. * {Number} options.partSize
  54. */
  55. proto.multipartUploadCopy = async function multipartUploadCopy(name, sourceData, options = {}) {
  56. this.resetCancelFlag();
  57. const { versionId = null } = options;
  58. const metaOpt = {
  59. versionId
  60. };
  61. const objectMeta = await this._getObjectMeta(sourceData.sourceBucketName, sourceData.sourceKey, metaOpt);
  62. const fileSize = objectMeta.res.headers['content-length'];
  63. sourceData.startOffset = sourceData.startOffset || 0;
  64. sourceData.endOffset = sourceData.endOffset || fileSize;
  65. if (options.checkpoint && options.checkpoint.uploadId) {
  66. return await this._resumeMultipartCopy(options.checkpoint, sourceData, options);
  67. }
  68. const minPartSize = 100 * 1024;
  69. const copySize = sourceData.endOffset - sourceData.startOffset;
  70. if (copySize < minPartSize) {
  71. throw new Error(`copySize must not be smaller than ${minPartSize}`);
  72. }
  73. if (options.partSize && options.partSize < minPartSize) {
  74. throw new Error(`partSize must not be smaller than ${minPartSize}`);
  75. }
  76. const init = await this.initMultipartUpload(name, options);
  77. const { uploadId } = init;
  78. const partSize = this._getPartSize(copySize, options.partSize);
  79. const checkpoint = {
  80. name,
  81. copySize,
  82. partSize,
  83. uploadId,
  84. doneParts: []
  85. };
  86. if (options && options.progress) {
  87. await options.progress(0, checkpoint, init.res);
  88. }
  89. return await this._resumeMultipartCopy(checkpoint, sourceData, options);
  90. };
  91. /*
  92. * Resume multipart copy from checkpoint. The checkpoint will be
  93. * updated after each successful part copy.
  94. * @param {Object} checkpoint the checkpoint
  95. * @param {Object} options
  96. */
  97. proto._resumeMultipartCopy = async function _resumeMultipartCopy(checkpoint, sourceData, options) {
  98. if (this.isCancel()) {
  99. throw this._makeCancelEvent();
  100. }
  101. const { versionId = null } = options;
  102. const metaOpt = {
  103. versionId
  104. };
  105. const { copySize, partSize, uploadId, doneParts, name } = checkpoint;
  106. const partOffs = this._divideMultipartCopyParts(copySize, partSize, sourceData.startOffset);
  107. const numParts = partOffs.length;
  108. const uploadPartCopyOptions = {
  109. headers: {}
  110. };
  111. if (options.copyheaders) {
  112. copy(options.copyheaders).to(uploadPartCopyOptions.headers);
  113. }
  114. if (versionId) {
  115. copy(metaOpt).to(uploadPartCopyOptions);
  116. }
  117. const uploadPartJob = function uploadPartJob(self, partNo, source) {
  118. return new Promise(async (resolve, reject) => {
  119. try {
  120. if (!self.isCancel()) {
  121. const pi = partOffs[partNo - 1];
  122. const range = `${pi.start}-${pi.end - 1}`;
  123. let result;
  124. try {
  125. result = await self.uploadPartCopy(name, uploadId, partNo, range, source, uploadPartCopyOptions);
  126. } catch (error) {
  127. if (error.status === 404) {
  128. throw self._makeAbortEvent();
  129. }
  130. throw error;
  131. }
  132. if (!self.isCancel()) {
  133. debug(`content-range ${result.res.headers['content-range']}`);
  134. doneParts.push({
  135. number: partNo,
  136. etag: result.res.headers.etag
  137. });
  138. checkpoint.doneParts = doneParts;
  139. if (options && options.progress) {
  140. await options.progress(doneParts.length / numParts, checkpoint, result.res);
  141. }
  142. }
  143. }
  144. resolve();
  145. } catch (err) {
  146. err.partNum = partNo;
  147. reject(err);
  148. }
  149. });
  150. };
  151. const all = Array.from(new Array(numParts), (x, i) => i + 1);
  152. const done = doneParts.map(p => p.number);
  153. const todo = all.filter(p => done.indexOf(p) < 0);
  154. const defaultParallel = 5;
  155. const parallel = options.parallel || defaultParallel;
  156. if (this.checkBrowserAndVersion('Internet Explorer', '10') || parallel === 1) {
  157. for (let i = 0; i < todo.length; i++) {
  158. if (this.isCancel()) {
  159. throw this._makeCancelEvent();
  160. }
  161. /* eslint no-await-in-loop: [0] */
  162. await uploadPartJob(this, todo[i], sourceData);
  163. }
  164. } else {
  165. // upload in parallel
  166. const errors = await this._parallelNode(todo, parallel, uploadPartJob, sourceData);
  167. const abortEvent = errors.find(err => err.name === 'abort');
  168. if (abortEvent) throw abortEvent;
  169. if (this.isCancel()) {
  170. throw this._makeCancelEvent();
  171. }
  172. // check errors after all jobs are completed
  173. if (errors && errors.length > 0) {
  174. const err = errors[0];
  175. err.message = `Failed to copy some parts with error: ${err.toString()} part_num: ${err.partNum}`;
  176. throw err;
  177. }
  178. }
  179. return await this.completeMultipartUpload(name, uploadId, doneParts, options);
  180. };
  181. proto._divideMultipartCopyParts = function _divideMultipartCopyParts(fileSize, partSize, startOffset) {
  182. const numParts = Math.ceil(fileSize / partSize);
  183. const partOffs = [];
  184. for (let i = 0; i < numParts; i++) {
  185. const start = partSize * i + startOffset;
  186. const end = Math.min(start + partSize, fileSize + startOffset);
  187. partOffs.push({
  188. start,
  189. end
  190. });
  191. }
  192. return partOffs;
  193. };
  194. /**
  195. * Get Object Meta
  196. * @param {String} bucket bucket name
  197. * @param {String} name object name
  198. * @param {Object} options
  199. */
  200. proto._getObjectMeta = async function _getObjectMeta(bucket, name, options) {
  201. const currentBucket = this.getBucket();
  202. this.setBucket(bucket);
  203. const data = await this.head(name, options);
  204. this.setBucket(currentBucket);
  205. return data;
  206. };