@paulbeliavskis/git-grove 0.1.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 (108) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +90 -0
  3. package/bin/grove.js +2 -0
  4. package/dist/cli.d.ts +2 -0
  5. package/dist/cli.d.ts.map +1 -0
  6. package/dist/cli.js +111 -0
  7. package/dist/cli.js.map +1 -0
  8. package/dist/commands/cleanup.d.ts +4 -0
  9. package/dist/commands/cleanup.d.ts.map +1 -0
  10. package/dist/commands/cleanup.js +95 -0
  11. package/dist/commands/cleanup.js.map +1 -0
  12. package/dist/commands/code.d.ts +2 -0
  13. package/dist/commands/code.d.ts.map +1 -0
  14. package/dist/commands/code.js +59 -0
  15. package/dist/commands/code.js.map +1 -0
  16. package/dist/commands/init.d.ts +4 -0
  17. package/dist/commands/init.d.ts.map +1 -0
  18. package/dist/commands/init.js +135 -0
  19. package/dist/commands/init.js.map +1 -0
  20. package/dist/commands/list.d.ts +4 -0
  21. package/dist/commands/list.d.ts.map +1 -0
  22. package/dist/commands/list.js +42 -0
  23. package/dist/commands/list.js.map +1 -0
  24. package/dist/commands/path.d.ts +2 -0
  25. package/dist/commands/path.d.ts.map +1 -0
  26. package/dist/commands/path.js +36 -0
  27. package/dist/commands/path.js.map +1 -0
  28. package/dist/commands/remove.d.ts +2 -0
  29. package/dist/commands/remove.d.ts.map +1 -0
  30. package/dist/commands/remove.js +50 -0
  31. package/dist/commands/remove.js.map +1 -0
  32. package/dist/commands/repos.d.ts +2 -0
  33. package/dist/commands/repos.d.ts.map +1 -0
  34. package/dist/commands/repos.js +39 -0
  35. package/dist/commands/repos.js.map +1 -0
  36. package/dist/commands/review.d.ts +4 -0
  37. package/dist/commands/review.d.ts.map +1 -0
  38. package/dist/commands/review.js +96 -0
  39. package/dist/commands/review.js.map +1 -0
  40. package/dist/commands/setup.d.ts +2 -0
  41. package/dist/commands/setup.d.ts.map +1 -0
  42. package/dist/commands/setup.js +227 -0
  43. package/dist/commands/setup.js.map +1 -0
  44. package/dist/commands/shell-init.d.ts +2 -0
  45. package/dist/commands/shell-init.d.ts.map +1 -0
  46. package/dist/commands/shell-init.js +77 -0
  47. package/dist/commands/shell-init.js.map +1 -0
  48. package/dist/commands/sprint.d.ts +4 -0
  49. package/dist/commands/sprint.d.ts.map +1 -0
  50. package/dist/commands/sprint.js +237 -0
  51. package/dist/commands/sprint.js.map +1 -0
  52. package/dist/commands/status.d.ts +4 -0
  53. package/dist/commands/status.d.ts.map +1 -0
  54. package/dist/commands/status.js +47 -0
  55. package/dist/commands/status.js.map +1 -0
  56. package/dist/commands/work.d.ts +4 -0
  57. package/dist/commands/work.d.ts.map +1 -0
  58. package/dist/commands/work.js +104 -0
  59. package/dist/commands/work.js.map +1 -0
  60. package/dist/lib/ai.d.ts +14 -0
  61. package/dist/lib/ai.d.ts.map +1 -0
  62. package/dist/lib/ai.js +36 -0
  63. package/dist/lib/ai.js.map +1 -0
  64. package/dist/lib/config.d.ts +11 -0
  65. package/dist/lib/config.d.ts.map +1 -0
  66. package/dist/lib/config.js +44 -0
  67. package/dist/lib/config.js.map +1 -0
  68. package/dist/lib/credentials.d.ts +11 -0
  69. package/dist/lib/credentials.d.ts.map +1 -0
  70. package/dist/lib/credentials.js +151 -0
  71. package/dist/lib/credentials.js.map +1 -0
  72. package/dist/lib/git.d.ts +23 -0
  73. package/dist/lib/git.d.ts.map +1 -0
  74. package/dist/lib/git.js +136 -0
  75. package/dist/lib/git.js.map +1 -0
  76. package/dist/lib/hooks.d.ts +3 -0
  77. package/dist/lib/hooks.d.ts.map +1 -0
  78. package/dist/lib/hooks.js +44 -0
  79. package/dist/lib/hooks.js.map +1 -0
  80. package/dist/lib/jira.d.ts +21 -0
  81. package/dist/lib/jira.d.ts.map +1 -0
  82. package/dist/lib/jira.js +47 -0
  83. package/dist/lib/jira.js.map +1 -0
  84. package/dist/lib/platform.d.ts +4 -0
  85. package/dist/lib/platform.d.ts.map +1 -0
  86. package/dist/lib/platform.js +10 -0
  87. package/dist/lib/platform.js.map +1 -0
  88. package/dist/lib/providers/bitbucket.d.ts +3 -0
  89. package/dist/lib/providers/bitbucket.d.ts.map +1 -0
  90. package/dist/lib/providers/bitbucket.js +49 -0
  91. package/dist/lib/providers/bitbucket.js.map +1 -0
  92. package/dist/lib/providers/github.d.ts +3 -0
  93. package/dist/lib/providers/github.d.ts.map +1 -0
  94. package/dist/lib/providers/github.js +48 -0
  95. package/dist/lib/providers/github.js.map +1 -0
  96. package/dist/lib/providers/types.d.ts +3 -0
  97. package/dist/lib/providers/types.d.ts.map +1 -0
  98. package/dist/lib/providers/types.js +13 -0
  99. package/dist/lib/providers/types.js.map +1 -0
  100. package/dist/lib/types.d.ts +89 -0
  101. package/dist/lib/types.d.ts.map +1 -0
  102. package/dist/lib/types.js +2 -0
  103. package/dist/lib/types.js.map +1 -0
  104. package/dist/lib/ui.d.ts +14 -0
  105. package/dist/lib/ui.d.ts.map +1 -0
  106. package/dist/lib/ui.js +32 -0
  107. package/dist/lib/ui.js.map +1 -0
  108. package/package.json +56 -0
