@originals/sdk 1.8.1 → 1.8.2

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 (145) hide show
  1. package/dist/utils/hash.js +1 -0
  2. package/package.json +6 -5
  3. package/src/adapters/FeeOracleMock.ts +9 -0
  4. package/src/adapters/index.ts +5 -0
  5. package/src/adapters/providers/OrdHttpProvider.ts +126 -0
  6. package/src/adapters/providers/OrdMockProvider.ts +101 -0
  7. package/src/adapters/types.ts +66 -0
  8. package/src/bitcoin/BitcoinManager.ts +329 -0
  9. package/src/bitcoin/BroadcastClient.ts +54 -0
  10. package/src/bitcoin/OrdinalsClient.ts +120 -0
  11. package/src/bitcoin/PSBTBuilder.ts +106 -0
  12. package/src/bitcoin/fee-calculation.ts +38 -0
  13. package/src/bitcoin/providers/OrdNodeProvider.ts +92 -0
  14. package/src/bitcoin/providers/OrdinalsProvider.ts +56 -0
  15. package/src/bitcoin/providers/types.ts +59 -0
  16. package/src/bitcoin/transactions/commit.ts +465 -0
  17. package/src/bitcoin/transactions/index.ts +13 -0
  18. package/src/bitcoin/transfer.ts +43 -0
  19. package/src/bitcoin/utxo-selection.ts +322 -0
  20. package/src/bitcoin/utxo.ts +113 -0
  21. package/src/cel/ExternalReferenceManager.ts +87 -0
  22. package/src/cel/OriginalsCel.ts +460 -0
  23. package/src/cel/algorithms/createEventLog.ts +68 -0
  24. package/src/cel/algorithms/deactivateEventLog.ts +109 -0
  25. package/src/cel/algorithms/index.ts +11 -0
  26. package/src/cel/algorithms/updateEventLog.ts +99 -0
  27. package/src/cel/algorithms/verifyEventLog.ts +306 -0
  28. package/src/cel/algorithms/witnessEvent.ts +87 -0
  29. package/src/cel/cli/create.ts +330 -0
  30. package/src/cel/cli/index.ts +383 -0
  31. package/src/cel/cli/inspect.ts +549 -0
  32. package/src/cel/cli/migrate.ts +473 -0
  33. package/src/cel/cli/verify.ts +249 -0
  34. package/src/cel/hash.ts +71 -0
  35. package/src/cel/index.ts +16 -0
  36. package/src/cel/layers/BtcoCelManager.ts +408 -0
  37. package/src/cel/layers/PeerCelManager.ts +371 -0
  38. package/src/cel/layers/WebVHCelManager.ts +361 -0
  39. package/src/cel/layers/index.ts +27 -0
  40. package/src/cel/serialization/cbor.ts +189 -0
  41. package/src/cel/serialization/index.ts +10 -0
  42. package/src/cel/serialization/json.ts +209 -0
  43. package/src/cel/types.ts +160 -0
  44. package/src/cel/witnesses/BitcoinWitness.ts +184 -0
  45. package/src/cel/witnesses/HttpWitness.ts +241 -0
  46. package/src/cel/witnesses/WitnessService.ts +51 -0
  47. package/src/cel/witnesses/index.ts +11 -0
  48. package/src/contexts/credentials-v1.json +237 -0
  49. package/src/contexts/credentials-v2-examples.json +5 -0
  50. package/src/contexts/credentials-v2.json +340 -0
  51. package/src/contexts/credentials.json +237 -0
  52. package/src/contexts/data-integrity-v2.json +81 -0
  53. package/src/contexts/dids.json +58 -0
  54. package/src/contexts/ed255192020.json +93 -0
  55. package/src/contexts/ordinals-plus.json +23 -0
  56. package/src/contexts/originals.json +22 -0
  57. package/src/core/OriginalsSDK.ts +420 -0
  58. package/src/crypto/Multikey.ts +194 -0
  59. package/src/crypto/Signer.ts +262 -0
  60. package/src/crypto/noble-init.ts +138 -0
  61. package/src/did/BtcoDidResolver.ts +231 -0
  62. package/src/did/DIDManager.ts +705 -0
  63. package/src/did/Ed25519Verifier.ts +68 -0
  64. package/src/did/KeyManager.ts +239 -0
  65. package/src/did/WebVHManager.ts +499 -0
  66. package/src/did/createBtcoDidDocument.ts +60 -0
  67. package/src/did/providers/OrdinalsClientProviderAdapter.ts +68 -0
  68. package/src/events/EventEmitter.ts +222 -0
  69. package/src/events/index.ts +19 -0
  70. package/src/events/types.ts +331 -0
  71. package/src/examples/basic-usage.ts +78 -0
  72. package/src/examples/create-module-original.ts +435 -0
  73. package/src/examples/full-lifecycle-flow.ts +514 -0
  74. package/src/examples/run.ts +60 -0
  75. package/src/index.ts +204 -0
  76. package/src/kinds/KindRegistry.ts +320 -0
  77. package/src/kinds/index.ts +74 -0
  78. package/src/kinds/types.ts +470 -0
  79. package/src/kinds/validators/AgentValidator.ts +257 -0
  80. package/src/kinds/validators/AppValidator.ts +211 -0
  81. package/src/kinds/validators/DatasetValidator.ts +242 -0
  82. package/src/kinds/validators/DocumentValidator.ts +311 -0
  83. package/src/kinds/validators/MediaValidator.ts +269 -0
  84. package/src/kinds/validators/ModuleValidator.ts +225 -0
  85. package/src/kinds/validators/base.ts +276 -0
  86. package/src/kinds/validators/index.ts +12 -0
  87. package/src/lifecycle/BatchOperations.ts +381 -0
  88. package/src/lifecycle/LifecycleManager.ts +2156 -0
  89. package/src/lifecycle/OriginalsAsset.ts +524 -0
  90. package/src/lifecycle/ProvenanceQuery.ts +280 -0
  91. package/src/lifecycle/ResourceVersioning.ts +163 -0
  92. package/src/migration/MigrationManager.ts +587 -0
  93. package/src/migration/audit/AuditLogger.ts +176 -0
  94. package/src/migration/checkpoint/CheckpointManager.ts +112 -0
  95. package/src/migration/checkpoint/CheckpointStorage.ts +101 -0
  96. package/src/migration/index.ts +33 -0
  97. package/src/migration/operations/BaseMigration.ts +126 -0
  98. package/src/migration/operations/PeerToBtcoMigration.ts +105 -0
  99. package/src/migration/operations/PeerToWebvhMigration.ts +62 -0
  100. package/src/migration/operations/WebvhToBtcoMigration.ts +105 -0
  101. package/src/migration/rollback/RollbackManager.ts +170 -0
  102. package/src/migration/state/StateMachine.ts +92 -0
  103. package/src/migration/state/StateTracker.ts +156 -0
  104. package/src/migration/types.ts +356 -0
  105. package/src/migration/validation/BitcoinValidator.ts +107 -0
  106. package/src/migration/validation/CredentialValidator.ts +62 -0
  107. package/src/migration/validation/DIDCompatibilityValidator.ts +151 -0
  108. package/src/migration/validation/LifecycleValidator.ts +64 -0
  109. package/src/migration/validation/StorageValidator.ts +79 -0
  110. package/src/migration/validation/ValidationPipeline.ts +213 -0
  111. package/src/resources/ResourceManager.ts +655 -0
  112. package/src/resources/index.ts +21 -0
  113. package/src/resources/types.ts +202 -0
  114. package/src/storage/LocalStorageAdapter.ts +64 -0
  115. package/src/storage/MemoryStorageAdapter.ts +29 -0
  116. package/src/storage/StorageAdapter.ts +25 -0
  117. package/src/storage/index.ts +3 -0
  118. package/src/types/bitcoin.ts +98 -0
  119. package/src/types/common.ts +92 -0
  120. package/src/types/credentials.ts +89 -0
  121. package/src/types/did.ts +31 -0
  122. package/src/types/external-shims.d.ts +53 -0
  123. package/src/types/index.ts +7 -0
  124. package/src/types/network.ts +178 -0
  125. package/src/utils/EventLogger.ts +298 -0
  126. package/src/utils/Logger.ts +324 -0
  127. package/src/utils/MetricsCollector.ts +358 -0
  128. package/src/utils/bitcoin-address.ts +132 -0
  129. package/src/utils/cbor.ts +31 -0
  130. package/src/utils/encoding.ts +135 -0
  131. package/src/utils/hash.ts +12 -0
  132. package/src/utils/retry.ts +46 -0
  133. package/src/utils/satoshi-validation.ts +196 -0
  134. package/src/utils/serialization.ts +102 -0
  135. package/src/utils/telemetry.ts +44 -0
  136. package/src/utils/validation.ts +123 -0
  137. package/src/vc/CredentialManager.ts +955 -0
  138. package/src/vc/Issuer.ts +105 -0
  139. package/src/vc/Verifier.ts +54 -0
  140. package/src/vc/cryptosuites/bbs.ts +253 -0
  141. package/src/vc/cryptosuites/bbsSimple.ts +21 -0
  142. package/src/vc/cryptosuites/eddsa.ts +99 -0
  143. package/src/vc/documentLoader.ts +81 -0
  144. package/src/vc/proofs/data-integrity.ts +33 -0
  145. package/src/vc/utils/jsonld.ts +18 -0
