git-compass 0.2.0

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.
Files changed (74) hide show
  1. package/README.md +114 -0
  2. package/dist/__tests__/commands.test.d.ts +2 -0
  3. package/dist/__tests__/commands.test.d.ts.map +1 -0
  4. package/dist/__tests__/commands.test.js +26 -0
  5. package/dist/__tests__/commands.test.js.map +1 -0
  6. package/dist/__tests__/sanity.test.d.ts +2 -0
  7. package/dist/__tests__/sanity.test.d.ts.map +1 -0
  8. package/dist/__tests__/sanity.test.js +7 -0
  9. package/dist/__tests__/sanity.test.js.map +1 -0
  10. package/dist/bin/git-compass.d.ts +3 -0
  11. package/dist/bin/git-compass.d.ts.map +1 -0
  12. package/dist/bin/git-compass.js +4 -0
  13. package/dist/bin/git-compass.js.map +1 -0
  14. package/dist/commands/analyze-all.d.ts +3 -0
  15. package/dist/commands/analyze-all.d.ts.map +1 -0
  16. package/dist/commands/analyze-all.js +136 -0
  17. package/dist/commands/analyze-all.js.map +1 -0
  18. package/dist/commands/analyze.d.ts +3 -0
  19. package/dist/commands/analyze.d.ts.map +1 -0
  20. package/dist/commands/analyze.js +202 -0
  21. package/dist/commands/analyze.js.map +1 -0
  22. package/dist/commands/config.d.ts +3 -0
  23. package/dist/commands/config.d.ts.map +1 -0
  24. package/dist/commands/config.js +78 -0
  25. package/dist/commands/config.js.map +1 -0
  26. package/dist/commands/query.d.ts +3 -0
  27. package/dist/commands/query.d.ts.map +1 -0
  28. package/dist/commands/query.js +78 -0
  29. package/dist/commands/query.js.map +1 -0
  30. package/dist/commands/watch.d.ts +3 -0
  31. package/dist/commands/watch.d.ts.map +1 -0
  32. package/dist/commands/watch.js +49 -0
  33. package/dist/commands/watch.js.map +1 -0
  34. package/dist/config/__tests__/config.test.d.ts +2 -0
  35. package/dist/config/__tests__/config.test.d.ts.map +1 -0
  36. package/dist/config/__tests__/config.test.js +45 -0
  37. package/dist/config/__tests__/config.test.js.map +1 -0
  38. package/dist/config/index.d.ts +10 -0
  39. package/dist/config/index.d.ts.map +1 -0
  40. package/dist/config/index.js +82 -0
  41. package/dist/config/index.js.map +1 -0
  42. package/dist/constants/index.d.ts +15 -0
  43. package/dist/constants/index.d.ts.map +1 -0
  44. package/dist/constants/index.js +15 -0
  45. package/dist/constants/index.js.map +1 -0
  46. package/dist/formatters/__tests__/console.test.d.ts +2 -0
  47. package/dist/formatters/__tests__/console.test.d.ts.map +1 -0
  48. package/dist/formatters/__tests__/console.test.js +40 -0
  49. package/dist/formatters/__tests__/console.test.js.map +1 -0
  50. package/dist/formatters/console.d.ts +3 -0
  51. package/dist/formatters/console.d.ts.map +1 -0
  52. package/dist/formatters/console.js +148 -0
  53. package/dist/formatters/console.js.map +1 -0
  54. package/dist/formatters/report-gen.d.ts +3 -0
  55. package/dist/formatters/report-gen.d.ts.map +1 -0
  56. package/dist/formatters/report-gen.js +8 -0
  57. package/dist/formatters/report-gen.js.map +1 -0
  58. package/dist/index.d.ts +2 -0
  59. package/dist/index.d.ts.map +1 -0
  60. package/dist/index.js +47 -0
  61. package/dist/index.js.map +1 -0
  62. package/dist/utils/__tests__/cache.test.d.ts +2 -0
  63. package/dist/utils/__tests__/cache.test.d.ts.map +1 -0
  64. package/dist/utils/__tests__/cache.test.js +38 -0
  65. package/dist/utils/__tests__/cache.test.js.map +1 -0
  66. package/dist/utils/cache.d.ts +12 -0
  67. package/dist/utils/cache.d.ts.map +1 -0
  68. package/dist/utils/cache.js +47 -0
  69. package/dist/utils/cache.js.map +1 -0
  70. package/dist/utils/gitignore.d.ts +5 -0
  71. package/dist/utils/gitignore.d.ts.map +1 -0
  72. package/dist/utils/gitignore.js +36 -0
  73. package/dist/utils/gitignore.js.map +1 -0
  74. package/package.json +64 -0
