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
|
|
|
|
*/
|
|
|
|
|
|
|
|
import * as semver from 'semver';
|
|
|
|
import {GithubClient, GithubRepo} from '../../utils/git/github';
|
|
|
|
|
|
|
|
/** Type describing a Github repository with corresponding API client. */
|
|
|
|
export interface GithubRepoWithApi extends GithubRepo {
|
|
|
|
/** API client that can access the repository. */
|
|
|
|
api: GithubClient;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Type describing a version-branch. */
|
|
|
|
export interface VersionBranch {
|
|
|
|
/** Name of the branch in Git. e.g. `10.0.x`. */
|
|
|
|
name: string;
|
|
|
|
/**
|
|
|
|
* Parsed SemVer version for the version-branch. Version branches technically do
|
|
|
|
* not follow the SemVer format, but we can have representative SemVer versions
|
|
|
|
* that can be used for comparisons, sorting and other checks.
|
|
|
|
*/
|
|
|
|
parsed: semver.SemVer;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Regular expression that matches version-branches. */
|
2020-10-10 15:02:47 -04:00
|
|
|
const versionBranchNameRegex = /^(\d+)\.(\d+)\.x$/;
|
2020-09-09 08:42:34 -04:00
|
|
|
|
|
|
|
/** Gets the version of a given branch by reading the `package.json` upstream. */
|
|
|
|
export async function getVersionOfBranch(
|
|
|
|
repo: GithubRepoWithApi, branchName: string): Promise<semver.SemVer> {
|
2021-06-25 18:45:48 -04:00
|
|
|
const {data} = await repo.api.repos.getContent(
|
2020-09-09 08:42:34 -04:00
|
|
|
{owner: repo.owner, repo: repo.name, path: '/package.json', ref: branchName});
|
2021-06-25 18:45:48 -04:00
|
|
|
// Workaround for: https://github.com/octokit/rest.js/issues/32.
|
|
|
|
// TODO: Remove cast once types of Octokit `getContent` are fixed.
|
|
|
|
const content = (data as {content?: string}).content;
|
|
|
|
if (!content) {
|
|
|
|
throw Error(`Unable to read "package.json" file from repository.`);
|
|
|
|
}
|
2021-05-27 15:14:36 -04:00
|
|
|
const {version} = JSON.parse(Buffer.from(content, 'base64').toString()) as
|
2021-02-04 17:42:42 -05:00
|
|
|
{version: string, [key: string]: any};
|
2020-09-09 08:42:34 -04:00
|
|
|
const parsedVersion = semver.parse(version);
|
|
|
|
if (parsedVersion === null) {
|
|
|
|
throw Error(`Invalid version detected in following branch: ${branchName}.`);
|
|
|
|
}
|
|
|
|
return parsedVersion;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Whether the given branch corresponds to a version branch. */
|
|
|
|
export function isVersionBranch(branchName: string): boolean {
|
|
|
|
return versionBranchNameRegex.test(branchName);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts a given version-branch into a SemVer version that can be used with SemVer
|
|
|
|
* utilities. e.g. to determine semantic order, extract major digit, compare.
|
|
|
|
*
|
|
|
|
* For example `10.0.x` will become `10.0.0` in SemVer. The patch digit is not
|
|
|
|
* relevant but needed for parsing. SemVer does not allow `x` as patch digit.
|
|
|
|
*/
|
|
|
|
export function getVersionForVersionBranch(branchName: string): semver.SemVer|null {
|
|
|
|
return semver.parse(branchName.replace(versionBranchNameRegex, '$1.$2.0'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the version branches for the specified major versions in descending
|
|
|
|
* order. i.e. latest version branches first.
|
|
|
|
*/
|
|
|
|
export async function getBranchesForMajorVersions(
|
|
|
|
repo: GithubRepoWithApi, majorVersions: number[]): Promise<VersionBranch[]> {
|
2021-06-25 18:48:24 -04:00
|
|
|
const branchData = await repo.api.paginate(
|
|
|
|
repo.api.repos.listBranches, {owner: repo.owner, repo: repo.name, protected: true});
|
2020-09-09 08:42:34 -04:00
|
|
|
const branches: VersionBranch[] = [];
|
|
|
|
|
|
|
|
for (const {name} of branchData) {
|
|
|
|
if (!isVersionBranch(name)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Convert the version-branch into a SemVer version that can be used with the
|
|
|
|
// SemVer utilities. e.g. to determine semantic order, compare versions.
|
|
|
|
const parsed = getVersionForVersionBranch(name);
|
|
|
|
// Collect all version-branches that match the specified major versions.
|
|
|
|
if (parsed !== null && majorVersions.includes(parsed.major)) {
|
|
|
|
branches.push({name, parsed});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort captured version-branches in descending order.
|
|
|
|
return branches.sort((a, b) => semver.rcompare(a.parsed, b.parsed));
|
|
|
|
}
|