ankui 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 (270) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +369 -0
  3. package/dist/cli.d.ts +2 -0
  4. package/dist/cli.js +192 -0
  5. package/dist/cli.js.map +1 -0
  6. package/dist/commands/access.d.ts +6 -0
  7. package/dist/commands/access.js +12 -0
  8. package/dist/commands/access.js.map +1 -0
  9. package/dist/commands/caps.d.ts +6 -0
  10. package/dist/commands/caps.js +12 -0
  11. package/dist/commands/caps.js.map +1 -0
  12. package/dist/commands/discover.d.ts +10 -0
  13. package/dist/commands/discover.js +120 -0
  14. package/dist/commands/discover.js.map +1 -0
  15. package/dist/commands/doctor.d.ts +6 -0
  16. package/dist/commands/doctor.js +12 -0
  17. package/dist/commands/doctor.js.map +1 -0
  18. package/dist/commands/launch-tui.d.ts +15 -0
  19. package/dist/commands/launch-tui.js +53 -0
  20. package/dist/commands/launch-tui.js.map +1 -0
  21. package/dist/commands/list.d.ts +11 -0
  22. package/dist/commands/list.js +51 -0
  23. package/dist/commands/list.js.map +1 -0
  24. package/dist/commands/mcp.d.ts +6 -0
  25. package/dist/commands/mcp.js +12 -0
  26. package/dist/commands/mcp.js.map +1 -0
  27. package/dist/commands/scan-all.d.ts +15 -0
  28. package/dist/commands/scan-all.js +32 -0
  29. package/dist/commands/scan-all.js.map +1 -0
  30. package/dist/commands/show.d.ts +7 -0
  31. package/dist/commands/show.js +12 -0
  32. package/dist/commands/show.js.map +1 -0
  33. package/dist/commands/watch.d.ts +32 -0
  34. package/dist/commands/watch.js +205 -0
  35. package/dist/commands/watch.js.map +1 -0
  36. package/dist/config/ankui-config.d.ts +13 -0
  37. package/dist/config/ankui-config.js +123 -0
  38. package/dist/config/ankui-config.js.map +1 -0
  39. package/dist/scanner/access-review.d.ts +2 -0
  40. package/dist/scanner/access-review.js +183 -0
  41. package/dist/scanner/access-review.js.map +1 -0
  42. package/dist/scanner/adapters/claude.d.ts +2 -0
  43. package/dist/scanner/adapters/claude.js +298 -0
  44. package/dist/scanner/adapters/claude.js.map +1 -0
  45. package/dist/scanner/adapters/codex.d.ts +2 -0
  46. package/dist/scanner/adapters/codex.js +152 -0
  47. package/dist/scanner/adapters/codex.js.map +1 -0
  48. package/dist/scanner/adapters/cursor.d.ts +2 -0
  49. package/dist/scanner/adapters/cursor.js +114 -0
  50. package/dist/scanner/adapters/cursor.js.map +1 -0
  51. package/dist/scanner/adapters/gemini.d.ts +2 -0
  52. package/dist/scanner/adapters/gemini.js +191 -0
  53. package/dist/scanner/adapters/gemini.js.map +1 -0
  54. package/dist/scanner/adapters/index.d.ts +35 -0
  55. package/dist/scanner/adapters/index.js +85 -0
  56. package/dist/scanner/adapters/index.js.map +1 -0
  57. package/dist/scanner/adapters/opencode.d.ts +2 -0
  58. package/dist/scanner/adapters/opencode.js +231 -0
  59. package/dist/scanner/adapters/opencode.js.map +1 -0
  60. package/dist/scanner/adapters/shared.d.ts +57 -0
  61. package/dist/scanner/adapters/shared.js +239 -0
  62. package/dist/scanner/adapters/shared.js.map +1 -0
  63. package/dist/scanner/adapters/skills-sh.d.ts +2 -0
  64. package/dist/scanner/adapters/skills-sh.js +71 -0
  65. package/dist/scanner/adapters/skills-sh.js.map +1 -0
  66. package/dist/scanner/capability-map.d.ts +8 -0
  67. package/dist/scanner/capability-map.js +57 -0
  68. package/dist/scanner/capability-map.js.map +1 -0
  69. package/dist/scanner/discovery.d.ts +31 -0
  70. package/dist/scanner/discovery.js +320 -0
  71. package/dist/scanner/discovery.js.map +1 -0
  72. package/dist/scanner/filesystem-crawler.d.ts +32 -0
  73. package/dist/scanner/filesystem-crawler.js +210 -0
  74. package/dist/scanner/filesystem-crawler.js.map +1 -0
  75. package/dist/scanner/index.d.ts +8 -0
  76. package/dist/scanner/index.js +120 -0
  77. package/dist/scanner/index.js.map +1 -0
  78. package/dist/scanner/multi-project.d.ts +31 -0
  79. package/dist/scanner/multi-project.js +227 -0
  80. package/dist/scanner/multi-project.js.map +1 -0
  81. package/dist/scanner/parallel.d.ts +4 -0
  82. package/dist/scanner/parallel.js +41 -0
  83. package/dist/scanner/parallel.js.map +1 -0
  84. package/dist/scanner/parsing.d.ts +9 -0
  85. package/dist/scanner/parsing.js +135 -0
  86. package/dist/scanner/parsing.js.map +1 -0
  87. package/dist/scanner/paths.d.ts +8 -0
  88. package/dist/scanner/paths.js +39 -0
  89. package/dist/scanner/paths.js.map +1 -0
  90. package/dist/scanner/preview.d.ts +12 -0
  91. package/dist/scanner/preview.js +29 -0
  92. package/dist/scanner/preview.js.map +1 -0
  93. package/dist/scanner/project-discovery.d.ts +11 -0
  94. package/dist/scanner/project-discovery.js +35 -0
  95. package/dist/scanner/project-discovery.js.map +1 -0
  96. package/dist/scanner/ripgrep.d.ts +11 -0
  97. package/dist/scanner/ripgrep.js +61 -0
  98. package/dist/scanner/ripgrep.js.map +1 -0
  99. package/dist/scanner/safety.d.ts +37 -0
  100. package/dist/scanner/safety.js +330 -0
  101. package/dist/scanner/safety.js.map +1 -0
  102. package/dist/scanner/skill-naming.d.ts +5 -0
  103. package/dist/scanner/skill-naming.js +53 -0
  104. package/dist/scanner/skill-naming.js.map +1 -0
  105. package/dist/scanner/watcher.d.ts +15 -0
  106. package/dist/scanner/watcher.js +71 -0
  107. package/dist/scanner/watcher.js.map +1 -0
  108. package/dist/tui/App.d.ts +33 -0
  109. package/dist/tui/App.js +201 -0
  110. package/dist/tui/App.js.map +1 -0
  111. package/dist/tui/components/Breadcrumb.d.ts +10 -0
  112. package/dist/tui/components/Breadcrumb.js +11 -0
  113. package/dist/tui/components/Breadcrumb.js.map +1 -0
  114. package/dist/tui/components/DisclosureRow.d.ts +17 -0
  115. package/dist/tui/components/DisclosureRow.js +13 -0
  116. package/dist/tui/components/DisclosureRow.js.map +1 -0
  117. package/dist/tui/components/DotLeaderRow.d.ts +17 -0
  118. package/dist/tui/components/DotLeaderRow.js +34 -0
  119. package/dist/tui/components/DotLeaderRow.js.map +1 -0
  120. package/dist/tui/components/EmptyStateWhisper.d.ts +5 -0
  121. package/dist/tui/components/EmptyStateWhisper.js +6 -0
  122. package/dist/tui/components/EmptyStateWhisper.js.map +1 -0
  123. package/dist/tui/components/Frame.d.ts +15 -0
  124. package/dist/tui/components/Frame.js +21 -0
  125. package/dist/tui/components/Frame.js.map +1 -0
  126. package/dist/tui/components/IdleWhisper.d.ts +11 -0
  127. package/dist/tui/components/IdleWhisper.js +12 -0
  128. package/dist/tui/components/IdleWhisper.js.map +1 -0
  129. package/dist/tui/components/KeyHint.d.ts +13 -0
  130. package/dist/tui/components/KeyHint.js +12 -0
  131. package/dist/tui/components/KeyHint.js.map +1 -0
  132. package/dist/tui/components/LoadingSplash.d.ts +12 -0
  133. package/dist/tui/components/LoadingSplash.js +55 -0
  134. package/dist/tui/components/LoadingSplash.js.map +1 -0
  135. package/dist/tui/components/ProgressBar.d.ts +16 -0
  136. package/dist/tui/components/ProgressBar.js +24 -0
  137. package/dist/tui/components/ProgressBar.js.map +1 -0
  138. package/dist/tui/components/SearchBox.d.ts +5 -0
  139. package/dist/tui/components/SearchBox.js +6 -0
  140. package/dist/tui/components/SearchBox.js.map +1 -0
  141. package/dist/tui/components/SectionHeader.d.ts +12 -0
  142. package/dist/tui/components/SectionHeader.js +14 -0
  143. package/dist/tui/components/SectionHeader.js.map +1 -0
  144. package/dist/tui/components/SkillViewport.d.ts +8 -0
  145. package/dist/tui/components/SkillViewport.js +42 -0
  146. package/dist/tui/components/SkillViewport.js.map +1 -0
  147. package/dist/tui/components/Spinner.d.ts +11 -0
  148. package/dist/tui/components/Spinner.js +13 -0
  149. package/dist/tui/components/Spinner.js.map +1 -0
  150. package/dist/tui/components/StatusPill.d.ts +13 -0
  151. package/dist/tui/components/StatusPill.js +15 -0
  152. package/dist/tui/components/StatusPill.js.map +1 -0
  153. package/dist/tui/components/TabBar.d.ts +20 -0
  154. package/dist/tui/components/TabBar.js +26 -0
  155. package/dist/tui/components/TabBar.js.map +1 -0
  156. package/dist/tui/components/TextInput.d.ts +13 -0
  157. package/dist/tui/components/TextInput.js +34 -0
  158. package/dist/tui/components/TextInput.js.map +1 -0
  159. package/dist/tui/hooks/use-idle-whisper.d.ts +17 -0
  160. package/dist/tui/hooks/use-idle-whisper.js +66 -0
  161. package/dist/tui/hooks/use-idle-whisper.js.map +1 -0
  162. package/dist/tui/hooks/use-rotating-message.d.ts +20 -0
  163. package/dist/tui/hooks/use-rotating-message.js +34 -0
  164. package/dist/tui/hooks/use-rotating-message.js.map +1 -0
  165. package/dist/tui/input/use-keys.d.ts +16 -0
  166. package/dist/tui/input/use-keys.js +32 -0
  167. package/dist/tui/input/use-keys.js.map +1 -0
  168. package/dist/tui/messages.d.ts +18 -0
  169. package/dist/tui/messages.js +59 -0
  170. package/dist/tui/messages.js.map +1 -0
  171. package/dist/tui/render.d.ts +12 -0
  172. package/dist/tui/render.js +112 -0
  173. package/dist/tui/render.js.map +1 -0
  174. package/dist/tui/screens/AccessTab.d.ts +6 -0
  175. package/dist/tui/screens/AccessTab.js +26 -0
  176. package/dist/tui/screens/AccessTab.js.map +1 -0
  177. package/dist/tui/screens/DoctorTab.d.ts +6 -0
  178. package/dist/tui/screens/DoctorTab.js +24 -0
  179. package/dist/tui/screens/DoctorTab.js.map +1 -0
  180. package/dist/tui/screens/FirstRunScan.d.ts +11 -0
  181. package/dist/tui/screens/FirstRunScan.js +128 -0
  182. package/dist/tui/screens/FirstRunScan.js.map +1 -0
  183. package/dist/tui/screens/McpsTab.d.ts +6 -0
  184. package/dist/tui/screens/McpsTab.js +30 -0
  185. package/dist/tui/screens/McpsTab.js.map +1 -0
  186. package/dist/tui/screens/Overview.d.ts +6 -0
  187. package/dist/tui/screens/Overview.js +75 -0
  188. package/dist/tui/screens/Overview.js.map +1 -0
  189. package/dist/tui/screens/ProjectDrillIn.d.ts +10 -0
  190. package/dist/tui/screens/ProjectDrillIn.js +16 -0
  191. package/dist/tui/screens/ProjectDrillIn.js.map +1 -0
  192. package/dist/tui/screens/Settings.d.ts +8 -0
  193. package/dist/tui/screens/Settings.js +71 -0
  194. package/dist/tui/screens/Settings.js.map +1 -0
  195. package/dist/tui/screens/ToolTab.d.ts +9 -0
  196. package/dist/tui/screens/ToolTab.js +38 -0
  197. package/dist/tui/screens/ToolTab.js.map +1 -0
  198. package/dist/tui/screens/UserScopeDrillIn.d.ts +13 -0
  199. package/dist/tui/screens/UserScopeDrillIn.js +15 -0
  200. package/dist/tui/screens/UserScopeDrillIn.js.map +1 -0
  201. package/dist/tui/state/navigation.d.ts +1 -0
  202. package/dist/tui/state/navigation.js +11 -0
  203. package/dist/tui/state/navigation.js.map +1 -0
  204. package/dist/tui/state/settings-state.d.ts +22 -0
  205. package/dist/tui/state/settings-state.js +30 -0
  206. package/dist/tui/state/settings-state.js.map +1 -0
  207. package/dist/tui/state/tui-state.d.ts +57 -0
  208. package/dist/tui/state/tui-state.js +86 -0
  209. package/dist/tui/state/tui-state.js.map +1 -0
  210. package/dist/tui/theme/borders.d.ts +20 -0
  211. package/dist/tui/theme/borders.js +25 -0
  212. package/dist/tui/theme/borders.js.map +1 -0
  213. package/dist/tui/theme/colors.d.ts +14 -0
  214. package/dist/tui/theme/colors.js +18 -0
  215. package/dist/tui/theme/colors.js.map +1 -0
  216. package/dist/tui/theme/icons.d.ts +30 -0
  217. package/dist/tui/theme/icons.js +51 -0
  218. package/dist/tui/theme/icons.js.map +1 -0
  219. package/dist/tui/util/doctor-grouping.d.ts +27 -0
  220. package/dist/tui/util/doctor-grouping.js +67 -0
  221. package/dist/tui/util/doctor-grouping.js.map +1 -0
  222. package/dist/tui/util/finding-grouping.d.ts +24 -0
  223. package/dist/tui/util/finding-grouping.js +43 -0
  224. package/dist/tui/util/finding-grouping.js.map +1 -0
  225. package/dist/tui/util/mcp-grouping.d.ts +27 -0
  226. package/dist/tui/util/mcp-grouping.js +96 -0
  227. package/dist/tui/util/mcp-grouping.js.map +1 -0
  228. package/dist/tui/util/scan-history.d.ts +15 -0
  229. package/dist/tui/util/scan-history.js +43 -0
  230. package/dist/tui/util/scan-history.js.map +1 -0
  231. package/dist/tui/util/skill-filter.d.ts +2 -0
  232. package/dist/tui/util/skill-filter.js +7 -0
  233. package/dist/tui/util/skill-filter.js.map +1 -0
  234. package/dist/tui/util/skill-grouping.d.ts +3 -0
  235. package/dist/tui/util/skill-grouping.js +31 -0
  236. package/dist/tui/util/skill-grouping.js.map +1 -0
  237. package/dist/types.d.ts +151 -0
  238. package/dist/types.js +143 -0
  239. package/dist/types.js.map +1 -0
  240. package/dist/utils/errors.d.ts +1 -0
  241. package/dist/utils/errors.js +7 -0
  242. package/dist/utils/errors.js.map +1 -0
  243. package/dist/utils/format-access.d.ts +3 -0
  244. package/dist/utils/format-access.js +94 -0
  245. package/dist/utils/format-access.js.map +1 -0
  246. package/dist/utils/format-caps.d.ts +3 -0
  247. package/dist/utils/format-caps.js +139 -0
  248. package/dist/utils/format-caps.js.map +1 -0
  249. package/dist/utils/format-doctor.d.ts +3 -0
  250. package/dist/utils/format-doctor.js +105 -0
  251. package/dist/utils/format-doctor.js.map +1 -0
  252. package/dist/utils/format-list.d.ts +7 -0
  253. package/dist/utils/format-list.js +112 -0
  254. package/dist/utils/format-list.js.map +1 -0
  255. package/dist/utils/format-mcp.d.ts +3 -0
  256. package/dist/utils/format-mcp.js +130 -0
  257. package/dist/utils/format-mcp.js.map +1 -0
  258. package/dist/utils/format-multi-project.d.ts +3 -0
  259. package/dist/utils/format-multi-project.js +100 -0
  260. package/dist/utils/format-multi-project.js.map +1 -0
  261. package/dist/utils/format-show.d.ts +6 -0
  262. package/dist/utils/format-show.js +143 -0
  263. package/dist/utils/format-show.js.map +1 -0
  264. package/dist/utils/format.d.ts +3 -0
  265. package/dist/utils/format.js +53 -0
  266. package/dist/utils/format.js.map +1 -0
  267. package/dist/utils/paths.d.ts +1 -0
  268. package/dist/utils/paths.js +10 -0
  269. package/dist/utils/paths.js.map +1 -0
  270. package/package.json +61 -0
