@ucptools/validator 1.0.0 → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (236) hide show
  1. package/.claude/settings.local.json +60 -0
  2. package/.vercel/README.txt +11 -0
  3. package/.vercel/project.json +1 -0
  4. package/dist/cli/index.d.ts +6 -0
  5. package/dist/cli/index.d.ts.map +1 -0
  6. package/dist/cli/index.js +279 -0
  7. package/dist/cli/index.js.map +1 -0
  8. package/dist/compliance/compliance-generator.d.ts +34 -0
  9. package/dist/compliance/compliance-generator.d.ts.map +1 -0
  10. package/dist/compliance/compliance-generator.js +320 -0
  11. package/dist/compliance/compliance-generator.js.map +1 -0
  12. package/dist/compliance/index.d.ts +8 -0
  13. package/dist/compliance/index.d.ts.map +1 -0
  14. package/dist/compliance/index.js +17 -0
  15. package/dist/compliance/index.js.map +1 -0
  16. package/dist/compliance/templates.d.ts +34 -0
  17. package/dist/compliance/templates.d.ts.map +1 -0
  18. package/{src/compliance/templates.ts → dist/compliance/templates.js} +117 -155
  19. package/dist/compliance/templates.js.map +1 -0
  20. package/dist/compliance/types.d.ts +64 -0
  21. package/dist/compliance/types.d.ts.map +1 -0
  22. package/dist/compliance/types.js +64 -0
  23. package/dist/compliance/types.js.map +1 -0
  24. package/dist/db/index.d.ts +11 -0
  25. package/dist/db/index.d.ts.map +1 -0
  26. package/dist/db/index.js +63 -0
  27. package/dist/db/index.js.map +1 -0
  28. package/dist/db/schema.d.ts +444 -0
  29. package/dist/db/schema.d.ts.map +1 -0
  30. package/dist/db/schema.js +65 -0
  31. package/dist/db/schema.js.map +1 -0
  32. package/dist/feed-analyzer/feed-analyzer.d.ts +26 -0
  33. package/dist/feed-analyzer/feed-analyzer.d.ts.map +1 -0
  34. package/{src/feed-analyzer/feed-analyzer.ts → dist/feed-analyzer/feed-analyzer.js} +642 -726
  35. package/dist/feed-analyzer/feed-analyzer.js.map +1 -0
  36. package/dist/feed-analyzer/index.d.ts +8 -0
  37. package/dist/feed-analyzer/index.d.ts.map +1 -0
  38. package/dist/feed-analyzer/index.js +19 -0
  39. package/dist/feed-analyzer/index.js.map +1 -0
  40. package/dist/feed-analyzer/types.d.ts +204 -0
  41. package/dist/feed-analyzer/types.d.ts.map +1 -0
  42. package/dist/feed-analyzer/types.js +162 -0
  43. package/dist/feed-analyzer/types.js.map +1 -0
  44. package/{src/generator/index.ts → dist/generator/index.d.ts} +1 -1
  45. package/dist/generator/index.d.ts.map +1 -0
  46. package/dist/generator/index.js +13 -0
  47. package/dist/generator/index.js.map +1 -0
  48. package/dist/generator/key-generator.d.ts +24 -0
  49. package/dist/generator/key-generator.d.ts.map +1 -0
  50. package/dist/generator/key-generator.js +144 -0
  51. package/dist/generator/key-generator.js.map +1 -0
  52. package/dist/generator/profile-builder.d.ts +15 -0
  53. package/dist/generator/profile-builder.d.ts.map +1 -0
  54. package/dist/generator/profile-builder.js +338 -0
  55. package/dist/generator/profile-builder.js.map +1 -0
  56. package/dist/hosting/artifacts-generator.d.ts +10 -0
  57. package/dist/hosting/artifacts-generator.d.ts.map +1 -0
  58. package/{src/hosting/artifacts-generator.ts → dist/hosting/artifacts-generator.js} +191 -241
  59. package/dist/hosting/artifacts-generator.js.map +1 -0
  60. package/{src/hosting/index.ts → dist/hosting/index.d.ts} +1 -1
  61. package/dist/hosting/index.d.ts.map +1 -0
  62. package/dist/hosting/index.js +10 -0
  63. package/dist/hosting/index.js.map +1 -0
  64. package/dist/index.d.ts +18 -0
  65. package/dist/index.d.ts.map +1 -0
  66. package/dist/index.js +78 -0
  67. package/dist/index.js.map +1 -0
  68. package/{src/security/index.ts → dist/security/index.d.ts} +8 -15
  69. package/dist/security/index.d.ts.map +1 -0
  70. package/dist/security/index.js +12 -0
  71. package/dist/security/index.js.map +1 -0
  72. package/dist/security/security-scanner.d.ts +10 -0
  73. package/dist/security/security-scanner.d.ts.map +1 -0
  74. package/dist/security/security-scanner.js +541 -0
  75. package/dist/security/security-scanner.js.map +1 -0
  76. package/dist/security/types.d.ts +48 -0
  77. package/dist/security/types.d.ts.map +1 -0
  78. package/dist/security/types.js +21 -0
  79. package/dist/security/types.js.map +1 -0
  80. package/dist/services/directory.d.ts +104 -0
  81. package/dist/services/directory.d.ts.map +1 -0
  82. package/dist/services/directory.js +333 -0
  83. package/dist/services/directory.js.map +1 -0
  84. package/dist/simulator/agent-simulator.d.ts +69 -0
  85. package/dist/simulator/agent-simulator.d.ts.map +1 -0
  86. package/{src/simulator/agent-simulator.ts → dist/simulator/agent-simulator.js} +650 -941
  87. package/dist/simulator/agent-simulator.js.map +1 -0
  88. package/{src/simulator/index.ts → dist/simulator/index.d.ts} +7 -7
  89. package/dist/simulator/index.d.ts.map +1 -0
  90. package/dist/simulator/index.js +23 -0
  91. package/dist/simulator/index.js.map +1 -0
  92. package/{src/simulator/types.ts → dist/simulator/types.d.ts} +145 -170
  93. package/dist/simulator/types.d.ts.map +1 -0
  94. package/dist/simulator/types.js +18 -0
  95. package/dist/simulator/types.js.map +1 -0
  96. package/dist/types/generator.d.ts +106 -0
  97. package/dist/types/generator.d.ts.map +1 -0
  98. package/dist/types/generator.js +6 -0
  99. package/dist/types/generator.js.map +1 -0
  100. package/{src/types/index.ts → dist/types/index.d.ts} +1 -1
  101. package/dist/types/index.d.ts.map +1 -0
  102. package/dist/types/index.js +23 -0
  103. package/dist/types/index.js.map +1 -0
  104. package/dist/types/ucp-profile.d.ts +103 -0
  105. package/dist/types/ucp-profile.d.ts.map +1 -0
  106. package/dist/types/ucp-profile.js +45 -0
  107. package/dist/types/ucp-profile.js.map +1 -0
  108. package/dist/types/validation.d.ts +68 -0
  109. package/dist/types/validation.d.ts.map +1 -0
  110. package/dist/types/validation.js +32 -0
  111. package/dist/types/validation.js.map +1 -0
  112. package/dist/validator/index.d.ts +26 -0
  113. package/dist/validator/index.d.ts.map +1 -0
  114. package/dist/validator/index.js +161 -0
  115. package/dist/validator/index.js.map +1 -0
  116. package/dist/validator/network-validator.d.ts +28 -0
  117. package/dist/validator/network-validator.d.ts.map +1 -0
  118. package/dist/validator/network-validator.js +319 -0
  119. package/dist/validator/network-validator.js.map +1 -0
  120. package/dist/validator/rules-validator.d.ts +11 -0
  121. package/dist/validator/rules-validator.d.ts.map +1 -0
  122. package/dist/validator/rules-validator.js +257 -0
  123. package/dist/validator/rules-validator.js.map +1 -0
  124. package/dist/validator/sdk-validator.d.ts +58 -0
  125. package/dist/validator/sdk-validator.d.ts.map +1 -0
  126. package/{src/validator/sdk-validator.ts → dist/validator/sdk-validator.js} +273 -330
  127. package/dist/validator/sdk-validator.js.map +1 -0
  128. package/dist/validator/structural-validator.d.ts +11 -0
  129. package/dist/validator/structural-validator.d.ts.map +1 -0
  130. package/dist/validator/structural-validator.js +415 -0
  131. package/dist/validator/structural-validator.js.map +1 -0
  132. package/package.json +1 -1
  133. package/publish-output.txt +0 -0
  134. package/CLAUDE.md +0 -109
  135. package/api/analyze-feed.js +0 -140
  136. package/api/badge.js +0 -185
  137. package/api/benchmark.js +0 -177
  138. package/api/directory-stats.ts +0 -29
  139. package/api/directory.ts +0 -73
  140. package/api/generate-compliance.js +0 -143
  141. package/api/generate-schema.js +0 -457
  142. package/api/generate.js +0 -132
  143. package/api/security-scan.js +0 -133
  144. package/api/simulate.js +0 -187
  145. package/api/tsconfig.json +0 -10
  146. package/api/validate.js +0 -1351
  147. package/apify-actor/.actor/actor.json +0 -68
  148. package/apify-actor/.actor/input_schema.json +0 -32
  149. package/apify-actor/APIFY-STORE-LISTING.md +0 -412
  150. package/apify-actor/Dockerfile +0 -8
  151. package/apify-actor/README.md +0 -166
  152. package/apify-actor/main.ts +0 -111
  153. package/apify-actor/package.json +0 -17
  154. package/apify-actor/src/main.js +0 -199
  155. package/docs/BRAND-IDENTITY.md +0 -238
  156. package/docs/BRAND-STYLE-GUIDE.md +0 -356
  157. package/drizzle/0000_black_king_cobra.sql +0 -39
  158. package/drizzle/meta/0000_snapshot.json +0 -309
  159. package/drizzle/meta/_journal.json +0 -13
  160. package/drizzle.config.ts +0 -10
  161. package/public/.well-known/ucp +0 -25
  162. package/public/android-chrome-192x192.png +0 -0
  163. package/public/android-chrome-512x512.png +0 -0
  164. package/public/apple-touch-icon.png +0 -0
  165. package/public/brand.css +0 -321
  166. package/public/directory.html +0 -701
  167. package/public/favicon-16x16.png +0 -0
  168. package/public/favicon-32x32.png +0 -0
  169. package/public/favicon.ico +0 -0
  170. package/public/guides/bigcommerce.html +0 -743
  171. package/public/guides/fastucp.html +0 -838
  172. package/public/guides/magento.html +0 -779
  173. package/public/guides/shopify.html +0 -726
  174. package/public/guides/squarespace.html +0 -749
  175. package/public/guides/wix.html +0 -747
  176. package/public/guides/woocommerce.html +0 -733
  177. package/public/index.html +0 -3835
  178. package/public/learn.html +0 -396
  179. package/public/logo.jpeg +0 -0
  180. package/public/og-image-icon.png +0 -0
  181. package/public/og-image.png +0 -0
  182. package/public/robots.txt +0 -6
  183. package/public/site.webmanifest +0 -31
  184. package/public/sitemap.xml +0 -69
  185. package/public/social/linkedin-banner-1128x191.png +0 -0
  186. package/public/social/temp.PNG +0 -0
  187. package/public/social/x-header-1500x500.png +0 -0
  188. package/public/verify.html +0 -410
  189. package/scripts/generate-favicons.js +0 -44
  190. package/scripts/generate-ico.js +0 -23
  191. package/scripts/generate-og-image.js +0 -45
  192. package/scripts/reset-db.ts +0 -77
  193. package/scripts/seed-db.ts +0 -71
  194. package/scripts/setup-benchmark-db.js +0 -70
  195. package/src/api/server.ts +0 -266
  196. package/src/cli/index.ts +0 -302
  197. package/src/compliance/compliance-generator.ts +0 -452
  198. package/src/compliance/index.ts +0 -28
  199. package/src/compliance/types.ts +0 -170
  200. package/src/db/index.ts +0 -28
  201. package/src/db/schema.ts +0 -84
  202. package/src/feed-analyzer/index.ts +0 -34
  203. package/src/feed-analyzer/types.ts +0 -354
  204. package/src/generator/key-generator.ts +0 -124
  205. package/src/generator/profile-builder.ts +0 -402
  206. package/src/index.ts +0 -105
  207. package/src/security/security-scanner.ts +0 -604
  208. package/src/security/types.ts +0 -55
  209. package/src/services/directory.ts +0 -434
  210. package/src/types/generator.ts +0 -140
  211. package/src/types/ucp-profile.ts +0 -140
  212. package/src/types/validation.ts +0 -89
  213. package/src/validator/index.ts +0 -194
  214. package/src/validator/network-validator.ts +0 -417
  215. package/src/validator/rules-validator.ts +0 -297
  216. package/src/validator/structural-validator.ts +0 -476
  217. package/tests/fixtures/non-compliant-profile.json +0 -25
  218. package/tests/fixtures/official-sample-profile.json +0 -75
  219. package/tests/integration/benchmark.test.ts +0 -207
  220. package/tests/integration/database.test.ts +0 -163
  221. package/tests/integration/directory-api.test.ts +0 -268
  222. package/tests/integration/simulate-api.test.ts +0 -230
  223. package/tests/integration/validate-api.test.ts +0 -269
  224. package/tests/setup.ts +0 -15
  225. package/tests/unit/agent-simulator.test.ts +0 -575
  226. package/tests/unit/compliance-generator.test.ts +0 -374
  227. package/tests/unit/directory-service.test.ts +0 -272
  228. package/tests/unit/feed-analyzer.test.ts +0 -517
  229. package/tests/unit/lint-suggestions.test.ts +0 -423
  230. package/tests/unit/official-samples.test.ts +0 -211
  231. package/tests/unit/pdf-report.test.ts +0 -390
  232. package/tests/unit/sdk-validator.test.ts +0 -531
  233. package/tests/unit/security-scanner.test.ts +0 -410
  234. package/tests/unit/validation.test.ts +0 -390
  235. package/vercel.json +0 -34
  236. package/vitest.config.ts +0 -22
