Skip to content

Commit

Permalink
feat: add PrereleaseVersioningStrategy (#1981)
Browse files Browse the repository at this point in the history
* feat: add PrereleaseVersioningStrategy

* test: fix snapshot test

* test: add test for wrapping number

* test: add all versioning test cases

* fix: implement logic to keep version and bump patch
  • Loading branch information
chingor13 authored Jul 11, 2023
1 parent b2de23b commit b070888
Show file tree
Hide file tree
Showing 4 changed files with 498 additions and 1 deletion.
2 changes: 1 addition & 1 deletion __snapshots__/cli.js
Original file line number Diff line number Diff line change
Expand Up @@ -196,7 +196,7 @@ Options:
generated? [boolean] [default: false]
--versioning-strategy strategy used for bumping versions
[choices: "always-bump-major", "always-bump-minor", "always-bump-patch",
"default", "service-pack"] [default: "default"]
"default", "prerelease", "service-pack"] [default: "default"]
--changelog-path where can the CHANGELOG be found in the
project? [string] [default: "CHANGELOG.md"]
--changelog-type type of changelog to build
Expand Down
2 changes: 2 additions & 0 deletions src/factories/versioning-strategy-factory.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import {AlwaysBumpMajor} from '../versioning-strategies/always-bump-major';
import {ServicePackVersioningStrategy} from '../versioning-strategies/service-pack';
import {GitHub} from '../github';
import {ConfigurationError} from '../errors';
import {PrereleaseVersioningStrategy} from '../versioning-strategies/prerelease';

export type VersioningStrategyType = string;

Expand All @@ -40,6 +41,7 @@ const versioningTypes: Record<string, VersioningStrategyBuilder> = {
'always-bump-minor': options => new AlwaysBumpMinor(options),
'always-bump-major': options => new AlwaysBumpMajor(options),
'service-pack': options => new ServicePackVersioningStrategy(options),
prerelease: options => new PrereleaseVersioningStrategy(options),
};

export function buildVersioningStrategy(
Expand Down
203 changes: 203 additions & 0 deletions src/versioning-strategies/prerelease.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,203 @@
// Copyright 2023 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

import {DefaultVersioningStrategy} from './default';
import {Version} from '../version';
import {ConventionalCommit} from '..';
import {VersionUpdater, CustomVersionUpdate} from '../versioning-strategy';

const PRERELEASE_PATTERN = /^(?<type>[a-z]+)(?<number>\d+)$/;

class PrereleasePatchVersionUpdate implements VersionUpdater {
/**
* Returns the new bumped version
*
* @param {Version} version The current version
* @returns {Version} The bumped version
*/
bump(version: Version): Version {
if (version.preRelease) {
const match = version.preRelease.match(PRERELEASE_PATTERN);
if (match?.groups) {
const numberLength = match.groups.number.length;
const nextPrereleaseNumber = Number(match.groups.number) + 1;
const paddedNextPrereleaseNumber = `${nextPrereleaseNumber}`.padStart(
numberLength,
'0'
);
const nextPrerelease = `${match.groups.type}${paddedNextPrereleaseNumber}`;
return new Version(
version.major,
version.minor,
version.patch,
nextPrerelease,
version.build
);
}
}
return new Version(
version.major,
version.minor,
version.patch + 1,
version.preRelease,
version.build
);
}
}

class PrereleaseMinorVersionUpdate implements VersionUpdater {
/**
* Returns the new bumped version
*
* @param {Version} version The current version
* @returns {Version} The bumped version
*/
bump(version: Version): Version {
if (version.preRelease) {
const match = version.preRelease.match(PRERELEASE_PATTERN);
if (match?.groups) {
const numberLength = match.groups.number.length;
const prereleaseNumber = Number(match.groups.number);

let nextPrereleaseNumber = 1;
let nextMinorNumber = version.minor + 1;
let nextPatchNumber = 0;
if (version.patch === 0) {
// this is already the next minor candidate, then bump the pre-release number
nextPrereleaseNumber = prereleaseNumber + 1;
nextMinorNumber = version.minor;
nextPatchNumber = version.patch;
}

const paddedNextPrereleaseNumber = `${nextPrereleaseNumber}`.padStart(
numberLength,
'0'
);
const nextPrerelease = `${match.groups.type}${paddedNextPrereleaseNumber}`;
return new Version(
version.major,
nextMinorNumber,
nextPatchNumber,
nextPrerelease,
version.build
);
}
}
return new Version(
version.major,
version.minor + 1,
0,
version.preRelease,
version.build
);
}
}

class PrereleaseMajorVersionUpdate implements VersionUpdater {
/**
* Returns the new bumped version
*
* @param {Version} version The current version
* @returns {Version} The bumped version
*/
bump(version: Version): Version {
if (version.preRelease) {
const match = version.preRelease.match(PRERELEASE_PATTERN);
if (match?.groups) {
const numberLength = match.groups.number.length;
const prereleaseNumber = Number(match.groups.number);

let nextPrereleaseNumber = 1;
let nextMajorNumber = version.major + 1;
let nextMinorNumber = 0;
let nextPatchNumber = 0;
if (version.patch === 0 && version.minor === 0) {
// this is already the next major candidate, then bump the pre-release number
nextPrereleaseNumber = prereleaseNumber + 1;
nextMajorNumber = version.major;
nextMinorNumber = version.minor;
nextPatchNumber = version.patch;
}

const paddedNextPrereleaseNumber = `${nextPrereleaseNumber}`.padStart(
numberLength,
'0'
);
const nextPrerelease = `${match.groups.type}${paddedNextPrereleaseNumber}`;
return new Version(
nextMajorNumber,
nextMinorNumber,
nextPatchNumber,
nextPrerelease,
version.build
);
}
}
return new Version(
version.major + 1,
0,
0,
version.preRelease,
version.build
);
}
}

/**
* This versioning strategy will increment the pre-release number for patch
* bumps if there is a pre-release number (preserving any leading 0s).
* Example: 1.2.3-beta01 -> 1.2.3-beta02.
*/
export class PrereleaseVersioningStrategy extends DefaultVersioningStrategy {
determineReleaseType(
version: Version,
commits: ConventionalCommit[]
): VersionUpdater {
// iterate through list of commits and find biggest commit type
let breaking = 0;
let features = 0;
for (const commit of commits) {
const releaseAs = commit.notes.find(note => note.title === 'RELEASE AS');
if (releaseAs) {
// commits are handled newest to oldest, so take the first one (newest) found
this.logger.debug(
`found Release-As: ${releaseAs.text}, forcing version`
);
return new CustomVersionUpdate(
Version.parse(releaseAs.text).toString()
);
}
if (commit.breaking) {
breaking++;
} else if (commit.type === 'feat' || commit.type === 'feature') {
features++;
}
}

if (breaking > 0) {
if (version.major < 1 && this.bumpMinorPreMajor) {
return new PrereleaseMinorVersionUpdate();
} else {
return new PrereleaseMajorVersionUpdate();
}
} else if (features > 0) {
if (version.major < 1 && this.bumpPatchForMinorPreMajor) {
return new PrereleasePatchVersionUpdate();
} else {
return new PrereleaseMinorVersionUpdate();
}
}
return new PrereleasePatchVersionUpdate();
}
}
Loading

0 comments on commit b070888

Please sign in to comment.