@pugi/cli 0.1.0-beta.4 → 0.1.0-beta.40
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/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-mascot.ansi +15 -25
- package/bin/run.js +33 -1
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +15 -0
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash-classifier.js +108 -1
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/diff-capture.js +73 -0
- package/dist/core/context/index.js +7 -0
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +86 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/dispatch.js +218 -2
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +115 -5
- package/dist/core/engine/budgets.js +98 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +860 -211
- package/dist/core/engine/prompts.js +88 -2
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +992 -36
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/hooks/events.js +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/mcp/client.js +75 -6
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +24 -2
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory-sync/queue.js +158 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/history.js +11 -1
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/session.js +1899 -38
- package/dist/core/repl/slash-commands.js +406 -21
- package/dist/core/repl/store/session-store.js +31 -2
- package/dist/core/repl/workspace-context.js +22 -0
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/session.js +92 -0
- package/dist/core/settings.js +80 -0
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +113 -24
- package/dist/core/subagents/index.js +18 -5
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +3073 -321
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +242 -11
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +390 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +184 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +879 -0
- package/dist/runtime/commands/memory.js +508 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +17 -2
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +32 -0
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/version.js +65 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +213 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/file-tools.js +85 -14
- package/dist/tools/lsp-tools.js +189 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/registry.js +46 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/web-fetch.js +147 -2
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +10 -0
- package/dist/tui/ask-modal.js +2 -2
- package/dist/tui/ask-user-question-prompt.js +192 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +82 -8
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +69 -2
- package/dist/tui/markdown-render.js +4 -4
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +303 -13
- package/dist/tui/repl-splash.js +2 -2
- package/dist/tui/repl.js +72 -14
- package/dist/tui/splash.js +1 -1
- package/dist/tui/status-bar.js +94 -16
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/tool-stream-pane.js +52 -3
- package/dist/tui/update-banner.js +20 -2
- package/dist/tui/vim-input.js +267 -0
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +12 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
package/dist/tui/status-bar.js
CHANGED
|
@@ -1,5 +1,12 @@
|
|
|
1
|
-
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
1
|
+
import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
2
|
import { Box, Text } from 'ink';
|
|
3
|
+
import { formatCostUsd, formatTokens } from '../core/repl/model-pricing.js';
|
|
4
|
+
/**
|
|
5
|
+
* Window during which the per-turn delta flash stays visible on the
|
|
6
|
+
* cost-meter row. CEO spec: ~2 seconds after completion. Past that, the
|
|
7
|
+
* flash dimms out and the row shows session totals only.
|
|
8
|
+
*/
|
|
9
|
+
const TURN_DELTA_FLASH_MS = 2_000;
|
|
3
10
|
/**
|
|
4
11
|
* Cyan dot glyphs across the pulse cycle. Three steps keep the motion
|
|
5
12
|
* subtle - a true gradient would force an Ink rerender on every
|
|
@@ -17,8 +24,73 @@ export function StatusBar(props) {
|
|
|
17
24
|
// first. When the connection is healthy (`on_watch` / `connecting`),
|
|
18
25
|
// the FSM dispatch state takes over to show the dispatch lifecycle
|
|
19
26
|
// (`dispatching` / `tool: read` / `aborting` / etc.).
|
|
20
|
-
const status = composeStatusLabel(props.connection, props.dispatchState, props.dispatchToolLabel);
|
|
21
|
-
|
|
27
|
+
const status = composeStatusLabel(props.connection, props.dispatchState, props.dispatchToolLabel, props.lastCompletedOutcome);
|
|
28
|
+
// α7 cost-meter sprint — the cost row anchors above the legacy
|
|
29
|
+
// dispatch-state line so the operator's eye lands on the meter first
|
|
30
|
+
// (matches Claude Code TUI footer rhythm). The session-elapsed slot
|
|
31
|
+
// uses sessionStartedAtEpochMs (REPL boot), distinct from the
|
|
32
|
+
// per-brief `elapsedLabel` on the row below.
|
|
33
|
+
const costRow = renderCostMeterRow(props.sessionTokensIn ?? 0, props.sessionTokensOut ?? 0, props.sessionCostUsd ?? 0, props.sessionStartedAtEpochMs, now);
|
|
34
|
+
const deltaFlash = renderTurnDeltaFlash(props.lastTurnDelta, now);
|
|
35
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsxs(Box, { children: [_jsx(Text, { color: "cyan", children: `↑ ${costRow.tokensInLabel}` }), _jsx(Text, { dimColor: true, children: ' ' }), _jsx(Text, { color: "cyan", children: `↓ ${costRow.tokensOutLabel}` }), _jsx(Text, { dimColor: true, children: ` · ` }), _jsx(Text, { bold: true, children: costRow.costLabel }), _jsx(Text, { dimColor: true, children: ` · ${costRow.elapsedLabel}` }), deltaFlash ? (_jsxs(_Fragment, { children: [_jsx(Text, { dimColor: true, children: ' ' }), _jsx(Text, { color: "green", children: deltaFlash })] })) : null] }), _jsxs(Box, { children: [_jsx(Text, { color: status.color, children: `${glyph ?? '●'} ${status.label}` }), _jsx(Text, { dimColor: true, children: ` · ${props.activeAgentCount} agents · ` }), _jsx(Text, { children: `↓ ${tokenLabel} tokens` }), _jsx(Text, { dimColor: true, children: ` · ${elapsedLabel}` }), typeof props.externalDispatchCount === 'number' && props.externalDispatchCount > 0 ? (_jsx(Text, { color: "yellow", children: ` · ${props.externalDispatchCount} dispatch${props.externalDispatchCount === 1 ? '' : 'es'} active. /cancel к manage.` })) : null] }), _jsx(Box, { children: _jsx(Text, { dimColor: true, children: `${formatCount(props.pugiMdCount)} PUGI.md · ${formatCount(props.mcpServerCount)} MCP · ${formatCount(props.skillCount)} skills · ${formatQuota(props.quotaPct)} quota` }) })] }));
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* α7 cost-meter sprint — assemble the cost-meter row labels. Pure helper
|
|
39
|
+
* so the snapshot tests assert the formatted shape without standing up
|
|
40
|
+
* an Ink renderer.
|
|
41
|
+
*/
|
|
42
|
+
export function renderCostMeterRow(tokensIn, tokensOut, costUsd, sessionStartedAtEpochMs, nowEpochMs) {
|
|
43
|
+
const elapsedMs = typeof sessionStartedAtEpochMs === 'number'
|
|
44
|
+
? Math.max(0, nowEpochMs - sessionStartedAtEpochMs)
|
|
45
|
+
: 0;
|
|
46
|
+
return {
|
|
47
|
+
tokensInLabel: formatTokens(tokensIn),
|
|
48
|
+
tokensOutLabel: formatTokens(tokensOut),
|
|
49
|
+
costLabel: formatCostUsd(costUsd),
|
|
50
|
+
elapsedLabel: formatElapsedShort(elapsedMs),
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* α7 cost-meter sprint — render the per-turn delta flash when the most
|
|
55
|
+
* recent turn completed within the flash window. Returns null when no
|
|
56
|
+
* turn has completed yet OR the flash has expired (the elapsed slot
|
|
57
|
+
* gets the slot back). The flash format mirrors the spec:
|
|
58
|
+
*
|
|
59
|
+
* `+200/+1.1k +$0.01`
|
|
60
|
+
*
|
|
61
|
+
* Exported for snapshot tests.
|
|
62
|
+
*/
|
|
63
|
+
export function renderTurnDeltaFlash(delta, nowEpochMs) {
|
|
64
|
+
if (!delta)
|
|
65
|
+
return null;
|
|
66
|
+
const elapsedSinceMs = nowEpochMs - delta.completedAtEpochMs;
|
|
67
|
+
if (elapsedSinceMs < 0 || elapsedSinceMs > TURN_DELTA_FLASH_MS)
|
|
68
|
+
return null;
|
|
69
|
+
const inLabel = formatTokens(delta.tokensIn);
|
|
70
|
+
const outLabel = formatTokens(delta.tokensOut);
|
|
71
|
+
const costLabel = delta.costUsd > 0 ? ` +${formatCostUsd(delta.costUsd)}` : '';
|
|
72
|
+
return `+${inLabel}/+${outLabel}${costLabel}`;
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* α7 cost-meter sprint — local copy of the session elapsed formatter.
|
|
76
|
+
* Mirrors the helper in `core/repl/session.ts` so the status bar stays
|
|
77
|
+
* a pure leaf component without a circular import on session.ts (the
|
|
78
|
+
* model-pricing module's `formatDuration` is similar but ships an
|
|
79
|
+
* `XhYm` ceiling that does not match the CEO spec's `2m44s` shape).
|
|
80
|
+
*/
|
|
81
|
+
function formatElapsedShort(elapsedMs) {
|
|
82
|
+
if (!Number.isFinite(elapsedMs) || elapsedMs <= 0)
|
|
83
|
+
return '0s';
|
|
84
|
+
const totalSec = Math.floor(elapsedMs / 1000);
|
|
85
|
+
if (totalSec < 60)
|
|
86
|
+
return `${totalSec}s`;
|
|
87
|
+
const min = Math.floor(totalSec / 60);
|
|
88
|
+
const sec = totalSec % 60;
|
|
89
|
+
if (min < 60)
|
|
90
|
+
return `${min}m${sec.toString().padStart(2, '0')}s`;
|
|
91
|
+
const hr = Math.floor(min / 60);
|
|
92
|
+
const restMin = min % 60;
|
|
93
|
+
return `${hr}h${restMin.toString().padStart(2, '0')}m`;
|
|
22
94
|
}
|
|
23
95
|
/**
|
|
24
96
|
* Render a count badge — number if defined, `—` placeholder otherwise.
|
|
@@ -75,7 +147,7 @@ export function connectionLabel(connection) {
|
|
|
75
147
|
* `tool: <kind>` upstream so we just concatenate; null falls through
|
|
76
148
|
* to the bare `tool` placeholder.
|
|
77
149
|
*/
|
|
78
|
-
function composeStatusLabel(connection, dispatchState, toolLabel) {
|
|
150
|
+
function composeStatusLabel(connection, dispatchState, toolLabel, lastCompletedOutcome) {
|
|
79
151
|
// Transport health wins.
|
|
80
152
|
if (connection === 'offline' || connection === 'reconnecting') {
|
|
81
153
|
return connectionLabel(connection);
|
|
@@ -93,6 +165,16 @@ function composeStatusLabel(connection, dispatchState, toolLabel) {
|
|
|
93
165
|
case 'awaiting_response':
|
|
94
166
|
return { label: 'dispatching', color: 'cyan' };
|
|
95
167
|
case 'completed':
|
|
168
|
+
// Branch on the work-done outcome so the bottom-bar tells the
|
|
169
|
+
// same truth as the agent-tree (2026-05-26 — memory
|
|
170
|
+
// feedback_no_fake_dispatch_promises). `'replied'` = text-only
|
|
171
|
+
// turn, render with the same neutral gray + arrow used in the
|
|
172
|
+
// agent-tree. `'shipped'` = real side-effect (or older server
|
|
173
|
+
// that omits the outcome field). Defaults to `'shipped'` so
|
|
174
|
+
// older callers without the prop wired stay back-compat.
|
|
175
|
+
if (lastCompletedOutcome === 'replied') {
|
|
176
|
+
return { label: 'replied', color: 'gray' };
|
|
177
|
+
}
|
|
96
178
|
return { label: 'shipped', color: 'green' };
|
|
97
179
|
case 'idle':
|
|
98
180
|
case undefined:
|
|
@@ -111,18 +193,14 @@ function formatElapsed(startedAt, now) {
|
|
|
111
193
|
const seconds = Math.floor((ms % 60_000) / 1000);
|
|
112
194
|
return `${minutes}m ${seconds.toString().padStart(2, '0')}s`;
|
|
113
195
|
}
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
if (total < 1_000_000)
|
|
123
|
-
return `${(total / 1_000).toFixed(1)}k`;
|
|
124
|
-
return `${(total / 1_000_000).toFixed(1)}m`;
|
|
125
|
-
}
|
|
196
|
+
// `formatTokens` for the downstream-throughput slot is imported from
|
|
197
|
+
// `core/repl/model-pricing.ts` — single source of truth for token
|
|
198
|
+
// formatting across the cost-meter row, `/cost` slash, and the legacy
|
|
199
|
+
// downstream-tokens slot. The shape is identical to the prior local
|
|
200
|
+
// helper (`<1000` raw, `<1m` one-decimal k, `≥1m` one-decimal m); the
|
|
201
|
+
// only semantic difference is non-finite / negative inputs render as
|
|
202
|
+
// `0` instead of throwing, matching the cost-meter row's defensive
|
|
203
|
+
// posture.
|
|
126
204
|
function clampPhase(phase) {
|
|
127
205
|
if (typeof phase !== 'number' || Number.isNaN(phase))
|
|
128
206
|
return 0;
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { Box, Text } from 'ink';
|
|
3
|
+
export function StatusTable({ snapshot }) {
|
|
4
|
+
const labelWidth = Math.max('Label'.length, ...snapshot.fields.map((f) => f.label.length));
|
|
5
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Box, { marginBottom: 1, children: _jsx(Text, { bold: true, children: "Pugi status" }) }), snapshot.fields.map((field) => (_jsxs(Box, { children: [_jsxs(Text, { children: [field.label.padEnd(labelWidth, ' '), " "] }), field.available ? (_jsx(Text, { children: field.value })) : (_jsx(Text, { dimColor: true, children: field.value }))] }, field.key))), _jsx(Box, { marginTop: 1, children: _jsxs(Text, { dimColor: true, children: ["CLI ", snapshot.meta.cliVersion, " Node ", snapshot.meta.nodeVersion, " cwd ", snapshot.meta.cwd] }) })] }));
|
|
6
|
+
}
|
|
7
|
+
//# sourceMappingURL=status-table.js.map
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { Box, Text } from 'ink';
|
|
3
|
+
/**
|
|
4
|
+
* Curated ASCII pug corpus. Five variants — wide enough that repeat
|
|
5
|
+
* invocations look fresh, narrow enough that every entry stays
|
|
6
|
+
* hand-vetted (no procedural slop). Each art block intentionally fits
|
|
7
|
+
* inside an 80-column terminal so the surrounding box border does not
|
|
8
|
+
* wrap on narrow shells.
|
|
9
|
+
*
|
|
10
|
+
* The trailing newline at the end of each `art` string is intentional —
|
|
11
|
+
* keeps the renderer's join logic uniform between the boxed and the
|
|
12
|
+
* `--ascii-only` paths.
|
|
13
|
+
*/
|
|
14
|
+
export const PUG_STICKERS = Object.freeze([
|
|
15
|
+
{
|
|
16
|
+
id: 'classic-face',
|
|
17
|
+
caption: 'classic pug face',
|
|
18
|
+
art: [
|
|
19
|
+
' _._ _,-\'""`-._',
|
|
20
|
+
' (,-.`._,\'( |\\`-/|',
|
|
21
|
+
' `-.-\' \\ )-`( , o o)',
|
|
22
|
+
' `- \\`_`"\'-',
|
|
23
|
+
].join('\n'),
|
|
24
|
+
},
|
|
25
|
+
{
|
|
26
|
+
id: 'sit-pose',
|
|
27
|
+
caption: 'sit, stay, ship',
|
|
28
|
+
art: [
|
|
29
|
+
' /\\___/\\',
|
|
30
|
+
' ( o o )',
|
|
31
|
+
' ( =^= )',
|
|
32
|
+
' (______)',
|
|
33
|
+
].join('\n'),
|
|
34
|
+
},
|
|
35
|
+
{
|
|
36
|
+
id: 'peek',
|
|
37
|
+
caption: 'peek-a-pug',
|
|
38
|
+
art: [
|
|
39
|
+
' __',
|
|
40
|
+
' ___/ \\___',
|
|
41
|
+
' / o o \\',
|
|
42
|
+
' | > ^ < |',
|
|
43
|
+
' \\__________/',
|
|
44
|
+
].join('\n'),
|
|
45
|
+
},
|
|
46
|
+
{
|
|
47
|
+
id: 'sleepy',
|
|
48
|
+
caption: 'sleepy pug, no Zzz today',
|
|
49
|
+
art: [
|
|
50
|
+
' .--.',
|
|
51
|
+
' / - -\\',
|
|
52
|
+
' ( ^ ^ )',
|
|
53
|
+
' \\ ^^ /',
|
|
54
|
+
' `----\'',
|
|
55
|
+
].join('\n'),
|
|
56
|
+
},
|
|
57
|
+
{
|
|
58
|
+
id: 'shipping',
|
|
59
|
+
caption: 'shipping pug',
|
|
60
|
+
art: [
|
|
61
|
+
' .---. .---.',
|
|
62
|
+
' |o_o| |o_o|',
|
|
63
|
+
' \\_^_/ \\_^_/',
|
|
64
|
+
' /| |\\ /| |\\',
|
|
65
|
+
' shipped • shipped',
|
|
66
|
+
].join('\n'),
|
|
67
|
+
},
|
|
68
|
+
]);
|
|
69
|
+
/**
|
|
70
|
+
* Curated rotating-quote pool. Brand voice gate (brandbook §08):
|
|
71
|
+
* `brief / dispatch / stop / agents / quit / shipped` are the power
|
|
72
|
+
* words; quotes lean on those and на the operator-mode register.
|
|
73
|
+
* Adding lines: keep each ≤ 64 chars so the boxed renderer never wraps,
|
|
74
|
+
* stay в the operator's voice, no AI attribution, no hype.
|
|
75
|
+
*/
|
|
76
|
+
export const PUG_QUOTES = Object.freeze([
|
|
77
|
+
'Pugi: your engineering co-pilot.',
|
|
78
|
+
'Brief it. It ships.',
|
|
79
|
+
'Built for operators, not for benchmarks.',
|
|
80
|
+
'Pugi: твой инженерный напарник.',
|
|
81
|
+
'Dispatch agents, not promises.',
|
|
82
|
+
'Small CLI. Loud workforce.',
|
|
83
|
+
'Engineering at the speed of brief.',
|
|
84
|
+
'Pugi: shipping is the default mode.',
|
|
85
|
+
]);
|
|
86
|
+
/**
|
|
87
|
+
* Clamp a raw rng draw to a safe array index. Handles every hostile
|
|
88
|
+
* shape the spec exercises:
|
|
89
|
+
* - rng returns NaN → fall back to 0
|
|
90
|
+
* - rng returns 1.0 → clamp to length-1 (Math.floor would land at n)
|
|
91
|
+
* - rng returns -ε → clamp to 0
|
|
92
|
+
* The caller hands в the corpus length; the helper never touches the
|
|
93
|
+
* corpus itself so it stays trivially testable.
|
|
94
|
+
*/
|
|
95
|
+
function safeIndex(raw, length) {
|
|
96
|
+
if (!Number.isFinite(raw))
|
|
97
|
+
return 0;
|
|
98
|
+
const floored = Math.floor(raw);
|
|
99
|
+
if (floored < 0)
|
|
100
|
+
return 0;
|
|
101
|
+
if (floored >= length)
|
|
102
|
+
return length - 1;
|
|
103
|
+
return floored;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Pick one art variant. Defaults to `Math.random` but the caller can
|
|
107
|
+
* inject a deterministic source — the spec uses a sequence-driven
|
|
108
|
+
* stub to assert the picker hits each entry в the corpus.
|
|
109
|
+
*/
|
|
110
|
+
export function pickArtVariant(rng = Math.random) {
|
|
111
|
+
const raw = rng() * PUG_STICKERS.length;
|
|
112
|
+
return PUG_STICKERS[safeIndex(raw, PUG_STICKERS.length)];
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Pick one rotating brand quote. Same contract as `pickArtVariant` —
|
|
116
|
+
* test-injectable rng so the spec can pin the chosen index.
|
|
117
|
+
*/
|
|
118
|
+
export function pickQuote(rng = Math.random) {
|
|
119
|
+
const raw = rng() * PUG_QUOTES.length;
|
|
120
|
+
return PUG_QUOTES[safeIndex(raw, PUG_QUOTES.length)];
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Plain-text renderer for the `--ascii-only` flag and the non-TTY shell
|
|
124
|
+
* path. Emits the art verbatim, then a blank line, then the quote.
|
|
125
|
+
* No box border — scripting use-case (`pugi stickers --ascii-only`
|
|
126
|
+
* piped to `figlet`, `lolcat`, or a regression-fixture file) gets a
|
|
127
|
+
* stable contract free of decorative ANSI noise.
|
|
128
|
+
*/
|
|
129
|
+
export function renderPugStickersText(art, quote) {
|
|
130
|
+
return `${art.art}\n\n${quote}`;
|
|
131
|
+
}
|
|
132
|
+
export function PugStickersArt({ art, quote }) {
|
|
133
|
+
const lines = art.art.split('\n');
|
|
134
|
+
return (_jsxs(Box, { flexDirection: "column", borderStyle: "round", paddingX: 1, children: [_jsxs(Box, { marginBottom: 1, children: [_jsx(Text, { bold: true, children: "Pugi stickers" }), _jsxs(Text, { dimColor: true, children: [" \u2014 ", art.caption] })] }), _jsx(Box, { flexDirection: "column", children: lines.map((line, i) => (_jsx(Text, { children: line }, `art-${i}`))) }), _jsx(Box, { marginTop: 1, children: _jsxs(Text, { dimColor: true, children: ["\"", quote, "\""] }) })] }));
|
|
135
|
+
}
|
|
136
|
+
//# sourceMappingURL=stickers-art.js.map
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { Box, Text } from 'ink';
|
|
3
|
+
import { OUTPUT_STYLES, OUTPUT_STYLE_SLUGS, } from '../core/output-style/presets.js';
|
|
4
|
+
import { useTheme } from '../core/theme/context.js';
|
|
5
|
+
/**
|
|
6
|
+
* Banner above the table. Plain text (not bold) so the prefix `*`
|
|
7
|
+
* remains the dominant active-row cue.
|
|
8
|
+
*/
|
|
9
|
+
function buildBanner(active, source) {
|
|
10
|
+
return `Active style: ${active} (${source})`;
|
|
11
|
+
}
|
|
12
|
+
export function StyleTable({ active, source }) {
|
|
13
|
+
// Leak L30 (2026-05-27): the active-row marker color flows through
|
|
14
|
+
// the theme so `colorblind` operators see cyan instead of green
|
|
15
|
+
// (which their palette re-maps to `success`). Falls back to the
|
|
16
|
+
// default theme's `success` token when no provider is mounted.
|
|
17
|
+
const theme = useTheme();
|
|
18
|
+
const slugWidth = Math.max('NAME'.length, ...OUTPUT_STYLE_SLUGS.map((slug) => slug.length));
|
|
19
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Box, { marginBottom: 1, children: _jsx(Text, { bold: true, children: "Pugi output styles" }) }), _jsx(Box, { children: _jsx(Text, { dimColor: true, children: ` ${'NAME'.padEnd(slugWidth)} GLOSS` }) }), OUTPUT_STYLE_SLUGS.map((slug) => (_jsx(StyleRow, { slug: slug, active: active, slugWidth: slugWidth, activeColor: theme.success }, slug))), _jsx(Box, { marginTop: 1, children: _jsx(Text, { children: buildBanner(active, source) }) })] }));
|
|
20
|
+
}
|
|
21
|
+
function StyleRow({ slug, active, slugWidth, activeColor }) {
|
|
22
|
+
const isActive = slug === active;
|
|
23
|
+
const marker = isActive ? '*' : ' ';
|
|
24
|
+
const slugPart = slug.padEnd(slugWidth, ' ');
|
|
25
|
+
const gloss = OUTPUT_STYLES[slug].gloss;
|
|
26
|
+
return (_jsxs(Box, { children: [_jsx(Text, { color: isActive ? activeColor : undefined, bold: isActive, children: `${marker} ${slugPart}` }), _jsx(Text, { children: ` ${gloss}` })] }));
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=style-table.js.map
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { Box, Text } from 'ink';
|
|
3
|
+
import { compileSampleRow, THEMES, THEME_SLUGS, } from '../core/theme/presets.js';
|
|
4
|
+
/**
|
|
5
|
+
* Banner above the table. Plain text (not bold) so the prefix `*`
|
|
6
|
+
* remains the dominant active-row cue. Mirrors `<StyleTable>` so the
|
|
7
|
+
* Settings-group surfaces read identically.
|
|
8
|
+
*/
|
|
9
|
+
function buildBanner(active, source) {
|
|
10
|
+
return `Active theme: ${active} (${source})`;
|
|
11
|
+
}
|
|
12
|
+
export function ThemeTable({ active, source }) {
|
|
13
|
+
const slugWidth = Math.max('NAME'.length, ...THEME_SLUGS.map((slug) => slug.length));
|
|
14
|
+
// The gloss column gets sized to the widest gloss + 2 padding so
|
|
15
|
+
// the sample column lines up. Computed once per render so the
|
|
16
|
+
// layout stays stable when the catalogue grows.
|
|
17
|
+
const glossWidth = Math.max('GLOSS'.length, ...THEME_SLUGS.map((slug) => THEMES[slug].gloss.length));
|
|
18
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Box, { marginBottom: 1, children: _jsx(Text, { bold: true, children: "Pugi themes" }) }), _jsx(Box, { children: _jsx(Text, { dimColor: true, children: ` ${'NAME'.padEnd(slugWidth)} ${'GLOSS'.padEnd(glossWidth)} SAMPLE` }) }), THEME_SLUGS.map((slug) => (_jsx(ThemeRow, { slug: slug, active: active, slugWidth: slugWidth, glossWidth: glossWidth }, slug))), _jsx(Box, { marginTop: 1, children: _jsx(Text, { children: buildBanner(active, source) }) })] }));
|
|
19
|
+
}
|
|
20
|
+
function ThemeRow({ slug, active, slugWidth, glossWidth }) {
|
|
21
|
+
const isActive = slug === active;
|
|
22
|
+
const marker = isActive ? '*' : ' ';
|
|
23
|
+
const slugPart = slug.padEnd(slugWidth, ' ');
|
|
24
|
+
const preset = THEMES[slug];
|
|
25
|
+
const gloss = preset.gloss.padEnd(glossWidth, ' ');
|
|
26
|
+
const sample = compileSampleRow(slug);
|
|
27
|
+
return (_jsxs(Box, { children: [_jsx(Text, { color: isActive ? preset.colors.accent : undefined, bold: isActive, children: `${marker} ${slugPart}` }), _jsx(Text, { children: ` ${gloss} ` }), _jsx(Text, { color: preset.colors.foreground, children: sample.foreground }), _jsx(Text, { children: ' ' }), _jsx(Text, { color: preset.colors.accent, children: sample.accent }), _jsx(Text, { children: ' ' }), _jsx(Text, { color: preset.colors.success, children: sample.success }), _jsx(Text, { children: ' ' }), _jsx(Text, { color: preset.colors.warning, children: sample.warning }), _jsx(Text, { children: ' ' }), _jsx(Text, { color: preset.colors.error, children: sample.error })] }));
|
|
28
|
+
}
|
|
29
|
+
//# sourceMappingURL=theme-table.js.map
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
2
|
import { Box, Text } from 'ink';
|
|
3
|
+
import { RESULT_PREVIEW_MAX_CHARS, STREAMING_DELTA_MAX_CHARS, } from '../core/repl/session.js';
|
|
3
4
|
const DEFAULT_COLLAPSE_THRESHOLD = 5;
|
|
4
5
|
const DEFAULT_MAX_ROWS = 8;
|
|
5
6
|
export function ToolStreamPane(props) {
|
|
@@ -28,7 +29,38 @@ function ToolCallRow({ call, collapseThreshold, }) {
|
|
|
28
29
|
const label = formatToolLabel(call.tool, call.args);
|
|
29
30
|
const summary = formatSummary(call);
|
|
30
31
|
const showHint = (call.resultLines ?? 0) > collapseThreshold;
|
|
31
|
-
|
|
32
|
+
// Wave 6 small-CC-parity batch (2026-05-27): on a `running` row,
|
|
33
|
+
// surface the rolling streaming delta as a dim inline preview after
|
|
34
|
+
// the label. On a completed row, the same slot carries the
|
|
35
|
+
// `resultPreview` quoted head. Either way the row stays single-line —
|
|
36
|
+
// both fields are clamped к their respective char ceilings upstream.
|
|
37
|
+
const inlineTail = call.status === 'running'
|
|
38
|
+
? call.streamingDelta
|
|
39
|
+
: call.resultPreview;
|
|
40
|
+
// Error rows: render the label too in red so the eye lands on the
|
|
41
|
+
// failure even при peripheral attention. The other states keep the
|
|
42
|
+
// glyph as the only color signal.
|
|
43
|
+
const labelColor = call.status === 'error' ? 'red' : undefined;
|
|
44
|
+
return (_jsxs(Box, { children: [_jsx(Text, { color: color, children: glyph }), _jsx(Text, { children: ' ' }), _jsx(Text, { bold: true, color: labelColor, children: label }), _jsx(Text, { dimColor: true, children: ` ${summary}` }), inlineTail ? (_jsx(Text, { dimColor: true, children: ` ${formatInlineTail(call.status, inlineTail)}` })) : null, showHint ? (_jsx(Text, { dimColor: true, children: ` · ${call.resultLines} lines, Ctrl+O to expand` })) : null] }));
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Wave 6 small-CC-parity batch (2026-05-27): render the inline tail
|
|
48
|
+
* slot for either the live `streamingDelta` (during `running`) or the
|
|
49
|
+
* collapsed `resultPreview` (after completion). Pure helper so the
|
|
50
|
+
* spec can assert the exact shape.
|
|
51
|
+
*
|
|
52
|
+
* running → `… npm WARN deprecated…`
|
|
53
|
+
* ok / error → `"<preview head>"`
|
|
54
|
+
*/
|
|
55
|
+
export function formatInlineTail(status, tail) {
|
|
56
|
+
if (status === 'running') {
|
|
57
|
+
return tail;
|
|
58
|
+
}
|
|
59
|
+
// Wrap the completed preview in quotes so the operator's eye groups
|
|
60
|
+
// the preview block visually distinct from the canonical detail
|
|
61
|
+
// (`OK`, `+12 -0`). Mirrors the Claude Code TUI's quoted-preview
|
|
62
|
+
// pattern.
|
|
63
|
+
return `"${tail}"`;
|
|
32
64
|
}
|
|
33
65
|
function statusGlyph(status) {
|
|
34
66
|
switch (status) {
|
|
@@ -52,18 +84,35 @@ function statusColor(status) {
|
|
|
52
84
|
}
|
|
53
85
|
/**
|
|
54
86
|
* Render the canonical `Tool(args)` form. Tool names are capitalised
|
|
55
|
-
* the way Claude Code shows them; args are truncated to
|
|
56
|
-
*
|
|
87
|
+
* the way Claude Code shows them; args are truncated to keep the row
|
|
88
|
+
* single-line even on 80-col terminals. Cap = 60 chars (chosen empirically
|
|
89
|
+
* to leave room for the glyph + 1-space gap + summary + optional
|
|
90
|
+
* inline tail without overflow on narrow shells).
|
|
57
91
|
*/
|
|
58
92
|
function formatToolLabel(tool, args) {
|
|
59
93
|
const name = toolDisplayName(tool);
|
|
60
94
|
const trimmedArgs = args.length > 60 ? `${args.slice(0, 57)}…` : args;
|
|
61
95
|
return `${name}(${trimmedArgs})`;
|
|
62
96
|
}
|
|
97
|
+
/**
|
|
98
|
+
* Re-export the upstream char caps so the operator-facing constants
|
|
99
|
+
* have one source of truth. The session module owns the canonical
|
|
100
|
+
* values (used both during ingest и during the pane's render lookup);
|
|
101
|
+
* tests assert against these names rather than literal numbers so a
|
|
102
|
+
* future tuning сtays diff-friendly.
|
|
103
|
+
*/
|
|
104
|
+
export { RESULT_PREVIEW_MAX_CHARS, STREAMING_DELTA_MAX_CHARS };
|
|
63
105
|
function toolDisplayName(tool) {
|
|
64
106
|
switch (tool) {
|
|
65
107
|
case 'read':
|
|
66
108
|
return 'Read';
|
|
109
|
+
case 'write':
|
|
110
|
+
// 2026-05-27 — Write is the most operator-visible tool for the
|
|
111
|
+
// codegen-dispatch surface (Hiroshi writing index.html / style.css
|
|
112
|
+
// / script.js for a tic-tac-toe brief). Add the display name so
|
|
113
|
+
// the tool stream pane renders ✓ Write(index.html) instead of an
|
|
114
|
+
// unlabeled placeholder. Mirrors the Claude Code Write rendering.
|
|
115
|
+
return 'Write';
|
|
67
116
|
case 'edit':
|
|
68
117
|
return 'Edit';
|
|
69
118
|
case 'bash':
|
|
@@ -1,8 +1,26 @@
|
|
|
1
1
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
2
|
import { Box, Text } from 'ink';
|
|
3
|
-
import { upgradeCommand } from '../runtime/update-check.js';
|
|
3
|
+
import { compareVersions, upgradeCommand, } from '../runtime/update-check.js';
|
|
4
|
+
import { getCachedServerRecommendation } from '../core/transport/version-interceptor.js';
|
|
5
|
+
/**
|
|
6
|
+
* Resolve the `latest` value the banner should show. Exported so the
|
|
7
|
+
* spec can lock the merge logic without rendering Ink.
|
|
8
|
+
*/
|
|
9
|
+
export function resolveDisplayedLatest(npmLatest, serverRecommended) {
|
|
10
|
+
if (!serverRecommended)
|
|
11
|
+
return npmLatest;
|
|
12
|
+
return compareVersions(serverRecommended, npmLatest) > 0
|
|
13
|
+
? serverRecommended
|
|
14
|
+
: npmLatest;
|
|
15
|
+
}
|
|
4
16
|
export function UpdateBanner({ result }) {
|
|
5
17
|
const command = upgradeCommand(result.method);
|
|
6
|
-
|
|
18
|
+
// Read the cache lazily inside the render so a server response that
|
|
19
|
+
// landed AFTER the banner was constructed still shows up on the next
|
|
20
|
+
// re-render. The cache lookup is a single map read — cheap enough to
|
|
21
|
+
// do per render.
|
|
22
|
+
const serverRecommended = getCachedServerRecommendation();
|
|
23
|
+
const displayedLatest = resolveDisplayedLatest(result.latest, serverRecommended);
|
|
24
|
+
return (_jsxs(Box, { flexDirection: "column", marginBottom: 1, children: [_jsxs(Box, { children: [_jsx(Text, { dimColor: true, children: '─ ' }), _jsx(Text, { bold: true, color: "#3da9fc", children: 'Pugi ' }), _jsx(Text, { children: result.installed }), _jsx(Text, { dimColor: true, children: ' (installed) → ' }), _jsx(Text, { bold: true, children: displayedLatest }), _jsx(Text, { dimColor: true, children: ' (latest)' })] }), _jsxs(Box, { children: [_jsx(Text, { dimColor: true, children: ' Update: ' }), _jsx(Text, { color: "#3da9fc", children: command })] }), _jsx(Box, { children: _jsx(Text, { dimColor: true, children: ' Skip with PUGI_SKIP_UPDATE_BANNER=1' }) })] }));
|
|
7
25
|
}
|
|
8
26
|
//# sourceMappingURL=update-banner.js.map
|