@@ -1,476 +0,0 @@
1
- /**
2
- * Structural Validator
3
- * Validates UCP Profile JSON structure (no network calls)
4
- */
5
-
6
- import type { UcpProfile, UcpService, UcpCapability } from '../types/ucp-profile.js';
7
- import type { ValidationIssue } from '../types/validation.js';
8
- import { ValidationErrorCodes } from '../types/validation.js';
9
-
10
- /**
11
- * Version format regex (YYYY-MM-DD)
12
- */
13
- const VERSION_REGEX = /^\d{4}-\d{2}-\d{2}$/;
14
-
15
- /**
16
- * Validate the structural integrity of a UCP profile
17
- * Returns an array of validation issues
18
- */
19
- export function validateStructure(profile: unknown): ValidationIssue[] {
20
- const issues: ValidationIssue[] = [];
21
-
22
- // Check if it's an object
23
- if (!profile || typeof profile !== 'object') {
24
- issues.push({
25
- severity: 'error',
26
- code: ValidationErrorCodes.MISSING_UCP_OBJECT,
27
- path: '$',
28
- message: 'Profile must be a JSON object',
29
- hint: 'Ensure your profile is valid JSON and contains a root object',
30
- });
31
- return issues;
32
- }
33
-
34
- const profileObj = profile as Record<string, unknown>;
35
-
36
- // Check for ucp root object
37
- if (!profileObj.ucp || typeof profileObj.ucp !== 'object') {
38
- issues.push({
39
- severity: 'error',
40
- code: ValidationErrorCodes.MISSING_UCP_OBJECT,
41
- path: '$.ucp',
42
- message: 'Missing required "ucp" object at root level',
43
- hint: 'Add a "ucp" object containing version, services, and capabilities',
44
- });
45
- return issues;
46
- }
47
-
48
- const ucp = profileObj.ucp as Record<string, unknown>;
49
-
50
- // Validate version
51
- issues.push(...validateVersion(ucp));
52
-
53
- // Validate services
54
- issues.push(...validateServices(ucp));
55
-
56
- // Validate capabilities
57
- issues.push(...validateCapabilities(ucp));
58
-
59
- // Validate signing_keys if present
60
- if (profileObj.signing_keys !== undefined) {
61
- issues.push(...validateSigningKeys(profileObj.signing_keys));
62
- }
63
-
64
- return issues;
65
- }
66
-
67
- /**
68
- * Validate UCP version field
69
- */
70
- function validateVersion(ucp: Record<string, unknown>): ValidationIssue[] {
71
- const issues: ValidationIssue[] = [];
72
-
73
- if (!ucp.version) {
74
- issues.push({
75
- severity: 'error',
76
- code: ValidationErrorCodes.MISSING_VERSION,
77
- path: '$.ucp.version',
78
- message: 'Missing required "version" field in ucp object',
79
- hint: 'Add version field with format "YYYY-MM-DD" (e.g., "2026-01-11")',
80
- });
81
- } else if (typeof ucp.version !== 'string') {
82
- issues.push({
83
- severity: 'error',
84
- code: ValidationErrorCodes.INVALID_VERSION_FORMAT,
85
- path: '$.ucp.version',
86
- message: 'Version must be a string',
87
- hint: 'Use format "YYYY-MM-DD" (e.g., "2026-01-11")',
88
- });
89
- } else if (!VERSION_REGEX.test(ucp.version)) {
90
- issues.push({
91
- severity: 'error',
92
- code: ValidationErrorCodes.INVALID_VERSION_FORMAT,
93
- path: '$.ucp.version',
94
- message: `Invalid version format: "${ucp.version}"`,
95
- hint: 'Use format "YYYY-MM-DD" (e.g., "2026-01-11")',
96
- });
97
- }
98
-
99
- return issues;
100
- }
101
-
102
- /**
103
- * Validate services object
104
- */
105
- function validateServices(ucp: Record<string, unknown>): ValidationIssue[] {
106
- const issues: ValidationIssue[] = [];
107
-
108
- if (!ucp.services) {
109
- issues.push({
110
- severity: 'error',
111
- code: ValidationErrorCodes.MISSING_SERVICES,
112
- path: '$.ucp.services',
113
- message: 'Missing required "services" field in ucp object',
114
- hint: 'Add a services object with at least one service definition',
115
- });
116
- return issues;
117
- }
118
-
119
- if (typeof ucp.services !== 'object' || Array.isArray(ucp.services)) {
120
- issues.push({
121
- severity: 'error',
122
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
123
- path: '$.ucp.services',
124
- message: 'Services must be an object (not an array)',
125
- hint: 'Use format: { "dev.ucp.shopping": { ... } }',
126
- });
127
- return issues;
128
- }
129
-
130
- const services = ucp.services as Record<string, unknown>;
131
-
132
- // Check for at least one service
133
- if (Object.keys(services).length === 0) {
134
- issues.push({
135
- severity: 'warn',
136
- code: ValidationErrorCodes.MISSING_SERVICES,
137
- path: '$.ucp.services',
138
- message: 'Services object is empty',
139
- hint: 'Add at least one service (e.g., "dev.ucp.shopping")',
140
- });
141
- }
142
-
143
- // Validate each service
144
- for (const [serviceName, service] of Object.entries(services)) {
145
- issues.push(...validateService(serviceName, service));
146
- }
147
-
148
- return issues;
149
- }
150
-
151
- /**
152
- * Validate individual service definition
153
- */
154
- function validateService(name: string, service: unknown): ValidationIssue[] {
155
- const issues: ValidationIssue[] = [];
156
- const path = `$.ucp.services["${name}"]`;
157
-
158
- if (!service || typeof service !== 'object') {
159
- issues.push({
160
- severity: 'error',
161
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
162
- path,
163
- message: `Service "${name}" must be an object`,
164
- });
165
- return issues;
166
- }
167
-
168
- const svc = service as Record<string, unknown>;
169
-
170
- // Check required version
171
- if (!svc.version) {
172
- issues.push({
173
- severity: 'error',
174
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
175
- path: `${path}.version`,
176
- message: `Service "${name}" missing required "version" field`,
177
- });
178
- } else if (typeof svc.version === 'string' && !VERSION_REGEX.test(svc.version)) {
179
- issues.push({
180
- severity: 'error',
181
- code: ValidationErrorCodes.INVALID_VERSION_FORMAT,
182
- path: `${path}.version`,
183
- message: `Invalid version format in service "${name}"`,
184
- hint: 'Use format "YYYY-MM-DD"',
185
- });
186
- }
187
-
188
- // Check required spec
189
- if (!svc.spec) {
190
- issues.push({
191
- severity: 'error',
192
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
193
- path: `${path}.spec`,
194
- message: `Service "${name}" missing required "spec" field`,
195
- });
196
- }
197
-
198
- // Check for at least one transport
199
- const hasTransport = svc.rest || svc.mcp || svc.a2a || svc.embedded;
200
- if (!hasTransport) {
201
- issues.push({
202
- severity: 'warn',
203
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
204
- path,
205
- message: `Service "${name}" has no transport bindings`,
206
- hint: 'Add at least one transport: rest, mcp, a2a, or embedded',
207
- });
208
- }
209
-
210
- // Validate REST transport if present
211
- if (svc.rest) {
212
- issues.push(...validateRestTransport(svc.rest, `${path}.rest`));
213
- }
214
-
215
- // Validate MCP transport if present
216
- if (svc.mcp) {
217
- issues.push(...validateMcpTransport(svc.mcp, `${path}.mcp`));
218
- }
219
-
220
- return issues;
221
- }
222
-
223
- /**
224
- * Validate REST transport binding
225
- */
226
- function validateRestTransport(transport: unknown, path: string): ValidationIssue[] {
227
- const issues: ValidationIssue[] = [];
228
-
229
- if (!transport || typeof transport !== 'object') {
230
- issues.push({
231
- severity: 'error',
232
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
233
- path,
234
- message: 'REST transport must be an object',
235
- });
236
- return issues;
237
- }
238
-
239
- const rest = transport as Record<string, unknown>;
240
-
241
- if (!rest.schema) {
242
- issues.push({
243
- severity: 'error',
244
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
245
- path: `${path}.schema`,
246
- message: 'REST transport missing required "schema" field',
247
- });
248
- }
249
-
250
- if (!rest.endpoint) {
251
- issues.push({
252
- severity: 'error',
253
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
254
- path: `${path}.endpoint`,
255
- message: 'REST transport missing required "endpoint" field',
256
- });
257
- }
258
-
259
- return issues;
260
- }
261
-
262
- /**
263
- * Validate MCP transport binding
264
- */
265
- function validateMcpTransport(transport: unknown, path: string): ValidationIssue[] {
266
- const issues: ValidationIssue[] = [];
267
-
268
- if (!transport || typeof transport !== 'object') {
269
- issues.push({
270
- severity: 'error',
271
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
272
- path,
273
- message: 'MCP transport must be an object',
274
- });
275
- return issues;
276
- }
277
-
278
- const mcp = transport as Record<string, unknown>;
279
-
280
- if (!mcp.schema) {
281
- issues.push({
282
- severity: 'error',
283
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
284
- path: `${path}.schema`,
285
- message: 'MCP transport missing required "schema" field',
286
- });
287
- }
288
-
289
- if (!mcp.endpoint) {
290
- issues.push({
291
- severity: 'error',
292
- code: ValidationErrorCodes.INVALID_SERVICE_STRUCTURE,
293
- path: `${path}.endpoint`,
294
- message: 'MCP transport missing required "endpoint" field',
295
- });
296
- }
297
-
298
- return issues;
299
- }
300
-
301
- /**
302
- * Validate capabilities array
303
- */
304
- function validateCapabilities(ucp: Record<string, unknown>): ValidationIssue[] {
305
- const issues: ValidationIssue[] = [];
306
-
307
- if (!ucp.capabilities) {
308
- issues.push({
309
- severity: 'error',
310
- code: ValidationErrorCodes.MISSING_CAPABILITIES,
311
- path: '$.ucp.capabilities',
312
- message: 'Missing required "capabilities" field in ucp object',
313
- hint: 'Add a capabilities array with at least one capability',
314
- });
315
- return issues;
316
- }
317
-
318
- if (!Array.isArray(ucp.capabilities)) {
319
- issues.push({
320
- severity: 'error',
321
- code: ValidationErrorCodes.INVALID_CAPABILITY_STRUCTURE,
322
- path: '$.ucp.capabilities',
323
- message: 'Capabilities must be an array',
324
- });
325
- return issues;
326
- }
327
-
328
- if (ucp.capabilities.length === 0) {
329
- issues.push({
330
- severity: 'warn',
331
- code: ValidationErrorCodes.MISSING_CAPABILITIES,
332
- path: '$.ucp.capabilities',
333
- message: 'Capabilities array is empty',
334
- hint: 'Add at least one capability (e.g., dev.ucp.shopping.checkout)',
335
- });
336
- }
337
-
338
- // Validate each capability
339
- for (let i = 0; i < ucp.capabilities.length; i++) {
340
- issues.push(...validateCapability(ucp.capabilities[i], i));
341
- }
342
-
343
- return issues;
344
- }
345
-
346
- /**
347
- * Validate individual capability
348
- */
349
- function validateCapability(capability: unknown, index: number): ValidationIssue[] {
350
- const issues: ValidationIssue[] = [];
351
- const path = `$.ucp.capabilities[${index}]`;
352
-
353
- if (!capability || typeof capability !== 'object') {
354
- issues.push({
355
- severity: 'error',
356
- code: ValidationErrorCodes.INVALID_CAPABILITY_STRUCTURE,
357
- path,
358
- message: `Capability at index ${index} must be an object`,
359
- });
360
- return issues;
361
- }
362
-
363
- const cap = capability as Record<string, unknown>;
364
-
365
- // Check required fields
366
- if (!cap.name) {
367
- issues.push({
368
- severity: 'error',
369
- code: ValidationErrorCodes.INVALID_CAPABILITY_STRUCTURE,
370
- path: `${path}.name`,
371
- message: 'Capability missing required "name" field',
372
- });
373
- }
374
-
375
- if (!cap.version) {
376
- issues.push({
377
- severity: 'error',
378
- code: ValidationErrorCodes.INVALID_CAPABILITY_STRUCTURE,
379
- path: `${path}.version`,
380
- message: 'Capability missing required "version" field',
381
- });
382
- } else if (typeof cap.version === 'string' && !VERSION_REGEX.test(cap.version)) {
383
- issues.push({
384
- severity: 'error',
385
- code: ValidationErrorCodes.INVALID_VERSION_FORMAT,
386
- path: `${path}.version`,
387
- message: `Invalid version format: "${cap.version}"`,
388
- hint: 'Use format "YYYY-MM-DD"',
389
- });
390
- }
391
-
392
- if (!cap.spec) {
393
- issues.push({
394
- severity: 'error',
395
- code: ValidationErrorCodes.INVALID_CAPABILITY_STRUCTURE,
396
- path: `${path}.spec`,
397
- message: 'Capability missing required "spec" field',
398
- });
399
- }
400
-
401
- if (!cap.schema) {
402
- issues.push({
403
- severity: 'error',
404
- code: ValidationErrorCodes.INVALID_CAPABILITY_STRUCTURE,
405
- path: `${path}.schema`,
406
- message: 'Capability missing required "schema" field',
407
- });
408
- }
409
-
410
- return issues;
411
- }
412
-
413
- /**
414
- * Validate signing_keys structure
415
- * Accepts signing_keys as a direct array of JWKs (official UCP format)
416
- */
417
- function validateSigningKeys(signingKeys: unknown): ValidationIssue[] {
418
- const issues: ValidationIssue[] = [];
419
- const path = '$.signing_keys';
420
-
421
- // signing_keys should be an array of JWKs
422
- if (!Array.isArray(signingKeys)) {
423
- issues.push({
424
- severity: 'error',
425
- code: ValidationErrorCodes.INVALID_SIGNING_KEY,
426
- path,
427
- message: 'signing_keys must be an array of JWK objects',
428
- });
429
- return issues;
430
- }
431
-
432
- for (let i = 0; i < signingKeys.length; i++) {
433
- issues.push(...validateJwk(signingKeys[i], `${path}[${i}]`));
434
- }
435
-
436
- return issues;
437
- }
438
-
439
- /**
440
- * Validate JWK structure
441
- */
442
- function validateJwk(jwk: unknown, path: string): ValidationIssue[] {
443
- const issues: ValidationIssue[] = [];
444
-
445
- if (!jwk || typeof jwk !== 'object') {
446
- issues.push({
447
- severity: 'error',
448
- code: ValidationErrorCodes.INVALID_SIGNING_KEY,
449
- path,
450
- message: 'JWK must be an object',
451
- });
452
- return issues;
453
- }
454
-
455
- const key = jwk as Record<string, unknown>;
456
-
457
- if (!key.kty) {
458
- issues.push({
459
- severity: 'error',
460
- code: ValidationErrorCodes.INVALID_SIGNING_KEY,
461
- path: `${path}.kty`,
462
- message: 'JWK missing required "kty" field',
463
- });
464
- }
465
-
466
- if (!key.kid) {
467
- issues.push({
468
- severity: 'error',
469
- code: ValidationErrorCodes.INVALID_SIGNING_KEY,
470
- path: `${path}.kid`,
471
- message: 'JWK missing required "kid" field',
472
- });
473
- }
474
-
475
- return issues;
476
- }
@@ -1,25 +0,0 @@
1
- {
2
- "ucp": {
3
- "version": "1.0",
4
- "services": {
5
- "checkout": "/api/ucp/checkout",
6
- "orders": "/api/ucp/orders",
7
- "products": "/api/ucp/products"
8
- },
9
- "capabilities": [
10
- "dev.ucp.shopping.checkout",
11
- "dev.ucp.shopping.catalog",
12
- "dev.ucp.shopping.fulfillment"
13
- ],
14
- "merchant": {
15
- "name": "Test Merchant",
16
- "description": "A non-compliant but working UCP profile",
17
- "website": "https://example.com"
18
- },
19
- "sandbox": true
20
- },
21
- "payment": {
22
- "sandbox_mode": true,
23
- "accepted_tokens": ["sandbox_*", "test"]
24
- }
25
- }
@@ -1,75 +0,0 @@
1
- {
2
- "ucp": {
3
- "version": "2026-01-11",
4
- "services": {
5
- "dev.ucp.shopping": {
6
- "version": "2026-01-11",
7
- "spec": "https://ucp.dev/specs/shopping",
8
- "rest": {
9
- "schema": "https://ucp.dev/services/shopping/openapi.json",
10
- "endpoint": "https://example.com/api"
11
- }
12
- }
13
- },
14
- "capabilities": [
15
- {
16
- "name": "dev.ucp.shopping.checkout",
17
- "version": "2026-01-11",
18
- "spec": "https://ucp.dev/specs/shopping/checkout",
19
- "schema": "https://ucp.dev/schemas/shopping/checkout.json"
20
- },
21
- {
22
- "name": "dev.ucp.shopping.order",
23
- "version": "2026-01-11",
24
- "spec": "https://ucp.dev/specs/shopping/order",
25
- "schema": "https://ucp.dev/schemas/shopping/order.json",
26
- "extends": "dev.ucp.shopping.checkout"
27
- },
28
- {
29
- "name": "dev.ucp.shopping.fulfillment",
30
- "version": "2026-01-11",
31
- "spec": "https://ucp.dev/specs/shopping/fulfillment",
32
- "schema": "https://ucp.dev/schemas/shopping/fulfillment.json",
33
- "extends": "dev.ucp.shopping.order"
34
- },
35
- {
36
- "name": "dev.ucp.shopping.discount",
37
- "version": "2026-01-11",
38
- "spec": "https://ucp.dev/specs/shopping/discount",
39
- "schema": "https://ucp.dev/schemas/shopping/discount.json",
40
- "extends": "dev.ucp.shopping.checkout"
41
- },
42
- {
43
- "name": "dev.ucp.shopping.buyer_consent",
44
- "version": "2026-01-11",
45
- "spec": "https://ucp.dev/specs/shopping/buyer_consent",
46
- "schema": "https://ucp.dev/schemas/shopping/buyer_consent.json",
47
- "extends": "dev.ucp.shopping.checkout"
48
- }
49
- ]
50
- },
51
- "signing_keys": [
52
- {
53
- "kty": "EC",
54
- "crv": "P-256",
55
- "x": "test-x-coordinate",
56
- "y": "test-y-coordinate",
57
- "kid": "key-2026-01"
58
- }
59
- ],
60
- "payment": {
61
- "handlers": [
62
- {
63
- "id": "mock_payment",
64
- "name": "com.example.mock_payment",
65
- "version": "2026-01-11",
66
- "spec": "https://example.com/payment/spec",
67
- "config_schema": "https://example.com/payment/config.json",
68
- "instrument_schemas": ["https://example.com/payment/instrument.json"],
69
- "config": {
70
- "mode": "sandbox"
71
- }
72
- }
73
- ]
74
- }
75
- }