@renxqoo/renx-code 0.0.9 → 0.0.12

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 (286) hide show
  1. package/README.md +31 -143
  2. package/bin/renx.cjs +79 -42
  3. package/bin/renx.exe +0 -0
  4. package/package.json +10 -28
  5. package/src/App.tsx +0 -297
  6. package/src/agent/runtime/event-format.ts +0 -258
  7. package/src/agent/runtime/model-types.ts +0 -13
  8. package/src/agent/runtime/runtime.context-usage.test.ts +0 -192
  9. package/src/agent/runtime/runtime.error-handling.test.ts +0 -235
  10. package/src/agent/runtime/runtime.simple.test.ts +0 -16
  11. package/src/agent/runtime/runtime.test.ts +0 -296
  12. package/src/agent/runtime/runtime.ts +0 -875
  13. package/src/agent/runtime/runtime.usage-forwarding.test.ts +0 -228
  14. package/src/agent/runtime/source-modules.test.ts +0 -38
  15. package/src/agent/runtime/source-modules.ts +0 -370
  16. package/src/agent/runtime/tool-call-buffer.test.ts +0 -65
  17. package/src/agent/runtime/tool-call-buffer.ts +0 -60
  18. package/src/agent/runtime/tool-confirmation.test.ts +0 -56
  19. package/src/agent/runtime/tool-confirmation.ts +0 -15
  20. package/src/agent/runtime/types.ts +0 -99
  21. package/src/commands/slash-commands.test.ts +0 -216
  22. package/src/commands/slash-commands.ts +0 -64
  23. package/src/components/chat/assistant-reply.test.tsx +0 -47
  24. package/src/components/chat/assistant-reply.tsx +0 -136
  25. package/src/components/chat/assistant-segment.test.ts +0 -99
  26. package/src/components/chat/assistant-segment.tsx +0 -125
  27. package/src/components/chat/assistant-tool-group.tsx +0 -900
  28. package/src/components/chat/code-block.test.tsx +0 -206
  29. package/src/components/chat/code-block.tsx +0 -313
  30. package/src/components/chat/prompt-card.tsx +0 -81
  31. package/src/components/chat/segment-groups.test.ts +0 -52
  32. package/src/components/chat/segment-groups.ts +0 -106
  33. package/src/components/chat/turn-item.tsx +0 -39
  34. package/src/components/conversation-panel.tsx +0 -43
  35. package/src/components/file-mention-menu.tsx +0 -77
  36. package/src/components/file-picker-dialog.tsx +0 -206
  37. package/src/components/footer-hints.tsx +0 -75
  38. package/src/components/model-picker-dialog.tsx +0 -248
  39. package/src/components/prompt.tsx +0 -233
  40. package/src/components/slash-command-menu.tsx +0 -65
  41. package/src/components/tool-confirm-dialog-content.test.ts +0 -103
  42. package/src/components/tool-confirm-dialog-content.ts +0 -186
  43. package/src/components/tool-confirm-dialog.tsx +0 -187
  44. package/src/components/tool-display-config.ts +0 -119
  45. package/src/context-usage-regressions.test.ts +0 -26
  46. package/src/files/attachment-capabilities.test.ts +0 -30
  47. package/src/files/attachment-capabilities.ts +0 -50
  48. package/src/files/attachment-content.ts +0 -153
  49. package/src/files/file-mention-query.test.ts +0 -34
  50. package/src/files/file-mention-query.ts +0 -32
  51. package/src/files/prompt-display.ts +0 -13
  52. package/src/files/types.ts +0 -5
  53. package/src/files/workspace-files.ts +0 -61
  54. package/src/hooks/agent-event-handlers.test.ts +0 -207
  55. package/src/hooks/agent-event-handlers.ts +0 -196
  56. package/src/hooks/chat-local-replies.fixed.test.ts +0 -119
  57. package/src/hooks/chat-local-replies.test.ts +0 -153
  58. package/src/hooks/chat-local-replies.ts +0 -63
  59. package/src/hooks/turn-updater.test.ts +0 -70
  60. package/src/hooks/turn-updater.ts +0 -166
  61. package/src/hooks/use-agent-chat.context.test.ts +0 -10
  62. package/src/hooks/use-agent-chat.status.test.ts +0 -14
  63. package/src/hooks/use-agent-chat.test.ts +0 -80
  64. package/src/hooks/use-agent-chat.ts +0 -621
  65. package/src/hooks/use-file-mention-menu.ts +0 -196
  66. package/src/hooks/use-file-picker.ts +0 -185
  67. package/src/hooks/use-model-picker.ts +0 -196
  68. package/src/hooks/use-slash-command-menu.ts +0 -154
  69. package/src/index.tsx +0 -55
  70. package/src/runtime/clipboard.test.ts +0 -43
  71. package/src/runtime/clipboard.ts +0 -89
  72. package/src/runtime/exit.test.ts +0 -177
  73. package/src/runtime/exit.ts +0 -98
  74. package/src/runtime/runtime-support.test.ts +0 -31
  75. package/src/runtime/terminal-theme.test.ts +0 -55
  76. package/src/runtime/terminal-theme.ts +0 -196
  77. package/src/types/chat.ts +0 -32
  78. package/src/types/message-content.ts +0 -48
  79. package/src/ui/open-code-theme.ts +0 -176
  80. package/src/ui/opencode-markdown.ts +0 -211
  81. package/src/ui/theme.simple.test.ts +0 -52
  82. package/src/ui/theme.test.ts +0 -151
  83. package/src/ui/theme.ts +0 -152
  84. package/src/utils/time.test.ts +0 -144
  85. package/src/utils/time.ts +0 -7
  86. package/tsconfig.json +0 -30
  87. package/vendor/agent-root/src/agent/ENTERPRISE_ACCEPTANCE_CHECKLIST.md +0 -95
  88. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.html +0 -1345
  89. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.md +0 -1353
  90. package/vendor/agent-root/src/agent/ERROR_CONTRACT.md +0 -60
  91. package/vendor/agent-root/src/agent/TEST_COVERAGE_ANALYSIS.md +0 -278
  92. package/vendor/agent-root/src/agent/__test__/error-contract.test.ts +0 -72
  93. package/vendor/agent-root/src/agent/__test__/types.test.ts +0 -137
  94. package/vendor/agent-root/src/agent/agent/__test__/abort-runtime.test.ts +0 -83
  95. package/vendor/agent-root/src/agent/agent/__test__/callback-safety.test.ts +0 -34
  96. package/vendor/agent-root/src/agent/agent/__test__/compaction.test.ts +0 -323
  97. package/vendor/agent-root/src/agent/agent/__test__/concurrency.test.ts +0 -290
  98. package/vendor/agent-root/src/agent/agent/__test__/error-normalizer.test.ts +0 -377
  99. package/vendor/agent-root/src/agent/agent/__test__/error.test.ts +0 -212
  100. package/vendor/agent-root/src/agent/agent/__test__/fault-injection.test.ts +0 -295
  101. package/vendor/agent-root/src/agent/agent/__test__/index.test.ts +0 -3607
  102. package/vendor/agent-root/src/agent/agent/__test__/logger.test.ts +0 -35
  103. package/vendor/agent-root/src/agent/agent/__test__/message-utils.test.ts +0 -517
  104. package/vendor/agent-root/src/agent/agent/__test__/telemetry.test.ts +0 -97
  105. package/vendor/agent-root/src/agent/agent/__test__/timeout-budget.test.ts +0 -479
  106. package/vendor/agent-root/src/agent/agent/__test__/tool-call-merge.test.ts +0 -80
  107. package/vendor/agent-root/src/agent/agent/__test__/tool-execution-ledger.test.ts +0 -76
  108. package/vendor/agent-root/src/agent/agent/__test__/write-buffer.test.ts +0 -173
  109. package/vendor/agent-root/src/agent/agent/__test__/write-file-session.test.ts +0 -109
  110. package/vendor/agent-root/src/agent/agent/abort-runtime.ts +0 -71
  111. package/vendor/agent-root/src/agent/agent/callback-safety.ts +0 -33
  112. package/vendor/agent-root/src/agent/agent/compaction.ts +0 -291
  113. package/vendor/agent-root/src/agent/agent/concurrency.ts +0 -103
  114. package/vendor/agent-root/src/agent/agent/error-normalizer.ts +0 -190
  115. package/vendor/agent-root/src/agent/agent/error.ts +0 -198
  116. package/vendor/agent-root/src/agent/agent/index.ts +0 -1772
  117. package/vendor/agent-root/src/agent/agent/logger.ts +0 -65
  118. package/vendor/agent-root/src/agent/agent/message-utils.ts +0 -101
  119. package/vendor/agent-root/src/agent/agent/stream-events.ts +0 -61
  120. package/vendor/agent-root/src/agent/agent/telemetry.ts +0 -123
  121. package/vendor/agent-root/src/agent/agent/timeout-budget.ts +0 -227
  122. package/vendor/agent-root/src/agent/agent/tool-call-merge.ts +0 -111
  123. package/vendor/agent-root/src/agent/agent/tool-execution-ledger.ts +0 -164
  124. package/vendor/agent-root/src/agent/agent/write-buffer.ts +0 -188
  125. package/vendor/agent-root/src/agent/agent/write-file-session.ts +0 -238
  126. package/vendor/agent-root/src/agent/app/__test__/agent-app-service.test.ts +0 -1053
  127. package/vendor/agent-root/src/agent/app/__test__/minimal-agent-application.test.ts +0 -158
  128. package/vendor/agent-root/src/agent/app/__test__/sqlite-agent-app-store.test.ts +0 -437
  129. package/vendor/agent-root/src/agent/app/agent-app-service.ts +0 -748
  130. package/vendor/agent-root/src/agent/app/contracts.ts +0 -109
  131. package/vendor/agent-root/src/agent/app/index.ts +0 -5
  132. package/vendor/agent-root/src/agent/app/minimal-agent-application.ts +0 -151
  133. package/vendor/agent-root/src/agent/app/ports.ts +0 -72
  134. package/vendor/agent-root/src/agent/app/sqlite-agent-app-store.ts +0 -1182
  135. package/vendor/agent-root/src/agent/app/sqlite-client.ts +0 -177
  136. package/vendor/agent-root/src/agent/docs/cli-app-layer/00-README.md +0 -36
  137. package/vendor/agent-root/src/agent/docs/cli-app-layer/01-scope-and-goals.md +0 -33
  138. package/vendor/agent-root/src/agent/docs/cli-app-layer/02-architecture-overview.md +0 -40
  139. package/vendor/agent-root/src/agent/docs/cli-app-layer/03-domain-model-and-contracts.md +0 -91
  140. package/vendor/agent-root/src/agent/docs/cli-app-layer/04-ports-and-interfaces.md +0 -116
  141. package/vendor/agent-root/src/agent/docs/cli-app-layer/05-run-orchestration-and-state-machine.md +0 -52
  142. package/vendor/agent-root/src/agent/docs/cli-app-layer/06-cli-commands-and-ux.md +0 -53
  143. package/vendor/agent-root/src/agent/docs/cli-app-layer/07-storage-design-local.md +0 -52
  144. package/vendor/agent-root/src/agent/docs/cli-app-layer/08-error-and-observability.md +0 -40
  145. package/vendor/agent-root/src/agent/docs/cli-app-layer/09-security-and-policy-boundary.md +0 -19
  146. package/vendor/agent-root/src/agent/docs/cli-app-layer/10-test-plan-and-acceptance.md +0 -28
  147. package/vendor/agent-root/src/agent/docs/cli-app-layer/11-implementation-phases.md +0 -26
  148. package/vendor/agent-root/src/agent/docs/cli-app-layer/12-open-questions-and-risks.md +0 -30
  149. package/vendor/agent-root/src/agent/docs/cli-app-layer/13-sqlite-schema-fields-and-rationale.md +0 -567
  150. package/vendor/agent-root/src/agent/docs/cli-app-layer/14-project-flow-mermaid.md +0 -583
  151. package/vendor/agent-root/src/agent/docs/cli-app-layer/15-openclaw-style-project-blueprint.md +0 -972
  152. package/vendor/agent-root/src/agent/error-contract.ts +0 -154
  153. package/vendor/agent-root/src/agent/prompts/system.ts +0 -246
  154. package/vendor/agent-root/src/agent/prompts/system1.ts +0 -208
  155. package/vendor/agent-root/src/agent/storage/__test__/file-history-store.test.ts +0 -98
  156. package/vendor/agent-root/src/agent/storage/file-history-store.ts +0 -313
  157. package/vendor/agent-root/src/agent/storage/file-storage-config.ts +0 -94
  158. package/vendor/agent-root/src/agent/storage/file-system.ts +0 -31
  159. package/vendor/agent-root/src/agent/storage/file-write-service.ts +0 -21
  160. package/vendor/agent-root/src/agent/tool/__test__/base-tool.test.ts +0 -413
  161. package/vendor/agent-root/src/agent/tool/__test__/bash-policy.test.ts +0 -356
  162. package/vendor/agent-root/src/agent/tool/__test__/bash.mocked-coverage.test.ts +0 -375
  163. package/vendor/agent-root/src/agent/tool/__test__/bash.test.ts +0 -372
  164. package/vendor/agent-root/src/agent/tool/__test__/error.test.ts +0 -108
  165. package/vendor/agent-root/src/agent/tool/__test__/file-edit-tool.test.ts +0 -258
  166. package/vendor/agent-root/src/agent/tool/__test__/file-history-tools.test.ts +0 -121
  167. package/vendor/agent-root/src/agent/tool/__test__/file-read-tool.test.ts +0 -210
  168. package/vendor/agent-root/src/agent/tool/__test__/glob.test.ts +0 -139
  169. package/vendor/agent-root/src/agent/tool/__test__/grep.mocked-coverage.test.ts +0 -456
  170. package/vendor/agent-root/src/agent/tool/__test__/grep.test.ts +0 -192
  171. package/vendor/agent-root/src/agent/tool/__test__/lsp.test.ts +0 -300
  172. package/vendor/agent-root/src/agent/tool/__test__/outside-workspace-confirmation.test.ts +0 -214
  173. package/vendor/agent-root/src/agent/tool/__test__/path-security.test.ts +0 -336
  174. package/vendor/agent-root/src/agent/tool/__test__/skill-loader.test.ts +0 -494
  175. package/vendor/agent-root/src/agent/tool/__test__/skill-parser.test.ts +0 -543
  176. package/vendor/agent-root/src/agent/tool/__test__/skill-tool.test.ts +0 -172
  177. package/vendor/agent-root/src/agent/tool/__test__/task-concurrency-and-version.test.ts +0 -116
  178. package/vendor/agent-root/src/agent/tool/__test__/task-create-get-list-update.test.ts +0 -267
  179. package/vendor/agent-root/src/agent/tool/__test__/task-create.test.ts +0 -519
  180. package/vendor/agent-root/src/agent/tool/__test__/task-errors.test.ts +0 -225
  181. package/vendor/agent-root/src/agent/tool/__test__/task-output-blocking.test.ts +0 -223
  182. package/vendor/agent-root/src/agent/tool/__test__/task-output.test.ts +0 -184
  183. package/vendor/agent-root/src/agent/tool/__test__/task-parent-abort.test.ts +0 -287
  184. package/vendor/agent-root/src/agent/tool/__test__/task-real-runner-adapter.test.ts +0 -190
  185. package/vendor/agent-root/src/agent/tool/__test__/task-run-lifecycle.test.ts +0 -352
  186. package/vendor/agent-root/src/agent/tool/__test__/task-store-runner-branches.test.ts +0 -395
  187. package/vendor/agent-root/src/agent/tool/__test__/task-store.test.ts +0 -391
  188. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config-integration.test.ts +0 -176
  189. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config.test.ts +0 -68
  190. package/vendor/agent-root/src/agent/tool/__test__/task-tools-core-edges.test.ts +0 -630
  191. package/vendor/agent-root/src/agent/tool/__test__/task-tools-runtime-edges.test.ts +0 -732
  192. package/vendor/agent-root/src/agent/tool/__test__/task-types.test.ts +0 -494
  193. package/vendor/agent-root/src/agent/tool/__test__/task-utils-branches.test.ts +0 -175
  194. package/vendor/agent-root/src/agent/tool/__test__/tool-manager.test.ts +0 -505
  195. package/vendor/agent-root/src/agent/tool/__test__/types.test.ts +0 -55
  196. package/vendor/agent-root/src/agent/tool/__test__/web-fetch.test.ts +0 -244
  197. package/vendor/agent-root/src/agent/tool/__test__/web-search.test.ts +0 -290
  198. package/vendor/agent-root/src/agent/tool/__test__/write-file.test.ts +0 -368
  199. package/vendor/agent-root/src/agent/tool/base-tool.ts +0 -345
  200. package/vendor/agent-root/src/agent/tool/bash-policy.ts +0 -636
  201. package/vendor/agent-root/src/agent/tool/bash.ts +0 -688
  202. package/vendor/agent-root/src/agent/tool/error.ts +0 -131
  203. package/vendor/agent-root/src/agent/tool/file-edit-tool.ts +0 -264
  204. package/vendor/agent-root/src/agent/tool/file-history-list.ts +0 -103
  205. package/vendor/agent-root/src/agent/tool/file-history-restore.ts +0 -149
  206. package/vendor/agent-root/src/agent/tool/file-read-tool.ts +0 -211
  207. package/vendor/agent-root/src/agent/tool/glob.ts +0 -171
  208. package/vendor/agent-root/src/agent/tool/grep.ts +0 -496
  209. package/vendor/agent-root/src/agent/tool/lsp.ts +0 -481
  210. package/vendor/agent-root/src/agent/tool/path-security.ts +0 -117
  211. package/vendor/agent-root/src/agent/tool/search/common.ts +0 -153
  212. package/vendor/agent-root/src/agent/tool/skill/index.ts +0 -13
  213. package/vendor/agent-root/src/agent/tool/skill/loader.ts +0 -229
  214. package/vendor/agent-root/src/agent/tool/skill/parser.ts +0 -124
  215. package/vendor/agent-root/src/agent/tool/skill/types.ts +0 -27
  216. package/vendor/agent-root/src/agent/tool/skill-tool.ts +0 -143
  217. package/vendor/agent-root/src/agent/tool/task-create.ts +0 -186
  218. package/vendor/agent-root/src/agent/tool/task-errors.ts +0 -42
  219. package/vendor/agent-root/src/agent/tool/task-get.ts +0 -116
  220. package/vendor/agent-root/src/agent/tool/task-graph.ts +0 -78
  221. package/vendor/agent-root/src/agent/tool/task-list.ts +0 -141
  222. package/vendor/agent-root/src/agent/tool/task-mock-runner-adapter.ts +0 -232
  223. package/vendor/agent-root/src/agent/tool/task-output.ts +0 -223
  224. package/vendor/agent-root/src/agent/tool/task-parent-abort.ts +0 -115
  225. package/vendor/agent-root/src/agent/tool/task-real-runner-adapter.ts +0 -336
  226. package/vendor/agent-root/src/agent/tool/task-runner-adapter.ts +0 -55
  227. package/vendor/agent-root/src/agent/tool/task-stop.ts +0 -187
  228. package/vendor/agent-root/src/agent/tool/task-store.ts +0 -217
  229. package/vendor/agent-root/src/agent/tool/task-subagent-config.ts +0 -149
  230. package/vendor/agent-root/src/agent/tool/task-types.ts +0 -264
  231. package/vendor/agent-root/src/agent/tool/task-update.ts +0 -315
  232. package/vendor/agent-root/src/agent/tool/task.ts +0 -209
  233. package/vendor/agent-root/src/agent/tool/tool-manager.ts +0 -361
  234. package/vendor/agent-root/src/agent/tool/tool-prompts.ts +0 -242
  235. package/vendor/agent-root/src/agent/tool/types.ts +0 -116
  236. package/vendor/agent-root/src/agent/tool/web-fetch.ts +0 -227
  237. package/vendor/agent-root/src/agent/tool/web-search.ts +0 -208
  238. package/vendor/agent-root/src/agent/tool/write-file.ts +0 -497
  239. package/vendor/agent-root/src/agent/types.ts +0 -232
  240. package/vendor/agent-root/src/agent/utils/__tests__/index.test.ts +0 -18
  241. package/vendor/agent-root/src/agent/utils/__tests__/message-utils.test.ts +0 -610
  242. package/vendor/agent-root/src/agent/utils/__tests__/message.test.ts +0 -223
  243. package/vendor/agent-root/src/agent/utils/__tests__/token.test.ts +0 -42
  244. package/vendor/agent-root/src/agent/utils/index.ts +0 -16
  245. package/vendor/agent-root/src/agent/utils/message.ts +0 -171
  246. package/vendor/agent-root/src/agent/utils/token.ts +0 -28
  247. package/vendor/agent-root/src/config/__tests__/load-config-to-env.test.ts +0 -238
  248. package/vendor/agent-root/src/config/__tests__/loader.test.ts +0 -361
  249. package/vendor/agent-root/src/config/__tests__/runtime.test.ts +0 -88
  250. package/vendor/agent-root/src/config/index.ts +0 -55
  251. package/vendor/agent-root/src/config/loader.ts +0 -494
  252. package/vendor/agent-root/src/config/paths.ts +0 -30
  253. package/vendor/agent-root/src/config/runtime.ts +0 -163
  254. package/vendor/agent-root/src/config/types.ts +0 -96
  255. package/vendor/agent-root/src/logger/index.ts +0 -57
  256. package/vendor/agent-root/src/logger/logger.ts +0 -819
  257. package/vendor/agent-root/src/logger/types.ts +0 -150
  258. package/vendor/agent-root/src/providers/__tests__/errors.test.ts +0 -441
  259. package/vendor/agent-root/src/providers/__tests__/index.test.ts +0 -16
  260. package/vendor/agent-root/src/providers/__tests__/openai-compatible.options.test.ts +0 -318
  261. package/vendor/agent-root/src/providers/__tests__/openai-compatible.test.ts +0 -600
  262. package/vendor/agent-root/src/providers/__tests__/registry.test.ts +0 -523
  263. package/vendor/agent-root/src/providers/__tests__/responses-adapter.test.ts +0 -298
  264. package/vendor/agent-root/src/providers/adapters/__tests__/anthropic.test.ts +0 -354
  265. package/vendor/agent-root/src/providers/adapters/__tests__/kimi.test.ts +0 -58
  266. package/vendor/agent-root/src/providers/adapters/__tests__/standard.test.ts +0 -261
  267. package/vendor/agent-root/src/providers/adapters/anthropic.ts +0 -572
  268. package/vendor/agent-root/src/providers/adapters/base.ts +0 -131
  269. package/vendor/agent-root/src/providers/adapters/kimi.ts +0 -48
  270. package/vendor/agent-root/src/providers/adapters/responses.ts +0 -732
  271. package/vendor/agent-root/src/providers/adapters/standard.ts +0 -120
  272. package/vendor/agent-root/src/providers/http/__tests__/client.timeout.test.ts +0 -313
  273. package/vendor/agent-root/src/providers/http/client.ts +0 -289
  274. package/vendor/agent-root/src/providers/http/stream-parser.ts +0 -109
  275. package/vendor/agent-root/src/providers/index.ts +0 -76
  276. package/vendor/agent-root/src/providers/kimi-headers.ts +0 -177
  277. package/vendor/agent-root/src/providers/openai-compatible.ts +0 -387
  278. package/vendor/agent-root/src/providers/registry/model-config.ts +0 -477
  279. package/vendor/agent-root/src/providers/registry/provider-factory.ts +0 -127
  280. package/vendor/agent-root/src/providers/registry.ts +0 -135
  281. package/vendor/agent-root/src/providers/types/api.ts +0 -284
  282. package/vendor/agent-root/src/providers/types/config.ts +0 -58
  283. package/vendor/agent-root/src/providers/types/errors.ts +0 -323
  284. package/vendor/agent-root/src/providers/types/index.ts +0 -72
  285. package/vendor/agent-root/src/providers/types/provider.ts +0 -45
  286. package/vendor/agent-root/src/providers/types/registry.ts +0 -68
