jsii-release
Version:
Release jsii modules to multiple package managers
247 lines • 36.5 kB
JavaScript
;
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 () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__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) {
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(props.dir ?? path.join(process.cwd(), 'dist', 'go'));
this.gitBranch = props.branch ?? 'main';
this.dryRun = props.dryRun ?? false;
const gitUsername = props.username ?? git.username();
const gitUseremail = props.email ?? 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');
const branchExists = git.branchExistsOnRemote(repoURL, this.gitBranch);
if (!branchExists) {
console.log(`Remote branch '${this.gitBranch}' not found, continuing with default branch.`);
}
const cloneOptions = {
tags: true, // we need to know about all tags to not re-create an existing one
branch: branchExists ? this.gitBranch : undefined,
};
git.clone(repoURL, repoDir, cloneOptions);
const cwd = process.cwd();
try {
process.chdir(repoDir);
return this.doRelease(modules, repoDir);
}
finally {
process.chdir(cwd);
}
}
doRelease(modules, repoDir) {
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 = this.gitCommitMessage ?? 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 => git.rm(path.join(repoDir, f), { recursive: true }));
}
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)) {
git.rm(path.join(repoDir, p), { recursive: true });
}
}
}
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   * The git clone depth.\n   *\n   * Usually only the latest commit is required.\n   *\n   * @default 1\n   */\n  readonly cloneDepth?: number;\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\n    const branchExists = git.branchExistsOnRemote(repoURL, this.gitBranch);\n    if (!branchExists) {\n      console.log(`Remote branch '${this.gitBranch}' not found, continuing with default branch.`);\n    }\n    const cloneOptions = {\n      tags: true, // we need to know about all tags to not re-create an existing one\n      branch: branchExists ? this.gitBranch : undefined,\n    };\n\n    git.clone(repoURL, repoDir, cloneOptions);\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 => git.rm(path.join(repoDir, f), { recursive: true }));\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          git.rm(path.join(repoDir, p), { recursive: true });\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"]}