enya-agent 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 (389) hide show
  1. package/.env.example +20 -0
  2. package/.github/workflows/ci.yml +70 -0
  3. package/.github/workflows/publish.yml +250 -0
  4. package/.gitmodules +3 -0
  5. package/Cargo.lock +3584 -0
  6. package/Cargo.toml +97 -0
  7. package/crates/enact/Cargo.toml +27 -0
  8. package/crates/enact/src/lib.rs +60 -0
  9. package/crates/enact-a2a/Cargo.toml +25 -0
  10. package/crates/enact-a2a/src/lib.rs +411 -0
  11. package/crates/enact-channels/Cargo.toml +64 -0
  12. package/crates/enact-channels/examples/README.md +80 -0
  13. package/crates/enact-channels/examples/channel_bot.rs +169 -0
  14. package/crates/enact-channels/examples/telegram-echo.rs +34 -0
  15. package/crates/enact-channels/examples/whatsapp-echo.rs +142 -0
  16. package/crates/enact-channels/src/config.rs +213 -0
  17. package/crates/enact-channels/src/lib.rs +25 -0
  18. package/crates/enact-channels/src/runtime.rs +237 -0
  19. package/crates/enact-channels/src/security/mod.rs +5 -0
  20. package/crates/enact-channels/src/security/pairing.rs +205 -0
  21. package/crates/enact-channels/src/teams.rs +601 -0
  22. package/crates/enact-channels/src/telegram.rs +2833 -0
  23. package/crates/enact-channels/src/traits.rs +200 -0
  24. package/crates/enact-channels/src/webhook.rs +262 -0
  25. package/crates/enact-channels/src/whatsapp.rs +310 -0
  26. package/crates/enact-cli/Cargo.toml +40 -0
  27. package/crates/enact-cli/src/commands/doctor.rs +62 -0
  28. package/crates/enact-cli/src/commands/mod.rs +3 -0
  29. package/crates/enact-cli/src/commands/run.rs +69 -0
  30. package/crates/enact-cli/src/commands/serve.rs +81 -0
  31. package/crates/enact-cli/src/config.rs +2 -0
  32. package/crates/enact-cli/src/main.rs +79 -0
  33. package/crates/enact-config/Cargo.toml +36 -0
  34. package/crates/enact-config/ENV_VAR_MAPPING.md +135 -0
  35. package/crates/enact-config/QUICK_REFERENCE.md +92 -0
  36. package/crates/enact-config/README.md +107 -0
  37. package/crates/enact-config/TESTING.md +161 -0
  38. package/crates/enact-config/examples/test-env-vars.rs +100 -0
  39. package/crates/enact-config/src/config.rs +399 -0
  40. package/crates/enact-config/src/encrypted_store.rs +211 -0
  41. package/crates/enact-config/src/lib.rs +298 -0
  42. package/crates/enact-config/src/secrets.rs +149 -0
  43. package/crates/enact-config/src/sync.rs +260 -0
  44. package/crates/enact-config/test-env-vars.sh +34 -0
  45. package/crates/enact-config/tests/README.md +99 -0
  46. package/crates/enact-config/tests/config_integration_test.rs +202 -0
  47. package/crates/enact-config/tests/security_test.rs +140 -0
  48. package/crates/enact-context/Cargo.toml +41 -0
  49. package/crates/enact-context/src/budget.rs +314 -0
  50. package/crates/enact-context/src/calibrator.rs +535 -0
  51. package/crates/enact-context/src/compactor.rs +392 -0
  52. package/crates/enact-context/src/condenser.rs +826 -0
  53. package/crates/enact-context/src/lib.rs +94 -0
  54. package/crates/enact-context/src/segment.rs +238 -0
  55. package/crates/enact-context/src/step_context.rs +645 -0
  56. package/crates/enact-context/src/token_counter.rs +148 -0
  57. package/crates/enact-context/src/window.rs +372 -0
  58. package/crates/enact-core/Cargo.toml +42 -0
  59. package/crates/enact-core/README.md +98 -0
  60. package/crates/enact-core/src/background/executor.rs +524 -0
  61. package/crates/enact-core/src/background/mod.rs +48 -0
  62. package/crates/enact-core/src/background/target_binding.rs +390 -0
  63. package/crates/enact-core/src/background/trigger.rs +511 -0
  64. package/crates/enact-core/src/callable/callable.rs +152 -0
  65. package/crates/enact-core/src/callable/composite.rs +817 -0
  66. package/crates/enact-core/src/callable/graph.rs +104 -0
  67. package/crates/enact-core/src/callable/llm.rs +211 -0
  68. package/crates/enact-core/src/callable/mod.rs +64 -0
  69. package/crates/enact-core/src/callable/registry.rs +206 -0
  70. package/crates/enact-core/src/context/execution_context.rs +757 -0
  71. package/crates/enact-core/src/context/invocation.rs +99 -0
  72. package/crates/enact-core/src/context/mod.rs +50 -0
  73. package/crates/enact-core/src/context/tenant.rs +175 -0
  74. package/crates/enact-core/src/context/trace.rs +127 -0
  75. package/crates/enact-core/src/flow/conditional.rs +293 -0
  76. package/crates/enact-core/src/flow/mod.rs +43 -0
  77. package/crates/enact-core/src/flow/parallel.rs +437 -0
  78. package/crates/enact-core/src/flow/repeat.rs +534 -0
  79. package/crates/enact-core/src/flow/sequential.rs +248 -0
  80. package/crates/enact-core/src/graph/checkpoint.rs +79 -0
  81. package/crates/enact-core/src/graph/checkpoint_store.rs +76 -0
  82. package/crates/enact-core/src/graph/compiled.rs +189 -0
  83. package/crates/enact-core/src/graph/edge.rs +59 -0
  84. package/crates/enact-core/src/graph/graph_schema.rs +218 -0
  85. package/crates/enact-core/src/graph/loader.rs +155 -0
  86. package/crates/enact-core/src/graph/mod.rs +18 -0
  87. package/crates/enact-core/src/graph/node/function.rs +49 -0
  88. package/crates/enact-core/src/graph/node/mod.rs +48 -0
  89. package/crates/enact-core/src/graph/schema.rs +62 -0
  90. package/crates/enact-core/src/inbox/message.rs +405 -0
  91. package/crates/enact-core/src/inbox/mod.rs +31 -0
  92. package/crates/enact-core/src/inbox/store.rs +355 -0
  93. package/crates/enact-core/src/kernel/artifact/filesystem.rs +546 -0
  94. package/crates/enact-core/src/kernel/artifact/metadata.rs +283 -0
  95. package/crates/enact-core/src/kernel/artifact/mod.rs +27 -0
  96. package/crates/enact-core/src/kernel/artifact/store.rs +427 -0
  97. package/crates/enact-core/src/kernel/enforcement.rs +1315 -0
  98. package/crates/enact-core/src/kernel/error.rs +1200 -0
  99. package/crates/enact-core/src/kernel/event.rs +1394 -0
  100. package/crates/enact-core/src/kernel/execution_model.rs +831 -0
  101. package/crates/enact-core/src/kernel/execution_state.rs +189 -0
  102. package/crates/enact-core/src/kernel/execution_strategy.rs +117 -0
  103. package/crates/enact-core/src/kernel/ids.rs +2086 -0
  104. package/crates/enact-core/src/kernel/interrupt.rs +125 -0
  105. package/crates/enact-core/src/kernel/kernel.rs +1283 -0
  106. package/crates/enact-core/src/kernel/mod.rs +205 -0
  107. package/crates/enact-core/src/kernel/persistence/event_store.rs +270 -0
  108. package/crates/enact-core/src/kernel/persistence/message_store.rs +908 -0
  109. package/crates/enact-core/src/kernel/persistence/mod.rs +102 -0
  110. package/crates/enact-core/src/kernel/persistence/state_store.rs +228 -0
  111. package/crates/enact-core/src/kernel/persistence/vector_store.rs +299 -0
  112. package/crates/enact-core/src/kernel/reducer.rs +808 -0
  113. package/crates/enact-core/src/kernel/replay.rs +153 -0
  114. package/crates/enact-core/src/lib.rs +413 -0
  115. package/crates/enact-core/src/memory/episodic.rs +0 -0
  116. package/crates/enact-core/src/memory/mod.rs +6 -0
  117. package/crates/enact-core/src/memory/semantic.rs +0 -0
  118. package/crates/enact-core/src/memory/trait.rs +0 -0
  119. package/crates/enact-core/src/memory/vector_db.rs +0 -0
  120. package/crates/enact-core/src/memory/working.rs +0 -0
  121. package/crates/enact-core/src/policy/execution_policy.rs +292 -0
  122. package/crates/enact-core/src/policy/filters.rs +458 -0
  123. package/crates/enact-core/src/policy/input_processor.rs +407 -0
  124. package/crates/enact-core/src/policy/long_running.rs +134 -0
  125. package/crates/enact-core/src/policy/mod.rs +193 -0
  126. package/crates/enact-core/src/policy/pii_input.rs +274 -0
  127. package/crates/enact-core/src/policy/tenant_policy.rs +453 -0
  128. package/crates/enact-core/src/policy/tool_policy.rs +407 -0
  129. package/crates/enact-core/src/providers/mod.rs +63 -0
  130. package/crates/enact-core/src/providers/trait.rs +292 -0
  131. package/crates/enact-core/src/runner/callbacks.rs +6 -0
  132. package/crates/enact-core/src/runner/execution_runner.rs +476 -0
  133. package/crates/enact-core/src/runner/loop.rs +117 -0
  134. package/crates/enact-core/src/runner/mod.rs +58 -0
  135. package/crates/enact-core/src/runner/protected_runner.rs +280 -0
  136. package/crates/enact-core/src/signal/inmemory.rs +231 -0
  137. package/crates/enact-core/src/signal/mod.rs +108 -0
  138. package/crates/enact-core/src/streaming/event_logger.rs +195 -0
  139. package/crates/enact-core/src/streaming/event_stream.rs +1423 -0
  140. package/crates/enact-core/src/streaming/mod.rs +108 -0
  141. package/crates/enact-core/src/streaming/pause_cancel.rs +0 -0
  142. package/crates/enact-core/src/streaming/protected_emitter.rs +173 -0
  143. package/crates/enact-core/src/streaming/protection/context.rs +136 -0
  144. package/crates/enact-core/src/streaming/protection/encryption.rs +289 -0
  145. package/crates/enact-core/src/streaming/protection/mod.rs +43 -0
  146. package/crates/enact-core/src/streaming/protection/pii_protection.rs +243 -0
  147. package/crates/enact-core/src/streaming/protection/processor.rs +166 -0
  148. package/crates/enact-core/src/streaming/sse.rs +0 -0
  149. package/crates/enact-core/src/telemetry/exporter.rs +0 -0
  150. package/crates/enact-core/src/telemetry/init.rs +0 -0
  151. package/crates/enact-core/src/telemetry/mod.rs +49 -0
  152. package/crates/enact-core/src/telemetry/spans.rs +245 -0
  153. package/crates/enact-core/src/tool/agent_tool.rs +177 -0
  154. package/crates/enact-core/src/tool/browser/mod.rs +0 -0
  155. package/crates/enact-core/src/tool/browser/webdriver.rs +0 -0
  156. package/crates/enact-core/src/tool/cost.rs +247 -0
  157. package/crates/enact-core/src/tool/discovery.rs +0 -0
  158. package/crates/enact-core/src/tool/dispatcher.rs +347 -0
  159. package/crates/enact-core/src/tool/filesystem.rs +231 -0
  160. package/crates/enact-core/src/tool/function.rs +99 -0
  161. package/crates/enact-core/src/tool/git.rs +162 -0
  162. package/crates/enact-core/src/tool/http.rs +214 -0
  163. package/crates/enact-core/src/tool/mcp/client.rs +0 -0
  164. package/crates/enact-core/src/tool/mcp/mod.rs +0 -0
  165. package/crates/enact-core/src/tool/mod.rs +51 -0
  166. package/crates/enact-core/src/tool/reasoning/debugging.rs +0 -0
  167. package/crates/enact-core/src/tool/reasoning/mcts.rs +0 -0
  168. package/crates/enact-core/src/tool/reasoning/mod.rs +0 -0
  169. package/crates/enact-core/src/tool/reasoning/sequential.rs +0 -0
  170. package/crates/enact-core/src/tool/sandbox/dagger.rs +0 -0
  171. package/crates/enact-core/src/tool/sandbox/mod.rs +0 -0
  172. package/crates/enact-core/src/tool/shell.rs +147 -0
  173. package/crates/enact-core/src/tool/trait.rs +33 -0
  174. package/crates/enact-core/src/tool/web_search.rs +277 -0
  175. package/crates/enact-core/src/util/config.rs +0 -0
  176. package/crates/enact-core/src/util/errors.rs +0 -0
  177. package/crates/enact-core/src/util/mod.rs +6 -0
  178. package/crates/enact-core/tests/airgapped_e2e_test.rs +291 -0
  179. package/crates/enact-core/tests/e2e_agentic_loop.rs +119 -0
  180. package/crates/enact-core/tests/e2e_test.rs +259 -0
  181. package/crates/enact-core/tests/graph_test.rs +130 -0
  182. package/crates/enact-core/tests/stream_event_id_validation.rs +435 -0
  183. package/crates/enact-cron/Cargo.toml +28 -0
  184. package/crates/enact-cron/src/lib.rs +44 -0
  185. package/crates/enact-cron/src/schedule.rs +156 -0
  186. package/crates/enact-cron/src/store.rs +589 -0
  187. package/crates/enact-cron/src/types.rs +148 -0
  188. package/crates/enact-gateway/Cargo.toml +31 -0
  189. package/crates/enact-gateway/README.md +30 -0
  190. package/crates/enact-gateway/examples/whatsapp-gateway-runner-mock.rs +59 -0
  191. package/crates/enact-gateway/examples/whatsapp-gateway.rs +42 -0
  192. package/crates/enact-gateway/src/lib.rs +582 -0
  193. package/crates/enact-mcp/Cargo.toml +24 -0
  194. package/crates/enact-mcp/src/lib.rs +178 -0
  195. package/crates/enact-memory/Cargo.toml +25 -0
  196. package/crates/enact-memory/src/backend.rs +20 -0
  197. package/crates/enact-memory/src/chunker.rs +230 -0
  198. package/crates/enact-memory/src/embeddings.rs +221 -0
  199. package/crates/enact-memory/src/lib.rs +67 -0
  200. package/crates/enact-memory/src/markdown.rs +127 -0
  201. package/crates/enact-memory/src/none.rs +61 -0
  202. package/crates/enact-memory/src/sqlite.rs +276 -0
  203. package/crates/enact-memory/src/traits.rs +65 -0
  204. package/crates/enact-memory/src/vector.rs +198 -0
  205. package/crates/enact-oauth/Cargo.toml +27 -0
  206. package/crates/enact-oauth/src/lib.rs +584 -0
  207. package/crates/enact-observability/Cargo.toml +22 -0
  208. package/crates/enact-observability/src/lib.rs +197 -0
  209. package/crates/enact-providers/Cargo.toml +33 -0
  210. package/crates/enact-providers/examples/hello-agent.rs +33 -0
  211. package/crates/enact-providers/src/anthropic.rs +182 -0
  212. package/crates/enact-providers/src/azure.rs +96 -0
  213. package/crates/enact-providers/src/bridge.rs +221 -0
  214. package/crates/enact-providers/src/gemini.rs +227 -0
  215. package/crates/enact-providers/src/http.rs +78 -0
  216. package/crates/enact-providers/src/lib.rs +53 -0
  217. package/crates/enact-providers/src/openai_compatible.rs +167 -0
  218. package/crates/enact-providers/src/openrouter.rs +33 -0
  219. package/crates/enact-runner/Cargo.toml +24 -0
  220. package/crates/enact-runner/README.md +76 -0
  221. package/crates/enact-runner/src/compaction.rs +225 -0
  222. package/crates/enact-runner/src/config.rs +118 -0
  223. package/crates/enact-runner/src/lib.rs +63 -0
  224. package/crates/enact-runner/src/loop_driver.rs +414 -0
  225. package/crates/enact-runner/src/parser.rs +421 -0
  226. package/crates/enact-runner/src/retry.rs +262 -0
  227. package/crates/enact-runner/tests/integration.rs +278 -0
  228. package/crates/enact-security/Cargo.toml +22 -0
  229. package/crates/enact-security/src/audit.rs +375 -0
  230. package/crates/enact-security/src/lib.rs +37 -0
  231. package/crates/enact-security/src/policy.rs +406 -0
  232. package/crates/enact-skills/Cargo.toml +25 -0
  233. package/crates/enact-skills/src/lib.rs +506 -0
  234. package/crates/enact-tools/Cargo.toml +22 -0
  235. package/crates/enact-tools/src/file_read.rs +166 -0
  236. package/crates/enact-tools/src/file_write.rs +216 -0
  237. package/crates/enact-tools/src/git_operations.rs +513 -0
  238. package/crates/enact-tools/src/http_request.rs +417 -0
  239. package/crates/enact-tools/src/lib.rs +104 -0
  240. package/crates/enact-tools/src/security.rs +227 -0
  241. package/crates/enact-tools/src/shell.rs +191 -0
  242. package/crates/enact-tools/src/traits.rs +159 -0
  243. package/docs/Makefile +74 -0
  244. package/docs/config.toml +62 -0
  245. package/docs/content/_index.md +174 -0
  246. package/docs/content/a2a/_index.md +431 -0
  247. package/docs/content/api/_index.md +323 -0
  248. package/docs/content/channels/_index.md +160 -0
  249. package/docs/content/channels/teams.md +205 -0
  250. package/docs/content/channels/telegram.md +182 -0
  251. package/docs/content/channels/webhook.md +423 -0
  252. package/docs/content/channels/whatsapp.md +240 -0
  253. package/docs/content/cli/_index.md +261 -0
  254. package/docs/content/concepts/_index.md +273 -0
  255. package/docs/content/configuration/_index.md +241 -0
  256. package/docs/content/cron/_index.md +248 -0
  257. package/docs/content/developers/_index.md +278 -0
  258. package/docs/content/getting-started/_index.md +180 -0
  259. package/docs/content/installation/_index.md +186 -0
  260. package/docs/content/installation/uninstall.md +101 -0
  261. package/docs/content/installation/updating.md +120 -0
  262. package/docs/content/mcp/_index.md +215 -0
  263. package/docs/content/memory/_index.md +163 -0
  264. package/docs/content/oauth/_index.md +515 -0
  265. package/docs/content/providers/_index.md +206 -0
  266. package/docs/content/roadmap/_index.md +199 -0
  267. package/docs/content/security/_index.md +219 -0
  268. package/docs/content/skills/_index.md +228 -0
  269. package/docs/content/tools/_index.md +485 -0
  270. package/docs/content/troubleshooting/_index.md +259 -0
  271. package/docs/content/yaml-schema/_index.md +294 -0
  272. package/docs/static/giallo-dark.css +91 -0
  273. package/docs/static/giallo-light.css +91 -0
  274. package/docs/themes/tanuki/.github/workflows/deploy.yml +44 -0
  275. package/docs/themes/tanuki/LICENSE +21 -0
  276. package/docs/themes/tanuki/README.md +166 -0
  277. package/docs/themes/tanuki/examples/blog/config.toml +58 -0
  278. package/docs/themes/tanuki/examples/blog/content/_index.md +4 -0
  279. package/docs/themes/tanuki/examples/blog/content/about.md +33 -0
  280. package/docs/themes/tanuki/examples/blog/content/blog/_index.md +7 -0
  281. package/docs/themes/tanuki/examples/blog/content/blog/api-design-best-practices.md +245 -0
  282. package/docs/themes/tanuki/examples/blog/content/blog/building-accessible-websites.md +147 -0
  283. package/docs/themes/tanuki/examples/blog/content/blog/css-grid-vs-flexbox.md +165 -0
  284. package/docs/themes/tanuki/examples/blog/content/blog/customizing-catppuccin-colors.md +137 -0
  285. package/docs/themes/tanuki/examples/blog/content/blog/dark-mode-best-practices.md +82 -0
  286. package/docs/themes/tanuki/examples/blog/content/blog/docker-essentials.md +301 -0
  287. package/docs/themes/tanuki/examples/blog/content/blog/getting-started-with-zola.md +129 -0
  288. package/docs/themes/tanuki/examples/blog/content/blog/git-workflow-for-content.md +112 -0
  289. package/docs/themes/tanuki/examples/blog/content/blog/introduction-to-webassembly.md +183 -0
  290. package/docs/themes/tanuki/examples/blog/content/blog/modern-javascript-features.md +234 -0
  291. package/docs/themes/tanuki/examples/blog/content/blog/testing-strategies.md +311 -0
  292. package/docs/themes/tanuki/examples/blog/content/blog/typography-for-developers.md +104 -0
  293. package/docs/themes/tanuki/examples/blog/content/blog/welcome-to-tanuki.md +67 -0
  294. package/docs/themes/tanuki/examples/blog/content/blog/why-static-sites.md +85 -0
  295. package/docs/themes/tanuki/examples/blog/content/projects.md +64 -0
  296. package/docs/themes/tanuki/examples/book/config.toml +17 -0
  297. package/docs/themes/tanuki/examples/book/content/_index.md +12 -0
  298. package/docs/themes/tanuki/examples/book/content/chapter-1.md +90 -0
  299. package/docs/themes/tanuki/examples/book/content/chapter-2.md +143 -0
  300. package/docs/themes/tanuki/examples/book/content/chapter-3.md +217 -0
  301. package/docs/themes/tanuki/examples/book/content/chapter-4.md +224 -0
  302. package/docs/themes/tanuki/examples/book/content/chapter-5.md +297 -0
  303. package/docs/themes/tanuki/examples/book/content/print.md +6 -0
  304. package/docs/themes/tanuki/examples/docs/config.toml +28 -0
  305. package/docs/themes/tanuki/examples/docs/content/_index.md +20 -0
  306. package/docs/themes/tanuki/examples/docs/content/components.md +156 -0
  307. package/docs/themes/tanuki/examples/docs/content/configuration.md +94 -0
  308. package/docs/themes/tanuki/examples/docs/content/customization.md +202 -0
  309. package/docs/themes/tanuki/examples/docs/content/deployment.md +204 -0
  310. package/docs/themes/tanuki/examples/docs/content/installation.md +59 -0
  311. package/docs/themes/tanuki/examples/docs/content/print.md +6 -0
  312. package/docs/themes/tanuki/examples/docs/static/img/tanuki-icon.avif +0 -0
  313. package/docs/themes/tanuki/examples/index.html +2104 -0
  314. package/docs/themes/tanuki/mise.toml +108 -0
  315. package/docs/themes/tanuki/sass/base/_catppuccin.scss +164 -0
  316. package/docs/themes/tanuki/sass/base/_fonts.scss +64 -0
  317. package/docs/themes/tanuki/sass/base/_reset.scss +152 -0
  318. package/docs/themes/tanuki/sass/base/_typography.scss +523 -0
  319. package/docs/themes/tanuki/sass/components/_buttons.scss +209 -0
  320. package/docs/themes/tanuki/sass/components/_code.scss +457 -0
  321. package/docs/themes/tanuki/sass/components/_landing.scss +633 -0
  322. package/docs/themes/tanuki/sass/components/_layout.scss +294 -0
  323. package/docs/themes/tanuki/sass/components/_navigation.scss +1200 -0
  324. package/docs/themes/tanuki/sass/components/_print.scss +237 -0
  325. package/docs/themes/tanuki/sass/components/_search.scss +224 -0
  326. package/docs/themes/tanuki/sass/components/_sidebar.scss +473 -0
  327. package/docs/themes/tanuki/sass/components/_theme-toggle.scss +186 -0
  328. package/docs/themes/tanuki/sass/modes/_blog.scss +366 -0
  329. package/docs/themes/tanuki/sass/modes/_product.scss +875 -0
  330. package/docs/themes/tanuki/sass/modes/_raskell.scss +1696 -0
  331. package/docs/themes/tanuki/sass/patterns/_buttons.scss +183 -0
  332. package/docs/themes/tanuki/sass/patterns/_cards.scss +144 -0
  333. package/docs/themes/tanuki/sass/patterns/_index.scss +9 -0
  334. package/docs/themes/tanuki/sass/patterns/_lists.scss +259 -0
  335. package/docs/themes/tanuki/sass/patterns/_sections.scss +243 -0
  336. package/docs/themes/tanuki/sass/style.scss +47 -0
  337. package/docs/themes/tanuki/sass/tokens/_colors.scss +139 -0
  338. package/docs/themes/tanuki/sass/tokens/_spacing.scss +100 -0
  339. package/docs/themes/tanuki/sass/tokens/_typography.scss +186 -0
  340. package/docs/themes/tanuki/screenshot.png +0 -0
  341. package/docs/themes/tanuki/sentinel.kdl +59 -0
  342. package/docs/themes/tanuki/static/elasticlunr.min.js +10 -0
  343. package/docs/themes/tanuki/static/fonts/GEIST-LICENSE.txt +92 -0
  344. package/docs/themes/tanuki/static/fonts/Geist-Variable.woff2 +0 -0
  345. package/docs/themes/tanuki/static/fonts/GeistMono-Variable.woff2 +0 -0
  346. package/docs/themes/tanuki/static/img/tanuki-icon.avif +0 -0
  347. package/docs/themes/tanuki/static/img/tanuki-icon.png +0 -0
  348. package/docs/themes/tanuki/static/js/anchors.js +18 -0
  349. package/docs/themes/tanuki/static/js/app.js +274 -0
  350. package/docs/themes/tanuki/static/js/code.js +394 -0
  351. package/docs/themes/tanuki/static/js/navigation.js +778 -0
  352. package/docs/themes/tanuki/static/js/scroll-to-top.js +33 -0
  353. package/docs/themes/tanuki/static/js/search-raskell.js +240 -0
  354. package/docs/themes/tanuki/static/js/search.js +215 -0
  355. package/docs/themes/tanuki/static/js/theme.js +169 -0
  356. package/docs/themes/tanuki/static/syntax-dark.css +151 -0
  357. package/docs/themes/tanuki/static/syntax-light.css +151 -0
  358. package/docs/themes/tanuki/static/wasm/sentinel_playground_wasm.js +486 -0
  359. package/docs/themes/tanuki/static/wasm/sentinel_playground_wasm_bg.wasm +0 -0
  360. package/docs/themes/tanuki/templates/404.html +52 -0
  361. package/docs/themes/tanuki/templates/base.html +428 -0
  362. package/docs/themes/tanuki/templates/blog.html +66 -0
  363. package/docs/themes/tanuki/templates/home.html +108 -0
  364. package/docs/themes/tanuki/templates/index.html +178 -0
  365. package/docs/themes/tanuki/templates/landing.html +168 -0
  366. package/docs/themes/tanuki/templates/macros/nav.html +128 -0
  367. package/docs/themes/tanuki/templates/macros/posts.html +101 -0
  368. package/docs/themes/tanuki/templates/macros/ui.html +159 -0
  369. package/docs/themes/tanuki/templates/page.html +135 -0
  370. package/docs/themes/tanuki/templates/partials/footer.html +38 -0
  371. package/docs/themes/tanuki/templates/partials/header.html +366 -0
  372. package/docs/themes/tanuki/templates/partials/nav-buttons.html +55 -0
  373. package/docs/themes/tanuki/templates/partials/nav-overlay.html +81 -0
  374. package/docs/themes/tanuki/templates/partials/page-toc-panel.html +43 -0
  375. package/docs/themes/tanuki/templates/partials/search.html +52 -0
  376. package/docs/themes/tanuki/templates/partials/sidebar.html +107 -0
  377. package/docs/themes/tanuki/templates/partials/theme-toggle.html +35 -0
  378. package/docs/themes/tanuki/templates/partials/toc-overlay.html +146 -0
  379. package/docs/themes/tanuki/templates/partials/version-picker.html +38 -0
  380. package/docs/themes/tanuki/templates/print.html +244 -0
  381. package/docs/themes/tanuki/templates/section.html +186 -0
  382. package/docs/themes/tanuki/templates/taxonomy_list.html +18 -0
  383. package/docs/themes/tanuki/templates/taxonomy_single.html +31 -0
  384. package/docs/themes/tanuki/theme.toml +58 -0
  385. package/examples/hello-agent.rs +55 -0
  386. package/package.json +36 -0
  387. package/proto/config.proto +60 -0
  388. package/proto/events.proto +0 -0
  389. package/proto/runtime.proto +215 -0
