mirror of
https://github.com/electron/electron.git
synced 2026-01-08 07:04:01 -05:00
270 lines
11 KiB
TypeScript
270 lines
11 KiB
TypeScript
import { expect } from 'chai';
|
|
import * as sinon from 'sinon';
|
|
|
|
import { SpawnSyncReturns } from 'node:child_process';
|
|
import * as path from 'node:path';
|
|
|
|
import * as notes from '../script/release/notes/notes';
|
|
|
|
/* Fake a git spawnSync that only returns the specific
|
|
commits that we want to test */
|
|
|
|
class Commit {
|
|
sha1: string;
|
|
subject: string;
|
|
constructor (sha1: string, subject: string) {
|
|
this.sha1 = sha1;
|
|
this.subject = subject;
|
|
}
|
|
}
|
|
|
|
class GitFake {
|
|
branches: {
|
|
[key: string]: Commit[],
|
|
};
|
|
|
|
constructor () {
|
|
this.branches = {};
|
|
}
|
|
|
|
setBranch (name: string, commits: Array<Commit>): void {
|
|
this.branches[name] = commits;
|
|
}
|
|
|
|
// find the newest shared commit between branches a and b
|
|
mergeBase (a: string, b:string): string {
|
|
for (const commit of [...this.branches[a].reverse()]) {
|
|
if (this.branches[b].map((commit: Commit) => commit.sha1).includes(commit.sha1)) {
|
|
return commit.sha1;
|
|
}
|
|
}
|
|
console.error('test error: branches not related');
|
|
return '';
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
exec (command: string, args: string[], options?: any): SpawnSyncReturns<string> {
|
|
let stdout = '';
|
|
const stderr = '';
|
|
const status = 0;
|
|
|
|
if (args.length === 3 && args[0] === 'merge-base') {
|
|
// expected form: `git merge-base branchName1 branchName2`
|
|
const a: string = args[1]!;
|
|
const b: string = args[2]!;
|
|
stdout = this.mergeBase(a, b);
|
|
} else if (args.length === 3 && args[0] === 'log' && args[1] === '--format=%H') {
|
|
// expected form: `git log --format=%H branchName
|
|
const branch: string = args[2]!;
|
|
stdout = this.branches[branch].map((commit: Commit) => commit.sha1).join('\n');
|
|
} else if (args.length > 1 && args[0] === 'log' && args.includes('--format=%H,%s')) {
|
|
// expected form: `git log --format=%H,%s sha1..branchName
|
|
const [start, branch] = args[args.length - 1].split('..');
|
|
const lines : string[] = [];
|
|
let started = false;
|
|
for (const commit of this.branches[branch]) {
|
|
started = started || commit.sha1 === start;
|
|
if (started) {
|
|
lines.push(`${commit.sha1},${commit.subject}` /* %H,%s */);
|
|
}
|
|
}
|
|
stdout = lines.join('\n');
|
|
} else if (args.length === 6 &&
|
|
args[0] === 'branch' &&
|
|
args[1] === '--all' &&
|
|
args[2] === '--contains' &&
|
|
args[3].endsWith('-x-y')) {
|
|
// "what branch is this tag in?"
|
|
// git branch --all --contains ${ref} --sort version:refname
|
|
stdout = args[3];
|
|
} else {
|
|
console.error('unhandled git spawnSync():', args);
|
|
}
|
|
|
|
return { status, stdout, stderr, pid: 0, output: [null, stdout, stderr], signal: null };
|
|
}
|
|
}
|
|
|
|
describe('release notes', () => {
|
|
const sandbox = sinon.createSandbox();
|
|
const gitFake = new GitFake();
|
|
|
|
const oldBranch = '8-x-y';
|
|
const newBranch = '9-x-y';
|
|
|
|
// commits shared by both oldBranch and newBranch
|
|
const sharedHistory = [
|
|
new Commit('2abea22b4bffa1626a521711bacec7cd51425818', "fix: explicitly cancel redirects when mode is 'error' (#20686)"),
|
|
new Commit('467409458e716c68b35fa935d556050ca6bed1c4', 'build: add support for automated minor releases (#20620)') // merge-base
|
|
];
|
|
|
|
// these commits came after newBranch was created
|
|
const newBreaking = new Commit('2fad53e66b1a2cb6f7dad88fe9bb62d7a461fe98', 'refactor: use v8 serialization for ipc (#20214)');
|
|
const newFeat = new Commit('89eb309d0b22bd4aec058ffaf983e81e56a5c378', 'feat: allow GUID parameter to avoid systray demotion on Windows (#21891)');
|
|
const newFix = new Commit('0600420bac25439fc2067d51c6aaa4ee11770577', "fix: don't allow window to go behind menu bar on mac (#22828)");
|
|
const oldFix = new Commit('f77bd19a70ac2d708d17ddbe4dc12745ca3a8577', 'fix: prevent menu gc during popup (#20785)');
|
|
|
|
// a bug that's fixed in both branches by separate PRs
|
|
const newTropFix = new Commit('a6ff42c190cb5caf8f3e217748e49183a951491b', 'fix: workaround for hang when preventDefault-ing nativeWindowOpen (#22750)');
|
|
const oldTropFix = new Commit('8751f485c5a6c8c78990bfd55a4350700f81f8cd', 'fix: workaround for hang when preventDefault-ing nativeWindowOpen (#22749)');
|
|
|
|
// a PR that has unusual note formatting
|
|
const sublist = new Commit('61dc1c88fd34a3e8fff80c80ed79d0455970e610', 'fix: client area inset calculation when maximized for frameless windows (#25052) (#25216)');
|
|
|
|
before(() => {
|
|
// location of release-notes' octokit reply cache
|
|
const fixtureDir = path.resolve(__dirname, 'fixtures', 'release-notes');
|
|
process.env.NOTES_CACHE_PATH = path.resolve(fixtureDir, 'cache');
|
|
});
|
|
|
|
beforeEach(() => {
|
|
const wrapper = (command: unknown, args: unknown, options?: unknown) => {
|
|
if (command === 'git' && Array.isArray(args)) {
|
|
return gitFake.exec(command as string, args as string[], options);
|
|
}
|
|
// Default behavior for non-git commands
|
|
return { status: 0, stdout: '', stderr: '', pid: 0, output: [null, '', ''], signal: null };
|
|
};
|
|
sandbox.stub(require('node:child_process'), 'spawnSync').callsFake(wrapper);
|
|
|
|
gitFake.setBranch(oldBranch, [...sharedHistory, oldFix]);
|
|
});
|
|
|
|
afterEach(() => {
|
|
sandbox.restore();
|
|
});
|
|
|
|
describe('trop annotations', () => {
|
|
it('shows sibling branches', async function () {
|
|
const version = 'v9.0.0';
|
|
gitFake.setBranch(oldBranch, [...sharedHistory, oldTropFix]);
|
|
gitFake.setBranch(newBranch, [...sharedHistory, newTropFix]);
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
expect(results.fix).to.have.lengthOf(1);
|
|
console.log(results.fix);
|
|
expect(results.fix[0].trops).to.have.keys('8-x-y', '9-x-y');
|
|
});
|
|
});
|
|
|
|
// use case: A malicious contributor could edit the text of their 'Notes:'
|
|
// in the PR body after a PR's been merged and the maintainers have moved on.
|
|
// So instead always use the release-clerk PR comment
|
|
it('uses the release-clerk text', async function () {
|
|
// realText source: ${fixtureDir}/electron-electron-issue-21891-comments
|
|
const realText = 'Added GUID parameter to Tray API to avoid system tray icon demotion on Windows';
|
|
const testCommit = new Commit('89eb309d0b22bd4aec058ffaf983e81e56a5c378', 'feat: lole u got troled hard (#21891)');
|
|
const version = 'v9.0.0';
|
|
|
|
gitFake.setBranch(newBranch, [...sharedHistory, testCommit]);
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
expect(results.feat).to.have.lengthOf(1);
|
|
expect(results.feat[0].hash).to.equal(testCommit.sha1);
|
|
expect(results.feat[0].note).to.equal(realText);
|
|
});
|
|
|
|
describe('rendering', () => {
|
|
it('removes redundant bullet points', async function () {
|
|
const testCommit = sublist;
|
|
const version = 'v10.1.1';
|
|
|
|
gitFake.setBranch(newBranch, [...sharedHistory, testCommit]);
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
const rendered: any = await notes.render(results);
|
|
|
|
expect(rendered).to.not.include('* *');
|
|
});
|
|
|
|
it('indents sublists', async function () {
|
|
const testCommit = sublist;
|
|
const version = 'v10.1.1';
|
|
|
|
gitFake.setBranch(newBranch, [...sharedHistory, testCommit]);
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
const rendered: any = await notes.render(results);
|
|
|
|
expect(rendered).to.include([
|
|
'* Fixed the following issues for frameless when maximized on Windows:',
|
|
' * fix unreachable task bar when auto hidden with position top',
|
|
' * fix 1px extending to secondary monitor',
|
|
' * fix 1px overflowing into taskbar at certain resolutions',
|
|
' * fix white line on top of window under 4k resolutions. [#25216]'].join('\n'));
|
|
});
|
|
});
|
|
// test that when you feed in different semantic commit types,
|
|
// the parser returns them in the results' correct category
|
|
describe('semantic commit', () => {
|
|
const version = 'v9.0.0';
|
|
|
|
it("honors 'feat' type", async function () {
|
|
const testCommit = newFeat;
|
|
gitFake.setBranch(newBranch, [...sharedHistory, testCommit]);
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
expect(results.feat).to.have.lengthOf(1);
|
|
expect(results.feat[0].hash).to.equal(testCommit.sha1);
|
|
});
|
|
|
|
it("honors 'fix' type", async function () {
|
|
const testCommit = newFix;
|
|
gitFake.setBranch(newBranch, [...sharedHistory, testCommit]);
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
expect(results.fix).to.have.lengthOf(1);
|
|
expect(results.fix[0].hash).to.equal(testCommit.sha1);
|
|
});
|
|
|
|
it("honors 'BREAKING CHANGE' message", async function () {
|
|
const testCommit = newBreaking;
|
|
gitFake.setBranch(newBranch, [...sharedHistory, testCommit]);
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
expect(results.breaking).to.have.lengthOf(1);
|
|
expect(results.breaking[0].hash).to.equal(testCommit.sha1);
|
|
});
|
|
});
|
|
// test that when you have multiple stack updates only the
|
|
// latest will be kept
|
|
describe('superseding stack updates', () => {
|
|
const oldBranch = '27-x-y';
|
|
const newBranch = '28-x-y';
|
|
|
|
const version = 'v28.0.0';
|
|
|
|
it('with different major versions', async function () {
|
|
const mostRecentCommit = new Commit('9d0e6d09f0be0abbeae46dd3d66afd96d2daacaa', 'chore: bump chromium to 119.0.6043.0');
|
|
|
|
const sharedChromiumHistory = [
|
|
new Commit('029127a8b6f7c511fca4612748ad5b50e43aadaa', 'chore: bump chromium to 118.0.5993.0') // merge-base
|
|
];
|
|
const chromiumPatchUpdates = [
|
|
new Commit('d9ba26273ad3e7a34c905eccbd5dabda4eb7b402', 'chore: bump chromium to 118.0.5991.0'),
|
|
mostRecentCommit,
|
|
new Commit('d6c8ff2e7050f30dffd784915bcbd2a9f993cdb2', 'chore: bump chromium to 119.0.6029.0')
|
|
];
|
|
|
|
gitFake.setBranch(oldBranch, sharedChromiumHistory);
|
|
gitFake.setBranch(newBranch, [...sharedChromiumHistory, ...chromiumPatchUpdates]);
|
|
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
expect(results.other).to.have.lengthOf(1);
|
|
expect(results.other[0].hash).to.equal(mostRecentCommit.sha1);
|
|
});
|
|
it('with different build versions', async function () {
|
|
const mostRecentCommit = new Commit('8f7a48879ef8633a76279803637cdee7f7c6cd4f', 'chore: bump chromium to 119.0.6045.0');
|
|
|
|
const sharedChromiumHistory = [
|
|
new Commit('029127a8b6f7c511fca4612748ad5b50e43aadaa', 'chore: bump chromium to 118.0.5993.0') // merge-base
|
|
];
|
|
const chromiumPatchUpdates = [
|
|
mostRecentCommit,
|
|
new Commit('9d0e6d09f0be0abbeae46dd3d66afd96d2daacaa', 'chore: bump chromium to 119.0.6043.0'),
|
|
new Commit('d6c8ff2e7050f30dffd784915bcbd2a9f993cdb2', 'chore: bump chromium to 119.0.6029.0')
|
|
];
|
|
|
|
gitFake.setBranch(oldBranch, sharedChromiumHistory);
|
|
gitFake.setBranch(newBranch, [...sharedChromiumHistory, ...chromiumPatchUpdates]);
|
|
|
|
const results: any = await notes.get(oldBranch, newBranch, version);
|
|
expect(results.other).to.have.lengthOf(1);
|
|
expect(results.other[0].hash).to.equal(mostRecentCommit.sha1);
|
|
});
|
|
});
|
|
});
|