@@ -0,0 +1,148 @@
1
+ import chalk from "chalk";
2
+ import boxen from "boxen";
3
+ import { table, getBorderCharacters } from "table";
4
+ export function printConsoleReport(result, detailLevel = "normal", showAI = false) {
5
+ try {
6
+ const { meta, hotspots, riskScores, contributors, burnout, coupling, knowledge, impact, rot } = result;
7
+ const isVerbose = detailLevel === "verbose";
8
+ const isSummary = detailLevel === "summary";
9
+ const limit = isVerbose ? 10 : 5;
10
+ // 1. Header Summary (Always shown)
11
+ console.log(boxen(`${chalk.cyan.bold("Git Compass ANALYSIS SUMMARY")}\n` +
12
+ `${chalk.gray("────────────────────────")}\n` +
13
+ `${chalk.white.bold("Repo: ")} ${chalk.cyan(meta.repoPath)}\n` +
14
+ `${chalk.white.bold("Branch: ")} ${chalk.yellow(meta.branch)}\n` +
15
+ `${chalk.white.bold("Window: ")} ${chalk.magenta(meta.window)}\n` +
16
+ `${chalk.white.bold("Commits:")} ${chalk.green(meta.commitCount)}`, {
17
+ padding: 1,
18
+ margin: { top: 1, bottom: 1 },
19
+ borderStyle: "round",
20
+ borderColor: "cyan",
21
+ title: "Repository Metadata",
22
+ titleAlignment: "left"
23
+ }));
24
+ if (isSummary) {
25
+ printHealthIndicators(impact, rot, !!result.aiSummary);
26
+ return;
27
+ }
28
+ // 2. AI Insights (Intuitive Format)
29
+ if (showAI && result.aiSummary) {
30
+ console.log(boxen(chalk.white(result.aiSummary.digest) + "\n\n" +
31
+ chalk.gray.italic(`Provider: ${result.aiSummary.provider} | Model: ${result.aiSummary.model}`), {
32
+ padding: 1,
33
+ margin: { bottom: 1 },
34
+ borderStyle: "double",
35
+ borderColor: "magenta",
36
+ title: "AI ARCHITECTURAL INSIGHTS",
37
+ titleAlignment: "center",
38
+ width: 80 // Limit width for better readability of structured text
39
+ }));
40
+ }
41
+ // 3. Hotspots Table
42
+ if (hotspots.length > 0) {
43
+ console.log(chalk.yellow.bold("\nTop Hotspots"));
44
+ const hotspotData = [
45
+ [chalk.bold("File"), chalk.bold("Changes"), chalk.bold("Authors"), chalk.bold("Risk Level")]
46
+ ];
47
+ hotspots.slice(0, limit).forEach(h => {
48
+ const fileRisk = riskScores.find(r => r.path === h.path);
49
+ const riskLevel = fileRisk?.level || "low";
50
+ const riskColor = getRiskColor(riskLevel);
51
+ hotspotData.push([
52
+ chalk.white(h.path),
53
+ chalk.cyan(h.changeCount.toString()),
54
+ chalk.magenta(h.uniqueAuthors.toString()),
55
+ chalk.hex(riskColor).bold(riskLevel.toUpperCase())
56
+ ]);
57
+ });
58
+ console.log(table(hotspotData, {
59
+ border: getBorderCharacters("ramac"),
60
+ header: {
61
+ alignment: 'center',
62
+ content: 'Frequently Edited Files'
63
+ }
64
+ }));
65
+ }
66
+ // 4. High Risk Alert
67
+ const highRisk = riskScores.filter(r => r.level === "high" || r.level === "critical");
68
+ if (highRisk.length > 0) {
69
+ console.log(chalk.red.bold("\nHigh Risk Alert"));
70
+ highRisk.slice(0, isVerbose ? 10 : 3).forEach(r => {
71
+ const color = r.level === "critical" ? chalk.bgRed.white.bold : chalk.red.bold;
72
+ console.log(` ${color(" " + r.level.toUpperCase() + " ")} ${chalk.white(r.path)} ${chalk.gray("(Score: " + r.score + ")")}`);
73
+ });
74
+ }
75
+ // 5. Contributor Stats
76
+ if (contributors.length > 0) {
77
+ console.log(chalk.green.bold("\nTop Contributors"));
78
+ const contribData = [
79
+ [chalk.bold("Author"), chalk.bold("Commits"), chalk.bold("Activity"), chalk.bold("Status")]
80
+ ];
81
+ contributors.slice(0, limit).forEach(c => {
82
+ const burnoutInfo = burnout.contributors.find(b => b.author === c.author);
83
+ const isBurnedOut = burnoutInfo && (burnoutInfo.afterHoursPercent > 30 || burnoutInfo.weekendPercent > 30);
84
+ contribData.push([
85
+ chalk.white(c.author),
86
+ chalk.green(c.commitCount.toString()),
87
+ chalk.cyan(`${c.activeDays} days`),
88
+ isBurnedOut ? chalk.bgRed.white.bold(" BURNOUT ") : chalk.bgGreen.black.bold(" STABLE ")
89
+ ]);
90
+ });
91
+ console.log(table(contribData, { border: getBorderCharacters("ramac") }));
92
+ }
93
+ // 6. Deep Insights (Verbose only)
94
+ if (isVerbose) {
95
+ const insightsData = [];
96
+ if (coupling.length > 0) {
97
+ insightsData.push([chalk.bold("Temporal Coupling")]);
98
+ coupling.slice(0, 5).forEach(c => {
99
+ insightsData.push([` ${chalk.white(c.head)} ↔ ${chalk.white(c.tail)}\n ${chalk.gray("Strength: ")}${chalk.magenta((c.coupling * 100).toFixed(0) + "%")}`]);
100
+ });
101
+ }
102
+ if (knowledge.length > 0) {
103
+ insightsData.push([chalk.bold("Knowledge Silos")]);
104
+ knowledge.slice(0, 5).forEach(k => {
105
+ const color = k.riskLevel === "high" ? chalk.red : k.riskLevel === "medium" ? chalk.yellow : chalk.white;
106
+ insightsData.push([` ${chalk.white(k.path)}\n ${chalk.gray("Owner: ")}${color(k.mainContributor)} ${chalk.gray("(" + k.authorshipPercent + "%)")}`]);
107
+ });
108
+ }
109
+ if (insightsData.length > 0) {
110
+ console.log(chalk.blue.bold("\nDeep Architecture Insights"));
111
+ console.log(table(insightsData, { border: getBorderCharacters("ramac") }));
112
+ }
113
+ }
114
+ // 7. Health Indicators & Footer Tip
115
+ printHealthIndicators(impact, rot, showAI);
116
+ }
117
+ catch (err) {
118
+ console.error(chalk.red("\nError printing report:"), err);
119
+ }
120
+ }
121
+ function printHealthIndicators(impact, rot, showAI) {
122
+ const avgImpact = impact.length > 0 ? (impact.reduce((acc, i) => acc + i.blastRadius, 0) / impact.length).toFixed(2) : 0;
123
+ const footerContent = [
124
+ `${chalk.bold("Overall Health Indicators")}`,
125
+ `${chalk.gray("────────────────────────")}`,
126
+ `${chalk.white("Average Blast Radius: ")} ${chalk.yellow(avgImpact + " files")}`,
127
+ `${chalk.white("Abandoned Files (Rot): ")} ${chalk.red(rot.length)}`
128
+ ];
129
+ if (!showAI) {
130
+ footerContent.push("");
131
+ footerContent.push(`${chalk.gray.italic("Tip: Run 'git-compass config set-ai' to unlock AI-powered insights (use --ai flag).")}`);
132
+ }
133
+ console.log(boxen(footerContent.join("\n"), {
134
+ padding: 1,
135
+ margin: { top: 1, bottom: 1 },
136
+ borderStyle: "round",
137
+ borderColor: "gray"
138
+ }));
139
+ }
140
+ function getRiskColor(level) {
141
+ switch (level) {
142
+ case "critical": return "#FF0000";
143
+ case "high": return "#FF4500";
144
+ case "medium": return "#FFA500";
145
+ default: return "#00FF00";
146
+ }
147
+ }
148
+ //# sourceMappingURL=console.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"console.js","sourceRoot":"","sources":["../../src/formatters/console.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { AnalysisResult } from "@git-compass/core";
2
+ export declare function exportJson(result: AnalysisResult, outputPath: string): Promise<string>;
3
+ //# sourceMappingURL=report-gen.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"report-gen.d.ts","sourceRoot":"","sources":["../../src/formatters/report-gen.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAExD,wBAAsB,UAAU,CAAC,MAAM,EAAE,cAAc,EAAE,UAAU,EAAE,MAAM,mBAI1E"}
@@ -0,0 +1,8 @@
1
+ import fs from "fs/promises";
2
+ import path from "path";
3
+ export async function exportJson(result, outputPath) {
4
+ const fullPath = path.resolve(outputPath);
5
+ await fs.writeFile(fullPath, JSON.stringify(result, null, 2), "utf-8");
6
+ return fullPath;
7
+ }
8
+ //# sourceMappingURL=report-gen.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"report-gen.js","sourceRoot":"","sources":["../../src/formatters/report-gen.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,aAAa,CAAC;AAC7B,OAAO,IAAI,MAAM,MAAM,CAAC;AAIxB,MAAM,CAAC,KAAK,UAAU,UAAU,CAAC,MAAsB,EAAE,UAAkB;IACzE,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC1C,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IACvE,OAAO,QAAQ,CAAC;AAClB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function run(): void;
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AA4CA,wBAAgB,GAAG,SAYlB"}
package/dist/index.js ADDED
@@ -0,0 +1,47 @@
1
+ import { Command } from "commander";
2
+ import { configCommand } from "./commands/config.js";
3
+ import { analyzeCommand } from "./commands/analyze.js";
4
+ import { analyzeAllCommand } from "./commands/analyze-all.js";
5
+ import { watchCommand } from "./commands/watch.js";
6
+ import { queryCommand } from "./commands/query.js";
7
+ import chalk from "chalk";
8
+ const program = new Command();
9
+ program
10
+ .name("git-compass")
11
+ .description("Git repository analytics from your terminal")
12
+ .version("0.1.0");
13
+ program.addCommand(configCommand);
14
+ program.addCommand(analyzeCommand);
15
+ program.addCommand(analyzeAllCommand);
16
+ program.addCommand(watchCommand);
17
+ program.addCommand(queryCommand);
18
+ // Handle unknown commands gracefully
19
+ program.on("command:*", (cmds) => {
20
+ const unknownCommand = cmds[0];
21
+ const availableCommands = program.commands.map(cmd => cmd.name());
22
+ console.error(`\n${chalk.red.bold("Error:")} Unknown command ${chalk.yellow(unknownCommand)}`);
23
+ // Simple "did you mean" logic
24
+ const suggestion = availableCommands.find(c => {
25
+ // Basic prefix or substring match for simplicity
26
+ return c.startsWith(unknownCommand.slice(0, 3)) || unknownCommand.includes(c);
27
+ });
28
+ if (suggestion) {
29
+ console.log(`Did you mean ${chalk.green(suggestion)}?`);
30
+ }
31
+ console.log(`Run ${chalk.cyan("git-compass --help")} to see all available commands.\n`);
32
+ process.exit(1);
33
+ });
34
+ export function run() {
35
+ try {
36
+ program.parse(process.argv);
37
+ // Show help if no arguments provided
38
+ if (!process.argv.slice(2).length) {
39
+ program.outputHelp();
40
+ }
41
+ }
42
+ catch (err) {
43
+ console.error(`\n${chalk.red.bold("Fatal Error:")} ${err.message}`);
44
+ process.exit(1);
45
+ }
46
+ }
47
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AACvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,2BAA2B,CAAC;AAC9D,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,KAAK,MAAM,OAAO,CAAC;AAE1B,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,aAAa,CAAC;KACnB,WAAW,CAAC,6CAA6C,CAAC;KAC1D,OAAO,CAAC,OAAO,CAAC,CAAC;AAEpB,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;AAClC,OAAO,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;AACnC,OAAO,CAAC,UAAU,CAAC,iBAAiB,CAAC,CAAC;AACtC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;AACjC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;AAEjC,qCAAqC;AACrC,OAAO,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,EAAE;IAC/B,MAAM,cAAc,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;IAC/B,MAAM,iBAAiB,GAAG,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC;IAElE,OAAO,CAAC,KAAK,CACX,KAAK,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,oBAAoB,KAAK,CAAC,MAAM,CAAC,cAAc,CAAC,EAAE,CAChF,CAAC;IAEF,8BAA8B;IAC9B,MAAM,UAAU,GAAG,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE;QAC5C,iDAAiD;QACjD,OAAO,CAAC,CAAC,UAAU,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,cAAc,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;IAChF,CAAC,CAAC,CAAC;IAEH,IAAI,UAAU,EAAE,CAAC;QACf,OAAO,CAAC,GAAG,CAAC,gBAAgB,KAAK,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;IAC1D,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,OAAO,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,mCAAmC,CAAC,CAAC;IACxF,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC;AAEH,MAAM,UAAU,GAAG;IACjB,IAAI,CAAC;QACH,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAE5B,qCAAqC;QACrC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;YAClC,OAAO,CAAC,UAAU,EAAE,CAAC;QACvB,CAAC;IACH,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,KAAK,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,cAAc,CAAC,IAAK,GAAa,CAAC,OAAO,EAAE,CAAC,CAAC;QAC/E,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=cache.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cache.test.d.ts","sourceRoot":"","sources":["../../../src/utils/__tests__/cache.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,38 @@
1
+ import { describe, it, expect, vi, beforeEach } from "vitest";
2
+ import { getCachePath, loadCache, getCachedResult, updateCache } from "../cache.js";
3
+ import fs from "fs/promises";
4
+ import path from "path";
5
+ vi.mock("fs/promises");
6
+ describe("Cache Utils", () => {
7
+ beforeEach(() => {
8
+ vi.resetAllMocks();
9
+ });
10
+ it("should generate correct cache path", async () => {
11
+ const repoRoot = "/test/repo";
12
+ const cachePath = await getCachePath(repoRoot);
13
+ expect(cachePath).toBe(path.join(repoRoot, ".git-compass", "cache.json"));
14
+ });
15
+ it("should load empty cache if file does not exist", async () => {
16
+ vi.mocked(fs.readFile).mockRejectedValue(new Error("File not found"));
17
+ const cache = await loadCache("/path/to/cache.json");
18
+ expect(cache).toEqual({});
19
+ });
20
+ it("should retrieve cached result", () => {
21
+ const cache = {
22
+ "repo": {
23
+ "hash1": { data: "test" }
24
+ }
25
+ };
26
+ const result = getCachedResult(cache, "/test/repo", "hash1");
27
+ expect(result).toEqual({ data: "test" });
28
+ });
29
+ it("should update cache accurately", () => {
30
+ const cache = {};
31
+ const updated = updateCache(cache, "/test/repo", "hash1", { some: "data" });
32
+ expect(updated["repo"]).toBeDefined();
33
+ if (updated["repo"]) {
34
+ expect(updated["repo"]["hash1"]).toEqual({ some: "data" });
35
+ }
36
+ });
37
+ });
38
+ //# sourceMappingURL=cache.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cache.test.js","sourceRoot":"","sources":["../../../src/utils/__tests__/cache.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AAC9D,OAAO,EACL,YAAY,EACZ,SAAS,EACT,eAAe,EACf,WAAW,EACZ,MAAM,aAAa,CAAC;AACrB,OAAO,EAAE,MAAM,aAAa,CAAC;AAC7B,OAAO,IAAI,MAAM,MAAM,CAAC;AAExB,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;AAEvB,QAAQ,CAAC,aAAa,EAAE,GAAG,EAAE;IAC3B,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,MAAM,QAAQ,GAAG,YAAY,CAAC;QAC9B,MAAM,SAAS,GAAG,MAAM,YAAY,CAAC,QAAQ,CAAC,CAAC;QAC/C,MAAM,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,cAAc,EAAE,YAAY,CAAC,CAAC,CAAC;IAC5E,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK,IAAI,EAAE;QAC9D,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,iBAAiB,CAAC,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC;QACtE,MAAM,KAAK,GAAG,MAAM,SAAS,CAAC,qBAAqB,CAAC,CAAC;QACrD,MAAM,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IAC5B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+BAA+B,EAAE,GAAG,EAAE;QACvC,MAAM,KAAK,GAAG;YACZ,MAAM,EAAE;gBACN,OAAO,EAAE,EAAE,IAAI,EAAE,MAAM,EAAS;aACjC;SACF,CAAC;QACF,MAAM,MAAM,GAAG,eAAe,CAAC,KAAK,EAAE,YAAY,EAAE,OAAO,CAAC,CAAC;QAC7D,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;QACxC,MAAM,KAAK,GAAG,EAAE,CAAC;QACjB,MAAM,OAAO,GAAG,WAAW,CAAC,KAAK,EAAE,YAAY,EAAE,OAAO,EAAE,EAAE,IAAI,EAAE,MAAM,EAAS,CAAC,CAAC;QACnF,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;QACtC,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YACpB,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;QAC7D,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { AnalysisResult } from "@git-compass/core";
2
+ export interface CacheData {
3
+ [repoKey: string]: {
4
+ [commitHash: string]: AnalysisResult;
5
+ };
6
+ }
7
+ export declare function getCachePath(repoRoot: string): Promise<string>;
8
+ export declare function loadCache(cachePath: string): Promise<CacheData>;
9
+ export declare function saveCache(cachePath: string, data: CacheData): Promise<void>;
10
+ export declare function getCachedResult(cache: CacheData, repoRoot: string, commitHash: string): AnalysisResult | null;
11
+ export declare function updateCache(cache: CacheData, repoRoot: string, commitHash: string, result: AnalysisResult): CacheData;
12
+ //# sourceMappingURL=cache.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cache.d.ts","sourceRoot":"","sources":["../../src/utils/cache.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAExD,MAAM,WAAW,SAAS;IACxB,CAAC,OAAO,EAAE,MAAM,GAAG;QACjB,CAAC,UAAU,EAAE,MAAM,GAAG,cAAc,CAAC;KACtC,CAAC;CACH;AAID,wBAAsB,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAEpE;AAED,wBAAsB,SAAS,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,SAAS,CAAC,CAOrE;AAED,wBAAsB,SAAS,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,GAAG,OAAO,CAAC,IAAI,CAAC,CAOjF;AAED,wBAAgB,eAAe,CAC7B,KAAK,EAAE,SAAS,EAChB,QAAQ,EAAE,MAAM,EAChB,UAAU,EAAE,MAAM,GACjB,cAAc,GAAG,IAAI,CAGvB;AAED,wBAAgB,WAAW,CACzB,KAAK,EAAE,SAAS,EAChB,QAAQ,EAAE,MAAM,EAChB,UAAU,EAAE,MAAM,EAClB,MAAM,EAAE,cAAc,GACrB,SAAS,CAoBX"}
@@ -0,0 +1,47 @@
1
+ import fs from "fs/promises";
2
+ import path from "path";
3
+ const MAX_CACHE_ENTRIES_PER_REPO = 3;
4
+ export async function getCachePath(repoRoot) {
5
+ return path.join(repoRoot, ".git-compass", "cache.json");
6
+ }
7
+ export async function loadCache(cachePath) {
8
+ try {
9
+ const data = await fs.readFile(cachePath, "utf-8");
10
+ return JSON.parse(data);
11
+ }
12
+ catch (err) {
13
+ return {};
14
+ }
15
+ }
16
+ export async function saveCache(cachePath, data) {
17
+ try {
18
+ await fs.mkdir(path.dirname(cachePath), { recursive: true });
19
+ await fs.writeFile(cachePath, JSON.stringify(data, null, 2), "utf-8");
20
+ }
21
+ catch (err) {
22
+ console.error("Failed to save cache:", err);
23
+ }
24
+ }
25
+ export function getCachedResult(cache, repoRoot, commitHash) {
26
+ const repoName = path.basename(repoRoot);
27
+ return cache[repoName]?.[commitHash] || null;
28
+ }
29
+ export function updateCache(cache, repoRoot, commitHash, result) {
30
+ const repoName = path.basename(repoRoot);
31
+ if (!cache[repoName]) {
32
+ cache[repoName] = {};
33
+ }
34
+ // Add new result
35
+ cache[repoName][commitHash] = result;
36
+ // Enforce size limit (keep only the most recent entries)
37
+ const hashes = Object.keys(cache[repoName]);
38
+ if (hashes.length > MAX_CACHE_ENTRIES_PER_REPO) {
39
+ // Basic strategy: Remove oldest added hash (first key in object)
40
+ // In a more advanced version we would use timestamps, but for "lightweight" this is fine.
41
+ const hashToRemove = hashes[0];
42
+ if (hashToRemove)
43
+ delete cache[repoName][hashToRemove];
44
+ }
45
+ return cache;
46
+ }
47
+ //# sourceMappingURL=cache.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cache.js","sourceRoot":"","sources":["../../src/utils/cache.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,aAAa,CAAC;AAC7B,OAAO,IAAI,MAAM,MAAM,CAAC;AASxB,MAAM,0BAA0B,GAAG,CAAC,CAAC;AAErC,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,QAAgB;IACjD,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,cAAc,EAAE,YAAY,CAAC,CAAC;AAC3D,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,SAAS,CAAC,SAAiB;IAC/C,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QACnD,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,SAAS,CAAC,SAAiB,EAAE,IAAe;IAChE,IAAI,CAAC;QACH,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC7D,MAAM,EAAE,CAAC,SAAS,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IACxE,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,uBAAuB,EAAE,GAAG,CAAC,CAAC;IAC9C,CAAC;AACH,CAAC;AAED,MAAM,UAAU,eAAe,CAC7B,KAAgB,EAChB,QAAgB,EAChB,UAAkB;IAElB,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IACzC,OAAO,KAAK,CAAC,QAAQ,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,IAAI,CAAC;AAC/C,CAAC;AAED,MAAM,UAAU,WAAW,CACzB,KAAgB,EAChB,QAAgB,EAChB,UAAkB,EAClB,MAAsB;IAEtB,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IAEzC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,EAAE,CAAC;QACrB,KAAK,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;IACvB,CAAC;IAED,iBAAiB;IACjB,KAAK,CAAC,QAAQ,CAAC,CAAC,UAAU,CAAC,GAAG,MAAM,CAAC;IAErC,yDAAyD;IACzD,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC;IAC5C,IAAI,MAAM,CAAC,MAAM,GAAG,0BAA0B,EAAE,CAAC;QAC/C,iEAAiE;QACjE,0FAA0F;QAC1F,MAAM,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QAC/B,IAAI,YAAY;YAAE,OAAO,KAAK,CAAC,QAAQ,CAAC,CAAC,YAAY,CAAC,CAAC;IACzD,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Ensures that specific entries are present in the .gitignore file at the repo root.
3
+ */
4
+ export declare function ensureGitIgnore(repoRoot: string, entries: string[]): Promise<void>;
5
+ //# sourceMappingURL=gitignore.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"gitignore.d.ts","sourceRoot":"","sources":["../../src/utils/gitignore.ts"],"names":[],"mappings":"AAGA;;GAEG;AACH,wBAAsB,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CA+BxF"}
@@ -0,0 +1,36 @@
1
+ import fs from "fs/promises";
2
+ import path from "path";
3
+ /**
4
+ * Ensures that specific entries are present in the .gitignore file at the repo root.
5
+ */
6
+ export async function ensureGitIgnore(repoRoot, entries) {
7
+ const gitIgnorePath = path.join(repoRoot, ".gitignore");
8
+ try {
9
+ let content = "";
10
+ try {
11
+ content = await fs.readFile(gitIgnorePath, "utf-8");
12
+ }
13
+ catch (err) {
14
+ // If .gitignore doesn't exist, we'll create it
15
+ }
16
+ const lines = content.split(/\r?\n/);
17
+ let modified = false;
18
+ for (const entry of entries) {
19
+ if (!lines.some(line => line.trim() === entry)) {
20
+ lines.push(entry);
21
+ modified = true;
22
+ }
23
+ }
24
+ if (modified) {
25
+ // Ensure there's a newline at the end if we added something
26
+ if (lines[lines.length - 1] !== "") {
27
+ lines.push("");
28
+ }
29
+ await fs.writeFile(gitIgnorePath, lines.join("\n"), "utf-8");
30
+ }
31
+ }
32
+ catch (err) {
33
+ console.warn(`Could not update .gitignore: ${err.message}`);
34
+ }
35
+ }
36
+ //# sourceMappingURL=gitignore.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"gitignore.js","sourceRoot":"","sources":["../../src/utils/gitignore.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,aAAa,CAAC;AAC7B,OAAO,IAAI,MAAM,MAAM,CAAC;AAExB;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,eAAe,CAAC,QAAgB,EAAE,OAAiB;IACvE,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;IAExD,IAAI,CAAC;QACH,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,IAAI,CAAC;YACH,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;QACtD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,+CAA+C;QACjD,CAAC;QAED,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QACrC,IAAI,QAAQ,GAAG,KAAK,CAAC;QAErB,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC5B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,KAAK,CAAC,EAAE,CAAC;gBAC/C,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAClB,QAAQ,GAAG,IAAI,CAAC;YAClB,CAAC;QACH,CAAC;QAED,IAAI,QAAQ,EAAE,CAAC;YACb,4DAA4D;YAC5D,IAAI,KAAK,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC;gBACnC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACjB,CAAC;YACD,MAAM,EAAE,CAAC,SAAS,CAAC,aAAa,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,OAAO,CAAC,CAAC;QAC/D,CAAC;IACH,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,IAAI,CAAC,gCAAiC,GAAa,CAAC,OAAO,EAAE,CAAC,CAAC;IACzE,CAAC;AACH,CAAC"}
package/package.json ADDED
@@ -0,0 +1,64 @@
1
+ {
2
+ "name": "git-compass",
3
+ "version": "0.2.0",
4
+ "description": "Git repository analytics from your terminal",
5
+ "keywords": [
6
+ "git",
7
+ "analytics",
8
+ "hotspots",
9
+ "churn",
10
+ "bus-factor",
11
+ "code-quality",
12
+ "git-history",
13
+ "ai",
14
+ "llm",
15
+ "anthropic",
16
+ "developer-tools"
17
+ ],
18
+ "type": "module",
19
+ "bin": {
20
+ "git-compass": "./dist/bin/git-compass.js"
21
+ },
22
+ "main": "./dist/index.js",
23
+ "license": "MIT",
24
+ "author": "git-compass",
25
+ "publishConfig": {
26
+ "access": "public"
27
+ },
28
+ "repository": {
29
+ "type": "git",
30
+ "url": "git+https://github.com/HUMBLEF0OL/git-compass.git",
31
+ "directory": "packages/cli"
32
+ },
33
+ "bugs": {
34
+ "url": "https://github.com/HUMBLEF0OL/git-compass/issues"
35
+ },
36
+ "homepage": "https://github.com/HUMBLEF0OL/git-compass/tree/main/packages/cli#readme",
37
+ "files": [
38
+ "dist",
39
+ "README.md"
40
+ ],
41
+ "dependencies": {
42
+ "@inquirer/prompts": "8.3.2",
43
+ "boxen": "^7.1.1",
44
+ "chalk": "^5.3.0",
45
+ "chokidar": "^3.6.0",
46
+ "commander": "^12.1.0",
47
+ "dotenv": "^16.4.5",
48
+ "inquirer": "13.3.2",
49
+ "ora": "^8.0.1",
50
+ "table": "^6.8.2",
51
+ "@git-compass/core": "0.1.0"
52
+ },
53
+ "devDependencies": {
54
+ "@types/node": "^20.14.0",
55
+ "tsx": "^4.21.0",
56
+ "typescript": "^5.4.5",
57
+ "vitest": "^1.6.0"
58
+ },
59
+ "scripts": {
60
+ "build": "tsc",
61
+ "dev": "tsc --watch",
62
+ "test": "vitest run"
63
+ }
64
+ }