@@ -1,494 +0,0 @@
1
- import { describe, expect, it, beforeEach, afterEach } from 'vitest';
2
- import * as fs from 'node:fs/promises';
3
- import * as os from 'node:os';
4
- import * as path from 'node:path';
5
- import {
6
- SkillLoader,
7
- getSkillLoader,
8
- initializeSkillLoader,
9
- resetSkillLoader,
10
- } from '../skill/loader';
11
-
12
- describe('SkillLoader', () => {
13
- let tempDir: string;
14
- let skillDir: string;
15
- let skillFile: string;
16
-
17
- beforeEach(async () => {
18
- tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'skill-loader-test-'));
19
- skillDir = path.join(tempDir, 'test-skill');
20
- await fs.mkdir(skillDir, { recursive: true });
21
- skillFile = path.join(skillDir, 'SKILL.md');
22
-
23
- await fs.writeFile(
24
- skillFile,
25
- `---
26
- name: test-skill
27
- description: A test skill
28
- ---
29
- # Test Skill
30
- This is a test skill.`
31
- );
32
- });
33
-
34
- afterEach(async () => {
35
- await fs.rm(tempDir, { recursive: true, force: true });
36
- resetSkillLoader();
37
- });
38
-
39
- describe('constructor', () => {
40
- it('creates loader with default options', () => {
41
- const loader = new SkillLoader();
42
- expect(loader).toBeDefined();
43
- });
44
-
45
- it('creates loader with custom skill roots', () => {
46
- const loader = new SkillLoader({ skillRoots: [tempDir] });
47
- expect(loader).toBeDefined();
48
- });
49
-
50
- it('creates loader with custom working directory', () => {
51
- const loader = new SkillLoader({ workingDir: tempDir });
52
- expect(loader).toBeDefined();
53
- });
54
-
55
- it('creates loader with both custom options', () => {
56
- const loader = new SkillLoader({
57
- skillRoots: [tempDir],
58
- workingDir: tempDir,
59
- });
60
- expect(loader).toBeDefined();
61
- });
62
- });
63
-
64
- describe('initialize', () => {
65
- it('initializes loader and discovers skills', async () => {
66
- const loader = new SkillLoader({ skillRoots: [tempDir] });
67
- await loader.initialize();
68
-
69
- const metadata = loader.getAllMetadata();
70
- expect(metadata).toHaveLength(1);
71
- expect(metadata[0].name).toBe('test-skill');
72
- expect(metadata[0].description).toBe('A test skill');
73
- });
74
-
75
- it('does not reinitialize if already initialized', async () => {
76
- const loader = new SkillLoader({ skillRoots: [tempDir] });
77
- await loader.initialize();
78
-
79
- // Add another skill after initialization
80
- const skillDir2 = path.join(tempDir, 'test-skill-2');
81
- await fs.mkdir(skillDir2, { recursive: true });
82
- await fs.writeFile(
83
- path.join(skillDir2, 'SKILL.md'),
84
- `---
85
- name: test-skill-2
86
- description: Another test skill
87
- ---
88
- # Test Skill 2`
89
- );
90
-
91
- // Should not discover the new skill
92
- const metadata = loader.getAllMetadata();
93
- expect(metadata).toHaveLength(1);
94
- });
95
-
96
- it('handles empty skill roots', async () => {
97
- // When skillRoots is empty array, it should use default skill roots
98
- const loader = new SkillLoader({ skillRoots: [] });
99
- await loader.initialize();
100
-
101
- const metadata = loader.getAllMetadata();
102
- // Should discover skills from default roots (if any exist)
103
- expect(Array.isArray(metadata)).toBe(true);
104
- });
105
-
106
- it('handles non-existent skill roots', async () => {
107
- const loader = new SkillLoader({ skillRoots: ['/non/existent/path'] });
108
- await loader.initialize();
109
-
110
- const metadata = loader.getAllMetadata();
111
- expect(metadata).toHaveLength(0);
112
- });
113
-
114
- it('handles skill roots that are files', async () => {
115
- const filePath = path.join(tempDir, 'file.txt');
116
- await fs.writeFile(filePath, 'test');
117
-
118
- const loader = new SkillLoader({ skillRoots: [filePath] });
119
- await loader.initialize();
120
-
121
- const metadata = loader.getAllMetadata();
122
- expect(metadata).toHaveLength(0);
123
- });
124
-
125
- it('handles skill roots with permission errors', async () => {
126
- const restrictedDir = path.join(tempDir, 'restricted');
127
- await fs.mkdir(restrictedDir, { recursive: true });
128
- await fs.chmod(restrictedDir, 0o000);
129
-
130
- try {
131
- const loader = new SkillLoader({ skillRoots: [restrictedDir] });
132
- await loader.initialize();
133
-
134
- const metadata = loader.getAllMetadata();
135
- expect(metadata).toHaveLength(0);
136
- } finally {
137
- await fs.chmod(restrictedDir, 0o755);
138
- }
139
- });
140
-
141
- it('handles skill files with invalid frontmatter', async () => {
142
- const invalidSkillDir = path.join(tempDir, 'invalid-skill');
143
- await fs.mkdir(invalidSkillDir, { recursive: true });
144
- await fs.writeFile(
145
- path.join(invalidSkillDir, 'SKILL.md'),
146
- `---
147
- name: invalid-skill
148
- ---
149
- # Invalid Skill`
150
- );
151
-
152
- const loader = new SkillLoader({ skillRoots: [tempDir] });
153
- await loader.initialize();
154
-
155
- const metadata = loader.getAllMetadata();
156
- // When frontmatter is invalid, it uses fallback name (directory name) and derives description
157
- expect(metadata).toHaveLength(2); // Both skills
158
- expect(metadata.map((m) => m.name)).toContain('test-skill');
159
- expect(metadata.map((m) => m.name)).toContain('invalid-skill');
160
- });
161
-
162
- it('handles skill files with invalid names', async () => {
163
- const invalidNameDir = path.join(tempDir, 'Invalid-Name');
164
- await fs.mkdir(invalidNameDir, { recursive: true });
165
- await fs.writeFile(
166
- path.join(invalidNameDir, 'SKILL.md'),
167
- `---
168
- name: Invalid-Name
169
- description: Invalid name skill
170
- ---
171
- # Invalid Name Skill`
172
- );
173
-
174
- const loader = new SkillLoader({ skillRoots: [tempDir] });
175
- await loader.initialize();
176
-
177
- const metadata = loader.getAllMetadata();
178
- expect(metadata).toHaveLength(1); // Only the valid skill
179
- expect(metadata[0].name).toBe('test-skill');
180
- });
181
-
182
- it('handles skill files with empty descriptions', async () => {
183
- const emptyDescDir = path.join(tempDir, 'empty-desc');
184
- await fs.mkdir(emptyDescDir, { recursive: true });
185
- await fs.writeFile(
186
- path.join(emptyDescDir, 'SKILL.md'),
187
- `---
188
- name: empty-desc
189
- description:
190
- ---
191
- # Empty Description Skill`
192
- );
193
-
194
- const loader = new SkillLoader({ skillRoots: [tempDir] });
195
- await loader.initialize();
196
-
197
- const metadata = loader.getAllMetadata();
198
- // When description is empty, it derives description from markdown
199
- expect(metadata).toHaveLength(2); // Both skills
200
- expect(metadata.map((m) => m.name)).toContain('test-skill');
201
- expect(metadata.map((m) => m.name)).toContain('empty-desc');
202
- });
203
-
204
- it('handles skill files that cannot be read', async () => {
205
- // Skip on Windows as chmod doesn't work the same way
206
- if (process.platform === 'win32') {
207
- return;
208
- }
209
- const unreadableDir = path.join(tempDir, 'unreadable');
210
- await fs.mkdir(unreadableDir, { recursive: true });
211
- const unreadableFile = path.join(unreadableDir, 'SKILL.md');
212
- await fs.writeFile(
213
- unreadableFile,
214
- `---
215
- name: unreadable
216
- description: Unreadable skill
217
- ---
218
- # Unreadable Skill`
219
- );
220
- await fs.chmod(unreadableFile, 0o000);
221
-
222
- try {
223
- const loader = new SkillLoader({ skillRoots: [tempDir] });
224
- await loader.initialize();
225
-
226
- const metadata = loader.getAllMetadata();
227
- expect(metadata).toHaveLength(1); // Only the valid skill
228
- expect(metadata[0].name).toBe('test-skill');
229
- } finally {
230
- await fs.chmod(unreadableFile, 0o644);
231
- }
232
- });
233
-
234
- it('handles duplicate skill names', async () => {
235
- const duplicateDir = path.join(tempDir, 'duplicate');
236
- await fs.mkdir(duplicateDir, { recursive: true });
237
- await fs.writeFile(
238
- path.join(duplicateDir, 'SKILL.md'),
239
- `---
240
- name: test-skill
241
- description: Duplicate skill
242
- ---
243
- # Duplicate Skill`
244
- );
245
-
246
- const loader = new SkillLoader({ skillRoots: [tempDir] });
247
- await loader.initialize();
248
-
249
- const metadata = loader.getAllMetadata();
250
- expect(metadata).toHaveLength(1); // Only the first skill
251
- expect(metadata[0].name).toBe('test-skill');
252
- // The first discovered skill's description is kept
253
- // Note: The order of discovery depends on directory traversal order
254
- expect(['A test skill', 'Duplicate skill']).toContain(metadata[0].description);
255
- });
256
-
257
- it('handles nested skill directories', async () => {
258
- const nestedDir = path.join(tempDir, 'nested', 'deep', 'skill');
259
- await fs.mkdir(nestedDir, { recursive: true });
260
- await fs.writeFile(
261
- path.join(nestedDir, 'SKILL.md'),
262
- `---
263
- name: nested-skill
264
- description: Nested skill
265
- ---
266
- # Nested Skill`
267
- );
268
-
269
- const loader = new SkillLoader({ skillRoots: [tempDir] });
270
- await loader.initialize();
271
-
272
- const metadata = loader.getAllMetadata();
273
- expect(metadata).toHaveLength(2); // Both skills
274
- expect(metadata.map((m) => m.name)).toContain('test-skill');
275
- expect(metadata.map((m) => m.name)).toContain('nested-skill');
276
- });
277
-
278
- it('handles circular symlinks', async () => {
279
- const symlinkDir = path.join(tempDir, 'symlink');
280
- await fs.symlink(tempDir, symlinkDir);
281
-
282
- const loader = new SkillLoader({ skillRoots: [tempDir] });
283
- await loader.initialize();
284
-
285
- const metadata = loader.getAllMetadata();
286
- expect(metadata).toHaveLength(1); // Should not hang or crash
287
- });
288
- });
289
-
290
- describe('getAllMetadata', () => {
291
- it('returns empty array when no skills', async () => {
292
- // Create a new empty directory for this test
293
- const emptyDir = await fs.mkdtemp(path.join(os.tmpdir(), 'empty-skill-'));
294
- try {
295
- const loader = new SkillLoader({ skillRoots: [emptyDir] });
296
- await loader.initialize();
297
-
298
- const metadata = loader.getAllMetadata();
299
- expect(metadata).toHaveLength(0);
300
- } finally {
301
- await fs.rm(emptyDir, { recursive: true, force: true });
302
- }
303
- });
304
-
305
- it('returns skills sorted by name', async () => {
306
- // Create another skill
307
- const skillDir2 = path.join(tempDir, 'another-skill');
308
- await fs.mkdir(skillDir2, { recursive: true });
309
- await fs.writeFile(
310
- path.join(skillDir2, 'SKILL.md'),
311
- `---
312
- name: another-skill
313
- description: Another test skill
314
- ---
315
- # Another Test Skill`
316
- );
317
-
318
- const loader = new SkillLoader({ skillRoots: [tempDir] });
319
- await loader.initialize();
320
-
321
- const metadata = loader.getAllMetadata();
322
- expect(metadata).toHaveLength(2);
323
- expect(metadata[0].name).toBe('another-skill');
324
- expect(metadata[1].name).toBe('test-skill');
325
- });
326
- });
327
-
328
- describe('hasSkill', () => {
329
- it('returns true for existing skill', async () => {
330
- const loader = new SkillLoader({ skillRoots: [tempDir] });
331
- await loader.initialize();
332
-
333
- expect(loader.hasSkill('test-skill')).toBe(true);
334
- });
335
-
336
- it('returns false for non-existing skill', async () => {
337
- const loader = new SkillLoader({ skillRoots: [tempDir] });
338
- await loader.initialize();
339
-
340
- expect(loader.hasSkill('non-existing')).toBe(false);
341
- });
342
-
343
- it('returns false before initialization', () => {
344
- const loader = new SkillLoader({ skillRoots: [tempDir] });
345
-
346
- expect(loader.hasSkill('test-skill')).toBe(false);
347
- });
348
- });
349
-
350
- describe('loadSkill', () => {
351
- it('loads skill content', async () => {
352
- const loader = new SkillLoader({ skillRoots: [tempDir] });
353
- await loader.initialize();
354
-
355
- const skill = await loader.loadSkill('test-skill');
356
- expect(skill).toBeDefined();
357
- expect(skill!.metadata.name).toBe('test-skill');
358
- expect(skill!.content).toContain('# Test Skill');
359
- expect(skill!.content).toContain('This is a test skill.');
360
- expect(skill!.loadedAt).toBeGreaterThan(0);
361
- });
362
-
363
- it('caches loaded skills', async () => {
364
- const loader = new SkillLoader({ skillRoots: [tempDir] });
365
- await loader.initialize();
366
-
367
- const skill1 = await loader.loadSkill('test-skill');
368
- const skill2 = await loader.loadSkill('test-skill');
369
-
370
- expect(skill1).toBe(skill2); // Same object reference
371
- });
372
-
373
- it('returns null for non-existing skill', async () => {
374
- const loader = new SkillLoader({ skillRoots: [tempDir] });
375
- await loader.initialize();
376
-
377
- const skill = await loader.loadSkill('non-existing');
378
- expect(skill).toBeNull();
379
- });
380
-
381
- it('returns null for skill that cannot be read', async () => {
382
- const loader = new SkillLoader({ skillRoots: [tempDir] });
383
- await loader.initialize();
384
-
385
- // Remove the skill file
386
- await fs.rm(skillFile);
387
-
388
- const skill = await loader.loadSkill('test-skill');
389
- expect(skill).toBeNull();
390
- });
391
-
392
- it('returns null before initialization', async () => {
393
- const loader = new SkillLoader({ skillRoots: [tempDir] });
394
-
395
- const skill = await loader.loadSkill('test-skill');
396
- expect(skill).toBeNull();
397
- });
398
-
399
- it('extracts file refs and shell commands', async () => {
400
- // Create a skill with file refs and shell commands
401
- const skillDir2 = path.join(tempDir, 'rich-skill');
402
- await fs.mkdir(skillDir2, { recursive: true });
403
- await fs.writeFile(
404
- path.join(skillDir2, 'SKILL.md'),
405
- `---
406
- name: rich-skill
407
- description: Rich skill
408
- ---
409
- # Rich Skill
410
- Check @file.txt and run !\`ls -la\`.`
411
- );
412
-
413
- const loader = new SkillLoader({ skillRoots: [tempDir] });
414
- await loader.initialize();
415
-
416
- const skill = await loader.loadSkill('rich-skill');
417
- expect(skill).toBeDefined();
418
- expect(skill!.fileRefs).toContain('file.txt');
419
- expect(skill!.shellCommands).toContain('ls -la');
420
- });
421
- });
422
- });
423
-
424
- describe('getSkillLoader', () => {
425
- afterEach(() => {
426
- resetSkillLoader();
427
- });
428
-
429
- it('returns same loader for same options', () => {
430
- const loader1 = getSkillLoader({ workingDir: '/tmp' });
431
- const loader2 = getSkillLoader({ workingDir: '/tmp' });
432
-
433
- expect(loader1).toBe(loader2);
434
- });
435
-
436
- it('returns different loader for different options', () => {
437
- const loader1 = getSkillLoader({ workingDir: '/tmp' });
438
- const loader2 = getSkillLoader({ workingDir: '/var' });
439
-
440
- expect(loader1).not.toBe(loader2);
441
- });
442
-
443
- it('returns different loader for different skill roots', () => {
444
- const loader1 = getSkillLoader({ skillRoots: ['/tmp'] });
445
- const loader2 = getSkillLoader({ skillRoots: ['/var'] });
446
-
447
- expect(loader1).not.toBe(loader2);
448
- });
449
-
450
- it('returns same loader for equivalent options', () => {
451
- const loader1 = getSkillLoader({ skillRoots: ['/tmp', '/var'] });
452
- const loader2 = getSkillLoader({ skillRoots: ['/var', '/tmp'] });
453
-
454
- expect(loader1).toBe(loader2);
455
- });
456
- });
457
-
458
- describe('initializeSkillLoader', () => {
459
- afterEach(() => {
460
- resetSkillLoader();
461
- });
462
-
463
- it('initializes and returns loader', async () => {
464
- const tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'skill-init-test-'));
465
- try {
466
- const skillDir = path.join(tempDir, 'test-skill');
467
- await fs.mkdir(skillDir, { recursive: true });
468
- await fs.writeFile(
469
- path.join(skillDir, 'SKILL.md'),
470
- `---
471
- name: test-skill
472
- description: A test skill
473
- ---
474
- # Test Skill`
475
- );
476
-
477
- const loader = await initializeSkillLoader({ skillRoots: [tempDir] });
478
- expect(loader).toBeDefined();
479
- expect(loader.hasSkill('test-skill')).toBe(true);
480
- } finally {
481
- await fs.rm(tempDir, { recursive: true, force: true });
482
- }
483
- });
484
- });
485
-
486
- describe('resetSkillLoader', () => {
487
- it('resets global loader', () => {
488
- const loader1 = getSkillLoader();
489
- resetSkillLoader();
490
- const loader2 = getSkillLoader();
491
-
492
- expect(loader1).not.toBe(loader2);
493
- });
494
- });