Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Creating wonder-stuff-ci package! #628

Merged
merged 9 commits into from
Apr 12, 2023
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion packages/wonder-stuff-ci/README.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
# wonder-stuff-ci

This Wonder Stuff package contains functions for creating automation and scripts.
This Wonder Stuff package contains functions for automation and scripts.
2 changes: 1 addition & 1 deletion packages/wonder-stuff-ci/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
},
"name": "@khanacademy/wonder-stuff-ci",
"version": "0.0.1",
"description": "Functions for creating automation and scripts.",
"description": "Functions for automation and scripts.",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

praise: Thanks!

"module": "dist/es/index.js",
"main": "dist/index.js",
"types": "dist/index.d.ts",
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,9 @@
import {buffer} from "stream/consumers";
import {execProm, bufferToString} from "../exec-prom";
import {bufferToString} from "../buffer-to-string";

describe("#bufferToString", () => {
it.each(["testing", Buffer.from("testing")])(
"the buffer to string function returns the correct value",
(testCase: any) => {
(testCase: string | Buffer) => {
// Arrange
const input = testCase;

Expand Down
45 changes: 45 additions & 0 deletions packages/wonder-stuff-ci/src/__tests__/compare-versions.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
import {compareVersions} from "../compare-versions";

describe("#compareVersions", () => {
it.each(["7.8.1", "7.9.0", "7.10.0", "8.0.0"])(
"return 1 if version 1 is greater than version 2",
(testCase: string) => {
// Arrange
const version1 = testCase;
const version2 = "7.8.0";

// Act
const result = compareVersions(version1, version2);

// Assert
expect(result).toBe(1);
},
);

it.each(["7.7.1", "7.6.0", "6.10.0"])(
"return -1 if version 1 is less than version 2",
(testCase: string) => {
// Arrange
const version1 = testCase;
const version2 = "7.8.0";

// Act
const result = compareVersions(version1, version2);

// Assert
expect(result).toBe(-1);
},
);

it("return 0 if version 1 is equal to version 2", () => {
// Arrange
const version1 = "7.8.0";
const version2 = "7.8.0";

// Act
const result = compareVersions(version1, version2);

// Assert
expect(result).toBe(0);
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import {extractMobileReleaseInfoFromBranchName} from "../extract-mobile-release-info-from-branch-name";

describe("#extractMobileReleaseInfoFromBranchName", () => {
it.each([
"release/unified/7.8.0",
"release/android/7.8.0",
"release/ios/7.8.0",
])("get the version from the release branch", (testCase: string | null) => {
// Arrange
const releaseBranch = testCase;

// Act
const result = extractMobileReleaseInfoFromBranchName(releaseBranch);

// Assert
expect(result?.version).toBe("7.8.0");
});

it.each(["release/testing", "android/7.8.0", "ios/7.8.0", null])(
"return null if the branch is not a release branch",
(testCase: string | null) => {
// Arrange
const releaseBranch = testCase;

// Act
const result =
extractMobileReleaseInfoFromBranchName(releaseBranch);

// Assert
expect(result).toBe(null);
},
);

it("get the correct prefix string", () => {
// Arrange
const branchName = "release/unified/7.8.0";

// Act
const result = extractMobileReleaseInfoFromBranchName(branchName);

// Assert
expect(result?.prefix).toBe("release/unified/");
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import {getMobileReleaseTags} from "../get-mobile-release-tags";

jest.mock("../get-tags-from-git");

describe("#getMobileReleaseTags", () => {
it("get the tags in sorted order by ascending version value", async () => {
// Arrange
jest.spyOn(
require("../get-tags-from-git"),
"getTagsFromGit",
).mockReturnValue(["android-7.10.0", "unified-7.8.0", "unified-7.9.0"]);

// Act
const result = await getMobileReleaseTags();

// Assert
expect(result).toStrictEqual([
"unified-7.8.0",
"unified-7.9.0",
"android-7.10.0",
]);
});
});
Original file line number Diff line number Diff line change
@@ -1,16 +1,18 @@
import {allMobileReleaseTags} from "../mobile-release-git-utils";
import execProm from "../exec-prom";
import {getTagsFromGit} from "../get-tags-from-git";

jest.mock("../exec-prom");
import {execAsync} from "../exec-async";

describe("#allMobileReleaseTags", () => {
jest.mock("../exec-async");

describe("#getTagsFromGit", () => {
it("return an array of release tags split by new line", async () => {
// Arrange
jest.spyOn(require("../exec-prom"), "execProm").mockReturnValue({
jest.spyOn(require("../exec-async"), "execAsync").mockReturnValue({
stdout: "android-7.10.0\nunified-7.8.0\nunified-7.9.0",
});

// Act
const result = await allMobileReleaseTags();
const result = await getTagsFromGit();

// Assert
expect(result).toStrictEqual([
Expand Down
112 changes: 0 additions & 112 deletions packages/wonder-stuff-ci/src/__tests__/mobile-release-utils.test.ts

This file was deleted.

10 changes: 10 additions & 0 deletions packages/wonder-stuff-ci/src/buffer-to-string.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
/**
* Coerce a buffer or string into a string.
*
* @param {Buffer | string} input The `Buffer` or `string`
* that should be coerced to a string.
* @returns {string} The `string` representation of the given parameter.
*/
export const bufferToString = (input: Buffer | string): string => {
return typeof input === "string" ? input : input.toString("utf8");
};
20 changes: 20 additions & 0 deletions packages/wonder-stuff-ci/src/compare-versions.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
/**
* Compare two versions or tags
* @param {string} v1 A version of the form `<num>.<num>.<num>` or tag of the form `<tag>-<num>.<num>.<num>`.
* @param {string} v2 A version of the form `<num>.<num>.<num>` or tag of the form `<tag>-<num>.<num>.<num>`.
* @returns {number} 1 if v1 > v2, -1 if v1 < v2, 0 if v1 == v2
*/
export const compareVersions = (v1: string, v2: string) => {
const v1v = v1.includes("-") ? v1.split("-")[1] : v1;
const v2v = v2.includes("-") ? v2.split("-")[1] : v2;
const v1p = v1v.replace(/^v/g, "").split(".");
const v2p = v2v.replace(/^v/g, "").split(".");
for (let i = 0; i < v1p.length || i < v2p.length; i++) {
const p1 = +v1p[i] || 0;
const p2 = +v2p[i] || 0;
if (+p1 !== +p2) {
return p1 > p2 ? 1 : -1;
}
}
return 0;
};
7 changes: 7 additions & 0 deletions packages/wonder-stuff-ci/src/exec-async.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import {exec} from "child_process";
import util from "util";

/**
* A simple promisified version of child_process.exec, so we can `await` it
*/
export const execAsync = util.promisify(exec);
13 changes: 0 additions & 13 deletions packages/wonder-stuff-ci/src/exec-prom.ts

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
/**
* Extract release version and prefix from mobile release branch.
*
* Example, given the branch `release/unified/v7.8.0`:
* {
* prefix: "release/unified/",
* version: "7.8.0"
* }
*
* @param {string} The release branch of the form `[release/[unified|ios|android]]/[v]<num>.<num>.<num>[-extra]`.
* @returns {Object} The release version and prefix, if found; otherwise, `null`.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

suggestion: If you create the type as suggested below and then replace {Object} with {MobileReleaseInfo}, which gives folks a little nicer info in their IDEs.

*/
export const extractMobileReleaseInfoFromBranchName = (arg: string | null) => {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

suggestion: Currently, the return type is inferred from the code and as such, there isn't a concrete way to refer to this type. It would be helpful for calling code if there was one as it can be used by them to type their own arguments and variables.

Something like:

type MobileReleaseInfo = {
    prefix: "release/unified/" | "release/ios" | "release/android",
    version: string,
};

And then:

Suggested change
export const extractMobileReleaseInfoFromBranchName = (arg: string | null) => {
export const extractMobileReleaseInfoFromBranchName =
(arg: string | null): MobileReleaseInfo => {

if (!arg) {
return null;
}

const match = arg.match(
/^(release\/(ios|android|unified)\/)?v?(\d+\.\d+\.\d+(-\w*)*)$/i,
);
return match && match.length >= 3 && match[3]
? {prefix: match[1] || "release/unified/", version: match[3]}
: null;
};
20 changes: 20 additions & 0 deletions packages/wonder-stuff-ci/src/get-mobile-release-tags.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import {compareVersions} from "./compare-versions";
import {getTagsFromGit} from "./get-tags-from-git";

/**
* Get mobile release tags.
*
* Tags are filtered to only include those matching our tag version format
* (`<tag>-<num>.<num>.<num>`), then they are sorted by the version information from
* earliest version to most recent.
*
* @returns {Promise<Array<string>>} all release tags sorted creation time ascending
*/
export const getMobileReleaseTags = async (): Promise<Array<string>> => {
const tags = await getTagsFromGit();
return tags
.filter((tag) =>
tag.match(/^(ios|android|unified)-(\d+\.\d+\.\d+(-\w*)*)$/i),
)
.sort(compareVersions);
};
13 changes: 13 additions & 0 deletions packages/wonder-stuff-ci/src/get-tags-from-git.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import {execAsync} from "./exec-async";

/**
* Get all tags from git.
*
* @returns {Promise<Array<string>>} A promise of all git tags sorted by creation time ascending.
*/
export const getTagsFromGit = async (): Promise<Array<string>> => {
// Why not use simple-git here? Because for some reason it takes like 100x as long.
await execAsync("git fetch --tags");
const {stdout} = await execAsync("git tag");
return stdout.split("\n").filter(Boolean);
};
Loading