@@ -0,0 +1,36 @@
1
+ import { loadConfig } from '../lib/config.js';
2
+ import { listWorktrees } from '../lib/git.js';
3
+ export function pathCommand(name) {
4
+ const config = loadConfig();
5
+ const allWorktrees = [];
6
+ for (const repo of Object.values(config.repos)) {
7
+ const worktrees = listWorktrees(repo.gitRoot);
8
+ const nonMain = worktrees.filter((wt) => !wt.bare &&
9
+ wt.path.replace(/\\/g, '/') !== repo.gitRoot.replace(/\\/g, '/'));
10
+ allWorktrees.push(...nonMain);
11
+ }
12
+ const match = fuzzyMatch(allWorktrees, name);
13
+ if (!match) {
14
+ process.stderr.write(`No worktree matching "${name}" found.\n`);
15
+ process.exit(1);
16
+ }
17
+ // Output only the path to stdout (pipe-friendly)
18
+ process.stdout.write(match.path);
19
+ }
20
+ function fuzzyMatch(worktrees, query) {
21
+ const q = query.toLowerCase();
22
+ // Exact match
23
+ const exact = worktrees.find((wt) => wt.folderName.toLowerCase() === q);
24
+ if (exact)
25
+ return exact;
26
+ // Prefix match
27
+ const prefix = worktrees.find((wt) => wt.folderName.toLowerCase().startsWith(q));
28
+ if (prefix)
29
+ return prefix;
30
+ // Substring match
31
+ const substr = worktrees.find((wt) => wt.folderName.toLowerCase().includes(q));
32
+ if (substr)
33
+ return substr;
34
+ return undefined;
35
+ }
36
+ //# sourceMappingURL=path.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"path.js","sourceRoot":"","sources":["../../src/commands/path.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAE9C,MAAM,UAAU,WAAW,CAAC,IAAY;IACtC,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,YAAY,GAA2C,EAAE,CAAC;IAEhE,KAAK,MAAM,IAAI,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC;QAC/C,MAAM,SAAS,GAAG,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC9C,MAAM,OAAO,GAAG,SAAS,CAAC,MAAM,CAC9B,CAAC,EAAE,EAAE,EAAE,CACL,CAAC,EAAE,CAAC,IAAI;YACR,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,KAAK,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CACnE,CAAC;QACF,YAAY,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,CAAC;IAChC,CAAC;IAED,MAAM,KAAK,GAAG,UAAU,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC;IAE7C,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,yBAAyB,IAAI,YAAY,CAAC,CAAC;QAChE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,iDAAiD;IACjD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;AACnC,CAAC;AAED,SAAS,UAAU,CACjB,SAAiD,EACjD,KAAa;IAEb,MAAM,CAAC,GAAG,KAAK,CAAC,WAAW,EAAE,CAAC;IAE9B,cAAc;IACd,MAAM,KAAK,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,UAAU,CAAC,WAAW,EAAE,KAAK,CAAC,CAAC,CAAC;IACxE,IAAI,KAAK;QAAE,OAAO,KAAK,CAAC;IAExB,eAAe;IACf,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,CACnC,EAAE,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,UAAU,CAAC,CAAC,CAAC,CAC1C,CAAC;IACF,IAAI,MAAM;QAAE,OAAO,MAAM,CAAC;IAE1B,kBAAkB;IAClB,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,CACnC,EAAE,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,CACxC,CAAC;IACF,IAAI,MAAM;QAAE,OAAO,MAAM,CAAC;IAE1B,OAAO,SAAS,CAAC;AACnB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function removeCommand(name?: string): Promise<void>;
2
+ //# sourceMappingURL=remove.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"remove.d.ts","sourceRoot":"","sources":["../../src/commands/remove.ts"],"names":[],"mappings":"AAIA,wBAAsB,aAAa,CAAC,IAAI,CAAC,EAAE,MAAM,iBAmDhD"}
@@ -0,0 +1,50 @@
1
+ import inquirer from 'inquirer';
2
+ import { loadConfig, saveConfig } from '../lib/config.js';
3
+ import { success, warn, error } from '../lib/ui.js';
4
+ export async function removeCommand(name) {
5
+ const config = loadConfig();
6
+ const repos = Object.keys(config.repos);
7
+ if (repos.length === 0) {
8
+ warn('No repos registered.');
9
+ return;
10
+ }
11
+ let alias;
12
+ if (name) {
13
+ if (!config.repos[name]) {
14
+ error(`Repo "${name}" not found.`);
15
+ return;
16
+ }
17
+ alias = name;
18
+ }
19
+ else {
20
+ const { selected } = await inquirer.prompt([
21
+ {
22
+ type: 'list',
23
+ name: 'selected',
24
+ message: 'Select repo to remove:',
25
+ choices: repos,
26
+ },
27
+ ]);
28
+ alias = selected;
29
+ }
30
+ const { confirmed } = await inquirer.prompt([
31
+ {
32
+ type: 'confirm',
33
+ name: 'confirmed',
34
+ message: `Unregister "${alias}"? (worktrees on disk are not affected)`,
35
+ default: false,
36
+ },
37
+ ]);
38
+ if (!confirmed) {
39
+ warn('Cancelled.');
40
+ return;
41
+ }
42
+ delete config.repos[alias];
43
+ if (config.defaults.repo === alias) {
44
+ const remaining = Object.keys(config.repos);
45
+ config.defaults.repo = remaining[0] || undefined;
46
+ }
47
+ saveConfig(config);
48
+ success(`Repo "${alias}" removed from grove config.`);
49
+ }
50
+ //# sourceMappingURL=remove.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"remove.js","sourceRoot":"","sources":["../../src/commands/remove.ts"],"names":[],"mappings":"AAAA,OAAO,QAAQ,MAAM,UAAU,CAAC;AAChC,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC1D,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,cAAc,CAAC;AAEpD,MAAM,CAAC,KAAK,UAAU,aAAa,CAAC,IAAa;IAC/C,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IAExC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACvB,IAAI,CAAC,sBAAsB,CAAC,CAAC;QAC7B,OAAO;IACT,CAAC;IAED,IAAI,KAAa,CAAC;IAClB,IAAI,IAAI,EAAE,CAAC;QACT,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;YACxB,KAAK,CAAC,SAAS,IAAI,cAAc,CAAC,CAAC;YACnC,OAAO;QACT,CAAC;QACD,KAAK,GAAG,IAAI,CAAC;IACf,CAAC;SAAM,CAAC;QACN,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC;YACzC;gBACE,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE,UAAU;gBAChB,OAAO,EAAE,wBAAwB;gBACjC,OAAO,EAAE,KAAK;aACf;SACF,CAAC,CAAC;QACH,KAAK,GAAG,QAAQ,CAAC;IACnB,CAAC;IAED,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC;QAC1C;YACE,IAAI,EAAE,SAAS;YACf,IAAI,EAAE,WAAW;YACjB,OAAO,EAAE,eAAe,KAAK,yCAAyC;YACtE,OAAO,EAAE,KAAK;SACf;KACF,CAAC,CAAC;IAEH,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,IAAI,CAAC,YAAY,CAAC,CAAC;QACnB,OAAO;IACT,CAAC;IAED,OAAO,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IAE3B,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,KAAK,EAAE,CAAC;QACnC,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC5C,MAAM,CAAC,QAAQ,CAAC,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC;IACnD,CAAC;IAED,UAAU,CAAC,MAAM,CAAC,CAAC;IACnB,OAAO,CAAC,SAAS,KAAK,8BAA8B,CAAC,CAAC;AACxD,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function reposCommand(): void;
2
+ //# sourceMappingURL=repos.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"repos.d.ts","sourceRoot":"","sources":["../../src/commands/repos.ts"],"names":[],"mappings":"AAIA,wBAAgB,YAAY,SA4C3B"}
@@ -0,0 +1,39 @@
1
+ import { loadConfig } from '../lib/config.js';
2
+ import { listWorktrees } from '../lib/git.js';
3
+ import { createTable, warn } from '../lib/ui.js';
4
+ export function reposCommand() {
5
+ const config = loadConfig();
6
+ const repos = Object.entries(config.repos);
7
+ if (repos.length === 0) {
8
+ warn('No repos registered. Run `grove init` first.');
9
+ return;
10
+ }
11
+ const table = createTable({
12
+ head: ['Alias', 'Git Root', 'Worktrees', 'Provider', 'Remote'],
13
+ });
14
+ for (const [alias, repo] of repos) {
15
+ let worktreeCount;
16
+ try {
17
+ const wts = listWorktrees(repo.gitRoot);
18
+ worktreeCount = wts.filter((wt) => !wt.bare &&
19
+ wt.path.replace(/\\/g, '/') !== repo.gitRoot.replace(/\\/g, '/')).length;
20
+ }
21
+ catch {
22
+ worktreeCount = 0;
23
+ }
24
+ const provider = repo.provider;
25
+ const remote = provider.type === 'bitbucket'
26
+ ? `${provider.workspace}/${provider.repoSlug}`
27
+ : `${provider.owner}/${provider.repo}`;
28
+ const isDefault = config.defaults.repo === alias ? ' (default)' : '';
29
+ table.push([
30
+ `${alias}${isDefault}`,
31
+ repo.gitRoot,
32
+ String(worktreeCount),
33
+ provider.type,
34
+ remote,
35
+ ]);
36
+ }
37
+ console.log(table.toString());
38
+ }
39
+ //# sourceMappingURL=repos.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"repos.js","sourceRoot":"","sources":["../../src/commands/repos.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,WAAW,EAAE,IAAI,EAAE,MAAM,cAAc,CAAC;AAEjD,MAAM,UAAU,YAAY;IAC1B,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,KAAK,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IAE3C,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACvB,IAAI,CAAC,8CAA8C,CAAC,CAAC;QACrD,OAAO;IACT,CAAC;IAED,MAAM,KAAK,GAAG,WAAW,CAAC;QACxB,IAAI,EAAE,CAAC,OAAO,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,QAAQ,CAAC;KAC/D,CAAC,CAAC;IAEH,KAAK,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,KAAK,EAAE,CAAC;QAClC,IAAI,aAAqB,CAAC;QAC1B,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACxC,aAAa,GAAG,GAAG,CAAC,MAAM,CACxB,CAAC,EAAE,EAAE,EAAE,CACL,CAAC,EAAE,CAAC,IAAI;gBACR,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,KAAK,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CACnE,CAAC,MAAM,CAAC;QACX,CAAC;QAAC,MAAM,CAAC;YACP,aAAa,GAAG,CAAC,CAAC;QACpB,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;QAC/B,MAAM,MAAM,GACV,QAAQ,CAAC,IAAI,KAAK,WAAW;YAC3B,CAAC,CAAC,GAAG,QAAQ,CAAC,SAAS,IAAI,QAAQ,CAAC,QAAQ,EAAE;YAC9C,CAAC,CAAC,GAAG,QAAQ,CAAC,KAAK,IAAI,QAAQ,CAAC,IAAI,EAAE,CAAC;QAE3C,MAAM,SAAS,GAAG,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,CAAC;QAErE,KAAK,CAAC,IAAI,CAAC;YACT,GAAG,KAAK,GAAG,SAAS,EAAE;YACtB,IAAI,CAAC,OAAO;YACZ,MAAM,CAAC,aAAa,CAAC;YACrB,QAAQ,CAAC,IAAI;YACb,MAAM;SACP,CAAC,CAAC;IACL,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;AAChC,CAAC"}
@@ -0,0 +1,4 @@
1
+ export declare function reviewCommand(options: {
2
+ repo?: string;
3
+ }): Promise<void>;
4
+ //# sourceMappingURL=review.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review.d.ts","sourceRoot":"","sources":["../../src/commands/review.ts"],"names":[],"mappings":"AAOA,wBAAsB,aAAa,CAAC,OAAO,EAAE;IAAE,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,iBAkG7D"}
@@ -0,0 +1,96 @@
1
+ import inquirer from 'inquirer';
2
+ import { loadConfig, getRepoConfig } from '../lib/config.js';
3
+ import { createProvider } from '../lib/providers/types.js';
4
+ import { launchAiReview } from '../lib/ai.js';
5
+ import { createTable, info, error, ora, truncate } from '../lib/ui.js';
6
+ import { workCommand } from './work.js';
7
+ export async function reviewCommand(options) {
8
+ const config = loadConfig();
9
+ const { name, repo } = getRepoConfig(config, options.repo);
10
+ info(`Fetching PRs for ${name} (${repo.provider.type})...`);
11
+ const spinner = ora('Loading pull requests...').start();
12
+ let prs;
13
+ try {
14
+ const provider = createProvider(repo.provider);
15
+ prs = await provider.listPRs();
16
+ spinner.succeed(`Found ${prs.length} open PR(s)`);
17
+ }
18
+ catch (err) {
19
+ spinner.fail('Failed to fetch PRs');
20
+ error(err.message);
21
+ return;
22
+ }
23
+ if (prs.length === 0) {
24
+ info('No open PRs matching target branches.');
25
+ return;
26
+ }
27
+ // Display PR table
28
+ const table = createTable({
29
+ head: ['PR#', 'Title', 'Branch', 'Target'],
30
+ });
31
+ for (const pr of prs) {
32
+ table.push([
33
+ String(pr.id),
34
+ truncate(pr.title, 60),
35
+ pr.sourceBranch,
36
+ pr.targetBranch,
37
+ ]);
38
+ }
39
+ console.log(table.toString());
40
+ // Select PR
41
+ const { selectedPr } = await inquirer.prompt([
42
+ {
43
+ type: 'list',
44
+ name: 'selectedPr',
45
+ message: 'Select a PR:',
46
+ choices: [
47
+ ...prs.map((pr) => ({
48
+ name: `#${pr.id} - ${truncate(pr.title, 50)} (${pr.sourceBranch})`,
49
+ value: pr,
50
+ })),
51
+ new inquirer.Separator(),
52
+ { name: 'Cancel', value: null },
53
+ ],
54
+ },
55
+ ]);
56
+ if (!selectedPr)
57
+ return;
58
+ // Choose action - only show AI options if configured
59
+ const hasAi = !!config.ai?.command;
60
+ const choices = [
61
+ { name: 'Create worktree only', value: 'worktree' },
62
+ ...(hasAi
63
+ ? [
64
+ { name: 'Create worktree + AI review', value: 'both' },
65
+ { name: 'AI review only (no worktree)', value: 'ai' },
66
+ ]
67
+ : []),
68
+ ];
69
+ const { action } = await inquirer.prompt([
70
+ {
71
+ type: 'list',
72
+ name: 'action',
73
+ message: 'What would you like to do?',
74
+ choices,
75
+ },
76
+ ]);
77
+ let worktreePath = null;
78
+ if (action === 'worktree' || action === 'both') {
79
+ worktreePath = await workCommand(selectedPr.sourceBranch, { repo: name });
80
+ }
81
+ if (action === 'ai' || action === 'both') {
82
+ const cwd = worktreePath || repo.gitRoot;
83
+ info(`Launching AI review in ${cwd}...`);
84
+ try {
85
+ launchAiReview(config.ai, repo.provider, {
86
+ prId: selectedPr.id,
87
+ branch: selectedPr.sourceBranch,
88
+ worktreePath: cwd,
89
+ }, cwd);
90
+ }
91
+ catch (err) {
92
+ error(`AI review failed: ${err.message}`);
93
+ }
94
+ }
95
+ }
96
+ //# sourceMappingURL=review.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review.js","sourceRoot":"","sources":["../../src/commands/review.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function setupCommand(): Promise<void>;
2
+ //# sourceMappingURL=setup.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.d.ts","sourceRoot":"","sources":["../../src/commands/setup.ts"],"names":[],"mappings":"AAUA,wBAAsB,YAAY,kBA+HjC"}
@@ -0,0 +1,227 @@
1
+ import inquirer from 'inquirer';
2
+ import { loadConfig, saveConfig, configExists, getConfigPath } from '../lib/config.js';
3
+ import { detectBitbucketCredentials, detectGitHubCredentials, isCommandAvailable, } from '../lib/credentials.js';
4
+ import { chalk, info, success } from '../lib/ui.js';
5
+ export async function setupCommand() {
6
+ const config = configExists() ? loadConfig() : { version: 1, repos: {}, defaults: {} };
7
+ const configPath = getConfigPath();
8
+ // --- Print current state ---
9
+ console.log(chalk.bold('\nGrove Setup\n'));
10
+ // Config file
11
+ console.log(chalk.bold('Config'));
12
+ console.log(` Path: ${configPath}`);
13
+ console.log(` Exists: ${configExists() ? chalk.green('yes') : chalk.yellow('no')}`);
14
+ console.log('');
15
+ // Repos
16
+ console.log(chalk.bold('Repos'));
17
+ const repos = Object.entries(config.repos);
18
+ if (repos.length === 0) {
19
+ console.log(chalk.yellow(' None registered. Run `grove init` in a git repo.'));
20
+ }
21
+ else {
22
+ for (const [alias, repo] of repos) {
23
+ const isDefault = config.defaults.repo === alias ? chalk.dim(' (default)') : '';
24
+ const provider = repo.provider;
25
+ const remote = provider.type === 'bitbucket'
26
+ ? `${provider.workspace}/${provider.repoSlug}`
27
+ : `${provider.owner}/${provider.repo}`;
28
+ console.log(` ${chalk.green(alias)}${isDefault} - ${provider.type} (${remote})`);
29
+ }
30
+ }
31
+ console.log('');
32
+ // Credentials - only show for providers in use
33
+ const providers = new Set(repos.map(([, r]) => r.provider.type));
34
+ if (providers.size > 0) {
35
+ console.log(chalk.bold('Credentials'));
36
+ if (providers.has('bitbucket')) {
37
+ const bbCreds = detectBitbucketCredentials();
38
+ if (bbCreds) {
39
+ console.log(` Bitbucket: ${chalk.green('found')} via ${bbCreds.source}`);
40
+ }
41
+ else {
42
+ console.log(` Bitbucket: ${chalk.yellow('not found')}`);
43
+ }
44
+ }
45
+ if (providers.has('github')) {
46
+ const ghCreds = detectGitHubCredentials();
47
+ if (ghCreds) {
48
+ console.log(` GitHub: ${chalk.green('found')} via ${ghCreds.source}`);
49
+ }
50
+ else {
51
+ console.log(` GitHub: ${chalk.yellow('not found')}`);
52
+ }
53
+ }
54
+ console.log('');
55
+ }
56
+ // Jira
57
+ console.log(chalk.bold('Jira'));
58
+ if (config.jira?.host && config.jira.email && config.jira.token) {
59
+ console.log(` Host: ${chalk.green(config.jira.host)}`);
60
+ console.log(` Email: ${config.jira.email}`);
61
+ console.log(` Token: ${chalk.dim('****' + config.jira.token.slice(-4))}`);
62
+ if (config.jira.boardId) {
63
+ console.log(` Board: id ${config.jira.boardId}`);
64
+ }
65
+ }
66
+ else {
67
+ console.log(chalk.yellow(' Not configured. Run `grove sprint` or configure here.'));
68
+ }
69
+ console.log('');
70
+ // AI tool
71
+ console.log(chalk.bold('AI Tool'));
72
+ if (config.ai?.command) {
73
+ const available = isCommandAvailable(config.ai.command);
74
+ console.log(` Command: ${config.ai.command} ${available ? chalk.green('(installed)') : chalk.red('(not found in PATH)')}`);
75
+ console.log(` Prompt: ${config.ai.reviewPrompt.slice(0, 80)}${config.ai.reviewPrompt.length > 80 ? '...' : ''}`);
76
+ if (config.ai.args.length > 0) {
77
+ console.log(` Args: ${config.ai.args.join(' ')}`);
78
+ }
79
+ }
80
+ else {
81
+ console.log(chalk.yellow(' Not configured. AI review features are disabled.'));
82
+ }
83
+ console.log('');
84
+ // Shell integration
85
+ console.log(chalk.bold('Shell'));
86
+ const groveAvailable = isCommandAvailable('grove');
87
+ console.log(` grove CLI: ${groveAvailable ? chalk.green('in PATH') : chalk.yellow('not in PATH (run npm link)')}`);
88
+ console.log('');
89
+ // --- Menu ---
90
+ const { action } = await inquirer.prompt([
91
+ {
92
+ type: 'list',
93
+ name: 'action',
94
+ message: 'What would you like to configure?',
95
+ choices: [
96
+ { name: 'Jira credentials', value: 'jira' },
97
+ { name: 'AI tool', value: 'ai' },
98
+ { name: 'Default repo', value: 'default', disabled: repos.length === 0 ? 'no repos registered' : false },
99
+ { name: 'Done', value: 'done' },
100
+ ],
101
+ },
102
+ ]);
103
+ if (action === 'done')
104
+ return;
105
+ if (action === 'jira') {
106
+ await configureJira(config);
107
+ }
108
+ if (action === 'ai') {
109
+ await configureAi(config);
110
+ }
111
+ if (action === 'default') {
112
+ const { repo } = await inquirer.prompt([
113
+ {
114
+ type: 'list',
115
+ name: 'repo',
116
+ message: 'Default repo:',
117
+ choices: Object.keys(config.repos),
118
+ default: config.defaults.repo,
119
+ },
120
+ ]);
121
+ config.defaults.repo = repo;
122
+ saveConfig(config);
123
+ success(`Default repo set to "${repo}".`);
124
+ }
125
+ }
126
+ async function configureJira(config) {
127
+ info('Generate an API token at: https://id.atlassian.com/manage-profile/security/api-tokens');
128
+ console.log('');
129
+ const hasToken = !!config.jira?.token;
130
+ const answers = await inquirer.prompt([
131
+ {
132
+ type: 'input',
133
+ name: 'host',
134
+ message: 'Jira Cloud host (e.g. yoursite.atlassian.net):',
135
+ default: config.jira?.host,
136
+ validate: (v) => v.includes('.') || 'Enter a valid hostname',
137
+ },
138
+ {
139
+ type: 'input',
140
+ name: 'email',
141
+ message: 'Atlassian account email:',
142
+ default: config.jira?.email,
143
+ validate: (v) => v.includes('@') || 'Enter a valid email',
144
+ },
145
+ {
146
+ type: 'password',
147
+ name: 'token',
148
+ message: hasToken
149
+ ? 'Atlassian API token (leave blank to keep existing):'
150
+ : 'Atlassian API token:',
151
+ mask: '*',
152
+ validate: (v) => v.length > 0 || hasToken || 'Token is required',
153
+ },
154
+ ]);
155
+ config.jira = {
156
+ host: answers.host,
157
+ email: answers.email,
158
+ token: answers.token || config.jira?.token || '',
159
+ boardId: config.jira?.boardId,
160
+ };
161
+ saveConfig(config);
162
+ success('Jira credentials saved.');
163
+ }
164
+ async function configureAi(config) {
165
+ // Detect available AI tools
166
+ const tools = ['claude', 'cursor', 'aider', 'copilot', 'gh'];
167
+ const detected = tools.filter(isCommandAvailable);
168
+ const currentCommand = config.ai?.command || '';
169
+ const { command } = await inquirer.prompt([
170
+ {
171
+ type: 'list',
172
+ name: 'command',
173
+ message: 'AI CLI tool:',
174
+ choices: [
175
+ ...detected.map((t) => ({
176
+ name: `${t} (detected)`,
177
+ value: t,
178
+ })),
179
+ new inquirer.Separator(),
180
+ { name: 'Other (enter manually)', value: '__other__' },
181
+ ...(config.ai ? [{ name: 'Remove AI config', value: '__remove__' }] : []),
182
+ ],
183
+ default: detected.includes(currentCommand) ? currentCommand : undefined,
184
+ },
185
+ ]);
186
+ if (command === '__remove__') {
187
+ delete config.ai;
188
+ saveConfig(config);
189
+ success('AI config removed.');
190
+ return;
191
+ }
192
+ let finalCommand = command;
193
+ if (command === '__other__') {
194
+ const { custom } = await inquirer.prompt([
195
+ { type: 'input', name: 'custom', message: 'AI CLI command:' },
196
+ ]);
197
+ finalCommand = custom;
198
+ }
199
+ // Determine default prompt based on what repos use
200
+ const providers = new Set(Object.values(config.repos).map((r) => r.provider.type));
201
+ let defaultPrompt = config.ai?.reviewPrompt || '';
202
+ if (!defaultPrompt) {
203
+ if (providers.has('bitbucket')) {
204
+ defaultPrompt = 'Review PR #{prId} in {repoSlug}. The source branch is {branch} in workspace {workspace}. Provide a thorough code review analyzing code quality, potential bugs, and suggestions.';
205
+ }
206
+ else {
207
+ defaultPrompt = 'Review PR #{prId} in {repoSlug}. The source branch is {branch}. Provide a thorough code review analyzing code quality, potential bugs, and suggestions.';
208
+ }
209
+ }
210
+ const { reviewPrompt } = await inquirer.prompt([
211
+ {
212
+ type: 'input',
213
+ name: 'reviewPrompt',
214
+ message: 'Review prompt template:',
215
+ default: defaultPrompt,
216
+ },
217
+ ]);
218
+ const ai = {
219
+ command: finalCommand,
220
+ reviewPrompt,
221
+ args: config.ai?.args || [],
222
+ };
223
+ config.ai = ai;
224
+ saveConfig(config);
225
+ success(`AI tool configured: ${finalCommand}`);
226
+ }
227
+ //# sourceMappingURL=setup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.js","sourceRoot":"","sources":["../../src/commands/setup.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function shellInitCommand(shell?: string): void;
2
+ //# sourceMappingURL=shell-init.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shell-init.d.ts","sourceRoot":"","sources":["../../src/commands/shell-init.ts"],"names":[],"mappings":"AAOA,wBAAgB,gBAAgB,CAAC,KAAK,CAAC,EAAE,MAAM,QAiB9C"}
@@ -0,0 +1,77 @@
1
+ import { dirname } from 'path';
2
+ import { fileURLToPath } from 'url';
3
+ import { error } from '../lib/ui.js';
4
+ const __dirname = dirname(fileURLToPath(import.meta.url));
5
+ export function shellInitCommand(shell) {
6
+ const detected = shell || detectShell();
7
+ switch (detected) {
8
+ case 'powershell':
9
+ case 'pwsh':
10
+ console.log(getPowerShellInit());
11
+ break;
12
+ case 'bash':
13
+ case 'zsh':
14
+ case 'sh':
15
+ console.log(getBashInit());
16
+ break;
17
+ default:
18
+ error(`Unsupported shell: ${detected}. Use 'powershell' or 'bash'.`);
19
+ process.exit(1);
20
+ }
21
+ }
22
+ function detectShell() {
23
+ const shell = process.env.SHELL || process.env.PSModulePath ? 'powershell' : 'bash';
24
+ return shell;
25
+ }
26
+ function getPowerShellInit() {
27
+ return `
28
+ function gw {
29
+ param([string]$Name)
30
+ if (-not $Name) {
31
+ grove list
32
+ return
33
+ }
34
+ $path = grove path $Name 2>$null
35
+ if ($LASTEXITCODE -eq 0 -and $path) {
36
+ Set-Location $path
37
+ } else {
38
+ Write-Host "Worktree '$Name' not found." -ForegroundColor Red
39
+ }
40
+ }
41
+
42
+ Register-ArgumentCompleter -CommandName gw -ParameterName Name -ScriptBlock {
43
+ param($commandName, $parameterName, $wordToComplete, $commandAst, $fakeBoundParameters)
44
+ $names = grove list --names-only 2>$null
45
+ if ($names) {
46
+ $names -split "\\r?\\n" | Where-Object { $_ -like "$wordToComplete*" } | ForEach-Object {
47
+ [System.Management.Automation.CompletionResult]::new($_, $_, 'ParameterValue', $_)
48
+ }
49
+ }
50
+ }
51
+ `.trim();
52
+ }
53
+ function getBashInit() {
54
+ return `
55
+ gw() {
56
+ if [ -z "$1" ]; then
57
+ grove list
58
+ return
59
+ fi
60
+ local path
61
+ path=$(grove path "$1" 2>/dev/null)
62
+ if [ $? -eq 0 ] && [ -n "$path" ]; then
63
+ cd "$path"
64
+ else
65
+ echo "Worktree '$1' not found." >&2
66
+ fi
67
+ }
68
+
69
+ _gw_completions() {
70
+ local names
71
+ names=$(grove list --names-only 2>/dev/null)
72
+ COMPREPLY=($(compgen -W "$names" -- "\${COMP_WORDS[COMP_CWORD]}"))
73
+ }
74
+ complete -F _gw_completions gw
75
+ `.trim();
76
+ }
77
+ //# sourceMappingURL=shell-init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shell-init.js","sourceRoot":"","sources":["../../src/commands/shell-init.ts"],"names":[],"mappings":"AACA,OAAO,EAAQ,OAAO,EAAE,MAAM,MAAM,CAAC;AACrC,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAC;AACpC,OAAO,EAAE,KAAK,EAAE,MAAM,cAAc,CAAC;AAErC,MAAM,SAAS,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;AAE1D,MAAM,UAAU,gBAAgB,CAAC,KAAc;IAC7C,MAAM,QAAQ,GAAG,KAAK,IAAI,WAAW,EAAE,CAAC;IAExC,QAAQ,QAAQ,EAAE,CAAC;QACjB,KAAK,YAAY,CAAC;QAClB,KAAK,MAAM;YACT,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,CAAC,CAAC;YACjC,MAAM;QACR,KAAK,MAAM,CAAC;QACZ,KAAK,KAAK,CAAC;QACX,KAAK,IAAI;YACP,OAAO,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;YAC3B,MAAM;QACR;YACE,KAAK,CAAC,sBAAsB,QAAQ,+BAA+B,CAAC,CAAC;YACrE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACpB,CAAC;AACH,CAAC;AAED,SAAS,WAAW;IAClB,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,KAAK,IAAI,OAAO,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,MAAM,CAAC;IACpF,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAS,iBAAiB;IACxB,OAAO;;;;;;;;;;;;;;;;;;;;;;;;CAwBR,CAAC,IAAI,EAAE,CAAC;AACT,CAAC;AAED,SAAS,WAAW;IAClB,OAAO;;;;;;;;;;;;;;;;;;;;;CAqBR,CAAC,IAAI,EAAE,CAAC;AACT,CAAC"}
@@ -0,0 +1,4 @@
1
+ export declare function sprintCommand(options: {
2
+ repo?: string;
3
+ }): Promise<string | null>;
4
+ //# sourceMappingURL=sprint.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sprint.d.ts","sourceRoot":"","sources":["../../src/commands/sprint.ts"],"names":[],"mappings":"AA+DA,wBAAsB,aAAa,CAAC,OAAO,EAAE;IAAE,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAyMtF"}