harnery 0.0.1 → 0.2.1

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 (445) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +84 -2
  3. package/bin/agent-coord +42 -0
  4. package/bin/agent-hook +44 -0
  5. package/bin/harn +40 -0
  6. package/dist/cli.d.ts +9 -0
  7. package/dist/cli.d.ts.map +1 -0
  8. package/dist/cli.js +18 -0
  9. package/dist/commander.d.ts +128 -0
  10. package/dist/commander.d.ts.map +1 -0
  11. package/dist/commander.js +126 -0
  12. package/dist/commands/agents.d.ts +18 -0
  13. package/dist/commands/agents.d.ts.map +1 -0
  14. package/dist/commands/agents.js +3946 -0
  15. package/dist/commands/backup.d.ts +22 -0
  16. package/dist/commands/backup.d.ts.map +1 -0
  17. package/dist/commands/backup.js +262 -0
  18. package/dist/commands/browse-ai.d.ts +4 -0
  19. package/dist/commands/browse-ai.d.ts.map +1 -0
  20. package/dist/commands/browse-ai.js +156 -0
  21. package/dist/commands/browse.d.ts +4 -0
  22. package/dist/commands/browse.d.ts.map +1 -0
  23. package/dist/commands/browse.js +590 -0
  24. package/dist/commands/callers.d.ts +4 -0
  25. package/dist/commands/callers.d.ts.map +1 -0
  26. package/dist/commands/callers.js +276 -0
  27. package/dist/commands/completion.d.ts +17 -0
  28. package/dist/commands/completion.d.ts.map +1 -0
  29. package/dist/commands/completion.js +158 -0
  30. package/dist/commands/config-get.d.ts +4 -0
  31. package/dist/commands/config-get.d.ts.map +1 -0
  32. package/dist/commands/config-get.js +131 -0
  33. package/dist/commands/context.d.ts +11 -0
  34. package/dist/commands/context.d.ts.map +1 -0
  35. package/dist/commands/context.js +185 -0
  36. package/dist/commands/cookies.d.ts +4 -0
  37. package/dist/commands/cookies.d.ts.map +1 -0
  38. package/dist/commands/cookies.js +140 -0
  39. package/dist/commands/docs.d.ts +4 -0
  40. package/dist/commands/docs.d.ts.map +1 -0
  41. package/dist/commands/docs.js +137 -0
  42. package/dist/commands/doctor.d.ts +25 -0
  43. package/dist/commands/doctor.d.ts.map +1 -0
  44. package/dist/commands/doctor.js +200 -0
  45. package/dist/commands/edit-batch.d.ts +18 -0
  46. package/dist/commands/edit-batch.d.ts.map +1 -0
  47. package/dist/commands/edit-batch.js +172 -0
  48. package/dist/commands/eml.d.ts +4 -0
  49. package/dist/commands/eml.d.ts.map +1 -0
  50. package/dist/commands/eml.js +428 -0
  51. package/dist/commands/env.d.ts +4 -0
  52. package/dist/commands/env.d.ts.map +1 -0
  53. package/dist/commands/env.js +201 -0
  54. package/dist/commands/fetch.d.ts +4 -0
  55. package/dist/commands/fetch.d.ts.map +1 -0
  56. package/dist/commands/fetch.js +99 -0
  57. package/dist/commands/file-history.d.ts +4 -0
  58. package/dist/commands/file-history.d.ts.map +1 -0
  59. package/dist/commands/file-history.js +152 -0
  60. package/dist/commands/grep.d.ts +4 -0
  61. package/dist/commands/grep.d.ts.map +1 -0
  62. package/dist/commands/grep.js +317 -0
  63. package/dist/commands/init.d.ts +82 -0
  64. package/dist/commands/init.d.ts.map +1 -0
  65. package/dist/commands/init.js +288 -0
  66. package/dist/commands/outline.d.ts +4 -0
  67. package/dist/commands/outline.d.ts.map +1 -0
  68. package/dist/commands/outline.js +509 -0
  69. package/dist/commands/presence.d.ts +12 -0
  70. package/dist/commands/presence.d.ts.map +1 -0
  71. package/dist/commands/presence.js +123 -0
  72. package/dist/commands/read.d.ts +7 -0
  73. package/dist/commands/read.d.ts.map +1 -0
  74. package/dist/commands/read.js +46 -0
  75. package/dist/commands/scratch.d.ts +4 -0
  76. package/dist/commands/scratch.d.ts.map +1 -0
  77. package/dist/commands/scratch.js +426 -0
  78. package/dist/commands/session.d.ts +4 -0
  79. package/dist/commands/session.d.ts.map +1 -0
  80. package/dist/commands/session.js +162 -0
  81. package/dist/commands/sync.d.ts +24 -0
  82. package/dist/commands/sync.d.ts.map +1 -0
  83. package/dist/commands/sync.js +275 -0
  84. package/dist/commands/toc.d.ts +5 -0
  85. package/dist/commands/toc.d.ts.map +1 -0
  86. package/dist/commands/toc.js +153 -0
  87. package/dist/commands/tokens.d.ts +4 -0
  88. package/dist/commands/tokens.d.ts.map +1 -0
  89. package/dist/commands/tokens.js +48 -0
  90. package/dist/commands/tunnel.d.ts +4 -0
  91. package/dist/commands/tunnel.d.ts.map +1 -0
  92. package/dist/commands/tunnel.js +513 -0
  93. package/dist/commands/uninstall.d.ts +22 -0
  94. package/dist/commands/uninstall.d.ts.map +1 -0
  95. package/dist/commands/uninstall.js +126 -0
  96. package/dist/commands/web.d.ts +4 -0
  97. package/dist/commands/web.d.ts.map +1 -0
  98. package/dist/commands/web.js +165 -0
  99. package/dist/core/agents/canonical-emit.d.ts +27 -0
  100. package/dist/core/agents/canonical-emit.d.ts.map +1 -0
  101. package/dist/core/agents/canonical-emit.js +72 -0
  102. package/dist/core/agents/cli-emit.d.ts +27 -0
  103. package/dist/core/agents/cli-emit.d.ts.map +1 -0
  104. package/dist/core/agents/cli-emit.js +57 -0
  105. package/dist/core/agents/cli.d.ts +10 -0
  106. package/dist/core/agents/cli.d.ts.map +1 -0
  107. package/dist/core/agents/cli.js +757 -0
  108. package/dist/core/agents/codex-replay.d.ts +29 -0
  109. package/dist/core/agents/codex-replay.d.ts.map +1 -0
  110. package/dist/core/agents/codex-replay.js +138 -0
  111. package/dist/core/agents/coord-client.d.ts +98 -0
  112. package/dist/core/agents/coord-client.d.ts.map +1 -0
  113. package/dist/core/agents/coord-client.js +212 -0
  114. package/dist/core/agents/events/consume.d.ts +59 -0
  115. package/dist/core/agents/events/consume.d.ts.map +1 -0
  116. package/dist/core/agents/events/consume.js +147 -0
  117. package/dist/core/agents/events/emit.d.ts +42 -0
  118. package/dist/core/agents/events/emit.d.ts.map +1 -0
  119. package/dist/core/agents/events/emit.js +70 -0
  120. package/dist/core/agents/events/ulid.d.ts +11 -0
  121. package/dist/core/agents/events/ulid.d.ts.map +1 -0
  122. package/dist/core/agents/events/ulid.js +47 -0
  123. package/dist/core/agents/index.d.ts +14 -0
  124. package/dist/core/agents/index.d.ts.map +1 -0
  125. package/dist/core/agents/index.js +13 -0
  126. package/dist/core/agents/paths.d.ts +6 -0
  127. package/dist/core/agents/paths.d.ts.map +1 -0
  128. package/dist/core/agents/paths.js +17 -0
  129. package/dist/core/agents/render/prompt-context.d.ts +43 -0
  130. package/dist/core/agents/render/prompt-context.d.ts.map +1 -0
  131. package/dist/core/agents/render/prompt-context.js +335 -0
  132. package/dist/core/agents/render/session-context.d.ts +39 -0
  133. package/dist/core/agents/render/session-context.d.ts.map +1 -0
  134. package/dist/core/agents/render/session-context.js +283 -0
  135. package/dist/core/agents/rules/claim-conflict.d.ts +35 -0
  136. package/dist/core/agents/rules/claim-conflict.d.ts.map +1 -0
  137. package/dist/core/agents/rules/claim-conflict.js +244 -0
  138. package/dist/core/agents/rules/commit-conflict.d.ts +59 -0
  139. package/dist/core/agents/rules/commit-conflict.d.ts.map +1 -0
  140. package/dist/core/agents/rules/commit-conflict.js +244 -0
  141. package/dist/core/agents/rules/stop-hook.d.ts +44 -0
  142. package/dist/core/agents/rules/stop-hook.d.ts.map +1 -0
  143. package/dist/core/agents/rules/stop-hook.js +161 -0
  144. package/dist/core/agents/session-events.d.ts +41 -0
  145. package/dist/core/agents/session-events.d.ts.map +1 -0
  146. package/dist/core/agents/session-events.js +205 -0
  147. package/dist/core/agents/state/activity-log.d.ts +18 -0
  148. package/dist/core/agents/state/activity-log.d.ts.map +1 -0
  149. package/dist/core/agents/state/activity-log.js +34 -0
  150. package/dist/core/agents/state/council.d.ts +39 -0
  151. package/dist/core/agents/state/council.d.ts.map +1 -0
  152. package/dist/core/agents/state/council.js +216 -0
  153. package/dist/core/agents/state/heartbeat-projector.d.ts +59 -0
  154. package/dist/core/agents/state/heartbeat-projector.d.ts.map +1 -0
  155. package/dist/core/agents/state/heartbeat-projector.js +436 -0
  156. package/dist/core/agents/state/heartbeat-writer.d.ts +64 -0
  157. package/dist/core/agents/state/heartbeat-writer.d.ts.map +1 -0
  158. package/dist/core/agents/state/heartbeat-writer.js +271 -0
  159. package/dist/core/agents/state/names.d.ts +35 -0
  160. package/dist/core/agents/state/names.d.ts.map +1 -0
  161. package/dist/core/agents/state/names.js +376 -0
  162. package/dist/core/agents/state/pidmap.d.ts +11 -0
  163. package/dist/core/agents/state/pidmap.d.ts.map +1 -0
  164. package/dist/core/agents/state/pidmap.js +32 -0
  165. package/dist/core/agents/state/scratch.d.ts +27 -0
  166. package/dist/core/agents/state/scratch.d.ts.map +1 -0
  167. package/dist/core/agents/state/scratch.js +90 -0
  168. package/dist/core/agents/state/shell-mutation.d.ts +17 -0
  169. package/dist/core/agents/state/shell-mutation.d.ts.map +1 -0
  170. package/dist/core/agents/state/shell-mutation.js +41 -0
  171. package/dist/core/agents/state/stale-sweep.d.ts +16 -0
  172. package/dist/core/agents/state/stale-sweep.d.ts.map +1 -0
  173. package/dist/core/agents/state/stale-sweep.js +166 -0
  174. package/dist/core/config.d.ts +29 -0
  175. package/dist/core/config.d.ts.map +1 -0
  176. package/dist/core/config.js +108 -0
  177. package/dist/core/hooks/cli.d.ts +21 -0
  178. package/dist/core/hooks/cli.d.ts.map +1 -0
  179. package/dist/core/hooks/cli.js +1123 -0
  180. package/dist/core/hooks/effects/image-capture.d.ts +43 -0
  181. package/dist/core/hooks/effects/image-capture.d.ts.map +1 -0
  182. package/dist/core/hooks/effects/image-capture.js +288 -0
  183. package/dist/core/hooks/effects/index.d.ts +64 -0
  184. package/dist/core/hooks/effects/index.d.ts.map +1 -0
  185. package/dist/core/hooks/effects/index.js +197 -0
  186. package/dist/core/hooks/events/emit.d.ts +31 -0
  187. package/dist/core/hooks/events/emit.d.ts.map +1 -0
  188. package/dist/core/hooks/events/emit.js +89 -0
  189. package/dist/core/hooks/events/schema.d.ts +235 -0
  190. package/dist/core/hooks/events/schema.d.ts.map +1 -0
  191. package/dist/core/hooks/events/schema.js +12 -0
  192. package/dist/core/hooks/events/ulid.d.ts +10 -0
  193. package/dist/core/hooks/events/ulid.d.ts.map +1 -0
  194. package/dist/core/hooks/events/ulid.js +47 -0
  195. package/dist/core/hooks/harness/detect.d.ts +9 -0
  196. package/dist/core/hooks/harness/detect.d.ts.map +1 -0
  197. package/dist/core/hooks/harness/detect.js +29 -0
  198. package/dist/core/hooks/harness/events.d.ts +45 -0
  199. package/dist/core/hooks/harness/events.d.ts.map +1 -0
  200. package/dist/core/hooks/harness/events.js +71 -0
  201. package/dist/core/hooks/harness/output.d.ts +46 -0
  202. package/dist/core/hooks/harness/output.d.ts.map +1 -0
  203. package/dist/core/hooks/harness/output.js +87 -0
  204. package/dist/core/hooks/harness/parse.d.ts +67 -0
  205. package/dist/core/hooks/harness/parse.d.ts.map +1 -0
  206. package/dist/core/hooks/harness/parse.js +132 -0
  207. package/dist/core/hooks/index.d.ts +8 -0
  208. package/dist/core/hooks/index.d.ts.map +1 -0
  209. package/dist/core/hooks/index.js +7 -0
  210. package/dist/core/hooks/resolve/anchor.d.ts +37 -0
  211. package/dist/core/hooks/resolve/anchor.d.ts.map +1 -0
  212. package/dist/core/hooks/resolve/anchor.js +48 -0
  213. package/dist/core/hooks/resolve/coord-root.d.ts +6 -0
  214. package/dist/core/hooks/resolve/coord-root.d.ts.map +1 -0
  215. package/dist/core/hooks/resolve/coord-root.js +27 -0
  216. package/dist/core/hooks/resolve/intent.d.ts +33 -0
  217. package/dist/core/hooks/resolve/intent.d.ts.map +1 -0
  218. package/dist/core/hooks/resolve/intent.js +79 -0
  219. package/dist/core/hooks/resolve/owner.d.ts +42 -0
  220. package/dist/core/hooks/resolve/owner.d.ts.map +1 -0
  221. package/dist/core/hooks/resolve/owner.js +140 -0
  222. package/dist/core/hooks/resolve/transcript.d.ts +26 -0
  223. package/dist/core/hooks/resolve/transcript.d.ts.map +1 -0
  224. package/dist/core/hooks/resolve/transcript.js +73 -0
  225. package/dist/index.d.ts +15 -0
  226. package/dist/index.d.ts.map +1 -0
  227. package/dist/index.js +13 -0
  228. package/dist/lib/agent-browser/client.d.ts +99 -0
  229. package/dist/lib/agent-browser/client.d.ts.map +1 -0
  230. package/dist/lib/agent-browser/client.js +177 -0
  231. package/dist/lib/agent-browser/index.d.ts +2 -0
  232. package/dist/lib/agent-browser/index.d.ts.map +1 -0
  233. package/dist/lib/agent-browser/index.js +1 -0
  234. package/dist/lib/browser/client.d.ts +193 -0
  235. package/dist/lib/browser/client.d.ts.map +1 -0
  236. package/dist/lib/browser/client.js +325 -0
  237. package/dist/lib/browser/dev-overlay.d.ts +23 -0
  238. package/dist/lib/browser/dev-overlay.d.ts.map +1 -0
  239. package/dist/lib/browser/dev-overlay.js +153 -0
  240. package/dist/lib/browser/index.d.ts +5 -0
  241. package/dist/lib/browser/index.d.ts.map +1 -0
  242. package/dist/lib/browser/index.js +2 -0
  243. package/dist/lib/browser/layout.d.ts +79 -0
  244. package/dist/lib/browser/layout.d.ts.map +1 -0
  245. package/dist/lib/browser/layout.js +220 -0
  246. package/dist/lib/browser/visibility.d.ts +86 -0
  247. package/dist/lib/browser/visibility.d.ts.map +1 -0
  248. package/dist/lib/browser/visibility.js +333 -0
  249. package/dist/lib/browser/visual-diff.d.ts +38 -0
  250. package/dist/lib/browser/visual-diff.d.ts.map +1 -0
  251. package/dist/lib/browser/visual-diff.js +107 -0
  252. package/dist/lib/completion/bash.d.ts +25 -0
  253. package/dist/lib/completion/bash.d.ts.map +1 -0
  254. package/dist/lib/completion/bash.js +284 -0
  255. package/dist/lib/completion/fish.d.ts +16 -0
  256. package/dist/lib/completion/fish.d.ts.map +1 -0
  257. package/dist/lib/completion/fish.js +118 -0
  258. package/dist/lib/completion/index.d.ts +5 -0
  259. package/dist/lib/completion/index.d.ts.map +1 -0
  260. package/dist/lib/completion/index.js +4 -0
  261. package/dist/lib/completion/walk.d.ts +68 -0
  262. package/dist/lib/completion/walk.d.ts.map +1 -0
  263. package/dist/lib/completion/walk.js +102 -0
  264. package/dist/lib/completion/zsh.d.ts +13 -0
  265. package/dist/lib/completion/zsh.d.ts.map +1 -0
  266. package/dist/lib/completion/zsh.js +249 -0
  267. package/dist/lib/context/index.d.ts +107 -0
  268. package/dist/lib/context/index.d.ts.map +1 -0
  269. package/dist/lib/context/index.js +275 -0
  270. package/dist/lib/cookies/client.d.ts +131 -0
  271. package/dist/lib/cookies/client.d.ts.map +1 -0
  272. package/dist/lib/cookies/client.js +239 -0
  273. package/dist/lib/cookies/index.d.ts +2 -0
  274. package/dist/lib/cookies/index.d.ts.map +1 -0
  275. package/dist/lib/cookies/index.js +1 -0
  276. package/dist/lib/council/index.d.ts +266 -0
  277. package/dist/lib/council/index.d.ts.map +1 -0
  278. package/dist/lib/council/index.js +674 -0
  279. package/dist/lib/docs-index.d.ts +28 -0
  280. package/dist/lib/docs-index.d.ts.map +1 -0
  281. package/dist/lib/docs-index.js +169 -0
  282. package/dist/lib/docs-lint.d.ts +26 -0
  283. package/dist/lib/docs-lint.d.ts.map +1 -0
  284. package/dist/lib/docs-lint.js +378 -0
  285. package/dist/lib/docs-sweep.d.ts +34 -0
  286. package/dist/lib/docs-sweep.d.ts.map +1 -0
  287. package/dist/lib/docs-sweep.js +304 -0
  288. package/dist/lib/docs.d.ts +27 -0
  289. package/dist/lib/docs.d.ts.map +1 -0
  290. package/dist/lib/docs.js +142 -0
  291. package/dist/lib/env.d.ts +11 -0
  292. package/dist/lib/env.d.ts.map +1 -0
  293. package/dist/lib/env.js +12 -0
  294. package/dist/lib/exec.d.ts +32 -0
  295. package/dist/lib/exec.d.ts.map +1 -0
  296. package/dist/lib/exec.js +54 -0
  297. package/dist/lib/format.d.ts +29 -0
  298. package/dist/lib/format.d.ts.map +1 -0
  299. package/dist/lib/format.js +139 -0
  300. package/dist/lib/http/client.d.ts +56 -0
  301. package/dist/lib/http/client.d.ts.map +1 -0
  302. package/dist/lib/http/client.js +160 -0
  303. package/dist/lib/http/index.d.ts +2 -0
  304. package/dist/lib/http/index.d.ts.map +1 -0
  305. package/dist/lib/http/index.js +1 -0
  306. package/dist/lib/identities/index.d.ts +77 -0
  307. package/dist/lib/identities/index.d.ts.map +1 -0
  308. package/dist/lib/identities/index.js +190 -0
  309. package/dist/lib/machine.d.ts +19 -0
  310. package/dist/lib/machine.d.ts.map +1 -0
  311. package/dist/lib/machine.js +61 -0
  312. package/dist/lib/presence.d.ts +48 -0
  313. package/dist/lib/presence.d.ts.map +1 -0
  314. package/dist/lib/presence.js +123 -0
  315. package/dist/lib/readability/client.d.ts +39 -0
  316. package/dist/lib/readability/client.d.ts.map +1 -0
  317. package/dist/lib/readability/client.js +121 -0
  318. package/dist/lib/readability/index.d.ts +2 -0
  319. package/dist/lib/readability/index.d.ts.map +1 -0
  320. package/dist/lib/readability/index.js +1 -0
  321. package/dist/lib/scratch/index.d.ts +74 -0
  322. package/dist/lib/scratch/index.d.ts.map +1 -0
  323. package/dist/lib/scratch/index.js +393 -0
  324. package/dist/lib/tunnel/gate.d.ts +12 -0
  325. package/dist/lib/tunnel/gate.d.ts.map +1 -0
  326. package/dist/lib/tunnel/gate.js +101 -0
  327. package/dist/lib/tunnel/state.d.ts +34 -0
  328. package/dist/lib/tunnel/state.d.ts.map +1 -0
  329. package/dist/lib/tunnel/state.js +132 -0
  330. package/package.json +160 -8
  331. package/schemas/.gitkeep +0 -0
  332. package/schemas/config.schema.json +109 -0
  333. package/src/cli.ts +22 -0
  334. package/src/commander.ts +242 -0
  335. package/src/commands/.gitkeep +0 -0
  336. package/src/commands/agents.ts +4567 -0
  337. package/src/commands/backup.ts +305 -0
  338. package/src/commands/browse-ai.ts +198 -0
  339. package/src/commands/browse.ts +849 -0
  340. package/src/commands/callers.ts +363 -0
  341. package/src/commands/completion.ts +193 -0
  342. package/src/commands/config-get.ts +161 -0
  343. package/src/commands/context.ts +209 -0
  344. package/src/commands/cookies.ts +198 -0
  345. package/src/commands/docs.ts +174 -0
  346. package/src/commands/doctor.ts +231 -0
  347. package/src/commands/edit-batch.ts +233 -0
  348. package/src/commands/eml.ts +519 -0
  349. package/src/commands/env.ts +254 -0
  350. package/src/commands/fetch.ts +136 -0
  351. package/src/commands/file-history.ts +202 -0
  352. package/src/commands/grep.ts +371 -0
  353. package/src/commands/init.ts +335 -0
  354. package/src/commands/outline.ts +583 -0
  355. package/src/commands/presence.ts +152 -0
  356. package/src/commands/read.ts +64 -0
  357. package/src/commands/scratch.ts +445 -0
  358. package/src/commands/session.ts +187 -0
  359. package/src/commands/sync.ts +306 -0
  360. package/src/commands/toc.ts +218 -0
  361. package/src/commands/tokens.ts +79 -0
  362. package/src/commands/tunnel.ts +633 -0
  363. package/src/commands/uninstall.ts +144 -0
  364. package/src/commands/web.ts +193 -0
  365. package/src/core/agents/canonical-emit.ts +77 -0
  366. package/src/core/agents/cli-emit.ts +64 -0
  367. package/src/core/agents/cli.ts +838 -0
  368. package/src/core/agents/codex-replay.ts +163 -0
  369. package/src/core/agents/coord-client.ts +249 -0
  370. package/src/core/agents/events/consume.ts +196 -0
  371. package/src/core/agents/events/emit.ts +108 -0
  372. package/src/core/agents/events/ulid.ts +51 -0
  373. package/src/core/agents/index.ts +14 -0
  374. package/src/core/agents/paths.ts +16 -0
  375. package/src/core/agents/render/prompt-context.ts +401 -0
  376. package/src/core/agents/render/session-context.ts +341 -0
  377. package/src/core/agents/rules/claim-conflict.ts +282 -0
  378. package/src/core/agents/rules/commit-conflict.ts +303 -0
  379. package/src/core/agents/rules/stop-hook.ts +229 -0
  380. package/src/core/agents/session-events.ts +228 -0
  381. package/src/core/agents/state/activity-log.ts +33 -0
  382. package/src/core/agents/state/council.ts +265 -0
  383. package/src/core/agents/state/heartbeat-projector.ts +488 -0
  384. package/src/core/agents/state/heartbeat-writer.ts +333 -0
  385. package/src/core/agents/state/names.ts +399 -0
  386. package/src/core/agents/state/pidmap.ts +38 -0
  387. package/src/core/agents/state/scratch.ts +121 -0
  388. package/src/core/agents/state/shell-mutation.ts +44 -0
  389. package/src/core/agents/state/stale-sweep.ts +190 -0
  390. package/src/core/config.ts +111 -0
  391. package/src/core/hooks/cli.ts +1247 -0
  392. package/src/core/hooks/effects/image-capture.ts +330 -0
  393. package/src/core/hooks/effects/index.ts +210 -0
  394. package/src/core/hooks/events/emit.ts +120 -0
  395. package/src/core/hooks/events/schema.ts +430 -0
  396. package/src/core/hooks/events/ulid.ts +51 -0
  397. package/src/core/hooks/harness/detect.ts +30 -0
  398. package/src/core/hooks/harness/events.ts +102 -0
  399. package/src/core/hooks/harness/output.ts +100 -0
  400. package/src/core/hooks/harness/parse.ts +180 -0
  401. package/src/core/hooks/index.ts +16 -0
  402. package/src/core/hooks/resolve/anchor.ts +51 -0
  403. package/src/core/hooks/resolve/coord-root.ts +25 -0
  404. package/src/core/hooks/resolve/intent.ts +89 -0
  405. package/src/core/hooks/resolve/owner.ts +140 -0
  406. package/src/core/hooks/resolve/transcript.ts +72 -0
  407. package/src/hooks/.gitkeep +0 -0
  408. package/src/index.ts +15 -0
  409. package/src/lib/agent-browser/client.ts +239 -0
  410. package/src/lib/agent-browser/index.ts +1 -0
  411. package/src/lib/browser/client.ts +449 -0
  412. package/src/lib/browser/dev-overlay.ts +207 -0
  413. package/src/lib/browser/index.ts +24 -0
  414. package/src/lib/browser/layout.ts +288 -0
  415. package/src/lib/browser/visibility.ts +419 -0
  416. package/src/lib/browser/visual-diff.ts +150 -0
  417. package/src/lib/completion/bash.ts +291 -0
  418. package/src/lib/completion/fish.ts +134 -0
  419. package/src/lib/completion/index.ts +10 -0
  420. package/src/lib/completion/walk.ts +184 -0
  421. package/src/lib/completion/zsh.ts +262 -0
  422. package/src/lib/context/index.ts +386 -0
  423. package/src/lib/cookies/client.ts +301 -0
  424. package/src/lib/cookies/index.ts +13 -0
  425. package/src/lib/council/index.ts +803 -0
  426. package/src/lib/docs-index.ts +216 -0
  427. package/src/lib/docs-lint.ts +413 -0
  428. package/src/lib/docs-sweep.ts +348 -0
  429. package/src/lib/docs.ts +199 -0
  430. package/src/lib/env.ts +12 -0
  431. package/src/lib/exec.ts +74 -0
  432. package/src/lib/format.ts +147 -0
  433. package/src/lib/http/client.ts +211 -0
  434. package/src/lib/http/index.ts +1 -0
  435. package/src/lib/identities/index.ts +210 -0
  436. package/src/lib/machine.ts +61 -0
  437. package/src/lib/presence.ts +154 -0
  438. package/src/lib/readability/client.ts +169 -0
  439. package/src/lib/readability/index.ts +5 -0
  440. package/src/lib/readability/turndown-plugin-gfm.d.ts +10 -0
  441. package/src/lib/scratch/index.ts +470 -0
  442. package/src/lib/tunnel/gate.ts +113 -0
  443. package/src/lib/tunnel/state.ts +167 -0
  444. package/src/web/.gitkeep +0 -0
  445. package/index.js +0 -1
