2020-09-09 08:42:34 -04:00
|
|
|
/**
|
|
|
|
* @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.io/license
|
|
|
|
*/
|
|
|
|
|
2020-09-09 08:48:11 -04:00
|
|
|
import * as semver from 'semver';
|
|
|
|
|
|
|
|
import {ReleaseConfig} from '../config/index';
|
|
|
|
|
|
|
|
import {fetchProjectNpmPackageInfo} from './npm-registry';
|
|
|
|
|
2021-05-17 13:00:50 -04:00
|
|
|
/** Type describing a NPM dist tag indicating long-term support. */
|
|
|
|
export type LtsNpmDistTag = `v${number}-lts`;
|
|
|
|
|
2020-09-09 08:48:11 -04:00
|
|
|
/** Interface describing determined LTS branches. */
|
|
|
|
export interface LtsBranches {
|
|
|
|
/** List of active LTS version branches. */
|
|
|
|
active: LtsBranch[];
|
|
|
|
/** List of inactive LTS version branches. */
|
|
|
|
inactive: LtsBranch[];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Interface describing an LTS version branch. */
|
|
|
|
export interface LtsBranch {
|
|
|
|
/** Name of the branch. */
|
|
|
|
name: string;
|
|
|
|
/** Most recent version for the given LTS branch. */
|
|
|
|
version: semver.SemVer;
|
|
|
|
/** NPM dist tag for the LTS version. */
|
2021-05-17 13:00:50 -04:00
|
|
|
npmDistTag: LtsNpmDistTag;
|
2020-09-09 08:48:11 -04:00
|
|
|
}
|
|
|
|
|
2020-09-09 08:42:34 -04:00
|
|
|
/**
|
|
|
|
* Number of months a major version in Angular is actively supported. See:
|
|
|
|
* https://angular.io/guide/releases#support-policy-and-schedule.
|
|
|
|
*/
|
2020-10-10 15:02:47 -04:00
|
|
|
const majorActiveSupportDuration = 6;
|
2020-09-09 08:42:34 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of months a major version has active long-term support. See:
|
|
|
|
* https://angular.io/guide/releases#support-policy-and-schedule.
|
|
|
|
*/
|
2020-10-10 15:02:47 -04:00
|
|
|
const majorLongTermSupportDuration = 12;
|
2020-09-09 08:42:34 -04:00
|
|
|
|
2020-09-09 08:48:11 -04:00
|
|
|
/** Regular expression that matches LTS NPM dist tags. */
|
2020-10-10 15:02:47 -04:00
|
|
|
const ltsNpmDistTagRegex = /^v(\d+)-lts$/;
|
2020-09-09 08:48:11 -04:00
|
|
|
|
|
|
|
/** Finds all long-term support release trains from the specified NPM package. */
|
|
|
|
export async function fetchLongTermSupportBranchesFromNpm(config: ReleaseConfig):
|
|
|
|
Promise<LtsBranches> {
|
|
|
|
const {'dist-tags': distTags, time} = await fetchProjectNpmPackageInfo(config);
|
|
|
|
const today = new Date();
|
|
|
|
const active: LtsBranch[] = [];
|
|
|
|
const inactive: LtsBranch[] = [];
|
|
|
|
|
|
|
|
// Iterate through the NPM package information and determine active/inactive LTS versions with
|
2020-10-10 15:02:47 -04:00
|
|
|
// their corresponding branches. We assume that an LTS tagged version in NPM belongs to the
|
2020-09-09 08:48:11 -04:00
|
|
|
// last-minor branch of a given major (i.e. we assume there are no outdated LTS NPM dist tags).
|
|
|
|
for (const npmDistTag in distTags) {
|
2021-05-17 13:00:50 -04:00
|
|
|
if (isLtsDistTag(npmDistTag)) {
|
2020-09-09 08:48:11 -04:00
|
|
|
const version = semver.parse(distTags[npmDistTag])!;
|
|
|
|
const branchName = `${version.major}.${version.minor}.x`;
|
|
|
|
const majorReleaseDate = new Date(time[`${version.major}.0.0`]);
|
|
|
|
const ltsEndDate = computeLtsEndDateOfMajor(majorReleaseDate);
|
|
|
|
const ltsBranch: LtsBranch = {name: branchName, version, npmDistTag};
|
|
|
|
// Depending on whether the LTS phase is still active, add the branch
|
2020-10-10 15:02:47 -04:00
|
|
|
// to the list of active or inactive LTS branches.
|
2020-09-09 08:48:11 -04:00
|
|
|
if (today <= ltsEndDate) {
|
|
|
|
active.push(ltsBranch);
|
|
|
|
} else {
|
|
|
|
inactive.push(ltsBranch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort LTS branches in descending order. i.e. most recent ones first.
|
|
|
|
active.sort((a, b) => semver.rcompare(a.version, b.version));
|
|
|
|
inactive.sort((a, b) => semver.rcompare(a.version, b.version));
|
|
|
|
|
|
|
|
return {active, inactive};
|
|
|
|
}
|
|
|
|
|
2021-05-17 13:00:50 -04:00
|
|
|
/** Gets whether the specified tag corresponds to a LTS dist tag. */
|
|
|
|
export function isLtsDistTag(tagName: string): tagName is LtsNpmDistTag {
|
|
|
|
return ltsNpmDistTagRegex.test(tagName);
|
|
|
|
}
|
|
|
|
|
2020-09-09 08:42:34 -04:00
|
|
|
/**
|
|
|
|
* Computes the date when long-term support ends for a major released at the
|
|
|
|
* specified date.
|
|
|
|
*/
|
|
|
|
export function computeLtsEndDateOfMajor(majorReleaseDate: Date): Date {
|
|
|
|
return new Date(
|
|
|
|
majorReleaseDate.getFullYear(),
|
2020-10-10 15:02:47 -04:00
|
|
|
majorReleaseDate.getMonth() + majorActiveSupportDuration + majorLongTermSupportDuration,
|
2020-09-09 08:42:34 -04:00
|
|
|
majorReleaseDate.getDate(), majorReleaseDate.getHours(), majorReleaseDate.getMinutes(),
|
|
|
|
majorReleaseDate.getSeconds(), majorReleaseDate.getMilliseconds());
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Gets the long-term support NPM dist tag for a given major version. */
|
2021-05-17 13:00:50 -04:00
|
|
|
export function getLtsNpmDistTagOfMajor(major: number): LtsNpmDistTag {
|
2020-09-09 08:42:34 -04:00
|
|
|
// LTS versions should be tagged in NPM in the following format: `v{major}-lts`.
|
2021-05-17 13:00:50 -04:00
|
|
|
return `v${major}-lts` as const;
|
2020-09-09 08:42:34 -04:00
|
|
|
}
|