@@ -0,0 +1,506 @@
1
+ use anyhow::Result;
2
+ use directories::UserDirs;
3
+ use serde::{Deserialize, Serialize};
4
+ use std::collections::HashMap;
5
+ use std::path::{Path, PathBuf};
6
+ use std::process::Command;
7
+ use std::time::{Duration, SystemTime};
8
+
9
+ const OPEN_SKILLS_REPO_URL: &str = "https://github.com/besoeasy/open-skills";
10
+ const OPEN_SKILLS_SYNC_MARKER: &str = ".enact-open-skills-sync";
11
+ const OPEN_SKILLS_SYNC_INTERVAL_SECS: u64 = 60 * 60 * 24 * 7;
12
+
13
+ /// A skill is a user-defined or community-built capability.
14
+ /// Skills live in `~/.enact/skills/<name>/SKILL.toml`
15
+ /// and can include tool definitions, prompts, and automation scripts.
16
+ #[derive(Debug, Clone, Serialize, Deserialize)]
17
+ pub struct Skill {
18
+ pub name: String,
19
+ pub description: String,
20
+ pub version: String,
21
+ #[serde(default)]
22
+ pub author: Option<String>,
23
+ #[serde(default)]
24
+ pub tags: Vec<String>,
25
+ #[serde(default)]
26
+ pub tools: Vec<SkillTool>,
27
+ #[serde(default)]
28
+ pub prompts: Vec<String>,
29
+ #[serde(skip)]
30
+ pub location: Option<PathBuf>,
31
+ }
32
+
33
+ /// A tool defined by a skill (shell command, HTTP call, etc.)
34
+ #[derive(Debug, Clone, Serialize, Deserialize)]
35
+ pub struct SkillTool {
36
+ pub name: String,
37
+ pub description: String,
38
+ /// "shell", "http", "script"
39
+ pub kind: String,
40
+ /// The command/URL/script to execute
41
+ pub command: String,
42
+ #[serde(default)]
43
+ pub args: HashMap<String, String>,
44
+ }
45
+
46
+ /// Skill manifest parsed from SKILL.toml
47
+ #[derive(Debug, Clone, Serialize, Deserialize)]
48
+ struct SkillManifest {
49
+ skill: SkillMeta,
50
+ #[serde(default)]
51
+ tools: Vec<SkillTool>,
52
+ #[serde(default)]
53
+ prompts: Vec<String>,
54
+ }
55
+
56
+ #[derive(Debug, Clone, Serialize, Deserialize)]
57
+ struct SkillMeta {
58
+ name: String,
59
+ description: String,
60
+ #[serde(default = "default_version")]
61
+ version: String,
62
+ #[serde(default)]
63
+ author: Option<String>,
64
+ #[serde(default)]
65
+ tags: Vec<String>,
66
+ }
67
+
68
+ fn default_version() -> String {
69
+ "0.1.0".to_string()
70
+ }
71
+
72
+ /// Load all skills from the workspace skills directory
73
+ pub fn load_skills(workspace_dir: &Path) -> Vec<Skill> {
74
+ let mut skills = Vec::new();
75
+
76
+ if let Some(open_skills_dir) = ensure_open_skills_repo() {
77
+ skills.extend(load_open_skills(&open_skills_dir));
78
+ }
79
+
80
+ skills.extend(load_workspace_skills(workspace_dir));
81
+ skills
82
+ }
83
+
84
+ fn load_workspace_skills(workspace_dir: &Path) -> Vec<Skill> {
85
+ let skills_dir = workspace_dir.join("skills");
86
+ load_skills_from_directory(&skills_dir)
87
+ }
88
+
89
+ fn load_skills_from_directory(skills_dir: &Path) -> Vec<Skill> {
90
+ if !skills_dir.exists() {
91
+ return Vec::new();
92
+ }
93
+
94
+ let mut skills = Vec::new();
95
+
96
+ let Ok(entries) = std::fs::read_dir(skills_dir) else {
97
+ return skills;
98
+ };
99
+
100
+ for entry in entries.flatten() {
101
+ let path = entry.path();
102
+ if !path.is_dir() {
103
+ continue;
104
+ }
105
+
106
+ // Try SKILL.toml first, then SKILL.md
107
+ let manifest_path = path.join("SKILL.toml");
108
+ let md_path = path.join("SKILL.md");
109
+
110
+ if manifest_path.exists() {
111
+ if let Ok(skill) = load_skill_toml(&manifest_path) {
112
+ skills.push(skill);
113
+ }
114
+ } else if md_path.exists() {
115
+ if let Ok(skill) = load_skill_md(&md_path, &path) {
116
+ skills.push(skill);
117
+ }
118
+ }
119
+ }
120
+
121
+ skills
122
+ }
123
+
124
+ fn load_open_skills(repo_dir: &Path) -> Vec<Skill> {
125
+ let mut skills = Vec::new();
126
+
127
+ let Ok(entries) = std::fs::read_dir(repo_dir) else {
128
+ return skills;
129
+ };
130
+
131
+ for entry in entries.flatten() {
132
+ let path = entry.path();
133
+ if !path.is_file() {
134
+ continue;
135
+ }
136
+
137
+ let is_markdown = path
138
+ .extension()
139
+ .and_then(|ext| ext.to_str())
140
+ .is_some_and(|ext| ext.eq_ignore_ascii_case("md"));
141
+ if !is_markdown {
142
+ continue;
143
+ }
144
+
145
+ let is_readme = path
146
+ .file_name()
147
+ .and_then(|name| name.to_str())
148
+ .is_some_and(|name| name.eq_ignore_ascii_case("README.md"));
149
+ if is_readme {
150
+ continue;
151
+ }
152
+
153
+ if let Ok(skill) = load_open_skill_md(&path) {
154
+ skills.push(skill);
155
+ }
156
+ }
157
+
158
+ skills
159
+ }
160
+
161
+ fn open_skills_enabled() -> bool {
162
+ if let Ok(raw) = std::env::var("ENACT_OPEN_SKILLS_ENABLED") {
163
+ let value = raw.trim().to_ascii_lowercase();
164
+ return !matches!(value.as_str(), "0" | "false" | "off" | "no");
165
+ }
166
+
167
+ // Keep tests deterministic and network-free by default.
168
+ !cfg!(test)
169
+ }
170
+
171
+ fn resolve_open_skills_dir() -> Option<PathBuf> {
172
+ if let Ok(path) = std::env::var("ENACT_OPEN_SKILLS_DIR") {
173
+ let trimmed = path.trim();
174
+ if !trimmed.is_empty() {
175
+ return Some(PathBuf::from(trimmed));
176
+ }
177
+ }
178
+
179
+ UserDirs::new().map(|dirs| dirs.home_dir().join("open-skills"))
180
+ }
181
+
182
+ fn ensure_open_skills_repo() -> Option<PathBuf> {
183
+ if !open_skills_enabled() {
184
+ return None;
185
+ }
186
+
187
+ let repo_dir = resolve_open_skills_dir()?;
188
+
189
+ if !repo_dir.exists() {
190
+ if !clone_open_skills_repo(&repo_dir) {
191
+ return None;
192
+ }
193
+ let _ = mark_open_skills_synced(&repo_dir);
194
+ return Some(repo_dir);
195
+ }
196
+
197
+ if should_sync_open_skills(&repo_dir) {
198
+ if pull_open_skills_repo(&repo_dir) {
199
+ let _ = mark_open_skills_synced(&repo_dir);
200
+ } else {
201
+ tracing::warn!(
202
+ "open-skills update failed; using local copy from {}",
203
+ repo_dir.display()
204
+ );
205
+ }
206
+ }
207
+
208
+ Some(repo_dir)
209
+ }
210
+
211
+ fn clone_open_skills_repo(repo_dir: &Path) -> bool {
212
+ if let Some(parent) = repo_dir.parent() {
213
+ if let Err(err) = std::fs::create_dir_all(parent) {
214
+ tracing::warn!(
215
+ "failed to create open-skills parent directory {}: {err}",
216
+ parent.display()
217
+ );
218
+ return false;
219
+ }
220
+ }
221
+
222
+ let output = Command::new("git")
223
+ .args(["clone", "--depth", "1", OPEN_SKILLS_REPO_URL])
224
+ .arg(repo_dir)
225
+ .output();
226
+
227
+ match output {
228
+ Ok(result) if result.status.success() => {
229
+ tracing::info!("initialized open-skills at {}", repo_dir.display());
230
+ true
231
+ }
232
+ Ok(result) => {
233
+ let stderr = String::from_utf8_lossy(&result.stderr);
234
+ tracing::warn!("failed to clone open-skills: {stderr}");
235
+ false
236
+ }
237
+ Err(err) => {
238
+ tracing::warn!("failed to run git clone for open-skills: {err}");
239
+ false
240
+ }
241
+ }
242
+ }
243
+
244
+ fn pull_open_skills_repo(repo_dir: &Path) -> bool {
245
+ // If user points to a non-git directory via env var, keep using it without pulling.
246
+ if !repo_dir.join(".git").exists() {
247
+ return true;
248
+ }
249
+
250
+ let output = Command::new("git")
251
+ .arg("-C")
252
+ .arg(repo_dir)
253
+ .args(["pull", "--ff-only"])
254
+ .output();
255
+
256
+ match output {
257
+ Ok(result) if result.status.success() => true,
258
+ Ok(result) => {
259
+ let stderr = String::from_utf8_lossy(&result.stderr);
260
+ tracing::warn!("failed to pull open-skills updates: {stderr}");
261
+ false
262
+ }
263
+ Err(err) => {
264
+ tracing::warn!("failed to run git pull for open-skills: {err}");
265
+ false
266
+ }
267
+ }
268
+ }
269
+
270
+ fn should_sync_open_skills(repo_dir: &Path) -> bool {
271
+ let marker = repo_dir.join(OPEN_SKILLS_SYNC_MARKER);
272
+ let Ok(metadata) = std::fs::metadata(marker) else {
273
+ return true;
274
+ };
275
+ let Ok(modified_at) = metadata.modified() else {
276
+ return true;
277
+ };
278
+ let Ok(age) = SystemTime::now().duration_since(modified_at) else {
279
+ return true;
280
+ };
281
+
282
+ age >= Duration::from_secs(OPEN_SKILLS_SYNC_INTERVAL_SECS)
283
+ }
284
+
285
+ fn mark_open_skills_synced(repo_dir: &Path) -> Result<()> {
286
+ std::fs::write(repo_dir.join(OPEN_SKILLS_SYNC_MARKER), b"synced")?;
287
+ Ok(())
288
+ }
289
+
290
+ /// Load a skill from a SKILL.toml manifest
291
+ fn load_skill_toml(path: &Path) -> Result<Skill> {
292
+ let content = std::fs::read_to_string(path)?;
293
+ let manifest: SkillManifest = toml::from_str(&content)?;
294
+
295
+ Ok(Skill {
296
+ name: manifest.skill.name,
297
+ description: manifest.skill.description,
298
+ version: manifest.skill.version,
299
+ author: manifest.skill.author,
300
+ tags: manifest.skill.tags,
301
+ tools: manifest.tools,
302
+ prompts: manifest.prompts,
303
+ location: Some(path.to_path_buf()),
304
+ })
305
+ }
306
+
307
+ /// Load a skill from a SKILL.md file (simpler format)
308
+ fn load_skill_md(path: &Path, dir: &Path) -> Result<Skill> {
309
+ let content = std::fs::read_to_string(path)?;
310
+ let name = dir
311
+ .file_name()
312
+ .and_then(|n| n.to_str())
313
+ .unwrap_or("unknown")
314
+ .to_string();
315
+
316
+ Ok(Skill {
317
+ name,
318
+ description: extract_description(&content),
319
+ version: "0.1.0".to_string(),
320
+ author: None,
321
+ tags: Vec::new(),
322
+ tools: Vec::new(),
323
+ prompts: vec![content],
324
+ location: Some(path.to_path_buf()),
325
+ })
326
+ }
327
+
328
+ fn load_open_skill_md(path: &Path) -> Result<Skill> {
329
+ let content = std::fs::read_to_string(path)?;
330
+ let name = path
331
+ .file_stem()
332
+ .and_then(|n| n.to_str())
333
+ .unwrap_or("open-skill")
334
+ .to_string();
335
+
336
+ Ok(Skill {
337
+ name,
338
+ description: extract_description(&content),
339
+ version: "open-skills".to_string(),
340
+ author: Some("besoeasy/open-skills".to_string()),
341
+ tags: vec!["open-skills".to_string()],
342
+ tools: Vec::new(),
343
+ prompts: vec![content],
344
+ location: Some(path.to_path_buf()),
345
+ })
346
+ }
347
+
348
+ fn extract_description(content: &str) -> String {
349
+ content
350
+ .lines()
351
+ .find(|line| !line.starts_with('#') && !line.trim().is_empty())
352
+ .unwrap_or("No description")
353
+ .trim()
354
+ .to_string()
355
+ }
356
+
357
+ /// Build a system prompt addition from all loaded skills
358
+ pub fn skills_to_prompt(skills: &[Skill]) -> String {
359
+ use std::fmt::Write;
360
+
361
+ if skills.is_empty() {
362
+ return String::new();
363
+ }
364
+
365
+ let mut prompt = String::from("\n## Active Skills\n\n");
366
+
367
+ for skill in skills {
368
+ let _ = writeln!(prompt, "### {} (v{})", skill.name, skill.version);
369
+ let _ = writeln!(prompt, "{}", skill.description);
370
+
371
+ if !skill.tools.is_empty() {
372
+ prompt.push_str("Tools:\n");
373
+ for tool in &skill.tools {
374
+ let _ = writeln!(
375
+ prompt,
376
+ "- **{}**: {} ({})",
377
+ tool.name, tool.description, tool.kind
378
+ );
379
+ }
380
+ }
381
+
382
+ for p in &skill.prompts {
383
+ prompt.push_str(p);
384
+ prompt.push('\n');
385
+ }
386
+
387
+ prompt.push('\n');
388
+ }
389
+
390
+ prompt
391
+ }
392
+
393
+ /// Get the skills directory path
394
+ pub fn skills_dir(workspace_dir: &Path) -> PathBuf {
395
+ workspace_dir.join("skills")
396
+ }
397
+
398
+ /// Initialize the skills directory with a README
399
+ pub fn init_skills_dir(workspace_dir: &Path) -> Result<()> {
400
+ let dir = skills_dir(workspace_dir);
401
+ std::fs::create_dir_all(&dir)?;
402
+
403
+ let readme = dir.join("README.md");
404
+ if !readme.exists() {
405
+ std::fs::write(
406
+ &readme,
407
+ "# Enact Skills\n\n\
408
+ Each subdirectory is a skill. Create a `SKILL.toml` or `SKILL.md` file inside.\n\n\
409
+ ## SKILL.toml format\n\n\
410
+ ```toml\n\
411
+ [skill]\n\
412
+ name = \"my-skill\"\n\
413
+ description = \"What this skill does\"\n\
414
+ version = \"0.1.0\"\n\
415
+ author = \"your-name\"\n\
416
+ tags = [\"productivity\", \"automation\"]\n\n\
417
+ [[tools]]\n\
418
+ name = \"my_tool\"\n\
419
+ description = \"What this tool does\"\n\
420
+ kind = \"shell\"\n\
421
+ command = \"echo hello\"\n\
422
+ ```\n\n\
423
+ ## SKILL.md format (simpler)\n\n\
424
+ Just write a markdown file with instructions for the agent.\n\
425
+ The agent will read it and follow the instructions.\n",
426
+ )?;
427
+ }
428
+
429
+ Ok(())
430
+ }
431
+
432
+ #[cfg(test)]
433
+ #[allow(clippy::similar_names)]
434
+ mod tests {
435
+ use super::*;
436
+ use std::fs;
437
+
438
+ #[test]
439
+ fn load_empty_skills_dir() {
440
+ let dir = tempfile::tempdir().unwrap();
441
+ let skills = load_skills(dir.path());
442
+ assert!(skills.is_empty());
443
+ }
444
+
445
+ #[test]
446
+ fn load_skill_from_toml() {
447
+ let dir = tempfile::tempdir().unwrap();
448
+ let skills_dir = dir.path().join("skills");
449
+ let skill_dir = skills_dir.join("test-skill");
450
+ fs::create_dir_all(&skill_dir).unwrap();
451
+
452
+ fs::write(
453
+ skill_dir.join("SKILL.toml"),
454
+ r#"
455
+ [skill]
456
+ name = "test-skill"
457
+ description = "A test skill"
458
+ version = "1.0.0"
459
+ tags = ["test"]
460
+
461
+ [[tools]]
462
+ name = "hello"
463
+ description = "Says hello"
464
+ kind = "shell"
465
+ command = "echo hello"
466
+ "#,
467
+ )
468
+ .unwrap();
469
+
470
+ let skills = load_skills(dir.path());
471
+ assert_eq!(skills.len(), 1);
472
+ assert_eq!(skills[0].name, "test-skill");
473
+ assert_eq!(skills[0].tools.len(), 1);
474
+ assert_eq!(skills[0].tools[0].name, "hello");
475
+ }
476
+
477
+ #[test]
478
+ fn skills_to_prompt_empty() {
479
+ let prompt = skills_to_prompt(&[]);
480
+ assert!(prompt.is_empty());
481
+ }
482
+
483
+ #[test]
484
+ fn skills_to_prompt_with_skills() {
485
+ let skills = vec![Skill {
486
+ name: "test".to_string(),
487
+ description: "A test".to_string(),
488
+ version: "1.0.0".to_string(),
489
+ author: None,
490
+ tags: vec![],
491
+ tools: vec![],
492
+ prompts: vec!["Do the thing.".to_string()],
493
+ location: None,
494
+ }];
495
+ let prompt = skills_to_prompt(&skills);
496
+ assert!(prompt.contains("test"));
497
+ assert!(prompt.contains("Do the thing"));
498
+ }
499
+
500
+ #[test]
501
+ fn init_skills_creates_readme() {
502
+ let dir = tempfile::tempdir().unwrap();
503
+ init_skills_dir(dir.path()).unwrap();
504
+ assert!(dir.path().join("skills").join("README.md").exists());
505
+ }
506
+ }
@@ -0,0 +1,22 @@
1
+ [package]
2
+ name = "enact-tools"
3
+ version.workspace = true
4
+ edition.workspace = true
5
+ license.workspace = true
6
+ description = "Built-in tools for Enact agents"
7
+ repository.workspace = true
8
+ homepage.workspace = true
9
+ keywords = ["tools", "http", "browser", "utilities"]
10
+ categories.workspace = true
11
+
12
+ [dependencies]
13
+ anyhow.workspace = true
14
+ async-trait.workspace = true
15
+ tokio.workspace = true
16
+ serde.workspace = true
17
+ serde_json.workspace = true
18
+ tracing.workspace = true
19
+ reqwest.workspace = true
20
+
21
+ [dev-dependencies]
22
+ tempfile.workspace = true
@@ -0,0 +1,166 @@
1
+ //! File read tool with path sandboxing
2
+
3
+ use crate::security::SecurityPolicy;
4
+ use crate::traits::{Tool, ToolResult};
5
+ use async_trait::async_trait;
6
+ use serde_json::json;
7
+ use std::sync::Arc;
8
+
9
+ const MAX_FILE_SIZE_BYTES: u64 = 10 * 1024 * 1024;
10
+
11
+ /// Read file contents with path sandboxing
12
+ pub struct FileReadTool {
13
+ security: Arc<SecurityPolicy>,
14
+ }
15
+
16
+ impl FileReadTool {
17
+ pub fn new(security: Arc<SecurityPolicy>) -> Self {
18
+ Self { security }
19
+ }
20
+ }
21
+
22
+ #[async_trait]
23
+ impl Tool for FileReadTool {
24
+ fn name(&self) -> &str {
25
+ "file_read"
26
+ }
27
+
28
+ fn description(&self) -> &str {
29
+ "Read the contents of a file in the workspace"
30
+ }
31
+
32
+ fn parameters_schema(&self) -> serde_json::Value {
33
+ json!({
34
+ "type": "object",
35
+ "properties": {
36
+ "path": {
37
+ "type": "string",
38
+ "description": "Relative path to the file within the workspace"
39
+ }
40
+ },
41
+ "required": ["path"]
42
+ })
43
+ }
44
+
45
+ async fn execute(&self, args: serde_json::Value) -> anyhow::Result<ToolResult> {
46
+ let path = args
47
+ .get("path")
48
+ .and_then(|v| v.as_str())
49
+ .ok_or_else(|| anyhow::anyhow!("Missing 'path' parameter"))?;
50
+
51
+ if self.security.is_rate_limited() {
52
+ return Ok(ToolResult::failure(
53
+ "Rate limit exceeded: too many actions in the last hour",
54
+ ));
55
+ }
56
+
57
+ // Security check: validate path is within workspace
58
+ if !self.security.is_path_allowed(path) {
59
+ return Ok(ToolResult::failure(format!(
60
+ "Path not allowed by security policy: {path}"
61
+ )));
62
+ }
63
+
64
+ // Record action BEFORE canonicalization
65
+ if !self.security.record_action() {
66
+ return Ok(ToolResult::failure(
67
+ "Rate limit exceeded: action budget exhausted",
68
+ ));
69
+ }
70
+
71
+ let full_path = self.security.workspace_dir.join(path);
72
+
73
+ // Resolve path before reading to block symlink escapes
74
+ let resolved_path = match tokio::fs::canonicalize(&full_path).await {
75
+ Ok(p) => p,
76
+ Err(e) => {
77
+ return Ok(ToolResult::failure(format!(
78
+ "Failed to resolve file path: {e}"
79
+ )));
80
+ }
81
+ };
82
+
83
+ if !self.security.is_resolved_path_allowed(&resolved_path) {
84
+ return Ok(ToolResult::failure(format!(
85
+ "Resolved path escapes workspace: {}",
86
+ resolved_path.display()
87
+ )));
88
+ }
89
+
90
+ // Check file size
91
+ match tokio::fs::metadata(&resolved_path).await {
92
+ Ok(meta) => {
93
+ if meta.len() > MAX_FILE_SIZE_BYTES {
94
+ return Ok(ToolResult::failure(format!(
95
+ "File too large: {} bytes (limit: {MAX_FILE_SIZE_BYTES} bytes)",
96
+ meta.len()
97
+ )));
98
+ }
99
+ }
100
+ Err(e) => {
101
+ return Ok(ToolResult::failure(format!(
102
+ "Failed to read file metadata: {e}"
103
+ )));
104
+ }
105
+ }
106
+
107
+ match tokio::fs::read_to_string(&resolved_path).await {
108
+ Ok(contents) => Ok(ToolResult::success(contents)),
109
+ Err(e) => Ok(ToolResult::failure(format!("Failed to read file: {e}"))),
110
+ }
111
+ }
112
+ }
113
+
114
+ #[cfg(test)]
115
+ mod tests {
116
+ use super::*;
117
+ use crate::security::AutonomyLevel;
118
+
119
+ fn test_security(workspace: std::path::PathBuf) -> Arc<SecurityPolicy> {
120
+ Arc::new(SecurityPolicy {
121
+ autonomy: AutonomyLevel::Supervised,
122
+ workspace_dir: workspace,
123
+ ..SecurityPolicy::default()
124
+ })
125
+ }
126
+
127
+ #[test]
128
+ fn file_read_name() {
129
+ let tool = FileReadTool::new(test_security(std::env::temp_dir()));
130
+ assert_eq!(tool.name(), "file_read");
131
+ }
132
+
133
+ #[tokio::test]
134
+ async fn file_read_blocks_path_traversal() {
135
+ let dir = std::env::temp_dir().join("enact_test_file_read_traversal");
136
+ let _ = tokio::fs::remove_dir_all(&dir).await;
137
+ tokio::fs::create_dir_all(&dir).await.unwrap();
138
+
139
+ let tool = FileReadTool::new(test_security(dir.clone()));
140
+ let result = tool
141
+ .execute(json!({"path": "../../../etc/passwd"}))
142
+ .await
143
+ .unwrap();
144
+ assert!(!result.success);
145
+ assert!(result.error.as_ref().unwrap().contains("not allowed"));
146
+
147
+ let _ = tokio::fs::remove_dir_all(&dir).await;
148
+ }
149
+
150
+ #[tokio::test]
151
+ async fn file_read_existing_file() {
152
+ let dir = std::env::temp_dir().join("enact_test_file_read");
153
+ let _ = tokio::fs::remove_dir_all(&dir).await;
154
+ tokio::fs::create_dir_all(&dir).await.unwrap();
155
+ tokio::fs::write(dir.join("test.txt"), "hello world")
156
+ .await
157
+ .unwrap();
158
+
159
+ let tool = FileReadTool::new(test_security(dir.clone()));
160
+ let result = tool.execute(json!({"path": "test.txt"})).await.unwrap();
161
+ assert!(result.success);
162
+ assert_eq!(result.output, "hello world");
163
+
164
+ let _ = tokio::fs::remove_dir_all(&dir).await;
165
+ }
166
+ }