@a5c-ai/krate 5.0.1-staging.04a3db697

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 (246) hide show
  1. package/Dockerfile +31 -0
  2. package/README.md +183 -0
  3. package/bin/krate-demo.mjs +23 -0
  4. package/bin/krate-server.mjs +14 -0
  5. package/dist/krate-controller-ui.json +3067 -0
  6. package/dist/krate-lifecycle.json +201 -0
  7. package/dist/krate-runtime-snapshot.json +2955 -0
  8. package/dist/krate-summary.json +722 -0
  9. package/docs/README.md +61 -0
  10. package/docs/agents/README.md +83 -0
  11. package/docs/agents/acceptance-test-matrix.md +193 -0
  12. package/docs/agents/agent-mux-adapter-contract.md +167 -0
  13. package/docs/agents/agent-mux-source-map.md +310 -0
  14. package/docs/agents/agent-run-memory-import-spec.md +256 -0
  15. package/docs/agents/agent-stack-management-spec.md +421 -0
  16. package/docs/agents/api-contract-spec.md +309 -0
  17. package/docs/agents/artifacts-writeback-spec.md +145 -0
  18. package/docs/agents/chart-packaging-spec.md +128 -0
  19. package/docs/agents/ci-orchestration-spec.md +140 -0
  20. package/docs/agents/context-assembly-spec.md +219 -0
  21. package/docs/agents/controller-reconciliation-spec.md +255 -0
  22. package/docs/agents/crd-schema-spec.md +315 -0
  23. package/docs/agents/decision-log-open-questions.md +169 -0
  24. package/docs/agents/developer-implementation-checklist.md +329 -0
  25. package/docs/agents/dispatching-design.md +262 -0
  26. package/docs/agents/gaps-agent-mux-to-krate-crds.md +298 -0
  27. package/docs/agents/glossary.md +66 -0
  28. package/docs/agents/implementation-blueprint.md +324 -0
  29. package/docs/agents/implementation-rollout-slices.md +251 -0
  30. package/docs/agents/memory-context-integration-spec.md +194 -0
  31. package/docs/agents/memory-ontology-schema-spec.md +253 -0
  32. package/docs/agents/memory-operations-runbook.md +121 -0
  33. package/docs/agents/mvp-vertical-slice-spec.md +146 -0
  34. package/docs/agents/observability-audit-spec.md +265 -0
  35. package/docs/agents/operator-runbook.md +174 -0
  36. package/docs/agents/org-memory-api-payload-examples.md +333 -0
  37. package/docs/agents/org-memory-controller-sequence-spec.md +181 -0
  38. package/docs/agents/org-memory-e2e-fixture-plan.md +161 -0
  39. package/docs/agents/org-memory-ui-implementation-map.md +114 -0
  40. package/docs/agents/org-memory-vertical-slice-spec.md +168 -0
  41. package/docs/agents/org-resource-model-delta-spec.md +111 -0
  42. package/docs/agents/org-route-resource-model-spec.md +183 -0
  43. package/docs/agents/org-scoping-namespace-spec.md +114 -0
  44. package/docs/agents/rbac-secrets-management-spec.md +406 -0
  45. package/docs/agents/repository-page-integration-spec.md +255 -0
  46. package/docs/agents/resource-contract-examples.md +808 -0
  47. package/docs/agents/resource-relationship-map.md +190 -0
  48. package/docs/agents/security-threat-model.md +188 -0
  49. package/docs/agents/shared-memory-company-brain-spec.md +358 -0
  50. package/docs/agents/storage-migration-spec.md +168 -0
  51. package/docs/agents/subagent-orchestration-spec.md +152 -0
  52. package/docs/agents/system-overview.md +88 -0
  53. package/docs/agents/tools-mcp-skills-spec.md +189 -0
  54. package/docs/agents/traceability-matrix.md +79 -0
  55. package/docs/agents/ui-flow-spec.md +211 -0
  56. package/docs/agents/ui-ux-system-spec.md +426 -0
  57. package/docs/agents/workspace-lifecycle-spec.md +166 -0
  58. package/docs/architecture-spec.md +78 -0
  59. package/docs/components/control-plane.md +78 -0
  60. package/docs/components/data-plane.md +69 -0
  61. package/docs/components/hooks-events.md +67 -0
  62. package/docs/components/identity-rbac-policy.md +73 -0
  63. package/docs/components/kubevela-oam.md +70 -0
  64. package/docs/components/operations-publishing.md +81 -0
  65. package/docs/components/runners-ci.md +66 -0
  66. package/docs/components/web-ui.md +94 -0
  67. package/docs/external/README.md +47 -0
  68. package/docs/external/bidirectional-sync-design.md +134 -0
  69. package/docs/external/cicd-interface.md +64 -0
  70. package/docs/external/external-backend-controllers.md +170 -0
  71. package/docs/external/external-backend-crds.md +234 -0
  72. package/docs/external/external-backend-ui-spec.md +151 -0
  73. package/docs/external/external-backend-ux-flows.md +115 -0
  74. package/docs/external/external-object-mapping.md +125 -0
  75. package/docs/external/git-forge-interface.md +68 -0
  76. package/docs/external/github-integration-design.md +151 -0
  77. package/docs/external/issue-tracking-interface.md +66 -0
  78. package/docs/external/provider-capability-manifests.md +204 -0
  79. package/docs/external/provider-catalog.md +139 -0
  80. package/docs/external/provider-rollout-testing.md +78 -0
  81. package/docs/external/research-results.md +48 -0
  82. package/docs/external/security-auth-permissions.md +81 -0
  83. package/docs/external/sync-state-machines.md +108 -0
  84. package/docs/external/unified-external-backend-model.md +107 -0
  85. package/docs/external/user-facing-changes.md +67 -0
  86. package/docs/gaps.md +161 -0
  87. package/docs/install.md +94 -0
  88. package/docs/krate-design.md +334 -0
  89. package/docs/local-minikube.md +55 -0
  90. package/docs/ontology/README.md +32 -0
  91. package/docs/ontology/bounded-contexts.md +29 -0
  92. package/docs/ontology/events-and-hooks.md +32 -0
  93. package/docs/ontology/oam-kubevela.md +32 -0
  94. package/docs/ontology/operations-and-release.md +25 -0
  95. package/docs/ontology/personas-and-actors.md +32 -0
  96. package/docs/ontology/policies-and-invariants.md +33 -0
  97. package/docs/ontology/problem-space.md +30 -0
  98. package/docs/ontology/resource-contracts.md +40 -0
  99. package/docs/ontology/resource-taxonomy.md +42 -0
  100. package/docs/ontology/runners-and-ci.md +29 -0
  101. package/docs/ontology/solution-space.md +24 -0
  102. package/docs/ontology/storage-and-data-boundaries.md +29 -0
  103. package/docs/ontology/validation-matrix.md +24 -0
  104. package/docs/ontology/web-ui-excellent-flows.md +32 -0
  105. package/docs/ontology/workflows.md +39 -0
  106. package/docs/ontology/world.md +35 -0
  107. package/docs/product-requirements.md +62 -0
  108. package/docs/roadmap-mvp.md +87 -0
  109. package/docs/system-requirements.md +90 -0
  110. package/docs/tests/README.md +53 -0
  111. package/docs/tests/agent-qa-plan.md +63 -0
  112. package/docs/tests/browser-ui-tests.md +62 -0
  113. package/docs/tests/ci-quality-gates.md +48 -0
  114. package/docs/tests/coverage-model.md +64 -0
  115. package/docs/tests/e2e-scenario-tests.md +53 -0
  116. package/docs/tests/fixtures-test-data.md +63 -0
  117. package/docs/tests/observability-reliability-tests.md +54 -0
  118. package/docs/tests/product-test-matrix.md +145 -0
  119. package/docs/tests/qa-adoption-roadmap.md +130 -0
  120. package/docs/tests/qa-automation-plan.md +101 -0
  121. package/docs/tests/security-compliance-tests.md +57 -0
  122. package/docs/tests/test-framework-tools.md +88 -0
  123. package/docs/tests/test-suite-layout.md +121 -0
  124. package/docs/tests/unit-integration-tests.md +48 -0
  125. package/docs/todo-kyverno +714 -0
  126. package/docs/todos.md +4 -0
  127. package/docs/user-stories.md +78 -0
  128. package/examples/minikube-demo.yaml +190 -0
  129. package/examples/oam-application.yaml +23 -0
  130. package/examples/policy-kyverno-pr-title.yaml +18 -0
  131. package/package.json +63 -0
  132. package/scripts/build.mjs +29 -0
  133. package/scripts/setup-minikube.mjs +65 -0
  134. package/scripts/smoke.mjs +37 -0
  135. package/scripts/validate-doc-coverage.mjs +152 -0
  136. package/scripts/validate-package.mjs +93 -0
  137. package/scripts/validate-ui.mjs +236 -0
  138. package/src/agent-adapter-controller.js +169 -0
  139. package/src/agent-approval-controller.js +170 -0
  140. package/src/agent-context-bundles.js +242 -0
  141. package/src/agent-dispatch-controller.js +209 -0
  142. package/src/agent-gateway-config-controller.js +147 -0
  143. package/src/agent-memory-controller.js +357 -0
  144. package/src/agent-memory-import.js +327 -0
  145. package/src/agent-memory-query.js +292 -0
  146. package/src/agent-memory-repository-source-controller.js +255 -0
  147. package/src/agent-mux-client.js +280 -0
  148. package/src/agent-permission-review.js +250 -0
  149. package/src/agent-project-controller.js +117 -0
  150. package/src/agent-provider-config-controller.js +150 -0
  151. package/src/agent-secret-config-grant-controller.js +282 -0
  152. package/src/agent-session-transcript-controller.js +189 -0
  153. package/src/agent-stack-controller.js +347 -0
  154. package/src/agent-subagent-controller.js +160 -0
  155. package/src/agent-transport-binding-controller.js +121 -0
  156. package/src/agent-trigger-controller.js +321 -0
  157. package/src/agent-workspace-controller.js +447 -0
  158. package/src/agent-writeback-controller.js +302 -0
  159. package/src/api-controller.js +541 -0
  160. package/src/argocd-gitops.js +43 -0
  161. package/src/async-controller.js +207 -0
  162. package/src/audit-controller.js +191 -0
  163. package/src/auth.js +307 -0
  164. package/src/component-catalog.js +41 -0
  165. package/src/control-plane.js +136 -0
  166. package/src/controller-client.js +50 -0
  167. package/src/controller-ui.js +551 -0
  168. package/src/data-plane.js +178 -0
  169. package/src/event-bus.js +61 -0
  170. package/src/external/conflict-controller.js +225 -0
  171. package/src/external/github/auth.js +96 -0
  172. package/src/external/github/cicd.js +180 -0
  173. package/src/external/github/git-forge.js +240 -0
  174. package/src/external/github/index.js +144 -0
  175. package/src/external/github/issue-tracking.js +163 -0
  176. package/src/external/provider-adapter.js +161 -0
  177. package/src/external/provider-resource-factory.js +161 -0
  178. package/src/external/sync-controller.js +235 -0
  179. package/src/external/webhook-controller.js +144 -0
  180. package/src/external/write-controller.js +283 -0
  181. package/src/gitea-backend.js +95 -0
  182. package/src/gitea-service.js +173 -0
  183. package/src/handoff.js +98 -0
  184. package/src/hooks-events.js +63 -0
  185. package/src/http-server.js +377 -0
  186. package/src/identity-policy.js +86 -0
  187. package/src/index.js +55 -0
  188. package/src/kubernetes-controller-async.js +511 -0
  189. package/src/kubernetes-controller.js +878 -0
  190. package/src/kubernetes-resource-gateway.js +48 -0
  191. package/src/operations.js +112 -0
  192. package/src/org-scoping.js +5 -0
  193. package/src/resource-model.js +221 -0
  194. package/src/runners-ci.js +48 -0
  195. package/src/runtime.js +196 -0
  196. package/src/snapshot-cache.js +157 -0
  197. package/src/web-ui.js +40 -0
  198. package/tests/agent-adapter-controller.test.js +361 -0
  199. package/tests/agent-approval-controller.test.js +173 -0
  200. package/tests/agent-context-bundles.test.js +278 -0
  201. package/tests/agent-dispatch-controller.test.js +315 -0
  202. package/tests/agent-gateway-config-controller.test.js +386 -0
  203. package/tests/agent-memory-controller.test.js +308 -0
  204. package/tests/agent-memory-import-snapshot.test.js +477 -0
  205. package/tests/agent-memory-query.test.js +404 -0
  206. package/tests/agent-memory-repository-source.test.js +514 -0
  207. package/tests/agent-mux-client.test.js +204 -0
  208. package/tests/agent-permission-review-v2.test.js +317 -0
  209. package/tests/agent-permission-review.test.js +209 -0
  210. package/tests/agent-project-controller.test.js +302 -0
  211. package/tests/agent-provider-config-controller.test.js +376 -0
  212. package/tests/agent-resources.test.js +228 -0
  213. package/tests/agent-secret-config-grant.test.js +231 -0
  214. package/tests/agent-session-transcript-controller.test.js +499 -0
  215. package/tests/agent-stack-controller.test.js +221 -0
  216. package/tests/agent-subagent-controller.test.js +201 -0
  217. package/tests/agent-transport-binding-controller.test.js +294 -0
  218. package/tests/agent-trigger-controller.test.js +211 -0
  219. package/tests/agent-trigger-routes.test.js +190 -0
  220. package/tests/agent-trigger-sources.test.js +245 -0
  221. package/tests/agent-workspace-controller.test.js +181 -0
  222. package/tests/agent-writeback.test.js +292 -0
  223. package/tests/approval-persistence.test.js +171 -0
  224. package/tests/async-controller.test.js +252 -0
  225. package/tests/audit-controller.test.js +227 -0
  226. package/tests/deployment.test.js +396 -0
  227. package/tests/e2e/lifecycle.test.js +117 -0
  228. package/tests/external-github-forge.test.js +560 -0
  229. package/tests/external-github-issues-cicd.test.js +520 -0
  230. package/tests/external-integration.test.js +470 -0
  231. package/tests/external-persistence.test.js +340 -0
  232. package/tests/external-provider-adapter.test.js +365 -0
  233. package/tests/external-resource-model.test.js +215 -0
  234. package/tests/external-webhook-sync.test.js +287 -0
  235. package/tests/external-write-conflict.test.js +353 -0
  236. package/tests/gitea-service.test.js +253 -0
  237. package/tests/health-check-real.test.js +165 -0
  238. package/tests/integration/full-flow.test.js +266 -0
  239. package/tests/krate.test.js +727 -0
  240. package/tests/memory-search-wiring.test.js +270 -0
  241. package/tests/org-scoping.test.js +687 -0
  242. package/tests/session-cookie-hmac.test.js +151 -0
  243. package/tests/snapshot-performance.test.js +247 -0
  244. package/tests/sse-events.test.js +107 -0
  245. package/tests/workspace-volumes.test.js +312 -0
  246. package/tests/writeback-persistence.test.js +207 -0