@@ -0,0 +1,470 @@
1
+ /**
2
+ * Originals Kind System Types
3
+ *
4
+ * Defines the typed "Kinds" for Originals - a classification system that
5
+ * enables kind-specific validation, metadata schemas, and behaviors.
6
+ */
7
+
8
+ import type { AssetResource } from '../types/common';
9
+
10
+ /**
11
+ * The supported Original kinds
12
+ * Each kind represents a category of digital asset with specific requirements
13
+ */
14
+ export enum OriginalKind {
15
+ /** Executable application with runtime and entrypoint */
16
+ App = 'originals:kind:app',
17
+
18
+ /** AI agent or autonomous system with capabilities and model info */
19
+ Agent = 'originals:kind:agent',
20
+
21
+ /** Reusable code module with exports and dependencies */
22
+ Module = 'originals:kind:module',
23
+
24
+ /** Structured data collection with schema definition */
25
+ Dataset = 'originals:kind:dataset',
26
+
27
+ /** Media content (image, audio, video) with format metadata */
28
+ Media = 'originals:kind:media',
29
+
30
+ /** Text document with formatting and sections */
31
+ Document = 'originals:kind:document',
32
+ }
33
+
34
+ /**
35
+ * Reference to a dependency Original
36
+ */
37
+ export interface DependencyRef {
38
+ /** DID of the dependency Original */
39
+ did: string;
40
+
41
+ /** Semantic version constraint (e.g., "^1.0.0", ">=2.0.0") */
42
+ version?: string;
43
+
44
+ /** Human-readable name of the dependency */
45
+ name?: string;
46
+
47
+ /** Whether this dependency is required (default: true) */
48
+ optional?: boolean;
49
+ }
50
+
51
+ /**
52
+ * Base manifest fields shared by all kinds
53
+ */
54
+ export interface BaseManifest {
55
+ /** Human-readable name */
56
+ name: string;
57
+
58
+ /** Semantic version string (e.g., "1.0.0") */
59
+ version: string;
60
+
61
+ /** Optional description */
62
+ description?: string;
63
+
64
+ /** Resources associated with this Original */
65
+ resources: AssetResource[];
66
+
67
+ /** Dependencies on other Originals */
68
+ dependencies?: DependencyRef[];
69
+
70
+ /** Free-form tags for categorization */
71
+ tags?: string[];
72
+
73
+ /** Author information */
74
+ author?: {
75
+ name?: string;
76
+ did?: string;
77
+ email?: string;
78
+ url?: string;
79
+ };
80
+
81
+ /** License identifier (SPDX) */
82
+ license?: string;
83
+
84
+ /** URL for more information */
85
+ homepage?: string;
86
+
87
+ /** Repository URL */
88
+ repository?: string;
89
+ }
90
+
91
+ /**
92
+ * App-specific metadata
93
+ */
94
+ export interface AppMetadata {
95
+ /** Runtime environment (e.g., "node", "browser", "deno", "bun") */
96
+ runtime: string;
97
+
98
+ /** Entrypoint resource ID or path */
99
+ entrypoint: string;
100
+
101
+ /** Required runtime version */
102
+ runtimeVersion?: string;
103
+
104
+ /** Minimum required runtime version */
105
+ minRuntimeVersion?: string;
106
+
107
+ /** App permissions required */
108
+ permissions?: string[];
109
+
110
+ /** Environment variables expected */
111
+ env?: Record<string, {
112
+ description?: string;
113
+ required?: boolean;
114
+ default?: string;
115
+ }>;
116
+
117
+ /** Supported platforms */
118
+ platforms?: ('linux' | 'darwin' | 'windows' | 'web')[];
119
+
120
+ /** App icons by size */
121
+ icons?: Record<string, string>;
122
+
123
+ /** CLI commands exposed */
124
+ commands?: Record<string, {
125
+ description: string;
126
+ args?: string[];
127
+ }>;
128
+ }
129
+
130
+ /**
131
+ * Agent-specific metadata
132
+ */
133
+ export interface AgentMetadata {
134
+ /** Agent capabilities/skills */
135
+ capabilities: string[];
136
+
137
+ /** Model information if AI-based */
138
+ model?: {
139
+ provider?: string;
140
+ name: string;
141
+ version?: string;
142
+ parameters?: Record<string, unknown>;
143
+ };
144
+
145
+ /** Input types the agent accepts */
146
+ inputTypes?: string[];
147
+
148
+ /** Output types the agent produces */
149
+ outputTypes?: string[];
150
+
151
+ /** Memory/state configuration */
152
+ memory?: {
153
+ type: 'stateless' | 'session' | 'persistent';
154
+ maxSize?: number;
155
+ };
156
+
157
+ /** System prompt or instructions */
158
+ systemPrompt?: string;
159
+
160
+ /** Tools/functions the agent can use */
161
+ tools?: Array<{
162
+ name: string;
163
+ description: string;
164
+ parameters?: Record<string, unknown>;
165
+ }>;
166
+
167
+ /** Rate limiting configuration */
168
+ rateLimit?: {
169
+ requestsPerMinute?: number;
170
+ tokensPerMinute?: number;
171
+ };
172
+ }
173
+
174
+ /**
175
+ * Module-specific metadata
176
+ */
177
+ export interface ModuleMetadata {
178
+ /** Module format (esm, commonjs, umd, etc.) */
179
+ format: 'esm' | 'commonjs' | 'umd' | 'amd' | 'iife';
180
+
181
+ /** Main entrypoint file (resource ID) */
182
+ main: string;
183
+
184
+ /** Types definition file (resource ID) */
185
+ types?: string;
186
+
187
+ /** Exported symbols/functions */
188
+ exports?: Record<string, string | { import?: string; require?: string; types?: string }>;
189
+
190
+ /** Peer dependencies (expected to be provided by consumer) */
191
+ peerDependencies?: Record<string, string>;
192
+
193
+ /** Browser-specific entrypoint */
194
+ browser?: string;
195
+
196
+ /** Files included in the module */
197
+ files?: string[];
198
+
199
+ /** Side effects declaration */
200
+ sideEffects?: boolean | string[];
201
+
202
+ /** TypeScript configuration */
203
+ typescript?: {
204
+ strict?: boolean;
205
+ target?: string;
206
+ moduleResolution?: string;
207
+ };
208
+ }
209
+
210
+ /**
211
+ * Dataset-specific metadata
212
+ */
213
+ export interface DatasetMetadata {
214
+ /** Schema definition (JSON Schema, or URL to schema) */
215
+ schema: Record<string, unknown> | string;
216
+
217
+ /** Data format (csv, json, parquet, etc.) */
218
+ format: string;
219
+
220
+ /** Number of records/rows */
221
+ recordCount?: number;
222
+
223
+ /** Data columns/fields */
224
+ columns?: Array<{
225
+ name: string;
226
+ type: string;
227
+ description?: string;
228
+ nullable?: boolean;
229
+ }>;
230
+
231
+ /** Data source information */
232
+ source?: {
233
+ origin?: string;
234
+ collectedAt?: string;
235
+ methodology?: string;
236
+ };
237
+
238
+ /** Dataset statistics */
239
+ statistics?: {
240
+ sizeBytes?: number;
241
+ compression?: string;
242
+ checksums?: Record<string, string>;
243
+ };
244
+
245
+ /** License specific to the data */
246
+ dataLicense?: string;
247
+
248
+ /** Privacy classification */
249
+ privacy?: 'public' | 'internal' | 'confidential' | 'restricted';
250
+
251
+ /** Update frequency */
252
+ updateFrequency?: 'realtime' | 'hourly' | 'daily' | 'weekly' | 'monthly' | 'static';
253
+ }
254
+
255
+ /**
256
+ * Media-specific metadata
257
+ */
258
+ export interface MediaMetadata {
259
+ /** Media type (image, audio, video, 3d) */
260
+ mediaType: 'image' | 'audio' | 'video' | '3d' | 'animation';
261
+
262
+ /** MIME type of the primary media resource */
263
+ mimeType: string;
264
+
265
+ /** Dimensions (for images and video) */
266
+ dimensions?: {
267
+ width: number;
268
+ height: number;
269
+ aspectRatio?: string;
270
+ };
271
+
272
+ /** Duration in seconds (for audio/video) */
273
+ duration?: number;
274
+
275
+ /** Frame rate (for video/animation) */
276
+ frameRate?: number;
277
+
278
+ /** Audio channels (for audio/video) */
279
+ audioChannels?: number;
280
+
281
+ /** Sample rate (for audio) */
282
+ sampleRate?: number;
283
+
284
+ /** Codec information */
285
+ codec?: string;
286
+
287
+ /** Bitrate in kbps */
288
+ bitrate?: number;
289
+
290
+ /** Color space */
291
+ colorSpace?: string;
292
+
293
+ /** Thumbnail resource ID */
294
+ thumbnail?: string;
295
+
296
+ /** Preview/low-res version resource ID */
297
+ preview?: string;
298
+
299
+ /** Alt text for accessibility */
300
+ altText?: string;
301
+
302
+ /** Caption or transcript */
303
+ caption?: string;
304
+
305
+ /** EXIF or similar metadata */
306
+ exif?: Record<string, unknown>;
307
+ }
308
+
309
+ /**
310
+ * Document-specific metadata
311
+ */
312
+ export interface DocumentMetadata {
313
+ /** Document format (markdown, html, pdf, docx, etc.) */
314
+ format: 'markdown' | 'html' | 'pdf' | 'docx' | 'txt' | 'asciidoc' | 'rst' | 'latex';
315
+
316
+ /** Document language (ISO 639-1) */
317
+ language?: string;
318
+
319
+ /** Main content resource ID */
320
+ content: string;
321
+
322
+ /** Table of contents */
323
+ toc?: Array<{
324
+ title: string;
325
+ level: number;
326
+ anchor?: string;
327
+ }>;
328
+
329
+ /** Page count */
330
+ pageCount?: number;
331
+
332
+ /** Word count */
333
+ wordCount?: number;
334
+
335
+ /** Reading time in minutes */
336
+ readingTime?: number;
337
+
338
+ /** Keywords */
339
+ keywords?: string[];
340
+
341
+ /** Abstract or summary */
342
+ abstract?: string;
343
+
344
+ /** Bibliography/references */
345
+ references?: Array<{
346
+ id: string;
347
+ title: string;
348
+ authors?: string[];
349
+ year?: number;
350
+ url?: string;
351
+ doi?: string;
352
+ }>;
353
+
354
+ /** Document status */
355
+ status?: 'draft' | 'review' | 'published' | 'archived';
356
+
357
+ /** Revision number */
358
+ revision?: number;
359
+ }
360
+
361
+ /**
362
+ * Maps each kind to its specific metadata type
363
+ */
364
+ export interface KindMetadataMap {
365
+ [OriginalKind.App]: AppMetadata;
366
+ [OriginalKind.Agent]: AgentMetadata;
367
+ [OriginalKind.Module]: ModuleMetadata;
368
+ [OriginalKind.Dataset]: DatasetMetadata;
369
+ [OriginalKind.Media]: MediaMetadata;
370
+ [OriginalKind.Document]: DocumentMetadata;
371
+ }
372
+
373
+ /**
374
+ * Type-safe metadata accessor
375
+ */
376
+ export type KindMetadata<K extends OriginalKind> = K extends keyof KindMetadataMap
377
+ ? KindMetadataMap[K]
378
+ : never;
379
+
380
+ /**
381
+ * Generic Original manifest with kind-specific metadata
382
+ */
383
+ export interface OriginalManifest<K extends OriginalKind = OriginalKind> extends BaseManifest {
384
+ /** The kind of Original */
385
+ kind: K;
386
+
387
+ /** Kind-specific metadata */
388
+ metadata: KindMetadata<K>;
389
+ }
390
+
391
+ /**
392
+ * Type-specific manifest aliases for convenience
393
+ */
394
+ export type AppManifest = OriginalManifest<OriginalKind.App>;
395
+ export type AgentManifest = OriginalManifest<OriginalKind.Agent>;
396
+ export type ModuleManifest = OriginalManifest<OriginalKind.Module>;
397
+ export type DatasetManifest = OriginalManifest<OriginalKind.Dataset>;
398
+ export type MediaManifest = OriginalManifest<OriginalKind.Media>;
399
+ export type DocumentManifest = OriginalManifest<OriginalKind.Document>;
400
+
401
+ /**
402
+ * Union type of all possible manifests
403
+ */
404
+ export type AnyManifest =
405
+ | AppManifest
406
+ | AgentManifest
407
+ | ModuleManifest
408
+ | DatasetManifest
409
+ | MediaManifest
410
+ | DocumentManifest;
411
+
412
+ /**
413
+ * Result of validating a manifest
414
+ */
415
+ export interface ValidationResult {
416
+ /** Whether the manifest is valid */
417
+ isValid: boolean;
418
+
419
+ /** Validation errors if any */
420
+ errors: ValidationError[];
421
+
422
+ /** Validation warnings (non-fatal) */
423
+ warnings: ValidationWarning[];
424
+ }
425
+
426
+ /**
427
+ * A validation error
428
+ */
429
+ export interface ValidationError {
430
+ /** Error code */
431
+ code: string;
432
+
433
+ /** Human-readable message */
434
+ message: string;
435
+
436
+ /** Path to the invalid field (e.g., "metadata.entrypoint") */
437
+ path?: string;
438
+
439
+ /** The invalid value */
440
+ value?: unknown;
441
+ }
442
+
443
+ /**
444
+ * A validation warning (non-fatal issue)
445
+ */
446
+ export interface ValidationWarning {
447
+ /** Warning code */
448
+ code: string;
449
+
450
+ /** Human-readable message */
451
+ message: string;
452
+
453
+ /** Path to the field with warning */
454
+ path?: string;
455
+
456
+ /** Suggested fix */
457
+ suggestion?: string;
458
+ }
459
+
460
+ /**
461
+ * Options for creating a typed Original
462
+ */
463
+ export interface CreateTypedOriginalOptions {
464
+ /** Skip validation (not recommended) */
465
+ skipValidation?: boolean;
466
+
467
+ /** Treat warnings as errors */
468
+ strictMode?: boolean;
469
+ }
470
+
@@ -0,0 +1,257 @@
1
+ /**
2
+ * Agent Kind Validator
3
+ *
4
+ * Validates manifests for AI agents or autonomous systems with capabilities and model info.
5
+ */
6
+
7
+ import { OriginalKind, type OriginalManifest, type ValidationResult, type AgentMetadata } from '../types';
8
+ import { BaseKindValidator, ValidationUtils } from './base';
9
+
10
+ /**
11
+ * Known AI model providers
12
+ */
13
+ const KNOWN_PROVIDERS = [
14
+ 'openai', 'anthropic', 'google', 'meta', 'mistral', 'cohere',
15
+ 'huggingface', 'replicate', 'together', 'groq', 'perplexity',
16
+ 'fireworks', 'local', 'custom',
17
+ ];
18
+
19
+ /**
20
+ * Valid memory types
21
+ */
22
+ const VALID_MEMORY_TYPES = ['stateless', 'session', 'persistent'];
23
+
24
+ /**
25
+ * Validator for Agent Originals
26
+ */
27
+ export class AgentValidator extends BaseKindValidator<OriginalKind.Agent> {
28
+ readonly kind = OriginalKind.Agent;
29
+
30
+ protected validateKind(manifest: OriginalManifest<OriginalKind.Agent>): ValidationResult {
31
+ const errors: ValidationResult['errors'] = [];
32
+ const warnings: ValidationResult['warnings'] = [];
33
+ const metadata = manifest.metadata as AgentMetadata;
34
+
35
+ // Validate metadata exists
36
+ if (!metadata || typeof metadata !== 'object') {
37
+ return ValidationUtils.failure([
38
+ ValidationUtils.error('MISSING_METADATA', 'Agent manifest must have metadata', 'metadata'),
39
+ ]);
40
+ }
41
+
42
+ // Validate capabilities (required)
43
+ if (!metadata.capabilities) {
44
+ errors.push(ValidationUtils.error(
45
+ 'MISSING_CAPABILITIES',
46
+ 'Agent must specify capabilities',
47
+ 'metadata.capabilities',
48
+ ));
49
+ } else if (!Array.isArray(metadata.capabilities)) {
50
+ errors.push(ValidationUtils.error(
51
+ 'INVALID_CAPABILITIES',
52
+ 'Capabilities must be an array of strings',
53
+ 'metadata.capabilities',
54
+ ));
55
+ } else if (metadata.capabilities.length === 0) {
56
+ errors.push(ValidationUtils.error(
57
+ 'EMPTY_CAPABILITIES',
58
+ 'Agent must have at least one capability',
59
+ 'metadata.capabilities',
60
+ ));
61
+ } else {
62
+ for (let i = 0; i < metadata.capabilities.length; i++) {
63
+ if (typeof metadata.capabilities[i] !== 'string') {
64
+ errors.push(ValidationUtils.error(
65
+ 'INVALID_CAPABILITY',
66
+ `Capability at index ${i} must be a string`,
67
+ `metadata.capabilities[${i}]`,
68
+ ));
69
+ } else if (metadata.capabilities[i].length === 0) {
70
+ errors.push(ValidationUtils.error(
71
+ 'EMPTY_CAPABILITY',
72
+ `Capability at index ${i} cannot be empty`,
73
+ `metadata.capabilities[${i}]`,
74
+ ));
75
+ }
76
+ }
77
+ }
78
+
79
+ // Validate model if specified
80
+ if (metadata.model) {
81
+ if (typeof metadata.model !== 'object') {
82
+ errors.push(ValidationUtils.error(
83
+ 'INVALID_MODEL',
84
+ 'Model must be an object',
85
+ 'metadata.model',
86
+ ));
87
+ } else {
88
+ // Model name is required
89
+ if (!metadata.model.name || typeof metadata.model.name !== 'string') {
90
+ errors.push(ValidationUtils.error(
91
+ 'MISSING_MODEL_NAME',
92
+ 'Model must have a name',
93
+ 'metadata.model.name',
94
+ ));
95
+ }
96
+
97
+ // Validate provider if specified
98
+ if (metadata.model.provider) {
99
+ const normalizedProvider = metadata.model.provider.toLowerCase();
100
+ if (!KNOWN_PROVIDERS.includes(normalizedProvider)) {
101
+ warnings.push(ValidationUtils.warning(
102
+ 'UNKNOWN_PROVIDER',
103
+ `Model provider "${metadata.model.provider}" is not a commonly recognized provider`,
104
+ 'metadata.model.provider',
105
+ `Consider using one of: ${KNOWN_PROVIDERS.join(', ')}`,
106
+ ));
107
+ }
108
+ }
109
+ }
110
+ }
111
+
112
+ // Validate input/output types if specified
113
+ if (metadata.inputTypes) {
114
+ if (!Array.isArray(metadata.inputTypes)) {
115
+ errors.push(ValidationUtils.error(
116
+ 'INVALID_INPUT_TYPES',
117
+ 'Input types must be an array',
118
+ 'metadata.inputTypes',
119
+ ));
120
+ }
121
+ }
122
+
123
+ if (metadata.outputTypes) {
124
+ if (!Array.isArray(metadata.outputTypes)) {
125
+ errors.push(ValidationUtils.error(
126
+ 'INVALID_OUTPUT_TYPES',
127
+ 'Output types must be an array',
128
+ 'metadata.outputTypes',
129
+ ));
130
+ }
131
+ }
132
+
133
+ // Validate memory if specified
134
+ if (metadata.memory) {
135
+ if (typeof metadata.memory !== 'object') {
136
+ errors.push(ValidationUtils.error(
137
+ 'INVALID_MEMORY',
138
+ 'Memory must be an object',
139
+ 'metadata.memory',
140
+ ));
141
+ } else {
142
+ if (!metadata.memory.type || !VALID_MEMORY_TYPES.includes(metadata.memory.type)) {
143
+ errors.push(ValidationUtils.error(
144
+ 'INVALID_MEMORY_TYPE',
145
+ `Memory type must be one of: ${VALID_MEMORY_TYPES.join(', ')}`,
146
+ 'metadata.memory.type',
147
+ metadata.memory.type,
148
+ ));
149
+ }
150
+
151
+ if (metadata.memory.maxSize !== undefined &&
152
+ (typeof metadata.memory.maxSize !== 'number' || metadata.memory.maxSize <= 0)) {
153
+ errors.push(ValidationUtils.error(
154
+ 'INVALID_MEMORY_SIZE',
155
+ 'Memory maxSize must be a positive number',
156
+ 'metadata.memory.maxSize',
157
+ ));
158
+ }
159
+ }
160
+ }
161
+
162
+ // Validate tools if specified
163
+ if (metadata.tools) {
164
+ if (!Array.isArray(metadata.tools)) {
165
+ errors.push(ValidationUtils.error(
166
+ 'INVALID_TOOLS',
167
+ 'Tools must be an array',
168
+ 'metadata.tools',
169
+ ));
170
+ } else {
171
+ for (let i = 0; i < metadata.tools.length; i++) {
172
+ const tool = metadata.tools[i];
173
+ const toolPath = `metadata.tools[${i}]`;
174
+
175
+ if (!tool || typeof tool !== 'object') {
176
+ errors.push(ValidationUtils.error(
177
+ 'INVALID_TOOL',
178
+ `Tool at index ${i} must be an object`,
179
+ toolPath,
180
+ ));
181
+ continue;
182
+ }
183
+
184
+ if (!tool.name || typeof tool.name !== 'string') {
185
+ errors.push(ValidationUtils.error(
186
+ 'MISSING_TOOL_NAME',
187
+ `Tool at index ${i} must have a name`,
188
+ `${toolPath}.name`,
189
+ ));
190
+ }
191
+
192
+ if (!tool.description || typeof tool.description !== 'string') {
193
+ warnings.push(ValidationUtils.warning(
194
+ 'MISSING_TOOL_DESC',
195
+ `Tool "${tool.name || i}" should have a description`,
196
+ `${toolPath}.description`,
197
+ ));
198
+ }
199
+ }
200
+ }
201
+ }
202
+
203
+ // Validate rate limits if specified
204
+ if (metadata.rateLimit) {
205
+ if (typeof metadata.rateLimit !== 'object') {
206
+ errors.push(ValidationUtils.error(
207
+ 'INVALID_RATE_LIMIT',
208
+ 'Rate limit must be an object',
209
+ 'metadata.rateLimit',
210
+ ));
211
+ } else {
212
+ if (metadata.rateLimit.requestsPerMinute !== undefined &&
213
+ (typeof metadata.rateLimit.requestsPerMinute !== 'number' ||
214
+ metadata.rateLimit.requestsPerMinute <= 0)) {
215
+ errors.push(ValidationUtils.error(
216
+ 'INVALID_RPM',
217
+ 'requestsPerMinute must be a positive number',
218
+ 'metadata.rateLimit.requestsPerMinute',
219
+ ));
220
+ }
221
+
222
+ if (metadata.rateLimit.tokensPerMinute !== undefined &&
223
+ (typeof metadata.rateLimit.tokensPerMinute !== 'number' ||
224
+ metadata.rateLimit.tokensPerMinute <= 0)) {
225
+ errors.push(ValidationUtils.error(
226
+ 'INVALID_TPM',
227
+ 'tokensPerMinute must be a positive number',
228
+ 'metadata.rateLimit.tokensPerMinute',
229
+ ));
230
+ }
231
+ }
232
+ }
233
+
234
+ // Suggest adding model info for AI agents
235
+ if (!metadata.model) {
236
+ warnings.push(ValidationUtils.warning(
237
+ 'MISSING_MODEL',
238
+ 'Consider adding model information if this is an AI-based agent',
239
+ 'metadata.model',
240
+ ));
241
+ }
242
+
243
+ // Suggest adding system prompt
244
+ if (!metadata.systemPrompt) {
245
+ warnings.push(ValidationUtils.warning(
246
+ 'MISSING_SYSTEM_PROMPT',
247
+ 'Consider adding a system prompt to define the agent\'s behavior',
248
+ 'metadata.systemPrompt',
249
+ ));
250
+ }
251
+
252
+ return errors.length > 0
253
+ ? ValidationUtils.failure(errors, warnings)
254
+ : ValidationUtils.success(warnings);
255
+ }
256
+ }
257
+