angular-cli/scripts/snapshots.mts
Joey Perrott fcf6426732 ci: remove circleci as we no longer rely on it
Remove CircleCI artifacts and configuration
2024-11-12 13:09:11 -08:00

192 lines
5.9 KiB
TypeScript

/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.dev/license
*/
import { execSync, spawnSync } from 'node:child_process';
import * as fs from 'node:fs';
import * as os from 'node:os';
import * as path from 'node:path';
import { fileURLToPath } from 'node:url';
import build from './build.mjs';
import jsonHelp, { createTemporaryProject } from './json-help.mjs';
import { PackageInfo, packages } from './packages.mjs';
const __dirname = path.dirname(fileURLToPath(import.meta.url));
// Added to the README.md of the snapshot. This is markdown.
const readmeHeaderFn = (name: string, snapshotRepo: string) => `
# Snapshot build of ${name}
This repository is a snapshot of a commit on the original repository. The original code used to
generate this is located at http://github.com/angular/angular-cli.
We do not accept PRs or Issues opened on this repository. You should not use this over a tested and
released version of this package.
To test this snapshot in your own project, use
\`\`\`bash
npm install git+https://github.com/${snapshotRepo}.git
\`\`\`
----
`;
function _copy(from: string, to: string) {
fs.readdirSync(from).forEach((name) => {
const fromPath = path.join(from, name);
const toPath = path.join(to, name);
if (fs.statSync(fromPath).isDirectory()) {
if (!fs.existsSync(toPath)) {
fs.mkdirSync(toPath);
}
_copy(fromPath, toPath);
} else {
fs.writeFileSync(toPath, fs.readFileSync(fromPath));
}
});
}
const monorepoData = JSON.parse(fs.readFileSync('./.monorepo.json', 'utf-8'));
function _exec(command: string, args: string[], opts: { cwd?: string }) {
const { status, error, stdout } = spawnSync(command, args, {
stdio: ['ignore', 'pipe', 'inherit'],
...opts,
});
if (status != 0) {
console.error(`Command failed: ${command} ${args.map((x) => JSON.stringify(x)).join(', ')}`);
throw error;
}
return stdout.toString('utf-8');
}
let gitShaCache: string | undefined;
async function _publishSnapshot(
pkg: PackageInfo,
branch: string,
message: string,
githubToken: string,
) {
const snapshotRepo = monorepoData.packages[pkg.name]?.snapshotRepo;
if (!snapshotRepo) {
console.warn(`Skipping ${pkg.name}.`);
return;
}
console.info(`Publishing ${pkg.name} to repo ${snapshotRepo}.`);
const root = process.cwd();
console.debug('Temporary directory: ' + root);
const url = `https://${githubToken ? githubToken + '@' : ''}github.com/${snapshotRepo}.git`;
const destPath = path.join(root, path.basename(snapshotRepo));
_exec('git', ['clone', url], { cwd: root });
if (branch) {
// Try to checkout an existing branch, otherwise create it.
try {
_exec('git', ['checkout', branch], { cwd: destPath });
} catch {
_exec('git', ['checkout', '-b', branch], { cwd: destPath });
}
}
// Clear snapshot directory before publishing to remove deleted build files.
try {
_exec('git', ['rm', '-rf', './'], { cwd: destPath });
} catch {
// Ignore errors on delete. :shrug:
}
_copy(path.join(__dirname, '../dist', pkg.name), destPath);
if (githubToken) {
_exec('git', ['config', 'commit.gpgSign', 'false'], { cwd: destPath });
}
// Add the header to the existing README.md (or create a README if it doesn't exist).
const readmePath = path.join(destPath, 'README.md');
let readme = readmeHeaderFn(pkg.name, snapshotRepo);
try {
readme += fs.readFileSync(readmePath, 'utf-8');
} catch {}
fs.writeFileSync(readmePath, readme);
// Make sure that every snapshots is unique (otherwise we would need to make sure git accepts
// empty commits).
fs.writeFileSync(path.join(destPath, 'uniqueId'), '' + new Date());
// Ensure we call git from within this repo
gitShaCache ??= _exec('git', ['log', '--format=%h', '-n1'], { cwd: __dirname }).trim();
// Commit and push.
_exec('git', ['add', '.'], { cwd: destPath });
_exec('git', ['commit', '-a', '-m', message], { cwd: destPath });
_exec('git', ['tag', gitShaCache], { cwd: destPath });
_exec('git', ['push', 'origin', branch], { cwd: destPath });
_exec('git', ['push', '--tags', 'origin', branch], { cwd: destPath });
}
export interface SnapshotsOptions {
force?: boolean;
githubToken?: string;
branch?: string;
}
export default async function (opts: SnapshotsOptions) {
// Get the SHA.
if (execSync(`git status --porcelain`).toString() && !opts.force) {
console.error('You cannot run snapshots with local changes.');
process.exit(1);
}
const root = fs.mkdtempSync(path.join(os.tmpdir(), 'angular-cli-publish-'));
const message = execSync(`git log --format="%h %s" -n1`).toString().trim();
let branch = opts.branch || 'main';
// CIRCLE_BRANCH
if (typeof process.env['CIRCLE_BRANCH'] == 'string') {
branch = '' + process.env['CIRCLE_BRANCH'];
}
const githubToken = (opts.githubToken || process.env.SNAPSHOT_BUILDS_GITHUB_TOKEN || '').trim();
if (githubToken) {
console.info('Setting up global git name.');
_exec('git', ['config', '--global', 'user.email', 'dev-infra@angular.dev'], {});
_exec('git', ['config', '--global', 'user.name', 'Angular Builds'], {});
_exec('git', ['config', '--global', 'push.default', 'simple'], {});
}
// This is needed as otherwise when we run `devkit admin create` after `bazel build` the `dist`
// will be overridden with the output of the legacy build.
const temporaryProjectRoot = await createTemporaryProject();
// Run build.
console.info('Building...');
await build({ snapshot: true });
await jsonHelp({ temporaryProjectRoot });
if (!githubToken) {
console.info('No token given, skipping actual publishing...');
return 0;
}
for (const pkg of packages) {
process.chdir(root);
await _publishSnapshot(pkg, branch, message, githubToken);
}
return 0;
}