UNPKG

aws-delivlib

Version:

A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.

242 lines • 33 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.get = exports.download = exports.jsonGet = exports.PyPIArtifactIntegrity = exports.NpmArtifactIntegrity = exports.RepositoryIntegrity = exports.ArtifactIntegrity = void 0; const child_process_1 = require("child_process"); const os = __importStar(require("os")); const path = __importStar(require("path")); // eslint-disable-next-line import/no-extraneous-dependencies const adm_zip_1 = __importDefault(require("adm-zip")); // eslint-disable-next-line import/no-extraneous-dependencies const follow = __importStar(require("follow-redirects")); // eslint-disable-next-line import/no-extraneous-dependencies const fs = __importStar(require("fs-extra")); // eslint-disable-next-line import/no-extraneous-dependencies const jstream = __importStar(require("JSONStream")); // eslint-disable-next-line import/no-extraneous-dependencies const tar = __importStar(require("tar")); /** * Integrity class for validating a local artifact against its published counterpart. * * Implementations differ based on the package manager in question. */ class ArtifactIntegrity { /** * Validate a local artifact against its published counterpart. * * @param localArtifactDir The directory of the local artifact. Must contain exactly one file with the appropriate extenstion. */ async validate(localArtifactDir) { const artifactPath = this.findOne(localArtifactDir); const name = this.constructor.name; this.log(`Validating ${artifactPath}`); const workdir = fs.mkdtempSync(path.join(os.tmpdir(), 'integrity-check')); try { const downloaded = path.join(workdir, `${name}.downloaded`); const remote = path.join(workdir, `${name}.remote`); const local = path.join(workdir, `${name}.local`); // parse the artifact name into a package. const pkg = this.parseArtifactName(path.basename(artifactPath)); fs.mkdirSync(remote); fs.mkdirSync(local); // download the package this.log(`Downloading ${pkg.name}@${pkg.version} to ${downloaded}`); await this.download(pkg, downloaded); // extract the downlaoded package this.log(`Extracting remote artifact from ${downloaded} to ${remote}`); await this.extract(downloaded, remote); // extract the local artfiact this.log(`Extracting local artifact from ${artifactPath} to ${local}`); await this.extract(artifactPath, local); this.log(`Comparing ${local} <> ${remote}`); try { (0, child_process_1.execSync)(`diff ${local} ${remote}`, { stdio: ['ignore', 'inherit', 'inherit'] }); } catch (error) { throw new Error(`${name} validation failed`); } this.log('Success'); } finally { fs.removeSync(workdir); } } log(message) { console.log(`${this.constructor.name} | ${message} `); } findOne(dir) { const files = fs.readdirSync(dir).filter(f => f.endsWith(this.ext)); if (files.length === 0) { throw new Error(`No files found in ${dir} with extension ${this.ext}`); } const [first, ...rest] = files; if (rest.length > 0) { throw new Error(`Multiple files found in ${dir} with extension ${this.ext}: ${first}, ${rest.join(', ')}`); } return path.join(dir, first); } } exports.ArtifactIntegrity = ArtifactIntegrity; /** * Integrity class for validating the artifacts produced by this repository against their published counterparts. */ class RepositoryIntegrity { constructor(props) { this.props = props; } /** * Validate the artifacts of this repo against its published counterpart. */ async validate() { // note that run 'release' by default to preserve the version number. // this won't do a bump since the commit we are on is already tagged. const artifacts = this.props.repository.pack(this.props.packCommand ?? 'npx projen release'); let integrity = undefined; for (const artifact of artifacts) { switch (artifact.lang) { case 'js': integrity = new NpmArtifactIntegrity(); break; case 'python': integrity = new PyPIArtifactIntegrity(); break; case 'java': case 'dotnet': case 'go': // we don't have integrity checks for these // artifacts yet. break; default: throw new Error(`Unsupported artifact language: ${artifact.lang}`); } if (integrity) { await integrity.validate(artifact.directory); } } console.log('Validation done'); } } exports.RepositoryIntegrity = RepositoryIntegrity; /** * NpmIntegrity is able to perform integrity checks against packages stored on npmjs.com */ class NpmArtifactIntegrity extends ArtifactIntegrity { constructor() { super(...arguments); this.ext = 'tgz'; } async download(pkg, target) { const tarballUrl = await jsonGet(`https://registry.npmjs.org/${encodeURIComponent(pkg.name)}/${encodeURIComponent(pkg.version)}`, ['dist', 'tarball']); await download(tarballUrl, target); } async extract(file, targetDir) { return tar.x({ cwd: targetDir, file: file, strip: 1 }); } parseArtifactName(artifactName) { // cdk8s@1.0.0-beta.59.jsii.tgz const jsiiArtifact = /(.*)@(.*)\.jsii./; // npm artifact: cdk8s-cli-1.0.0-beta59.tgz // yarn artifact: cdk8s-cli-v1.0.0-beta59.tgz (add a 'v' before the version) const npmOrYarnArtifact = /(.*)-v?(\d.*).(tgz|tar.gz)/; const regex = artifactName.includes('.jsii.') ? jsiiArtifact : npmOrYarnArtifact; const match = artifactName.match(regex); if (!match) { throw new Error(`Unable to parse artifact: ${artifactName}`); } return { name: match[1], version: match[2] }; } } exports.NpmArtifactIntegrity = NpmArtifactIntegrity; /** * PyPIIntegiry is able to perform integiry checks against packages stored on pypi.org */ class PyPIArtifactIntegrity extends ArtifactIntegrity { constructor() { super(...arguments); this.ext = 'whl'; } async download(pkg, target) { const files = await jsonGet(`https://pypi.org/pypi/${encodeURIComponent(pkg.name)}/json`, ['releases', pkg.version]); const wheels = files.filter((f) => f.url.endsWith('whl')).map((f) => f.url); if (wheels.length === 0) { throw new Error(`No wheels found for package ${pkg.name}-${pkg.version}`); } if (wheels.length > 1) { throw new Error(`Multiple wheels found for package ${pkg.name}-${pkg.version}: ${wheels.join(',')}`); } await download(wheels[0], target); } async extract(artifact, target) { const zip = new adm_zip_1.default(artifact); return zip.extractAllTo(target); } parseArtifactName(artifactName) { // cdk8s-1.0.0b63-py3-none-any.whl const regex = /(.*)-v?(\d.*)-py.*.whl/; const match = artifactName.match(regex); if (!match) { throw new Error(`Unable to parse artifact: ${artifactName}`); } return { name: match[1], version: match[2] }; } } exports.PyPIArtifactIntegrity = PyPIArtifactIntegrity; function jsonGet(url, jsonPath) { return get(url, (res, ok, ko) => { const json = jstream.parse(jsonPath); json.once('data', ok); json.once('error', ko); res.pipe(json, { end: true }); }, { headers: { 'Accept': 'application/json', 'Accept-Encoding': 'identity' } }); } exports.jsonGet = jsonGet; async function download(url, targetFile) { return get(url, (res, ok, ko) => { const file = fs.createWriteStream(targetFile); file.on('finish', ok); file.on('error', ko); res.pipe(file, { end: true }); }); } exports.download = download; async function get(url, handler, options = {}) { return new Promise((ok, ko) => { const request = follow.https.get(url, options, (res) => { if (res.statusCode !== 200) { const error = new Error(`GET ${url} - HTTP ${res.statusCode} (${res.statusMessage})`); Error.captureStackTrace(error); return ko(error); } res.once('error', ko); handler(res, ok, ko); }); request.on('error', ko); }); } exports.get = get; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integrity.js","sourceRoot":"","sources":["integrity.ts"],"names":[],"mappings":"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","sourcesContent":["import { execSync } from 'child_process';\nimport type { RequestOptions, IncomingMessage } from 'http';\nimport * as os from 'os';\nimport * as path from 'path';\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport AdmZip from 'adm-zip';\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport * as follow from 'follow-redirects';\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport * as fs from 'fs-extra';\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport * as jstream from 'JSONStream';\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport * as tar from 'tar';\nimport { Repository } from './repository';\n\n\n/**\n * Published package.\n */\nexport interface PublishedPackage {\n\n  /**\n   * Name of the package as stored in the package manager.\n   */\n  readonly name: string;\n\n  /**\n   * Version of the package as stored in the package manager.\n   */\n  readonly version: string;\n}\n\n/**\n * Integrity class for validating a local artifact against its published counterpart.\n *\n * Implementations differ based on the package manager in question.\n */\nexport abstract class ArtifactIntegrity {\n\n  /**\n   * The file extenstion of artifacts produced for this check. (e.g 'whl')\n   */\n  protected abstract readonly ext: string;\n\n  /**\n   * Download a package to the target file.\n   *\n   * @param pkg The package to download.\n   * @param targetFile The file path to download the package to.\n   */\n  protected abstract download(pkg: PublishedPackage, targetFile: string): Promise<void>;\n\n  /**\n   * Extract the artifact into the target directory.\n   *\n   * @param artifact Path to an artifact file.\n   * @param targetDir The directory to extract to. It will exist by the time this method is invoked.\n   */\n  protected abstract extract(artifact: string, targetDir: string): Promise<void>;\n\n  /**\n   * Parse a local artifact file name into a structured package.\n   *\n   * @param artifactName Base name of the local artifact file.\n   * @returns The package this artifact correlates to.\n   */\n  protected abstract parseArtifactName(artifactName: string): PublishedPackage;\n\n  /**\n   * Validate a local artifact against its published counterpart.\n   *\n   * @param localArtifactDir The directory of the local artifact. Must contain exactly one file with the appropriate extenstion.\n   */\n  public async validate(localArtifactDir: string) {\n\n    const artifactPath = this.findOne(localArtifactDir);\n    const name = this.constructor.name;\n\n    this.log(`Validating ${artifactPath}`);\n    const workdir = fs.mkdtempSync(path.join(os.tmpdir(), 'integrity-check'));\n\n    try {\n      const downloaded = path.join(workdir, `${name}.downloaded`);\n      const remote = path.join(workdir, `${name}.remote`);\n      const local = path.join(workdir, `${name}.local`);\n\n      // parse the artifact name into a package.\n      const pkg = this.parseArtifactName(path.basename(artifactPath));\n\n      fs.mkdirSync(remote);\n      fs.mkdirSync(local);\n\n      // download the package\n      this.log(`Downloading ${pkg.name}@${pkg.version} to ${downloaded}`);\n      await this.download(pkg, downloaded);\n\n      // extract the downlaoded package\n      this.log(`Extracting remote artifact from ${downloaded} to ${remote}`);\n      await this.extract(downloaded, remote);\n\n      // extract the local artfiact\n      this.log(`Extracting local artifact from ${artifactPath} to ${local}`);\n      await this.extract(artifactPath, local);\n\n      this.log(`Comparing ${local} <> ${remote}`);\n      try {\n        execSync(`diff ${local} ${remote}`, { stdio: ['ignore', 'inherit', 'inherit'] });\n      } catch (error) {\n        throw new Error(`${name} validation failed`);\n      }\n      this.log('Success');\n\n    } finally {\n      fs.removeSync(workdir);\n    }\n\n  }\n\n  protected log(message: string) {\n    console.log(`${this.constructor.name} | ${message} `);\n  }\n\n  private findOne(dir: string): string {\n    const files = fs.readdirSync(dir).filter(f => f.endsWith(this.ext));\n    if (files.length === 0) {\n      throw new Error(`No files found in ${dir} with extension ${this.ext}`);\n    }\n    const [first, ...rest] = files;\n    if (rest.length > 0) {\n      throw new Error(`Multiple files found in ${dir} with extension ${this.ext}: ${first}, ${rest.join(', ')}`);\n    }\n    return path.join(dir, first);\n  }\n\n}\n\n/**\n * Properties for `RepositoryIntegrity`.\n */\nexport interface RepositoryIntegrityProps {\n  /**\n   * Repository to validate.\n   */\n  readonly repository: Repository;\n\n  /**\n   * The command that produces the local artifacts.\n   *\n   * @default 'npx projen release'\n   */\n  readonly packCommand?: string;\n}\n\n/**\n * Integrity class for validating the artifacts produced by this repository against their published counterparts.\n */\nexport class RepositoryIntegrity {\n\n  public constructor(private readonly props: RepositoryIntegrityProps) {}\n\n  /**\n   * Validate the artifacts of this repo against its published counterpart.\n   */\n  public async validate() {\n\n    // note that run 'release' by default to preserve the version number.\n    // this won't do a bump since the commit we are on is already tagged.\n    const artifacts = this.props.repository.pack(this.props.packCommand ?? 'npx projen release');\n\n    let integrity = undefined;\n    for (const artifact of artifacts) {\n      switch (artifact.lang) {\n        case 'js':\n          integrity = new NpmArtifactIntegrity();\n          break;\n        case 'python':\n          integrity = new PyPIArtifactIntegrity();\n          break;\n        case 'java':\n        case 'dotnet':\n        case 'go':\n          // we don't have integrity checks for these\n          // artifacts yet.\n          break;\n        default:\n          throw new Error(`Unsupported artifact language: ${artifact.lang}`);\n      }\n      if (integrity) {\n        await integrity.validate(artifact.directory);\n      }\n    }\n    console.log('Validation done');\n  }\n\n}\n\n/**\n * NpmIntegrity is able to perform integrity checks against packages stored on npmjs.com\n */\nexport class NpmArtifactIntegrity extends ArtifactIntegrity {\n\n  protected readonly ext = 'tgz';\n\n  protected async download(pkg: PublishedPackage, target: string): Promise<void> {\n    const tarballUrl = await jsonGet(`https://registry.npmjs.org/${encodeURIComponent(pkg.name)}/${encodeURIComponent(pkg.version)}`, ['dist', 'tarball']);\n    await download(tarballUrl, target);\n  }\n\n  public async extract(file: string, targetDir: string): Promise<void> {\n    return tar.x({ cwd: targetDir, file: file, strip: 1 });\n  }\n\n  protected parseArtifactName(artifactName: string): PublishedPackage {\n\n    // cdk8s@1.0.0-beta.59.jsii.tgz\n    const jsiiArtifact = /(.*)@(.*)\\.jsii./;\n\n    // npm artifact: cdk8s-cli-1.0.0-beta59.tgz\n    // yarn artifact: cdk8s-cli-v1.0.0-beta59.tgz (add a 'v' before the version)\n    const npmOrYarnArtifact = /(.*)-v?(\\d.*).(tgz|tar.gz)/;\n\n    const regex = artifactName.includes('.jsii.') ? jsiiArtifact : npmOrYarnArtifact;\n\n    const match = artifactName.match(regex);\n    if (!match) {\n      throw new Error(`Unable to parse artifact: ${artifactName}`);\n    }\n\n    return { name: match[1], version: match[2] };\n\n  }\n\n}\n\n/**\n * PyPIIntegiry is able to perform integiry checks against packages stored on pypi.org\n */\nexport class PyPIArtifactIntegrity extends ArtifactIntegrity {\n\n  protected readonly ext = 'whl';\n\n  protected async download(pkg: PublishedPackage, target: string): Promise<void> {\n\n    const files = await jsonGet(`https://pypi.org/pypi/${encodeURIComponent(pkg.name)}/json`, ['releases', pkg.version]);\n    const wheels: string[] = files.filter((f: any) => f.url.endsWith('whl')).map((f: any) => f.url);\n\n    if (wheels.length === 0) {\n      throw new Error(`No wheels found for package ${pkg.name}-${pkg.version}`);\n    }\n\n    if (wheels.length > 1) {\n      throw new Error(`Multiple wheels found for package ${pkg.name}-${pkg.version}: ${wheels.join(',')}`);\n    }\n\n    await download(wheels[0], target);\n  }\n\n  public async extract(artifact: string, target: string): Promise<void> {\n    const zip = new AdmZip(artifact);\n    return zip.extractAllTo(target);\n  }\n\n  protected parseArtifactName(artifactName: string): PublishedPackage {\n\n    // cdk8s-1.0.0b63-py3-none-any.whl\n    const regex = /(.*)-v?(\\d.*)-py.*.whl/;\n\n    const match = artifactName.match(regex);\n    if (!match) {\n      throw new Error(`Unable to parse artifact: ${artifactName}`);\n    }\n\n    return { name: match[1], version: match[2] };\n\n  }\n\n}\n\nexport function jsonGet(url: string, jsonPath?: string[]): Promise<any> {\n  return get(url, (res, ok, ko) => {\n    const json = jstream.parse(jsonPath);\n    json.once('data', ok);\n    json.once('error', ko);\n\n    res.pipe(json, { end: true });\n  }, { headers: { 'Accept': 'application/json', 'Accept-Encoding': 'identity' } });\n}\n\nexport async function download(url: string, targetFile: string): Promise<any> {\n  return get(url, (res, ok, ko) => {\n    const file = fs.createWriteStream(targetFile);\n    file.on('finish', ok);\n    file.on('error', ko);\n    res.pipe(file, { end: true });\n  });\n}\n\nexport async function get(\n  url: string,\n  handler: (res: IncomingMessage, ok: (value: unknown) => void, ko: (err: Error) => void) => void,\n  options: RequestOptions = {}) {\n\n  return new Promise((ok, ko) => {\n    const request = follow.https.get(url, options, (res: IncomingMessage) => {\n      if (res.statusCode !== 200) {\n        const error = new Error(`GET ${url} - HTTP ${res.statusCode} (${res.statusMessage})`);\n        Error.captureStackTrace(error);\n        return ko(error);\n      }\n      res.once('error', ko);\n      handler(res, ok, ko);\n    });\n    request.on('error', ko);\n  });\n\n}\n"]}