UNPKG

@aws-cdk/core

Version:

AWS Cloud Development Kit Core Library

130 lines 18.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.contentFingerprint = exports.fingerprint = void 0; const crypto = require("crypto"); const fs = require("fs"); const path = require("path"); const ignore_1 = require("./ignore"); const options_1 = require("./options"); const utils_1 = require("./utils"); const BUFFER_SIZE = 8 * 1024; const CTRL_SOH = '\x01'; const CTRL_SOT = '\x02'; const CTRL_ETX = '\x03'; const CR = '\r'; const LF = '\n'; const CRLF = `${CR}${LF}`; /** * Produces fingerprint based on the contents of a single file or an entire directory tree. * * Line endings are converted from CRLF to LF. * * The fingerprint will also include: * 1. An extra string if defined in `options.extra`. * 2. The symlink follow mode value. * * @param fileOrDirectory The directory or file to fingerprint * @param options Fingerprinting options */ function fingerprint(fileOrDirectory, options = {}) { const hash = crypto.createHash('sha256'); _hashField(hash, 'options.extra', options.extraHash || ''); const follow = options.follow || options_1.SymlinkFollowMode.EXTERNAL; _hashField(hash, 'options.follow', follow); // Resolve symlinks in the initial path (for example, the root directory // might be symlinked). It's important that we know the absolute path, so we // can judge if further symlinks inside the target directory are within the // target or not (if we don't resolve, we would test w.r.t. the wrong path). fileOrDirectory = fs.realpathSync(fileOrDirectory); const isDir = fs.statSync(fileOrDirectory).isDirectory(); const rootDirectory = isDir ? fileOrDirectory : path.dirname(fileOrDirectory); const ignoreMode = options.ignoreMode || options_1.IgnoreMode.GLOB; if (ignoreMode != options_1.IgnoreMode.GLOB) { _hashField(hash, 'options.ignoreMode', ignoreMode); } const ignoreStrategy = ignore_1.IgnoreStrategy.fromCopyOptions(options, fileOrDirectory); _processFileOrDirectory(fileOrDirectory, isDir); return hash.digest('hex'); function _processFileOrDirectory(symbolicPath, isRootDir = false, realPath = symbolicPath) { if (!isRootDir && ignoreStrategy.ignores(symbolicPath)) { return; } const stat = fs.lstatSync(realPath); // Use relative path as hash component. Normalize it with forward slashes to ensure // same hash on Windows and Linux. const hashComponent = path.relative(fileOrDirectory, symbolicPath).replace(/\\/g, '/'); if (stat.isSymbolicLink()) { const linkTarget = fs.readlinkSync(realPath); const resolvedLinkTarget = path.resolve(path.dirname(realPath), linkTarget); if (utils_1.shouldFollow(follow, rootDirectory, resolvedLinkTarget)) { _processFileOrDirectory(symbolicPath, false, resolvedLinkTarget); } else { _hashField(hash, `link:${hashComponent}`, linkTarget); } } else if (stat.isFile()) { _hashField(hash, `file:${hashComponent}`, contentFingerprint(realPath)); } else if (stat.isDirectory()) { for (const item of fs.readdirSync(realPath).sort()) { _processFileOrDirectory(path.join(symbolicPath, item), false, path.join(realPath, item)); } } else { throw new Error(`Unable to hash ${symbolicPath}: it is neither a file nor a directory`); } } } exports.fingerprint = fingerprint; function contentFingerprint(file) { const hash = crypto.createHash('sha256'); const buffer = Buffer.alloc(BUFFER_SIZE); // eslint-disable-next-line no-bitwise const fd = fs.openSync(file, fs.constants.O_DSYNC | fs.constants.O_RDONLY | fs.constants.O_SYNC); let size = 0; let isBinary = false; let lastStr = ''; let read = 0; try { while ((read = fs.readSync(fd, buffer, 0, BUFFER_SIZE, null)) !== 0) { const slicedBuffer = buffer.slice(0, read); // Detect if file is binary by checking the first 8k bytes for the // null character (git like implementation) if (size === 0) { isBinary = slicedBuffer.indexOf(0) !== -1; } let dataBuffer = slicedBuffer; if (!isBinary) { // Line endings normalization (CRLF -> LF) const str = buffer.slice(0, read).toString(); // We are going to normalize line endings to LF. So if the current // buffer ends with CR, it could be that the next one starts with // LF so we need to save it for later use. if (new RegExp(`${CR}$`).test(str)) { lastStr += str; continue; } const data = lastStr + str; const normalizedData = data.replace(new RegExp(CRLF, 'g'), LF); dataBuffer = Buffer.from(normalizedData); lastStr = ''; } size += dataBuffer.length; hash.update(dataBuffer); } if (lastStr) { hash.update(Buffer.from(lastStr)); } } finally { fs.closeSync(fd); } return `${size}:${hash.digest('hex')}`; } exports.contentFingerprint = contentFingerprint; function _hashField(hash, header, value) { hash.update(CTRL_SOH).update(header).update(CTRL_SOT).update(value).update(CTRL_ETX); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fingerprint.js","sourceRoot":"","sources":["fingerprint.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { IgnoreStrategy } from './ignore';\nimport { FingerprintOptions, IgnoreMode, SymlinkFollowMode } from './options';\nimport { shouldFollow } from './utils';\n\nconst BUFFER_SIZE = 8 * 1024;\nconst CTRL_SOH = '\\x01';\nconst CTRL_SOT = '\\x02';\nconst CTRL_ETX = '\\x03';\nconst CR = '\\r';\nconst LF = '\\n';\nconst CRLF = `${CR}${LF}`;\n\n/**\n * Produces fingerprint based on the contents of a single file or an entire directory tree.\n *\n * Line endings are converted from CRLF to LF.\n *\n * The fingerprint will also include:\n * 1. An extra string if defined in `options.extra`.\n * 2. The symlink follow mode value.\n *\n * @param fileOrDirectory The directory or file to fingerprint\n * @param options Fingerprinting options\n */\nexport function fingerprint(fileOrDirectory: string, options: FingerprintOptions = { }) {\n  const hash = crypto.createHash('sha256');\n  _hashField(hash, 'options.extra', options.extraHash || '');\n  const follow = options.follow || SymlinkFollowMode.EXTERNAL;\n  _hashField(hash, 'options.follow', follow);\n\n  // Resolve symlinks in the initial path (for example, the root directory\n  // might be symlinked). It's important that we know the absolute path, so we\n  // can judge if further symlinks inside the target directory are within the\n  // target or not (if we don't resolve, we would test w.r.t. the wrong path).\n  fileOrDirectory = fs.realpathSync(fileOrDirectory);\n\n  const isDir = fs.statSync(fileOrDirectory).isDirectory();\n  const rootDirectory = isDir\n    ? fileOrDirectory\n    : path.dirname(fileOrDirectory);\n\n  const ignoreMode = options.ignoreMode || IgnoreMode.GLOB;\n  if (ignoreMode != IgnoreMode.GLOB) {\n    _hashField(hash, 'options.ignoreMode', ignoreMode);\n  }\n\n  const ignoreStrategy = IgnoreStrategy.fromCopyOptions(options, fileOrDirectory);\n  _processFileOrDirectory(fileOrDirectory, isDir);\n\n  return hash.digest('hex');\n\n  function _processFileOrDirectory(symbolicPath: string, isRootDir: boolean = false, realPath = symbolicPath) {\n    if (!isRootDir && ignoreStrategy.ignores(symbolicPath)) {\n      return;\n    }\n\n    const stat = fs.lstatSync(realPath);\n\n    // Use relative path as hash component. Normalize it with forward slashes to ensure\n    // same hash on Windows and Linux.\n    const hashComponent = path.relative(fileOrDirectory, symbolicPath).replace(/\\\\/g, '/');\n\n    if (stat.isSymbolicLink()) {\n      const linkTarget = fs.readlinkSync(realPath);\n      const resolvedLinkTarget = path.resolve(path.dirname(realPath), linkTarget);\n      if (shouldFollow(follow, rootDirectory, resolvedLinkTarget)) {\n        _processFileOrDirectory(symbolicPath, false, resolvedLinkTarget);\n      } else {\n        _hashField(hash, `link:${hashComponent}`, linkTarget);\n      }\n    } else if (stat.isFile()) {\n      _hashField(hash, `file:${hashComponent}`, contentFingerprint(realPath));\n    } else if (stat.isDirectory()) {\n      for (const item of fs.readdirSync(realPath).sort()) {\n        _processFileOrDirectory(path.join(symbolicPath, item), false, path.join(realPath, item));\n      }\n    } else {\n      throw new Error(`Unable to hash ${symbolicPath}: it is neither a file nor a directory`);\n    }\n  }\n}\n\nexport function contentFingerprint(file: string): string {\n  const hash = crypto.createHash('sha256');\n  const buffer = Buffer.alloc(BUFFER_SIZE);\n  // eslint-disable-next-line no-bitwise\n  const fd = fs.openSync(file, fs.constants.O_DSYNC | fs.constants.O_RDONLY | fs.constants.O_SYNC);\n  let size = 0;\n  let isBinary = false;\n  let lastStr = '';\n  let read = 0;\n  try {\n    while ((read = fs.readSync(fd, buffer, 0, BUFFER_SIZE, null)) !== 0) {\n      const slicedBuffer = buffer.slice(0, read);\n\n      // Detect if file is binary by checking the first 8k bytes for the\n      // null character (git like implementation)\n      if (size === 0) {\n        isBinary = slicedBuffer.indexOf(0) !== -1;\n      }\n\n      let dataBuffer = slicedBuffer;\n      if (!isBinary) { // Line endings normalization (CRLF -> LF)\n        const str = buffer.slice(0, read).toString();\n\n        // We are going to normalize line endings to LF. So if the current\n        // buffer ends with CR, it could be that the next one starts with\n        // LF so we need to save it for later use.\n        if (new RegExp(`${CR}$`).test(str)) {\n          lastStr += str;\n          continue;\n        }\n\n        const data = lastStr + str;\n        const normalizedData = data.replace(new RegExp(CRLF, 'g'), LF);\n        dataBuffer = Buffer.from(normalizedData);\n        lastStr = '';\n      }\n\n      size += dataBuffer.length;\n      hash.update(dataBuffer);\n    }\n\n    if (lastStr) {\n      hash.update(Buffer.from(lastStr));\n    }\n  } finally {\n    fs.closeSync(fd);\n  }\n  return `${size}:${hash.digest('hex')}`;\n}\n\nfunction _hashField(hash: crypto.Hash, header: string, value: string | Buffer | DataView) {\n  hash.update(CTRL_SOH).update(header).update(CTRL_SOT).update(value).update(CTRL_ETX);\n}\n"]}