@agenticmail/enterprise 0.3.2 → 0.4.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 (1106) hide show
  1. package/.github/CODEOWNERS +23 -0
  2. package/.github/workflows/publish-community-skills.yml +121 -0
  3. package/.github/workflows/validate-community-skills.yml +172 -0
  4. package/ARCHITECTURE.md +184 -167
  5. package/CLAUDE.md +40 -0
  6. package/CONTRIBUTING.md +254 -0
  7. package/DEPLOYMENT.md +1031 -0
  8. package/README.md +355 -174
  9. package/community-skills/_template/README.md +31 -0
  10. package/community-skills/_template/agenticmail-skill.json +48 -0
  11. package/community-skills/activecampaign/README.md +21 -0
  12. package/community-skills/activecampaign/agenticmail-skill.json +56 -0
  13. package/community-skills/adobe-sign/README.md +21 -0
  14. package/community-skills/adobe-sign/agenticmail-skill.json +72 -0
  15. package/community-skills/adp/README.md +21 -0
  16. package/community-skills/adp/agenticmail-skill.json +65 -0
  17. package/community-skills/airtable-bases/README.md +29 -0
  18. package/community-skills/airtable-bases/agenticmail-skill.json +69 -0
  19. package/community-skills/apollo-io/README.md +21 -0
  20. package/community-skills/apollo-io/agenticmail-skill.json +50 -0
  21. package/community-skills/asana-tasks/README.md +30 -0
  22. package/community-skills/asana-tasks/agenticmail-skill.json +70 -0
  23. package/community-skills/auth0/README.md +21 -0
  24. package/community-skills/auth0/agenticmail-skill.json +56 -0
  25. package/community-skills/aws-services/README.md +38 -0
  26. package/community-skills/aws-services/agenticmail-skill.json +70 -0
  27. package/community-skills/azure-devops/README.md +21 -0
  28. package/community-skills/azure-devops/agenticmail-skill.json +57 -0
  29. package/community-skills/bamboohr/README.md +21 -0
  30. package/community-skills/bamboohr/agenticmail-skill.json +56 -0
  31. package/community-skills/basecamp/README.md +21 -0
  32. package/community-skills/basecamp/agenticmail-skill.json +56 -0
  33. package/community-skills/bigcommerce/README.md +21 -0
  34. package/community-skills/bigcommerce/agenticmail-skill.json +55 -0
  35. package/community-skills/bitbucket-repos/README.md +29 -0
  36. package/community-skills/bitbucket-repos/agenticmail-skill.json +70 -0
  37. package/community-skills/box/README.md +21 -0
  38. package/community-skills/box/agenticmail-skill.json +48 -0
  39. package/community-skills/brex/README.md +21 -0
  40. package/community-skills/brex/agenticmail-skill.json +48 -0
  41. package/community-skills/buffer/README.md +21 -0
  42. package/community-skills/buffer/agenticmail-skill.json +50 -0
  43. package/community-skills/calendly/README.md +20 -0
  44. package/community-skills/calendly/agenticmail-skill.json +43 -0
  45. package/community-skills/canva-design/README.md +33 -0
  46. package/community-skills/canva-design/agenticmail-skill.json +60 -0
  47. package/community-skills/chargebee/README.md +21 -0
  48. package/community-skills/chargebee/agenticmail-skill.json +55 -0
  49. package/community-skills/circleci-pipelines/README.md +28 -0
  50. package/community-skills/circleci-pipelines/agenticmail-skill.json +66 -0
  51. package/community-skills/clickup/README.md +21 -0
  52. package/community-skills/clickup/agenticmail-skill.json +48 -0
  53. package/community-skills/close-crm/README.md +21 -0
  54. package/community-skills/close-crm/agenticmail-skill.json +49 -0
  55. package/community-skills/cloudflare-cdn/README.md +37 -0
  56. package/community-skills/cloudflare-cdn/agenticmail-skill.json +67 -0
  57. package/community-skills/confluence-wiki/README.md +31 -0
  58. package/community-skills/confluence-wiki/agenticmail-skill.json +74 -0
  59. package/community-skills/contentful/README.md +21 -0
  60. package/community-skills/contentful/agenticmail-skill.json +62 -0
  61. package/community-skills/copper-crm/README.md +21 -0
  62. package/community-skills/copper-crm/agenticmail-skill.json +56 -0
  63. package/community-skills/crisp/README.md +21 -0
  64. package/community-skills/crisp/agenticmail-skill.json +56 -0
  65. package/community-skills/crowdstrike/README.md +21 -0
  66. package/community-skills/crowdstrike/agenticmail-skill.json +69 -0
  67. package/community-skills/datadog-monitoring/README.md +37 -0
  68. package/community-skills/datadog-monitoring/agenticmail-skill.json +73 -0
  69. package/community-skills/digitalocean/README.md +21 -0
  70. package/community-skills/digitalocean/agenticmail-skill.json +48 -0
  71. package/community-skills/discord/README.md +45 -0
  72. package/community-skills/discord/agenticmail-skill.json +65 -0
  73. package/community-skills/docker-containers/README.md +38 -0
  74. package/community-skills/docker-containers/agenticmail-skill.json +70 -0
  75. package/community-skills/docusign-esign/README.md +35 -0
  76. package/community-skills/docusign-esign/agenticmail-skill.json +66 -0
  77. package/community-skills/drift/README.md +21 -0
  78. package/community-skills/drift/agenticmail-skill.json +50 -0
  79. package/community-skills/dropbox-storage/README.md +36 -0
  80. package/community-skills/dropbox-storage/agenticmail-skill.json +67 -0
  81. package/community-skills/figma-design/README.md +36 -0
  82. package/community-skills/figma-design/agenticmail-skill.json +66 -0
  83. package/community-skills/firebase/README.md +21 -0
  84. package/community-skills/firebase/agenticmail-skill.json +56 -0
  85. package/community-skills/flyio/README.md +21 -0
  86. package/community-skills/flyio/agenticmail-skill.json +56 -0
  87. package/community-skills/freshbooks/README.md +21 -0
  88. package/community-skills/freshbooks/agenticmail-skill.json +55 -0
  89. package/community-skills/freshdesk/README.md +21 -0
  90. package/community-skills/freshdesk/agenticmail-skill.json +57 -0
  91. package/community-skills/freshsales/README.md +21 -0
  92. package/community-skills/freshsales/agenticmail-skill.json +55 -0
  93. package/community-skills/freshservice/README.md +21 -0
  94. package/community-skills/freshservice/agenticmail-skill.json +56 -0
  95. package/community-skills/front/README.md +21 -0
  96. package/community-skills/front/agenticmail-skill.json +48 -0
  97. package/community-skills/github/README.md +49 -0
  98. package/community-skills/github/agenticmail-skill.json +73 -0
  99. package/community-skills/github-actions/README.md +29 -0
  100. package/community-skills/github-actions/agenticmail-skill.json +72 -0
  101. package/community-skills/gitlab-ci/README.md +51 -0
  102. package/community-skills/gitlab-ci/agenticmail-skill.json +66 -0
  103. package/community-skills/gong/README.md +21 -0
  104. package/community-skills/gong/agenticmail-skill.json +50 -0
  105. package/community-skills/google-ads/README.md +36 -0
  106. package/community-skills/google-ads/agenticmail-skill.json +58 -0
  107. package/community-skills/google-analytics/README.md +28 -0
  108. package/community-skills/google-analytics/agenticmail-skill.json +69 -0
  109. package/community-skills/google-cloud/README.md +33 -0
  110. package/community-skills/google-cloud/agenticmail-skill.json +59 -0
  111. package/community-skills/google-drive/README.md +36 -0
  112. package/community-skills/google-drive/agenticmail-skill.json +65 -0
  113. package/community-skills/gotomeeting/README.md +20 -0
  114. package/community-skills/gotomeeting/agenticmail-skill.json +45 -0
  115. package/community-skills/grafana/README.md +21 -0
  116. package/community-skills/grafana/agenticmail-skill.json +56 -0
  117. package/community-skills/greenhouse/README.md +21 -0
  118. package/community-skills/greenhouse/agenticmail-skill.json +49 -0
  119. package/community-skills/gusto/README.md +21 -0
  120. package/community-skills/gusto/agenticmail-skill.json +49 -0
  121. package/community-skills/hashicorp-vault/README.md +21 -0
  122. package/community-skills/hashicorp-vault/agenticmail-skill.json +58 -0
  123. package/community-skills/heroku/README.md +21 -0
  124. package/community-skills/heroku/agenticmail-skill.json +48 -0
  125. package/community-skills/hibob/README.md +21 -0
  126. package/community-skills/hibob/agenticmail-skill.json +48 -0
  127. package/community-skills/hootsuite/README.md +21 -0
  128. package/community-skills/hootsuite/agenticmail-skill.json +48 -0
  129. package/community-skills/hubspot-crm/README.md +36 -0
  130. package/community-skills/hubspot-crm/agenticmail-skill.json +70 -0
  131. package/community-skills/huggingface/README.md +21 -0
  132. package/community-skills/huggingface/agenticmail-skill.json +48 -0
  133. package/community-skills/index.json +1320 -0
  134. package/community-skills/intercom-support/README.md +36 -0
  135. package/community-skills/intercom-support/agenticmail-skill.json +64 -0
  136. package/community-skills/jira/README.md +53 -0
  137. package/community-skills/jira/agenticmail-skill.json +72 -0
  138. package/community-skills/klaviyo/README.md +21 -0
  139. package/community-skills/klaviyo/agenticmail-skill.json +48 -0
  140. package/community-skills/kubernetes-cluster/README.md +36 -0
  141. package/community-skills/kubernetes-cluster/agenticmail-skill.json +72 -0
  142. package/community-skills/lattice/README.md +21 -0
  143. package/community-skills/lattice/agenticmail-skill.json +49 -0
  144. package/community-skills/launchdarkly/README.md +21 -0
  145. package/community-skills/launchdarkly/agenticmail-skill.json +48 -0
  146. package/community-skills/lever/README.md +21 -0
  147. package/community-skills/lever/agenticmail-skill.json +49 -0
  148. package/community-skills/linear/README.md +29 -0
  149. package/community-skills/linear/agenticmail-skill.json +81 -0
  150. package/community-skills/linkedin/README.md +21 -0
  151. package/community-skills/linkedin/agenticmail-skill.json +48 -0
  152. package/community-skills/livechat/README.md +20 -0
  153. package/community-skills/livechat/agenticmail-skill.json +43 -0
  154. package/community-skills/loom-video/README.md +20 -0
  155. package/community-skills/loom-video/agenticmail-skill.json +44 -0
  156. package/community-skills/mailchimp-campaigns/README.md +37 -0
  157. package/community-skills/mailchimp-campaigns/agenticmail-skill.json +69 -0
  158. package/community-skills/mailgun/README.md +21 -0
  159. package/community-skills/mailgun/agenticmail-skill.json +71 -0
  160. package/community-skills/microsoft-teams/README.md +51 -0
  161. package/community-skills/microsoft-teams/agenticmail-skill.json +66 -0
  162. package/community-skills/miro-boards/README.md +20 -0
  163. package/community-skills/miro-boards/agenticmail-skill.json +44 -0
  164. package/community-skills/mixpanel-analytics/README.md +34 -0
  165. package/community-skills/mixpanel-analytics/agenticmail-skill.json +64 -0
  166. package/community-skills/monday-boards/README.md +28 -0
  167. package/community-skills/monday-boards/agenticmail-skill.json +68 -0
  168. package/community-skills/mongodb-atlas/README.md +21 -0
  169. package/community-skills/mongodb-atlas/agenticmail-skill.json +58 -0
  170. package/community-skills/neon/README.md +21 -0
  171. package/community-skills/neon/agenticmail-skill.json +50 -0
  172. package/community-skills/netlify/README.md +21 -0
  173. package/community-skills/netlify/agenticmail-skill.json +48 -0
  174. package/community-skills/netsuite/README.md +21 -0
  175. package/community-skills/netsuite/agenticmail-skill.json +57 -0
  176. package/community-skills/newrelic/README.md +21 -0
  177. package/community-skills/newrelic/agenticmail-skill.json +58 -0
  178. package/community-skills/notion/README.md +49 -0
  179. package/community-skills/notion/agenticmail-skill.json +72 -0
  180. package/community-skills/okta/README.md +21 -0
  181. package/community-skills/okta/agenticmail-skill.json +57 -0
  182. package/community-skills/openai/README.md +21 -0
  183. package/community-skills/openai/agenticmail-skill.json +48 -0
  184. package/community-skills/opsgenie/README.md +21 -0
  185. package/community-skills/opsgenie/agenticmail-skill.json +48 -0
  186. package/community-skills/outreach/README.md +21 -0
  187. package/community-skills/outreach/agenticmail-skill.json +50 -0
  188. package/community-skills/paddle/README.md +21 -0
  189. package/community-skills/paddle/agenticmail-skill.json +55 -0
  190. package/community-skills/pagerduty/README.md +21 -0
  191. package/community-skills/pagerduty/agenticmail-skill.json +48 -0
  192. package/community-skills/pandadoc/README.md +21 -0
  193. package/community-skills/pandadoc/agenticmail-skill.json +48 -0
  194. package/community-skills/paypal/README.md +21 -0
  195. package/community-skills/paypal/agenticmail-skill.json +55 -0
  196. package/community-skills/personio/README.md +21 -0
  197. package/community-skills/personio/agenticmail-skill.json +48 -0
  198. package/community-skills/pinecone/README.md +21 -0
  199. package/community-skills/pinecone/agenticmail-skill.json +48 -0
  200. package/community-skills/pipedrive-deals/README.md +33 -0
  201. package/community-skills/pipedrive-deals/agenticmail-skill.json +60 -0
  202. package/community-skills/plaid/README.md +21 -0
  203. package/community-skills/plaid/agenticmail-skill.json +65 -0
  204. package/community-skills/postmark/README.md +21 -0
  205. package/community-skills/postmark/agenticmail-skill.json +48 -0
  206. package/community-skills/power-automate/README.md +21 -0
  207. package/community-skills/power-automate/agenticmail-skill.json +59 -0
  208. package/community-skills/quickbooks-accounting/README.md +38 -0
  209. package/community-skills/quickbooks-accounting/agenticmail-skill.json +70 -0
  210. package/community-skills/recurly/README.md +21 -0
  211. package/community-skills/recurly/agenticmail-skill.json +48 -0
  212. package/community-skills/reddit/README.md +21 -0
  213. package/community-skills/reddit/agenticmail-skill.json +48 -0
  214. package/community-skills/render/README.md +21 -0
  215. package/community-skills/render/agenticmail-skill.json +48 -0
  216. package/community-skills/ringcentral/README.md +20 -0
  217. package/community-skills/ringcentral/agenticmail-skill.json +43 -0
  218. package/community-skills/rippling/README.md +21 -0
  219. package/community-skills/rippling/agenticmail-skill.json +48 -0
  220. package/community-skills/salesforce/README.md +53 -0
  221. package/community-skills/salesforce/agenticmail-skill.json +73 -0
  222. package/community-skills/salesloft/README.md +21 -0
  223. package/community-skills/salesloft/agenticmail-skill.json +48 -0
  224. package/community-skills/sanity/README.md +21 -0
  225. package/community-skills/sanity/agenticmail-skill.json +62 -0
  226. package/community-skills/sap/README.md +21 -0
  227. package/community-skills/sap/agenticmail-skill.json +63 -0
  228. package/community-skills/segment-cdp/README.md +37 -0
  229. package/community-skills/segment-cdp/agenticmail-skill.json +66 -0
  230. package/community-skills/sendgrid-email/README.md +51 -0
  231. package/community-skills/sendgrid-email/agenticmail-skill.json +71 -0
  232. package/community-skills/sentry/README.md +21 -0
  233. package/community-skills/sentry/agenticmail-skill.json +58 -0
  234. package/community-skills/servicenow/README.md +21 -0
  235. package/community-skills/servicenow/agenticmail-skill.json +56 -0
  236. package/community-skills/shopify/README.md +21 -0
  237. package/community-skills/shopify/agenticmail-skill.json +56 -0
  238. package/community-skills/shortcut/README.md +21 -0
  239. package/community-skills/shortcut/agenticmail-skill.json +48 -0
  240. package/community-skills/slack/README.md +45 -0
  241. package/community-skills/slack/agenticmail-skill.json +73 -0
  242. package/community-skills/smartsheet/README.md +21 -0
  243. package/community-skills/smartsheet/agenticmail-skill.json +48 -0
  244. package/community-skills/snowflake-warehouse/README.md +37 -0
  245. package/community-skills/snowflake-warehouse/agenticmail-skill.json +71 -0
  246. package/community-skills/snyk/README.md +21 -0
  247. package/community-skills/snyk/agenticmail-skill.json +48 -0
  248. package/community-skills/splunk/README.md +21 -0
  249. package/community-skills/splunk/agenticmail-skill.json +56 -0
  250. package/community-skills/square/README.md +21 -0
  251. package/community-skills/square/agenticmail-skill.json +55 -0
  252. package/community-skills/statuspage/README.md +21 -0
  253. package/community-skills/statuspage/agenticmail-skill.json +56 -0
  254. package/community-skills/stripe/README.md +45 -0
  255. package/community-skills/stripe/agenticmail-skill.json +66 -0
  256. package/community-skills/supabase/README.md +21 -0
  257. package/community-skills/supabase/agenticmail-skill.json +56 -0
  258. package/community-skills/teamwork/README.md +21 -0
  259. package/community-skills/teamwork/agenticmail-skill.json +56 -0
  260. package/community-skills/telegram-bot/README.md +20 -0
  261. package/community-skills/telegram-bot/agenticmail-skill.json +52 -0
  262. package/community-skills/terraform-iac/README.md +34 -0
  263. package/community-skills/terraform-iac/agenticmail-skill.json +66 -0
  264. package/community-skills/todoist-tasks/README.md +29 -0
  265. package/community-skills/todoist-tasks/agenticmail-skill.json +72 -0
  266. package/community-skills/trello-cards/README.md +30 -0
  267. package/community-skills/trello-cards/agenticmail-skill.json +72 -0
  268. package/community-skills/twilio-sms/README.md +47 -0
  269. package/community-skills/twilio-sms/agenticmail-skill.json +63 -0
  270. package/community-skills/twitter/README.md +21 -0
  271. package/community-skills/twitter/agenticmail-skill.json +48 -0
  272. package/community-skills/vercel-deployments/README.md +29 -0
  273. package/community-skills/vercel-deployments/agenticmail-skill.json +70 -0
  274. package/community-skills/weaviate/README.md +21 -0
  275. package/community-skills/weaviate/agenticmail-skill.json +56 -0
  276. package/community-skills/webex/README.md +20 -0
  277. package/community-skills/webex/agenticmail-skill.json +43 -0
  278. package/community-skills/webflow/README.md +21 -0
  279. package/community-skills/webflow/agenticmail-skill.json +48 -0
  280. package/community-skills/whatsapp-business/README.md +20 -0
  281. package/community-skills/whatsapp-business/agenticmail-skill.json +52 -0
  282. package/community-skills/whereby/README.md +20 -0
  283. package/community-skills/whereby/agenticmail-skill.json +43 -0
  284. package/community-skills/woocommerce/README.md +21 -0
  285. package/community-skills/woocommerce/agenticmail-skill.json +55 -0
  286. package/community-skills/wordpress/README.md +21 -0
  287. package/community-skills/wordpress/agenticmail-skill.json +56 -0
  288. package/community-skills/workday/README.md +21 -0
  289. package/community-skills/workday/agenticmail-skill.json +56 -0
  290. package/community-skills/wrike/README.md +21 -0
  291. package/community-skills/wrike/agenticmail-skill.json +48 -0
  292. package/community-skills/xero/README.md +21 -0
  293. package/community-skills/xero/agenticmail-skill.json +56 -0
  294. package/community-skills/youtube/README.md +21 -0
  295. package/community-skills/youtube/agenticmail-skill.json +52 -0
  296. package/community-skills/zendesk-tickets/README.md +37 -0
  297. package/community-skills/zendesk-tickets/agenticmail-skill.json +65 -0
  298. package/community-skills/zoho-crm/README.md +21 -0
  299. package/community-skills/zoho-crm/agenticmail-skill.json +74 -0
  300. package/community-skills/zoom-meetings/README.md +51 -0
  301. package/community-skills/zoom-meetings/agenticmail-skill.json +66 -0
  302. package/community-skills/zuora/README.md +21 -0
  303. package/community-skills/zuora/agenticmail-skill.json +55 -0
  304. package/dashboards/README.md +81 -70
  305. package/dashboards/django/app.py +117 -0
  306. package/dashboards/django/static/styles.css +284 -0
  307. package/dashboards/django/templates/agent_detail.html +501 -0
  308. package/dashboards/django/templates/agents.html +217 -0
  309. package/dashboards/django/templates/api_keys.html +41 -0
  310. package/dashboards/django/templates/audit.html +26 -0
  311. package/dashboards/django/templates/compliance.html +33 -0
  312. package/dashboards/django/templates/components/modal.html +6 -0
  313. package/dashboards/django/templates/components/pagination.html +9 -0
  314. package/dashboards/django/templates/components/stats.html +8 -0
  315. package/dashboards/django/templates/dashboard.html +24 -0
  316. package/dashboards/django/templates/dlp.html +70 -0
  317. package/dashboards/django/templates/guardrails.html +78 -0
  318. package/dashboards/django/templates/journal.html +39 -0
  319. package/dashboards/django/templates/layout.html +52 -0
  320. package/dashboards/django/templates/login.html +30 -0
  321. package/dashboards/django/templates/messages.html +38 -0
  322. package/dashboards/django/templates/settings.html +472 -0
  323. package/dashboards/django/templates/skills.html +66 -0
  324. package/dashboards/django/templates/users.html +34 -0
  325. package/dashboards/django/templates/vault.html +46 -0
  326. package/dashboards/django/utils/__init__.py +0 -0
  327. package/dashboards/django/utils/api.py +20 -0
  328. package/dashboards/django/utils/helpers.py +39 -0
  329. package/dashboards/django/views/__init__.py +38 -0
  330. package/dashboards/django/views/agents.py +343 -0
  331. package/dashboards/django/views/api_keys.py +47 -0
  332. package/dashboards/django/views/audit.py +35 -0
  333. package/dashboards/django/views/auth.py +34 -0
  334. package/dashboards/django/views/compliance.py +37 -0
  335. package/dashboards/django/views/dashboard.py +27 -0
  336. package/dashboards/django/views/dlp.py +53 -0
  337. package/dashboards/django/views/guardrails.py +61 -0
  338. package/dashboards/django/views/journal.py +41 -0
  339. package/dashboards/django/views/messages.py +65 -0
  340. package/dashboards/django/views/settings_view.py +335 -0
  341. package/dashboards/django/views/skills.py +50 -0
  342. package/dashboards/django/views/users.py +42 -0
  343. package/dashboards/django/views/vault.py +50 -0
  344. package/dashboards/dotnet/AgenticMailDashboard.csproj +10 -0
  345. package/dashboards/dotnet/Program.cs +53 -233
  346. package/dashboards/dotnet/Routes/AgentRoutes.cs +771 -0
  347. package/dashboards/dotnet/Routes/ApiKeyRoutes.cs +185 -0
  348. package/dashboards/dotnet/Routes/AuditRoutes.cs +86 -0
  349. package/dashboards/dotnet/Routes/AuthRoutes.cs +50 -0
  350. package/dashboards/dotnet/Routes/ComplianceRoutes.cs +143 -0
  351. package/dashboards/dotnet/Routes/DashboardRoutes.cs +81 -0
  352. package/dashboards/dotnet/Routes/DlpRoutes.cs +230 -0
  353. package/dashboards/dotnet/Routes/GuardrailRoutes.cs +285 -0
  354. package/dashboards/dotnet/Routes/JournalRoutes.cs +118 -0
  355. package/dashboards/dotnet/Routes/MessageRoutes.cs +167 -0
  356. package/dashboards/dotnet/Routes/SettingRoutes.cs +907 -0
  357. package/dashboards/dotnet/Routes/SkillRoutes.cs +198 -0
  358. package/dashboards/dotnet/Routes/UserRoutes.cs +111 -0
  359. package/dashboards/dotnet/Routes/VaultRoutes.cs +162 -0
  360. package/dashboards/dotnet/Services/ApiClient.cs +103 -0
  361. package/dashboards/dotnet/Services/HtmlBuilder.cs +249 -0
  362. package/dashboards/dotnet/wwwroot/styles.css +284 -0
  363. package/dashboards/express/app.js +37 -126
  364. package/dashboards/express/middleware/auth.js +10 -0
  365. package/dashboards/express/package-lock.json +922 -0
  366. package/dashboards/express/package.json +17 -0
  367. package/dashboards/express/public/styles.css +284 -0
  368. package/dashboards/express/routes/agents.js +771 -0
  369. package/dashboards/express/routes/apiKeys.js +107 -0
  370. package/dashboards/express/routes/audit.js +75 -0
  371. package/dashboards/express/routes/auth.js +38 -0
  372. package/dashboards/express/routes/compliance.js +93 -0
  373. package/dashboards/express/routes/dashboard.js +65 -0
  374. package/dashboards/express/routes/dlp.js +165 -0
  375. package/dashboards/express/routes/guardrails.js +195 -0
  376. package/dashboards/express/routes/journal.js +95 -0
  377. package/dashboards/express/routes/messages.js +97 -0
  378. package/dashboards/express/routes/settings.js +830 -0
  379. package/dashboards/express/routes/skills.js +141 -0
  380. package/dashboards/express/routes/users.js +69 -0
  381. package/dashboards/express/routes/vault.js +141 -0
  382. package/dashboards/express/utils/api.js +44 -0
  383. package/dashboards/express/utils/helpers.js +46 -0
  384. package/dashboards/express/views/components/modal.js +255 -0
  385. package/dashboards/express/views/components/stats.js +22 -0
  386. package/dashboards/express/views/components/table.js +31 -0
  387. package/dashboards/express/views/layout.js +73 -0
  388. package/dashboards/express/views/login.js +45 -0
  389. package/dashboards/go/go.mod +3 -0
  390. package/dashboards/go/handlers/agents.go +790 -0
  391. package/dashboards/go/handlers/apikeys.go +38 -0
  392. package/dashboards/go/handlers/audit.go +52 -0
  393. package/dashboards/go/handlers/auth.go +39 -0
  394. package/dashboards/go/handlers/compliance.go +58 -0
  395. package/dashboards/go/handlers/dashboard.go +43 -0
  396. package/dashboards/go/handlers/dlp.go +78 -0
  397. package/dashboards/go/handlers/guardrails.go +89 -0
  398. package/dashboards/go/handlers/journal.go +60 -0
  399. package/dashboards/go/handlers/messages.go +59 -0
  400. package/dashboards/go/handlers/settings.go +951 -0
  401. package/dashboards/go/handlers/skills.go +129 -0
  402. package/dashboards/go/handlers/users.go +54 -0
  403. package/dashboards/go/handlers/vault.go +85 -0
  404. package/dashboards/go/main.go +35 -485
  405. package/dashboards/go/middleware/auth.go +17 -0
  406. package/dashboards/go/services/api.go +38 -0
  407. package/dashboards/go/services/session.go +52 -0
  408. package/dashboards/go/static/styles.css +284 -0
  409. package/dashboards/go/templates/components.go +45 -0
  410. package/dashboards/go/templates/helpers.go +35 -0
  411. package/dashboards/go/templates/layout.go +59 -0
  412. package/dashboards/html/index.html +203 -450
  413. package/dashboards/html/package.json +11 -0
  414. package/dashboards/html/public/styles.css +149 -0
  415. package/dashboards/html/src/api.js +36 -0
  416. package/dashboards/html/src/components/badge.js +13 -0
  417. package/dashboards/html/src/components/layout.js +11 -0
  418. package/dashboards/html/src/components/modal.js +9 -0
  419. package/dashboards/html/src/components/pagination.js +12 -0
  420. package/dashboards/html/src/components/stat-card.js +14 -0
  421. package/dashboards/html/src/components/table.js +6 -0
  422. package/dashboards/html/src/main.js +83 -0
  423. package/dashboards/html/src/pages/agents.js +831 -0
  424. package/dashboards/html/src/pages/api-keys.js +80 -0
  425. package/dashboards/html/src/pages/audit.js +50 -0
  426. package/dashboards/html/src/pages/compliance.js +69 -0
  427. package/dashboards/html/src/pages/dashboard.js +25 -0
  428. package/dashboards/html/src/pages/dlp.js +112 -0
  429. package/dashboards/html/src/pages/guardrails.js +127 -0
  430. package/dashboards/html/src/pages/journal.js +54 -0
  431. package/dashboards/html/src/pages/login.js +84 -0
  432. package/dashboards/html/src/pages/messages.js +90 -0
  433. package/dashboards/html/src/pages/settings.js +656 -0
  434. package/dashboards/html/src/pages/skills.js +114 -0
  435. package/dashboards/html/src/pages/users.js +62 -0
  436. package/dashboards/html/src/pages/vault.js +105 -0
  437. package/dashboards/html/src/router.js +40 -0
  438. package/dashboards/html/src/utils/escape.js +7 -0
  439. package/dashboards/html/src/utils/time.js +16 -0
  440. package/dashboards/html/src/utils/toast.js +10 -0
  441. package/dashboards/java/AgenticMailDashboard.java +123 -346
  442. package/dashboards/java/handlers/AgentsHandler.java +807 -0
  443. package/dashboards/java/handlers/ApiKeysHandler.java +201 -0
  444. package/dashboards/java/handlers/AuditHandler.java +95 -0
  445. package/dashboards/java/handlers/AuthHandler.java +71 -0
  446. package/dashboards/java/handlers/ComplianceHandler.java +192 -0
  447. package/dashboards/java/handlers/DashboardHandler.java +98 -0
  448. package/dashboards/java/handlers/DlpHandler.java +230 -0
  449. package/dashboards/java/handlers/GuardrailsHandler.java +272 -0
  450. package/dashboards/java/handlers/JournalHandler.java +130 -0
  451. package/dashboards/java/handlers/MessagesHandler.java +168 -0
  452. package/dashboards/java/handlers/SettingsHandler.java +1017 -0
  453. package/dashboards/java/handlers/SkillsHandler.java +198 -0
  454. package/dashboards/java/handlers/UsersHandler.java +118 -0
  455. package/dashboards/java/handlers/VaultHandler.java +168 -0
  456. package/dashboards/java/services/ApiClient.java +233 -0
  457. package/dashboards/java/services/SessionManager.java +138 -0
  458. package/dashboards/java/static/styles.css +288 -0
  459. package/dashboards/java/templates/Components.java +142 -0
  460. package/dashboards/java/templates/Helpers.java +106 -0
  461. package/dashboards/java/templates/Layout.java +98 -0
  462. package/dashboards/laravel/controllers/AgentController.php +250 -0
  463. package/dashboards/laravel/controllers/ApiKeyController.php +73 -0
  464. package/dashboards/laravel/controllers/AuditController.php +31 -0
  465. package/dashboards/laravel/controllers/AuthController.php +44 -0
  466. package/dashboards/laravel/controllers/ComplianceController.php +55 -0
  467. package/dashboards/laravel/controllers/DashboardController.php +28 -0
  468. package/dashboards/laravel/controllers/DlpController.php +88 -0
  469. package/dashboards/laravel/controllers/GuardrailController.php +110 -0
  470. package/dashboards/laravel/controllers/JournalController.php +44 -0
  471. package/dashboards/laravel/controllers/MessageController.php +47 -0
  472. package/dashboards/laravel/controllers/SettingController.php +258 -0
  473. package/dashboards/laravel/controllers/SkillController.php +74 -0
  474. package/dashboards/laravel/controllers/UserController.php +47 -0
  475. package/dashboards/laravel/controllers/VaultController.php +94 -0
  476. package/dashboards/laravel/index.php +319 -0
  477. package/dashboards/laravel/lib/Api.php +73 -0
  478. package/dashboards/laravel/lib/Helpers.php +123 -0
  479. package/dashboards/laravel/public/styles.css +291 -0
  480. package/dashboards/laravel/views/agent-detail.php +534 -0
  481. package/dashboards/laravel/views/agents.php +204 -0
  482. package/dashboards/laravel/views/api-keys.php +55 -0
  483. package/dashboards/laravel/views/audit.php +30 -0
  484. package/dashboards/laravel/views/compliance.php +48 -0
  485. package/dashboards/laravel/views/components/modal.php +59 -0
  486. package/dashboards/laravel/views/components/stats.php +21 -0
  487. package/dashboards/laravel/views/components/table.php +35 -0
  488. package/dashboards/laravel/views/dashboard.php +31 -0
  489. package/dashboards/laravel/views/dlp.php +87 -0
  490. package/dashboards/laravel/views/guardrails.php +95 -0
  491. package/dashboards/laravel/views/journal.php +50 -0
  492. package/dashboards/laravel/views/layout.php +82 -0
  493. package/dashboards/laravel/views/login.php +45 -0
  494. package/dashboards/laravel/views/messages.php +60 -0
  495. package/dashboards/laravel/views/settings.php +622 -0
  496. package/dashboards/laravel/views/skills.php +74 -0
  497. package/dashboards/laravel/views/users.php +42 -0
  498. package/dashboards/laravel/views/vault.php +80 -0
  499. package/dashboards/php/components/layout.php +80 -0
  500. package/dashboards/php/components/modal.php +22 -0
  501. package/dashboards/php/components/stats.php +20 -0
  502. package/dashboards/php/components/table.php +33 -0
  503. package/dashboards/php/index.php +663 -381
  504. package/dashboards/php/lib/api.php +37 -0
  505. package/dashboards/php/lib/auth.php +47 -0
  506. package/dashboards/php/lib/helpers.php +69 -0
  507. package/dashboards/php/pages/agent-detail.php +552 -0
  508. package/dashboards/php/pages/agents.php +185 -0
  509. package/dashboards/php/pages/api-keys.php +34 -0
  510. package/dashboards/php/pages/audit.php +31 -0
  511. package/dashboards/php/pages/compliance.php +38 -0
  512. package/dashboards/php/pages/dashboard.php +29 -0
  513. package/dashboards/php/pages/dlp.php +65 -0
  514. package/dashboards/php/pages/guardrails.php +70 -0
  515. package/dashboards/php/pages/journal.php +53 -0
  516. package/dashboards/php/pages/login.php +34 -0
  517. package/dashboards/php/pages/messages.php +53 -0
  518. package/dashboards/php/pages/settings.php +687 -0
  519. package/dashboards/php/pages/skills.php +70 -0
  520. package/dashboards/php/pages/users.php +37 -0
  521. package/dashboards/php/pages/vault.php +70 -0
  522. package/dashboards/php/public/styles.css +81 -0
  523. package/dashboards/python/app.py +61 -259
  524. package/dashboards/python/routes/__init__.py +0 -0
  525. package/dashboards/python/routes/agents.py +296 -0
  526. package/dashboards/python/routes/api_keys.py +49 -0
  527. package/dashboards/python/routes/audit.py +33 -0
  528. package/dashboards/python/routes/auth.py +30 -0
  529. package/dashboards/python/routes/compliance.py +50 -0
  530. package/dashboards/python/routes/dashboard.py +32 -0
  531. package/dashboards/python/routes/dlp.py +64 -0
  532. package/dashboards/python/routes/guardrails.py +79 -0
  533. package/dashboards/python/routes/journal.py +40 -0
  534. package/dashboards/python/routes/messages.py +46 -0
  535. package/dashboards/python/routes/settings.py +260 -0
  536. package/dashboards/python/routes/skills.py +59 -0
  537. package/dashboards/python/routes/users.py +43 -0
  538. package/dashboards/python/routes/vault.py +59 -0
  539. package/dashboards/python/static/styles.css +288 -0
  540. package/dashboards/python/templates/agent_detail.html +552 -0
  541. package/dashboards/python/templates/agents.html +246 -0
  542. package/dashboards/python/templates/api_keys.html +64 -0
  543. package/dashboards/python/templates/audit.html +50 -0
  544. package/dashboards/python/templates/compliance.html +74 -0
  545. package/dashboards/python/templates/components/modal.html +43 -0
  546. package/dashboards/python/templates/components/stats.html +26 -0
  547. package/dashboards/python/templates/components/table.html +43 -0
  548. package/dashboards/python/templates/dashboard.html +34 -0
  549. package/dashboards/python/templates/dlp.html +159 -0
  550. package/dashboards/python/templates/guardrails.html +131 -0
  551. package/dashboards/python/templates/journal.html +66 -0
  552. package/dashboards/python/templates/layout.html +86 -0
  553. package/dashboards/python/templates/login.html +36 -0
  554. package/dashboards/python/templates/messages.html +90 -0
  555. package/dashboards/python/templates/settings.html +624 -0
  556. package/dashboards/python/templates/skills.html +89 -0
  557. package/dashboards/python/templates/users.html +70 -0
  558. package/dashboards/python/templates/vault.html +73 -0
  559. package/dashboards/python/utils/__init__.py +0 -0
  560. package/dashboards/python/utils/api.py +35 -0
  561. package/dashboards/python/utils/helpers.py +90 -0
  562. package/dashboards/rails/app.rb +56 -0
  563. package/dashboards/rails/helpers/api.rb +39 -0
  564. package/dashboards/rails/helpers/auth.rb +11 -0
  565. package/dashboards/rails/helpers/view.rb +95 -0
  566. package/dashboards/rails/public/styles.css +299 -0
  567. package/dashboards/rails/routes/agents.rb +173 -0
  568. package/dashboards/rails/routes/api_keys.rb +30 -0
  569. package/dashboards/rails/routes/audit.rb +15 -0
  570. package/dashboards/rails/routes/auth.rb +29 -0
  571. package/dashboards/rails/routes/compliance.rb +25 -0
  572. package/dashboards/rails/routes/dashboard.rb +13 -0
  573. package/dashboards/rails/routes/dlp.rb +40 -0
  574. package/dashboards/rails/routes/guardrails.rb +51 -0
  575. package/dashboards/rails/routes/journal.rb +24 -0
  576. package/dashboards/rails/routes/messages.rb +22 -0
  577. package/dashboards/rails/routes/settings.rb +196 -0
  578. package/dashboards/rails/routes/skills.rb +38 -0
  579. package/dashboards/rails/routes/users.rb +18 -0
  580. package/dashboards/rails/routes/vault.rb +37 -0
  581. package/dashboards/rails/views/agent_detail.erb +550 -0
  582. package/dashboards/rails/views/agents.erb +214 -0
  583. package/dashboards/rails/views/api_keys.erb +52 -0
  584. package/dashboards/rails/views/audit.erb +38 -0
  585. package/dashboards/rails/views/compliance.erb +69 -0
  586. package/dashboards/rails/views/dashboard.erb +46 -0
  587. package/dashboards/rails/views/dlp.erb +104 -0
  588. package/dashboards/rails/views/guardrails.erb +107 -0
  589. package/dashboards/rails/views/journal.erb +65 -0
  590. package/dashboards/rails/views/layout.erb +53 -0
  591. package/dashboards/rails/views/login.erb +47 -0
  592. package/dashboards/rails/views/messages.erb +62 -0
  593. package/dashboards/rails/views/settings.erb +578 -0
  594. package/dashboards/rails/views/skills.erb +66 -0
  595. package/dashboards/rails/views/users.erb +53 -0
  596. package/dashboards/rails/views/vault.erb +66 -0
  597. package/dashboards/ruby/app.rb +53 -182
  598. package/dashboards/ruby/helpers/api.rb +63 -0
  599. package/dashboards/ruby/helpers/auth.rb +20 -0
  600. package/dashboards/ruby/helpers/view.rb +91 -0
  601. package/dashboards/ruby/public/styles.css +294 -0
  602. package/dashboards/ruby/routes/agents.rb +197 -0
  603. package/dashboards/ruby/routes/api_keys.rb +43 -0
  604. package/dashboards/ruby/routes/audit.rb +15 -0
  605. package/dashboards/ruby/routes/auth.rb +29 -0
  606. package/dashboards/ruby/routes/compliance.rb +28 -0
  607. package/dashboards/ruby/routes/dashboard.rb +13 -0
  608. package/dashboards/ruby/routes/dlp.rb +55 -0
  609. package/dashboards/ruby/routes/guardrails.rb +77 -0
  610. package/dashboards/ruby/routes/journal.rb +24 -0
  611. package/dashboards/ruby/routes/messages.rb +27 -0
  612. package/dashboards/ruby/routes/settings.rb +214 -0
  613. package/dashboards/ruby/routes/skills.rb +50 -0
  614. package/dashboards/ruby/routes/users.rb +26 -0
  615. package/dashboards/ruby/routes/vault.rb +49 -0
  616. package/dashboards/ruby/views/agent_detail.erb +550 -0
  617. package/dashboards/ruby/views/agents.erb +214 -0
  618. package/dashboards/ruby/views/api_keys.erb +59 -0
  619. package/dashboards/ruby/views/audit.erb +38 -0
  620. package/dashboards/ruby/views/compliance.erb +45 -0
  621. package/dashboards/ruby/views/dashboard.erb +46 -0
  622. package/dashboards/ruby/views/dlp.erb +88 -0
  623. package/dashboards/ruby/views/guardrails.erb +103 -0
  624. package/dashboards/ruby/views/journal.erb +53 -0
  625. package/dashboards/ruby/views/layout.erb +52 -0
  626. package/dashboards/ruby/views/login.erb +31 -0
  627. package/dashboards/ruby/views/messages.erb +51 -0
  628. package/dashboards/ruby/views/settings.erb +579 -0
  629. package/dashboards/ruby/views/skills.erb +66 -0
  630. package/dashboards/ruby/views/users.erb +53 -0
  631. package/dashboards/ruby/views/vault.erb +66 -0
  632. package/dashboards/shared/styles.css +322 -0
  633. package/dist/{chunk-TVF23PUW.js → chunk-7FVRYOP4.js} +22 -7
  634. package/dist/chunk-DRXMYYKN.js +79 -0
  635. package/dist/chunk-E23VJ3QX.js +9427 -0
  636. package/dist/chunk-EOBN6RCA.js +12652 -0
  637. package/dist/chunk-FLRYMSKY.js +23 -0
  638. package/dist/chunk-HAUHDCUB.js +764 -0
  639. package/dist/chunk-HSF6OJ5Z.js +154 -0
  640. package/dist/{chunk-PNKVD2UK.js → chunk-KFQGP6VL.js} +7 -0
  641. package/dist/chunk-LKAFZ343.js +591 -0
  642. package/dist/{chunk-GXIEEA2T.js → chunk-NTVN3JHS.js} +10 -10
  643. package/dist/chunk-RO537U6H.js +70 -0
  644. package/dist/chunk-SGBTJHEF.js +2287 -0
  645. package/dist/chunk-SMUXH6FM.js +1943 -0
  646. package/dist/chunk-TY7NVD4U.js +275 -0
  647. package/dist/chunk-X6UVWFHW.js +3455 -0
  648. package/dist/{chunk-7RGCCHIT.js → chunk-XMDE2NGH.js} +2 -1
  649. package/dist/chunk-ZNR5DDTA.js +239 -0
  650. package/dist/cidr-LISVZSM2.js +17 -0
  651. package/dist/cli-build-skill-XOYECCLE.js +235 -0
  652. package/dist/cli-recover-SSGGSKZJ.js +97 -0
  653. package/dist/cli-submit-skill-RSBLF5XN.js +162 -0
  654. package/dist/cli-validate-WJBUOEGW.js +148 -0
  655. package/dist/cli-verify-V3GPFMWU.js +98 -0
  656. package/dist/cli.js +50 -376
  657. package/dist/config-store-CRMKWBON.js +58 -0
  658. package/dist/dashboard/app.js +246 -0
  659. package/dist/dashboard/assets/logo.png +0 -0
  660. package/dist/dashboard/components/error-boundary.js +21 -0
  661. package/dist/dashboard/components/help-button.js +34 -0
  662. package/dist/dashboard/components/icons.js +53 -0
  663. package/dist/dashboard/components/modal.js +124 -0
  664. package/dist/dashboard/components/persona-fields.js +271 -0
  665. package/dist/dashboard/components/settings-help.js +191 -0
  666. package/dist/dashboard/components/tag-input.js +96 -0
  667. package/dist/dashboard/components/utils.js +148 -0
  668. package/dist/dashboard/index.html +407 -874
  669. package/dist/dashboard/pages/activity.js +62 -0
  670. package/dist/dashboard/pages/agent-detail.js +3577 -0
  671. package/dist/dashboard/pages/agents.js +1072 -0
  672. package/dist/dashboard/pages/approvals.js +81 -0
  673. package/dist/dashboard/pages/audit.js +133 -0
  674. package/dist/dashboard/pages/community-skills.js +665 -0
  675. package/dist/dashboard/pages/compliance.js +81 -0
  676. package/dist/dashboard/pages/dashboard.js +150 -0
  677. package/dist/dashboard/pages/dlp.js +107 -0
  678. package/dist/dashboard/pages/domain-status.js +123 -0
  679. package/dist/dashboard/pages/guardrails.js +995 -0
  680. package/dist/dashboard/pages/journal.js +51 -0
  681. package/dist/dashboard/pages/knowledge-contributions.js +763 -0
  682. package/dist/dashboard/pages/knowledge.js +46 -0
  683. package/dist/dashboard/pages/login.js +491 -0
  684. package/dist/dashboard/pages/messages.js +274 -0
  685. package/dist/dashboard/pages/settings.js +1554 -0
  686. package/dist/dashboard/pages/skill-connections.js +452 -0
  687. package/dist/dashboard/pages/skills.js +433 -0
  688. package/dist/dashboard/pages/users.js +49 -0
  689. package/dist/dashboard/pages/vault.js +538 -0
  690. package/dist/dashboard/pages/workforce.js +569 -0
  691. package/dist/dashboard/vendor/react-dom.production.min.js +267 -0
  692. package/dist/dashboard/vendor/react.production.min.js +31 -0
  693. package/dist/db-adapter-5PWMLY67.js +7 -0
  694. package/dist/domain-lock-URIFILHB.js +7 -0
  695. package/dist/{dynamodb-CCGL2E77.js → dynamodb-CD3LREWD.js} +4 -4
  696. package/dist/{factory-HINWFYZ3.js → factory-FVJH5RRY.js} +2 -2
  697. package/dist/firewall-AHIRE6UB.js +10 -0
  698. package/dist/index.js +133 -448
  699. package/dist/{managed-RZITNPXG.js → managed-QVTONZ7E.js} +4 -2
  700. package/dist/{mongodb-ODTXIVPV.js → mongodb-73MGW7LD.js} +4 -4
  701. package/dist/{mysql-RM3S2FV5.js → mysql-A2RYNBQG.js} +59 -6
  702. package/dist/{postgres-LN7A6MGQ.js → postgres-OE23X46P.js} +63 -6
  703. package/dist/registry/cli.js +411 -0
  704. package/dist/routes-NJK5OI5N.js +5673 -0
  705. package/dist/runtime-SMA6JUMP.js +46 -0
  706. package/dist/server-ZT5NWHT4.js +11 -0
  707. package/dist/setup-RCYNX5NA.js +20 -0
  708. package/dist/skills-QNR3CCHA.js +14 -0
  709. package/dist/soul-templates.json +1525 -0
  710. package/dist/{sqlite-VLKVAJA4.js → sqlite-SFVNZPHX.js} +56 -8
  711. package/dist/{turso-LDWODSDI.js → turso-L6WQAY7U.js} +59 -6
  712. package/live-test.mjs +1 -1
  713. package/package.json +11 -10
  714. package/schemas/agenticmail-skill.v1.schema.json +180 -0
  715. package/serve.mjs +12 -2
  716. package/src/admin/routes.ts +536 -5
  717. package/src/agent-tools/common.ts +232 -0
  718. package/src/agent-tools/index.ts +246 -0
  719. package/src/agent-tools/merge.ts +62 -0
  720. package/src/agent-tools/middleware.ts +436 -0
  721. package/src/agent-tools/security.ts +352 -0
  722. package/src/agent-tools/tools/bash.ts +154 -0
  723. package/src/agent-tools/tools/browser.ts +236 -0
  724. package/src/agent-tools/tools/edit.ts +100 -0
  725. package/src/agent-tools/tools/enterprise-calendar.ts +561 -0
  726. package/src/agent-tools/tools/enterprise-code-sandbox.ts +395 -0
  727. package/src/agent-tools/tools/enterprise-database.ts +377 -0
  728. package/src/agent-tools/tools/enterprise-diff.ts +580 -0
  729. package/src/agent-tools/tools/enterprise-documents.ts +896 -0
  730. package/src/agent-tools/tools/enterprise-finance.ts +484 -0
  731. package/src/agent-tools/tools/enterprise-http.ts +477 -0
  732. package/src/agent-tools/tools/enterprise-knowledge-search.ts +369 -0
  733. package/src/agent-tools/tools/enterprise-logs.ts +479 -0
  734. package/src/agent-tools/tools/enterprise-notifications.ts +532 -0
  735. package/src/agent-tools/tools/enterprise-security-scan.ts +528 -0
  736. package/src/agent-tools/tools/enterprise-spreadsheet.ts +825 -0
  737. package/src/agent-tools/tools/enterprise-translation.ts +530 -0
  738. package/src/agent-tools/tools/enterprise-vision.ts +348 -0
  739. package/src/agent-tools/tools/enterprise-web-research.ts +422 -0
  740. package/src/agent-tools/tools/enterprise-workflow.ts +308 -0
  741. package/src/agent-tools/tools/glob.ts +129 -0
  742. package/src/agent-tools/tools/grep.ts +178 -0
  743. package/src/agent-tools/tools/memory.ts +194 -0
  744. package/src/agent-tools/tools/read.ts +160 -0
  745. package/src/agent-tools/tools/web-fetch-utils.ts +202 -0
  746. package/src/agent-tools/tools/web-fetch.ts +456 -0
  747. package/src/agent-tools/tools/web-search.ts +480 -0
  748. package/src/agent-tools/tools/web-shared.ts +232 -0
  749. package/src/agent-tools/tools/write.ts +68 -0
  750. package/src/agent-tools/types.ts +193 -0
  751. package/src/auth/routes.ts +916 -51
  752. package/src/cli.ts +74 -14
  753. package/src/dashboard/app.js +246 -0
  754. package/src/dashboard/assets/logo.png +0 -0
  755. package/src/dashboard/components/error-boundary.js +21 -0
  756. package/src/dashboard/components/help-button.js +34 -0
  757. package/src/dashboard/components/icons.js +53 -0
  758. package/src/dashboard/components/modal.js +124 -0
  759. package/src/dashboard/components/persona-fields.js +271 -0
  760. package/src/dashboard/components/settings-help.js +191 -0
  761. package/src/dashboard/components/tag-input.js +96 -0
  762. package/src/dashboard/components/utils.js +148 -0
  763. package/src/dashboard/index.html +407 -874
  764. package/src/dashboard/pages/activity.js +62 -0
  765. package/src/dashboard/pages/agent-detail.js +3577 -0
  766. package/src/dashboard/pages/agents.js +1072 -0
  767. package/src/dashboard/pages/approvals.js +81 -0
  768. package/src/dashboard/pages/audit.js +133 -0
  769. package/src/dashboard/pages/community-skills.js +665 -0
  770. package/src/dashboard/pages/compliance.js +81 -0
  771. package/src/dashboard/pages/dashboard.js +150 -0
  772. package/src/dashboard/pages/dlp.js +107 -0
  773. package/src/dashboard/pages/domain-status.js +123 -0
  774. package/src/dashboard/pages/guardrails.js +995 -0
  775. package/src/dashboard/pages/journal.js +51 -0
  776. package/src/dashboard/pages/knowledge-contributions.js +763 -0
  777. package/src/dashboard/pages/knowledge.js +46 -0
  778. package/src/dashboard/pages/login.js +491 -0
  779. package/src/dashboard/pages/messages.js +274 -0
  780. package/src/dashboard/pages/settings.js +1554 -0
  781. package/src/dashboard/pages/skill-connections.js +452 -0
  782. package/src/dashboard/pages/skills.js +433 -0
  783. package/src/dashboard/pages/users.js +49 -0
  784. package/src/dashboard/pages/vault.js +538 -0
  785. package/src/dashboard/pages/workforce.js +569 -0
  786. package/src/dashboard/vendor/react-dom.production.min.js +267 -0
  787. package/src/dashboard/vendor/react.production.min.js +31 -0
  788. package/src/db/adapter.ts +134 -1
  789. package/src/db/dynamodb.ts +2 -2
  790. package/src/db/mongodb.ts +2 -2
  791. package/src/db/mysql.ts +55 -1
  792. package/src/db/postgres.ts +59 -1
  793. package/src/db/proxy.ts +39 -0
  794. package/src/db/sql-schema.ts +2 -1
  795. package/src/db/sqlite.ts +51 -3
  796. package/src/db/turso.ts +55 -1
  797. package/src/deploy/fly.ts +2 -2
  798. package/src/deploy/managed.ts +32 -11
  799. package/src/domain-lock/cli-recover.ts +124 -0
  800. package/src/domain-lock/cli-verify.ts +122 -0
  801. package/src/domain-lock/index.ts +220 -0
  802. package/src/engine/activity-routes.ts +116 -0
  803. package/src/engine/activity.ts +62 -2
  804. package/src/engine/agent-config.ts +190 -24
  805. package/src/engine/agent-memory.ts +1062 -0
  806. package/src/engine/agent-routes.ts +379 -0
  807. package/src/engine/approvals.ts +309 -2
  808. package/src/engine/catalog-routes.ts +155 -0
  809. package/src/engine/cli-build-skill.ts +285 -0
  810. package/src/engine/cli-submit-skill.ts +200 -0
  811. package/src/engine/cli-validate.ts +188 -0
  812. package/src/engine/communication-routes.ts +139 -0
  813. package/src/engine/communication.ts +759 -0
  814. package/src/engine/community-registry.ts +1509 -0
  815. package/src/engine/community-routes.ts +233 -0
  816. package/src/engine/compliance-routes.ts +93 -0
  817. package/src/engine/compliance.ts +325 -0
  818. package/src/engine/db-adapter.ts +349 -1
  819. package/src/engine/db-schema.ts +971 -23
  820. package/src/engine/deploy-schema-routes.ts +176 -0
  821. package/src/engine/deployer.ts +6 -6
  822. package/src/engine/dlp-routes.ts +59 -0
  823. package/src/engine/dlp.ts +231 -0
  824. package/src/engine/guardrail-routes.ts +125 -0
  825. package/src/engine/guardrails.ts +403 -0
  826. package/src/engine/index.ts +131 -10
  827. package/src/engine/journal-routes.ts +56 -0
  828. package/src/engine/journal.ts +249 -0
  829. package/src/engine/knowledge-contribution-routes.ts +385 -0
  830. package/src/engine/knowledge-contribution.ts +1319 -0
  831. package/src/engine/knowledge-routes.ts +68 -0
  832. package/src/engine/knowledge.ts +73 -1
  833. package/src/engine/lifecycle.ts +487 -31
  834. package/src/engine/memory-routes.ts +142 -0
  835. package/src/engine/oauth-connect-routes.ts +366 -0
  836. package/src/engine/oauth-connect.ts +304 -0
  837. package/src/engine/onboarding-routes.ts +140 -0
  838. package/src/engine/onboarding.ts +574 -0
  839. package/src/engine/org-approval-routes.ts +146 -0
  840. package/src/engine/org-policies.ts +497 -0
  841. package/src/engine/policy-import-routes.ts +125 -0
  842. package/src/engine/policy-import.ts +1186 -0
  843. package/src/engine/policy-routes.ts +163 -0
  844. package/src/engine/routes.ts +239 -482
  845. package/src/engine/skill-updater-routes.ts +132 -0
  846. package/src/engine/skill-updater.ts +480 -0
  847. package/src/engine/skill-validator.ts +331 -0
  848. package/src/engine/skills/enterprise-calendar.ts +142 -0
  849. package/src/engine/skills/enterprise-code-sandbox.ts +112 -0
  850. package/src/engine/skills/enterprise-database.ts +122 -0
  851. package/src/engine/skills/enterprise-diff.ts +94 -0
  852. package/src/engine/skills/enterprise-documents.ts +161 -0
  853. package/src/engine/skills/enterprise-finance.ts +109 -0
  854. package/src/engine/skills/enterprise-http.ts +98 -0
  855. package/src/engine/skills/enterprise-knowledge-search.ts +106 -0
  856. package/src/engine/skills/enterprise-logs.ts +115 -0
  857. package/src/engine/skills/enterprise-notifications.ts +119 -0
  858. package/src/engine/skills/enterprise-security-scan.ts +124 -0
  859. package/src/engine/skills/enterprise-spreadsheet.ts +170 -0
  860. package/src/engine/skills/enterprise-translation.ts +111 -0
  861. package/src/engine/skills/enterprise-vision.ts +107 -0
  862. package/src/engine/skills/enterprise-web-research.ts +114 -0
  863. package/src/engine/skills/enterprise-workflow.ts +109 -0
  864. package/src/engine/skills/gws-admin.ts +17 -0
  865. package/src/engine/skills/gws-calendar.ts +19 -0
  866. package/src/engine/skills/gws-chat.ts +16 -0
  867. package/src/engine/skills/gws-docs.ts +17 -0
  868. package/src/engine/skills/gws-drive.ts +20 -0
  869. package/src/engine/skills/gws-forms.ts +16 -0
  870. package/src/engine/skills/gws-gmail.ts +20 -0
  871. package/src/engine/skills/gws-groups.ts +16 -0
  872. package/src/engine/skills/gws-keep.ts +16 -0
  873. package/src/engine/skills/gws-meet.ts +16 -0
  874. package/src/engine/skills/gws-sheets.ts +18 -0
  875. package/src/engine/skills/gws-sites.ts +15 -0
  876. package/src/engine/skills/gws-slides.ts +16 -0
  877. package/src/engine/skills/gws-vault.ts +16 -0
  878. package/src/engine/skills/index.ts +121 -0
  879. package/src/engine/skills/m365-admin.ts +17 -0
  880. package/src/engine/skills/m365-bookings.ts +16 -0
  881. package/src/engine/skills/m365-copilot.ts +16 -0
  882. package/src/engine/skills/m365-excel.ts +59 -0
  883. package/src/engine/skills/m365-forms.ts +16 -0
  884. package/src/engine/skills/m365-onedrive.ts +59 -0
  885. package/src/engine/skills/m365-onenote.ts +16 -0
  886. package/src/engine/skills/m365-outlook.ts +26 -0
  887. package/src/engine/skills/m365-planner.ts +17 -0
  888. package/src/engine/skills/m365-power-automate.ts +17 -0
  889. package/src/engine/skills/m365-power-bi.ts +18 -0
  890. package/src/engine/skills/m365-powerpoint.ts +32 -0
  891. package/src/engine/skills/m365-sharepoint.ts +19 -0
  892. package/src/engine/skills/m365-teams.ts +20 -0
  893. package/src/engine/skills/m365-todo.ts +16 -0
  894. package/src/engine/skills/m365-whiteboard.ts +15 -0
  895. package/src/engine/skills/m365-word.ts +41 -0
  896. package/src/engine/skills.ts +353 -17
  897. package/src/engine/soul-library.ts +142 -0
  898. package/src/engine/soul-templates.json +1525 -0
  899. package/src/engine/storage-manager.ts +252 -0
  900. package/src/engine/storage-routes.ts +113 -0
  901. package/src/engine/storage.ts +528 -0
  902. package/src/engine/tenant.ts +69 -5
  903. package/src/engine/tool-catalog.ts +177 -13
  904. package/src/engine/vault-routes.ts +130 -0
  905. package/src/engine/vault.ts +544 -0
  906. package/src/engine/workforce-routes.ts +282 -0
  907. package/src/engine/workforce.ts +957 -0
  908. package/src/index.ts +9 -1
  909. package/src/lib/cidr.ts +122 -0
  910. package/src/lib/config-store.ts +86 -0
  911. package/src/mcp/adapters/activecampaign.adapter.ts +391 -0
  912. package/src/mcp/adapters/adobe-sign.adapter.ts +469 -0
  913. package/src/mcp/adapters/adp.adapter.ts +358 -0
  914. package/src/mcp/adapters/airtable.adapter.ts +273 -0
  915. package/src/mcp/adapters/apollo.adapter.ts +420 -0
  916. package/src/mcp/adapters/asana.adapter.ts +315 -0
  917. package/src/mcp/adapters/auth0.adapter.ts +386 -0
  918. package/src/mcp/adapters/aws.adapter.ts +345 -0
  919. package/src/mcp/adapters/azure-devops.adapter.ts +389 -0
  920. package/src/mcp/adapters/bamboohr.adapter.ts +376 -0
  921. package/src/mcp/adapters/basecamp.adapter.ts +366 -0
  922. package/src/mcp/adapters/bigcommerce.adapter.ts +429 -0
  923. package/src/mcp/adapters/bitbucket.adapter.ts +260 -0
  924. package/src/mcp/adapters/box.adapter.ts +350 -0
  925. package/src/mcp/adapters/brex.adapter.ts +367 -0
  926. package/src/mcp/adapters/buffer.adapter.ts +303 -0
  927. package/src/mcp/adapters/calendly.adapter.ts +262 -0
  928. package/src/mcp/adapters/canva.adapter.ts +256 -0
  929. package/src/mcp/adapters/chargebee.adapter.ts +448 -0
  930. package/src/mcp/adapters/circleci.adapter.ts +216 -0
  931. package/src/mcp/adapters/clickup.adapter.ts +335 -0
  932. package/src/mcp/adapters/close.adapter.ts +390 -0
  933. package/src/mcp/adapters/cloudflare.adapter.ts +377 -0
  934. package/src/mcp/adapters/confluence.adapter.ts +301 -0
  935. package/src/mcp/adapters/contentful.adapter.ts +355 -0
  936. package/src/mcp/adapters/copper.adapter.ts +468 -0
  937. package/src/mcp/adapters/crisp.adapter.ts +415 -0
  938. package/src/mcp/adapters/crowdstrike.adapter.ts +413 -0
  939. package/src/mcp/adapters/datadog.adapter.ts +373 -0
  940. package/src/mcp/adapters/digitalocean.adapter.ts +336 -0
  941. package/src/mcp/adapters/discord.adapter.ts +248 -0
  942. package/src/mcp/adapters/docker.adapter.ts +238 -0
  943. package/src/mcp/adapters/docusign.adapter.ts +431 -0
  944. package/src/mcp/adapters/drift.adapter.ts +386 -0
  945. package/src/mcp/adapters/dropbox.adapter.ts +315 -0
  946. package/src/mcp/adapters/figma.adapter.ts +302 -0
  947. package/src/mcp/adapters/firebase.adapter.ts +446 -0
  948. package/src/mcp/adapters/flyio.adapter.ts +302 -0
  949. package/src/mcp/adapters/freshbooks.adapter.ts +474 -0
  950. package/src/mcp/adapters/freshdesk.adapter.ts +441 -0
  951. package/src/mcp/adapters/freshsales.adapter.ts +457 -0
  952. package/src/mcp/adapters/freshservice.adapter.ts +481 -0
  953. package/src/mcp/adapters/front.adapter.ts +357 -0
  954. package/src/mcp/adapters/github-actions.adapter.ts +329 -0
  955. package/src/mcp/adapters/github.adapter.ts +387 -0
  956. package/src/mcp/adapters/gitlab.adapter.ts +368 -0
  957. package/src/mcp/adapters/gong.adapter.ts +386 -0
  958. package/src/mcp/adapters/google-ads.adapter.ts +363 -0
  959. package/src/mcp/adapters/google-analytics.adapter.ts +316 -0
  960. package/src/mcp/adapters/google-cloud.adapter.ts +312 -0
  961. package/src/mcp/adapters/google-drive.adapter.ts +387 -0
  962. package/src/mcp/adapters/gotomeeting.adapter.ts +255 -0
  963. package/src/mcp/adapters/grafana.adapter.ts +361 -0
  964. package/src/mcp/adapters/greenhouse.adapter.ts +354 -0
  965. package/src/mcp/adapters/gusto.adapter.ts +329 -0
  966. package/src/mcp/adapters/hashicorp-vault.adapter.ts +355 -0
  967. package/src/mcp/adapters/heroku.adapter.ts +291 -0
  968. package/src/mcp/adapters/hibob.adapter.ts +334 -0
  969. package/src/mcp/adapters/hootsuite.adapter.ts +322 -0
  970. package/src/mcp/adapters/hubspot.adapter.ts +400 -0
  971. package/src/mcp/adapters/huggingface.adapter.ts +349 -0
  972. package/src/mcp/adapters/index.ts +530 -0
  973. package/src/mcp/adapters/intercom.adapter.ts +269 -0
  974. package/src/mcp/adapters/jira.adapter.ts +482 -0
  975. package/src/mcp/adapters/klaviyo.adapter.ts +353 -0
  976. package/src/mcp/adapters/kubernetes.adapter.ts +431 -0
  977. package/src/mcp/adapters/lattice.adapter.ts +339 -0
  978. package/src/mcp/adapters/launchdarkly.adapter.ts +368 -0
  979. package/src/mcp/adapters/lever.adapter.ts +347 -0
  980. package/src/mcp/adapters/linear.adapter.ts +300 -0
  981. package/src/mcp/adapters/linkedin.adapter.ts +331 -0
  982. package/src/mcp/adapters/livechat.adapter.ts +259 -0
  983. package/src/mcp/adapters/loom.adapter.ts +230 -0
  984. package/src/mcp/adapters/mailchimp.adapter.ts +394 -0
  985. package/src/mcp/adapters/mailgun.adapter.ts +425 -0
  986. package/src/mcp/adapters/microsoft-teams.adapter.ts +251 -0
  987. package/src/mcp/adapters/miro.adapter.ts +274 -0
  988. package/src/mcp/adapters/mixpanel.adapter.ts +324 -0
  989. package/src/mcp/adapters/monday.adapter.ts +308 -0
  990. package/src/mcp/adapters/mongodb-atlas.adapter.ts +345 -0
  991. package/src/mcp/adapters/neon.adapter.ts +312 -0
  992. package/src/mcp/adapters/netlify.adapter.ts +324 -0
  993. package/src/mcp/adapters/netsuite.adapter.ts +411 -0
  994. package/src/mcp/adapters/newrelic.adapter.ts +339 -0
  995. package/src/mcp/adapters/notion.adapter.ts +338 -0
  996. package/src/mcp/adapters/okta.adapter.ts +394 -0
  997. package/src/mcp/adapters/openai.adapter.ts +315 -0
  998. package/src/mcp/adapters/opsgenie.adapter.ts +375 -0
  999. package/src/mcp/adapters/outreach.adapter.ts +372 -0
  1000. package/src/mcp/adapters/paddle.adapter.ts +467 -0
  1001. package/src/mcp/adapters/pagerduty.adapter.ts +412 -0
  1002. package/src/mcp/adapters/pandadoc.adapter.ts +389 -0
  1003. package/src/mcp/adapters/paypal.adapter.ts +465 -0
  1004. package/src/mcp/adapters/personio.adapter.ts +401 -0
  1005. package/src/mcp/adapters/pinecone.adapter.ts +340 -0
  1006. package/src/mcp/adapters/pipedrive.adapter.ts +324 -0
  1007. package/src/mcp/adapters/plaid.adapter.ts +444 -0
  1008. package/src/mcp/adapters/postmark.adapter.ts +387 -0
  1009. package/src/mcp/adapters/power-automate.adapter.ts +388 -0
  1010. package/src/mcp/adapters/quickbooks.adapter.ts +431 -0
  1011. package/src/mcp/adapters/recurly.adapter.ts +433 -0
  1012. package/src/mcp/adapters/reddit.adapter.ts +371 -0
  1013. package/src/mcp/adapters/render.adapter.ts +332 -0
  1014. package/src/mcp/adapters/ringcentral.adapter.ts +281 -0
  1015. package/src/mcp/adapters/rippling.adapter.ts +287 -0
  1016. package/src/mcp/adapters/salesforce.adapter.ts +321 -0
  1017. package/src/mcp/adapters/salesloft.adapter.ts +413 -0
  1018. package/src/mcp/adapters/sanity.adapter.ts +363 -0
  1019. package/src/mcp/adapters/sap.adapter.ts +483 -0
  1020. package/src/mcp/adapters/segment.adapter.ts +260 -0
  1021. package/src/mcp/adapters/sendgrid.adapter.ts +265 -0
  1022. package/src/mcp/adapters/sentry.adapter.ts +331 -0
  1023. package/src/mcp/adapters/servicenow.adapter.ts +468 -0
  1024. package/src/mcp/adapters/shopify.adapter.ts +451 -0
  1025. package/src/mcp/adapters/shortcut.adapter.ts +290 -0
  1026. package/src/mcp/adapters/slack.adapter.ts +380 -0
  1027. package/src/mcp/adapters/smartsheet.adapter.ts +326 -0
  1028. package/src/mcp/adapters/snowflake.adapter.ts +347 -0
  1029. package/src/mcp/adapters/snyk.adapter.ts +394 -0
  1030. package/src/mcp/adapters/splunk.adapter.ts +403 -0
  1031. package/src/mcp/adapters/square.adapter.ts +467 -0
  1032. package/src/mcp/adapters/statuspage.adapter.ts +401 -0
  1033. package/src/mcp/adapters/stripe.adapter.ts +380 -0
  1034. package/src/mcp/adapters/supabase.adapter.ts +334 -0
  1035. package/src/mcp/adapters/teamwork.adapter.ts +404 -0
  1036. package/src/mcp/adapters/telegram.adapter.ts +299 -0
  1037. package/src/mcp/adapters/terraform.adapter.ts +300 -0
  1038. package/src/mcp/adapters/todoist.adapter.ts +239 -0
  1039. package/src/mcp/adapters/trello.adapter.ts +316 -0
  1040. package/src/mcp/adapters/twilio.adapter.ts +233 -0
  1041. package/src/mcp/adapters/twitter.adapter.ts +348 -0
  1042. package/src/mcp/adapters/vercel.adapter.ts +219 -0
  1043. package/src/mcp/adapters/weaviate.adapter.ts +371 -0
  1044. package/src/mcp/adapters/webex.adapter.ts +237 -0
  1045. package/src/mcp/adapters/webflow.adapter.ts +287 -0
  1046. package/src/mcp/adapters/whatsapp.adapter.ts +273 -0
  1047. package/src/mcp/adapters/whereby.adapter.ts +240 -0
  1048. package/src/mcp/adapters/woocommerce.adapter.ts +454 -0
  1049. package/src/mcp/adapters/wordpress.adapter.ts +455 -0
  1050. package/src/mcp/adapters/workday.adapter.ts +354 -0
  1051. package/src/mcp/adapters/wrike.adapter.ts +349 -0
  1052. package/src/mcp/adapters/xero.adapter.ts +472 -0
  1053. package/src/mcp/adapters/youtube.adapter.ts +401 -0
  1054. package/src/mcp/adapters/zendesk.adapter.ts +399 -0
  1055. package/src/mcp/adapters/zoho-crm.adapter.ts +410 -0
  1056. package/src/mcp/adapters/zoom.adapter.ts +241 -0
  1057. package/src/mcp/adapters/zuora.adapter.ts +476 -0
  1058. package/src/mcp/framework/api-executor.ts +192 -0
  1059. package/src/mcp/framework/aws-sigv4.ts +216 -0
  1060. package/src/mcp/framework/credential-resolver.ts +128 -0
  1061. package/src/mcp/framework/skill-mcp-framework.ts +226 -0
  1062. package/src/mcp/framework/types.ts +130 -0
  1063. package/src/mcp/index.ts +124 -0
  1064. package/src/middleware/egress-filter.ts +81 -0
  1065. package/src/middleware/firewall.ts +121 -0
  1066. package/src/middleware/index.ts +33 -0
  1067. package/src/registry/cli.ts +63 -0
  1068. package/src/registry/server.ts +504 -0
  1069. package/src/runtime/agent-loop.ts +504 -0
  1070. package/src/runtime/email-channel.ts +120 -0
  1071. package/src/runtime/followup.ts +211 -0
  1072. package/src/runtime/gateway.ts +260 -0
  1073. package/src/runtime/hooks.ts +472 -0
  1074. package/src/runtime/index.ts +679 -0
  1075. package/src/runtime/llm-client.ts +1019 -0
  1076. package/src/runtime/providers.ts +231 -0
  1077. package/src/runtime/session-manager.ts +340 -0
  1078. package/src/runtime/subagent.ts +154 -0
  1079. package/src/runtime/tool-executor.ts +202 -0
  1080. package/src/runtime/types.ts +215 -0
  1081. package/src/server.ts +157 -45
  1082. package/src/setup/company.ts +127 -8
  1083. package/src/setup/database.ts +2 -1
  1084. package/src/setup/domain.ts +33 -4
  1085. package/src/setup/index.ts +19 -7
  1086. package/src/setup/provision.ts +128 -8
  1087. package/src/setup/registration.ts +227 -0
  1088. package/src/types/hono-env.ts +14 -0
  1089. package/src/types/optional-deps.d.ts +10 -0
  1090. package/start-live.mjs +12 -3
  1091. package/test-integration.mjs +628 -257
  1092. package/dist/chunk-77IDQJL3.js +0 -7
  1093. package/dist/chunk-BE7MXVLA.js +0 -757
  1094. package/dist/chunk-BS2WCSHO.js +0 -48
  1095. package/dist/chunk-FL3VQBGL.js +0 -757
  1096. package/dist/chunk-IQWA44WT.js +0 -970
  1097. package/dist/chunk-N2JVTNNJ.js +0 -2553
  1098. package/dist/chunk-YFDSE4BW.js +0 -1355
  1099. package/dist/db-adapter-DEWEFNIV.js +0 -7
  1100. package/dist/factory-V37IG5AT.js +0 -9
  1101. package/dist/routes-2JEPIIKC.js +0 -441
  1102. package/dist/server-32YYCI3A.js +0 -8
  1103. package/dist/server-H3C6WUOS.js +0 -8
  1104. package/dist/sqlite-3K5YOZ4K.js +0 -439
  1105. package/src/engine/agenticmail-bridge.ts +0 -296
  1106. package/src/engine/openclaw-hook.ts +0 -371
