@ucptools/validator 1.0.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 (121) hide show
  1. package/CLAUDE.md +109 -0
  2. package/CONTRIBUTING.md +113 -0
  3. package/LICENSE +21 -0
  4. package/README.md +203 -0
  5. package/api/analyze-feed.js +140 -0
  6. package/api/badge.js +185 -0
  7. package/api/benchmark.js +177 -0
  8. package/api/directory-stats.ts +29 -0
  9. package/api/directory.ts +73 -0
  10. package/api/generate-compliance.js +143 -0
  11. package/api/generate-schema.js +457 -0
  12. package/api/generate.js +132 -0
  13. package/api/security-scan.js +133 -0
  14. package/api/simulate.js +187 -0
  15. package/api/tsconfig.json +10 -0
  16. package/api/validate.js +1351 -0
  17. package/apify-actor/.actor/actor.json +68 -0
  18. package/apify-actor/.actor/input_schema.json +32 -0
  19. package/apify-actor/APIFY-STORE-LISTING.md +412 -0
  20. package/apify-actor/Dockerfile +8 -0
  21. package/apify-actor/README.md +166 -0
  22. package/apify-actor/main.ts +111 -0
  23. package/apify-actor/package.json +17 -0
  24. package/apify-actor/src/main.js +199 -0
  25. package/docs/BRAND-IDENTITY.md +238 -0
  26. package/docs/BRAND-STYLE-GUIDE.md +356 -0
  27. package/drizzle/0000_black_king_cobra.sql +39 -0
  28. package/drizzle/meta/0000_snapshot.json +309 -0
  29. package/drizzle/meta/_journal.json +13 -0
  30. package/drizzle.config.ts +10 -0
  31. package/examples/full-profile.json +70 -0
  32. package/examples/minimal-profile.json +23 -0
  33. package/package.json +69 -0
  34. package/public/.well-known/ucp +25 -0
  35. package/public/android-chrome-192x192.png +0 -0
  36. package/public/android-chrome-512x512.png +0 -0
  37. package/public/apple-touch-icon.png +0 -0
  38. package/public/brand.css +321 -0
  39. package/public/directory.html +701 -0
  40. package/public/favicon-16x16.png +0 -0
  41. package/public/favicon-32x32.png +0 -0
  42. package/public/favicon.ico +0 -0
  43. package/public/guides/bigcommerce.html +743 -0
  44. package/public/guides/fastucp.html +838 -0
  45. package/public/guides/magento.html +779 -0
  46. package/public/guides/shopify.html +726 -0
  47. package/public/guides/squarespace.html +749 -0
  48. package/public/guides/wix.html +747 -0
  49. package/public/guides/woocommerce.html +733 -0
  50. package/public/index.html +3835 -0
  51. package/public/learn.html +396 -0
  52. package/public/logo.jpeg +0 -0
  53. package/public/og-image-icon.png +0 -0
  54. package/public/og-image.png +0 -0
  55. package/public/robots.txt +6 -0
  56. package/public/site.webmanifest +31 -0
  57. package/public/sitemap.xml +69 -0
  58. package/public/social/linkedin-banner-1128x191.png +0 -0
  59. package/public/social/temp.PNG +0 -0
  60. package/public/social/x-header-1500x500.png +0 -0
  61. package/public/verify.html +410 -0
  62. package/scripts/generate-favicons.js +44 -0
  63. package/scripts/generate-ico.js +23 -0
  64. package/scripts/generate-og-image.js +45 -0
  65. package/scripts/reset-db.ts +77 -0
  66. package/scripts/seed-db.ts +71 -0
  67. package/scripts/setup-benchmark-db.js +70 -0
  68. package/src/api/server.ts +266 -0
  69. package/src/cli/index.ts +302 -0
  70. package/src/compliance/compliance-generator.ts +452 -0
  71. package/src/compliance/index.ts +28 -0
  72. package/src/compliance/templates.ts +338 -0
  73. package/src/compliance/types.ts +170 -0
  74. package/src/db/index.ts +28 -0
  75. package/src/db/schema.ts +84 -0
  76. package/src/feed-analyzer/feed-analyzer.ts +726 -0
  77. package/src/feed-analyzer/index.ts +34 -0
  78. package/src/feed-analyzer/types.ts +354 -0
  79. package/src/generator/index.ts +7 -0
  80. package/src/generator/key-generator.ts +124 -0
  81. package/src/generator/profile-builder.ts +402 -0
  82. package/src/hosting/artifacts-generator.ts +679 -0
  83. package/src/hosting/index.ts +6 -0
  84. package/src/index.ts +105 -0
  85. package/src/security/index.ts +15 -0
  86. package/src/security/security-scanner.ts +604 -0
  87. package/src/security/types.ts +55 -0
  88. package/src/services/directory.ts +434 -0
  89. package/src/simulator/agent-simulator.ts +941 -0
  90. package/src/simulator/index.ts +7 -0
  91. package/src/simulator/types.ts +170 -0
  92. package/src/types/generator.ts +140 -0
  93. package/src/types/index.ts +7 -0
  94. package/src/types/ucp-profile.ts +140 -0
  95. package/src/types/validation.ts +89 -0
  96. package/src/validator/index.ts +194 -0
  97. package/src/validator/network-validator.ts +417 -0
  98. package/src/validator/rules-validator.ts +297 -0
  99. package/src/validator/sdk-validator.ts +330 -0
  100. package/src/validator/structural-validator.ts +476 -0
  101. package/tests/fixtures/non-compliant-profile.json +25 -0
  102. package/tests/fixtures/official-sample-profile.json +75 -0
  103. package/tests/integration/benchmark.test.ts +207 -0
  104. package/tests/integration/database.test.ts +163 -0
  105. package/tests/integration/directory-api.test.ts +268 -0
  106. package/tests/integration/simulate-api.test.ts +230 -0
  107. package/tests/integration/validate-api.test.ts +269 -0
  108. package/tests/setup.ts +15 -0
  109. package/tests/unit/agent-simulator.test.ts +575 -0
  110. package/tests/unit/compliance-generator.test.ts +374 -0
  111. package/tests/unit/directory-service.test.ts +272 -0
  112. package/tests/unit/feed-analyzer.test.ts +517 -0
  113. package/tests/unit/lint-suggestions.test.ts +423 -0
  114. package/tests/unit/official-samples.test.ts +211 -0
  115. package/tests/unit/pdf-report.test.ts +390 -0
  116. package/tests/unit/sdk-validator.test.ts +531 -0
  117. package/tests/unit/security-scanner.test.ts +410 -0
  118. package/tests/unit/validation.test.ts +390 -0
  119. package/tsconfig.json +20 -0
  120. package/vercel.json +34 -0
  121. package/vitest.config.ts +22 -0
