@productbrain/cli 0.1.0-beta.973 → 0.1.0-beta.978
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/__tests__/notice-marker.test.d.ts +2 -0
- package/dist/__tests__/notice-marker.test.d.ts.map +1 -0
- package/dist/__tests__/notice-marker.test.js +41 -0
- package/dist/__tests__/notice-marker.test.js.map +1 -0
- package/dist/__tests__/update-check.test.js +160 -1
- package/dist/__tests__/update-check.test.js.map +1 -1
- package/dist/__tests__/upgrade-runner.test.js +12 -0
- package/dist/__tests__/upgrade-runner.test.js.map +1 -1
- package/dist/__tests__/upgrade.test.d.ts +2 -0
- package/dist/__tests__/upgrade.test.d.ts.map +1 -0
- package/dist/__tests__/upgrade.test.js +56 -0
- package/dist/__tests__/upgrade.test.js.map +1 -0
- package/dist/commands/doctor.js +2 -2
- package/dist/commands/doctor.js.map +1 -1
- package/dist/commands/doctor.test.js +19 -0
- package/dist/commands/doctor.test.js.map +1 -1
- package/dist/commands/upgrade.d.ts.map +1 -1
- package/dist/commands/upgrade.js +31 -10
- package/dist/commands/upgrade.js.map +1 -1
- package/dist/lib/notice-marker.d.ts +3 -0
- package/dist/lib/notice-marker.d.ts.map +1 -0
- package/dist/lib/notice-marker.js +53 -0
- package/dist/lib/notice-marker.js.map +1 -0
- package/dist/lib/update-check.d.ts +26 -11
- package/dist/lib/update-check.d.ts.map +1 -1
- package/dist/lib/update-check.js +123 -73
- package/dist/lib/update-check.js.map +1 -1
- package/dist/lib/upgrade-runner.d.ts +2 -1
- package/dist/lib/upgrade-runner.d.ts.map +1 -1
- package/dist/lib/upgrade-runner.js +8 -7
- package/dist/lib/upgrade-runner.js.map +1 -1
- package/package.json +1 -1
package/dist/commands/upgrade.js
CHANGED
|
@@ -22,11 +22,12 @@ function UpgradeTui({ currentVersion, latestVersion, onDone, onExit, }) {
|
|
|
22
22
|
const [status, setStatus] = useState('running');
|
|
23
23
|
const [message, setMessage] = useState('');
|
|
24
24
|
const channel = useMemo(() => detectInstallChannel(), []);
|
|
25
|
-
const command = useMemo(() => buildUpgradeCommand(channel), [channel]);
|
|
25
|
+
const command = useMemo(() => buildUpgradeCommand(channel, latestVersion ?? undefined), [channel, latestVersion]);
|
|
26
26
|
useEffect(() => {
|
|
27
27
|
let mounted = true;
|
|
28
28
|
let timer;
|
|
29
29
|
spawnUpgrade(channel, {
|
|
30
|
+
version: latestVersion ?? undefined,
|
|
30
31
|
onOutput: (line) => {
|
|
31
32
|
if (!mounted)
|
|
32
33
|
return;
|
|
@@ -57,21 +58,41 @@ function UpgradeTui({ currentVersion, latestVersion, onDone, onExit, }) {
|
|
|
57
58
|
return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Text, { bold: true, color: "cyan", children: "Upgrade pb CLI" }), _jsxs(Box, { marginTop: 1, children: [_jsx(Text, { children: currentVersion }), _jsxs(Text, { color: "gray", children: [" ", '->', " "] }), _jsx(Text, { bold: true, children: target })] }), _jsxs(Box, { marginTop: 1, children: [_jsx(Text, { color: "gray", children: "Channel: " }), _jsx(Text, { children: command.channel }), _jsx(Text, { color: "gray", children: " Command: " }), _jsx(Text, { children: command.display })] }), _jsx(ProgressLines, { lines: lines }), _jsxs(Box, { marginTop: 1, children: [status === 'running' && _jsx(Text, { color: "yellow", children: "Installing..." }), status === 'succeeded' && _jsx(Text, { color: "green", children: message }), status === 'failed' && _jsx(Text, { color: "red", children: message })] })] }));
|
|
58
59
|
}
|
|
59
60
|
export async function runUpgrade({ currentVersion }) {
|
|
61
|
+
const latest = await resolveLatestVersion(currentVersion);
|
|
62
|
+
const hasNewer = latest !== null && isNewer(currentVersion, latest);
|
|
63
|
+
// Anti-downgrade (DEC-3): only act when we have a STRICTLY NEWER target. A null `latest`
|
|
64
|
+
// (offline, or the registry's beta tag is <= installed) must NOT fall back to a bare
|
|
65
|
+
// `@productbrain/cli@beta` install — that can downgrade a newer installed CLI. Treat
|
|
66
|
+
// "no strictly newer target" as a no-op everywhere, never an auto-install of a bare tag.
|
|
67
|
+
if (!hasNewer) {
|
|
68
|
+
if (isJsonMode()) {
|
|
69
|
+
process.stdout.write(JSON.stringify({ status: 'current', currentVersion, latest: latest ?? null }) + '\n');
|
|
70
|
+
}
|
|
71
|
+
else if (latest) {
|
|
72
|
+
process.stdout.write(`\n${bold('pb is already current')} (${currentVersion}).\n\n`);
|
|
73
|
+
}
|
|
74
|
+
else {
|
|
75
|
+
// Could not resolve a newer version (offline / registry unreachable). Surface the manual
|
|
76
|
+
// command instead of auto-installing a bare tag that could downgrade.
|
|
77
|
+
process.stdout.write('\n');
|
|
78
|
+
process.stdout.write(`${bold('pb upgrade')} ${dim('—')} could not determine a newer version (offline?).\n`);
|
|
79
|
+
process.stdout.write(`pb is at ${currentVersion}. To reinstall manually, run:\n\n`);
|
|
80
|
+
process.stdout.write(` ${UPGRADE_COMMAND}\n\n`);
|
|
81
|
+
}
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
// From here `latest` is non-null and strictly newer than the installed version.
|
|
60
85
|
if (isJsonMode()) {
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
hint: 'Run the command above to upgrade pb CLI.',
|
|
64
|
-
}) + '\n');
|
|
86
|
+
const command = buildUpgradeCommand(detectInstallChannel(), latest).display;
|
|
87
|
+
process.stdout.write(JSON.stringify({ command, latest, hint: 'Run the command above to upgrade pb CLI.' }) + '\n');
|
|
65
88
|
return;
|
|
66
89
|
}
|
|
67
|
-
const latest = await resolveLatestVersion();
|
|
68
90
|
if (!canUseInteractiveUpgrade()) {
|
|
91
|
+
const displayCmd = buildUpgradeCommand(detectInstallChannel(), latest).display;
|
|
69
92
|
process.stdout.write('\n');
|
|
70
93
|
process.stdout.write(`${bold('Upgrade pb CLI')} ${dim('—')} run the command below:\n\n`);
|
|
71
|
-
process.stdout.write(` ${
|
|
72
|
-
|
|
73
|
-
process.stdout.write(`${dim(currentVersion)} -> ${latest}\n\n`);
|
|
74
|
-
}
|
|
94
|
+
process.stdout.write(` ${displayCmd}\n\n`);
|
|
95
|
+
process.stdout.write(`${dim(currentVersion)} -> ${latest}\n\n`);
|
|
75
96
|
return;
|
|
76
97
|
}
|
|
77
98
|
const { render } = await import('ink');
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"upgrade.js","sourceRoot":"","sources":["../../src/commands/upgrade.tsx"],"names":[],"mappings":";AAAA,OAAO,KAAK,EAAE,EAAE,SAAS,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAC5D,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAChC,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAE,IAAI,EAAE,GAAG,EAAc,MAAM,iBAAiB,CAAC;AACxD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,oBAAoB,EAAE,mBAAmB,EAAE,YAAY,EAAE,MAAM,0BAA0B,CAAC;AACnG,OAAO,EAAE,OAAO,EAAE,oBAAoB,EAAE,MAAM,wBAAwB,CAAC;AAMvE,SAAS,wBAAwB;IAC/B,OAAO,OAAO,CACZ,OAAO,CAAC,MAAM,CAAC,KAAK;QACpB,OAAO,CAAC,KAAK,CAAC,KAAK;QACnB,CAAC,UAAU,EAAE;QACb,OAAO,CAAC,GAAG,CAAC,EAAE,KAAK,MAAM;QACzB,CAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAC/B,CAAC;AACJ,CAAC;AAED,SAAS,aAAa,CAAC,EAAE,KAAK,EAAuB;IACnD,MAAM,OAAO,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;IAChC,OAAO,CACL,KAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,SAAS,EAAE,CAAC,YACrC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,CACtB,KAAC,IAAI,IAAC,KAAK,EAAC,MAAM,gDAAuC,CAC1D,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC,CAC/B,KAAC,IAAI,IAA0B,KAAK,EAAC,MAAM,YAAE,IAAI,IAAtC,GAAG,KAAK,IAAI,IAAI,EAAE,CAA4B,CAC1D,CAAC,GACE,CACP,CAAC;AACJ,CAAC;AAED,SAAS,UAAU,CAAC,EAClB,cAAc,EACd,aAAa,EACb,MAAM,EACN,MAAM,GAMP;IACC,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAW,EAAE,CAAC,CAAC;IACjD,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,QAAQ,CAAqC,SAAS,CAAC,CAAC;IACpF,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,GAAG,QAAQ,CAAC,EAAE,CAAC,CAAC;IAC3C,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,oBAAoB,EAAE,EAAE,EAAE,CAAC,CAAC;IAC1D,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,mBAAmB,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;
|
|
1
|
+
{"version":3,"file":"upgrade.js","sourceRoot":"","sources":["../../src/commands/upgrade.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notice-marker.d.ts","sourceRoot":"","sources":["../../src/lib/notice-marker.ts"],"names":[],"mappings":"AAwCA,0EAA0E;AAC1E,wBAAgB,WAAW,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,GAAG,OAAO,CAU/D"}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Once-per-session (agent) / once-per-day (human) notice claim via O_EXCL.
|
|
3
|
+
* First process to create the marker wins (returns true => may print). All others,
|
|
4
|
+
* and any error, fail OPEN to no-print (return false). Markers self-expire by key
|
|
5
|
+
* (session id rotates per session; daily key rolls per UTC day). DEC-2.
|
|
6
|
+
*/
|
|
7
|
+
import { openSync, closeSync, mkdirSync, statSync, readdirSync, unlinkSync } from 'node:fs';
|
|
8
|
+
import { homedir } from 'node:os';
|
|
9
|
+
import { join } from 'node:path';
|
|
10
|
+
import { readSession } from './session.js';
|
|
11
|
+
const CONFIG_DIR = join(homedir(), '.config', 'productbrain');
|
|
12
|
+
const DAILY_STALE_MS = 7 * 24 * 60 * 60 * 1000;
|
|
13
|
+
function utcDateStamp() {
|
|
14
|
+
const d = new Date();
|
|
15
|
+
const p = (n) => String(n).padStart(2, '0');
|
|
16
|
+
return `${d.getUTCFullYear()}${p(d.getUTCMonth() + 1)}${p(d.getUTCDate())}`;
|
|
17
|
+
}
|
|
18
|
+
/** Channel keys are independent so the human throttle never dedups the agent one. */
|
|
19
|
+
function markerKey(channel) {
|
|
20
|
+
if (channel === 'human')
|
|
21
|
+
return `notice-human-${utcDateStamp()}`;
|
|
22
|
+
const session = readSession();
|
|
23
|
+
// If no active session, fall back to a daily key so the per-day throttle still holds.
|
|
24
|
+
return session?.sessionId ? `notice-session-${session.sessionId}` : `notice-agent-daily-${utcDateStamp()}`;
|
|
25
|
+
}
|
|
26
|
+
function cleanupOldNoticeMarkers() {
|
|
27
|
+
try {
|
|
28
|
+
for (const name of readdirSync(CONFIG_DIR)) {
|
|
29
|
+
if (!name.startsWith('notice-human-') &&
|
|
30
|
+
!name.startsWith('notice-agent-daily-') &&
|
|
31
|
+
!name.startsWith('notice-session-'))
|
|
32
|
+
continue;
|
|
33
|
+
const p = join(CONFIG_DIR, name);
|
|
34
|
+
if (Date.now() - statSync(p).mtimeMs > DAILY_STALE_MS)
|
|
35
|
+
unlinkSync(p);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
catch { /* best-effort */ }
|
|
39
|
+
}
|
|
40
|
+
/** Returns true exactly once per (channel, session/day). Never throws. */
|
|
41
|
+
export function claimNotice(channel) {
|
|
42
|
+
const path = join(CONFIG_DIR, markerKey(channel));
|
|
43
|
+
try {
|
|
44
|
+
mkdirSync(CONFIG_DIR, { recursive: true });
|
|
45
|
+
closeSync(openSync(path, 'wx')); // 'wx' = O_EXCL: throws EEXIST if present
|
|
46
|
+
cleanupOldNoticeMarkers();
|
|
47
|
+
return true;
|
|
48
|
+
}
|
|
49
|
+
catch {
|
|
50
|
+
return false; // already claimed, or fs error => fail open to no-print
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
//# sourceMappingURL=notice-marker.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notice-marker.js","sourceRoot":"","sources":["../../src/lib/notice-marker.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AACH,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAC5F,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAE3C,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,cAAc,CAAC,CAAC;AAC9D,MAAM,cAAc,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC;AAE/C,SAAS,YAAY;IACnB,MAAM,CAAC,GAAG,IAAI,IAAI,EAAE,CAAC;IACrB,MAAM,CAAC,GAAG,CAAC,CAAS,EAAE,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IACpD,OAAO,GAAG,CAAC,CAAC,cAAc,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,EAAE,CAAC;AAC9E,CAAC;AAED,qFAAqF;AACrF,SAAS,SAAS,CAAC,OAA0B;IAC3C,IAAI,OAAO,KAAK,OAAO;QAAE,OAAO,gBAAgB,YAAY,EAAE,EAAE,CAAC;IACjE,MAAM,OAAO,GAAG,WAAW,EAAE,CAAC;IAC9B,sFAAsF;IACtF,OAAO,OAAO,EAAE,SAAS,CAAC,CAAC,CAAC,kBAAkB,OAAO,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,sBAAsB,YAAY,EAAE,EAAE,CAAC;AAC7G,CAAC;AAED,SAAS,uBAAuB;IAC9B,IAAI,CAAC;QACH,KAAK,MAAM,IAAI,IAAI,WAAW,CAAC,UAAU,CAAC,EAAE,CAAC;YAC3C,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC;gBACjC,CAAC,IAAI,CAAC,UAAU,CAAC,qBAAqB,CAAC;gBACvC,CAAC,IAAI,CAAC,UAAU,CAAC,iBAAiB,CAAC;gBAAE,SAAS;YAClD,MAAM,CAAC,GAAG,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,CAAC;YACjC,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC,OAAO,GAAG,cAAc;gBAAE,UAAU,CAAC,CAAC,CAAC,CAAC;QACvE,CAAC;IACH,CAAC;IAAC,MAAM,CAAC,CAAC,iBAAiB,CAAC,CAAC;AAC/B,CAAC;AAED,0EAA0E;AAC1E,MAAM,UAAU,WAAW,CAAC,OAA0B;IACpD,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,EAAE,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC;IAClD,IAAI,CAAC;QACH,SAAS,CAAC,UAAU,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC3C,SAAS,CAAC,QAAQ,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,0CAA0C;QAC3E,uBAAuB,EAAE,CAAC;QAC1B,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC,CAAC,wDAAwD;IACxE,CAAC;AACH,CAAC"}
|
|
@@ -5,10 +5,11 @@
|
|
|
5
5
|
* a one-line notification when a newer version is available.
|
|
6
6
|
*
|
|
7
7
|
* Rules:
|
|
8
|
-
* -
|
|
9
|
-
* -
|
|
8
|
+
* - Synchronous: reads cache only, never awaits network, never delays the command
|
|
9
|
+
* - Multi-source cache at ~/.config/productbrain/update-check.json:
|
|
10
|
+
* server header is authoritative within 48h; npm is an isNewer-gated fallback
|
|
10
11
|
* - Silent on any error (network down, registry unreachable, etc.)
|
|
11
|
-
* - JSON mode:
|
|
12
|
+
* - JSON/agent mode: notice goes to stderr (stdout stays bare JSON), throttled once per session
|
|
12
13
|
* - Does NOT use undici/fetch directly; uses Node.js built-in https
|
|
13
14
|
*/
|
|
14
15
|
/**
|
|
@@ -17,25 +18,39 @@
|
|
|
17
18
|
* can fire it from their own discovery path while sharing the dedup guard.
|
|
18
19
|
*/
|
|
19
20
|
export declare function reportVersionSkewIfNew(latest: string | null, current?: string): void;
|
|
21
|
+
/**
|
|
22
|
+
* Hot-path resolution from cache only — NEVER fetches. The **newest** in-window source wins: the
|
|
23
|
+
* server header is preferred only when it is actually the highest version, so a server that lags
|
|
24
|
+
* npm (its committed `cliVersion.ts` value trails the run-numbered npm publish — see TEN-2199) can
|
|
25
|
+
* no longer mask a newer npm latest. Returns the best available value even when every source is
|
|
26
|
+
* stale ("better than nothing"), because the hot path (checkForUpdate) cannot block on a refresh.
|
|
27
|
+
*/
|
|
28
|
+
export declare function resolveCachedLatest(): string | null;
|
|
29
|
+
/** @deprecated use resolveCachedLatest. Kept for callers migrating. */
|
|
20
30
|
export declare function readCachedLatestVersion(): string | null;
|
|
21
31
|
/** Clear the post-upgrade sentinel and stale update cache on next CLI startup. */
|
|
22
32
|
export declare function clearPendingUpgradeCache(): void;
|
|
23
33
|
/** Written after a successful self-upgrade so the next CLI process drops stale cache. */
|
|
24
|
-
export declare function writeUpgradePendingSentinel(): void;
|
|
34
|
+
export declare function writeUpgradePendingSentinel(installedVersion?: string): void;
|
|
35
|
+
/** Test-only: reset module-level per-process guards so each test starts clean. */
|
|
36
|
+
export declare function _resetProcessGuardsForTest(): void;
|
|
25
37
|
/**
|
|
26
38
|
* Compare two semver strings. Returns true if `b` is strictly newer than `a`.
|
|
27
39
|
* Handles standard x.y.z and x.y.z-beta.N formats (numeric comparison per part).
|
|
28
40
|
*/
|
|
29
41
|
export declare function isNewer(current: string, latest: string): boolean;
|
|
30
|
-
/**
|
|
42
|
+
/** Record the server-supplied latest version (authoritative source). */
|
|
31
43
|
export declare function recordServerLatest(latestVersion: string | null): void;
|
|
32
|
-
/**
|
|
33
|
-
export declare function
|
|
44
|
+
/** Record an npm-resolved latest version — gated so it never stores <= installed. */
|
|
45
|
+
export declare function recordNpmLatest(latestVersion: string | null, currentVersion: string): void;
|
|
46
|
+
/** LIVE resolve — fetches npm and records it (gated). Call ONLY from doctor/upgrade, never the hot path. */
|
|
47
|
+
export declare function resolveLatestVersion(currentVersion?: string): Promise<string | null>;
|
|
34
48
|
/**
|
|
35
|
-
*
|
|
36
|
-
*
|
|
37
|
-
*
|
|
38
|
-
*
|
|
49
|
+
* Synchronously prints at most one update line. Reads cache only — never fetches.
|
|
50
|
+
* Reaches agents (JSON mode) on stderr; stdout stays bare JSON. Throttled once per
|
|
51
|
+
* session (agent) / once per UTC day (human TTY) via O_EXCL markers. Never throws.
|
|
52
|
+
* NOTE: a session with no authenticated server call writes no cache and fires no
|
|
53
|
+
* notice; `pb doctor` live-resolves as the backstop. DEC-2.
|
|
39
54
|
*/
|
|
40
55
|
export declare function checkForUpdate(currentVersion: string): void;
|
|
41
56
|
//# sourceMappingURL=update-check.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"update-check.d.ts","sourceRoot":"","sources":["../../src/lib/update-check.ts"],"names":[],"mappings":"AAAA
|
|
1
|
+
{"version":3,"file":"update-check.d.ts","sourceRoot":"","sources":["../../src/lib/update-check.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAyCH;;;;GAIG;AACH,wBAAgB,sBAAsB,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAQpF;AA+BD;;;;;;GAMG;AACH,wBAAgB,mBAAmB,IAAI,MAAM,GAAG,IAAI,CAUnD;AAYD,uEAAuE;AACvE,wBAAgB,uBAAuB,IAAI,MAAM,GAAG,IAAI,CAEvD;AAMD,kFAAkF;AAClF,wBAAgB,wBAAwB,IAAI,IAAI,CAU/C;AAED,yFAAyF;AACzF,wBAAgB,2BAA2B,CAAC,gBAAgB,CAAC,EAAE,MAAM,GAAG,IAAI,CAK3E;AAED,kFAAkF;AAClF,wBAAgB,0BAA0B,IAAI,IAAI,CAIjD;AA2BD;;;GAGG;AACH,wBAAgB,OAAO,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAuBhE;AAED,wEAAwE;AACxE,wBAAgB,kBAAkB,CAAC,aAAa,EAAE,MAAM,GAAG,IAAI,GAAG,IAAI,CAMrE;AAED,qFAAqF;AACrF,wBAAgB,eAAe,CAAC,aAAa,EAAE,MAAM,GAAG,IAAI,EAAE,cAAc,EAAE,MAAM,GAAG,IAAI,CAM1F;AAED,4GAA4G;AAC5G,wBAAsB,oBAAoB,CAAC,cAAc,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAmB1F;AAED;;;;;;GAMG;AACH,wBAAgB,cAAc,CAAC,cAAc,EAAE,MAAM,GAAG,IAAI,CAiB3D"}
|
package/dist/lib/update-check.js
CHANGED
|
@@ -5,10 +5,11 @@
|
|
|
5
5
|
* a one-line notification when a newer version is available.
|
|
6
6
|
*
|
|
7
7
|
* Rules:
|
|
8
|
-
* -
|
|
9
|
-
* -
|
|
8
|
+
* - Synchronous: reads cache only, never awaits network, never delays the command
|
|
9
|
+
* - Multi-source cache at ~/.config/productbrain/update-check.json:
|
|
10
|
+
* server header is authoritative within 48h; npm is an isNewer-gated fallback
|
|
10
11
|
* - Silent on any error (network down, registry unreachable, etc.)
|
|
11
|
-
* - JSON mode:
|
|
12
|
+
* - JSON/agent mode: notice goes to stderr (stdout stays bare JSON), throttled once per session
|
|
12
13
|
* - Does NOT use undici/fetch directly; uses Node.js built-in https
|
|
13
14
|
*/
|
|
14
15
|
import { readFileSync, writeFileSync, mkdirSync, existsSync, unlinkSync } from 'node:fs';
|
|
@@ -16,18 +17,20 @@ import { request } from 'node:https';
|
|
|
16
17
|
import { homedir } from 'node:os';
|
|
17
18
|
import { dirname, join } from 'node:path';
|
|
18
19
|
import { fileURLToPath } from 'node:url';
|
|
19
|
-
import { UPGRADE_COMMAND } from './constants.js';
|
|
20
20
|
import { isJsonMode } from './runner.js';
|
|
21
21
|
import { dim, yellow } from './style.js';
|
|
22
|
+
import { claimNotice } from './notice-marker.js';
|
|
22
23
|
import { trackEvent } from './telemetry.js';
|
|
23
24
|
const PKG_NAME = '@productbrain/cli';
|
|
24
|
-
const
|
|
25
|
+
const SERVER_FRESHNESS_MS = 48 * 60 * 60 * 1000;
|
|
26
|
+
// npm fallback freshness — mirrors the single-source 24h cache TTL that existed before the
|
|
27
|
+
// multi-source split, so a cached npm value is revalidated daily instead of served forever.
|
|
28
|
+
const NPM_FRESHNESS_MS = 24 * 60 * 60 * 1000;
|
|
25
29
|
const CONFIG_DIR = join(homedir(), '.config', 'productbrain');
|
|
26
30
|
const CACHE_PATH = join(CONFIG_DIR, 'update-check.json');
|
|
27
31
|
const UPGRADE_PENDING_PATH = join(CONFIG_DIR, 'upgrade-pending');
|
|
28
32
|
const NPM_BETA_URL = `https://registry.npmjs.org/${PKG_NAME}/beta`;
|
|
29
33
|
const CHANNEL_RANK = { alpha: -3, beta: -2, rc: -1 };
|
|
30
|
-
let cacheWrittenThisProcess = false;
|
|
31
34
|
let pendingUpgradeClearedThisProcess = false;
|
|
32
35
|
let versionSkewReportedThisProcess = false;
|
|
33
36
|
let cachedRunningCliVersion;
|
|
@@ -64,38 +67,66 @@ export function reportVersionSkewIfNew(latest, current) {
|
|
|
64
67
|
versionSkewReportedThisProcess = true;
|
|
65
68
|
trackEvent('cli.version_skew.detected', { current: running, latest });
|
|
66
69
|
}
|
|
67
|
-
|
|
68
|
-
function readCache() {
|
|
70
|
+
function readCacheFile() {
|
|
69
71
|
try {
|
|
70
72
|
if (!existsSync(CACHE_PATH))
|
|
71
|
-
return
|
|
72
|
-
|
|
73
|
-
const entry = JSON.parse(raw);
|
|
74
|
-
if (Date.now() - entry.checkedAt < CACHE_TTL_MS) {
|
|
75
|
-
return entry.latestVersion;
|
|
76
|
-
}
|
|
73
|
+
return {};
|
|
74
|
+
return JSON.parse(readFileSync(CACHE_PATH, 'utf8'));
|
|
77
75
|
}
|
|
78
76
|
catch {
|
|
79
|
-
|
|
77
|
+
return {};
|
|
80
78
|
}
|
|
81
|
-
return null;
|
|
82
|
-
}
|
|
83
|
-
export function readCachedLatestVersion() {
|
|
84
|
-
if (isVersionCheckDisabled())
|
|
85
|
-
return null;
|
|
86
|
-
return readCache();
|
|
87
79
|
}
|
|
88
|
-
|
|
89
|
-
function writeCache(latestVersion) {
|
|
80
|
+
function writeCacheFile(cache) {
|
|
90
81
|
try {
|
|
91
82
|
mkdirSync(CONFIG_DIR, { recursive: true });
|
|
92
|
-
|
|
93
|
-
writeFileSync(CACHE_PATH, JSON.stringify(entry), 'utf8');
|
|
94
|
-
cacheWrittenThisProcess = true;
|
|
83
|
+
writeFileSync(CACHE_PATH, JSON.stringify(cache), 'utf8');
|
|
95
84
|
}
|
|
96
|
-
catch {
|
|
97
|
-
|
|
85
|
+
catch { /* non-fatal */ }
|
|
86
|
+
}
|
|
87
|
+
/** Return the newest (highest by semver) of the supplied values, ignoring null/undefined. */
|
|
88
|
+
function newestOf(...values) {
|
|
89
|
+
let best = null;
|
|
90
|
+
for (const v of values) {
|
|
91
|
+
if (!v)
|
|
92
|
+
continue;
|
|
93
|
+
if (best === null || isNewer(best, v))
|
|
94
|
+
best = v;
|
|
98
95
|
}
|
|
96
|
+
return best;
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* Hot-path resolution from cache only — NEVER fetches. The **newest** in-window source wins: the
|
|
100
|
+
* server header is preferred only when it is actually the highest version, so a server that lags
|
|
101
|
+
* npm (its committed `cliVersion.ts` value trails the run-numbered npm publish — see TEN-2199) can
|
|
102
|
+
* no longer mask a newer npm latest. Returns the best available value even when every source is
|
|
103
|
+
* stale ("better than nothing"), because the hot path (checkForUpdate) cannot block on a refresh.
|
|
104
|
+
*/
|
|
105
|
+
export function resolveCachedLatest() {
|
|
106
|
+
if (isVersionCheckDisabled())
|
|
107
|
+
return null;
|
|
108
|
+
const cache = readCacheFile();
|
|
109
|
+
const now = Date.now();
|
|
110
|
+
const serverFresh = cache.server && now - cache.server.recordedAt < SERVER_FRESHNESS_MS ? cache.server.value : null;
|
|
111
|
+
// Hot path: an npm value is usable even when stale ("better than nothing"); the live resolver
|
|
112
|
+
// (resolveLatestVersion) is the one that enforces the npm freshness window and re-fetches.
|
|
113
|
+
const best = newestOf(serverFresh, cache.npm?.value);
|
|
114
|
+
if (best !== null)
|
|
115
|
+
return best;
|
|
116
|
+
return cache.server?.value ?? null; // stale server (>48h) as last resort — better than nothing
|
|
117
|
+
}
|
|
118
|
+
/** The in-window value from each source (null when the source is absent or past its freshness window). */
|
|
119
|
+
function readFreshSources() {
|
|
120
|
+
const cache = readCacheFile();
|
|
121
|
+
const now = Date.now();
|
|
122
|
+
return {
|
|
123
|
+
server: cache.server && now - cache.server.recordedAt < SERVER_FRESHNESS_MS ? cache.server.value : null,
|
|
124
|
+
npm: cache.npm && now - cache.npm.recordedAt < NPM_FRESHNESS_MS ? cache.npm.value : null,
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
/** @deprecated use resolveCachedLatest. Kept for callers migrating. */
|
|
128
|
+
export function readCachedLatestVersion() {
|
|
129
|
+
return resolveCachedLatest();
|
|
99
130
|
}
|
|
100
131
|
function isVersionCheckDisabled() {
|
|
101
132
|
return process.env.PB_NO_VERSION_CHECK === '1';
|
|
@@ -123,14 +154,18 @@ export function clearPendingUpgradeCache() {
|
|
|
123
154
|
}
|
|
124
155
|
}
|
|
125
156
|
/** Written after a successful self-upgrade so the next CLI process drops stale cache. */
|
|
126
|
-
export function writeUpgradePendingSentinel() {
|
|
157
|
+
export function writeUpgradePendingSentinel(installedVersion) {
|
|
127
158
|
try {
|
|
128
159
|
mkdirSync(CONFIG_DIR, { recursive: true });
|
|
129
|
-
writeFileSync(UPGRADE_PENDING_PATH, JSON.stringify({ upgradedAt: Date.now() }), 'utf8');
|
|
130
|
-
}
|
|
131
|
-
catch {
|
|
132
|
-
// Non-fatal: update checks will recover via normal cache TTL.
|
|
160
|
+
writeFileSync(UPGRADE_PENDING_PATH, JSON.stringify({ upgradedAt: Date.now(), version: installedVersion ?? null /* version: concrete installed version, recorded for post-upgrade diagnostics/future "upgraded to X" notice */ }), 'utf8');
|
|
133
161
|
}
|
|
162
|
+
catch { /* non-fatal — update checks recover via cache TTL */ }
|
|
163
|
+
}
|
|
164
|
+
/** Test-only: reset module-level per-process guards so each test starts clean. */
|
|
165
|
+
export function _resetProcessGuardsForTest() {
|
|
166
|
+
pendingUpgradeClearedThisProcess = false;
|
|
167
|
+
versionSkewReportedThisProcess = false;
|
|
168
|
+
cachedRunningCliVersion = undefined;
|
|
134
169
|
}
|
|
135
170
|
/** Fetch latest version from npm registry. Returns null on any error. */
|
|
136
171
|
function fetchLatestVersion() {
|
|
@@ -188,59 +223,74 @@ export function isNewer(current, latest) {
|
|
|
188
223
|
}
|
|
189
224
|
return false;
|
|
190
225
|
}
|
|
191
|
-
/**
|
|
226
|
+
/** Record the server-supplied latest version (authoritative source). */
|
|
192
227
|
export function recordServerLatest(latestVersion) {
|
|
193
|
-
if (isVersionCheckDisabled())
|
|
228
|
+
if (isVersionCheckDisabled() || !latestVersion)
|
|
194
229
|
return;
|
|
195
|
-
|
|
230
|
+
const cache = readCacheFile();
|
|
231
|
+
cache.server = { value: latestVersion, recordedAt: Date.now() };
|
|
232
|
+
writeCacheFile(cache);
|
|
233
|
+
reportVersionSkewIfNew(latestVersion);
|
|
234
|
+
}
|
|
235
|
+
/** Record an npm-resolved latest version — gated so it never stores <= installed. */
|
|
236
|
+
export function recordNpmLatest(latestVersion, currentVersion) {
|
|
237
|
+
if (isVersionCheckDisabled() || !latestVersion)
|
|
196
238
|
return;
|
|
197
|
-
if (
|
|
239
|
+
if (!isNewer(currentVersion, latestVersion))
|
|
198
240
|
return;
|
|
199
|
-
|
|
200
|
-
|
|
241
|
+
const cache = readCacheFile();
|
|
242
|
+
cache.npm = { value: latestVersion, recordedAt: Date.now() };
|
|
243
|
+
writeCacheFile(cache);
|
|
201
244
|
}
|
|
202
|
-
/**
|
|
203
|
-
export async function resolveLatestVersion() {
|
|
245
|
+
/** LIVE resolve — fetches npm and records it (gated). Call ONLY from doctor/upgrade, never the hot path. */
|
|
246
|
+
export async function resolveLatestVersion(currentVersion) {
|
|
204
247
|
if (isVersionCheckDisabled())
|
|
205
248
|
return null;
|
|
206
|
-
const
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
249
|
+
const fresh = readFreshSources();
|
|
250
|
+
// Skip the network ONLY when npm itself is within its freshness window. A fresh server header
|
|
251
|
+
// alone cannot prove npm isn't newer (the hot path never populates npm, and the server's committed
|
|
252
|
+
// cliVersion.ts value structurally trails the run-numbered npm publish — TEN-2199), so a
|
|
253
|
+
// server-only cache MUST trigger a live npm fetch or the lagging server would mask the true latest.
|
|
254
|
+
if (fresh.npm !== null)
|
|
255
|
+
return newestOf(fresh.server, fresh.npm);
|
|
256
|
+
const fetched = await fetchLatestVersion();
|
|
257
|
+
if (fetched !== null) {
|
|
258
|
+
// The live registry is ground truth. Record only when strictly newer (the cache never stores a
|
|
259
|
+
// value <= installed), but compare against any fresh server header so a genuinely-newer server
|
|
260
|
+
// still wins. A successful fetch REPLACES a stale npm entry rather than falling back to it.
|
|
261
|
+
if (currentVersion)
|
|
262
|
+
recordNpmLatest(fetched, currentVersion);
|
|
263
|
+
return newestOf(fresh.server, fetched);
|
|
264
|
+
}
|
|
265
|
+
// Fetch failed (offline / registry unreachable): fall back to any cached value — better than
|
|
266
|
+
// nothing, and the documented `pb doctor` offline backstop.
|
|
267
|
+
return resolveCachedLatest();
|
|
213
268
|
}
|
|
214
269
|
/**
|
|
215
|
-
*
|
|
216
|
-
*
|
|
217
|
-
*
|
|
218
|
-
*
|
|
270
|
+
* Synchronously prints at most one update line. Reads cache only — never fetches.
|
|
271
|
+
* Reaches agents (JSON mode) on stderr; stdout stays bare JSON. Throttled once per
|
|
272
|
+
* session (agent) / once per UTC day (human TTY) via O_EXCL markers. Never throws.
|
|
273
|
+
* NOTE: a session with no authenticated server call writes no cache and fires no
|
|
274
|
+
* notice; `pb doctor` live-resolves as the backstop. DEC-2.
|
|
219
275
|
*/
|
|
220
276
|
export function checkForUpdate(currentVersion) {
|
|
221
277
|
clearPendingUpgradeCache();
|
|
222
278
|
if (isVersionCheckDisabled())
|
|
223
279
|
return;
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
});
|
|
240
|
-
}
|
|
241
|
-
catch {
|
|
242
|
-
// Never surface update-check errors to the user
|
|
243
|
-
}
|
|
244
|
-
})();
|
|
280
|
+
try {
|
|
281
|
+
const latest = resolveCachedLatest();
|
|
282
|
+
if (!latest || !isNewer(currentVersion, latest))
|
|
283
|
+
return;
|
|
284
|
+
reportVersionSkewIfNew(latest, currentVersion);
|
|
285
|
+
const channel = isJsonMode() ? 'agent' : 'human';
|
|
286
|
+
if (!claimNotice(channel))
|
|
287
|
+
return;
|
|
288
|
+
const line = `${yellow('pb:')} update available — you're on ${dim(currentVersion)}, latest is ${latest}. ` +
|
|
289
|
+
`Recommended: tell your operator and run ${dim('pb upgrade')}.\n`;
|
|
290
|
+
process.stderr.write('\n' + line);
|
|
291
|
+
}
|
|
292
|
+
catch {
|
|
293
|
+
// Advisory notice — never block or crash a command.
|
|
294
|
+
}
|
|
245
295
|
}
|
|
246
296
|
//# sourceMappingURL=update-check.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"update-check.js","sourceRoot":"","sources":["../../src/lib/update-check.ts"],"names":[],"mappings":"AAAA
|
|
1
|
+
{"version":3,"file":"update-check.js","sourceRoot":"","sources":["../../src/lib/update-check.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAEH,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,SAAS,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACzF,OAAO,EAAE,OAAO,EAAE,MAAM,YAAY,CAAC;AACrC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,GAAG,EAAE,MAAM,EAAE,MAAM,YAAY,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAE5C,MAAM,QAAQ,GAAG,mBAAmB,CAAC;AACrC,MAAM,mBAAmB,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC;AAChD,2FAA2F;AAC3F,4FAA4F;AAC5F,MAAM,gBAAgB,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC;AAC7C,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,cAAc,CAAC,CAAC;AAC9D,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,EAAE,mBAAmB,CAAC,CAAC;AACzD,MAAM,oBAAoB,GAAG,IAAI,CAAC,UAAU,EAAE,iBAAiB,CAAC,CAAC;AACjE,MAAM,YAAY,GAAG,8BAA8B,QAAQ,OAAO,CAAC;AACnE,MAAM,YAAY,GAA2B,EAAE,KAAK,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC;AAE7E,IAAI,gCAAgC,GAAG,KAAK,CAAC;AAC7C,IAAI,8BAA8B,GAAG,KAAK,CAAC;AAC3C,IAAI,uBAAkD,CAAC;AAEvD,8FAA8F;AAC9F,SAAS,qBAAqB;IAC5B,IAAI,uBAAuB,KAAK,SAAS;QAAE,OAAO,uBAAuB,CAAC;IAC1E,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;QACrD,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,cAAc,CAAC,CAAC;QACvD,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,OAAO,EAAE,MAAM,CAAC,CAAyB,CAAC;QAC9E,uBAAuB,GAAG,GAAG,CAAC,OAAO,IAAI,IAAI,CAAC;IAChD,CAAC;IAAC,MAAM,CAAC;QACP,uBAAuB,GAAG,IAAI,CAAC;IACjC,CAAC;IACD,OAAO,uBAAuB,CAAC;AACjC,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,sBAAsB,CAAC,MAAqB,EAAE,OAAgB;IAC5E,IAAI,8BAA8B;QAAE,OAAO;IAC3C,IAAI,CAAC,MAAM;QAAE,OAAO;IACpB,MAAM,OAAO,GAAG,OAAO,IAAI,qBAAqB,EAAE,CAAC;IACnD,IAAI,CAAC,OAAO;QAAE,OAAO;IACrB,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,MAAM,CAAC;QAAE,OAAO;IACtC,8BAA8B,GAAG,IAAI,CAAC;IACtC,UAAU,CAAC,2BAA2B,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;AACxE,CAAC;AAKD,SAAS,aAAa;IACpB,IAAI,CAAC;QACH,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;YAAE,OAAO,EAAE,CAAC;QACvC,OAAO,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,UAAU,EAAE,MAAM,CAAC,CAAc,CAAC;IACnE,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,SAAS,cAAc,CAAC,KAAgB;IACtC,IAAI,CAAC;QACH,SAAS,CAAC,UAAU,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC3C,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,MAAM,CAAC,CAAC;IAC3D,CAAC;IAAC,MAAM,CAAC,CAAC,eAAe,CAAC,CAAC;AAC7B,CAAC;AAED,6FAA6F;AAC7F,SAAS,QAAQ,CAAC,GAAG,MAAwC;IAC3D,IAAI,IAAI,GAAkB,IAAI,CAAC;IAC/B,KAAK,MAAM,CAAC,IAAI,MAAM,EAAE,CAAC;QACvB,IAAI,CAAC,CAAC;YAAE,SAAS;QACjB,IAAI,IAAI,KAAK,IAAI,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;YAAE,IAAI,GAAG,CAAC,CAAC;IAClD,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,mBAAmB;IACjC,IAAI,sBAAsB,EAAE;QAAE,OAAO,IAAI,CAAC;IAC1C,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;IAC9B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IACvB,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,IAAI,GAAG,GAAG,KAAK,CAAC,MAAM,CAAC,UAAU,GAAG,mBAAmB,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;IACpH,8FAA8F;IAC9F,2FAA2F;IAC3F,MAAM,IAAI,GAAG,QAAQ,CAAC,WAAW,EAAE,KAAK,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;IACrD,IAAI,IAAI,KAAK,IAAI;QAAE,OAAO,IAAI,CAAC;IAC/B,OAAO,KAAK,CAAC,MAAM,EAAE,KAAK,IAAI,IAAI,CAAC,CAAC,2DAA2D;AACjG,CAAC;AAED,0GAA0G;AAC1G,SAAS,gBAAgB;IACvB,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;IAC9B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IACvB,OAAO;QACL,MAAM,EAAE,KAAK,CAAC,MAAM,IAAI,GAAG,GAAG,KAAK,CAAC,MAAM,CAAC,UAAU,GAAG,mBAAmB,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI;QACvG,GAAG,EAAE,KAAK,CAAC,GAAG,IAAI,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC,UAAU,GAAG,gBAAgB,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI;KACzF,CAAC;AACJ,CAAC;AAED,uEAAuE;AACvE,MAAM,UAAU,uBAAuB;IACrC,OAAO,mBAAmB,EAAE,CAAC;AAC/B,CAAC;AAED,SAAS,sBAAsB;IAC7B,OAAO,OAAO,CAAC,GAAG,CAAC,mBAAmB,KAAK,GAAG,CAAC;AACjD,CAAC;AAED,kFAAkF;AAClF,MAAM,UAAU,wBAAwB;IACtC,IAAI,gCAAgC;QAAE,OAAO;IAC7C,gCAAgC,GAAG,IAAI,CAAC;IACxC,IAAI,CAAC;QACH,IAAI,CAAC,UAAU,CAAC,oBAAoB,CAAC;YAAE,OAAO;QAC9C,IAAI,CAAC;YAAC,UAAU,CAAC,oBAAoB,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,eAAe,CAAC,CAAC;QACnE,IAAI,CAAC;YAAC,IAAI,UAAU,CAAC,UAAU,CAAC;gBAAE,UAAU,CAAC,UAAU,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,eAAe,CAAC,CAAC;IACvF,CAAC;IAAC,MAAM,CAAC;QACP,iDAAiD;IACnD,CAAC;AACH,CAAC;AAED,yFAAyF;AACzF,MAAM,UAAU,2BAA2B,CAAC,gBAAyB;IACnE,IAAI,CAAC;QACH,SAAS,CAAC,UAAU,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC3C,aAAa,CAAC,oBAAoB,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,OAAO,EAAE,gBAAgB,IAAI,IAAI,CAAC,8GAA8G,EAAE,CAAC,EAAE,MAAM,CAAC,CAAC;IAC5O,CAAC;IAAC,MAAM,CAAC,CAAC,qDAAqD,CAAC,CAAC;AACnE,CAAC;AAED,kFAAkF;AAClF,MAAM,UAAU,0BAA0B;IACxC,gCAAgC,GAAG,KAAK,CAAC;IACzC,8BAA8B,GAAG,KAAK,CAAC;IACvC,uBAAuB,GAAG,SAAS,CAAC;AACtC,CAAC;AAED,yEAAyE;AACzE,SAAS,kBAAkB;IACzB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC7B,MAAM,GAAG,GAAG,OAAO,CAAC,YAAY,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE,QAAQ,EAAE,kBAAkB,EAAE,EAAE,EAAE,CAAC,GAAG,EAAE,EAAE;YACtG,IAAI,IAAI,GAAG,EAAE,CAAC;YACd,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAa,EAAE,EAAE,GAAG,IAAI,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YACjE,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;gBACjB,IAAI,CAAC;oBACH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAyB,CAAC;oBACxD,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC,CAAC;gBAClC,CAAC;gBAAC,MAAM,CAAC;oBACP,OAAO,CAAC,IAAI,CAAC,CAAC;gBAChB,CAAC;YACH,CAAC,CAAC,CAAC;YACH,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;QACvC,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;QACrC,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,GAAG,EAAE;YACxB,GAAG,CAAC,OAAO,EAAE,CAAC;YACd,OAAO,CAAC,IAAI,CAAC,CAAC;QAChB,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,GAAG,EAAE,CAAC;IACZ,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,OAAO,CAAC,OAAe,EAAE,MAAc;IACrD,IAAI,OAAO,KAAK,MAAM;QAAE,OAAO,KAAK,CAAC;IACrC,MAAM,YAAY,GAAG,CAAC,CAAS,EAAY,EAAE;QAC3C,MAAM,KAAK,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QAClC,MAAM,CAAC,OAAO,EAAE,GAAG,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACxC,MAAM,KAAK,GAAG,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;QAC1E,IAAI,GAAG,KAAK,SAAS;YAAE,OAAO,CAAC,GAAG,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAE/C,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;QACjD,MAAM,OAAO,GAAG,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC;QAChD,MAAM,MAAM,GAAG,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACvD,OAAO,CAAC,GAAG,KAAK,EAAE,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,EAAE,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACvF,CAAC,CAAC;IACF,MAAM,CAAC,GAAG,YAAY,CAAC,OAAO,CAAC,CAAC;IAChC,MAAM,CAAC,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC;IAC/B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC;IACzC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QAC7B,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACrB,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACrB,IAAI,EAAE,GAAG,EAAE;YAAE,OAAO,IAAI,CAAC;QACzB,IAAI,EAAE,GAAG,EAAE;YAAE,OAAO,KAAK,CAAC;IAC5B,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,wEAAwE;AACxE,MAAM,UAAU,kBAAkB,CAAC,aAA4B;IAC7D,IAAI,sBAAsB,EAAE,IAAI,CAAC,aAAa;QAAE,OAAO;IACvD,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;IAC9B,KAAK,CAAC,MAAM,GAAG,EAAE,KAAK,EAAE,aAAa,EAAE,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;IAChE,cAAc,CAAC,KAAK,CAAC,CAAC;IACtB,sBAAsB,CAAC,aAAa,CAAC,CAAC;AACxC,CAAC;AAED,qFAAqF;AACrF,MAAM,UAAU,eAAe,CAAC,aAA4B,EAAE,cAAsB;IAClF,IAAI,sBAAsB,EAAE,IAAI,CAAC,aAAa;QAAE,OAAO;IACvD,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE,aAAa,CAAC;QAAE,OAAO;IACpD,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;IAC9B,KAAK,CAAC,GAAG,GAAG,EAAE,KAAK,EAAE,aAAa,EAAE,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;IAC7D,cAAc,CAAC,KAAK,CAAC,CAAC;AACxB,CAAC;AAED,4GAA4G;AAC5G,MAAM,CAAC,KAAK,UAAU,oBAAoB,CAAC,cAAuB;IAChE,IAAI,sBAAsB,EAAE;QAAE,OAAO,IAAI,CAAC;IAC1C,MAAM,KAAK,GAAG,gBAAgB,EAAE,CAAC;IACjC,8FAA8F;IAC9F,mGAAmG;IACnG,yFAAyF;IACzF,oGAAoG;IACpG,IAAI,KAAK,CAAC,GAAG,KAAK,IAAI;QAAE,OAAO,QAAQ,CAAC,KAAK,CAAC,MAAM,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC;IACjE,MAAM,OAAO,GAAG,MAAM,kBAAkB,EAAE,CAAC;IAC3C,IAAI,OAAO,KAAK,IAAI,EAAE,CAAC;QACrB,+FAA+F;QAC/F,+FAA+F;QAC/F,4FAA4F;QAC5F,IAAI,cAAc;YAAE,eAAe,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;QAC7D,OAAO,QAAQ,CAAC,KAAK,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACzC,CAAC;IACD,6FAA6F;IAC7F,4DAA4D;IAC5D,OAAO,mBAAmB,EAAE,CAAC;AAC/B,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,cAAc,CAAC,cAAsB;IACnD,wBAAwB,EAAE,CAAC;IAC3B,IAAI,sBAAsB,EAAE;QAAE,OAAO;IACrC,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,mBAAmB,EAAE,CAAC;QACrC,IAAI,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE,MAAM,CAAC;YAAE,OAAO;QACxD,sBAAsB,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC;QAE/C,MAAM,OAAO,GAAsB,UAAU,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC;QACpE,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC;YAAE,OAAO;QAElC,MAAM,IAAI,GAAG,GAAG,MAAM,CAAC,KAAK,CAAC,iCAAiC,GAAG,CAAC,cAAc,CAAC,eAAe,MAAM,IAAI;YACxG,2CAA2C,GAAG,CAAC,YAAY,CAAC,KAAK,CAAC;QACpE,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;IACpC,CAAC;IAAC,MAAM,CAAC;QACP,oDAAoD;IACtD,CAAC;AACH,CAAC"}
|
|
@@ -14,8 +14,9 @@ export type UpgradeResult = {
|
|
|
14
14
|
export type SpawnUpgradeOptions = {
|
|
15
15
|
onOutput?: (line: string) => void;
|
|
16
16
|
env?: NodeJS.ProcessEnv;
|
|
17
|
+
version?: string;
|
|
17
18
|
};
|
|
18
19
|
export declare function detectInstallChannel(execPath?: string, env?: NodeJS.ProcessEnv): InstallChannel;
|
|
19
|
-
export declare function buildUpgradeCommand(channel: InstallChannel): UpgradeCommand;
|
|
20
|
+
export declare function buildUpgradeCommand(channel: InstallChannel, version?: string): UpgradeCommand;
|
|
20
21
|
export declare function spawnUpgrade(channel?: InstallChannel, options?: SpawnUpgradeOptions): Promise<UpgradeResult>;
|
|
21
22
|
//# sourceMappingURL=upgrade-runner.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"upgrade-runner.d.ts","sourceRoot":"","sources":["../../src/lib/upgrade-runner.ts"],"names":[],"mappings":"AAOA,MAAM,MAAM,cAAc,GAAG,KAAK,GAAG,MAAM,GAAG,MAAM,GAAG,KAAK,GAAG,MAAM,GAAG,SAAS,CAAC;AAElF,MAAM,MAAM,cAAc,GAAG;IAC3B,OAAO,EAAE,cAAc,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,aAAa,GAAG;IAC1B,OAAO,EAAE,cAAc,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,EAAE,CAAC;CAClB,CAAC;AAEF,MAAM,MAAM,mBAAmB,GAAG;IAChC,QAAQ,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IAClC,GAAG,CAAC,EAAE,MAAM,CAAC,UAAU,CAAC;
|
|
1
|
+
{"version":3,"file":"upgrade-runner.d.ts","sourceRoot":"","sources":["../../src/lib/upgrade-runner.ts"],"names":[],"mappings":"AAOA,MAAM,MAAM,cAAc,GAAG,KAAK,GAAG,MAAM,GAAG,MAAM,GAAG,KAAK,GAAG,MAAM,GAAG,SAAS,CAAC;AAElF,MAAM,MAAM,cAAc,GAAG;IAC3B,OAAO,EAAE,cAAc,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,aAAa,GAAG;IAC1B,OAAO,EAAE,cAAc,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,EAAE,CAAC;CAClB,CAAC;AAEF,MAAM,MAAM,mBAAmB,GAAG;IAChC,QAAQ,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IAClC,GAAG,CAAC,EAAE,MAAM,CAAC,UAAU,CAAC;IACxB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB,CAAC;AAYF,wBAAgB,oBAAoB,CAAC,QAAQ,SAAmB,EAAE,GAAG,GAAE,MAAM,CAAC,UAAwB,GAAG,cAAc,CActH;AAED,wBAAgB,mBAAmB,CAAC,OAAO,EAAE,cAAc,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,cAAc,CAe7F;AAaD,wBAAgB,YAAY,CAC1B,OAAO,GAAE,cAAuC,EAChD,OAAO,GAAE,mBAAwB,GAChC,OAAO,CAAC,aAAa,CAAC,CAiDxB"}
|
|
@@ -36,19 +36,20 @@ export function detectInstallChannel(execPath = process.execPath, env = process.
|
|
|
36
36
|
return 'npm';
|
|
37
37
|
return 'unknown';
|
|
38
38
|
}
|
|
39
|
-
export function buildUpgradeCommand(channel) {
|
|
39
|
+
export function buildUpgradeCommand(channel, version) {
|
|
40
|
+
const spec = version ? `@productbrain/cli@${version}` : '@productbrain/cli@beta';
|
|
40
41
|
switch (channel) {
|
|
41
42
|
case 'pnpm':
|
|
42
|
-
return { channel, command: 'pnpm', args: ['add', '-g',
|
|
43
|
+
return { channel, command: 'pnpm', args: ['add', '-g', spec], display: `pnpm add -g ${spec}` };
|
|
43
44
|
case 'yarn':
|
|
44
|
-
return { channel, command: 'yarn', args: ['global', 'add',
|
|
45
|
+
return { channel, command: 'yarn', args: ['global', 'add', spec], display: `yarn global add ${spec}` };
|
|
45
46
|
case 'bun':
|
|
46
|
-
return { channel, command: 'bun', args: ['add', '-g',
|
|
47
|
+
return { channel, command: 'bun', args: ['add', '-g', spec], display: `bun add -g ${spec}` };
|
|
47
48
|
case 'brew':
|
|
48
49
|
return { channel, command: 'brew', args: ['upgrade', 'productbrain'], display: 'brew upgrade productbrain' };
|
|
49
50
|
case 'npm':
|
|
50
51
|
case 'unknown':
|
|
51
|
-
return { channel, command: 'npm', args: ['install', '-g',
|
|
52
|
+
return { channel, command: 'npm', args: ['install', '-g', spec], display: version ? `npm install -g ${spec}` : UPGRADE_COMMAND };
|
|
52
53
|
}
|
|
53
54
|
}
|
|
54
55
|
function failureGuidance(message) {
|
|
@@ -62,7 +63,7 @@ function failureGuidance(message) {
|
|
|
62
63
|
return 'Run the displayed upgrade command manually, then retry pb doctor.';
|
|
63
64
|
}
|
|
64
65
|
export function spawnUpgrade(channel = detectInstallChannel(), options = {}) {
|
|
65
|
-
const spec = buildUpgradeCommand(channel);
|
|
66
|
+
const spec = buildUpgradeCommand(channel, options.version);
|
|
66
67
|
const output = [];
|
|
67
68
|
trackEvent('cli.upgrade.initiated', { channel: spec.channel });
|
|
68
69
|
return new Promise((resolve, reject) => {
|
|
@@ -100,7 +101,7 @@ export function spawnUpgrade(channel = detectInstallChannel(), options = {}) {
|
|
|
100
101
|
}));
|
|
101
102
|
return;
|
|
102
103
|
}
|
|
103
|
-
writeUpgradePendingSentinel();
|
|
104
|
+
writeUpgradePendingSentinel(options.version);
|
|
104
105
|
trackEvent('cli.upgrade.succeeded', { channel: spec.channel });
|
|
105
106
|
resolve({ channel: spec.channel, command: spec.display, exitCode, output });
|
|
106
107
|
});
|