mirror of
https://github.com/openclaw/openclaw.git
synced 2026-02-19 18:39:20 -05:00
test(status): add coverage for update summary + timestamps
This commit is contained in:
108
src/commands/status.update.test.ts
Normal file
108
src/commands/status.update.test.ts
Normal file
@@ -0,0 +1,108 @@
|
||||
import { describe, expect, it } from "vitest";
|
||||
import type { UpdateCheckResult } from "../infra/update-check.js";
|
||||
import { VERSION } from "../version.js";
|
||||
import { formatUpdateOneLiner, resolveUpdateAvailability } from "./status.update.js";
|
||||
|
||||
function buildUpdate(partial: Partial<UpdateCheckResult>): UpdateCheckResult {
|
||||
return {
|
||||
root: null,
|
||||
installKind: "unknown",
|
||||
packageManager: "unknown",
|
||||
...partial,
|
||||
};
|
||||
}
|
||||
|
||||
function nextMajorVersion(version: string): string {
|
||||
const [majorPart] = version.split(".");
|
||||
const major = Number.parseInt(majorPart ?? "", 10);
|
||||
if (Number.isFinite(major) && major >= 0) {
|
||||
return `${major + 1}.0.0`;
|
||||
}
|
||||
return "999999.0.0";
|
||||
}
|
||||
|
||||
describe("resolveUpdateAvailability", () => {
|
||||
it("flags git update when behind upstream", () => {
|
||||
const update = buildUpdate({
|
||||
installKind: "git",
|
||||
git: {
|
||||
root: "/tmp/repo",
|
||||
sha: null,
|
||||
tag: null,
|
||||
branch: "main",
|
||||
upstream: "origin/main",
|
||||
dirty: false,
|
||||
ahead: 0,
|
||||
behind: 3,
|
||||
fetchOk: true,
|
||||
},
|
||||
});
|
||||
expect(resolveUpdateAvailability(update)).toEqual({
|
||||
available: true,
|
||||
hasGitUpdate: true,
|
||||
hasRegistryUpdate: false,
|
||||
latestVersion: null,
|
||||
gitBehind: 3,
|
||||
});
|
||||
});
|
||||
|
||||
it("flags registry update when latest version is newer", () => {
|
||||
const latestVersion = nextMajorVersion(VERSION);
|
||||
const update = buildUpdate({
|
||||
installKind: "package",
|
||||
packageManager: "pnpm",
|
||||
registry: { latestVersion },
|
||||
});
|
||||
const availability = resolveUpdateAvailability(update);
|
||||
expect(availability.available).toBe(true);
|
||||
expect(availability.hasGitUpdate).toBe(false);
|
||||
expect(availability.hasRegistryUpdate).toBe(true);
|
||||
expect(availability.latestVersion).toBe(latestVersion);
|
||||
});
|
||||
});
|
||||
|
||||
describe("formatUpdateOneLiner", () => {
|
||||
it("renders git status and registry latest summary", () => {
|
||||
const update = buildUpdate({
|
||||
installKind: "git",
|
||||
git: {
|
||||
root: "/tmp/repo",
|
||||
sha: "abc123456789",
|
||||
tag: null,
|
||||
branch: "main",
|
||||
upstream: "origin/main",
|
||||
dirty: true,
|
||||
ahead: 0,
|
||||
behind: 2,
|
||||
fetchOk: true,
|
||||
},
|
||||
registry: { latestVersion: VERSION },
|
||||
deps: {
|
||||
manager: "pnpm",
|
||||
status: "ok",
|
||||
lockfilePath: "pnpm-lock.yaml",
|
||||
markerPath: "node_modules/.modules.yaml",
|
||||
},
|
||||
});
|
||||
|
||||
expect(formatUpdateOneLiner(update)).toBe(
|
||||
`Update: git main · ↔ origin/main · dirty · behind 2 · npm latest ${VERSION} · deps ok`,
|
||||
);
|
||||
});
|
||||
|
||||
it("renders package-manager mode with registry error", () => {
|
||||
const update = buildUpdate({
|
||||
installKind: "package",
|
||||
packageManager: "npm",
|
||||
registry: { latestVersion: null, error: "offline" },
|
||||
deps: {
|
||||
manager: "npm",
|
||||
status: "missing",
|
||||
lockfilePath: "package-lock.json",
|
||||
markerPath: "node_modules",
|
||||
},
|
||||
});
|
||||
|
||||
expect(formatUpdateOneLiner(update)).toBe("Update: npm · npm latest unknown · deps missing");
|
||||
});
|
||||
});
|
||||
58
src/logging/timestamps.test.ts
Normal file
58
src/logging/timestamps.test.ts
Normal file
@@ -0,0 +1,58 @@
|
||||
import { describe, expect, it } from "vitest";
|
||||
import { formatLocalIsoWithOffset } from "./timestamps.js";
|
||||
|
||||
function buildFakeDate(parts: {
|
||||
year: number;
|
||||
month: number;
|
||||
day: number;
|
||||
hour: number;
|
||||
minute: number;
|
||||
second: number;
|
||||
millisecond: number;
|
||||
timezoneOffsetMinutes: number;
|
||||
}): Date {
|
||||
return {
|
||||
getFullYear: () => parts.year,
|
||||
getMonth: () => parts.month - 1,
|
||||
getDate: () => parts.day,
|
||||
getHours: () => parts.hour,
|
||||
getMinutes: () => parts.minute,
|
||||
getSeconds: () => parts.second,
|
||||
getMilliseconds: () => parts.millisecond,
|
||||
getTimezoneOffset: () => parts.timezoneOffsetMinutes,
|
||||
} as unknown as Date;
|
||||
}
|
||||
|
||||
describe("formatLocalIsoWithOffset", () => {
|
||||
it("formats positive offset with millisecond padding", () => {
|
||||
const value = formatLocalIsoWithOffset(
|
||||
buildFakeDate({
|
||||
year: 2026,
|
||||
month: 1,
|
||||
day: 2,
|
||||
hour: 3,
|
||||
minute: 4,
|
||||
second: 5,
|
||||
millisecond: 6,
|
||||
timezoneOffsetMinutes: -150, // UTC+02:30
|
||||
}),
|
||||
);
|
||||
expect(value).toBe("2026-01-02T03:04:05.006+02:30");
|
||||
});
|
||||
|
||||
it("formats negative offset", () => {
|
||||
const value = formatLocalIsoWithOffset(
|
||||
buildFakeDate({
|
||||
year: 2026,
|
||||
month: 12,
|
||||
day: 31,
|
||||
hour: 23,
|
||||
minute: 59,
|
||||
second: 58,
|
||||
millisecond: 321,
|
||||
timezoneOffsetMinutes: 300, // UTC-05:00
|
||||
}),
|
||||
);
|
||||
expect(value).toBe("2026-12-31T23:59:58.321-05:00");
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user