UNPKG

jsii-release

Version:

Release jsii modules to multiple package managers

231 lines 34.4 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; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.GoReleaser = void 0; const path = __importStar(require("path")); const fs = __importStar(require("fs-extra")); const git = __importStar(require("../help/git")); const os = __importStar(require("../help/os")); const shell = __importStar(require("../help/shell")); /** * Release a set of Golang modules. */ class GoReleaser { constructor(props) { var _a, _b, _c, _d, _e; if (!shell.which('git')) { throw new Error('git must be available to in order to be able to push Go code to GitHub'); } this.version = props.version; this.gitCommitMessage = props.message; this.dir = path.resolve((_a = props.dir) !== null && _a !== void 0 ? _a : path.join(process.cwd(), 'dist', 'go')); this.gitBranch = (_b = props.branch) !== null && _b !== void 0 ? _b : 'main'; this.dryRun = (_c = props.dryRun) !== null && _c !== void 0 ? _c : false; const gitUsername = (_d = props.username) !== null && _d !== void 0 ? _d : git.username(); const gitUseremail = (_e = props.email) !== null && _e !== void 0 ? _e : git.email(); if (!gitUsername) { throw new Error('Unable to detect username. either configure a git user.name or pass GIT_USER_NAME env variable'); } if (!gitUseremail) { throw new Error('Unable to detect user email. either configure a git user.email or pass GIT_USER_EMAIL env variable'); } this.gitUseremail = gitUseremail; this.gitUsername = gitUsername; } /** * Run the release process. * * @returns metadata about the release. */ release() { const modules = this.collectModules(this.dir); if (modules.length === 0) { console.log('No modules detected. Skipping'); return {}; } console.log('Detected modules:'); modules.forEach(m => console.log(` - ${m.modFile}`)); const repoURL = this.extractRepoURL(modules); const repoDir = path.join(os.mkdtempSync(), 'repo'); git.clone(repoURL, repoDir); const cwd = process.cwd(); try { process.chdir(repoDir); return this.doRelease(modules, repoDir); } finally { process.chdir(cwd); } } doRelease(modules, repoDir) { var _a; git.identify(this.gitUsername, this.gitUseremail); git.checkout(this.gitBranch, { createIfMissing: true }); this.syncRepo(repoDir); git.add('.'); if (!git.diffIndex()) { console.log('No changes. Skipping release'); return {}; } const commitMessage = (_a = this.gitCommitMessage) !== null && _a !== void 0 ? _a : this.buildReleaseMessage(modules); git.commit(commitMessage); const tags = []; for (const module of modules) { const name = this.buildTagName(module); const created = git.tag(name); if (created) { tags.push(name); } } if (tags.length === 0) { console.log('All tags already exist. Skipping release'); return {}; } const refs = [...tags, this.gitBranch]; if (this.dryRun) { console.log('==========================================='); console.log(' 🏜️ DRY-RUN MODE 🏜️'); console.log('==========================================='); refs.forEach(t => console.log(`Remote ref will be updated: ${t}`)); } else { refs.forEach(t => git.push(t)); } return { tags, commitMessage, repoDir }; } collectModules(dir) { const modules = []; for (const p of [...fs.readdirSync(dir), '.']) { const modFile = path.join(dir, p, 'go.mod'); if (fs.existsSync(modFile)) { modules.push(this.parseModule(modFile)); } } return modules; } parseModule(modFile) { const version = this.extractVersion(path.dirname(modFile)); const majorVersion = Number(version.split('.')[0]); // extract the module decleration (e.g 'module github.com/aws/constructs-go/constructs/v3') const match = fs.readFileSync(modFile).toString().match(/module (.*)/); if (!match || !match[1]) { throw new Error(`Unable to detected module declaration in ${modFile}`); } // e.g 'github.com/aws/constructs-go/constructs/v3' const cannonicalName = match[1]; // e.g ['github.com', 'aws', 'constructs-go', 'constructs', 'v3'] const cannonicalNameParts = cannonicalName.split('/'); // e.g 'github.com/aws/constructs-go' const repoURL = cannonicalNameParts.slice(0, 3).join('/'); // e.g 'v3' const majorVersionSuffix = majorVersion > 1 ? `/v${majorVersion}` : ''; if (!cannonicalName.endsWith(majorVersionSuffix)) { throw new Error(`Module declaration in '${modFile}' expected to end with '${majorVersionSuffix}' since its major version is larger than 1`); } if (!repoURL.startsWith('github.com')) { if (!(git.detectSSH() || git.detectGHE())) { throw new Error(`Repository must be hosted on github.com. Found: '${repoURL}' in ${modFile}`); } } let repoPath = cannonicalNameParts .slice(3) // e.g ['constructs', 'v3'] .join('/'); // e.g 'constructs/v3' // we could have something like // constructs/v3 // or something like // constructsv3/v3 // we only want to strip the last 'v3' const split = repoPath.split('/'); if (split[split.length - 1] === `v${majorVersion}`) { split.pop(); repoPath = split.join('/'); } // strip '/' if exists (wont exist for top level modules) repoPath = repoPath.endsWith('/') ? repoPath.substr(0, repoPath.length - 1) : repoPath; return { modFile, version, cannonicalName, repoPath, repoURL }; } buildTagName(m) { return m.repoPath === '' ? `v${m.version}` : `${m.repoPath}/v${m.version}`; } buildReleaseMessage(modules) { const semantic = 'chore(release)'; const versions = new Set(modules.map(m => m.version)); if (versions.size === 1) { // single version (e.g chore(release): v1.2.3) return `${semantic}: v${versions.values().next().value}`; } else { // multiple versions (e.g chore(release): chore(release): module1@v1.2.3 module2@v1.2.3) return `${semantic}: ${modules.map(m => `${m.repoPath ? `${m.repoPath}@` : ''}v${m.version}`).join(' ')}`; } } syncRepo(repoDir) { const topLevel = path.join(repoDir, 'go.mod'); if (fs.existsSync(topLevel)) { // with top level modules we sync the entire repository // so we just empty it out fs.readdirSync(repoDir) .filter(f => f !== '.git') .forEach(f => fs.removeSync(path.join(repoDir, f))); } else { // otherwise, we selectively remove the submodules only. for (const p of fs.readdirSync(repoDir)) { const submodule = path.join(repoDir, p, 'go.mod'); if (fs.existsSync(submodule)) { fs.removeSync(path.join(repoDir, p)); } } } fs.copySync(this.dir, repoDir, { recursive: true }); } extractRepoURL(modules) { const repos = new Set(modules.map(m => m.repoURL)); if (repos.size === 0) { throw new Error('Unable to detect repository from module files.'); } if (repos.size > 1) { throw new Error('Multiple repositories found in module files'); } return repos.values().next().value; } extractVersion(moduleDirectory) { const versionFile = path.join(moduleDirectory, 'version'); const repoVersion = this.version; const moduleVersion = fs.existsSync(versionFile) ? fs.readFileSync(versionFile).toString() : undefined; if (repoVersion && moduleVersion && repoVersion !== moduleVersion) { throw new Error(`Repo version (${repoVersion}) conflicts with module version (${moduleVersion}) for module in ${moduleDirectory}`); } // just take the one thats defined, they have to the same if both are. const version = moduleVersion ? moduleVersion : repoVersion; if (!version) { throw new Error(`Unable to determine version of module ${moduleDirectory}. ` + "Either include a 'version' file, or specify a global version using the VERSION environment variable."); } return version; } } exports.GoReleaser = GoReleaser; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"go.js","sourceRoot":"","sources":["../../src/targets/go.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as fs from 'fs-extra';\nimport * as git from '../help/git';\nimport * as os from '../help/os';\nimport * as shell from '../help/shell';\n\n/**\n * Encapsulates some information about the release.\n */\nexport interface GoRelease {\n\n  /**\n   * The tags the release created.\n   */\n  readonly tags?: string[];\n\n  /**\n   * The commit message of the release.\n   */\n  readonly commitMessage?: string;\n\n  /**\n   * The directory where the repository was released from.\n   */\n  readonly repoDir?: string;\n}\n\n/**\n * Properties for `GoReleaser`.\n */\nexport interface GoReleaserProps {\n\n  /**\n   * The source code directory.\n   *\n   * @default 'dist/go'\n   */\n  readonly dir?: string;\n\n  /**\n   * Execute a dry run only.\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * The branch to push to.\n   *\n   * @default 'main'\n   */\n  readonly branch?: string;\n\n  /**\n   * The username to use for the commit.\n   *\n   * @default - taken from git config. throws if not configured.\n   */\n  readonly username?: string;\n\n  /**\n   * The email to use for the commit.\n   *\n   * @default - taken from git config. throws if not configured.\n   */\n  readonly email?: string;\n\n  /**\n   * The version.\n   *\n   * @default - taken from the 'version'. throws if doesn't exist.\n   */\n  readonly version?: string;\n\n  /**\n   * The message to use for the commit marking the release.\n   */\n  readonly message?: string;\n}\n\n/**\n * Information about a specific module.\n */\nexport interface GoModule {\n\n  /**\n   * Path to the mod file.\n   */\n  readonly modFile: string;\n\n  /**\n   * The version of the module.\n   */\n  readonly version: string;\n\n  /**\n   * The cannonical name of the module. (e.g 'github.com/aws/constructs-go/constructs/v3`)\n   */\n  readonly cannonicalName: string;\n\n  /**\n   * The path inside the repository the module is located in. (e.g 'constructs')\n   */\n  readonly repoPath: string;\n\n  /**\n   * The repository URL. (e.g 'github.com/aws/constructs')\n   */\n  readonly repoURL: string;\n\n}\n\n/**\n * Release a set of Golang modules.\n */\nexport class GoReleaser {\n\n  private readonly version?: string;\n  private readonly gitCommitMessage?: string;\n\n  private readonly dir: string;\n  private readonly dryRun: boolean;\n  private readonly gitBranch: string;\n  private readonly gitUsername: string;\n  private readonly gitUseremail: string;\n\n  constructor(props: GoReleaserProps) {\n\n    if (!shell.which('git')) {\n      throw new Error('git must be available to in order to be able to push Go code to GitHub');\n    }\n\n    this.version = props.version;\n    this.gitCommitMessage = props.message;\n    this.dir = path.resolve(props.dir ?? path.join(process.cwd(), 'dist', 'go'));\n    this.gitBranch = props.branch ?? 'main';\n    this.dryRun = props.dryRun ?? false;\n\n    const gitUsername = props.username ?? git.username();\n    const gitUseremail = props.email ?? git.email();\n\n    if (!gitUsername) {\n      throw new Error('Unable to detect username. either configure a git user.name or pass GIT_USER_NAME env variable');\n    }\n\n    if (!gitUseremail) {\n      throw new Error('Unable to detect user email. either configure a git user.email or pass GIT_USER_EMAIL env variable');\n    }\n\n    this.gitUseremail = gitUseremail;\n    this.gitUsername = gitUsername;\n  }\n\n  /**\n   * Run the release process.\n   *\n   * @returns metadata about the release.\n   */\n  public release(): GoRelease {\n    const modules = this.collectModules(this.dir);\n    if (modules.length === 0) {\n      console.log('No modules detected. Skipping');\n      return {};\n    }\n\n    console.log('Detected modules:');\n    modules.forEach(m => console.log(` - ${m.modFile}`));\n\n    const repoURL = this.extractRepoURL(modules);\n    const repoDir = path.join(os.mkdtempSync(), 'repo');\n    git.clone(repoURL, repoDir);\n\n    const cwd = process.cwd();\n    try {\n      process.chdir(repoDir);\n      return this.doRelease(modules, repoDir);\n    } finally {\n      process.chdir(cwd);\n    }\n\n  }\n\n  private doRelease(modules: GoModule[], repoDir: string): GoRelease {\n\n    git.identify(this.gitUsername, this.gitUseremail);\n    git.checkout(this.gitBranch, { createIfMissing: true });\n    this.syncRepo(repoDir);\n\n    git.add('.');\n    if (!git.diffIndex()) {\n      console.log('No changes. Skipping release');\n      return {};\n    }\n\n    const commitMessage = this.gitCommitMessage ?? this.buildReleaseMessage(modules);\n    git.commit(commitMessage);\n\n    const tags = [];\n    for (const module of modules) {\n      const name = this.buildTagName(module);\n      const created = git.tag(name);\n      if (created) { tags.push(name); }\n    }\n\n    if (tags.length === 0) {\n      console.log('All tags already exist. Skipping release');\n      return {};\n    }\n\n    const refs = [...tags, this.gitBranch];\n\n    if (this.dryRun) {\n      console.log('===========================================');\n      console.log('            🏜️ DRY-RUN MODE 🏜️');\n      console.log('===========================================');\n      refs.forEach(t => console.log(`Remote ref will be updated: ${t}`));\n    } else {\n      refs.forEach(t => git.push(t));\n    }\n    return { tags, commitMessage, repoDir };\n  }\n\n  private collectModules(dir: string): GoModule[] {\n    const modules: GoModule[] = [];\n    for (const p of [...fs.readdirSync(dir), '.']) {\n      const modFile = path.join(dir, p, 'go.mod');\n      if (fs.existsSync(modFile)) {\n        modules.push(this.parseModule(modFile));\n      }\n    }\n    return modules;\n  }\n\n  private parseModule(modFile: string): GoModule {\n\n    const version = this.extractVersion(path.dirname(modFile));\n    const majorVersion = Number(version.split('.')[0]);\n\n    // extract the module decleration (e.g 'module github.com/aws/constructs-go/constructs/v3')\n    const match = fs.readFileSync(modFile).toString().match(/module (.*)/);\n    if (!match || !match[1]) {\n      throw new Error(`Unable to detected module declaration in ${modFile}`);\n    }\n\n    // e.g 'github.com/aws/constructs-go/constructs/v3'\n    const cannonicalName = match[1];\n\n    // e.g ['github.com', 'aws', 'constructs-go', 'constructs', 'v3']\n    const cannonicalNameParts = cannonicalName.split('/');\n\n    // e.g 'github.com/aws/constructs-go'\n    const repoURL = cannonicalNameParts.slice(0, 3).join('/');\n\n    // e.g 'v3'\n    const majorVersionSuffix = majorVersion > 1 ? `/v${majorVersion}` : '';\n\n    if (!cannonicalName.endsWith(majorVersionSuffix)) {\n      throw new Error(`Module declaration in '${modFile}' expected to end with '${majorVersionSuffix}' since its major version is larger than 1`);\n    }\n\n    if (!repoURL.startsWith('github.com')) {\n      if (!(git.detectSSH() || git.detectGHE())) {\n        throw new Error(`Repository must be hosted on github.com. Found: '${repoURL}' in ${modFile}`);\n      }\n    }\n    let repoPath = cannonicalNameParts\n      .slice(3) // e.g ['constructs', 'v3']\n      .join('/'); // e.g 'constructs/v3'\n\n    // we could have something like\n    // constructs/v3\n    // or something like\n    // constructsv3/v3\n    // we only want to strip the last 'v3'\n    const split = repoPath.split('/');\n    if (split[split.length - 1] === `v${majorVersion}`) {\n      split.pop();\n      repoPath = split.join('/');\n    }\n\n    // strip '/' if exists (wont exist for top level modules)\n    repoPath = repoPath.endsWith('/') ? repoPath.substr(0, repoPath.length - 1) : repoPath;\n\n    return { modFile, version, cannonicalName, repoPath, repoURL };\n\n  }\n\n  private buildTagName(m: GoModule) {\n    return m.repoPath === '' ? `v${m.version}` : `${m.repoPath}/v${m.version}`;\n  }\n\n  private buildReleaseMessage(modules: GoModule[]) {\n    const semantic = 'chore(release)';\n    const versions = new Set(modules.map(m => m.version));\n    if (versions.size === 1) {\n      // single version (e.g chore(release): v1.2.3)\n      return `${semantic}: v${versions.values().next().value}`;\n    } else {\n      // multiple versions (e.g chore(release): chore(release): module1@v1.2.3  module2@v1.2.3)\n      return `${semantic}: ${modules.map(m => `${m.repoPath ? `${m.repoPath}@` : ''}v${m.version}`).join(' ')}`;\n    }\n  }\n\n  private syncRepo(repoDir: string) {\n    const topLevel = path.join(repoDir, 'go.mod');\n    if (fs.existsSync(topLevel)) {\n      // with top level modules we sync the entire repository\n      // so we just empty it out\n      fs.readdirSync(repoDir)\n        .filter(f => f !== '.git')\n        .forEach(f => fs.removeSync(path.join(repoDir, f)));\n    } else {\n      // otherwise, we selectively remove the submodules only.\n      for (const p of fs.readdirSync(repoDir)) {\n        const submodule = path.join(repoDir, p, 'go.mod');\n        if (fs.existsSync(submodule)) {\n          fs.removeSync(path.join(repoDir, p));\n        }\n      }\n    }\n    fs.copySync(this.dir, repoDir, { recursive: true });\n  }\n\n  private extractRepoURL(modules: GoModule[]): string {\n    const repos = new Set<string>(modules.map(m => m.repoURL));\n    if (repos.size === 0) {\n      throw new Error('Unable to detect repository from module files.');\n    }\n    if (repos.size > 1) {\n      throw new Error('Multiple repositories found in module files');\n    }\n    return repos.values().next().value;\n  }\n\n  private extractVersion(moduleDirectory: string): string {\n\n    const versionFile = path.join(moduleDirectory, 'version');\n\n    const repoVersion = this.version;\n    const moduleVersion = fs.existsSync(versionFile) ? fs.readFileSync(versionFile).toString() : undefined;\n\n    if (repoVersion && moduleVersion && repoVersion !== moduleVersion) {\n      throw new Error(`Repo version (${repoVersion}) conflicts with module version (${moduleVersion}) for module in ${moduleDirectory}`);\n    }\n\n    // just take the one thats defined, they have to the same if both are.\n    const version = moduleVersion ? moduleVersion : repoVersion;\n\n    if (!version) {\n      throw new Error(`Unable to determine version of module ${moduleDirectory}. `\n        + \"Either include a 'version' file, or specify a global version using the VERSION environment variable.\");\n    }\n\n    return version;\n\n  }\n\n}\n"]}