@@ -0,0 +1,209 @@
1
+ import { describe, it } from 'node:test';
2
+ import assert from 'node:assert/strict';
3
+ import { createPermissionReviewer } from '../src/agent-permission-review.js';
4
+ import { createResource } from '../src/resource-model.js';
5
+
6
+ function makeStack(name, overrides = {}) {
7
+ return createResource('AgentStack', { name }, {
8
+ organizationRef: 'default',
9
+ baseAgent: 'claude-code',
10
+ adapter: 'babysitter',
11
+ runtimeIdentity: { serviceAccountRef: overrides.serviceAccountRef || 'sa-agent' },
12
+ ...(overrides.spec || {})
13
+ });
14
+ }
15
+
16
+ function makeServiceAccount(name) {
17
+ return createResource('AgentServiceAccount', { name }, {
18
+ organizationRef: 'default',
19
+ namespace: 'krate-agents',
20
+ serviceAccountName: name
21
+ });
22
+ }
23
+
24
+ function makeRoleBinding(name, subject) {
25
+ return createResource('AgentRoleBinding', { name }, {
26
+ organizationRef: 'default',
27
+ subject,
28
+ roleRef: 'agent-role',
29
+ scope: 'namespace'
30
+ });
31
+ }
32
+
33
+ function makeSecretGrant(name, subject, purpose, overrides = {}) {
34
+ return createResource('AgentSecretGrant', { name }, {
35
+ organizationRef: 'default',
36
+ subject,
37
+ secretRef: 'api-keys',
38
+ purpose,
39
+ ...overrides
40
+ });
41
+ }
42
+
43
+ function makeMcpServer(name, overrides = {}) {
44
+ return createResource('AgentMcpServer', { name }, {
45
+ organizationRef: 'default',
46
+ transport: 'stdio',
47
+ scope: 'workspace',
48
+ ...overrides
49
+ });
50
+ }
51
+
52
+ function makeModelProviderGrant(subject) {
53
+ return makeSecretGrant('sg-model', subject, 'model-provider');
54
+ }
55
+
56
+ const baseInput = {
57
+ repository: 'my-repo',
58
+ ref: 'refs/heads/main',
59
+ actor: 'user-1',
60
+ agentStack: 'test-stack',
61
+ triggerSource: 'manual',
62
+ taskKind: 'fix'
63
+ };
64
+
65
+ describe('agent permission review', () => {
66
+ it('fully granted stack returns allowed', () => {
67
+ const reviewer = createPermissionReviewer();
68
+ const resources = {
69
+ AgentStack: [makeStack('test-stack')],
70
+ AgentServiceAccount: [makeServiceAccount('sa-agent')],
71
+ AgentRoleBinding: [makeRoleBinding('rb-1', 'sa-agent')],
72
+ AgentSecretGrant: [makeModelProviderGrant('sa-agent')],
73
+ AgentConfigGrant: [],
74
+ AgentMcpServer: []
75
+ };
76
+ const result = reviewer.reviewPermissions({ ...baseInput, resources });
77
+ assert.equal(result.decision, 'allowed');
78
+ assert.ok(result.grants.length > 0, 'should have at least one grant');
79
+ assert.ok(result.grants.some((g) => g.kind === 'AgentServiceAccount' && g.status === 'bound'));
80
+ assert.ok(result.grants.some((g) => g.kind === 'AgentRoleBinding' && g.status === 'bound'));
81
+ assert.ok(typeof result.digest === 'string' && result.digest.length > 0);
82
+ });
83
+
84
+ it('missing service account returns denied with missing-runtime-identity reason', () => {
85
+ const reviewer = createPermissionReviewer();
86
+ const resources = {
87
+ AgentStack: [makeStack('test-stack', { serviceAccountRef: 'nonexistent-sa' })],
88
+ AgentServiceAccount: [],
89
+ AgentRoleBinding: [],
90
+ AgentSecretGrant: [],
91
+ AgentConfigGrant: [],
92
+ AgentMcpServer: []
93
+ };
94
+ const result = reviewer.reviewPermissions({ ...baseInput, resources });
95
+ assert.equal(result.decision, 'denied');
96
+ assert.ok(result.reasons.some((r) => r.severity === 'error' && r.message.includes('Missing AgentServiceAccount')));
97
+ });
98
+
99
+ it('missing secret grant returns denied with missingGrants information', () => {
100
+ const reviewer = createPermissionReviewer();
101
+ const mcpServer = makeMcpServer('mcp-github', { secretRef: 'github-token' });
102
+ const stack = makeStack('test-stack', {
103
+ spec: {
104
+ organizationRef: 'default',
105
+ baseAgent: 'claude-code',
106
+ adapter: 'babysitter',
107
+ runtimeIdentity: { serviceAccountRef: 'sa-agent' },
108
+ mcpServerRefs: ['mcp-github']
109
+ }
110
+ });
111
+ const resources = {
112
+ AgentStack: [stack],
113
+ AgentServiceAccount: [makeServiceAccount('sa-agent')],
114
+ AgentRoleBinding: [makeRoleBinding('rb-1', 'sa-agent')],
115
+ AgentSecretGrant: [],
116
+ AgentConfigGrant: [],
117
+ AgentMcpServer: [mcpServer]
118
+ };
119
+ const result = reviewer.reviewPermissions({ ...baseInput, resources });
120
+ assert.equal(result.decision, 'denied');
121
+ assert.ok(result.reasons.some((r) => r.severity === 'error' && r.message.includes('Missing AgentSecretGrant')));
122
+ });
123
+
124
+ it('grant requiring approval returns requires-approval', () => {
125
+ const reviewer = createPermissionReviewer();
126
+ const mcpServer = makeMcpServer('mcp-prod', { secretRef: 'prod-secret' });
127
+ const stack = makeStack('test-stack', {
128
+ spec: {
129
+ organizationRef: 'default',
130
+ baseAgent: 'claude-code',
131
+ adapter: 'babysitter',
132
+ runtimeIdentity: { serviceAccountRef: 'sa-agent' },
133
+ mcpServerRefs: ['mcp-prod']
134
+ }
135
+ });
136
+ const mcpGrant = makeSecretGrant('sg-prod', 'sa-agent', 'mcp-server:mcp-prod', {
137
+ requiredApproval: 'always'
138
+ });
139
+ const resources = {
140
+ AgentStack: [stack],
141
+ AgentServiceAccount: [makeServiceAccount('sa-agent')],
142
+ AgentRoleBinding: [makeRoleBinding('rb-1', 'sa-agent')],
143
+ AgentSecretGrant: [mcpGrant, makeModelProviderGrant('sa-agent')],
144
+ AgentConfigGrant: [],
145
+ AgentMcpServer: [mcpServer]
146
+ };
147
+ const result = reviewer.reviewPermissions({ ...baseInput, resources });
148
+ assert.equal(result.decision, 'requires-approval');
149
+ assert.ok(result.grants.some((g) => g.status === 'requires-approval' && g.requiredApproval === 'always'));
150
+ });
151
+
152
+ it('empty capabilities stack returns allowed', () => {
153
+ const reviewer = createPermissionReviewer();
154
+ const stack = makeStack('test-stack');
155
+ const resources = {
156
+ AgentStack: [stack],
157
+ AgentServiceAccount: [makeServiceAccount('sa-agent')],
158
+ AgentRoleBinding: [makeRoleBinding('rb-1', 'sa-agent')],
159
+ AgentSecretGrant: [makeModelProviderGrant('sa-agent')],
160
+ AgentConfigGrant: [],
161
+ AgentMcpServer: []
162
+ };
163
+ const result = reviewer.reviewPermissions({
164
+ ...baseInput,
165
+ toolRefs: [],
166
+ skillRefs: [],
167
+ mcpServerRefs: [],
168
+ contextLabelRefs: [],
169
+ resources
170
+ });
171
+ assert.equal(result.decision, 'allowed');
172
+ });
173
+
174
+ it('same input produces deterministic digest', () => {
175
+ const reviewer = createPermissionReviewer();
176
+ const resources = {
177
+ AgentStack: [makeStack('test-stack')],
178
+ AgentServiceAccount: [makeServiceAccount('sa-agent')],
179
+ AgentRoleBinding: [makeRoleBinding('rb-1', 'sa-agent')],
180
+ AgentSecretGrant: [makeModelProviderGrant('sa-agent')],
181
+ AgentConfigGrant: [],
182
+ AgentMcpServer: []
183
+ };
184
+ const result1 = reviewer.reviewPermissions({ ...baseInput, resources });
185
+ const result2 = reviewer.reviewPermissions({ ...baseInput, resources });
186
+ assert.equal(result1.digest, result2.digest);
187
+ assert.ok(typeof result1.digest === 'string' && result1.digest.length === 64, 'digest should be a 64-char hex SHA-256');
188
+ });
189
+
190
+ it('createPermissionSnapshot returns frozen object with timestamp', () => {
191
+ const reviewer = createPermissionReviewer();
192
+ const resources = {
193
+ AgentStack: [makeStack('test-stack')],
194
+ AgentServiceAccount: [makeServiceAccount('sa-agent')],
195
+ AgentRoleBinding: [makeRoleBinding('rb-1', 'sa-agent')],
196
+ AgentSecretGrant: [makeModelProviderGrant('sa-agent')],
197
+ AgentConfigGrant: [],
198
+ AgentMcpServer: []
199
+ };
200
+ const review = reviewer.reviewPermissions({ ...baseInput, resources });
201
+ const snapshot = reviewer.createPermissionSnapshot(review);
202
+ assert.ok(Object.isFrozen(snapshot), 'snapshot should be frozen');
203
+ assert.ok(typeof snapshot.snapshotAt === 'string' && snapshot.snapshotAt.length > 0, 'snapshot should have a timestamp');
204
+ assert.equal(snapshot.frozen, true);
205
+ assert.equal(snapshot.digest, review.digest);
206
+ assert.equal(snapshot.decision, review.decision);
207
+ assert.throws(() => { snapshot.decision = 'denied'; }, TypeError);
208
+ });
209
+ });
@@ -0,0 +1,302 @@
1
+ import assert from 'node:assert/strict';
2
+ import test from 'node:test';
3
+ import { createAgentProjectController, validateAgentProject, createResource, AGENT_PROJECT_CONTROLLER_BOUNDARY } from '../src/index.js';
4
+
5
+ // ---------------------------------------------------------------------------
6
+ // Acceptance criteria: Slice 1.2d — Agent Project Controller
7
+ //
8
+ // A KrateProject groups issues and tasks into a kanban board with workflow
9
+ // columns, default column assignment, issue tracking, and board state management.
10
+ //
11
+ // All tests in this file are expected to FAIL until the controller is
12
+ // implemented and exported from src/index.js.
13
+ // ---------------------------------------------------------------------------
14
+
15
+ function makeProject(name, overrides = {}) {
16
+ return createResource('KrateProject', { name, namespace: 'krate-org-default' }, {
17
+ organizationRef: 'default',
18
+ workflowColumns: [
19
+ { id: 'todo', displayName: 'To Do', color: '#888888', default: true },
20
+ { id: 'in-progress', displayName: 'In Progress', color: '#0075ca' },
21
+ { id: 'review', displayName: 'Review', color: '#e4e669' },
22
+ { id: 'done', displayName: 'Done', color: '#0e8a16' }
23
+ ],
24
+ ...overrides
25
+ });
26
+ }
27
+
28
+ // ---------------------------------------------------------------------------
29
+ // 1. Factory and shape
30
+ // ---------------------------------------------------------------------------
31
+
32
+ test('createAgentProjectController returns a controller with validate, getWorkflowColumns, getBoardState', () => {
33
+ const controller = createAgentProjectController();
34
+ assert.ok(controller, 'controller must be truthy');
35
+ assert.equal(typeof controller.validate, 'function', 'controller must expose a validate method');
36
+ assert.equal(typeof controller.getWorkflowColumns, 'function', 'controller must expose a getWorkflowColumns method');
37
+ assert.equal(typeof controller.getBoardState, 'function', 'controller must expose a getBoardState method');
38
+ assert.equal(controller.role, 'agent-project-controller', 'controller must declare its role');
39
+ });
40
+
41
+ // ---------------------------------------------------------------------------
42
+ // 2. validate — happy path
43
+ // ---------------------------------------------------------------------------
44
+
45
+ test('validate accepts valid project with name, organizationRef, workflowColumns', () => {
46
+ const controller = createAgentProjectController();
47
+ const project = makeProject('my-sprint');
48
+ const result = controller.validate(project);
49
+
50
+ assert.equal(result.valid, true, 'valid project must pass validation');
51
+ assert.ok(Array.isArray(result.errors), 'result must contain an errors array');
52
+ assert.equal(result.errors.length, 0, 'errors array must be empty for a valid project');
53
+ });
54
+
55
+ // ---------------------------------------------------------------------------
56
+ // 3. validate — missing name
57
+ // ---------------------------------------------------------------------------
58
+
59
+ test('validate rejects project with missing name', () => {
60
+ const controller = createAgentProjectController();
61
+ const project = {
62
+ apiVersion: 'krate.a5c.ai/v1alpha1',
63
+ kind: 'KrateProject',
64
+ metadata: { namespace: 'krate-org-default', labels: {}, annotations: {} },
65
+ spec: {
66
+ organizationRef: 'default',
67
+ workflowColumns: [{ id: 'todo', displayName: 'To Do', color: '#888888' }]
68
+ },
69
+ status: {}
70
+ };
71
+ const result = controller.validate(project);
72
+
73
+ assert.equal(result.valid, false, 'project without name must fail validation');
74
+ assert.ok(result.errors.length > 0, 'errors array must not be empty');
75
+ assert.ok(
76
+ result.errors.some((e) => /name/i.test(e)),
77
+ 'at least one error must mention "name"'
78
+ );
79
+ });
80
+
81
+ // ---------------------------------------------------------------------------
82
+ // 4. validate — missing organizationRef
83
+ // ---------------------------------------------------------------------------
84
+
85
+ test('validate rejects project with missing organizationRef', () => {
86
+ const controller = createAgentProjectController();
87
+ const project = makeProject('no-org-project', { organizationRef: undefined });
88
+ delete project.spec.organizationRef;
89
+ const result = controller.validate(project);
90
+
91
+ assert.equal(result.valid, false, 'project without organizationRef must fail validation');
92
+ assert.ok(result.errors.length > 0, 'errors array must not be empty');
93
+ assert.ok(
94
+ result.errors.some((e) => /organizationRef/i.test(e)),
95
+ 'at least one error must mention "organizationRef"'
96
+ );
97
+ });
98
+
99
+ // ---------------------------------------------------------------------------
100
+ // 5. validate — empty workflowColumns array
101
+ // ---------------------------------------------------------------------------
102
+
103
+ test('validate rejects project with empty workflowColumns array', () => {
104
+ const controller = createAgentProjectController();
105
+ const project = makeProject('no-columns-project', { workflowColumns: [] });
106
+ const result = controller.validate(project);
107
+
108
+ assert.equal(result.valid, false, 'project with empty workflowColumns must fail validation');
109
+ assert.ok(result.errors.length > 0, 'errors array must not be empty');
110
+ assert.ok(
111
+ result.errors.some((e) => /workflowColumns/i.test(e)),
112
+ 'at least one error must mention "workflowColumns"'
113
+ );
114
+ });
115
+
116
+ // ---------------------------------------------------------------------------
117
+ // 6. validate — duplicate column IDs
118
+ // ---------------------------------------------------------------------------
119
+
120
+ test('validate rejects project with duplicate column IDs', () => {
121
+ const controller = createAgentProjectController();
122
+ const project = makeProject('dup-columns-project', {
123
+ workflowColumns: [
124
+ { id: 'todo', displayName: 'To Do', color: '#888888' },
125
+ { id: 'todo', displayName: 'Also To Do', color: '#999999' },
126
+ { id: 'done', displayName: 'Done', color: '#0e8a16' }
127
+ ]
128
+ });
129
+ const result = controller.validate(project);
130
+
131
+ assert.equal(result.valid, false, 'project with duplicate column IDs must fail validation');
132
+ assert.ok(result.errors.length > 0, 'errors array must not be empty');
133
+ assert.ok(
134
+ result.errors.some((e) => /duplicate|column/i.test(e)),
135
+ 'at least one error must mention duplicate column IDs'
136
+ );
137
+ });
138
+
139
+ // ---------------------------------------------------------------------------
140
+ // 7. getWorkflowColumns — returns columns in order
141
+ // ---------------------------------------------------------------------------
142
+
143
+ test('getWorkflowColumns returns columns from spec with order preserved', () => {
144
+ const controller = createAgentProjectController();
145
+ const columns = [
146
+ { id: 'backlog', displayName: 'Backlog', color: '#cccccc' },
147
+ { id: 'todo', displayName: 'To Do', color: '#888888', default: true },
148
+ { id: 'doing', displayName: 'Doing', color: '#0075ca' },
149
+ { id: 'done', displayName: 'Done', color: '#0e8a16' }
150
+ ];
151
+ const project = makeProject('ordered-project', { workflowColumns: columns });
152
+ const result = controller.getWorkflowColumns(project);
153
+
154
+ assert.ok(Array.isArray(result), 'getWorkflowColumns must return an array');
155
+ assert.equal(result.length, 4, 'must return all four columns');
156
+ assert.equal(result[0].id, 'backlog', 'first column must be backlog');
157
+ assert.equal(result[1].id, 'todo', 'second column must be todo');
158
+ assert.equal(result[2].id, 'doing', 'third column must be doing');
159
+ assert.equal(result[3].id, 'done', 'fourth column must be done');
160
+ });
161
+
162
+ // ---------------------------------------------------------------------------
163
+ // 8. getDefaultColumn — returns marked default or first column
164
+ // ---------------------------------------------------------------------------
165
+
166
+ test('getDefaultColumn returns the column marked as default', () => {
167
+ const controller = createAgentProjectController();
168
+ const project = makeProject('default-col-project', {
169
+ workflowColumns: [
170
+ { id: 'todo', displayName: 'To Do', color: '#888888' },
171
+ { id: 'in-progress', displayName: 'In Progress', color: '#0075ca', default: true },
172
+ { id: 'done', displayName: 'Done', color: '#0e8a16' }
173
+ ]
174
+ });
175
+ const col = controller.getDefaultColumn(project);
176
+
177
+ assert.ok(col, 'getDefaultColumn must return a column');
178
+ assert.equal(col.id, 'in-progress', 'must return the column marked as default');
179
+ });
180
+
181
+ test('getDefaultColumn returns first column when none is marked as default', () => {
182
+ const controller = createAgentProjectController();
183
+ const project = makeProject('no-default-col-project', {
184
+ workflowColumns: [
185
+ { id: 'backlog', displayName: 'Backlog', color: '#cccccc' },
186
+ { id: 'todo', displayName: 'To Do', color: '#888888' },
187
+ { id: 'done', displayName: 'Done', color: '#0e8a16' }
188
+ ]
189
+ });
190
+ const col = controller.getDefaultColumn(project);
191
+
192
+ assert.ok(col, 'getDefaultColumn must return a column');
193
+ assert.equal(col.id, 'backlog', 'must return first column when no default is marked');
194
+ });
195
+
196
+ // ---------------------------------------------------------------------------
197
+ // 9. getBoardState — default is 'active'
198
+ // ---------------------------------------------------------------------------
199
+
200
+ test('getBoardState returns "active" by default when boardState is not set', () => {
201
+ const controller = createAgentProjectController();
202
+ const project = makeProject('active-project');
203
+ const state = controller.getBoardState(project);
204
+
205
+ assert.equal(state, 'active', 'getBoardState must return "active" by default');
206
+ });
207
+
208
+ // ---------------------------------------------------------------------------
209
+ // 10. getBoardState — returns spec.boardState when set
210
+ // ---------------------------------------------------------------------------
211
+
212
+ test('getBoardState returns spec.boardState when explicitly set', () => {
213
+ const controller = createAgentProjectController();
214
+ const project = makeProject('archived-project', { boardState: 'archived' });
215
+ const state = controller.getBoardState(project);
216
+
217
+ assert.equal(state, 'archived', 'getBoardState must return the spec boardState value');
218
+ });
219
+
220
+ // ---------------------------------------------------------------------------
221
+ // 11. validate — null resource
222
+ // ---------------------------------------------------------------------------
223
+
224
+ test('validate rejects null resource with a clear error', () => {
225
+ const controller = createAgentProjectController();
226
+ const result = controller.validate(null);
227
+
228
+ assert.equal(result.valid, false, 'null resource must fail validation');
229
+ assert.ok(result.errors.length > 0, 'errors array must not be empty');
230
+ assert.ok(
231
+ result.errors.some((e) => /null|undefined/i.test(e)),
232
+ 'error must mention null or undefined'
233
+ );
234
+ });
235
+
236
+ // ---------------------------------------------------------------------------
237
+ // 12. BOUNDARY exported with correct role
238
+ // ---------------------------------------------------------------------------
239
+
240
+ test('AGENT_PROJECT_CONTROLLER_BOUNDARY is exported and has correct role', () => {
241
+ assert.ok(AGENT_PROJECT_CONTROLLER_BOUNDARY, 'BOUNDARY must be exported');
242
+ assert.equal(
243
+ AGENT_PROJECT_CONTROLLER_BOUNDARY.role,
244
+ 'agent-project-controller',
245
+ 'BOUNDARY role must be "agent-project-controller"'
246
+ );
247
+ assert.ok(
248
+ Array.isArray(AGENT_PROJECT_CONTROLLER_BOUNDARY.owns),
249
+ 'BOUNDARY must declare owned concerns'
250
+ );
251
+ });
252
+
253
+ // ---------------------------------------------------------------------------
254
+ // 13. validateAgentProject — standalone export
255
+ // ---------------------------------------------------------------------------
256
+
257
+ test('validateAgentProject exported as standalone function follows existing pattern', () => {
258
+ assert.equal(typeof validateAgentProject, 'function', 'validateAgentProject must be a named export');
259
+
260
+ const project = makeProject('standalone-validate-project');
261
+ const result = validateAgentProject(project);
262
+
263
+ assert.ok(result, 'validateAgentProject must return a result');
264
+ assert.ok('valid' in result, 'result must have a valid property');
265
+ assert.ok(Array.isArray(result.errors), 'result must have an errors array');
266
+ assert.equal(result.valid, true, 'a fully-specified project must pass standalone validation');
267
+ });
268
+
269
+ // ---------------------------------------------------------------------------
270
+ // 14. validate — missing workflowColumns field entirely
271
+ // ---------------------------------------------------------------------------
272
+
273
+ test('validate rejects project with no workflowColumns field', () => {
274
+ const controller = createAgentProjectController();
275
+ const project = makeProject('missing-cols-project');
276
+ delete project.spec.workflowColumns;
277
+ const result = controller.validate(project);
278
+
279
+ assert.equal(result.valid, false, 'project missing workflowColumns must fail validation');
280
+ assert.ok(
281
+ result.errors.some((e) => /workflowColumns/i.test(e)),
282
+ 'at least one error must mention "workflowColumns"'
283
+ );
284
+ });
285
+
286
+ // ---------------------------------------------------------------------------
287
+ // 15. getWorkflowColumns — returns empty array when no spec
288
+ // ---------------------------------------------------------------------------
289
+
290
+ test('getWorkflowColumns handles resource with no spec gracefully', () => {
291
+ const controller = createAgentProjectController();
292
+ const resource = {
293
+ apiVersion: 'krate.a5c.ai/v1alpha1',
294
+ kind: 'KrateProject',
295
+ metadata: { name: 'spec-less-project', namespace: 'krate-org-default', labels: {}, annotations: {} },
296
+ status: {}
297
+ };
298
+ const result = controller.getWorkflowColumns(resource);
299
+
300
+ assert.ok(Array.isArray(result), 'getWorkflowColumns must return an array');
301
+ assert.equal(result.length, 0, 'must return empty array when spec is absent');
302
+ });