mirror of
https://github.com/game-ci/unity-builder.git
synced 2025-07-04 12:25:19 -04:00
Add tests for the versioning model
This commit is contained in:
parent
22c7d0e516
commit
b6f8040f4a
File diff suppressed because one or more lines are too long
@ -1,9 +1,25 @@
|
|||||||
expect.extend({
|
expect.extend({
|
||||||
|
toBeOfType(received, expectedType) {
|
||||||
|
const type = typeof received;
|
||||||
|
|
||||||
|
const pass = type === expectedType;
|
||||||
|
const message = () => `
|
||||||
|
Expected value to be of type ${this.utils.printExpected(expectedType)},
|
||||||
|
but received ${this.utils.printReceived(type)}`;
|
||||||
|
|
||||||
|
return {
|
||||||
|
message,
|
||||||
|
pass,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
toBeEitherAFunctionOrAnObject(received) {
|
toBeEitherAFunctionOrAnObject(received) {
|
||||||
const type = typeof received;
|
const type = typeof received;
|
||||||
|
|
||||||
const pass = ['object', 'function'].includes(type);
|
const pass = ['object', 'function'].includes(type);
|
||||||
const message = `Expected a function or an object, received ${type}`;
|
const message = () => `Expected a ${this.utils.printExpected('function')}
|
||||||
|
or an ${this.utils.printExpected('object')},
|
||||||
|
but received ${type}`;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
message,
|
message,
|
||||||
@ -11,5 +27,3 @@ expect.extend({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
jest.mock('./model/input');
|
|
||||||
|
@ -1,5 +0,0 @@
|
|||||||
export const mockDetermineVersion = jest.fn();
|
|
||||||
|
|
||||||
export default {
|
|
||||||
determineVersion: mockDetermineVersion,
|
|
||||||
};
|
|
9
src/model/__snapshots__/versioning.test.js.snap
Normal file
9
src/model/__snapshots__/versioning.test.js.snap
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`Versioning determineVersion throws for invalid strategy 0 1`] = `"Versioning strategy should be one of None, Semantic, Tag, Custom."`;
|
||||||
|
|
||||||
|
exports[`Versioning determineVersion throws for invalid strategy null 1`] = `"Versioning strategy should be one of None, Semantic, Tag, Custom."`;
|
||||||
|
|
||||||
|
exports[`Versioning determineVersion throws for invalid strategy somethingRandom 1`] = `"Versioning strategy should be one of None, Semantic, Tag, Custom."`;
|
||||||
|
|
||||||
|
exports[`Versioning determineVersion throws for invalid strategy undefined 1`] = `"Versioning strategy should be one of None, Semantic, Tag, Custom."`;
|
@ -1,5 +1,7 @@
|
|||||||
import Cache from './cache';
|
import Cache from './cache';
|
||||||
|
|
||||||
|
jest.mock('./input');
|
||||||
|
|
||||||
describe('Cache', () => {
|
describe('Cache', () => {
|
||||||
describe('Verification', () => {
|
describe('Verification', () => {
|
||||||
it('does not throw', () => {
|
it('does not throw', () => {
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
import { mockDetermineVersion } from './__mocks__/versioning';
|
|
||||||
import Input from './input';
|
import Input from './input';
|
||||||
|
import Versioning from './versioning';
|
||||||
|
|
||||||
jest.restoreAllMocks();
|
const determineVersion = jest
|
||||||
jest.mock('./versioning');
|
.spyOn(Versioning, 'determineVersion')
|
||||||
|
.mockImplementation(() => '1.3.37');
|
||||||
|
|
||||||
beforeEach(() => {
|
afterEach(() => {
|
||||||
mockDetermineVersion.mockClear();
|
jest.clearAllMocks();
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('Input', () => {
|
describe('Input', () => {
|
||||||
@ -18,11 +19,9 @@ describe('Input', () => {
|
|||||||
await expect(typeof (await Input.getFromUser())).toStrictEqual('object');
|
await expect(typeof (await Input.getFromUser())).toStrictEqual('object');
|
||||||
});
|
});
|
||||||
|
|
||||||
it.skip('calls version generator once', async () => {
|
it('calls version generator once', async () => {
|
||||||
await Input.getFromUser();
|
await Input.getFromUser();
|
||||||
|
expect(determineVersion).toHaveBeenCalledTimes(1);
|
||||||
// Todo - make sure the versioning mock is actually hit after restoreAllMocks is used.
|
|
||||||
expect(mockDetermineVersion).toHaveBeenCalledTimes(1);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
import Project from './project';
|
import Project from './project';
|
||||||
|
|
||||||
|
jest.mock('./input');
|
||||||
|
|
||||||
describe('Platform', () => {
|
describe('Platform', () => {
|
||||||
describe('relativePath', () => {
|
describe('relativePath', () => {
|
||||||
it('does not throw', () => {
|
it('does not throw', () => {
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
import * as core from '@actions/core';
|
import * as core from '@actions/core';
|
||||||
import { exec } from '@actions/exec';
|
|
||||||
import NotImplementedException from './error/not-implemented-exception';
|
import NotImplementedException from './error/not-implemented-exception';
|
||||||
import ValidationError from './error/validation-error';
|
import ValidationError from './error/validation-error';
|
||||||
import System from './system';
|
import System from './system';
|
||||||
@ -13,7 +12,8 @@ export default class Versioning {
|
|||||||
* Get the branch name of the (related) branch
|
* Get the branch name of the (related) branch
|
||||||
*/
|
*/
|
||||||
static get branch() {
|
static get branch() {
|
||||||
return this.headRef || this.ref.slice(11);
|
// Todo - use optional chaining (https://github.com/zeit/ncc/issues/534)
|
||||||
|
return this.headRef || (this.ref && this.ref.slice(11));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -113,22 +113,23 @@ export default class Versioning {
|
|||||||
*/
|
*/
|
||||||
static async parseSemanticVersion() {
|
static async parseSemanticVersion() {
|
||||||
const description = await this.getVersionDescription();
|
const description = await this.getVersionDescription();
|
||||||
const [match, tag, commits, hash] = this.descriptionRegex.exec(description);
|
|
||||||
|
|
||||||
if (!match) {
|
try {
|
||||||
throw new Error(`Failed to parse git describe output: "${description}"`);
|
const [match, tag, commits, hash] = this.descriptionRegex.exec(description);
|
||||||
|
|
||||||
|
return {
|
||||||
|
match,
|
||||||
|
tag,
|
||||||
|
commits,
|
||||||
|
hash,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
throw new Error(`Failed to parse git describe output: "${description}".`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
|
||||||
match,
|
|
||||||
tag,
|
|
||||||
commits,
|
|
||||||
hash,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static async fetchAll() {
|
static async fetchAll() {
|
||||||
await exec('git', ['fetch', '--all']);
|
await System.run('git', ['fetch', '--all']);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
245
src/model/versioning.test.js
Normal file
245
src/model/versioning.test.js
Normal file
@ -0,0 +1,245 @@
|
|||||||
|
import NotImplementedException from './error/not-implemented-exception';
|
||||||
|
import System from './system';
|
||||||
|
import Versioning from './versioning';
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
jest.restoreAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Versioning', () => {
|
||||||
|
describe('strategies', () => {
|
||||||
|
it('returns an object', () => {
|
||||||
|
expect(typeof Versioning.strategies).toStrictEqual('object');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('has items', () => {
|
||||||
|
expect(Object.values(Versioning.strategies).length).toBeGreaterThan(2);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('has an opt out option', () => {
|
||||||
|
expect(Versioning.strategies).toHaveProperty('None');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('has the semantic option', () => {
|
||||||
|
expect(Versioning.strategies).toHaveProperty('Semantic');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('has a strategy for tags', () => {
|
||||||
|
expect(Versioning.strategies).toHaveProperty('Tag');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('has an option that allows custom input', () => {
|
||||||
|
expect(Versioning.strategies).toHaveProperty('Custom');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('branch', () => {
|
||||||
|
it('returns headRef when set', () => {
|
||||||
|
const headReference = jest
|
||||||
|
.spyOn(Versioning, 'headRef', 'get')
|
||||||
|
.mockReturnValue('feature-branch-1');
|
||||||
|
|
||||||
|
expect(Versioning.branch).toStrictEqual('feature-branch-1');
|
||||||
|
expect(headReference).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns part of Ref when set', () => {
|
||||||
|
const reference = jest
|
||||||
|
.spyOn(Versioning, 'ref', 'get')
|
||||||
|
.mockReturnValue('refs/heads/feature-branch-2');
|
||||||
|
|
||||||
|
expect(Versioning.branch).toStrictEqual('feature-branch-2');
|
||||||
|
expect(reference).toHaveBeenCalledTimes(2);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('prefers headRef over ref when set', () => {
|
||||||
|
const headReference = jest
|
||||||
|
.spyOn(Versioning, 'headRef', 'get')
|
||||||
|
.mockReturnValue('feature-branch-1');
|
||||||
|
const reference = jest
|
||||||
|
.spyOn(Versioning, 'ref', 'get')
|
||||||
|
.mockReturnValue('refs/heads/feature-2');
|
||||||
|
|
||||||
|
expect(Versioning.branch).toStrictEqual('feature-branch-1');
|
||||||
|
expect(headReference).toHaveBeenCalledTimes(1);
|
||||||
|
expect(reference).toHaveBeenCalledTimes(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns undefined when headRef and ref are not set', () => {
|
||||||
|
const headReference = jest.spyOn(Versioning, 'headRef', 'get').mockReturnValue(undefined);
|
||||||
|
const reference = jest.spyOn(Versioning, 'ref', 'get').mockReturnValue(undefined);
|
||||||
|
|
||||||
|
expect(Versioning.branch).not.toBeDefined();
|
||||||
|
|
||||||
|
expect(headReference).toHaveBeenCalledTimes(1);
|
||||||
|
expect(reference).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('descriptionRegex', () => {
|
||||||
|
it('is a valid regex', () => {
|
||||||
|
expect(Versioning.descriptionRegex).toBeInstanceOf(RegExp);
|
||||||
|
});
|
||||||
|
|
||||||
|
test.each(['v1.1-1-g12345678', 'v0.1-2-g12345678', 'v0.0-500-gA9B6C3D0-dirty'])(
|
||||||
|
'is happy with valid %s',
|
||||||
|
description => {
|
||||||
|
expect(Versioning.descriptionRegex.test(description)).toBeTruthy();
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
test.each([null, undefined, 'v0', 'v0.1', 'v0.1.2', 'v0.1-2', 'v0.1-2-g'])(
|
||||||
|
'does not like %s',
|
||||||
|
description => {
|
||||||
|
expect(Versioning.descriptionRegex.test(description)).toBeFalsy();
|
||||||
|
// Also never expect without the v to work for any of these cases.
|
||||||
|
expect(Versioning.descriptionRegex.test(description?.substr(1))).toBeFalsy();
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('determineVersion', () => {
|
||||||
|
test.each([null, undefined, 0, 'somethingRandom'])(
|
||||||
|
'throws for invalid strategy %s',
|
||||||
|
async strategy => {
|
||||||
|
await expect(Versioning.determineVersion(strategy)).rejects.toThrowErrorMatchingSnapshot();
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
describe('opt out strategy', () => {
|
||||||
|
it("returns 'none'", async () => {
|
||||||
|
await expect(Versioning.determineVersion('None', 'v1.0')).resolves.toMatchInlineSnapshot(
|
||||||
|
`"none"`,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('custom strategy', () => {
|
||||||
|
test.each([null, undefined, 0, 'v0.1', '1', 'CamelCase', 'dashed-version'])(
|
||||||
|
'returns the inputVersion for %s',
|
||||||
|
async inputVersion => {
|
||||||
|
await expect(Versioning.determineVersion('Custom', inputVersion)).resolves.toStrictEqual(
|
||||||
|
inputVersion,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('semantic strategy', () => {
|
||||||
|
it('refers to generateSemanticVersion', async () => {
|
||||||
|
const generateSemanticVersion = jest
|
||||||
|
.spyOn(Versioning, 'generateSemanticVersion')
|
||||||
|
.mockResolvedValue('1.3.37');
|
||||||
|
|
||||||
|
await expect(Versioning.determineVersion('Semantic')).resolves.toStrictEqual('1.3.37');
|
||||||
|
expect(generateSemanticVersion).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('tag strategy', () => {
|
||||||
|
it('refers to generateTagVersion', async () => {
|
||||||
|
const generateTagVersion = jest
|
||||||
|
.spyOn(Versioning, 'generateTagVersion')
|
||||||
|
.mockResolvedValue('0.1');
|
||||||
|
|
||||||
|
await expect(Versioning.determineVersion('Tag')).resolves.toStrictEqual('0.1');
|
||||||
|
expect(generateTagVersion).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('not implemented strategy', () => {
|
||||||
|
it('throws a not implemented exception', async () => {
|
||||||
|
const strategy = 'Test';
|
||||||
|
jest.spyOn(Versioning, 'strategies', 'get').mockReturnValue({ [strategy]: strategy });
|
||||||
|
await expect(Versioning.determineVersion(strategy)).rejects.toThrowError(
|
||||||
|
NotImplementedException,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('generateTagVersion', () => {
|
||||||
|
it('removes the v', async () => {
|
||||||
|
jest.spyOn(Versioning, 'getTag').mockResolvedValue('v1.3.37');
|
||||||
|
await expect(Versioning.generateTagVersion()).resolves.toStrictEqual('1.3.37');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('parseSemanticVersion', () => {
|
||||||
|
it('returns the named parts', async () => {
|
||||||
|
jest.spyOn(Versioning, 'getVersionDescription').mockResolvedValue('v0.1-2-g12345678');
|
||||||
|
|
||||||
|
await expect(Versioning.parseSemanticVersion()).resolves.toMatchObject({
|
||||||
|
tag: '0.1',
|
||||||
|
commits: '2',
|
||||||
|
hash: '12345678',
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('throws when no match could be made', async () => {
|
||||||
|
jest.spyOn(Versioning, 'getVersionDescription').mockResolvedValue('no-match-can-be-made');
|
||||||
|
|
||||||
|
await expect(Versioning.parseSemanticVersion()).rejects.toThrowErrorMatchingInlineSnapshot(
|
||||||
|
`"Failed to parse git describe output: \\"no-match-can-be-made\\"."`,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getVersionDescription', () => {
|
||||||
|
it('returns the commands output', async () => {
|
||||||
|
const runOutput = 'someValue';
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue(runOutput);
|
||||||
|
await expect(Versioning.getVersionDescription()).resolves.toStrictEqual(runOutput);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('fetchAll', () => {
|
||||||
|
it('awaits the command', async () => {
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue(null);
|
||||||
|
await expect(Versioning.fetchAll()).resolves.not.toThrow();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('isDirty', () => {
|
||||||
|
it('returns true when there are files listed', async () => {
|
||||||
|
const runOutput = 'file.ext\nfile2.ext';
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue(runOutput);
|
||||||
|
await expect(Versioning.isDirty()).resolves.toStrictEqual(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns false when there is no output', async () => {
|
||||||
|
const runOutput = '';
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue(runOutput);
|
||||||
|
await expect(Versioning.isDirty()).resolves.toStrictEqual(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getTag', () => {
|
||||||
|
it('returns the commands output', async () => {
|
||||||
|
const runOutput = 'v1.0';
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue(runOutput);
|
||||||
|
await expect(Versioning.getTag()).resolves.toStrictEqual(runOutput);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('hasAnyVersionTags', () => {
|
||||||
|
it('returns false when the command returns 0', async () => {
|
||||||
|
const runOutput = '0';
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue(runOutput);
|
||||||
|
await expect(Versioning.hasAnyVersionTags()).resolves.toStrictEqual(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns true when the command returns >= 0', async () => {
|
||||||
|
const runOutput = '9';
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue(runOutput);
|
||||||
|
await expect(Versioning.hasAnyVersionTags()).resolves.toStrictEqual(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getTotalNumberOfCommits', () => {
|
||||||
|
it('returns a number from the command', async () => {
|
||||||
|
jest.spyOn(System, 'run').mockResolvedValue('9');
|
||||||
|
await expect(Versioning.getTotalNumberOfCommits()).resolves.toStrictEqual(9);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user