@diegopetrucci/pi-extensions 0.1.38 → 0.1.39

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/README.md CHANGED
@@ -9,6 +9,7 @@ A collection of [pi](https://github.com/earendil-works/pi-mono) agent extensions
9
9
  - [`context-cap`](./extensions/context-cap): Caps effective model context windows at 200k tokens by default so pi avoids the `dumb zone`; toggle temporarily with `/context-cap`.
10
10
  - [`context-inspector`](./extensions/context-inspector): Adds `/context`, a local self-contained HTML dashboard that breaks down where the current session context is going, with category overview, top offenders, and drilldown search.
11
11
  - [`dirty-repo-guard`](./extensions/dirty-repo-guard): Prompts before new sessions, session switches, or forks when the current git repo has uncommitted changes.
12
+ - [`git-footer`](./extensions/git-footer): Standalone footer extension that replaces pi's built-in footer with a TLH-style git summary: branch, dirty counts, ahead/behind, optional PR number, session name, plus context/model status. Not auto-loaded by the collection package because it conflicts with `minimal-footer`.
12
13
  - [`gnosis`](./extensions/gnosis): Exposes the `gn` repo-local knowledge base CLI as an agent tool for searching and recording durable project decisions, constraints, and intent.
13
14
  - [`inline-bash`](./extensions/inline-bash): Expands `!{command}` snippets in user prompts by running them through bash before the prompt reaches the agent.
14
15
  - [`librarian`](./extensions/librarian): Adds a GitHub research scout with a local repo checkout cache enabled by default under the OS user cache directory, toggleable with `/librarian-cache`, with cached repos expiring after 7 days of non-use.
@@ -0,0 +1,41 @@
1
+ # git-footer
2
+
3
+ A TLH-style git status footer for pi.
4
+
5
+ This package is standalone-only and is not auto-loaded by the `@diegopetrucci/pi-extensions` collection package because only one custom footer should win.
6
+
7
+ It replaces pi's built-in footer with a compact two-line layout whose first line mirrors how [`the-last-harness`](https://github.com/diegopetrucci/the-last-harness) summarizes repository state:
8
+
9
+ ```text
10
+ <repo> • <branch> • <git-status> • PR #<number> • <session-name>
11
+ ctx <percent>% • <model> <thinking> • <extension-statuses>
12
+ ```
13
+
14
+ Git status indicators:
15
+
16
+ - `!N`: conflicted paths
17
+ - `+N`: staged paths
18
+ - `~N`: unstaged paths
19
+ - `?N`: untracked paths
20
+ - `↑N`: commits ahead of upstream
21
+ - `↓N`: commits behind upstream
22
+
23
+ The extension polls git status in the background, caches the latest snapshot, and keeps footer rendering synchronous. It also performs a best-effort `gh pr view` lookup for the current branch; if `gh` is unavailable or the branch has no PR, the PR segment is omitted.
24
+
25
+ ## Install
26
+
27
+ ```bash
28
+ pi install npm:@diegopetrucci/pi-git-footer
29
+ ```
30
+
31
+ Then reload pi:
32
+
33
+ ```text
34
+ /reload
35
+ ```
36
+
37
+ ## Notes
38
+
39
+ - Replaces pi's built-in footer entirely.
40
+ - Intended as a separate footer extension; do not enable it at the same time as another custom-footer extension such as `minimal-footer` unless you want the last-loaded footer to win.
41
+ - `render()` never spawns subprocesses. Git and GitHub CLI lookups run on a short background interval with timeouts, and the footer reads only cached snapshots.
@@ -0,0 +1,533 @@
1
+ import { spawn } from "node:child_process";
2
+ import { basename } from "node:path";
3
+ import type { ExtensionAPI } from "@earendil-works/pi-coding-agent";
4
+ import { truncateToWidth } from "@earendil-works/pi-tui";
5
+
6
+ type GitStatusSnapshot = {
7
+ branch?: string;
8
+ staged: number;
9
+ unstaged: number;
10
+ untracked: number;
11
+ conflict: number;
12
+ ahead: number;
13
+ behind: number;
14
+ };
15
+
16
+ type PullRequestSnapshot = {
17
+ number?: number | string;
18
+ state?: string;
19
+ isDraft?: boolean;
20
+ url?: string;
21
+ title?: string;
22
+ };
23
+
24
+ type CommandResult = {
25
+ stdout: string;
26
+ stderr: string;
27
+ exitCode: number | null;
28
+ };
29
+
30
+ type CommandRunner = (
31
+ command: string,
32
+ args: readonly string[],
33
+ options: { cwd: string; signal: AbortSignal },
34
+ ) => Promise<CommandResult>;
35
+
36
+ type TimerHandle = unknown;
37
+
38
+ type Clock = {
39
+ setInterval(callback: () => void, ms: number): TimerHandle;
40
+ clearInterval(handle: TimerHandle): void;
41
+ };
42
+
43
+ type GitFooterCacheOptions = {
44
+ cwd: () => string;
45
+ runner?: CommandRunner;
46
+ clock?: Clock;
47
+ refreshIntervalMs?: number;
48
+ gitTimeoutMs?: number;
49
+ ghTimeoutMs?: number;
50
+ onChange?: () => void;
51
+ onBranchChangeSource?: (callback: () => void) => () => void;
52
+ };
53
+
54
+ const BRANCH_HEAD_PREFIX = "# branch.head ";
55
+ const BRANCH_AB_PREFIX = "# branch.ab ";
56
+ const FOOTER_SEPARATOR = " • ";
57
+ const DEFAULT_REFRESH_INTERVAL_MS = 8_000;
58
+ const DEFAULT_GIT_TIMEOUT_MS = 1_500;
59
+ const DEFAULT_GH_TIMEOUT_MS = 3_000;
60
+ const GIT_STATUS_ARGS = ["--no-optional-locks", "status", "--porcelain=v2", "--branch"] as const;
61
+ const GH_PR_VIEW_ARGS = ["pr", "view", "--json", "number,state,isDraft,url,title"] as const;
62
+
63
+ function createEmptyGitStatus(): GitStatusSnapshot {
64
+ return {
65
+ branch: undefined,
66
+ staged: 0,
67
+ unstaged: 0,
68
+ untracked: 0,
69
+ conflict: 0,
70
+ ahead: 0,
71
+ behind: 0,
72
+ };
73
+ }
74
+
75
+ function positiveCount(value: number): number {
76
+ return Number.isFinite(value) && value > 0 ? Math.trunc(value) : 0;
77
+ }
78
+
79
+ function addTrackedStatusCounts(status: GitStatusSnapshot, xy: string): void {
80
+ if (xy.length !== 2) return;
81
+ if (xy[0] !== ".") status.staged += 1;
82
+ if (xy[1] !== ".") status.unstaged += 1;
83
+ }
84
+
85
+ function parseBranchAheadBehind(line: string, status: GitStatusSnapshot): void {
86
+ const match = /^# branch\.ab \+(\d+) -(\d+)$/.exec(line);
87
+ if (!match) return;
88
+ status.ahead = Number.parseInt(match[1]!, 10);
89
+ status.behind = Number.parseInt(match[2]!, 10);
90
+ }
91
+
92
+ function normalizeBranchHead(value: string): string {
93
+ const branch = value.trim();
94
+ return branch === "(detached)" ? "detached" : branch;
95
+ }
96
+
97
+ function parseGitStatusPorcelainV2(output: string): GitStatusSnapshot {
98
+ const status = createEmptyGitStatus();
99
+
100
+ for (const rawLine of output.split("\n")) {
101
+ const line = rawLine.endsWith("\r") ? rawLine.slice(0, -1) : rawLine;
102
+ if (!line) continue;
103
+
104
+ if (line.startsWith(BRANCH_HEAD_PREFIX)) {
105
+ status.branch = normalizeBranchHead(line.slice(BRANCH_HEAD_PREFIX.length)) || undefined;
106
+ continue;
107
+ }
108
+
109
+ if (line.startsWith(BRANCH_AB_PREFIX)) {
110
+ parseBranchAheadBehind(line, status);
111
+ continue;
112
+ }
113
+
114
+ if (line.startsWith("1 ") || line.startsWith("2 ")) {
115
+ addTrackedStatusCounts(status, line.slice(2, 4));
116
+ continue;
117
+ }
118
+
119
+ if (line.startsWith("u ")) {
120
+ status.conflict += 1;
121
+ continue;
122
+ }
123
+
124
+ if (line.startsWith("? ")) status.untracked += 1;
125
+ }
126
+
127
+ return status;
128
+ }
129
+
130
+ function formatGitStatusFooterSegment(status: GitStatusSnapshot | undefined): string | undefined {
131
+ if (!status) return undefined;
132
+
133
+ const parts: string[] = [];
134
+ const indicators: Array<[string, number]> = [
135
+ ["!", positiveCount(status.conflict)],
136
+ ["+", positiveCount(status.staged)],
137
+ ["~", positiveCount(status.unstaged)],
138
+ ["?", positiveCount(status.untracked)],
139
+ ["↑", positiveCount(status.ahead)],
140
+ ["↓", positiveCount(status.behind)],
141
+ ];
142
+
143
+ for (const [prefix, count] of indicators) {
144
+ if (count > 0) parts.push(`${prefix}${count}`);
145
+ }
146
+
147
+ return parts.length > 0 ? parts.join(" ") : undefined;
148
+ }
149
+
150
+ function formatPullRequestFooterSegment(pullRequest: PullRequestSnapshot | undefined): string | undefined {
151
+ const value = pullRequest?.number;
152
+ if (Number.isSafeInteger(value) && Number(value) > 0) return `PR #${value}`;
153
+ if (typeof value === "string") {
154
+ const trimmed = value.trim();
155
+ if (/^[1-9]\d*$/.test(trimmed)) return `PR #${trimmed}`;
156
+ }
157
+ return undefined;
158
+ }
159
+
160
+ function formatGitFooterSegments(
161
+ status: GitStatusSnapshot | undefined,
162
+ pullRequest: PullRequestSnapshot | undefined,
163
+ ): string[] {
164
+ const segments: string[] = [];
165
+ const branch = typeof status?.branch === "string" ? status.branch.trim() : "";
166
+ if (branch) segments.push(branch);
167
+
168
+ const statusSegment = formatGitStatusFooterSegment(status);
169
+ if (statusSegment) segments.push(statusSegment);
170
+
171
+ const pullRequestSegment = formatPullRequestFooterSegment(pullRequest);
172
+ if (pullRequestSegment) segments.push(pullRequestSegment);
173
+
174
+ return segments;
175
+ }
176
+
177
+ function parsePullRequestJson(stdout: string): PullRequestSnapshot | undefined {
178
+ const trimmed = stdout.trim();
179
+ if (!trimmed) return undefined;
180
+ let parsed: unknown;
181
+ try {
182
+ parsed = JSON.parse(trimmed);
183
+ } catch {
184
+ return undefined;
185
+ }
186
+ if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) return undefined;
187
+
188
+ const record = parsed as Record<string, unknown>;
189
+ const snapshot: PullRequestSnapshot = {};
190
+ if (typeof record.number === "number" || typeof record.number === "string") {
191
+ snapshot.number = record.number;
192
+ }
193
+ if (typeof record.state === "string") snapshot.state = record.state;
194
+ if (typeof record.isDraft === "boolean") snapshot.isDraft = record.isDraft;
195
+ if (typeof record.url === "string") snapshot.url = record.url;
196
+ if (typeof record.title === "string") snapshot.title = record.title;
197
+ return snapshot;
198
+ }
199
+
200
+ function gitStatusSnapshotsEqual(
201
+ left: GitStatusSnapshot | undefined,
202
+ right: GitStatusSnapshot | undefined,
203
+ ): boolean {
204
+ return (
205
+ left?.branch === right?.branch
206
+ && left?.staged === right?.staged
207
+ && left?.unstaged === right?.unstaged
208
+ && left?.untracked === right?.untracked
209
+ && left?.conflict === right?.conflict
210
+ && left?.ahead === right?.ahead
211
+ && left?.behind === right?.behind
212
+ );
213
+ }
214
+
215
+ function pullRequestSnapshotsEqual(
216
+ left: PullRequestSnapshot | undefined,
217
+ right: PullRequestSnapshot | undefined,
218
+ ): boolean {
219
+ return (
220
+ left?.number === right?.number
221
+ && left?.state === right?.state
222
+ && left?.isDraft === right?.isDraft
223
+ && left?.url === right?.url
224
+ && left?.title === right?.title
225
+ );
226
+ }
227
+
228
+ function defaultRunner(
229
+ command: string,
230
+ args: readonly string[],
231
+ options: { cwd: string; signal: AbortSignal },
232
+ ): Promise<CommandResult> {
233
+ return new Promise((resolve, reject) => {
234
+ let child;
235
+ try {
236
+ child = spawn(command, [...args], {
237
+ cwd: options.cwd,
238
+ stdio: ["ignore", "pipe", "pipe"],
239
+ windowsHide: true,
240
+ });
241
+ } catch (error) {
242
+ reject(error);
243
+ return;
244
+ }
245
+
246
+ let stdout = "";
247
+ let stderr = "";
248
+ let settled = false;
249
+
250
+ const finish = (result: CommandResult | Error) => {
251
+ if (settled) return;
252
+ settled = true;
253
+ options.signal.removeEventListener("abort", onAbort);
254
+ if (result instanceof Error) reject(result);
255
+ else resolve(result);
256
+ };
257
+
258
+ const onAbort = () => {
259
+ try {
260
+ child.kill("SIGTERM");
261
+ } catch {
262
+ // Ignore: process may already be gone.
263
+ }
264
+ finish(new Error("aborted"));
265
+ };
266
+
267
+ child.stdout?.on("data", (chunk: Buffer | string) => {
268
+ stdout += typeof chunk === "string" ? chunk : chunk.toString("utf8");
269
+ });
270
+ child.stderr?.on("data", (chunk: Buffer | string) => {
271
+ stderr += typeof chunk === "string" ? chunk : chunk.toString("utf8");
272
+ });
273
+ child.on("error", finish);
274
+ child.on("close", (code) => finish({ stdout, stderr, exitCode: code }));
275
+
276
+ if (options.signal.aborted) {
277
+ onAbort();
278
+ return;
279
+ }
280
+ options.signal.addEventListener("abort", onAbort, { once: true });
281
+ });
282
+ }
283
+
284
+ function defaultClock(): Clock {
285
+ return {
286
+ setInterval(callback, ms) {
287
+ const handle = setInterval(callback, ms);
288
+ (handle as { unref?: () => void }).unref?.();
289
+ return handle;
290
+ },
291
+ clearInterval(handle) {
292
+ clearInterval(handle as ReturnType<typeof setInterval>);
293
+ },
294
+ };
295
+ }
296
+
297
+ class GitFooterCache {
298
+ private readonly cwd: () => string;
299
+ private readonly runner: CommandRunner;
300
+ private readonly clock: Clock;
301
+ private readonly refreshIntervalMs: number;
302
+ private readonly gitTimeoutMs: number;
303
+ private readonly ghTimeoutMs: number;
304
+ private readonly onChange: (() => void) | undefined;
305
+
306
+ private intervalHandle: TimerHandle | undefined;
307
+ private readonly inflightControllers = new Set<AbortController>();
308
+ private disposed = false;
309
+ private refreshInFlight: Promise<void> | undefined;
310
+ private branchChangeUnsubscribe: (() => void) | undefined;
311
+ private statusSnapshot: GitStatusSnapshot | undefined;
312
+ private pullRequestSnapshot: PullRequestSnapshot | undefined;
313
+ private lastSeenBranch: string | undefined;
314
+
315
+ constructor(options: GitFooterCacheOptions) {
316
+ this.cwd = options.cwd;
317
+ this.runner = options.runner ?? defaultRunner;
318
+ this.clock = options.clock ?? defaultClock();
319
+ this.refreshIntervalMs = options.refreshIntervalMs ?? DEFAULT_REFRESH_INTERVAL_MS;
320
+ this.gitTimeoutMs = options.gitTimeoutMs ?? DEFAULT_GIT_TIMEOUT_MS;
321
+ this.ghTimeoutMs = options.ghTimeoutMs ?? DEFAULT_GH_TIMEOUT_MS;
322
+ this.onChange = options.onChange;
323
+
324
+ this.intervalHandle = this.clock.setInterval(() => {
325
+ void this.refresh();
326
+ }, this.refreshIntervalMs);
327
+ void this.refresh();
328
+
329
+ if (options.onBranchChangeSource) {
330
+ this.branchChangeUnsubscribe = options.onBranchChangeSource(() => {
331
+ void this.refresh();
332
+ });
333
+ }
334
+ }
335
+
336
+ getStatusSnapshot(): GitStatusSnapshot | undefined {
337
+ return this.statusSnapshot;
338
+ }
339
+
340
+ getPullRequestSnapshot(): PullRequestSnapshot | undefined {
341
+ return this.pullRequestSnapshot;
342
+ }
343
+
344
+ refresh(): Promise<void> {
345
+ if (this.disposed) return Promise.resolve();
346
+ if (this.refreshInFlight) return this.refreshInFlight;
347
+ const run = this.runRefresh()
348
+ .finally(() => {
349
+ this.refreshInFlight = undefined;
350
+ })
351
+ .catch(() => undefined);
352
+ this.refreshInFlight = run;
353
+ return run;
354
+ }
355
+
356
+ private async runRefresh(): Promise<void> {
357
+ const previousStatusSnapshot = this.statusSnapshot;
358
+ const previousPullRequestSnapshot = this.pullRequestSnapshot;
359
+
360
+ const result = await this.fetchGitStatus();
361
+ if (this.disposed) return;
362
+ if (result.kind === "transient") return;
363
+ if (result.kind === "not-a-repo") {
364
+ this.statusSnapshot = undefined;
365
+ this.pullRequestSnapshot = undefined;
366
+ this.lastSeenBranch = undefined;
367
+ this.emitChangeIfSnapshotsChanged(previousStatusSnapshot, previousPullRequestSnapshot);
368
+ return;
369
+ }
370
+
371
+ const status = result.status;
372
+ this.statusSnapshot = status;
373
+
374
+ const branch = typeof status.branch === "string" ? status.branch : undefined;
375
+ const isValidBranch = !!branch && branch !== "detached";
376
+ const branchChanged = branch !== this.lastSeenBranch;
377
+
378
+ if (branchChanged) this.pullRequestSnapshot = undefined;
379
+ this.lastSeenBranch = branch;
380
+
381
+ if (!isValidBranch) {
382
+ this.pullRequestSnapshot = undefined;
383
+ this.emitChangeIfSnapshotsChanged(previousStatusSnapshot, previousPullRequestSnapshot);
384
+ return;
385
+ }
386
+
387
+ const pr = await this.fetchPullRequest();
388
+ if (this.disposed) return;
389
+ if (pr !== undefined) this.pullRequestSnapshot = pr;
390
+ else if (branchChanged) this.pullRequestSnapshot = undefined;
391
+
392
+ this.emitChangeIfSnapshotsChanged(previousStatusSnapshot, previousPullRequestSnapshot);
393
+ }
394
+
395
+ private emitChangeIfSnapshotsChanged(
396
+ previousStatusSnapshot: GitStatusSnapshot | undefined,
397
+ previousPullRequestSnapshot: PullRequestSnapshot | undefined,
398
+ ): void {
399
+ if (this.disposed) return;
400
+ if (
401
+ gitStatusSnapshotsEqual(previousStatusSnapshot, this.statusSnapshot)
402
+ && pullRequestSnapshotsEqual(previousPullRequestSnapshot, this.pullRequestSnapshot)
403
+ ) {
404
+ return;
405
+ }
406
+ try {
407
+ this.onChange?.();
408
+ } catch {
409
+ // Rendering hooks should not break refreshes.
410
+ }
411
+ }
412
+
413
+ private async fetchGitStatus(): Promise<
414
+ | { kind: "ok"; status: GitStatusSnapshot }
415
+ | { kind: "not-a-repo" }
416
+ | { kind: "transient" }
417
+ > {
418
+ const result = await this.runCommandSafely("git", GIT_STATUS_ARGS, this.gitTimeoutMs);
419
+ if (!result) return { kind: "transient" };
420
+ if (result.exitCode !== 0) return { kind: "not-a-repo" };
421
+ return { kind: "ok", status: parseGitStatusPorcelainV2(result.stdout) };
422
+ }
423
+
424
+ private async fetchPullRequest(): Promise<PullRequestSnapshot | undefined> {
425
+ const result = await this.runCommandSafely("gh", GH_PR_VIEW_ARGS, this.ghTimeoutMs);
426
+ if (!result || result.exitCode !== 0) return undefined;
427
+ return parsePullRequestJson(result.stdout);
428
+ }
429
+
430
+ private async runCommandSafely(
431
+ command: string,
432
+ args: readonly string[],
433
+ timeoutMs: number,
434
+ ): Promise<CommandResult | undefined> {
435
+ if (this.disposed) return undefined;
436
+ const controller = new AbortController();
437
+ this.inflightControllers.add(controller);
438
+ const timeoutId = setTimeout(() => controller.abort(), timeoutMs);
439
+ try {
440
+ return await this.runner(command, args, { cwd: this.cwd(), signal: controller.signal });
441
+ } catch {
442
+ return undefined;
443
+ } finally {
444
+ clearTimeout(timeoutId);
445
+ this.inflightControllers.delete(controller);
446
+ }
447
+ }
448
+
449
+ dispose(): void {
450
+ if (this.disposed) return;
451
+ this.disposed = true;
452
+ if (this.intervalHandle !== undefined) {
453
+ this.clock.clearInterval(this.intervalHandle);
454
+ this.intervalHandle = undefined;
455
+ }
456
+ if (this.branchChangeUnsubscribe) {
457
+ try {
458
+ this.branchChangeUnsubscribe();
459
+ } catch {
460
+ // Ignore misbehaving notifier.
461
+ }
462
+ this.branchChangeUnsubscribe = undefined;
463
+ }
464
+ for (const controller of this.inflightControllers) controller.abort();
465
+ this.inflightControllers.clear();
466
+ }
467
+ }
468
+
469
+ function composeFooterFirstLine(input: {
470
+ cwd: string;
471
+ sessionName?: string | null;
472
+ status?: GitStatusSnapshot;
473
+ pullRequest?: PullRequestSnapshot;
474
+ }): string {
475
+ const segments = [input.cwd];
476
+ if (input.status !== undefined) {
477
+ segments.push(...formatGitFooterSegments(input.status, input.pullRequest));
478
+ }
479
+ if (input.sessionName) segments.push(input.sessionName);
480
+ return segments.join(FOOTER_SEPARATOR);
481
+ }
482
+
483
+ function sanitizeFooterSegment(value: string): string {
484
+ return value.replace(/[\r\n\t]+/g, " ").trim();
485
+ }
486
+
487
+ export default function (pi: ExtensionAPI) {
488
+ pi.on("session_start", (_event, ctx) => {
489
+ let cache: GitFooterCache | undefined;
490
+
491
+ ctx.ui.setFooter((tui, theme, footerData) => {
492
+ cache = new GitFooterCache({
493
+ cwd: () => ctx.cwd,
494
+ onChange: () => tui.requestRender(),
495
+ onBranchChangeSource: (callback) => footerData.onBranchChange(callback),
496
+ });
497
+
498
+ return {
499
+ dispose() {
500
+ cache?.dispose();
501
+ cache = undefined;
502
+ },
503
+ invalidate() {},
504
+ render(width: number): string[] {
505
+ const status = cache?.getStatusSnapshot();
506
+ const pullRequest = cache?.getPullRequestSnapshot();
507
+ const firstLine = composeFooterFirstLine({
508
+ cwd: basename(ctx.cwd),
509
+ status,
510
+ pullRequest,
511
+ sessionName: pi.getSessionName(),
512
+ });
513
+
514
+ const usage = ctx.getContextUsage();
515
+ const context = usage?.percent == null ? "ctx ?" : `ctx ${usage.percent.toFixed(1)}%`;
516
+ const thinking = pi.getThinkingLevel();
517
+ const model = ctx.model?.id ?? "no-model";
518
+ const modelText = thinking === "off" ? model : `${model} ${thinking}`;
519
+ const statuses = [...footerData.getExtensionStatuses().values()]
520
+ .map(sanitizeFooterSegment)
521
+ .filter(Boolean);
522
+ const secondLine = [theme.fg("dim", context), theme.fg("dim", modelText), ...statuses]
523
+ .join(theme.fg("dim", FOOTER_SEPARATOR));
524
+
525
+ return [
526
+ truncateToWidth(theme.fg("dim", firstLine), width),
527
+ truncateToWidth(secondLine, width),
528
+ ];
529
+ },
530
+ };
531
+ });
532
+ });
533
+ }
@@ -0,0 +1,35 @@
1
+ {
2
+ "name": "@diegopetrucci/pi-git-footer",
3
+ "version": "0.1.0",
4
+ "description": "A TLH-style git status footer for pi.",
5
+ "keywords": [
6
+ "pi-package",
7
+ "pi",
8
+ "terminal",
9
+ "footer",
10
+ "git"
11
+ ],
12
+ "license": "MIT",
13
+ "repository": {
14
+ "type": "git",
15
+ "url": "git+https://github.com/diegopetrucci/pi-extensions.git",
16
+ "directory": "extensions/git-footer"
17
+ },
18
+ "files": [
19
+ "index.ts",
20
+ "README.md",
21
+ ".pi-fleet-tested-version"
22
+ ],
23
+ "publishConfig": {
24
+ "access": "public"
25
+ },
26
+ "pi": {
27
+ "extensions": [
28
+ "index.ts"
29
+ ]
30
+ },
31
+ "peerDependencies": {
32
+ "@earendil-works/pi-coding-agent": "*",
33
+ "@earendil-works/pi-tui": "*"
34
+ }
35
+ }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@diegopetrucci/pi-extensions",
3
- "version": "0.1.38",
3
+ "version": "0.1.39",
4
4
  "description": "A collection of pi extensions for context management, workflow audits, review-comment triage, notifications, brrr push alerts, safety guards, GitHub research, repo-local knowledge, todos, tool rendering, and model/provider helpers.",
5
5
  "keywords": [
6
6
  "pi-package",