mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-28 18:13:35 +00:00
232 lines
6.7 KiB
TypeScript
232 lines
6.7 KiB
TypeScript
import { execFileSync } from "node:child_process";
|
|
import { copyFileSync, mkdirSync, mkdtempSync, realpathSync, rmSync, writeFileSync } from "node:fs";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import { describe, expect, it } from "vitest";
|
|
import {
|
|
mobileReleaseRefFor,
|
|
parseArgs,
|
|
preflightMobileReleaseRef,
|
|
recordMobileReleaseRef,
|
|
resolveMobileReleaseRef,
|
|
} from "../../scripts/mobile-release-ref.ts";
|
|
|
|
const SCRIPT_PATH = path.join(process.cwd(), "scripts", "mobile-release-ref.ts");
|
|
|
|
function run(command: string, args: string[], cwd: string): string {
|
|
return execFileSync(command, args, {
|
|
cwd,
|
|
encoding: "utf8",
|
|
stdio: ["ignore", "pipe", "pipe"],
|
|
});
|
|
}
|
|
|
|
function git(cwd: string, args: string[]): string {
|
|
return run("git", args, cwd);
|
|
}
|
|
|
|
function createFixtureRepo(): { cleanup: () => void; remote: string; root: string; sha: string } {
|
|
const root = mkdtempSync(path.join(os.tmpdir(), "openclaw-mobile-release-ref-"));
|
|
const remote = path.join(root, "remote.git");
|
|
const checkout = path.join(root, "checkout");
|
|
|
|
git(root, ["init", "--bare", remote]);
|
|
git(root, ["clone", remote, checkout]);
|
|
git(checkout, ["config", "user.email", "release@example.com"]);
|
|
git(checkout, ["config", "user.name", "Release Test"]);
|
|
writeFileSync(path.join(checkout, "README.md"), "release\n", "utf8");
|
|
git(checkout, ["add", "README.md"]);
|
|
git(checkout, ["commit", "-m", "initial"]);
|
|
const sha = git(checkout, ["rev-parse", "HEAD"]).trim();
|
|
git(checkout, ["push", "origin", "HEAD:main"]);
|
|
|
|
return {
|
|
cleanup: () => rmSync(root, { force: true, recursive: true }),
|
|
remote: "origin",
|
|
root: checkout,
|
|
sha,
|
|
};
|
|
}
|
|
|
|
describe("mobile-release-ref", () => {
|
|
it("renders platform release refs from store identities", () => {
|
|
expect(mobileReleaseRefFor({ platform: "ios", version: "2026.6.10", build: "8" })).toBe(
|
|
"refs/openclaw/mobile-releases/ios/2026.6.10-8",
|
|
);
|
|
expect(
|
|
mobileReleaseRefFor({
|
|
platform: "android",
|
|
version: "2026.6.10",
|
|
versionCode: "2026061008",
|
|
}),
|
|
).toBe("refs/openclaw/mobile-releases/android/2026.6.10-2026061008");
|
|
});
|
|
|
|
it("validates platform-specific numeric identities", () => {
|
|
expect(() =>
|
|
mobileReleaseRefFor({ platform: "ios", version: "2026.6.10", build: "0" }),
|
|
).toThrow("Invalid iOS build");
|
|
expect(() =>
|
|
mobileReleaseRefFor({
|
|
platform: "android",
|
|
version: "2026.6.10",
|
|
versionCode: "not-a-code",
|
|
}),
|
|
).toThrow("Invalid Android versionCode");
|
|
expect(() =>
|
|
mobileReleaseRefFor({
|
|
platform: "android",
|
|
version: "2026.6.10",
|
|
versionCode: "2026061101",
|
|
}),
|
|
).toThrow("Expected 2026061001 through 2026061099");
|
|
expect(() =>
|
|
mobileReleaseRefFor({ platform: "ios", version: "2026.06.10", build: "8" }),
|
|
).toThrow("Invalid mobile release version");
|
|
});
|
|
|
|
it("parses CLI commands and rejects missing platform-specific fields", () => {
|
|
expect(
|
|
parseArgs([
|
|
"record",
|
|
"--",
|
|
"--platform",
|
|
"android",
|
|
"--version",
|
|
"2026.6.10",
|
|
"--version-code",
|
|
"2026061008",
|
|
"--sha",
|
|
"HEAD",
|
|
]),
|
|
).toMatchObject({
|
|
command: "record",
|
|
platform: "android",
|
|
version: "2026.6.10",
|
|
versionCode: "2026061008",
|
|
});
|
|
|
|
expect(() =>
|
|
mobileReleaseRefFor({
|
|
platform: "android",
|
|
version: "2026.6.10",
|
|
}),
|
|
).toThrow("Invalid Android versionCode");
|
|
});
|
|
|
|
it("creates, resolves, and idempotently accepts an existing same-SHA ref", () => {
|
|
const fixture = createFixtureRepo();
|
|
try {
|
|
const options = {
|
|
build: "8",
|
|
command: "record" as const,
|
|
platform: "ios" as const,
|
|
remote: fixture.remote,
|
|
rootDir: fixture.root,
|
|
sha: "HEAD",
|
|
version: "2026.6.10",
|
|
versionCode: null,
|
|
};
|
|
|
|
expect(preflightMobileReleaseRef(options).status).toBe("available");
|
|
expect(recordMobileReleaseRef(options)).toMatchObject({
|
|
ref: "refs/openclaw/mobile-releases/ios/2026.6.10-8",
|
|
sha: fixture.sha,
|
|
status: "created",
|
|
});
|
|
expect(recordMobileReleaseRef(options).status).toBe("already-recorded");
|
|
expect(resolveMobileReleaseRef(options)).toMatchObject({
|
|
ref: "refs/openclaw/mobile-releases/ios/2026.6.10-8",
|
|
sha: fixture.sha,
|
|
});
|
|
} finally {
|
|
fixture.cleanup();
|
|
}
|
|
});
|
|
|
|
it("rejects an existing ref at a different SHA", () => {
|
|
const fixture = createFixtureRepo();
|
|
try {
|
|
const first = {
|
|
build: null,
|
|
command: "record" as const,
|
|
platform: "android" as const,
|
|
remote: fixture.remote,
|
|
rootDir: fixture.root,
|
|
sha: "HEAD",
|
|
version: "2026.6.10",
|
|
versionCode: "2026061008",
|
|
};
|
|
recordMobileReleaseRef(first);
|
|
|
|
writeFileSync(path.join(fixture.root, "README.md"), "next\n", "utf8");
|
|
git(fixture.root, ["add", "README.md"]);
|
|
git(fixture.root, ["commit", "-m", "next"]);
|
|
|
|
expect(() => recordMobileReleaseRef(first)).toThrow("already points at");
|
|
} finally {
|
|
fixture.cleanup();
|
|
}
|
|
});
|
|
|
|
it("prints the resolved SHA from the CLI", () => {
|
|
const fixture = createFixtureRepo();
|
|
try {
|
|
recordMobileReleaseRef({
|
|
build: "9",
|
|
command: "record",
|
|
platform: "ios",
|
|
remote: fixture.remote,
|
|
rootDir: fixture.root,
|
|
sha: "HEAD",
|
|
version: "2026.6.10",
|
|
versionCode: null,
|
|
});
|
|
|
|
const stdout = run(
|
|
process.execPath,
|
|
[
|
|
"--import",
|
|
"tsx",
|
|
SCRIPT_PATH,
|
|
"resolve",
|
|
"--platform",
|
|
"ios",
|
|
"--version",
|
|
"2026.6.10",
|
|
"--build",
|
|
"9",
|
|
"--root",
|
|
fixture.root,
|
|
],
|
|
process.cwd(),
|
|
);
|
|
|
|
expect(stdout).toBe(`${fixture.sha}\trefs/openclaw/mobile-releases/ios/2026.6.10-9\n`);
|
|
} finally {
|
|
fixture.cleanup();
|
|
}
|
|
});
|
|
|
|
it("runs the CLI entrypoint from a path containing spaces", () => {
|
|
const root = mkdtempSync(path.join(os.tmpdir(), "openclaw mobile release ref-"));
|
|
try {
|
|
const scriptDir = path.join(root, "script dir");
|
|
const scriptPath = path.join(scriptDir, "mobile-release-ref.ts");
|
|
mkdirSync(scriptDir, { recursive: true });
|
|
writeFileSync(path.join(root, "package.json"), '{"type":"module"}\n', "utf8");
|
|
copyFileSync(SCRIPT_PATH, scriptPath);
|
|
|
|
const stdout = run(
|
|
process.execPath,
|
|
["--import", "tsx", realpathSync(scriptPath), "--help"],
|
|
process.cwd(),
|
|
);
|
|
|
|
expect(stdout).toContain("scripts/mobile-release-ref.ts preflight");
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
});
|