@@ -0,0 +1,402 @@
1
+ /**
2
+ * UCP Profile Builder
3
+ * Generates valid UCP Business Profiles from configuration inputs
4
+ */
5
+
6
+ import type {
7
+ UcpProfile,
8
+ UcpObject,
9
+ UcpService,
10
+ UcpCapability,
11
+ SigningKeys,
12
+ JwkKey,
13
+ } from '../types/ucp-profile.js';
14
+ import type {
15
+ GeneratorInput,
16
+ GeneratorOutput,
17
+ TransportConfig,
18
+ CapabilitySelection,
19
+ } from '../types/generator.js';
20
+ import {
21
+ CURRENT_UCP_VERSION,
22
+ KNOWN_CAPABILITIES,
23
+ KNOWN_SERVICES,
24
+ UCP_DEFAULTS,
25
+ } from '../types/ucp-profile.js';
26
+ import { generateSigningKeyPair } from './key-generator.js';
27
+
28
+ /**
29
+ * Build a complete UCP Profile from generator inputs
30
+ */
31
+ export async function buildProfile(input: GeneratorInput): Promise<GeneratorOutput> {
32
+ const ucpVersion = input.ucpVersion || CURRENT_UCP_VERSION;
33
+
34
+ // Build services
35
+ const services = buildServices(input.transport, ucpVersion);
36
+
37
+ // Build capabilities
38
+ const capabilities = buildCapabilities(input.capabilities, ucpVersion);
39
+
40
+ // Build the UCP object
41
+ const ucpObject: UcpObject = {
42
+ version: ucpVersion,
43
+ services,
44
+ capabilities,
45
+ };
46
+
47
+ // Build the full profile
48
+ const profile: UcpProfile = {
49
+ ucp: ucpObject,
50
+ };
51
+
52
+ // Handle signing keys if Order capability is enabled
53
+ let signingKeyPair: GeneratorOutput['signingKeyPair'] | undefined;
54
+
55
+ if (input.capabilities.order) {
56
+ if (input.security?.generateSigningKeys !== false) {
57
+ // Generate new signing keys
58
+ const keyPair = await generateSigningKeyPair(
59
+ input.security?.signingKeyAlgorithm || 'ES256'
60
+ );
61
+ profile.signing_keys = [keyPair.publicKey];
62
+ signingKeyPair = keyPair;
63
+ } else if (input.security?.uploadedPublicKey) {
64
+ // Use uploaded public key
65
+ const publicKey = parsePublicKey(input.security.uploadedPublicKey);
66
+ profile.signing_keys = [publicKey];
67
+ }
68
+ }
69
+
70
+ // Format the profile JSON
71
+ const profileJson = JSON.stringify(profile, null, 2);
72
+
73
+ // Generate install instructions
74
+ const installInstructions = generateInstallInstructions(
75
+ input.merchant.primaryDomain,
76
+ input.merchant.merchantId
77
+ );
78
+
79
+ return {
80
+ profile,
81
+ profileJson,
82
+ installInstructions,
83
+ signingKeyPair,
84
+ };
85
+ }
86
+
87
+ /**
88
+ * Build services configuration
89
+ */
90
+ function buildServices(
91
+ transport: TransportConfig,
92
+ ucpVersion: string
93
+ ): Record<string, UcpService> {
94
+ const services: Record<string, UcpService> = {};
95
+
96
+ // Always include the shopping service for MVP
97
+ const shoppingService: UcpService = {
98
+ version: ucpVersion,
99
+ spec: `${UCP_DEFAULTS.SPEC_BASE}overview/`,
100
+ };
101
+
102
+ // Add REST transport (required for MVP)
103
+ if (transport.rest) {
104
+ shoppingService.rest = {
105
+ schema: transport.rest.schemaUrl || UCP_DEFAULTS.SHOPPING_REST_SCHEMA,
106
+ endpoint: normalizeEndpoint(transport.rest.endpoint),
107
+ };
108
+ }
109
+
110
+ // Add MCP transport (optional)
111
+ if (transport.mcp) {
112
+ shoppingService.mcp = {
113
+ schema: transport.mcp.schemaUrl || `${UCP_DEFAULTS.SERVICE_SCHEMA_BASE}shopping/mcp.json`,
114
+ endpoint: normalizeEndpoint(transport.mcp.endpoint),
115
+ };
116
+ }
117
+
118
+ // Add A2A transport (optional)
119
+ if (transport.a2a) {
120
+ shoppingService.a2a = {
121
+ agentCard: transport.a2a.agentCardUrl,
122
+ };
123
+ }
124
+
125
+ // Add embedded transport (optional)
126
+ if (transport.embedded) {
127
+ shoppingService.embedded = {
128
+ schema: transport.embedded.schemaUrl,
129
+ };
130
+ }
131
+
132
+ services[KNOWN_SERVICES.SHOPPING] = shoppingService;
133
+
134
+ return services;
135
+ }
136
+
137
+ /**
138
+ * Build capabilities array based on selection
139
+ */
140
+ function buildCapabilities(
141
+ selection: CapabilitySelection,
142
+ ucpVersion: string
143
+ ): UcpCapability[] {
144
+ const capabilities: UcpCapability[] = [];
145
+
146
+ // Checkout capability (default ON)
147
+ if (selection.checkout !== false) {
148
+ capabilities.push({
149
+ name: KNOWN_CAPABILITIES.CHECKOUT,
150
+ version: ucpVersion,
151
+ spec: UCP_DEFAULTS.CHECKOUT_SPEC,
152
+ schema: UCP_DEFAULTS.CHECKOUT_SCHEMA,
153
+ });
154
+ }
155
+
156
+ // Order capability
157
+ if (selection.order) {
158
+ capabilities.push({
159
+ name: KNOWN_CAPABILITIES.ORDER,
160
+ version: ucpVersion,
161
+ spec: UCP_DEFAULTS.ORDER_SPEC,
162
+ schema: UCP_DEFAULTS.ORDER_SCHEMA,
163
+ });
164
+ }
165
+
166
+ // Fulfillment capability (extends Order)
167
+ if (selection.fulfillment) {
168
+ capabilities.push({
169
+ name: KNOWN_CAPABILITIES.FULFILLMENT,
170
+ version: ucpVersion,
171
+ spec: UCP_DEFAULTS.FULFILLMENT_SPEC,
172
+ schema: UCP_DEFAULTS.FULFILLMENT_SCHEMA,
173
+ extends: KNOWN_CAPABILITIES.ORDER,
174
+ });
175
+ }
176
+
177
+ // Discount capability
178
+ if (selection.discount) {
179
+ capabilities.push({
180
+ name: KNOWN_CAPABILITIES.DISCOUNT,
181
+ version: ucpVersion,
182
+ spec: UCP_DEFAULTS.DISCOUNT_SPEC,
183
+ schema: UCP_DEFAULTS.DISCOUNT_SCHEMA,
184
+ });
185
+ }
186
+
187
+ // Custom capabilities
188
+ if (selection.customCapabilities) {
189
+ for (const custom of selection.customCapabilities) {
190
+ const capability: UcpCapability = {
191
+ name: `${custom.namespace}.${custom.name}`,
192
+ version: custom.version,
193
+ spec: custom.specUrl,
194
+ schema: custom.schemaUrl,
195
+ };
196
+ if (custom.extendsCapability) {
197
+ capability.extends = custom.extendsCapability;
198
+ }
199
+ capabilities.push(capability);
200
+ }
201
+ }
202
+
203
+ return capabilities;
204
+ }
205
+
206
+ /**
207
+ * Normalize endpoint URL (ensure https, no trailing slash)
208
+ */
209
+ function normalizeEndpoint(endpoint: string): string {
210
+ let normalized = endpoint.trim();
211
+
212
+ // Ensure https
213
+ if (!normalized.startsWith('https://')) {
214
+ if (normalized.startsWith('http://')) {
215
+ normalized = normalized.replace('http://', 'https://');
216
+ } else {
217
+ normalized = `https://${normalized}`;
218
+ }
219
+ }
220
+
221
+ // Remove trailing slash
222
+ if (normalized.endsWith('/')) {
223
+ normalized = normalized.slice(0, -1);
224
+ }
225
+
226
+ return normalized;
227
+ }
228
+
229
+ /**
230
+ * Parse uploaded public key (PEM or JWK format)
231
+ */
232
+ function parsePublicKey(keyData: string): JwkKey {
233
+ // Try to parse as JSON (JWK)
234
+ try {
235
+ const parsed = JSON.parse(keyData);
236
+ if (parsed.kty) {
237
+ return parsed as JwkKey;
238
+ }
239
+ } catch {
240
+ // Not JSON, assume PEM format
241
+ }
242
+
243
+ // For PEM format, we'll need to convert (simplified for now)
244
+ throw new Error(
245
+ 'PEM format not yet supported. Please provide the public key in JWK format.'
246
+ );
247
+ }
248
+
249
+ /**
250
+ * Generate installation instructions markdown
251
+ */
252
+ function generateInstallInstructions(
253
+ merchantDomain: string,
254
+ merchantId: string
255
+ ): string {
256
+ return `# UCP Profile Installation Instructions
257
+
258
+ ## Profile Location
259
+
260
+ Your UCP Business Profile must be accessible at:
261
+ \`\`\`
262
+ https://${merchantDomain}/.well-known/ucp
263
+ \`\`\`
264
+
265
+ ## Installation Options
266
+
267
+ ### Option 1: Static File (Recommended for most setups)
268
+
269
+ 1. Save the \`ucp.json\` file to your web server
270
+ 2. Configure your server to serve it at \`/.well-known/ucp\`
271
+
272
+ #### Nginx Configuration
273
+ \`\`\`nginx
274
+ location = /.well-known/ucp {
275
+ alias /path/to/ucp.json;
276
+ default_type application/json;
277
+ add_header Access-Control-Allow-Origin "*";
278
+ add_header Cache-Control "public, max-age=3600";
279
+ }
280
+ \`\`\`
281
+
282
+ #### Apache Configuration
283
+ \`\`\`apache
284
+ <Directory "/path/to/.well-known">
285
+ Options -Indexes
286
+ AllowOverride None
287
+ Require all granted
288
+ </Directory>
289
+
290
+ <Files "ucp">
291
+ ForceType application/json
292
+ Header set Access-Control-Allow-Origin "*"
293
+ Header set Cache-Control "public, max-age=3600"
294
+ </Files>
295
+ \`\`\`
296
+
297
+ #### Vercel (vercel.json)
298
+ \`\`\`json
299
+ {
300
+ "rewrites": [
301
+ { "source": "/.well-known/ucp", "destination": "/ucp.json" }
302
+ ],
303
+ "headers": [
304
+ {
305
+ "source": "/.well-known/ucp",
306
+ "headers": [
307
+ { "key": "Content-Type", "value": "application/json" },
308
+ { "key": "Access-Control-Allow-Origin", "value": "*" }
309
+ ]
310
+ }
311
+ ]
312
+ }
313
+ \`\`\`
314
+
315
+ #### Netlify (_redirects)
316
+ \`\`\`
317
+ /.well-known/ucp /ucp.json 200
318
+ \`\`\`
319
+
320
+ ### Option 2: Edge Worker (Cloudflare)
321
+
322
+ This approach proxies the profile from our hosted service.
323
+
324
+ \`\`\`javascript
325
+ // Cloudflare Worker
326
+ export default {
327
+ async fetch(request, env) {
328
+ const profileUrl = 'https://profiles.ucp.tools/${merchantId}/ucp.json';
329
+
330
+ const response = await fetch(profileUrl, {
331
+ cf: { cacheTtl: 3600, cacheEverything: true }
332
+ });
333
+
334
+ return new Response(response.body, {
335
+ headers: {
336
+ 'Content-Type': 'application/json',
337
+ 'Access-Control-Allow-Origin': '*',
338
+ 'Cache-Control': 'public, max-age=3600'
339
+ }
340
+ });
341
+ }
342
+ };
343
+ \`\`\`
344
+
345
+ Route: \`${merchantDomain}/.well-known/ucp*\`
346
+
347
+ ### Option 3: Reverse Proxy (Nginx)
348
+
349
+ \`\`\`nginx
350
+ location = /.well-known/ucp {
351
+ proxy_pass https://profiles.ucp.tools/${merchantId}/ucp.json;
352
+ proxy_set_header Host profiles.ucp.tools;
353
+ proxy_cache_valid 200 1h;
354
+ add_header Access-Control-Allow-Origin "*";
355
+ }
356
+ \`\`\`
357
+
358
+ ## Verification
359
+
360
+ After installation, verify your profile:
361
+
362
+ 1. Open: https://${merchantDomain}/.well-known/ucp
363
+ 2. Confirm JSON response with correct structure
364
+ 3. Run validation: \`ucp-validate --remote https://${merchantDomain}\`
365
+
366
+ ## Support
367
+
368
+ For help, visit: https://ucp.tools
369
+ `;
370
+ }
371
+
372
+ /**
373
+ * Generate a minimal starter profile (for quick setup)
374
+ */
375
+ export function generateMinimalProfile(
376
+ endpoint: string,
377
+ ucpVersion: string = CURRENT_UCP_VERSION
378
+ ): UcpProfile {
379
+ return {
380
+ ucp: {
381
+ version: ucpVersion,
382
+ services: {
383
+ [KNOWN_SERVICES.SHOPPING]: {
384
+ version: ucpVersion,
385
+ spec: `${UCP_DEFAULTS.SPEC_BASE}overview/`,
386
+ rest: {
387
+ schema: UCP_DEFAULTS.SHOPPING_REST_SCHEMA,
388
+ endpoint: normalizeEndpoint(endpoint),
389
+ },
390
+ },
391
+ },
392
+ capabilities: [
393
+ {
394
+ name: KNOWN_CAPABILITIES.CHECKOUT,
395
+ version: ucpVersion,
396
+ spec: UCP_DEFAULTS.CHECKOUT_SPEC,
397
+ schema: UCP_DEFAULTS.CHECKOUT_SCHEMA,
398
+ },
399
+ ],
400
+ },
401
+ };
402
+ }