@@ -0,0 +1,583 @@
1
+ import { existsSync, readFileSync } from "node:fs";
2
+ import { extname, resolve } from "node:path";
3
+ import type { Command } from "commander";
4
+ import type * as TS from "typescript";
5
+ import type { EmitContext } from "../commander.ts";
6
+ import { resolveBinName } from "../core/config.ts";
7
+
8
+ /**
9
+ * `typescript` is a devDependency, not a runtime dependency, so it is loaded
10
+ * lazily here and only when outlining a TS/JS file. A static top-level import
11
+ * would make the whole CLI fail to boot for an end user who installed harnery
12
+ * without dev deps (every command is registered at startup). PHP/Python
13
+ * outlines use regex and never touch this.
14
+ */
15
+ async function loadTypeScript(): Promise<typeof TS> {
16
+ try {
17
+ return (await import("typescript")) as unknown as typeof TS;
18
+ } catch {
19
+ throw new Error(
20
+ "outlining TS/JS files needs the `typescript` package; install it (npm i -D typescript) or use outline on PHP/Python files",
21
+ );
22
+ }
23
+ }
24
+
25
+ /**
26
+ * `harn outline <file>`: print the structural skeleton of a code file (imports
27
+ * + top-level decls + line numbers). TS/JS/TSX/JSX use the TypeScript compiler
28
+ * AST; PHP and Python use a regex pass. Token-efficient substitute for reading
29
+ * a whole large file just to find one symbol.
30
+ *
31
+ * For markdown files, dispatch to `harn toc`.
32
+ */
33
+
34
+ interface OutlineOpts {
35
+ json?: boolean;
36
+ exportsOnly?: boolean;
37
+ imports?: boolean; // Commander sets to false on --no-imports
38
+ members?: boolean; // Commander sets to false on --no-members
39
+ }
40
+
41
+ interface SymbolEntry {
42
+ kind: string;
43
+ name: string;
44
+ signature?: string;
45
+ line: number;
46
+ exported?: boolean;
47
+ members?: SymbolEntry[];
48
+ }
49
+
50
+ interface OutlineResult {
51
+ file: string;
52
+ language: string;
53
+ total_lines: number;
54
+ imports: string[];
55
+ symbols: SymbolEntry[];
56
+ }
57
+
58
+ export function registerOutlineCommand(program: Command, emit: EmitContext): void {
59
+ program
60
+ .command("outline <file>")
61
+ .description(
62
+ `Print the structural skeleton of a code file (imports + top-level decls + line numbers). Supports TS/JS/TSX/JSX (AST), PHP/Python (regex). Use \`${resolveBinName()} toc\` for markdown.`,
63
+ )
64
+ .option("--json", "Structured JSON envelope")
65
+ .option("--exports-only", "Only show exported symbols")
66
+ .option("--no-imports", "Skip the imports summary block")
67
+ .option("--no-members", "Don't expand class/interface members")
68
+ .action(async (file: string, opts: OutlineOpts) => {
69
+ try {
70
+ const result = await runOutline(file, opts);
71
+ if (opts.json) {
72
+ emit.config({ format: "json" });
73
+ emit.data(result);
74
+ return;
75
+ }
76
+ emit.text(`${renderOutline(result, opts)}\n`);
77
+ } catch (err) {
78
+ emit.error({ code: "outline_failed", message: (err as Error).message });
79
+ process.exit(1);
80
+ }
81
+ });
82
+ }
83
+
84
+ async function runOutline(file: string, opts: OutlineOpts): Promise<OutlineResult> {
85
+ const absPath = resolve(file);
86
+ if (!existsSync(absPath)) throw new Error(`no such file: ${file}`);
87
+
88
+ const content = readFileSync(absPath, "utf8");
89
+ const total_lines = content.split("\n").length;
90
+ const ext = extname(absPath).toLowerCase();
91
+
92
+ let language = "";
93
+ let parsed: { imports: string[]; symbols: SymbolEntry[] };
94
+
95
+ if ([".ts", ".tsx", ".js", ".jsx", ".mts", ".cts", ".mjs", ".cjs"].includes(ext)) {
96
+ language = ext.slice(1);
97
+ parsed = await outlineTypeScript(content, absPath, ext);
98
+ } else if (ext === ".php") {
99
+ language = "php";
100
+ parsed = outlinePhp(content);
101
+ } else if (ext === ".py") {
102
+ language = "python";
103
+ parsed = outlinePython(content);
104
+ } else if ([".md", ".mdx"].includes(ext)) {
105
+ throw new Error(`use \`${resolveBinName()} toc ${file}\` for markdown files`);
106
+ } else {
107
+ throw new Error(
108
+ `unsupported file type: ${ext || "(no extension)"} (supported: ts/tsx/js/jsx/php/py)`,
109
+ );
110
+ }
111
+
112
+ if (opts.exportsOnly) {
113
+ parsed.symbols = parsed.symbols.filter((s) => s.exported);
114
+ }
115
+
116
+ return {
117
+ file,
118
+ language,
119
+ total_lines,
120
+ imports: parsed.imports,
121
+ symbols: parsed.symbols,
122
+ };
123
+ }
124
+
125
+ async function outlineTypeScript(
126
+ content: string,
127
+ path: string,
128
+ ext: string,
129
+ ): Promise<{ imports: string[]; symbols: SymbolEntry[] }> {
130
+ const ts = await loadTypeScript();
131
+ const isTsx = ext === ".tsx" || ext === ".jsx";
132
+ const scriptKind = isTsx ? ts.ScriptKind.TSX : ts.ScriptKind.TS;
133
+ const source = ts.createSourceFile(path, content, ts.ScriptTarget.Latest, true, scriptKind);
134
+
135
+ const imports: string[] = [];
136
+ const symbols: SymbolEntry[] = [];
137
+
138
+ const lineOf = (node: TS.Node) =>
139
+ source.getLineAndCharacterOfPosition(node.getStart(source)).line + 1;
140
+ const text = (node: TS.Node) =>
141
+ content.slice(node.getStart(source), node.getEnd()).replace(/\s+/g, " ").trim();
142
+ const isExported = (node: TS.Node): boolean => {
143
+ const mods = (node as { modifiers?: ReadonlyArray<TS.ModifierLike> }).modifiers;
144
+ return mods?.some((m) => m.kind === ts.SyntaxKind.ExportKeyword) ?? false;
145
+ };
146
+
147
+ for (const stmt of source.statements) {
148
+ if (ts.isImportDeclaration(stmt)) {
149
+ const moduleSpec = (stmt.moduleSpecifier as TS.StringLiteral).text;
150
+ const clause = stmt.importClause;
151
+ const isTypeOnly = clause?.isTypeOnly ? "type " : "";
152
+ let what = "";
153
+ if (clause?.name) what = clause.name.text;
154
+ if (clause?.namedBindings) {
155
+ if (ts.isNamespaceImport(clause.namedBindings)) {
156
+ what = `${what ? `${what}, ` : ""}* as ${clause.namedBindings.name.text}`;
157
+ } else {
158
+ const names = clause.namedBindings.elements.map((e) => e.name.text).join(", ");
159
+ what = `${what ? `${what}, ` : ""}{${names}}`;
160
+ }
161
+ }
162
+ imports.push(`${isTypeOnly}${what || "(side-effect)"} from "${moduleSpec}"`);
163
+ continue;
164
+ }
165
+
166
+ const exported = isExported(stmt);
167
+
168
+ if (ts.isFunctionDeclaration(stmt) && stmt.name) {
169
+ symbols.push({
170
+ kind: "function",
171
+ name: stmt.name.text,
172
+ signature: renderFnSig(ts, stmt, content, source),
173
+ line: lineOf(stmt),
174
+ exported,
175
+ });
176
+ } else if (ts.isClassDeclaration(stmt) && stmt.name) {
177
+ const heritage = (stmt.heritageClauses || [])
178
+ .map(
179
+ (h) =>
180
+ (h.token === ts.SyntaxKind.ExtendsKeyword ? "extends " : "implements ") +
181
+ h.types.map((t) => text(t.expression)).join(", "),
182
+ )
183
+ .join(" ");
184
+ const members: SymbolEntry[] = [];
185
+ for (const member of stmt.members) {
186
+ if (ts.isMethodDeclaration(member) && member.name) {
187
+ members.push({
188
+ kind: "method",
189
+ name: (member.name as TS.Identifier).text || text(member.name),
190
+ signature: renderFnSig(ts, member, content, source),
191
+ line: lineOf(member),
192
+ });
193
+ } else if (ts.isConstructorDeclaration(member)) {
194
+ members.push({
195
+ kind: "method",
196
+ name: "constructor",
197
+ signature: renderFnSig(ts, member, content, source),
198
+ line: lineOf(member),
199
+ });
200
+ } else if (ts.isPropertyDeclaration(member) && member.name) {
201
+ members.push({
202
+ kind: "property",
203
+ name: (member.name as TS.Identifier).text || text(member.name),
204
+ signature: member.type ? `: ${text(member.type)}` : "",
205
+ line: lineOf(member),
206
+ });
207
+ }
208
+ }
209
+ symbols.push({
210
+ kind: "class",
211
+ name: stmt.name.text,
212
+ signature: heritage,
213
+ line: lineOf(stmt),
214
+ exported,
215
+ members,
216
+ });
217
+ } else if (ts.isInterfaceDeclaration(stmt)) {
218
+ const heritage = (stmt.heritageClauses || [])
219
+ .flatMap((h) => h.types.map((t) => text(t.expression)))
220
+ .join(", ");
221
+ const members: SymbolEntry[] = [];
222
+ for (const member of stmt.members) {
223
+ if (ts.isPropertySignature(member) && member.name) {
224
+ members.push({
225
+ kind: "property",
226
+ name: (member.name as TS.Identifier).text || text(member.name),
227
+ signature: member.type ? `: ${text(member.type)}` : "",
228
+ line: lineOf(member),
229
+ });
230
+ } else if (ts.isMethodSignature(member) && member.name) {
231
+ members.push({
232
+ kind: "method",
233
+ name: (member.name as TS.Identifier).text || text(member.name),
234
+ signature: renderFnSig(ts, member, content, source),
235
+ line: lineOf(member),
236
+ });
237
+ }
238
+ }
239
+ symbols.push({
240
+ kind: "interface",
241
+ name: stmt.name.text,
242
+ signature: heritage ? `extends ${heritage}` : "",
243
+ line: lineOf(stmt),
244
+ exported,
245
+ members,
246
+ });
247
+ } else if (ts.isTypeAliasDeclaration(stmt)) {
248
+ const sigText = text(stmt.type);
249
+ symbols.push({
250
+ kind: "type",
251
+ name: stmt.name.text,
252
+ signature: `= ${sigText.slice(0, 80)}${sigText.length > 80 ? "…" : ""}`,
253
+ line: lineOf(stmt),
254
+ exported,
255
+ });
256
+ } else if (ts.isEnumDeclaration(stmt)) {
257
+ symbols.push({
258
+ kind: "enum",
259
+ name: stmt.name.text,
260
+ line: lineOf(stmt),
261
+ exported,
262
+ members: stmt.members.map((m) => ({
263
+ kind: "enumMember",
264
+ name: (m.name as TS.Identifier).text || text(m.name),
265
+ line: lineOf(m),
266
+ })),
267
+ });
268
+ } else if (ts.isVariableStatement(stmt)) {
269
+ const flags = stmt.declarationList.flags;
270
+ const isConst = (flags & ts.NodeFlags.Const) !== 0;
271
+ const isLet = (flags & ts.NodeFlags.Let) !== 0;
272
+ const kind = isConst ? "const" : isLet ? "let" : "var";
273
+ const stmtExported = isExported(stmt);
274
+ for (const decl of stmt.declarationList.declarations) {
275
+ if (!ts.isIdentifier(decl.name)) continue;
276
+ let sig = "";
277
+ if (decl.type) sig = `: ${text(decl.type)}`;
278
+ else if (
279
+ decl.initializer &&
280
+ (ts.isArrowFunction(decl.initializer) || ts.isFunctionExpression(decl.initializer))
281
+ ) {
282
+ sig = renderFnSig(ts, decl.initializer, content, source);
283
+ }
284
+ symbols.push({
285
+ kind,
286
+ name: decl.name.text,
287
+ signature: sig,
288
+ line: lineOf(decl),
289
+ exported: stmtExported,
290
+ });
291
+ }
292
+ } else if (ts.isExportDeclaration(stmt)) {
293
+ if (stmt.exportClause && ts.isNamedExports(stmt.exportClause)) {
294
+ const names = stmt.exportClause.elements.map((e) => e.name.text).join(", ");
295
+ const fromText = stmt.moduleSpecifier
296
+ ? ` from "${(stmt.moduleSpecifier as TS.StringLiteral).text}"`
297
+ : "";
298
+ symbols.push({
299
+ kind: "re-export",
300
+ name: `{${names}}${fromText}`,
301
+ line: lineOf(stmt),
302
+ exported: true,
303
+ });
304
+ } else if (stmt.moduleSpecifier) {
305
+ symbols.push({
306
+ kind: "re-export",
307
+ name: `* from "${(stmt.moduleSpecifier as TS.StringLiteral).text}"`,
308
+ line: lineOf(stmt),
309
+ exported: true,
310
+ });
311
+ }
312
+ } else if (ts.isModuleDeclaration(stmt) && stmt.name) {
313
+ symbols.push({
314
+ kind: "namespace",
315
+ name: (stmt.name as TS.Identifier).text || text(stmt.name),
316
+ line: lineOf(stmt),
317
+ exported,
318
+ });
319
+ }
320
+ }
321
+
322
+ return { imports, symbols };
323
+ }
324
+
325
+ function renderFnSig(
326
+ ts: typeof TS,
327
+ fn: TS.FunctionLikeDeclarationBase | TS.MethodSignature,
328
+ content: string,
329
+ source: TS.SourceFile,
330
+ ): string {
331
+ const params = (fn.parameters || [])
332
+ .map((p) => {
333
+ const name = ts.isIdentifier(p.name)
334
+ ? p.name.text
335
+ : content.slice(p.name.getStart(source), p.name.getEnd());
336
+ const type = p.type
337
+ ? content.slice(p.type.getStart(source), p.type.getEnd()).replace(/\s+/g, " ").trim()
338
+ : "";
339
+ const optional = p.questionToken ? "?" : "";
340
+ return type ? `${name}${optional}: ${type}` : `${name}${optional}`;
341
+ })
342
+ .join(", ");
343
+ const ret = fn.type
344
+ ? `: ${content.slice(fn.type.getStart(source), fn.type.getEnd()).replace(/\s+/g, " ").trim()}`
345
+ : "";
346
+ return `(${params})${ret}`;
347
+ }
348
+
349
+ function outlinePhp(content: string): { imports: string[]; symbols: SymbolEntry[] } {
350
+ const lines = content.split("\n");
351
+ const imports: string[] = [];
352
+ const symbols: SymbolEntry[] = [];
353
+ let currentContainer: SymbolEntry | null = null;
354
+
355
+ for (let i = 0; i < lines.length; i++) {
356
+ const line = lines[i];
357
+ const lineNum = i + 1;
358
+ const trimmed = line.trim();
359
+ const isIndented = /^[ \t]/.test(line);
360
+
361
+ if (
362
+ trimmed === "" ||
363
+ trimmed.startsWith("//") ||
364
+ trimmed.startsWith("#") ||
365
+ trimmed.startsWith("*") ||
366
+ trimmed.startsWith("/*")
367
+ )
368
+ continue;
369
+
370
+ const nsMatch = trimmed.match(/^namespace\s+([\w\\]+)\s*;/);
371
+ if (nsMatch && !isIndented) {
372
+ symbols.push({ kind: "namespace", name: nsMatch[1], line: lineNum });
373
+ continue;
374
+ }
375
+ const useMatch = trimmed.match(/^use\s+([\w\\]+)(?:\s+as\s+(\w+))?\s*;/);
376
+ if (useMatch && !isIndented) {
377
+ imports.push(useMatch[2] ? `${useMatch[1]} as ${useMatch[2]}` : useMatch[1]);
378
+ continue;
379
+ }
380
+ if (!isIndented) {
381
+ const classMatch = trimmed.match(
382
+ /^(?:abstract\s+|final\s+)?class\s+(\w+)(?:\s+extends\s+(\w+))?(?:\s+implements\s+([\w,\s]+))?/,
383
+ );
384
+ if (classMatch) {
385
+ currentContainer = {
386
+ kind: "class",
387
+ name: classMatch[1],
388
+ signature: [
389
+ classMatch[2] ? `extends ${classMatch[2]}` : "",
390
+ classMatch[3] ? `implements ${classMatch[3].trim()}` : "",
391
+ ]
392
+ .filter(Boolean)
393
+ .join(" "),
394
+ line: lineNum,
395
+ members: [],
396
+ };
397
+ symbols.push(currentContainer);
398
+ continue;
399
+ }
400
+ const interfaceMatch = trimmed.match(/^interface\s+(\w+)/);
401
+ if (interfaceMatch) {
402
+ currentContainer = {
403
+ kind: "interface",
404
+ name: interfaceMatch[1],
405
+ line: lineNum,
406
+ members: [],
407
+ };
408
+ symbols.push(currentContainer);
409
+ continue;
410
+ }
411
+ const traitMatch = trimmed.match(/^trait\s+(\w+)/);
412
+ if (traitMatch) {
413
+ currentContainer = { kind: "trait", name: traitMatch[1], line: lineNum, members: [] };
414
+ symbols.push(currentContainer);
415
+ continue;
416
+ }
417
+ }
418
+ const fnMatch = trimmed.match(
419
+ /^(?:public\s+|private\s+|protected\s+)?(?:static\s+)?function\s+(\w+)\s*\(([^)]*)\)/,
420
+ );
421
+ if (fnMatch) {
422
+ const entry: SymbolEntry = {
423
+ kind: isIndented && currentContainer ? "method" : "function",
424
+ name: fnMatch[1],
425
+ signature: `(${fnMatch[2].trim()})`,
426
+ line: lineNum,
427
+ };
428
+ if (isIndented && currentContainer) {
429
+ currentContainer.members!.push(entry);
430
+ } else {
431
+ symbols.push(entry);
432
+ }
433
+ }
434
+ }
435
+
436
+ return { imports, symbols };
437
+ }
438
+
439
+ function outlinePython(content: string): { imports: string[]; symbols: SymbolEntry[] } {
440
+ const lines = content.split("\n");
441
+ const imports: string[] = [];
442
+ const symbols: SymbolEntry[] = [];
443
+ let currentClass: SymbolEntry | null = null;
444
+
445
+ for (let i = 0; i < lines.length; i++) {
446
+ const line = lines[i];
447
+ const lineNum = i + 1;
448
+ const trimmed = line.trim();
449
+ const indent = line.match(/^ */)![0].length;
450
+
451
+ if (trimmed === "" || trimmed.startsWith("#")) continue;
452
+
453
+ const importMatch = trimmed.match(/^import\s+(.+)$/);
454
+ const fromMatch = trimmed.match(/^from\s+(\S+)\s+import\s+(.+)$/);
455
+ if (importMatch && indent === 0) {
456
+ imports.push(importMatch[1]);
457
+ continue;
458
+ }
459
+ if (fromMatch && indent === 0) {
460
+ imports.push(`${fromMatch[2].trim()} from ${fromMatch[1]}`);
461
+ continue;
462
+ }
463
+
464
+ const classMatch = trimmed.match(/^class\s+(\w+)(?:\(([^)]*)\))?:/);
465
+ if (classMatch && indent === 0) {
466
+ currentClass = {
467
+ kind: "class",
468
+ name: classMatch[1],
469
+ signature: classMatch[2] ? `(${classMatch[2]})` : "",
470
+ line: lineNum,
471
+ members: [],
472
+ };
473
+ symbols.push(currentClass);
474
+ continue;
475
+ }
476
+
477
+ const methodMatch = trimmed.match(/^(?:async\s+)?def\s+(\w+)\s*\(([^)]*)\)/);
478
+ if (methodMatch) {
479
+ if (indent === 0) {
480
+ symbols.push({
481
+ kind: "function",
482
+ name: methodMatch[1],
483
+ signature: `(${methodMatch[2].trim()})`,
484
+ line: lineNum,
485
+ });
486
+ currentClass = null;
487
+ } else if (currentClass) {
488
+ currentClass.members!.push({
489
+ kind: "method",
490
+ name: methodMatch[1],
491
+ signature: `(${methodMatch[2].trim()})`,
492
+ line: lineNum,
493
+ });
494
+ }
495
+ continue;
496
+ }
497
+
498
+ if (indent === 0 && trimmed.length > 0 && currentClass) {
499
+ currentClass = null;
500
+ }
501
+ }
502
+
503
+ return { imports, symbols };
504
+ }
505
+
506
+ function renderOutline(r: OutlineResult, opts: OutlineOpts): string {
507
+ const lines: string[] = [];
508
+ const symbolCount = r.symbols.length;
509
+ lines.push(
510
+ `outline · ${r.file} (${r.language}, ${r.total_lines} lines, ${symbolCount} top-level symbol${symbolCount === 1 ? "" : "s"})`,
511
+ );
512
+
513
+ if (opts.imports !== false && r.imports.length > 0) {
514
+ lines.push("");
515
+ if (r.imports.length <= 4) {
516
+ lines.push(`imports: ${r.imports.join(" · ")}`);
517
+ } else {
518
+ lines.push(`imports (${r.imports.length}):`);
519
+ for (const imp of r.imports) lines.push(` ${imp}`);
520
+ }
521
+ }
522
+
523
+ if (r.symbols.length > 0) {
524
+ lines.push("");
525
+ for (const sym of r.symbols) {
526
+ lines.push(renderSymbol(sym, 0, opts));
527
+ }
528
+ } else {
529
+ lines.push("");
530
+ lines.push("(no top-level symbols)");
531
+ }
532
+
533
+ return lines.join("\n");
534
+ }
535
+
536
+ function renderSymbol(s: SymbolEntry, depth: number, opts: OutlineOpts): string {
537
+ const indent = " ".repeat(depth);
538
+ const lineMark = `L${s.line}`.padEnd(6);
539
+ const exp = s.exported ? "★ " : " ";
540
+ const kindAbbr = kindToAbbr(s.kind);
541
+ const sig = s.signature ? ` ${s.signature}` : "";
542
+ const head = `${indent}${exp}${lineMark} ${kindAbbr} ${s.name}${sig}`;
543
+
544
+ if (opts.members !== false && s.members && s.members.length > 0) {
545
+ return [head, ...s.members.map((m) => renderSymbol(m, depth + 1, opts))].join("\n");
546
+ }
547
+ return head;
548
+ }
549
+
550
+ function kindToAbbr(kind: string): string {
551
+ switch (kind) {
552
+ case "function":
553
+ return "fn ";
554
+ case "class":
555
+ return "cl ";
556
+ case "interface":
557
+ return "if ";
558
+ case "type":
559
+ return "ty ";
560
+ case "const":
561
+ return "ct ";
562
+ case "let":
563
+ return "let ";
564
+ case "var":
565
+ return "var ";
566
+ case "enum":
567
+ return "en ";
568
+ case "enumMember":
569
+ return "em ";
570
+ case "method":
571
+ return "mt ";
572
+ case "property":
573
+ return "pr ";
574
+ case "namespace":
575
+ return "ns ";
576
+ case "trait":
577
+ return "tr ";
578
+ case "re-export":
579
+ return "rx ";
580
+ default:
581
+ return kind.slice(0, 4).padEnd(4);
582
+ }
583
+ }