@xen-orchestra/fs 1.0.0 → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/s3.js CHANGED
@@ -11,6 +11,8 @@ var _libStorage = require("@aws-sdk/lib-storage");
11
11
 
12
12
  var _nodeHttpHandler = require("@aws-sdk/node-http-handler");
13
13
 
14
+ var _middlewareApplyBodyChecksum = require("@aws-sdk/middleware-apply-body-checksum");
15
+
14
16
  var _assert = _interopRequireDefault(require("assert"));
15
17
 
16
18
  var _http = require("http");
@@ -103,6 +105,9 @@ let S3Handler = (_dec = (0, _decorateWith.decorateWith)(_retry.default.wrap, {
103
105
  })
104
106
  })
105
107
  });
108
+
109
+ this._s3.middlewareStack.use((0, _middlewareApplyBodyChecksum.getApplyMd5BodyChecksumPlugin)(this._s3.config));
110
+
106
111
  const parts = (0, _path.split)(path);
107
112
  this._bucket = parts.shift();
108
113
  this._dir = (0, _path.join)(...parts);
package/dist/s3.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/s3.js"],"names":["MIN_PART_SIZE","MAX_PART_SIZE","MAX_PARTS_COUNT","MAX_OBJECT_SIZE","IDEAL_FRAGMENT_SIZE","Math","ceil","warn","S3Handler","pRetry","wrap","delays","when","e","$metadata","httpStatusCode","onRetry","error","attemptNumber","delay","file","arguments","RemoteHandlerAbstract","constructor","remote","_opts","allowUnauthorized","host","path","username","password","protocol","region","url","_s3","S3Client","apiVersion","endpoint","forcePathStyle","credentials","accessKeyId","secretAccessKey","tls","requestHandler","NodeHttpHandler","socketTimeout","httpAgent","HttpAgent","keepAlive","httpsAgent","HttpsAgent","rejectUnauthorized","parts","_bucket","shift","_dir","type","_makeCopySource","_makeKey","_makePrefix","dir","_createParams","Bucket","Key","_multipartCopy","oldPath","newPath","size","_getSize","CopySource","multipartParams","send","CreateMultipartUploadCommand","start","partNumber","length","upload","UploadPartCopyCommand","CopySourceRange","min","PartNumber","push","ETag","CopyPartResult","CompleteMultipartUploadCommand","MultipartUpload","Parts","AbortMultipartUploadCommand","_copy","CopyObjectCommand","name","_isNotEmptyDir","result","ListObjectsV2Command","MaxKeys","Prefix","Contents","_isFile","HeadObjectCommand","_outputStream","input","validator","Body","PassThrough","Upload","client","queueSize","partSize","params","done","undefined","call","unlink","_writeFile","data","options","PutObjectCommand","_createReadStream","Error","code","GetObjectCommand","_unlink","DeleteObjectCommand","_list","NextContinuationToken","uniq","Set","Delimiter","ContinuationToken","IsTruncated","entry","CommonPrefixes","add","_mkdir","_rename","copy","fd","ContentLength","_read","buffer","position","Range","bytesRead","_rmdir","_rmtree","concurrency","_write","uploadParams","fileSize","resultBuffer","Buffer","alloc","max","bytesWritten","copyMultipartParams","prefixSize","suffixOffset","suffixSize","hasSuffix","editBuffer","editBufferOffset","prefixPosition","fragmentsCount","floor","prefixFragmentSize","prefixLastFragmentSize","i","fragmentEnd","assert","strictEqual","range","copyPrefixParams","part","downloadParams","prefixBuffer","concat","complementSize","complementOffset","prefixRange","complementBuffer","editParams","editPart","UploadPartCommand","suffixFragments","suffixFragmentOffset","suffixRange","copySuffixParams","suffixPart","_openFile","flags","_closeFile"],"mappings":"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","sourcesContent":["import {\n AbortMultipartUploadCommand,\n CompleteMultipartUploadCommand,\n CopyObjectCommand,\n CreateMultipartUploadCommand,\n DeleteObjectCommand,\n GetObjectCommand,\n HeadObjectCommand,\n ListObjectsV2Command,\n PutObjectCommand,\n S3Client,\n UploadPartCommand,\n UploadPartCopyCommand,\n} from '@aws-sdk/client-s3'\nimport { Upload } from '@aws-sdk/lib-storage'\nimport { NodeHttpHandler } from '@aws-sdk/node-http-handler'\nimport assert from 'assert'\nimport { Agent as HttpAgent } from 'http'\nimport { Agent as HttpsAgent } from 'https'\nimport pRetry from 'promise-toolbox/retry'\nimport { createLogger } from '@xen-orchestra/log'\nimport { decorateWith } from '@vates/decorate-with'\nimport { PassThrough, pipeline } from 'stream'\nimport { parse } from 'xo-remote-parser'\nimport copyStreamToBuffer from './_copyStreamToBuffer.js'\nimport createBufferFromStream from './_createBufferFromStream.js'\nimport guessAwsRegion from './_guessAwsRegion.js'\nimport RemoteHandlerAbstract from './abstract'\nimport { basename, join, split } from './_path'\nimport { asyncEach } from '@vates/async-each'\n\n// endpoints https://docs.aws.amazon.com/general/latest/gr/s3.html\n\n// limits: https://docs.aws.amazon.com/AmazonS3/latest/dev/qfacts.html\nconst MIN_PART_SIZE = 1024 * 1024 * 5 // 5MB\nconst MAX_PART_SIZE = 1024 * 1024 * 1024 * 5 // 5GB\nconst MAX_PARTS_COUNT = 10000\nconst MAX_OBJECT_SIZE = 1024 * 1024 * 1024 * 1024 * 5 // 5TB\nconst IDEAL_FRAGMENT_SIZE = Math.ceil(MAX_OBJECT_SIZE / MAX_PARTS_COUNT) // the smallest fragment size that still allows a 5TB upload in 10000 fragments, about 524MB\n\nconst { warn } = createLogger('xo:fs:s3')\n\nexport default class S3Handler extends RemoteHandlerAbstract {\n constructor(remote, _opts) {\n super(remote)\n const {\n allowUnauthorized,\n host,\n path,\n username,\n password,\n protocol,\n region = guessAwsRegion(host),\n } = parse(remote.url)\n\n this._s3 = new S3Client({\n apiVersion: '2006-03-01',\n endpoint: `${protocol}://${host}`,\n forcePathStyle: true,\n credentials: {\n accessKeyId: username,\n secretAccessKey: password,\n },\n tls: protocol === 'https',\n region,\n requestHandler: new NodeHttpHandler({\n socketTimeout: 600000,\n httpAgent: new HttpAgent({\n keepAlive: true,\n }),\n httpsAgent: new HttpsAgent({\n rejectUnauthorized: !allowUnauthorized,\n keepAlive: true,\n }),\n }),\n })\n\n const parts = split(path)\n this._bucket = parts.shift()\n this._dir = join(...parts)\n }\n\n get type() {\n return 's3'\n }\n\n _makeCopySource(path) {\n return join(this._bucket, this._dir, path)\n }\n\n _makeKey(file) {\n return join(this._dir, file)\n }\n\n _makePrefix(dir) {\n return join(this._dir, dir, '/')\n }\n\n _createParams(file) {\n return { Bucket: this._bucket, Key: this._makeKey(file) }\n }\n\n async _multipartCopy(oldPath, newPath) {\n const size = await this._getSize(oldPath)\n const CopySource = this._makeCopySource(oldPath)\n const multipartParams = await this._s3.send(new CreateMultipartUploadCommand({ ...this._createParams(newPath) }))\n try {\n const parts = []\n let start = 0\n while (start < size) {\n const partNumber = parts.length + 1\n const upload = await this._s3.send(\n new UploadPartCopyCommand({\n ...multipartParams,\n CopySource,\n CopySourceRange: `bytes=${start}-${Math.min(start + MAX_PART_SIZE, size) - 1}`,\n PartNumber: partNumber,\n })\n )\n parts.push({ ETag: upload.CopyPartResult.ETag, PartNumber: partNumber })\n start += MAX_PART_SIZE\n }\n await this._s3.send(\n new CompleteMultipartUploadCommand({\n ...multipartParams,\n MultipartUpload: { Parts: parts },\n })\n )\n } catch (e) {\n await this._s3.send(new AbortMultipartUploadCommand(multipartParams))\n throw e\n }\n }\n\n async _copy(oldPath, newPath) {\n const CopySource = this._makeCopySource(oldPath)\n try {\n await this._s3.send(\n new CopyObjectCommand({\n ...this._createParams(newPath),\n CopySource,\n })\n )\n } catch (e) {\n // object > 5GB must be copied part by part\n if (e.name === 'EntityTooLarge') {\n return this._multipartCopy(oldPath, newPath)\n }\n throw e\n }\n }\n\n async _isNotEmptyDir(path) {\n const result = await this._s3.send(\n new ListObjectsV2Command({\n Bucket: this._bucket,\n MaxKeys: 1,\n Prefix: this._makePrefix(path),\n })\n )\n return result.Contents?.length > 0\n }\n\n async _isFile(path) {\n try {\n await this._s3.send(new HeadObjectCommand(this._createParams(path)))\n return true\n } catch (error) {\n if (error.name === 'NotFound') {\n return false\n }\n throw error\n }\n }\n\n async _outputStream(path, input, { validator }) {\n // Workaround for \"ReferenceError: ReadableStream is not defined\"\n // https://github.com/aws/aws-sdk-js-v3/issues/2522\n const Body = new PassThrough()\n pipeline(input, Body, () => {})\n\n const upload = new Upload({\n client: this._s3,\n queueSize: 1,\n partSize: IDEAL_FRAGMENT_SIZE,\n params: {\n ...this._createParams(path),\n Body,\n },\n })\n\n await upload.done()\n\n if (validator !== undefined) {\n try {\n await validator.call(this, path)\n } catch (error) {\n await this.unlink(path)\n throw error\n }\n }\n }\n\n // some objectstorage provider like backblaze, can answer a 500/503 routinely\n // in this case we should retry, and let their load balancing do its magic\n // https://www.backblaze.com/b2/docs/calling.html#error_handling\n @decorateWith(pRetry.wrap, {\n delays: [100, 200, 500, 1000, 2000],\n when: e => e.$metadata?.httpStatusCode === 500,\n onRetry(error) {\n warn('retrying writing file', {\n attemptNumber: this.attemptNumber,\n delay: this.delay,\n error,\n file: this.arguments[0],\n })\n },\n })\n async _writeFile(file, data, options) {\n return this._s3.send(\n new PutObjectCommand({\n ...this._createParams(file),\n Body: data,\n })\n )\n }\n\n async _createReadStream(path, options) {\n if (!(await this._isFile(path))) {\n const error = new Error(`ENOENT: no such file '${path}'`)\n error.code = 'ENOENT'\n error.path = path\n throw error\n }\n\n return (await this._s3.send(new GetObjectCommand(this._createParams(path)))).Body\n }\n\n async _unlink(path) {\n await this._s3.send(new DeleteObjectCommand(this._createParams(path)))\n\n if (await this._isNotEmptyDir(path)) {\n const error = new Error(`EISDIR: illegal operation on a directory, unlink '${path}'`)\n error.code = 'EISDIR'\n error.path = path\n throw error\n }\n }\n\n async _list(dir) {\n let NextContinuationToken\n const uniq = new Set()\n const Prefix = this._makePrefix(dir)\n\n do {\n const result = await this._s3.send(\n new ListObjectsV2Command({\n Bucket: this._bucket,\n Prefix,\n Delimiter: '/',\n // will only return path until delimiters\n ContinuationToken: NextContinuationToken,\n })\n )\n\n if (result.IsTruncated) {\n warn(`need pagination to browse the directory ${dir} completely`)\n NextContinuationToken = result.NextContinuationToken\n } else {\n NextContinuationToken = undefined\n }\n\n // subdirectories\n for (const entry of result.CommonPrefixes ?? []) {\n uniq.add(basename(entry.Prefix))\n }\n\n // files\n for (const entry of result.Contents ?? []) {\n uniq.add(basename(entry.Key))\n }\n } while (NextContinuationToken !== undefined)\n\n return [...uniq]\n }\n\n async _mkdir(path) {\n if (await this._isFile(path)) {\n const error = new Error(`ENOTDIR: file already exists, mkdir '${path}'`)\n error.code = 'ENOTDIR'\n error.path = path\n throw error\n }\n // nothing to do, directories do not exist, they are part of the files' path\n }\n\n // s3 doesn't have a rename operation, so copy + delete source\n async _rename(oldPath, newPath) {\n await this.copy(oldPath, newPath)\n await this._s3.send(new DeleteObjectCommand(this._createParams(oldPath)))\n }\n\n async _getSize(file) {\n if (typeof file !== 'string') {\n file = file.fd\n }\n const result = await this._s3.send(new HeadObjectCommand(this._createParams(file)))\n return +result.ContentLength\n }\n\n async _read(file, buffer, position = 0) {\n if (typeof file !== 'string') {\n file = file.fd\n }\n const params = this._createParams(file)\n params.Range = `bytes=${position}-${position + buffer.length - 1}`\n try {\n const result = await this._s3.send(new GetObjectCommand(params))\n const bytesRead = await copyStreamToBuffer(result.Body, buffer)\n return { bytesRead, buffer }\n } catch (e) {\n if (e.name === 'NoSuchKey') {\n if (await this._isNotEmptyDir(file)) {\n const error = new Error(`${file} is a directory`)\n error.code = 'EISDIR'\n error.path = file\n throw error\n }\n }\n throw e\n }\n }\n\n async _rmdir(path) {\n if (await this._isNotEmptyDir(path)) {\n const error = new Error(`ENOTEMPTY: directory not empty, rmdir '${path}`)\n error.code = 'ENOTEMPTY'\n error.path = path\n throw error\n }\n\n // nothing to do, directories do not exist, they are part of the files' path\n }\n\n // reimplement _rmtree to handle efficiantly path with more than 1000 entries in trees\n // @todo : use parallel processing for unlink\n async _rmtree(path) {\n let NextContinuationToken\n const Prefix = this._makePrefix(path)\n do {\n const result = await this._s3.send(\n new ListObjectsV2Command({\n Bucket: this._bucket,\n Prefix,\n ContinuationToken: NextContinuationToken,\n })\n )\n\n NextContinuationToken = result.IsTruncated ? result.NextContinuationToken : undefined\n await asyncEach(\n result.Contents ?? [],\n async ({ Key }) => {\n // _unlink will add the prefix, but Key contains everything\n // also we don't need to check if we delete a directory, since the list only return files\n await this._s3.send(\n new DeleteObjectCommand({\n Bucket: this._bucket,\n Key,\n })\n )\n },\n {\n concurrency: 16,\n }\n )\n } while (NextContinuationToken !== undefined)\n }\n\n async _write(file, buffer, position) {\n if (typeof file !== 'string') {\n file = file.fd\n }\n const uploadParams = this._createParams(file)\n let fileSize\n try {\n fileSize = +(await this._s3.send(new HeadObjectCommand(uploadParams))).ContentLength\n } catch (e) {\n if (e.name === 'NotFound') {\n fileSize = 0\n } else {\n throw e\n }\n }\n if (fileSize < MIN_PART_SIZE) {\n const resultBuffer = Buffer.alloc(Math.max(fileSize, position + buffer.length))\n if (fileSize !== 0) {\n const result = await this._s3.send(new GetObjectCommand(uploadParams))\n await copyStreamToBuffer(result.Body, resultBuffer)\n } else {\n Buffer.alloc(0).copy(resultBuffer)\n }\n buffer.copy(resultBuffer, position)\n await this._s3.send(\n new PutObjectCommand({\n ...uploadParams,\n Body: resultBuffer,\n })\n )\n return { buffer, bytesWritten: buffer.length }\n } else {\n // using this trick: https://stackoverflow.com/a/38089437/72637\n // multipart fragments have a minimum size of 5Mo and a max of 5Go unless they are last\n // splitting the file in 3 parts: [prefix, edit, suffix]\n // if `prefix` is bigger than 5Mo, it will be sourced from uploadPartCopy()\n // otherwise otherwise it will be downloaded, concatenated to `edit`\n // `edit` will always be an upload part\n // `suffix` will always be sourced from uploadPartCopy()\n // Then everything will be sliced in 5Gb parts before getting uploaded\n const multipartParams = await this._s3.send(new CreateMultipartUploadCommand(uploadParams))\n const copyMultipartParams = {\n ...multipartParams,\n CopySource: this._makeCopySource(file),\n }\n try {\n const parts = []\n const prefixSize = position\n let suffixOffset = prefixSize + buffer.length\n let suffixSize = Math.max(0, fileSize - suffixOffset)\n let hasSuffix = suffixSize > 0\n let editBuffer = buffer\n let editBufferOffset = position\n let partNumber = 1\n let prefixPosition = 0\n // use floor() so that last fragment is handled in the if bellow\n let fragmentsCount = Math.floor(prefixSize / MAX_PART_SIZE)\n const prefixFragmentSize = MAX_PART_SIZE\n let prefixLastFragmentSize = prefixSize - prefixFragmentSize * fragmentsCount\n if (prefixLastFragmentSize >= MIN_PART_SIZE) {\n // the last fragment of the prefix is smaller than MAX_PART_SIZE, but bigger than the minimum\n // so we can copy it too\n fragmentsCount++\n prefixLastFragmentSize = 0\n }\n for (let i = 0; i < fragmentsCount; i++) {\n const fragmentEnd = Math.min(prefixPosition + prefixFragmentSize, prefixSize)\n assert.strictEqual(fragmentEnd - prefixPosition <= MAX_PART_SIZE, true)\n const range = `bytes=${prefixPosition}-${fragmentEnd - 1}`\n const copyPrefixParams = { ...copyMultipartParams, PartNumber: partNumber++, CopySourceRange: range }\n const part = await this._s3.send(new UploadPartCopyCommand(copyPrefixParams))\n parts.push({ ETag: part.CopyPartResult.ETag, PartNumber: copyPrefixParams.PartNumber })\n prefixPosition += prefixFragmentSize\n }\n if (prefixLastFragmentSize) {\n // grab everything from the prefix that was too small to be copied, download and merge to the edit buffer.\n const downloadParams = { ...uploadParams, Range: `bytes=${prefixPosition}-${prefixSize - 1}` }\n let prefixBuffer\n if (prefixSize > 0) {\n const result = await this._s3.send(new GetObjectCommand(downloadParams))\n prefixBuffer = await createBufferFromStream(result.Body)\n } else {\n prefixBuffer = Buffer.alloc(0)\n }\n editBuffer = Buffer.concat([prefixBuffer, buffer])\n editBufferOffset -= prefixLastFragmentSize\n }\n if (hasSuffix && editBuffer.length < MIN_PART_SIZE) {\n // the edit fragment is too short and is not the last fragment\n // let's steal from the suffix fragment to reach the minimum size\n // the suffix might be too short and itself entirely absorbed in the edit fragment, making it the last one.\n const complementSize = Math.min(MIN_PART_SIZE - editBuffer.length, suffixSize)\n const complementOffset = editBufferOffset + editBuffer.length\n suffixOffset += complementSize\n suffixSize -= complementSize\n hasSuffix = suffixSize > 0\n const prefixRange = `bytes=${complementOffset}-${complementOffset + complementSize - 1}`\n const downloadParams = { ...uploadParams, Range: prefixRange }\n const result = await this._s3.send(new GetObjectCommand(downloadParams))\n const complementBuffer = await createBufferFromStream(result.Body)\n editBuffer = Buffer.concat([editBuffer, complementBuffer])\n }\n const editParams = { ...multipartParams, Body: editBuffer, PartNumber: partNumber++ }\n const editPart = await this._s3.send(new UploadPartCommand(editParams))\n parts.push({ ETag: editPart.ETag, PartNumber: editParams.PartNumber })\n if (hasSuffix) {\n // use ceil because the last fragment can be arbitrarily small.\n const suffixFragments = Math.ceil(suffixSize / MAX_PART_SIZE)\n let suffixFragmentOffset = suffixOffset\n for (let i = 0; i < suffixFragments; i++) {\n const fragmentEnd = suffixFragmentOffset + MAX_PART_SIZE\n assert.strictEqual(Math.min(fileSize, fragmentEnd) - suffixFragmentOffset <= MAX_PART_SIZE, true)\n const suffixRange = `bytes=${suffixFragmentOffset}-${Math.min(fileSize, fragmentEnd) - 1}`\n const copySuffixParams = { ...copyMultipartParams, PartNumber: partNumber++, CopySourceRange: suffixRange }\n const suffixPart = (await this._s3.send(new UploadPartCopyCommand(copySuffixParams))).CopyPartResult\n parts.push({ ETag: suffixPart.ETag, PartNumber: copySuffixParams.PartNumber })\n suffixFragmentOffset = fragmentEnd\n }\n }\n await this._s3.send(\n new CompleteMultipartUploadCommand({\n ...multipartParams,\n MultipartUpload: { Parts: parts },\n })\n )\n } catch (e) {\n await this._s3.send(new AbortMultipartUploadCommand(multipartParams))\n throw e\n }\n }\n }\n\n async _openFile(path, flags) {\n return path\n }\n\n async _closeFile(fd) {}\n}\n"],"file":"s3.js"}
1
+ {"version":3,"sources":["../src/s3.js"],"names":["MIN_PART_SIZE","MAX_PART_SIZE","MAX_PARTS_COUNT","MAX_OBJECT_SIZE","IDEAL_FRAGMENT_SIZE","Math","ceil","warn","S3Handler","pRetry","wrap","delays","when","e","$metadata","httpStatusCode","onRetry","error","attemptNumber","delay","file","arguments","RemoteHandlerAbstract","constructor","remote","_opts","allowUnauthorized","host","path","username","password","protocol","region","url","_s3","S3Client","apiVersion","endpoint","forcePathStyle","credentials","accessKeyId","secretAccessKey","tls","requestHandler","NodeHttpHandler","socketTimeout","httpAgent","HttpAgent","keepAlive","httpsAgent","HttpsAgent","rejectUnauthorized","middlewareStack","use","config","parts","_bucket","shift","_dir","type","_makeCopySource","_makeKey","_makePrefix","dir","_createParams","Bucket","Key","_multipartCopy","oldPath","newPath","size","_getSize","CopySource","multipartParams","send","CreateMultipartUploadCommand","start","partNumber","length","upload","UploadPartCopyCommand","CopySourceRange","min","PartNumber","push","ETag","CopyPartResult","CompleteMultipartUploadCommand","MultipartUpload","Parts","AbortMultipartUploadCommand","_copy","CopyObjectCommand","name","_isNotEmptyDir","result","ListObjectsV2Command","MaxKeys","Prefix","Contents","_isFile","HeadObjectCommand","_outputStream","input","validator","Body","PassThrough","Upload","client","queueSize","partSize","params","done","undefined","call","unlink","_writeFile","data","options","PutObjectCommand","_createReadStream","Error","code","GetObjectCommand","_unlink","DeleteObjectCommand","_list","NextContinuationToken","uniq","Set","Delimiter","ContinuationToken","IsTruncated","entry","CommonPrefixes","add","_mkdir","_rename","copy","fd","ContentLength","_read","buffer","position","Range","bytesRead","_rmdir","_rmtree","concurrency","_write","uploadParams","fileSize","resultBuffer","Buffer","alloc","max","bytesWritten","copyMultipartParams","prefixSize","suffixOffset","suffixSize","hasSuffix","editBuffer","editBufferOffset","prefixPosition","fragmentsCount","floor","prefixFragmentSize","prefixLastFragmentSize","i","fragmentEnd","assert","strictEqual","range","copyPrefixParams","part","downloadParams","prefixBuffer","concat","complementSize","complementOffset","prefixRange","complementBuffer","editParams","editPart","UploadPartCommand","suffixFragments","suffixFragmentOffset","suffixRange","copySuffixParams","suffixPart","_openFile","flags","_closeFile"],"mappings":"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","sourcesContent":["import {\n AbortMultipartUploadCommand,\n CompleteMultipartUploadCommand,\n CopyObjectCommand,\n CreateMultipartUploadCommand,\n DeleteObjectCommand,\n GetObjectCommand,\n HeadObjectCommand,\n ListObjectsV2Command,\n PutObjectCommand,\n S3Client,\n UploadPartCommand,\n UploadPartCopyCommand,\n} from '@aws-sdk/client-s3'\nimport { Upload } from '@aws-sdk/lib-storage'\nimport { NodeHttpHandler } from '@aws-sdk/node-http-handler'\nimport { getApplyMd5BodyChecksumPlugin } from '@aws-sdk/middleware-apply-body-checksum'\nimport assert from 'assert'\nimport { Agent as HttpAgent } from 'http'\nimport { Agent as HttpsAgent } from 'https'\nimport pRetry from 'promise-toolbox/retry'\nimport { createLogger } from '@xen-orchestra/log'\nimport { decorateWith } from '@vates/decorate-with'\nimport { PassThrough, pipeline } from 'stream'\nimport { parse } from 'xo-remote-parser'\nimport copyStreamToBuffer from './_copyStreamToBuffer.js'\nimport createBufferFromStream from './_createBufferFromStream.js'\nimport guessAwsRegion from './_guessAwsRegion.js'\nimport RemoteHandlerAbstract from './abstract'\nimport { basename, join, split } from './_path'\nimport { asyncEach } from '@vates/async-each'\n\n// endpoints https://docs.aws.amazon.com/general/latest/gr/s3.html\n\n// limits: https://docs.aws.amazon.com/AmazonS3/latest/dev/qfacts.html\nconst MIN_PART_SIZE = 1024 * 1024 * 5 // 5MB\nconst MAX_PART_SIZE = 1024 * 1024 * 1024 * 5 // 5GB\nconst MAX_PARTS_COUNT = 10000\nconst MAX_OBJECT_SIZE = 1024 * 1024 * 1024 * 1024 * 5 // 5TB\nconst IDEAL_FRAGMENT_SIZE = Math.ceil(MAX_OBJECT_SIZE / MAX_PARTS_COUNT) // the smallest fragment size that still allows a 5TB upload in 10000 fragments, about 524MB\n\nconst { warn } = createLogger('xo:fs:s3')\n\nexport default class S3Handler extends RemoteHandlerAbstract {\n constructor(remote, _opts) {\n super(remote)\n const {\n allowUnauthorized,\n host,\n path,\n username,\n password,\n protocol,\n region = guessAwsRegion(host),\n } = parse(remote.url)\n\n this._s3 = new S3Client({\n apiVersion: '2006-03-01',\n endpoint: `${protocol}://${host}`,\n forcePathStyle: true,\n credentials: {\n accessKeyId: username,\n secretAccessKey: password,\n },\n tls: protocol === 'https',\n region,\n requestHandler: new NodeHttpHandler({\n socketTimeout: 600000,\n httpAgent: new HttpAgent({\n keepAlive: true,\n }),\n httpsAgent: new HttpsAgent({\n rejectUnauthorized: !allowUnauthorized,\n keepAlive: true,\n }),\n }),\n })\n\n // Workaround for https://github.com/aws/aws-sdk-js-v3/issues/2673\n this._s3.middlewareStack.use(\n getApplyMd5BodyChecksumPlugin(this._s3.config)\n )\n\n const parts = split(path)\n this._bucket = parts.shift()\n this._dir = join(...parts)\n }\n\n get type() {\n return 's3'\n }\n\n _makeCopySource(path) {\n return join(this._bucket, this._dir, path)\n }\n\n _makeKey(file) {\n return join(this._dir, file)\n }\n\n _makePrefix(dir) {\n return join(this._dir, dir, '/')\n }\n\n _createParams(file) {\n return { Bucket: this._bucket, Key: this._makeKey(file) }\n }\n\n async _multipartCopy(oldPath, newPath) {\n const size = await this._getSize(oldPath)\n const CopySource = this._makeCopySource(oldPath)\n const multipartParams = await this._s3.send(new CreateMultipartUploadCommand({ ...this._createParams(newPath) }))\n try {\n const parts = []\n let start = 0\n while (start < size) {\n const partNumber = parts.length + 1\n const upload = await this._s3.send(\n new UploadPartCopyCommand({\n ...multipartParams,\n CopySource,\n CopySourceRange: `bytes=${start}-${Math.min(start + MAX_PART_SIZE, size) - 1}`,\n PartNumber: partNumber,\n })\n )\n parts.push({ ETag: upload.CopyPartResult.ETag, PartNumber: partNumber })\n start += MAX_PART_SIZE\n }\n await this._s3.send(\n new CompleteMultipartUploadCommand({\n ...multipartParams,\n MultipartUpload: { Parts: parts },\n })\n )\n } catch (e) {\n await this._s3.send(new AbortMultipartUploadCommand(multipartParams))\n throw e\n }\n }\n\n async _copy(oldPath, newPath) {\n const CopySource = this._makeCopySource(oldPath)\n try {\n await this._s3.send(\n new CopyObjectCommand({\n ...this._createParams(newPath),\n CopySource,\n })\n )\n } catch (e) {\n // object > 5GB must be copied part by part\n if (e.name === 'EntityTooLarge') {\n return this._multipartCopy(oldPath, newPath)\n }\n throw e\n }\n }\n\n async _isNotEmptyDir(path) {\n const result = await this._s3.send(\n new ListObjectsV2Command({\n Bucket: this._bucket,\n MaxKeys: 1,\n Prefix: this._makePrefix(path),\n })\n )\n return result.Contents?.length > 0\n }\n\n async _isFile(path) {\n try {\n await this._s3.send(new HeadObjectCommand(this._createParams(path)))\n return true\n } catch (error) {\n if (error.name === 'NotFound') {\n return false\n }\n throw error\n }\n }\n\n async _outputStream(path, input, { validator }) {\n // Workaround for \"ReferenceError: ReadableStream is not defined\"\n // https://github.com/aws/aws-sdk-js-v3/issues/2522\n const Body = new PassThrough()\n pipeline(input, Body, () => {})\n\n const upload = new Upload({\n client: this._s3,\n queueSize: 1,\n partSize: IDEAL_FRAGMENT_SIZE,\n params: {\n ...this._createParams(path),\n Body,\n },\n })\n\n await upload.done()\n\n if (validator !== undefined) {\n try {\n await validator.call(this, path)\n } catch (error) {\n await this.unlink(path)\n throw error\n }\n }\n }\n\n // some objectstorage provider like backblaze, can answer a 500/503 routinely\n // in this case we should retry, and let their load balancing do its magic\n // https://www.backblaze.com/b2/docs/calling.html#error_handling\n @decorateWith(pRetry.wrap, {\n delays: [100, 200, 500, 1000, 2000],\n when: e => e.$metadata?.httpStatusCode === 500,\n onRetry(error) {\n warn('retrying writing file', {\n attemptNumber: this.attemptNumber,\n delay: this.delay,\n error,\n file: this.arguments[0],\n })\n },\n })\n async _writeFile(file, data, options) {\n return this._s3.send(\n new PutObjectCommand({\n ...this._createParams(file),\n Body: data,\n })\n )\n }\n\n async _createReadStream(path, options) {\n if (!(await this._isFile(path))) {\n const error = new Error(`ENOENT: no such file '${path}'`)\n error.code = 'ENOENT'\n error.path = path\n throw error\n }\n\n return (await this._s3.send(new GetObjectCommand(this._createParams(path)))).Body\n }\n\n async _unlink(path) {\n await this._s3.send(new DeleteObjectCommand(this._createParams(path)))\n\n if (await this._isNotEmptyDir(path)) {\n const error = new Error(`EISDIR: illegal operation on a directory, unlink '${path}'`)\n error.code = 'EISDIR'\n error.path = path\n throw error\n }\n }\n\n async _list(dir) {\n let NextContinuationToken\n const uniq = new Set()\n const Prefix = this._makePrefix(dir)\n\n do {\n const result = await this._s3.send(\n new ListObjectsV2Command({\n Bucket: this._bucket,\n Prefix,\n Delimiter: '/',\n // will only return path until delimiters\n ContinuationToken: NextContinuationToken,\n })\n )\n\n if (result.IsTruncated) {\n warn(`need pagination to browse the directory ${dir} completely`)\n NextContinuationToken = result.NextContinuationToken\n } else {\n NextContinuationToken = undefined\n }\n\n // subdirectories\n for (const entry of result.CommonPrefixes ?? []) {\n uniq.add(basename(entry.Prefix))\n }\n\n // files\n for (const entry of result.Contents ?? []) {\n uniq.add(basename(entry.Key))\n }\n } while (NextContinuationToken !== undefined)\n\n return [...uniq]\n }\n\n async _mkdir(path) {\n if (await this._isFile(path)) {\n const error = new Error(`ENOTDIR: file already exists, mkdir '${path}'`)\n error.code = 'ENOTDIR'\n error.path = path\n throw error\n }\n // nothing to do, directories do not exist, they are part of the files' path\n }\n\n // s3 doesn't have a rename operation, so copy + delete source\n async _rename(oldPath, newPath) {\n await this.copy(oldPath, newPath)\n await this._s3.send(new DeleteObjectCommand(this._createParams(oldPath)))\n }\n\n async _getSize(file) {\n if (typeof file !== 'string') {\n file = file.fd\n }\n const result = await this._s3.send(new HeadObjectCommand(this._createParams(file)))\n return +result.ContentLength\n }\n\n async _read(file, buffer, position = 0) {\n if (typeof file !== 'string') {\n file = file.fd\n }\n const params = this._createParams(file)\n params.Range = `bytes=${position}-${position + buffer.length - 1}`\n try {\n const result = await this._s3.send(new GetObjectCommand(params))\n const bytesRead = await copyStreamToBuffer(result.Body, buffer)\n return { bytesRead, buffer }\n } catch (e) {\n if (e.name === 'NoSuchKey') {\n if (await this._isNotEmptyDir(file)) {\n const error = new Error(`${file} is a directory`)\n error.code = 'EISDIR'\n error.path = file\n throw error\n }\n }\n throw e\n }\n }\n\n async _rmdir(path) {\n if (await this._isNotEmptyDir(path)) {\n const error = new Error(`ENOTEMPTY: directory not empty, rmdir '${path}`)\n error.code = 'ENOTEMPTY'\n error.path = path\n throw error\n }\n\n // nothing to do, directories do not exist, they are part of the files' path\n }\n\n // reimplement _rmtree to handle efficiantly path with more than 1000 entries in trees\n // @todo : use parallel processing for unlink\n async _rmtree(path) {\n let NextContinuationToken\n const Prefix = this._makePrefix(path)\n do {\n const result = await this._s3.send(\n new ListObjectsV2Command({\n Bucket: this._bucket,\n Prefix,\n ContinuationToken: NextContinuationToken,\n })\n )\n\n NextContinuationToken = result.IsTruncated ? result.NextContinuationToken : undefined\n await asyncEach(\n result.Contents ?? [],\n async ({ Key }) => {\n // _unlink will add the prefix, but Key contains everything\n // also we don't need to check if we delete a directory, since the list only return files\n await this._s3.send(\n new DeleteObjectCommand({\n Bucket: this._bucket,\n Key,\n })\n )\n },\n {\n concurrency: 16,\n }\n )\n } while (NextContinuationToken !== undefined)\n }\n\n async _write(file, buffer, position) {\n if (typeof file !== 'string') {\n file = file.fd\n }\n const uploadParams = this._createParams(file)\n let fileSize\n try {\n fileSize = +(await this._s3.send(new HeadObjectCommand(uploadParams))).ContentLength\n } catch (e) {\n if (e.name === 'NotFound') {\n fileSize = 0\n } else {\n throw e\n }\n }\n if (fileSize < MIN_PART_SIZE) {\n const resultBuffer = Buffer.alloc(Math.max(fileSize, position + buffer.length))\n if (fileSize !== 0) {\n const result = await this._s3.send(new GetObjectCommand(uploadParams))\n await copyStreamToBuffer(result.Body, resultBuffer)\n } else {\n Buffer.alloc(0).copy(resultBuffer)\n }\n buffer.copy(resultBuffer, position)\n await this._s3.send(\n new PutObjectCommand({\n ...uploadParams,\n Body: resultBuffer,\n })\n )\n return { buffer, bytesWritten: buffer.length }\n } else {\n // using this trick: https://stackoverflow.com/a/38089437/72637\n // multipart fragments have a minimum size of 5Mo and a max of 5Go unless they are last\n // splitting the file in 3 parts: [prefix, edit, suffix]\n // if `prefix` is bigger than 5Mo, it will be sourced from uploadPartCopy()\n // otherwise otherwise it will be downloaded, concatenated to `edit`\n // `edit` will always be an upload part\n // `suffix` will always be sourced from uploadPartCopy()\n // Then everything will be sliced in 5Gb parts before getting uploaded\n const multipartParams = await this._s3.send(new CreateMultipartUploadCommand(uploadParams))\n const copyMultipartParams = {\n ...multipartParams,\n CopySource: this._makeCopySource(file),\n }\n try {\n const parts = []\n const prefixSize = position\n let suffixOffset = prefixSize + buffer.length\n let suffixSize = Math.max(0, fileSize - suffixOffset)\n let hasSuffix = suffixSize > 0\n let editBuffer = buffer\n let editBufferOffset = position\n let partNumber = 1\n let prefixPosition = 0\n // use floor() so that last fragment is handled in the if bellow\n let fragmentsCount = Math.floor(prefixSize / MAX_PART_SIZE)\n const prefixFragmentSize = MAX_PART_SIZE\n let prefixLastFragmentSize = prefixSize - prefixFragmentSize * fragmentsCount\n if (prefixLastFragmentSize >= MIN_PART_SIZE) {\n // the last fragment of the prefix is smaller than MAX_PART_SIZE, but bigger than the minimum\n // so we can copy it too\n fragmentsCount++\n prefixLastFragmentSize = 0\n }\n for (let i = 0; i < fragmentsCount; i++) {\n const fragmentEnd = Math.min(prefixPosition + prefixFragmentSize, prefixSize)\n assert.strictEqual(fragmentEnd - prefixPosition <= MAX_PART_SIZE, true)\n const range = `bytes=${prefixPosition}-${fragmentEnd - 1}`\n const copyPrefixParams = { ...copyMultipartParams, PartNumber: partNumber++, CopySourceRange: range }\n const part = await this._s3.send(new UploadPartCopyCommand(copyPrefixParams))\n parts.push({ ETag: part.CopyPartResult.ETag, PartNumber: copyPrefixParams.PartNumber })\n prefixPosition += prefixFragmentSize\n }\n if (prefixLastFragmentSize) {\n // grab everything from the prefix that was too small to be copied, download and merge to the edit buffer.\n const downloadParams = { ...uploadParams, Range: `bytes=${prefixPosition}-${prefixSize - 1}` }\n let prefixBuffer\n if (prefixSize > 0) {\n const result = await this._s3.send(new GetObjectCommand(downloadParams))\n prefixBuffer = await createBufferFromStream(result.Body)\n } else {\n prefixBuffer = Buffer.alloc(0)\n }\n editBuffer = Buffer.concat([prefixBuffer, buffer])\n editBufferOffset -= prefixLastFragmentSize\n }\n if (hasSuffix && editBuffer.length < MIN_PART_SIZE) {\n // the edit fragment is too short and is not the last fragment\n // let's steal from the suffix fragment to reach the minimum size\n // the suffix might be too short and itself entirely absorbed in the edit fragment, making it the last one.\n const complementSize = Math.min(MIN_PART_SIZE - editBuffer.length, suffixSize)\n const complementOffset = editBufferOffset + editBuffer.length\n suffixOffset += complementSize\n suffixSize -= complementSize\n hasSuffix = suffixSize > 0\n const prefixRange = `bytes=${complementOffset}-${complementOffset + complementSize - 1}`\n const downloadParams = { ...uploadParams, Range: prefixRange }\n const result = await this._s3.send(new GetObjectCommand(downloadParams))\n const complementBuffer = await createBufferFromStream(result.Body)\n editBuffer = Buffer.concat([editBuffer, complementBuffer])\n }\n const editParams = { ...multipartParams, Body: editBuffer, PartNumber: partNumber++ }\n const editPart = await this._s3.send(new UploadPartCommand(editParams))\n parts.push({ ETag: editPart.ETag, PartNumber: editParams.PartNumber })\n if (hasSuffix) {\n // use ceil because the last fragment can be arbitrarily small.\n const suffixFragments = Math.ceil(suffixSize / MAX_PART_SIZE)\n let suffixFragmentOffset = suffixOffset\n for (let i = 0; i < suffixFragments; i++) {\n const fragmentEnd = suffixFragmentOffset + MAX_PART_SIZE\n assert.strictEqual(Math.min(fileSize, fragmentEnd) - suffixFragmentOffset <= MAX_PART_SIZE, true)\n const suffixRange = `bytes=${suffixFragmentOffset}-${Math.min(fileSize, fragmentEnd) - 1}`\n const copySuffixParams = { ...copyMultipartParams, PartNumber: partNumber++, CopySourceRange: suffixRange }\n const suffixPart = (await this._s3.send(new UploadPartCopyCommand(copySuffixParams))).CopyPartResult\n parts.push({ ETag: suffixPart.ETag, PartNumber: copySuffixParams.PartNumber })\n suffixFragmentOffset = fragmentEnd\n }\n }\n await this._s3.send(\n new CompleteMultipartUploadCommand({\n ...multipartParams,\n MultipartUpload: { Parts: parts },\n })\n )\n } catch (e) {\n await this._s3.send(new AbortMultipartUploadCommand(multipartParams))\n throw e\n }\n }\n }\n\n async _openFile(path, flags) {\n return path\n }\n\n async _closeFile(fd) {}\n}\n"],"file":"s3.js"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "private": false,
3
3
  "name": "@xen-orchestra/fs",
4
- "version": "1.0.0",
4
+ "version": "1.0.1",
5
5
  "license": "AGPL-3.0-or-later",
6
6
  "description": "The File System for Xen Orchestra backups.",
7
7
  "homepage": "https://github.com/vatesfr/xen-orchestra/tree/master/@xen-orchestra/fs",
@@ -20,6 +20,10 @@
20
20
  "node": ">=14"
21
21
  },