@@ -0,0 +1,1319 @@
1
+ /**
2
+ * Knowledge Contribution System
3
+ *
4
+ * Agents contribute knowledge from their memory to shared knowledge bases,
5
+ * categorized by role and domain. Future agents in the same role get
6
+ * bootstrapped with this collective knowledge.
7
+ *
8
+ * - Weekly scheduled contribution cycles
9
+ * - Role-based knowledge bases (support, sales, engineering, etc.)
10
+ * - Category taxonomy within each base
11
+ * - Quality scoring, deduplication, and decay
12
+ * - Contribution approval workflow (optional)
13
+ */
14
+
15
+ // ─── Types ──────────────────────────────────────────────
16
+
17
+ export type KnowledgeRole =
18
+ | 'support'
19
+ | 'sales'
20
+ | 'engineering'
21
+ | 'operations'
22
+ | 'hr'
23
+ | 'finance'
24
+ | 'legal'
25
+ | 'marketing'
26
+ | 'research'
27
+ | 'security'
28
+ | 'executive'
29
+ | 'general';
30
+
31
+ export interface KnowledgeBase {
32
+ id: string;
33
+ orgId: string;
34
+ name: string;
35
+ description: string;
36
+ role: KnowledgeRole;
37
+ categories: KnowledgeCategory[];
38
+ contributorCount: number;
39
+ entryCount: number;
40
+ lastContributionAt: string;
41
+ createdAt: string;
42
+ updatedAt: string;
43
+ }
44
+
45
+ export interface KnowledgeCategory {
46
+ id: string;
47
+ name: string;
48
+ description: string;
49
+ parentId?: string;
50
+ entryCount: number;
51
+ }
52
+
53
+ export interface KnowledgeEntry {
54
+ id: string;
55
+ baseId: string;
56
+ orgId: string;
57
+ categoryId: string;
58
+ title: string;
59
+ content: string;
60
+ summary: string;
61
+ tags: string[];
62
+ sourceAgentId: string;
63
+ sourceMemoryId?: string;
64
+ confidence: number;
65
+ qualityScore: number;
66
+ useCount: number;
67
+ voteUp: number;
68
+ voteDown: number;
69
+ status: 'pending' | 'approved' | 'rejected' | 'archived';
70
+ createdAt: string;
71
+ updatedAt: string;
72
+ expiresAt?: string;
73
+ }
74
+
75
+ export interface ContributionSchedule {
76
+ id: string;
77
+ orgId: string;
78
+ agentId: string;
79
+ baseId: string;
80
+ frequency: 'daily' | 'weekly' | 'biweekly' | 'monthly';
81
+ dayOfWeek?: number;
82
+ lastRunAt?: string;
83
+ nextRunAt: string;
84
+ enabled: boolean;
85
+ filters: {
86
+ minConfidence: number;
87
+ categories?: string[];
88
+ minImportance?: string;
89
+ };
90
+ createdAt: string;
91
+ }
92
+
93
+ export interface ContributionCycle {
94
+ id: string;
95
+ orgId: string;
96
+ agentId: string;
97
+ baseId: string;
98
+ scheduleId: string;
99
+ status: 'running' | 'completed' | 'failed';
100
+ memoriesScanned: number;
101
+ entriesContributed: number;
102
+ duplicatesSkipped: number;
103
+ startedAt: string;
104
+ completedAt?: string;
105
+ error?: string;
106
+ }
107
+
108
+ // ─── Helpers ─────────────────────────────────────────────
109
+
110
+ function uid(): string {
111
+ return Date.now().toString(36) + Math.random().toString(36).slice(2, 10);
112
+ }
113
+
114
+ /** Compute a simple Jaccard-like similarity between two strings based on word-level token overlap. */
115
+ function contentSimilarity(a: string, b: string): number {
116
+ const tokenize = (text: string): Set<string> => {
117
+ const tokens = new Set<string>();
118
+ const words = text.toLowerCase().replace(/[^a-z0-9\s]/g, '').split(/\s+/);
119
+ for (const w of words) {
120
+ if (w.length > 2) tokens.add(w);
121
+ }
122
+ return tokens;
123
+ };
124
+
125
+ const tokensA = tokenize(a);
126
+ const tokensB = tokenize(b);
127
+ if (tokensA.size === 0 || tokensB.size === 0) return 0;
128
+
129
+ let intersection = 0;
130
+ for (const t of tokensA) {
131
+ if (tokensB.has(t)) intersection++;
132
+ }
133
+
134
+ const union = new Set([...tokensA, ...tokensB]).size;
135
+ return union > 0 ? intersection / union : 0;
136
+ }
137
+
138
+ /** Compute a quality score from votes, usage, and freshness. */
139
+ function computeQualityScore(entry: {
140
+ voteUp: number;
141
+ voteDown: number;
142
+ useCount: number;
143
+ confidence: number;
144
+ createdAt: string;
145
+ }): number {
146
+ // Vote component: net votes (capped at reasonable range)
147
+ const netVotes = entry.voteUp - entry.voteDown;
148
+ const voteScore = Math.min(1, Math.max(0, (netVotes + 5) / 10)); // -5..+5 -> 0..1
149
+
150
+ // Usage component: logarithmic usage
151
+ const usageScore = Math.min(1, Math.log1p(entry.useCount) / Math.log1p(50)); // 0..50 -> 0..1
152
+
153
+ // Freshness component: decay over 180 days
154
+ const ageMs = Date.now() - new Date(entry.createdAt).getTime();
155
+ const ageDays = ageMs / 86_400_000;
156
+ const freshnessScore = Math.max(0.1, 1 - (ageDays / 180));
157
+
158
+ // Weighted combination
159
+ const quality = (
160
+ entry.confidence * 0.3 +
161
+ voteScore * 0.25 +
162
+ usageScore * 0.25 +
163
+ freshnessScore * 0.2
164
+ );
165
+
166
+ return parseFloat(Math.min(1, Math.max(0, quality)).toFixed(4));
167
+ }
168
+
169
+ /** Compute the next run timestamp from a frequency and optional dayOfWeek. */
170
+ function computeNextRunAt(frequency: ContributionSchedule['frequency'], dayOfWeek?: number): string {
171
+ const now = new Date();
172
+ let next: Date;
173
+
174
+ switch (frequency) {
175
+ case 'daily':
176
+ next = new Date(now.getTime() + 86_400_000);
177
+ break;
178
+ case 'weekly': {
179
+ next = new Date(now.getTime());
180
+ const targetDay = dayOfWeek ?? 1; // default Monday
181
+ const currentDay = now.getDay();
182
+ let daysUntil = targetDay - currentDay;
183
+ if (daysUntil <= 0) daysUntil += 7;
184
+ next.setDate(next.getDate() + daysUntil);
185
+ next.setHours(2, 0, 0, 0); // 2 AM
186
+ break;
187
+ }
188
+ case 'biweekly': {
189
+ next = new Date(now.getTime());
190
+ const targetDay2 = dayOfWeek ?? 1;
191
+ const currentDay2 = now.getDay();
192
+ let daysUntil2 = targetDay2 - currentDay2;
193
+ if (daysUntil2 <= 0) daysUntil2 += 7;
194
+ next.setDate(next.getDate() + daysUntil2 + 7);
195
+ next.setHours(2, 0, 0, 0);
196
+ break;
197
+ }
198
+ case 'monthly':
199
+ next = new Date(now.getFullYear(), now.getMonth() + 1, 1, 2, 0, 0, 0);
200
+ break;
201
+ default:
202
+ next = new Date(now.getTime() + 7 * 86_400_000);
203
+ }
204
+
205
+ return next.toISOString();
206
+ }
207
+
208
+ // ─── Available Roles ────────────────────────────────────
209
+
210
+ const AVAILABLE_ROLES: Array<{ id: KnowledgeRole; name: string; description: string }> = [
211
+ { id: 'support', name: 'Support', description: 'Customer support and success teams' },
212
+ { id: 'sales', name: 'Sales', description: 'Sales and business development teams' },
213
+ { id: 'engineering', name: 'Engineering', description: 'Software engineering and development teams' },
214
+ { id: 'operations', name: 'Operations', description: 'Business operations teams' },
215
+ { id: 'hr', name: 'Human Resources', description: 'HR and people operations teams' },
216
+ { id: 'finance', name: 'Finance', description: 'Finance and accounting teams' },
217
+ { id: 'legal', name: 'Legal', description: 'Legal and compliance teams' },
218
+ { id: 'marketing', name: 'Marketing', description: 'Marketing and growth teams' },
219
+ { id: 'research', name: 'Research', description: 'Research and data science teams' },
220
+ { id: 'security', name: 'Security', description: 'Information security teams' },
221
+ { id: 'executive', name: 'Executive', description: 'Executive leadership and strategy' },
222
+ { id: 'general', name: 'General', description: 'Cross-functional shared knowledge' },
223
+ ];
224
+
225
+ // ─── Default Category Taxonomy ──────────────────────────
226
+
227
+ const ROLE_CATEGORIES: Record<string, Omit<KnowledgeCategory, 'entryCount'>[]> = {
228
+ support: [
229
+ { id: 'troubleshooting', name: 'Troubleshooting', description: 'Common issues and solutions' },
230
+ { id: 'product-knowledge', name: 'Product Knowledge', description: 'Product features, limitations, and workarounds' },
231
+ { id: 'customer-patterns', name: 'Customer Patterns', description: 'Common customer behaviors and needs' },
232
+ { id: 'escalation-playbooks', name: 'Escalation Playbooks', description: 'When and how to escalate issues' },
233
+ { id: 'faq-responses', name: 'FAQ Responses', description: 'Frequently asked questions and best responses' },
234
+ { id: 'tools-integrations', name: 'Tools & Integrations', description: 'How to use support tools effectively' },
235
+ ],
236
+ sales: [
237
+ { id: 'objection-handling', name: 'Objection Handling', description: 'Responses to common objections' },
238
+ { id: 'competitor-intel', name: 'Competitor Intelligence', description: 'Competitor strengths, weaknesses, positioning' },
239
+ { id: 'pricing-strategies', name: 'Pricing Strategies', description: 'Effective pricing and discount approaches' },
240
+ { id: 'qualification-criteria', name: 'Qualification Criteria', description: 'Lead qualification patterns' },
241
+ { id: 'pitch-templates', name: 'Pitch Templates', description: 'Effective pitch structures and talking points' },
242
+ { id: 'deal-patterns', name: 'Deal Patterns', description: 'Successful deal closure patterns' },
243
+ ],
244
+ engineering: [
245
+ { id: 'architecture-decisions', name: 'Architecture Decisions', description: 'ADRs and design rationale' },
246
+ { id: 'debugging-patterns', name: 'Debugging Patterns', description: 'Common bugs and debugging approaches' },
247
+ { id: 'code-standards', name: 'Code Standards', description: 'Coding conventions and best practices' },
248
+ { id: 'infrastructure', name: 'Infrastructure', description: 'Infrastructure setup and operations knowledge' },
249
+ { id: 'incident-learnings', name: 'Incident Learnings', description: 'Post-mortem insights and prevention' },
250
+ { id: 'tool-configs', name: 'Tool Configurations', description: 'Development tool setups and configs' },
251
+ ],
252
+ operations: [
253
+ { id: 'process-docs', name: 'Process Documentation', description: 'Standard operating procedures' },
254
+ { id: 'vendor-management', name: 'Vendor Management', description: 'Vendor relationships and negotiations' },
255
+ { id: 'compliance-notes', name: 'Compliance Notes', description: 'Regulatory compliance knowledge' },
256
+ { id: 'efficiency-tips', name: 'Efficiency Tips', description: 'Process optimization insights' },
257
+ ],
258
+ hr: [
259
+ { id: 'policy-interpretations', name: 'Policy Interpretations', description: 'HR policy applications and edge cases' },
260
+ { id: 'interview-guides', name: 'Interview Guides', description: 'Interview questions and evaluation criteria' },
261
+ { id: 'onboarding-tips', name: 'Onboarding Tips', description: 'Employee onboarding best practices' },
262
+ { id: 'benefits-faq', name: 'Benefits FAQ', description: 'Benefits questions and answers' },
263
+ ],
264
+ finance: [
265
+ { id: 'accounting-procedures', name: 'Accounting Procedures', description: 'Standard accounting workflows' },
266
+ { id: 'tax-knowledge', name: 'Tax Knowledge', description: 'Tax rules and compliance notes' },
267
+ { id: 'budget-templates', name: 'Budget Templates', description: 'Budget planning templates and guides' },
268
+ { id: 'audit-prep', name: 'Audit Preparation', description: 'Audit preparation checklists and tips' },
269
+ ],
270
+ legal: [
271
+ { id: 'contract-templates', name: 'Contract Templates', description: 'Standard contract clauses and templates' },
272
+ { id: 'regulatory-updates', name: 'Regulatory Updates', description: 'Regulatory changes and implications' },
273
+ { id: 'risk-assessments', name: 'Risk Assessments', description: 'Legal risk evaluation patterns' },
274
+ ],
275
+ marketing: [
276
+ { id: 'campaign-playbooks', name: 'Campaign Playbooks', description: 'Successful campaign strategies' },
277
+ { id: 'content-guidelines', name: 'Content Guidelines', description: 'Brand voice and content standards' },
278
+ { id: 'channel-strategies', name: 'Channel Strategies', description: 'Channel-specific marketing approaches' },
279
+ { id: 'analytics-insights', name: 'Analytics Insights', description: 'Marketing analytics patterns' },
280
+ ],
281
+ research: [
282
+ { id: 'methodologies', name: 'Research Methodologies', description: 'Research methods and frameworks' },
283
+ { id: 'data-sources', name: 'Data Sources', description: 'Useful data sources and APIs' },
284
+ { id: 'findings-library', name: 'Findings Library', description: 'Key research findings and insights' },
285
+ ],
286
+ security: [
287
+ { id: 'threat-intel', name: 'Threat Intelligence', description: 'Known threats and attack patterns' },
288
+ { id: 'incident-response', name: 'Incident Response', description: 'IR procedures and playbooks' },
289
+ { id: 'security-configs', name: 'Security Configurations', description: 'Hardening guides and configs' },
290
+ ],
291
+ executive: [
292
+ { id: 'strategy-frameworks', name: 'Strategy Frameworks', description: 'Strategic planning frameworks' },
293
+ { id: 'market-analysis', name: 'Market Analysis', description: 'Market trends and analysis' },
294
+ { id: 'decision-frameworks', name: 'Decision Frameworks', description: 'Decision-making frameworks and criteria' },
295
+ ],
296
+ general: [
297
+ { id: 'best-practices', name: 'Best Practices', description: 'Cross-functional best practices' },
298
+ { id: 'tools-guides', name: 'Tools & Guides', description: 'Tool usage guides and tips' },
299
+ { id: 'lessons-learned', name: 'Lessons Learned', description: 'Cross-team lessons learned' },
300
+ { id: 'templates', name: 'Templates', description: 'Reusable templates and frameworks' },
301
+ ],
302
+ };
303
+
304
+ // ─── Memory Callback Type ───────────────────────────────
305
+
306
+ export type MemoryCallback = (agentId: string) => Promise<MemoryItem[]>;
307
+
308
+ /** Shape of a memory item returned by the memory callback. */
309
+ export interface MemoryItem {
310
+ id: string;
311
+ agentId: string;
312
+ orgId: string;
313
+ category: string;
314
+ title: string;
315
+ content: string;
316
+ source: string;
317
+ importance: string;
318
+ confidence: number;
319
+ tags: string[];
320
+ createdAt: string;
321
+ }
322
+
323
+ // ─── Knowledge Contribution Manager ────────────────────
324
+
325
+ export class KnowledgeContributionManager {
326
+ private bases = new Map<string, KnowledgeBase>();
327
+ private entries = new Map<string, KnowledgeEntry>();
328
+ private schedules = new Map<string, ContributionSchedule>();
329
+ private cycles = new Map<string, ContributionCycle>();
330
+ private memoryCallback?: MemoryCallback;
331
+ private schedulerTimer?: ReturnType<typeof setInterval>;
332
+
333
+ /** Per-base index: baseId -> Set of entry IDs for O(1) base lookups */
334
+ private baseEntryIndex = new Map<string, Set<string>>();
335
+ /** Per-agent index: agentId -> Set of entry IDs */
336
+ private agentEntryIndex = new Map<string, Set<string>>();
337
+ /** Per-agent schedule index: agentId -> scheduleId */
338
+ private agentScheduleIndex = new Map<string, string>();
339
+ /** Contributor tracking: baseId -> Set of agent IDs that contributed */
340
+ private baseContributors = new Map<string, Set<string>>();
341
+
342
+ constructor(opts?: { memoryCallback?: MemoryCallback }) {
343
+ this.memoryCallback = opts?.memoryCallback;
344
+ }
345
+
346
+ // ─── Index Helpers ──────────────────────────────────
347
+
348
+ private indexEntryAdd(entry: KnowledgeEntry): void {
349
+ // Base index
350
+ let baseSet = this.baseEntryIndex.get(entry.baseId);
351
+ if (!baseSet) { baseSet = new Set(); this.baseEntryIndex.set(entry.baseId, baseSet); }
352
+ baseSet.add(entry.id);
353
+
354
+ // Agent index
355
+ let agentSet = this.agentEntryIndex.get(entry.sourceAgentId);
356
+ if (!agentSet) { agentSet = new Set(); this.agentEntryIndex.set(entry.sourceAgentId, agentSet); }
357
+ agentSet.add(entry.id);
358
+
359
+ // Contributor tracking
360
+ let contributors = this.baseContributors.get(entry.baseId);
361
+ if (!contributors) { contributors = new Set(); this.baseContributors.set(entry.baseId, contributors); }
362
+ contributors.add(entry.sourceAgentId);
363
+ }
364
+
365
+ private indexEntryRemove(entry: KnowledgeEntry): void {
366
+ const baseSet = this.baseEntryIndex.get(entry.baseId);
367
+ if (baseSet) { baseSet.delete(entry.id); if (baseSet.size === 0) this.baseEntryIndex.delete(entry.baseId); }
368
+
369
+ const agentSet = this.agentEntryIndex.get(entry.sourceAgentId);
370
+ if (agentSet) { agentSet.delete(entry.id); if (agentSet.size === 0) this.agentEntryIndex.delete(entry.sourceAgentId); }
371
+ }
372
+
373
+ private getEntriesForBase(baseId: string): KnowledgeEntry[] {
374
+ const ids = this.baseEntryIndex.get(baseId);
375
+ if (!ids || ids.size === 0) return [];
376
+ const result: KnowledgeEntry[] = [];
377
+ for (const id of ids) {
378
+ const entry = this.entries.get(id);
379
+ if (entry) result.push(entry);
380
+ }
381
+ return result;
382
+ }
383
+
384
+ private getEntriesForAgent(agentId: string): KnowledgeEntry[] {
385
+ const ids = this.agentEntryIndex.get(agentId);
386
+ if (!ids || ids.size === 0) return [];
387
+ const result: KnowledgeEntry[] = [];
388
+ for (const id of ids) {
389
+ const entry = this.entries.get(id);
390
+ if (entry) result.push(entry);
391
+ }
392
+ return result;
393
+ }
394
+
395
+ // ─── Knowledge Base CRUD ────────────────────────────
396
+
397
+ /**
398
+ * Create a knowledge base.
399
+ * Accepts either (orgId, { name, description, role }) positional args
400
+ * or a single object { orgId, name, description, role, createdBy? }.
401
+ */
402
+ createBase(
403
+ orgIdOrOpts: string | { orgId: string; name: string; description: string; role: string; createdBy?: string },
404
+ opts?: { name: string; description: string; role: string },
405
+ ): KnowledgeBase {
406
+ let orgId: string;
407
+ let name: string;
408
+ let description: string;
409
+ let role: string;
410
+
411
+ if (typeof orgIdOrOpts === 'object') {
412
+ orgId = orgIdOrOpts.orgId;
413
+ name = orgIdOrOpts.name;
414
+ description = orgIdOrOpts.description;
415
+ role = orgIdOrOpts.role;
416
+ } else {
417
+ orgId = orgIdOrOpts;
418
+ if (!opts) throw new Error('Options required when orgId is a string');
419
+ name = opts.name;
420
+ description = opts.description;
421
+ role = opts.role;
422
+ }
423
+
424
+ const now = new Date().toISOString();
425
+ const roleCategories = ROLE_CATEGORIES[role] || ROLE_CATEGORIES.general;
426
+ const categories: KnowledgeCategory[] = roleCategories.map(c => ({
427
+ ...c,
428
+ entryCount: 0,
429
+ }));
430
+
431
+ const base: KnowledgeBase = {
432
+ id: uid(),
433
+ orgId,
434
+ name,
435
+ description,
436
+ role: role as KnowledgeRole,
437
+ categories,
438
+ contributorCount: 0,
439
+ entryCount: 0,
440
+ lastContributionAt: now,
441
+ createdAt: now,
442
+ updatedAt: now,
443
+ };
444
+
445
+ this.bases.set(base.id, base);
446
+ return base;
447
+ }
448
+
449
+ getBase(baseId: string): KnowledgeBase | undefined {
450
+ return this.bases.get(baseId);
451
+ }
452
+
453
+ /**
454
+ * List knowledge bases for an org.
455
+ * The second argument can be a string (role name) or { role? } filter object.
456
+ */
457
+ listBases(orgId: string, filters?: string | { role?: KnowledgeRole }): KnowledgeBase[] {
458
+ let results = Array.from(this.bases.values()).filter(b => b.orgId === orgId);
459
+
460
+ const roleFilter = typeof filters === 'string' ? filters : filters?.role;
461
+ if (roleFilter) {
462
+ results = results.filter(b => b.role === roleFilter);
463
+ }
464
+
465
+ return results;
466
+ }
467
+
468
+ /**
469
+ * Delete a knowledge base and all associated entries, schedules, and contributor tracking.
470
+ */
471
+ deleteBase(baseId: string, _userId?: string): void {
472
+ const base = this.bases.get(baseId);
473
+ if (!base) return;
474
+
475
+ // Remove all entries for this base
476
+ const entryIds = this.baseEntryIndex.get(baseId);
477
+ if (entryIds) {
478
+ for (const id of entryIds) {
479
+ const entry = this.entries.get(id);
480
+ if (entry) {
481
+ const agentSet = this.agentEntryIndex.get(entry.sourceAgentId);
482
+ if (agentSet) { agentSet.delete(id); if (agentSet.size === 0) this.agentEntryIndex.delete(entry.sourceAgentId); }
483
+ }
484
+ this.entries.delete(id);
485
+ }
486
+ this.baseEntryIndex.delete(baseId);
487
+ }
488
+
489
+ // Remove contributor tracking
490
+ this.baseContributors.delete(baseId);
491
+
492
+ // Remove schedules targeting this base
493
+ for (const [scheduleId, schedule] of Array.from(this.schedules.entries())) {
494
+ if (schedule.baseId === baseId) {
495
+ this.agentScheduleIndex.delete(schedule.agentId);
496
+ this.schedules.delete(scheduleId);
497
+ }
498
+ }
499
+
500
+ this.bases.delete(baseId);
501
+ }
502
+
503
+ // ─── Roles ──────────────────────────────────────────
504
+
505
+ /** List all available roles with metadata. */
506
+ listRoles(): Array<{ id: KnowledgeRole; name: string; description: string; categoryCount: number }> {
507
+ return AVAILABLE_ROLES.map(r => ({
508
+ ...r,
509
+ categoryCount: (ROLE_CATEGORIES[r.id] || []).length,
510
+ }));
511
+ }
512
+
513
+ // ─── Category Management ────────────────────────────
514
+
515
+ addCategory(baseId: string, category: { id?: string; name: string; description: string; parentId?: string }): KnowledgeCategory {
516
+ const base = this.bases.get(baseId);
517
+ if (!base) throw new Error(`Knowledge base ${baseId} not found`);
518
+
519
+ const cat: KnowledgeCategory = {
520
+ id: category.id || uid(),
521
+ name: category.name,
522
+ description: category.description,
523
+ parentId: category.parentId,
524
+ entryCount: 0,
525
+ };
526
+
527
+ // Check for duplicate ID
528
+ const existing = base.categories.find(c => c.id === cat.id);
529
+ if (existing) throw new Error(`Category ${cat.id} already exists in base ${baseId}`);
530
+
531
+ base.categories.push(cat);
532
+ base.updatedAt = new Date().toISOString();
533
+ return cat;
534
+ }
535
+
536
+ removeCategory(baseId: string, categoryId: string): void {
537
+ const base = this.bases.get(baseId);
538
+ if (!base) throw new Error(`Knowledge base ${baseId} not found`);
539
+
540
+ const idx = base.categories.findIndex(c => c.id === categoryId);
541
+ if (idx < 0) throw new Error(`Category ${categoryId} not found in base ${baseId}`);
542
+
543
+ base.categories.splice(idx, 1);
544
+ base.updatedAt = new Date().toISOString();
545
+ }
546
+
547
+ getCategoriesForRole(role: string): KnowledgeCategory[] {
548
+ const defs = ROLE_CATEGORIES[role] || ROLE_CATEGORIES.general;
549
+ return defs.map(d => ({ ...d, entryCount: 0 }));
550
+ }
551
+
552
+ // ─── Entries ────────────────────────────────────────
553
+
554
+ /**
555
+ * Low-level contribute: all fields provided explicitly including baseId and orgId.
556
+ * Used internally by contribution cycles.
557
+ */
558
+ contribute(opts: {
559
+ baseId: string;
560
+ orgId: string;
561
+ categoryId: string;
562
+ title: string;
563
+ content: string;
564
+ summary: string;
565
+ tags: string[];
566
+ sourceAgentId: string;
567
+ sourceMemoryId?: string;
568
+ confidence: number;
569
+ }): KnowledgeEntry {
570
+ const base = this.bases.get(opts.baseId);
571
+ if (!base) throw new Error(`Knowledge base ${opts.baseId} not found`);
572
+
573
+ // Validate category exists in the base
574
+ const category = base.categories.find(c => c.id === opts.categoryId);
575
+ if (!category) throw new Error(`Category ${opts.categoryId} not found in base ${opts.baseId}`);
576
+
577
+ const now = new Date().toISOString();
578
+
579
+ const entry: KnowledgeEntry = {
580
+ id: uid(),
581
+ baseId: opts.baseId,
582
+ orgId: opts.orgId,
583
+ categoryId: opts.categoryId,
584
+ title: opts.title,
585
+ content: opts.content,
586
+ summary: opts.summary,
587
+ tags: [...opts.tags],
588
+ sourceAgentId: opts.sourceAgentId,
589
+ sourceMemoryId: opts.sourceMemoryId,
590
+ confidence: Math.min(1, Math.max(0, opts.confidence)),
591
+ qualityScore: 0,
592
+ useCount: 0,
593
+ voteUp: 0,
594
+ voteDown: 0,
595
+ status: 'pending',
596
+ createdAt: now,
597
+ updatedAt: now,
598
+ };
599
+
600
+ // Compute initial quality score
601
+ entry.qualityScore = computeQualityScore(entry);
602
+
603
+ this.entries.set(entry.id, entry);
604
+ this.indexEntryAdd(entry);
605
+
606
+ // Update base stats
607
+ base.entryCount = (this.baseEntryIndex.get(opts.baseId)?.size) || 0;
608
+ base.contributorCount = (this.baseContributors.get(opts.baseId)?.size) || 0;
609
+ base.lastContributionAt = now;
610
+ base.updatedAt = now;
611
+
612
+ // Update category entry count
613
+ category.entryCount = this.getEntriesForBase(opts.baseId)
614
+ .filter(e => e.categoryId === opts.categoryId).length;
615
+
616
+ return entry;
617
+ }
618
+
619
+ /**
620
+ * Route-friendly contribute: baseId as first arg, entry fields as second arg.
621
+ * Resolves orgId from the base.
622
+ */
623
+ contributeEntry(baseId: string, opts: {
624
+ categoryId: string;
625
+ title: string;
626
+ content: string;
627
+ summary?: string;
628
+ tags?: string[];
629
+ sourceAgentId?: string;
630
+ sourceMemoryId?: string;
631
+ confidence?: number;
632
+ contributedBy?: string;
633
+ }): KnowledgeEntry {
634
+ const base = this.bases.get(baseId);
635
+ if (!base) throw new Error(`Knowledge base ${baseId} not found`);
636
+
637
+ return this.contribute({
638
+ baseId,
639
+ orgId: base.orgId,
640
+ categoryId: opts.categoryId,
641
+ title: opts.title,
642
+ content: opts.content,
643
+ summary: opts.summary || '',
644
+ tags: opts.tags || [],
645
+ sourceAgentId: opts.sourceAgentId || opts.contributedBy || 'unknown',
646
+ sourceMemoryId: opts.sourceMemoryId,
647
+ confidence: opts.confidence ?? 1.0,
648
+ });
649
+ }
650
+
651
+ getEntry(entryId: string): KnowledgeEntry | undefined {
652
+ return this.entries.get(entryId);
653
+ }
654
+
655
+ listEntries(baseId: string, filters?: {
656
+ categoryId?: string;
657
+ status?: KnowledgeEntry['status'];
658
+ minQuality?: number;
659
+ search?: string;
660
+ limit?: number;
661
+ offset?: number;
662
+ }): { entries: KnowledgeEntry[]; total: number } {
663
+ let results = this.getEntriesForBase(baseId);
664
+
665
+ if (filters?.categoryId) {
666
+ results = results.filter(e => e.categoryId === filters.categoryId);
667
+ }
668
+
669
+ if (filters?.status) {
670
+ results = results.filter(e => e.status === filters.status);
671
+ }
672
+
673
+ if (filters?.minQuality !== undefined) {
674
+ results = results.filter(e => e.qualityScore >= filters.minQuality!);
675
+ }
676
+
677
+ if (filters?.search) {
678
+ const q = filters.search.toLowerCase();
679
+ results = results.filter(e =>
680
+ e.title.toLowerCase().includes(q) ||
681
+ e.content.toLowerCase().includes(q) ||
682
+ e.summary.toLowerCase().includes(q) ||
683
+ e.tags.some(t => t.toLowerCase().includes(q))
684
+ );
685
+ }
686
+
687
+ // Sort by quality score descending
688
+ results.sort((a, b) => b.qualityScore - a.qualityScore);
689
+
690
+ const total = results.length;
691
+ const offset = filters?.offset || 0;
692
+ const limit = filters?.limit || 50;
693
+ const paged = results.slice(offset, offset + limit);
694
+
695
+ return { entries: paged, total };
696
+ }
697
+
698
+ approveEntry(entryId: string, _userId?: string): KnowledgeEntry | undefined {
699
+ const entry = this.entries.get(entryId);
700
+ if (!entry) return undefined;
701
+ entry.status = 'approved';
702
+ entry.updatedAt = new Date().toISOString();
703
+ return entry;
704
+ }
705
+
706
+ rejectEntry(entryId: string, _userId?: string): KnowledgeEntry | undefined {
707
+ const entry = this.entries.get(entryId);
708
+ if (!entry) return undefined;
709
+ entry.status = 'rejected';
710
+ entry.updatedAt = new Date().toISOString();
711
+ return entry;
712
+ }
713
+
714
+ archiveEntry(entryId: string, _userId?: string): KnowledgeEntry | undefined {
715
+ const entry = this.entries.get(entryId);
716
+ if (!entry) return undefined;
717
+ entry.status = 'archived';
718
+ entry.updatedAt = new Date().toISOString();
719
+ return entry;
720
+ }
721
+
722
+ /**
723
+ * Vote on an entry. Accepts either (entryId, direction) or (entryId, userId, direction).
724
+ */
725
+ vote(entryId: string, userIdOrDirection: string, directionArg?: 'up' | 'down'): KnowledgeEntry | undefined {
726
+ const entry = this.entries.get(entryId);
727
+ if (!entry) return undefined;
728
+
729
+ const direction: 'up' | 'down' = directionArg ?? (userIdOrDirection as 'up' | 'down');
730
+
731
+ if (direction === 'up') {
732
+ entry.voteUp++;
733
+ } else {
734
+ entry.voteDown++;
735
+ }
736
+
737
+ entry.qualityScore = computeQualityScore(entry);
738
+ entry.updatedAt = new Date().toISOString();
739
+ return entry;
740
+ }
741
+
742
+ /** Legacy alias for vote(). */
743
+ voteEntry(entryId: string, direction: 'up' | 'down'): void {
744
+ this.vote(entryId, direction);
745
+ }
746
+
747
+ recordUsage(entryId: string, _agentId?: string): void {
748
+ const entry = this.entries.get(entryId);
749
+ if (!entry) return;
750
+
751
+ entry.useCount++;
752
+ entry.qualityScore = computeQualityScore(entry);
753
+ entry.updatedAt = new Date().toISOString();
754
+ }
755
+
756
+ // ─── Deduplication ──────────────────────────────────
757
+
758
+ findSimilar(baseId: string, content: string, threshold: number = 0.6): KnowledgeEntry[] {
759
+ const baseEntries = this.getEntriesForBase(baseId);
760
+ const matches: Array<{ entry: KnowledgeEntry; similarity: number }> = [];
761
+
762
+ for (const entry of baseEntries) {
763
+ if (entry.status === 'rejected' || entry.status === 'archived') continue;
764
+
765
+ const similarity = contentSimilarity(content, entry.content);
766
+ if (similarity >= threshold) {
767
+ matches.push({ entry, similarity });
768
+ }
769
+ }
770
+
771
+ // Sort by similarity descending
772
+ matches.sort((a, b) => b.similarity - a.similarity);
773
+ return matches.map(m => m.entry);
774
+ }
775
+
776
+ // ─── Schedules ──────────────────────────────────────
777
+
778
+ /**
779
+ * Create a contribution schedule.
780
+ * Accepts filters with defaults for minConfidence.
781
+ */
782
+ createSchedule(opts: {
783
+ orgId: string;
784
+ agentId: string;
785
+ baseId: string;
786
+ frequency: ContributionSchedule['frequency'];
787
+ dayOfWeek?: number;
788
+ filters?: Partial<ContributionSchedule['filters']>;
789
+ createdBy?: string;
790
+ }): ContributionSchedule {
791
+ // Validate base exists
792
+ const base = this.bases.get(opts.baseId);
793
+ if (!base) throw new Error(`Knowledge base ${opts.baseId} not found`);
794
+
795
+ // Validate dayOfWeek range
796
+ if (opts.dayOfWeek !== undefined && (opts.dayOfWeek < 0 || opts.dayOfWeek > 6)) {
797
+ throw new Error('dayOfWeek must be between 0 (Sunday) and 6 (Saturday)');
798
+ }
799
+
800
+ const filters = opts.filters || {};
801
+
802
+ const schedule: ContributionSchedule = {
803
+ id: uid(),
804
+ orgId: opts.orgId,
805
+ agentId: opts.agentId,
806
+ baseId: opts.baseId,
807
+ frequency: opts.frequency,
808
+ dayOfWeek: opts.dayOfWeek,
809
+ nextRunAt: computeNextRunAt(opts.frequency, opts.dayOfWeek),
810
+ enabled: true,
811
+ filters: {
812
+ minConfidence: filters.minConfidence ?? 0.5,
813
+ categories: filters.categories,
814
+ minImportance: filters.minImportance,
815
+ },
816
+ createdAt: new Date().toISOString(),
817
+ };
818
+
819
+ this.schedules.set(schedule.id, schedule);
820
+ this.agentScheduleIndex.set(opts.agentId, schedule.id);
821
+
822
+ return schedule;
823
+ }
824
+
825
+ /** Get a schedule by agent ID. */
826
+ getSchedule(agentId: string): ContributionSchedule | undefined {
827
+ const scheduleId = this.agentScheduleIndex.get(agentId);
828
+ if (!scheduleId) return undefined;
829
+ return this.schedules.get(scheduleId);
830
+ }
831
+
832
+ /** Alias used by routes. */
833
+ getScheduleForAgent(agentId: string): ContributionSchedule | undefined {
834
+ return this.getSchedule(agentId);
835
+ }
836
+
837
+ listSchedules(orgId: string): ContributionSchedule[] {
838
+ return Array.from(this.schedules.values()).filter(s => s.orgId === orgId);
839
+ }
840
+
841
+ updateSchedule(
842
+ scheduleId: string,
843
+ updates: Partial<Pick<ContributionSchedule, 'frequency' | 'dayOfWeek' | 'enabled' | 'filters'>>,
844
+ _userId?: string,
845
+ ): ContributionSchedule | undefined {
846
+ const schedule = this.schedules.get(scheduleId);
847
+ if (!schedule) return undefined;
848
+
849
+ if (updates.frequency !== undefined) {
850
+ schedule.frequency = updates.frequency;
851
+ schedule.nextRunAt = computeNextRunAt(updates.frequency, updates.dayOfWeek ?? schedule.dayOfWeek);
852
+ }
853
+
854
+ if (updates.dayOfWeek !== undefined) {
855
+ if (updates.dayOfWeek < 0 || updates.dayOfWeek > 6) {
856
+ throw new Error('dayOfWeek must be between 0 (Sunday) and 6 (Saturday)');
857
+ }
858
+ schedule.dayOfWeek = updates.dayOfWeek;
859
+ }
860
+
861
+ if (updates.enabled !== undefined) {
862
+ schedule.enabled = updates.enabled;
863
+ }
864
+
865
+ if (updates.filters !== undefined) {
866
+ schedule.filters = { ...schedule.filters, ...updates.filters };
867
+ }
868
+
869
+ return schedule;
870
+ }
871
+
872
+ deleteSchedule(scheduleId: string, _userId?: string): void {
873
+ const schedule = this.schedules.get(scheduleId);
874
+ if (!schedule) return;
875
+
876
+ this.agentScheduleIndex.delete(schedule.agentId);
877
+ this.schedules.delete(scheduleId);
878
+ }
879
+
880
+ // ─── Contribution Cycles ────────────────────────────
881
+
882
+ async runContributionCycle(agentId: string): Promise<ContributionCycle> {
883
+ const schedule = this.getSchedule(agentId);
884
+ if (!schedule) throw new Error(`No contribution schedule found for agent ${agentId}`);
885
+
886
+ const base = this.bases.get(schedule.baseId);
887
+ if (!base) throw new Error(`Knowledge base ${schedule.baseId} not found`);
888
+
889
+ const now = new Date().toISOString();
890
+ const cycle: ContributionCycle = {
891
+ id: uid(),
892
+ orgId: schedule.orgId,
893
+ agentId,
894
+ baseId: schedule.baseId,
895
+ scheduleId: schedule.id,
896
+ status: 'running',
897
+ memoriesScanned: 0,
898
+ entriesContributed: 0,
899
+ duplicatesSkipped: 0,
900
+ startedAt: now,
901
+ };
902
+
903
+ this.cycles.set(cycle.id, cycle);
904
+
905
+ try {
906
+ // Fetch memories via callback
907
+ if (!this.memoryCallback) {
908
+ throw new Error('No memory callback configured — cannot fetch agent memories');
909
+ }
910
+
911
+ const memories = await this.memoryCallback(agentId);
912
+ cycle.memoriesScanned = memories.length;
913
+
914
+ // Filter memories based on schedule criteria
915
+ let filtered = memories.filter(m => m.confidence >= schedule.filters.minConfidence);
916
+
917
+ if (schedule.filters.minImportance) {
918
+ const importanceLevels: Record<string, number> = {
919
+ low: 1,
920
+ normal: 2,
921
+ high: 3,
922
+ critical: 4,
923
+ };
924
+ const minLevel = importanceLevels[schedule.filters.minImportance] || 0;
925
+ filtered = filtered.filter(m => (importanceLevels[m.importance] || 0) >= minLevel);
926
+ }
927
+
928
+ if (schedule.filters.categories && schedule.filters.categories.length > 0) {
929
+ filtered = filtered.filter(m => schedule.filters.categories!.includes(m.category));
930
+ }
931
+
932
+ // Process each filtered memory
933
+ for (const memory of filtered) {
934
+ // Check for duplicates via content similarity
935
+ const similars = this.findSimilar(schedule.baseId, memory.content, 0.6);
936
+ if (similars.length > 0) {
937
+ cycle.duplicatesSkipped++;
938
+ continue;
939
+ }
940
+
941
+ // Determine best category match from the base categories
942
+ const categoryId = this.matchCategory(memory, base.categories);
943
+
944
+ // Generate a summary from the content (first 200 chars or full content if shorter)
945
+ const summary = memory.content.length > 200
946
+ ? memory.content.slice(0, 200).replace(/\s+\S*$/, '') + '...'
947
+ : memory.content;
948
+
949
+ try {
950
+ this.contribute({
951
+ baseId: schedule.baseId,
952
+ orgId: schedule.orgId,
953
+ categoryId,
954
+ title: memory.title,
955
+ content: memory.content,
956
+ summary,
957
+ tags: memory.tags || [],
958
+ sourceAgentId: agentId,
959
+ sourceMemoryId: memory.id,
960
+ confidence: memory.confidence,
961
+ });
962
+
963
+ cycle.entriesContributed++;
964
+ } catch (err: any) {
965
+ // Log but continue with remaining memories
966
+ console.error(`[knowledge-contribution] Failed to contribute memory ${memory.id}:`, err.message);
967
+ }
968
+ }
969
+
970
+ cycle.status = 'completed';
971
+ cycle.completedAt = new Date().toISOString();
972
+
973
+ // Update schedule timestamps
974
+ schedule.lastRunAt = now;
975
+ schedule.nextRunAt = computeNextRunAt(schedule.frequency, schedule.dayOfWeek);
976
+
977
+ } catch (err: any) {
978
+ cycle.status = 'failed';
979
+ cycle.error = err.message;
980
+ cycle.completedAt = new Date().toISOString();
981
+ }
982
+
983
+ return cycle;
984
+ }
985
+
986
+ /** Alias used by routes: trigger a contribution cycle for an agent. */
987
+ async triggerContribution(agentId: string, _userId?: string): Promise<ContributionCycle> {
988
+ return this.runContributionCycle(agentId);
989
+ }
990
+
991
+ /** Find the best-matching category in a base for a given memory item. */
992
+ private matchCategory(memory: MemoryItem, categories: KnowledgeCategory[]): string {
993
+ if (categories.length === 0) return 'general';
994
+
995
+ // Try matching memory category/tags against base category IDs and names
996
+ const memCat = memory.category.toLowerCase();
997
+ const memTitle = memory.title.toLowerCase();
998
+ const memTags = memory.tags.map(t => t.toLowerCase());
999
+
1000
+ let bestMatch: string | undefined;
1001
+ let bestScore = 0;
1002
+
1003
+ for (const cat of categories) {
1004
+ let score = 0;
1005
+ const catId = cat.id.toLowerCase();
1006
+ const catName = cat.name.toLowerCase();
1007
+ const catDesc = cat.description.toLowerCase();
1008
+
1009
+ // Direct category match
1010
+ if (memCat.includes(catId) || catId.includes(memCat)) score += 3;
1011
+
1012
+ // Title overlap
1013
+ if (memTitle.includes(catId) || memTitle.includes(catName)) score += 2;
1014
+
1015
+ // Tag matches
1016
+ for (const tag of memTags) {
1017
+ if (catId.includes(tag) || catName.includes(tag) || tag.includes(catId)) score += 1;
1018
+ }
1019
+
1020
+ // Description keyword overlap
1021
+ const descWords = catDesc.split(/\s+/).filter(w => w.length > 3);
1022
+ for (const word of descWords) {
1023
+ if (memTitle.includes(word) || memory.content.toLowerCase().includes(word)) score += 0.5;
1024
+ }
1025
+
1026
+ if (score > bestScore) {
1027
+ bestScore = score;
1028
+ bestMatch = cat.id;
1029
+ }
1030
+ }
1031
+
1032
+ // Fall back to the first category if no meaningful match found
1033
+ return bestMatch || categories[0].id;
1034
+ }
1035
+
1036
+ checkDueSchedules(): ContributionSchedule[] {
1037
+ const now = new Date().toISOString();
1038
+ return Array.from(this.schedules.values()).filter(s =>
1039
+ s.enabled && s.nextRunAt <= now
1040
+ );
1041
+ }
1042
+
1043
+ async runDueContributions(): Promise<ContributionCycle[]> {
1044
+ const due = this.checkDueSchedules();
1045
+ const results: ContributionCycle[] = [];
1046
+
1047
+ for (const schedule of due) {
1048
+ try {
1049
+ const cycle = await this.runContributionCycle(schedule.agentId);
1050
+ results.push(cycle);
1051
+ } catch (err: any) {
1052
+ console.error(`[knowledge-contribution] Failed contribution cycle for agent ${schedule.agentId}:`, err.message);
1053
+ const failedCycle: ContributionCycle = {
1054
+ id: uid(),
1055
+ orgId: schedule.orgId,
1056
+ agentId: schedule.agentId,
1057
+ baseId: schedule.baseId,
1058
+ scheduleId: schedule.id,
1059
+ status: 'failed',
1060
+ memoriesScanned: 0,
1061
+ entriesContributed: 0,
1062
+ duplicatesSkipped: 0,
1063
+ startedAt: new Date().toISOString(),
1064
+ completedAt: new Date().toISOString(),
1065
+ error: err.message,
1066
+ };
1067
+ this.cycles.set(failedCycle.id, failedCycle);
1068
+ results.push(failedCycle);
1069
+ }
1070
+ }
1071
+
1072
+ return results;
1073
+ }
1074
+
1075
+ /** Alias used by routes. */
1076
+ async runDueCycles(_userId?: string): Promise<{ cyclesRun: number; completed: number; failed: number; cycles: ContributionCycle[] }> {
1077
+ const cycles = await this.runDueContributions();
1078
+ return {
1079
+ cyclesRun: cycles.length,
1080
+ completed: cycles.filter(c => c.status === 'completed').length,
1081
+ failed: cycles.filter(c => c.status === 'failed').length,
1082
+ cycles,
1083
+ };
1084
+ }
1085
+
1086
+ /** List contribution cycles with optional filters. */
1087
+ listCycles(opts?: { orgId?: string; agentId?: string; limit?: number }): ContributionCycle[] {
1088
+ let results = Array.from(this.cycles.values());
1089
+
1090
+ if (opts?.orgId) {
1091
+ results = results.filter(c => c.orgId === opts.orgId);
1092
+ }
1093
+ if (opts?.agentId) {
1094
+ results = results.filter(c => c.agentId === opts.agentId);
1095
+ }
1096
+
1097
+ // Sort by startedAt descending (most recent first)
1098
+ results.sort((a, b) => b.startedAt.localeCompare(a.startedAt));
1099
+
1100
+ const limit = opts?.limit || 50;
1101
+ return results.slice(0, limit);
1102
+ }
1103
+
1104
+ // ─── Bootstrap ──────────────────────────────────────
1105
+
1106
+ getBootstrapKnowledge(role: string, opts?: {
1107
+ categories?: string[];
1108
+ minQuality?: number;
1109
+ limit?: number;
1110
+ }): KnowledgeEntry[] {
1111
+ const minQuality = opts?.minQuality ?? 0.4;
1112
+ const limit = opts?.limit ?? 50;
1113
+
1114
+ // Find all bases matching the role across all orgs
1115
+ const matchingBases = Array.from(this.bases.values()).filter(b => b.role === role);
1116
+
1117
+ let entries: KnowledgeEntry[] = [];
1118
+ for (const base of matchingBases) {
1119
+ const baseEntries = this.getEntriesForBase(base.id);
1120
+ entries.push(...baseEntries);
1121
+ }
1122
+
1123
+ // Filter to approved/pending entries with sufficient quality
1124
+ entries = entries.filter(e =>
1125
+ (e.status === 'approved' || e.status === 'pending') &&
1126
+ e.qualityScore >= minQuality
1127
+ );
1128
+
1129
+ // Filter by categories if specified
1130
+ if (opts?.categories && opts.categories.length > 0) {
1131
+ entries = entries.filter(e => opts.categories!.includes(e.categoryId));
1132
+ }
1133
+
1134
+ // Sort by quality score descending
1135
+ entries.sort((a, b) => b.qualityScore - a.qualityScore);
1136
+
1137
+ return entries.slice(0, limit);
1138
+ }
1139
+
1140
+ // ─── Stats ──────────────────────────────────────────
1141
+
1142
+ getStats(orgId?: string): {
1143
+ totalBases: number;
1144
+ totalEntries: number;
1145
+ totalContributors: number;
1146
+ topCategories: Array<{ categoryId: string; count: number }>;
1147
+ recentContributions: KnowledgeEntry[];
1148
+ } {
1149
+ // If no orgId, aggregate across all orgs
1150
+ const allBases = orgId
1151
+ ? Array.from(this.bases.values()).filter(b => b.orgId === orgId)
1152
+ : Array.from(this.bases.values());
1153
+ const baseIds = new Set(allBases.map(b => b.id));
1154
+
1155
+ // Collect all entries
1156
+ const allEntries: KnowledgeEntry[] = [];
1157
+ for (const baseId of baseIds) {
1158
+ allEntries.push(...this.getEntriesForBase(baseId));
1159
+ }
1160
+
1161
+ // Count unique contributors
1162
+ const contributors = new Set<string>();
1163
+ for (const baseId of baseIds) {
1164
+ const c = this.baseContributors.get(baseId);
1165
+ if (c) for (const agentId of c) contributors.add(agentId);
1166
+ }
1167
+
1168
+ // Top categories
1169
+ const categoryCounts = new Map<string, number>();
1170
+ for (const entry of allEntries) {
1171
+ categoryCounts.set(entry.categoryId, (categoryCounts.get(entry.categoryId) || 0) + 1);
1172
+ }
1173
+ const topCategories = Array.from(categoryCounts.entries())
1174
+ .map(([categoryId, count]) => ({ categoryId, count }))
1175
+ .sort((a, b) => b.count - a.count)
1176
+ .slice(0, 10);
1177
+
1178
+ // Recent contributions (last 20)
1179
+ const recentContributions = [...allEntries]
1180
+ .sort((a, b) => b.createdAt.localeCompare(a.createdAt))
1181
+ .slice(0, 20);
1182
+
1183
+ return {
1184
+ totalBases: allBases.length,
1185
+ totalEntries: allEntries.length,
1186
+ totalContributors: contributors.size,
1187
+ topCategories,
1188
+ recentContributions,
1189
+ };
1190
+ }
1191
+
1192
+ getAgentContributions(agentId: string): {
1193
+ totalContributed: number;
1194
+ totalUsed: number;
1195
+ avgQuality: number;
1196
+ categories: Array<{ categoryId: string; count: number }>;
1197
+ } {
1198
+ const entries = this.getEntriesForAgent(agentId);
1199
+
1200
+ const totalUsed = entries.reduce((sum, e) => sum + e.useCount, 0);
1201
+ const avgQuality = entries.length > 0
1202
+ ? parseFloat((entries.reduce((sum, e) => sum + e.qualityScore, 0) / entries.length).toFixed(4))
1203
+ : 0;
1204
+
1205
+ const categoryCounts = new Map<string, number>();
1206
+ for (const entry of entries) {
1207
+ categoryCounts.set(entry.categoryId, (categoryCounts.get(entry.categoryId) || 0) + 1);
1208
+ }
1209
+ const categories = Array.from(categoryCounts.entries())
1210
+ .map(([categoryId, count]) => ({ categoryId, count }))
1211
+ .sort((a, b) => b.count - a.count);
1212
+
1213
+ return {
1214
+ totalContributed: entries.length,
1215
+ totalUsed,
1216
+ avgQuality,
1217
+ categories,
1218
+ };
1219
+ }
1220
+
1221
+ /** Alias used by routes. */
1222
+ getAgentStats(agentId: string): {
1223
+ totalContributed: number;
1224
+ totalUsed: number;
1225
+ avgQuality: number;
1226
+ categories: Array<{ categoryId: string; count: number }>;
1227
+ } {
1228
+ return this.getAgentContributions(agentId);
1229
+ }
1230
+
1231
+ // ─── Maintenance ────────────────────────────────────
1232
+
1233
+ decayQuality(): { entriesDecayed: number } {
1234
+ const now = Date.now();
1235
+ let entriesDecayed = 0;
1236
+
1237
+ for (const entry of this.entries.values()) {
1238
+ // Skip recently updated entries (less than 30 days old)
1239
+ const ageMs = now - new Date(entry.updatedAt).getTime();
1240
+ const ageDays = ageMs / 86_400_000;
1241
+ if (ageDays < 30) continue;
1242
+
1243
+ // Skip entries that are actively used
1244
+ const usageRate = entry.useCount / Math.max(1, ageDays / 7); // uses per week
1245
+ if (usageRate >= 1) continue;
1246
+
1247
+ // Decay: reduce quality score by a small amount
1248
+ const decayFactor = 0.95; // 5% decay per maintenance cycle
1249
+ const newQuality = parseFloat((entry.qualityScore * decayFactor).toFixed(4));
1250
+
1251
+ if (newQuality !== entry.qualityScore) {
1252
+ entry.qualityScore = newQuality;
1253
+ entry.updatedAt = new Date().toISOString();
1254
+ entriesDecayed++;
1255
+ }
1256
+ }
1257
+
1258
+ return { entriesDecayed };
1259
+ }
1260
+
1261
+ /** Alias used by routes. */
1262
+ runQualityDecay(_userId?: string): { entriesDecayed: number } {
1263
+ return this.decayQuality();
1264
+ }
1265
+
1266
+ archiveStale(maxAgeDays: number = 365): number {
1267
+ const cutoff = new Date(Date.now() - maxAgeDays * 86_400_000).toISOString();
1268
+ let archived = 0;
1269
+
1270
+ for (const entry of this.entries.values()) {
1271
+ if (entry.status === 'archived' || entry.status === 'rejected') continue;
1272
+
1273
+ // Archive if older than threshold AND low quality AND low usage
1274
+ if (entry.createdAt < cutoff && entry.qualityScore < 0.2 && entry.useCount < 3) {
1275
+ entry.status = 'archived';
1276
+ entry.updatedAt = new Date().toISOString();
1277
+ archived++;
1278
+ }
1279
+ }
1280
+
1281
+ return archived;
1282
+ }
1283
+
1284
+ /** Alias used by routes. */
1285
+ archiveStaleEntries(maxAgeDays: number = 90, _userId?: string): { entriesArchived: number } {
1286
+ const count = this.archiveStale(maxAgeDays);
1287
+ return { entriesArchived: count };
1288
+ }
1289
+
1290
+ // ─── Scheduler ──────────────────────────────────────
1291
+
1292
+ startScheduler(intervalMs: number = 60_000): void {
1293
+ this.stopScheduler();
1294
+
1295
+ this.schedulerTimer = setInterval(() => {
1296
+ this.runDueContributions().then(cycles => {
1297
+ if (cycles.length > 0) {
1298
+ const succeeded = cycles.filter(c => c.status === 'completed').length;
1299
+ const failed = cycles.filter(c => c.status === 'failed').length;
1300
+ console.log(`[knowledge-contribution] Ran ${cycles.length} contribution cycles: ${succeeded} completed, ${failed} failed`);
1301
+ }
1302
+ }).catch(err => {
1303
+ console.error('[knowledge-contribution] Scheduler error:', err.message);
1304
+ });
1305
+ }, intervalMs);
1306
+
1307
+ // Don't block process exit
1308
+ if (this.schedulerTimer && typeof this.schedulerTimer === 'object' && 'unref' in this.schedulerTimer) {
1309
+ this.schedulerTimer.unref();
1310
+ }
1311
+ }
1312
+
1313
+ stopScheduler(): void {
1314
+ if (this.schedulerTimer) {
1315
+ clearInterval(this.schedulerTimer);
1316
+ this.schedulerTimer = undefined;
1317
+ }
1318
+ }
1319
+ }