@@ -0,0 +1,210 @@
1
+ import fs from "node:fs/promises";
2
+ import path from "node:path";
3
+ import { createWarning } from "../types.js";
4
+ export const DEFAULT_MAX_DEPTH = 6;
5
+ export const DEFAULT_CRAWL_CONCURRENCY = 16;
6
+ export const DEFAULT_SKIP_NAMES = new Set([
7
+ // macOS junk
8
+ "Library",
9
+ "Music",
10
+ "Pictures",
11
+ "Movies",
12
+ "Applications",
13
+ "Public",
14
+ ".Trash",
15
+ // Package caches
16
+ "node_modules",
17
+ ".npm",
18
+ ".pnpm-store",
19
+ ".yarn",
20
+ ".cache",
21
+ ".cargo",
22
+ ".rustup",
23
+ ".gem",
24
+ ".go",
25
+ // Dev clutter
26
+ ".local",
27
+ ".config",
28
+ "vendor",
29
+ ".venv",
30
+ "venv",
31
+ "__pycache__",
32
+ ".next",
33
+ ".nuxt",
34
+ ".svelte-kit",
35
+ "dist",
36
+ "build",
37
+ "target",
38
+ // Build/IDE
39
+ ".idea",
40
+ ".vscode",
41
+ ".gradle",
42
+ ".m2"
43
+ ]);
44
+ export const DEFAULT_MARKER_DIRS = new Set([
45
+ ".claude",
46
+ ".codex",
47
+ ".cursor",
48
+ ".gemini",
49
+ ".opencode",
50
+ ".skills"
51
+ ]);
52
+ export const DEFAULT_MARKER_FILES = new Set([
53
+ "CLAUDE.md",
54
+ "AGENTS.md",
55
+ "GEMINI.md",
56
+ ".cursorrules",
57
+ ".mcp.json",
58
+ "opencode.json",
59
+ "opencode.jsonc"
60
+ ]);
61
+ export async function crawlForProjects(options) {
62
+ const rootDir = options.rootDir;
63
+ const maxDepth = options.maxDepth ?? DEFAULT_MAX_DEPTH;
64
+ const skipNames = options.skipNames ?? DEFAULT_SKIP_NAMES;
65
+ const markerDirs = options.markerDirs ?? DEFAULT_MARKER_DIRS;
66
+ const markerFiles = options.markerFiles ?? DEFAULT_MARKER_FILES;
67
+ const concurrency = options.concurrency ?? DEFAULT_CRAWL_CONCURRENCY;
68
+ const projects = [];
69
+ const warnings = [];
70
+ const start = Date.now();
71
+ let pathsVisited = 0;
72
+ const visited = new Set();
73
+ await visitDirectory({
74
+ dirPath: rootDir,
75
+ depth: 0,
76
+ maxDepth,
77
+ skipNames,
78
+ markerDirs,
79
+ markerFiles,
80
+ projects,
81
+ warnings,
82
+ onProject: options.onProject,
83
+ incrementVisited: () => {
84
+ pathsVisited += 1;
85
+ },
86
+ visited,
87
+ concurrency
88
+ });
89
+ return {
90
+ projects,
91
+ warnings,
92
+ stats: {
93
+ pathsVisited,
94
+ durationMs: Date.now() - start
95
+ }
96
+ };
97
+ }
98
+ async function visitDirectory(args) {
99
+ let realPath;
100
+ try {
101
+ realPath = await fs.realpath(args.dirPath);
102
+ }
103
+ catch (error) {
104
+ args.warnings.push(createWarning({
105
+ reason: classifyFsError(error),
106
+ path: args.dirPath,
107
+ message: `Cannot resolve realpath ${args.dirPath}: ${formatErrorMessage(error)}`
108
+ }));
109
+ return;
110
+ }
111
+ if (args.visited.has(realPath))
112
+ return;
113
+ args.visited.add(realPath);
114
+ args.incrementVisited();
115
+ let entries;
116
+ try {
117
+ entries = await fs.readdir(args.dirPath, { withFileTypes: true });
118
+ }
119
+ catch (error) {
120
+ args.warnings.push(createWarning({
121
+ reason: classifyFsError(error),
122
+ path: args.dirPath,
123
+ message: `Cannot read ${args.dirPath}: ${formatErrorMessage(error)}`
124
+ }));
125
+ return;
126
+ }
127
+ const matchedMarkers = [];
128
+ for (const entry of entries) {
129
+ if (entry.isDirectory() && args.markerDirs.has(entry.name)) {
130
+ matchedMarkers.push(entry.name);
131
+ }
132
+ else if (entry.isFile() && args.markerFiles.has(entry.name)) {
133
+ matchedMarkers.push(entry.name);
134
+ }
135
+ }
136
+ if (matchedMarkers.length > 0 && args.depth > 0) {
137
+ const project = {
138
+ projectPath: args.dirPath,
139
+ parentPath: path.dirname(args.dirPath),
140
+ markers: matchedMarkers,
141
+ depth: args.depth
142
+ };
143
+ args.projects.push(project);
144
+ args.onProject?.(project);
145
+ }
146
+ if (args.depth >= args.maxDepth)
147
+ return;
148
+ const childrenToVisit = [];
149
+ for (const entry of entries) {
150
+ if (!entry.isDirectory())
151
+ continue;
152
+ const childName = entry.name;
153
+ // Skip list — applies at any depth.
154
+ if (args.skipNames.has(childName))
155
+ continue;
156
+ // Marker directories are LEAVES. We've already recorded the parent above.
157
+ // Do not descend into them.
158
+ if (args.markerDirs.has(childName))
159
+ continue;
160
+ // Hidden dirs at any depth are skipped UNLESS the dir is itself a marker
161
+ // (handled by the markerDirs branch above). Reaching this point with a
162
+ // leading-dot child means it's hidden and NOT a marker → skip.
163
+ if (childName.startsWith("."))
164
+ continue;
165
+ childrenToVisit.push(path.join(args.dirPath, childName));
166
+ }
167
+ await runWithConcurrency(childrenToVisit, args.concurrency, async (childPath) => {
168
+ await visitDirectory({
169
+ ...args,
170
+ dirPath: childPath,
171
+ depth: args.depth + 1
172
+ });
173
+ });
174
+ }
175
+ async function runWithConcurrency(items, concurrency, worker) {
176
+ if (items.length === 0)
177
+ return;
178
+ const limit = Math.max(1, Math.floor(concurrency));
179
+ let nextIndex = 0;
180
+ async function next() {
181
+ while (true) {
182
+ const i = nextIndex;
183
+ nextIndex += 1;
184
+ if (i >= items.length)
185
+ return;
186
+ await worker(items[i]);
187
+ }
188
+ }
189
+ const workers = [];
190
+ for (let w = 0; w < Math.min(limit, items.length); w += 1) {
191
+ workers.push(next());
192
+ }
193
+ await Promise.all(workers);
194
+ }
195
+ function classifyFsError(error) {
196
+ if (isNodeError(error)) {
197
+ if (error.code === "EACCES" || error.code === "EPERM")
198
+ return "permission_denied";
199
+ if (error.code === "ENOENT")
200
+ return "permission_denied";
201
+ }
202
+ return "unknown";
203
+ }
204
+ function isNodeError(error) {
205
+ return error instanceof Error && "code" in error;
206
+ }
207
+ function formatErrorMessage(error) {
208
+ return error instanceof Error ? error.message : String(error);
209
+ }
210
+ //# sourceMappingURL=filesystem-crawler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"filesystem-crawler.js","sourceRoot":"","sources":["../../src/scanner/filesystem-crawler.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ import { type ScanResult } from "../types.js";
2
+ export interface ScanOptions {
3
+ cwd?: string;
4
+ homeDir?: string;
5
+ env?: Record<string, string | undefined>;
6
+ now?: Date;
7
+ }
8
+ export declare function scan(options?: ScanOptions): Promise<ScanResult>;
@@ -0,0 +1,120 @@
1
+ import os from "node:os";
2
+ import { createAllEmptyTools, createScanSummary, createToolStats } from "../types.js";
3
+ import { enrichSkill } from "./capability-map.js";
4
+ import { reviewTools } from "./access-review.js";
5
+ import { claudeAdapter } from "./adapters/claude.js";
6
+ import { codexAdapter } from "./adapters/codex.js";
7
+ import { cursorAdapter } from "./adapters/cursor.js";
8
+ import { geminiAdapter } from "./adapters/gemini.js";
9
+ import { opencodeAdapter } from "./adapters/opencode.js";
10
+ import { skillsShAdapter } from "./adapters/skills-sh.js";
11
+ import { runScannerAdapters } from "./adapters/index.js";
12
+ import { discover, isIgnoredByRootGitignore } from "./discovery.js";
13
+ export async function scan(options = {}) {
14
+ const cwd = options.cwd ?? process.cwd();
15
+ const homeDir = options.homeDir ?? os.homedir();
16
+ const tools = createAllEmptyTools();
17
+ const discovery = await discover({
18
+ cwd,
19
+ homeDir,
20
+ env: options.env ?? process.env
21
+ });
22
+ for (const discoveredPath of discovery.paths) {
23
+ const tool = findTool(tools, discoveredPath.toolId);
24
+ if (!tool) {
25
+ continue;
26
+ }
27
+ tool.detected = true;
28
+ if (!tool.detectedPaths.includes(discoveredPath.path)) {
29
+ tool.detectedPaths.push(discoveredPath.path);
30
+ }
31
+ }
32
+ for (const skippedPath of discovery.skippedPaths) {
33
+ if (!skippedPath.toolId) {
34
+ continue;
35
+ }
36
+ const tool = findTool(tools, skippedPath.toolId);
37
+ if (!tool || tool.warnings.some((warning) => warning.id === skippedPath.warning.id)) {
38
+ continue;
39
+ }
40
+ tool.warnings.push(skippedPath.warning);
41
+ }
42
+ const adapterResults = await runScannerAdapters([claudeAdapter, codexAdapter, cursorAdapter, geminiAdapter, opencodeAdapter, skillsShAdapter], {
43
+ cwd,
44
+ homeDir,
45
+ env: options.env ?? process.env,
46
+ discoveredPaths: discovery.paths,
47
+ isIgnored: (rel) => isIgnoredByRootGitignore(rel, discovery.gitignorePatterns)
48
+ });
49
+ for (const adapterResult of adapterResults) {
50
+ const tool = findTool(tools, adapterResult.toolId);
51
+ if (!tool) {
52
+ continue;
53
+ }
54
+ tool.skills.push(...adapterResult.skills);
55
+ tool.findings.push(...adapterResult.findings);
56
+ addWarnings(tool.warnings, adapterResult.warnings);
57
+ }
58
+ for (const tool of tools) {
59
+ for (const skill of tool.skills) {
60
+ enrichSkill(skill);
61
+ }
62
+ }
63
+ const reviewFindings = reviewTools(tools);
64
+ for (const finding of reviewFindings) {
65
+ for (const toolId of finding.toolIds) {
66
+ const tool = findTool(tools, toolId);
67
+ if (tool && !tool.findings.some((existing) => existing.id === finding.id)) {
68
+ tool.findings.push(finding);
69
+ }
70
+ }
71
+ }
72
+ for (const tool of tools) {
73
+ tool.stats = createToolStats(tool.skills, tool.findings);
74
+ }
75
+ const findings = dedupeFindings(tools.flatMap((tool) => tool.findings));
76
+ const warnings = dedupeWarnings([
77
+ ...discovery.warnings,
78
+ ...tools.flatMap((tool) => tool.warnings)
79
+ ]);
80
+ return {
81
+ scannedAt: (options.now ?? new Date()).toISOString(),
82
+ cwd,
83
+ homeDir,
84
+ tools,
85
+ findings,
86
+ warnings,
87
+ summary: createScanSummary(tools)
88
+ };
89
+ }
90
+ function findTool(tools, toolId) {
91
+ return tools.find((tool) => tool.id === toolId);
92
+ }
93
+ function addWarnings(target, warnings) {
94
+ const seen = new Set(target.map((warning) => warning.id));
95
+ for (const warning of warnings) {
96
+ if (seen.has(warning.id)) {
97
+ continue;
98
+ }
99
+ seen.add(warning.id);
100
+ target.push(warning);
101
+ }
102
+ }
103
+ function dedupeFindings(findings) {
104
+ const seen = new Set();
105
+ const deduped = [];
106
+ for (const finding of findings) {
107
+ if (seen.has(finding.id)) {
108
+ continue;
109
+ }
110
+ seen.add(finding.id);
111
+ deduped.push(finding);
112
+ }
113
+ return deduped;
114
+ }
115
+ function dedupeWarnings(warnings) {
116
+ const deduped = [];
117
+ addWarnings(deduped, warnings);
118
+ return deduped;
119
+ }
120
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/scanner/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,31 @@
1
+ import { type MultiProjectScanResult, type ScanResult, type Warning } from "../types.js";
2
+ import { type ScanOptions } from "./index.js";
3
+ export declare const PROJECT_MARKER_DIRS: readonly [".claude", ".codex", ".cursor", ".gemini", ".opencode", ".skills"];
4
+ export declare const PROJECT_MARKER_FILES: readonly ["CLAUDE.md", "AGENTS.md", "GEMINI.md", ".cursorrules", ".mcp.json", "opencode.json", "opencode.jsonc"];
5
+ export interface DiscoveredProject {
6
+ projectPath: string;
7
+ displayPath: string;
8
+ }
9
+ export interface DiscoverProjectsResult {
10
+ projects: DiscoveredProject[];
11
+ warnings: Warning[];
12
+ }
13
+ export declare function discoverProjects(devRoots: readonly string[], homeDir: string): Promise<DiscoverProjectsResult>;
14
+ export interface LoadAllScansOptions {
15
+ devRoots: readonly string[];
16
+ homeDir: string;
17
+ env: Record<string, string | undefined>;
18
+ now?: Date;
19
+ /** Per-project scan budget (ms). Default 5000. */
20
+ perProjectTimeoutMs?: number;
21
+ /** Concurrency for project scans. Default 10. */
22
+ concurrency?: number;
23
+ /** Test hook — replaces the underlying `scan()` implementation. Not for production. */
24
+ __scanForTesting?: (options: ScanOptions) => Promise<ScanResult>;
25
+ }
26
+ export declare function loadAllScans(options: LoadAllScansOptions): Promise<MultiProjectScanResult>;
27
+ export interface DevRootsConfig {
28
+ devRoots: string[];
29
+ warnings: Warning[];
30
+ }
31
+ export declare function readDevRootsConfig(homeDir: string): Promise<DevRootsConfig>;
@@ -0,0 +1,227 @@
1
+ import fs from "node:fs/promises";
2
+ import path from "node:path";
3
+ import { createMultiProjectTotals, createWarning } from "../types.js";
4
+ import { relativizeHome } from "../utils/paths.js";
5
+ import { scan as defaultScan } from "./index.js";
6
+ import { parallelMap } from "./parallel.js";
7
+ export const PROJECT_MARKER_DIRS = [
8
+ ".claude",
9
+ ".codex",
10
+ ".cursor",
11
+ ".gemini",
12
+ ".opencode",
13
+ ".skills"
14
+ ];
15
+ export const PROJECT_MARKER_FILES = [
16
+ "CLAUDE.md",
17
+ "AGENTS.md",
18
+ "GEMINI.md",
19
+ ".cursorrules",
20
+ ".mcp.json",
21
+ "opencode.json",
22
+ "opencode.jsonc"
23
+ ];
24
+ export async function discoverProjects(devRoots, homeDir) {
25
+ const seen = new Set();
26
+ const projects = [];
27
+ const warnings = [];
28
+ for (const devRoot of devRoots) {
29
+ let entries;
30
+ try {
31
+ entries = await fs.readdir(devRoot, { withFileTypes: true });
32
+ }
33
+ catch (error) {
34
+ warnings.push(createWarning({
35
+ reason: "permission_denied",
36
+ path: devRoot,
37
+ message: `Cannot read dev root ${devRoot}: ${formatErrorMessage(error)}`
38
+ }));
39
+ continue;
40
+ }
41
+ for (const entry of entries) {
42
+ if (!entry.isDirectory() && !entry.isSymbolicLink())
43
+ continue;
44
+ if (entry.name.startsWith("."))
45
+ continue;
46
+ const projectPath = path.join(devRoot, entry.name);
47
+ if (seen.has(projectPath))
48
+ continue;
49
+ if (await hasProjectMarker(projectPath)) {
50
+ seen.add(projectPath);
51
+ projects.push({
52
+ projectPath,
53
+ displayPath: relativizeHome(projectPath, homeDir)
54
+ });
55
+ }
56
+ }
57
+ }
58
+ return { projects, warnings };
59
+ }
60
+ async function hasProjectMarker(projectPath) {
61
+ for (const dirName of PROJECT_MARKER_DIRS) {
62
+ try {
63
+ const stat = await fs.stat(path.join(projectPath, dirName));
64
+ if (stat.isDirectory())
65
+ return true;
66
+ }
67
+ catch {
68
+ // not present — try next
69
+ }
70
+ }
71
+ for (const fileName of PROJECT_MARKER_FILES) {
72
+ try {
73
+ const stat = await fs.stat(path.join(projectPath, fileName));
74
+ if (stat.isFile())
75
+ return true;
76
+ }
77
+ catch {
78
+ // not present — try next
79
+ }
80
+ }
81
+ return false;
82
+ }
83
+ function formatErrorMessage(error) {
84
+ if (error instanceof Error)
85
+ return error.message;
86
+ return String(error);
87
+ }
88
+ function isMissingFileError(error) {
89
+ return Boolean(error &&
90
+ typeof error === "object" &&
91
+ "code" in error &&
92
+ error.code === "ENOENT");
93
+ }
94
+ const DEFAULT_PER_PROJECT_TIMEOUT_MS = 5000;
95
+ const DEFAULT_CONCURRENCY = 10;
96
+ export async function loadAllScans(options) {
97
+ const now = options.now ?? new Date();
98
+ const timeoutMs = options.perProjectTimeoutMs ?? DEFAULT_PER_PROJECT_TIMEOUT_MS;
99
+ const concurrency = options.concurrency ?? DEFAULT_CONCURRENCY;
100
+ const scanImpl = options.__scanForTesting ?? defaultScan;
101
+ const warnings = [];
102
+ const discovery = await discoverProjects(options.devRoots, options.homeDir);
103
+ warnings.push(...discovery.warnings);
104
+ const userScope = await scanImpl({
105
+ cwd: options.homeDir,
106
+ homeDir: options.homeDir,
107
+ env: options.env,
108
+ now
109
+ });
110
+ const outcomes = await parallelMap(discovery.projects, async (discovered) => {
111
+ try {
112
+ const scanResult = await runWithTimeout(() => scanImpl({
113
+ cwd: discovered.projectPath,
114
+ homeDir: options.homeDir,
115
+ env: options.env,
116
+ now
117
+ }), timeoutMs, `Project scan timed out after ${timeoutMs}ms`);
118
+ return {
119
+ project: {
120
+ projectPath: discovered.projectPath,
121
+ displayPath: discovered.displayPath,
122
+ scan: scanResult
123
+ }
124
+ };
125
+ }
126
+ catch (error) {
127
+ const reason = isTimeoutError(error) ? "adapter_timeout" : "non_disk_config_skipped";
128
+ return {
129
+ warning: createWarning({
130
+ reason,
131
+ path: discovered.projectPath,
132
+ message: error instanceof Error
133
+ ? error.message
134
+ : `Project scan failed: ${String(error)}`
135
+ })
136
+ };
137
+ }
138
+ }, { concurrency });
139
+ const projects = [];
140
+ for (const outcome of outcomes) {
141
+ if ("project" in outcome) {
142
+ projects.push(outcome.project);
143
+ }
144
+ else {
145
+ warnings.push(outcome.warning);
146
+ }
147
+ }
148
+ return {
149
+ scannedAt: now.toISOString(),
150
+ cwd: options.homeDir,
151
+ homeDir: options.homeDir,
152
+ devRoots: [...options.devRoots],
153
+ userScope,
154
+ projects,
155
+ warnings,
156
+ totals: createMultiProjectTotals({
157
+ userScopeSkillCount: userScope.tools.reduce((n, t) => n + t.skills.length, 0),
158
+ projectSkillCounts: projects.map((p) => p.scan.tools.reduce((n, t) => n + t.skills.length, 0))
159
+ })
160
+ };
161
+ }
162
+ class TimeoutError extends Error {
163
+ __ankuiTimeout = true;
164
+ }
165
+ function isTimeoutError(error) {
166
+ return Boolean(error && typeof error === "object" && error.__ankuiTimeout);
167
+ }
168
+ async function runWithTimeout(fn, ms, message) {
169
+ let timer;
170
+ try {
171
+ return await Promise.race([
172
+ fn(),
173
+ new Promise((_, reject) => {
174
+ timer = setTimeout(() => reject(new TimeoutError(message)), ms);
175
+ })
176
+ ]);
177
+ }
178
+ finally {
179
+ if (timer)
180
+ clearTimeout(timer);
181
+ }
182
+ }
183
+ export async function readDevRootsConfig(homeDir) {
184
+ const configPath = path.join(homeDir, ".config", "ankui", "config.json");
185
+ let raw;
186
+ try {
187
+ raw = await fs.readFile(configPath, "utf8");
188
+ }
189
+ catch (error) {
190
+ const missing = isMissingFileError(error);
191
+ const reason = missing ? "not_found" : "permission_denied";
192
+ const message = missing
193
+ ? `No Ankui config at ${configPath} yet — run \`ankui discover --apply\` to create one.`
194
+ : `Cannot read ${configPath}: ${formatErrorMessage(error)}`;
195
+ return {
196
+ devRoots: [],
197
+ warnings: [createWarning({ reason, path: configPath, message })]
198
+ };
199
+ }
200
+ let parsed;
201
+ try {
202
+ parsed = JSON.parse(raw);
203
+ }
204
+ catch (error) {
205
+ return {
206
+ devRoots: [],
207
+ warnings: [
208
+ createWarning({
209
+ reason: "parse_failed",
210
+ path: configPath,
211
+ message: `Failed to parse ${configPath}: ${formatErrorMessage(error)}`
212
+ })
213
+ ]
214
+ };
215
+ }
216
+ const devRoots = extractDevRoots(parsed);
217
+ return { devRoots, warnings: [] };
218
+ }
219
+ function extractDevRoots(parsed) {
220
+ if (!parsed || typeof parsed !== "object")
221
+ return [];
222
+ const candidate = parsed.devRoots;
223
+ if (!Array.isArray(candidate))
224
+ return [];
225
+ return candidate.filter((entry) => typeof entry === "string");
226
+ }
227
+ //# sourceMappingURL=multi-project.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"multi-project.js","sourceRoot":"","sources":["../../src/scanner/multi-project.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ export interface ParallelMapOptions {
2
+ concurrency: number;
3
+ }
4
+ export declare function parallelMap<T, R>(items: readonly T[], mapper: (item: T, index: number) => Promise<R>, options: ParallelMapOptions): Promise<R[]>;
@@ -0,0 +1,41 @@
1
+ export async function parallelMap(items, mapper, options) {
2
+ const { concurrency } = options;
3
+ if (!Number.isFinite(concurrency) || concurrency <= 0) {
4
+ throw new RangeError(`parallelMap: concurrency must be a positive finite number (got ${concurrency})`);
5
+ }
6
+ if (items.length === 0) {
7
+ return [];
8
+ }
9
+ const results = new Array(items.length);
10
+ let nextIndex = 0;
11
+ let firstError = undefined;
12
+ async function worker() {
13
+ while (true) {
14
+ if (firstError !== undefined)
15
+ return;
16
+ const i = nextIndex;
17
+ nextIndex += 1;
18
+ if (i >= items.length)
19
+ return;
20
+ try {
21
+ results[i] = await mapper(items[i], i);
22
+ }
23
+ catch (error) {
24
+ if (firstError === undefined)
25
+ firstError = error;
26
+ return;
27
+ }
28
+ }
29
+ }
30
+ const workerCount = Math.min(Math.floor(concurrency), items.length);
31
+ const workers = [];
32
+ for (let w = 0; w < workerCount; w += 1) {
33
+ workers.push(worker());
34
+ }
35
+ await Promise.all(workers);
36
+ if (firstError !== undefined) {
37
+ throw firstError;
38
+ }
39
+ return results;
40
+ }
41
+ //# sourceMappingURL=parallel.js.map