22
22
  "dependencies": {
23
+ "@aws-sdk/client-s3": "^3.54.0",
24
+ "@aws-sdk/lib-storage": "^3.54.0",
25
+ "@aws-sdk/middleware-apply-body-checksum": "^3.58.0",
26
+ "@aws-sdk/node-http-handler": "^3.54.0",
23
27
  "@marsaud/smb2": "^0.18.0",
24
28
  "@sindresorhus/df": "^3.1.1",
25
29
  "@vates/async-each": "^0.1.0",
@@ -27,9 +31,6 @@
27
31
  "@vates/decorate-with": "^2.0.0",
28
32
  "@xen-orchestra/async-map": "^0.1.2",
29
33
  "@xen-orchestra/log": "^0.3.0",
30
- "@aws-sdk/client-s3": "^3.54.0",
31
- "@aws-sdk/lib-storage": "^3.54.0",
32
- "@aws-sdk/node-http-handler": "^3.54.0",
33
34
  "bind-property-descriptor": "^2.0.0",
34
35
  "decorator-synchronized": "^0.6.0",
35
36
  "execa": "^5.0.0",
@@ -52,7 +53,7 @@
52
53
  "async-iterator-to-stream": "^1.1.0",
53
54
  "babel-plugin-lodash": "^3.3.2",
54
55
  "cross-env": "^7.0.2",
55
- "dotenv": "^15.0.0",
56
+ "dotenv": "^16.0.0",
56
57
  "rimraf": "^3.0.0"
57
58
  },
58
59
  "scripts": {