@originals/sdk 1.8.0 → 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,311 @@
1
+ /**
2
+ * Document Kind Validator
3
+ *
4
+ * Validates manifests for text documents with formatting and sections.
5
+ */
6
+
7
+ import { OriginalKind, type OriginalManifest, type ValidationResult, type DocumentMetadata } from '../types';
8
+ import { BaseKindValidator, ValidationUtils } from './base';
9
+
10
+ /**
11
+ * Valid document formats
12
+ */
13
+ const VALID_FORMATS = ['markdown', 'html', 'pdf', 'docx', 'txt', 'asciidoc', 'rst', 'latex'];
14
+
15
+ /**
16
+ * Valid document statuses
17
+ */
18
+ const VALID_STATUSES = ['draft', 'review', 'published', 'archived'];
19
+
20
+ /**
21
+ * Validator for Document Originals
22
+ */
23
+ export class DocumentValidator extends BaseKindValidator<OriginalKind.Document> {
24
+ readonly kind = OriginalKind.Document;
25
+
26
+ protected validateKind(manifest: OriginalManifest<OriginalKind.Document>): ValidationResult {
27
+ const errors: ValidationResult['errors'] = [];
28
+ const warnings: ValidationResult['warnings'] = [];
29
+ const metadata = manifest.metadata as DocumentMetadata;
30
+
31
+ // Validate metadata exists
32
+ if (!metadata || typeof metadata !== 'object') {
33
+ return ValidationUtils.failure([
34
+ ValidationUtils.error('MISSING_METADATA', 'Document manifest must have metadata', 'metadata'),
35
+ ]);
36
+ }
37
+
38
+ // Validate format (required)
39
+ if (!metadata.format) {
40
+ errors.push(ValidationUtils.error(
41
+ 'MISSING_FORMAT',
42
+ 'Document must specify a format',
43
+ 'metadata.format',
44
+ ));
45
+ } else if (!VALID_FORMATS.includes(metadata.format)) {
46
+ errors.push(ValidationUtils.error(
47
+ 'INVALID_FORMAT',
48
+ `Invalid document format: "${metadata.format}". Must be one of: ${VALID_FORMATS.join(', ')}`,
49
+ 'metadata.format',
50
+ metadata.format,
51
+ ));
52
+ }
53
+
54
+ // Validate content (required)
55
+ if (!metadata.content || typeof metadata.content !== 'string') {
56
+ errors.push(ValidationUtils.error(
57
+ 'MISSING_CONTENT',
58
+ 'Document must specify a content resource',
59
+ 'metadata.content',
60
+ ));
61
+ } else {
62
+ // Check if content references an existing resource
63
+ if (!ValidationUtils.resourceExists(metadata.content, manifest.resources)) {
64
+ warnings.push(ValidationUtils.warning(
65
+ 'CONTENT_NOT_RESOURCE',
66
+ `Content "${metadata.content}" does not match a resource ID`,
67
+ 'metadata.content',
68
+ 'Ensure the content field references a valid resource ID',
69
+ ));
70
+ }
71
+ }
72
+
73
+ // Validate language if specified
74
+ if (metadata.language) {
75
+ if (typeof metadata.language !== 'string') {
76
+ errors.push(ValidationUtils.error(
77
+ 'INVALID_LANGUAGE',
78
+ 'Language must be a string',
79
+ 'metadata.language',
80
+ ));
81
+ } else if (!ValidationUtils.isValidLanguageCode(metadata.language)) {
82
+ warnings.push(ValidationUtils.warning(
83
+ 'INVALID_LANGUAGE_CODE',
84
+ `Language "${metadata.language}" is not a valid ISO 639-1 code`,
85
+ 'metadata.language',
86
+ 'Use a 2-letter language code like "en", "es", "fr"',
87
+ ));
88
+ }
89
+ }
90
+
91
+ // Validate toc if specified
92
+ if (metadata.toc) {
93
+ if (!Array.isArray(metadata.toc)) {
94
+ errors.push(ValidationUtils.error(
95
+ 'INVALID_TOC',
96
+ 'Table of contents must be an array',
97
+ 'metadata.toc',
98
+ ));
99
+ } else {
100
+ for (let i = 0; i < metadata.toc.length; i++) {
101
+ const entry = metadata.toc[i];
102
+ const entryPath = `metadata.toc[${i}]`;
103
+
104
+ if (!entry || typeof entry !== 'object') {
105
+ errors.push(ValidationUtils.error(
106
+ 'INVALID_TOC_ENTRY',
107
+ `TOC entry at index ${i} must be an object`,
108
+ entryPath,
109
+ ));
110
+ continue;
111
+ }
112
+
113
+ if (!entry.title || typeof entry.title !== 'string') {
114
+ errors.push(ValidationUtils.error(
115
+ 'MISSING_TOC_TITLE',
116
+ `TOC entry at index ${i} must have a title`,
117
+ `${entryPath}.title`,
118
+ ));
119
+ }
120
+
121
+ if (typeof entry.level !== 'number' || entry.level < 1 || !Number.isInteger(entry.level)) {
122
+ errors.push(ValidationUtils.error(
123
+ 'INVALID_TOC_LEVEL',
124
+ `TOC entry at index ${i} must have a valid level (positive integer)`,
125
+ `${entryPath}.level`,
126
+ ));
127
+ }
128
+ }
129
+ }
130
+ }
131
+
132
+ // Validate pageCount if specified
133
+ if (metadata.pageCount !== undefined) {
134
+ if (typeof metadata.pageCount !== 'number' || metadata.pageCount <= 0 || !Number.isInteger(metadata.pageCount)) {
135
+ errors.push(ValidationUtils.error(
136
+ 'INVALID_PAGE_COUNT',
137
+ 'Page count must be a positive integer',
138
+ 'metadata.pageCount',
139
+ ));
140
+ }
141
+ }
142
+
143
+ // Validate wordCount if specified
144
+ if (metadata.wordCount !== undefined) {
145
+ if (typeof metadata.wordCount !== 'number' || metadata.wordCount < 0 || !Number.isInteger(metadata.wordCount)) {
146
+ errors.push(ValidationUtils.error(
147
+ 'INVALID_WORD_COUNT',
148
+ 'Word count must be a non-negative integer',
149
+ 'metadata.wordCount',
150
+ ));
151
+ }
152
+ }
153
+
154
+ // Validate readingTime if specified
155
+ if (metadata.readingTime !== undefined) {
156
+ if (typeof metadata.readingTime !== 'number' || metadata.readingTime <= 0) {
157
+ errors.push(ValidationUtils.error(
158
+ 'INVALID_READING_TIME',
159
+ 'Reading time must be a positive number (minutes)',
160
+ 'metadata.readingTime',
161
+ ));
162
+ }
163
+ }
164
+
165
+ // Validate keywords if specified
166
+ if (metadata.keywords) {
167
+ if (!Array.isArray(metadata.keywords)) {
168
+ errors.push(ValidationUtils.error(
169
+ 'INVALID_KEYWORDS',
170
+ 'Keywords must be an array of strings',
171
+ 'metadata.keywords',
172
+ ));
173
+ } else {
174
+ for (let i = 0; i < metadata.keywords.length; i++) {
175
+ if (typeof metadata.keywords[i] !== 'string') {
176
+ errors.push(ValidationUtils.error(
177
+ 'INVALID_KEYWORD',
178
+ `Keyword at index ${i} must be a string`,
179
+ `metadata.keywords[${i}]`,
180
+ ));
181
+ }
182
+ }
183
+ }
184
+ }
185
+
186
+ // Validate references if specified
187
+ if (metadata.references) {
188
+ if (!Array.isArray(metadata.references)) {
189
+ errors.push(ValidationUtils.error(
190
+ 'INVALID_REFERENCES',
191
+ 'References must be an array',
192
+ 'metadata.references',
193
+ ));
194
+ } else {
195
+ const refIds = new Set<string>();
196
+
197
+ for (let i = 0; i < metadata.references.length; i++) {
198
+ const ref = metadata.references[i];
199
+ const refPath = `metadata.references[${i}]`;
200
+
201
+ if (!ref || typeof ref !== 'object') {
202
+ errors.push(ValidationUtils.error(
203
+ 'INVALID_REFERENCE',
204
+ `Reference at index ${i} must be an object`,
205
+ refPath,
206
+ ));
207
+ continue;
208
+ }
209
+
210
+ if (!ref.id || typeof ref.id !== 'string') {
211
+ errors.push(ValidationUtils.error(
212
+ 'MISSING_REFERENCE_ID',
213
+ `Reference at index ${i} must have an id`,
214
+ `${refPath}.id`,
215
+ ));
216
+ } else {
217
+ if (refIds.has(ref.id)) {
218
+ errors.push(ValidationUtils.error(
219
+ 'DUPLICATE_REFERENCE_ID',
220
+ `Duplicate reference id: "${ref.id}"`,
221
+ `${refPath}.id`,
222
+ ));
223
+ }
224
+ refIds.add(ref.id);
225
+ }
226
+
227
+ if (!ref.title || typeof ref.title !== 'string') {
228
+ errors.push(ValidationUtils.error(
229
+ 'MISSING_REFERENCE_TITLE',
230
+ `Reference at index ${i} must have a title`,
231
+ `${refPath}.title`,
232
+ ));
233
+ }
234
+
235
+ // Validate URL if present
236
+ if (ref.url && !ValidationUtils.isValidURL(ref.url)) {
237
+ warnings.push(ValidationUtils.warning(
238
+ 'INVALID_REFERENCE_URL',
239
+ `Reference "${ref.id}" has an invalid URL`,
240
+ `${refPath}.url`,
241
+ ));
242
+ }
243
+ }
244
+ }
245
+ }
246
+
247
+ // Validate status if specified
248
+ if (metadata.status) {
249
+ if (!VALID_STATUSES.includes(metadata.status)) {
250
+ errors.push(ValidationUtils.error(
251
+ 'INVALID_STATUS',
252
+ `Invalid document status: "${metadata.status}". Must be one of: ${VALID_STATUSES.join(', ')}`,
253
+ 'metadata.status',
254
+ metadata.status,
255
+ ));
256
+ }
257
+ }
258
+
259
+ // Validate revision if specified
260
+ if (metadata.revision !== undefined) {
261
+ if (typeof metadata.revision !== 'number' || metadata.revision < 1 || !Number.isInteger(metadata.revision)) {
262
+ errors.push(ValidationUtils.error(
263
+ 'INVALID_REVISION',
264
+ 'Revision must be a positive integer',
265
+ 'metadata.revision',
266
+ ));
267
+ }
268
+ }
269
+
270
+ // Suggest adding language
271
+ if (!metadata.language) {
272
+ warnings.push(ValidationUtils.warning(
273
+ 'MISSING_LANGUAGE',
274
+ 'Consider specifying the document language',
275
+ 'metadata.language',
276
+ 'Add a language code like "en" for English',
277
+ ));
278
+ }
279
+
280
+ // Suggest adding abstract
281
+ if (!metadata.abstract) {
282
+ warnings.push(ValidationUtils.warning(
283
+ 'MISSING_ABSTRACT',
284
+ 'Consider adding an abstract or summary',
285
+ 'metadata.abstract',
286
+ ));
287
+ }
288
+
289
+ // Check that at least one document resource exists
290
+ const docResources = manifest.resources.filter(r =>
291
+ r.type === 'document' ||
292
+ r.type === 'text' ||
293
+ r.contentType.includes('text/') ||
294
+ r.contentType.includes('markdown') ||
295
+ r.contentType.includes('html') ||
296
+ r.contentType.includes('pdf')
297
+ );
298
+ if (docResources.length === 0) {
299
+ warnings.push(ValidationUtils.warning(
300
+ 'NO_DOCUMENT_RESOURCES',
301
+ 'No document resources found. Ensure resources have appropriate types',
302
+ 'resources',
303
+ ));
304
+ }
305
+
306
+ return errors.length > 0
307
+ ? ValidationUtils.failure(errors, warnings)
308
+ : ValidationUtils.success(warnings);
309
+ }
310
+ }
311
+
@@ -0,0 +1,269 @@
1
+ /**
2
+ * Media Kind Validator
3
+ *
4
+ * Validates manifests for media content (image, audio, video) with format metadata.
5
+ */
6
+
7
+ import { OriginalKind, type OriginalManifest, type ValidationResult, type MediaMetadata } from '../types';
8
+ import { BaseKindValidator, ValidationUtils } from './base';
9
+
10
+ /**
11
+ * Valid media types
12
+ */
13
+ const VALID_MEDIA_TYPES = ['image', 'audio', 'video', '3d', 'animation'];
14
+
15
+ /**
16
+ * Common MIME types by media type
17
+ */
18
+ const COMMON_MIME_TYPES: Record<string, string[]> = {
19
+ image: [
20
+ 'image/jpeg', 'image/png', 'image/gif', 'image/webp', 'image/svg+xml',
21
+ 'image/avif', 'image/bmp', 'image/tiff', 'image/heic', 'image/heif',
22
+ ],
23
+ audio: [
24
+ 'audio/mpeg', 'audio/mp3', 'audio/wav', 'audio/ogg', 'audio/flac',
25
+ 'audio/aac', 'audio/webm', 'audio/midi', 'audio/x-wav',
26
+ ],
27
+ video: [
28
+ 'video/mp4', 'video/webm', 'video/ogg', 'video/quicktime', 'video/x-msvideo',
29
+ 'video/x-matroska', 'video/mpeg',
30
+ ],
31
+ '3d': [
32
+ 'model/gltf+json', 'model/gltf-binary', 'model/obj', 'model/stl',
33
+ 'application/octet-stream',
34
+ ],
35
+ animation: [
36
+ 'image/gif', 'video/mp4', 'video/webm', 'application/json', // Lottie
37
+ ],
38
+ };
39
+
40
+ /**
41
+ * Validator for Media Originals
42
+ */
43
+ export class MediaValidator extends BaseKindValidator<OriginalKind.Media> {
44
+ readonly kind = OriginalKind.Media;
45
+
46
+ protected validateKind(manifest: OriginalManifest<OriginalKind.Media>): ValidationResult {
47
+ const errors: ValidationResult['errors'] = [];
48
+ const warnings: ValidationResult['warnings'] = [];
49
+ const metadata = manifest.metadata as MediaMetadata;
50
+
51
+ // Validate metadata exists
52
+ if (!metadata || typeof metadata !== 'object') {
53
+ return ValidationUtils.failure([
54
+ ValidationUtils.error('MISSING_METADATA', 'Media manifest must have metadata', 'metadata'),
55
+ ]);
56
+ }
57
+
58
+ // Validate mediaType (required)
59
+ if (!metadata.mediaType) {
60
+ errors.push(ValidationUtils.error(
61
+ 'MISSING_MEDIA_TYPE',
62
+ 'Media must specify a mediaType',
63
+ 'metadata.mediaType',
64
+ ));
65
+ } else if (!VALID_MEDIA_TYPES.includes(metadata.mediaType)) {
66
+ errors.push(ValidationUtils.error(
67
+ 'INVALID_MEDIA_TYPE',
68
+ `Invalid mediaType: "${metadata.mediaType}". Must be one of: ${VALID_MEDIA_TYPES.join(', ')}`,
69
+ 'metadata.mediaType',
70
+ metadata.mediaType,
71
+ ));
72
+ }
73
+
74
+ // Validate mimeType (required)
75
+ if (!metadata.mimeType || typeof metadata.mimeType !== 'string') {
76
+ errors.push(ValidationUtils.error(
77
+ 'MISSING_MIME_TYPE',
78
+ 'Media must specify a mimeType',
79
+ 'metadata.mimeType',
80
+ ));
81
+ } else if (!ValidationUtils.isValidMimeType(metadata.mimeType)) {
82
+ errors.push(ValidationUtils.error(
83
+ 'INVALID_MIME_TYPE',
84
+ `Invalid mimeType format: "${metadata.mimeType}"`,
85
+ 'metadata.mimeType',
86
+ metadata.mimeType,
87
+ ));
88
+ } else if (metadata.mediaType) {
89
+ // Check if MIME type matches the declared media type
90
+ const expectedMimeTypes = COMMON_MIME_TYPES[metadata.mediaType];
91
+ if (expectedMimeTypes && !expectedMimeTypes.includes(metadata.mimeType)) {
92
+ warnings.push(ValidationUtils.warning(
93
+ 'MIME_TYPE_MISMATCH',
94
+ `mimeType "${metadata.mimeType}" is not typically associated with mediaType "${metadata.mediaType}"`,
95
+ 'metadata.mimeType',
96
+ ));
97
+ }
98
+ }
99
+
100
+ // Validate dimensions for images and video
101
+ if (metadata.mediaType === 'image' || metadata.mediaType === 'video') {
102
+ if (metadata.dimensions) {
103
+ if (typeof metadata.dimensions !== 'object') {
104
+ errors.push(ValidationUtils.error(
105
+ 'INVALID_DIMENSIONS',
106
+ 'Dimensions must be an object',
107
+ 'metadata.dimensions',
108
+ ));
109
+ } else {
110
+ if (typeof metadata.dimensions.width !== 'number' || metadata.dimensions.width <= 0) {
111
+ errors.push(ValidationUtils.error(
112
+ 'INVALID_WIDTH',
113
+ 'Width must be a positive number',
114
+ 'metadata.dimensions.width',
115
+ ));
116
+ }
117
+ if (typeof metadata.dimensions.height !== 'number' || metadata.dimensions.height <= 0) {
118
+ errors.push(ValidationUtils.error(
119
+ 'INVALID_HEIGHT',
120
+ 'Height must be a positive number',
121
+ 'metadata.dimensions.height',
122
+ ));
123
+ }
124
+ }
125
+ } else {
126
+ warnings.push(ValidationUtils.warning(
127
+ 'MISSING_DIMENSIONS',
128
+ `Consider specifying dimensions for ${metadata.mediaType} content`,
129
+ 'metadata.dimensions',
130
+ ));
131
+ }
132
+ }
133
+
134
+ // Validate duration for audio and video
135
+ if (metadata.mediaType === 'audio' || metadata.mediaType === 'video') {
136
+ if (metadata.duration !== undefined) {
137
+ if (typeof metadata.duration !== 'number' || metadata.duration < 0) {
138
+ errors.push(ValidationUtils.error(
139
+ 'INVALID_DURATION',
140
+ 'Duration must be a non-negative number (seconds)',
141
+ 'metadata.duration',
142
+ ));
143
+ }
144
+ } else {
145
+ warnings.push(ValidationUtils.warning(
146
+ 'MISSING_DURATION',
147
+ `Consider specifying duration for ${metadata.mediaType} content`,
148
+ 'metadata.duration',
149
+ ));
150
+ }
151
+ }
152
+
153
+ // Validate frameRate for video
154
+ if (metadata.mediaType === 'video' || metadata.mediaType === 'animation') {
155
+ if (metadata.frameRate !== undefined) {
156
+ if (typeof metadata.frameRate !== 'number' || metadata.frameRate <= 0) {
157
+ errors.push(ValidationUtils.error(
158
+ 'INVALID_FRAME_RATE',
159
+ 'Frame rate must be a positive number',
160
+ 'metadata.frameRate',
161
+ ));
162
+ }
163
+ }
164
+ }
165
+
166
+ // Validate audio-specific fields
167
+ if (metadata.mediaType === 'audio' || metadata.mediaType === 'video') {
168
+ if (metadata.audioChannels !== undefined) {
169
+ if (typeof metadata.audioChannels !== 'number' ||
170
+ metadata.audioChannels <= 0 ||
171
+ !Number.isInteger(metadata.audioChannels)) {
172
+ errors.push(ValidationUtils.error(
173
+ 'INVALID_AUDIO_CHANNELS',
174
+ 'Audio channels must be a positive integer',
175
+ 'metadata.audioChannels',
176
+ ));
177
+ }
178
+ }
179
+
180
+ if (metadata.sampleRate !== undefined) {
181
+ if (typeof metadata.sampleRate !== 'number' || metadata.sampleRate <= 0) {
182
+ errors.push(ValidationUtils.error(
183
+ 'INVALID_SAMPLE_RATE',
184
+ 'Sample rate must be a positive number',
185
+ 'metadata.sampleRate',
186
+ ));
187
+ }
188
+ }
189
+ }
190
+
191
+ // Validate bitrate if specified
192
+ if (metadata.bitrate !== undefined) {
193
+ if (typeof metadata.bitrate !== 'number' || metadata.bitrate <= 0) {
194
+ errors.push(ValidationUtils.error(
195
+ 'INVALID_BITRATE',
196
+ 'Bitrate must be a positive number (kbps)',
197
+ 'metadata.bitrate',
198
+ ));
199
+ }
200
+ }
201
+
202
+ // Validate thumbnail if specified
203
+ if (metadata.thumbnail) {
204
+ if (typeof metadata.thumbnail !== 'string') {
205
+ errors.push(ValidationUtils.error(
206
+ 'INVALID_THUMBNAIL',
207
+ 'Thumbnail must be a string (resource ID)',
208
+ 'metadata.thumbnail',
209
+ ));
210
+ } else if (!ValidationUtils.resourceExists(metadata.thumbnail, manifest.resources)) {
211
+ warnings.push(ValidationUtils.warning(
212
+ 'THUMBNAIL_NOT_RESOURCE',
213
+ `Thumbnail "${metadata.thumbnail}" does not match a resource ID`,
214
+ 'metadata.thumbnail',
215
+ ));
216
+ }
217
+ }
218
+
219
+ // Validate preview if specified
220
+ if (metadata.preview) {
221
+ if (typeof metadata.preview !== 'string') {
222
+ errors.push(ValidationUtils.error(
223
+ 'INVALID_PREVIEW',
224
+ 'Preview must be a string (resource ID)',
225
+ 'metadata.preview',
226
+ ));
227
+ } else if (!ValidationUtils.resourceExists(metadata.preview, manifest.resources)) {
228
+ warnings.push(ValidationUtils.warning(
229
+ 'PREVIEW_NOT_RESOURCE',
230
+ `Preview "${metadata.preview}" does not match a resource ID`,
231
+ 'metadata.preview',
232
+ ));
233
+ }
234
+ }
235
+
236
+ // Suggest adding alt text for accessibility
237
+ if (!metadata.altText && metadata.mediaType === 'image') {
238
+ warnings.push(ValidationUtils.warning(
239
+ 'MISSING_ALT_TEXT',
240
+ 'Consider adding altText for accessibility',
241
+ 'metadata.altText',
242
+ ));
243
+ }
244
+
245
+ // Check that at least one media resource exists
246
+ const mediaResources = manifest.resources.filter(r =>
247
+ r.type === 'media' ||
248
+ r.type === 'image' ||
249
+ r.type === 'audio' ||
250
+ r.type === 'video' ||
251
+ r.contentType.startsWith('image/') ||
252
+ r.contentType.startsWith('audio/') ||
253
+ r.contentType.startsWith('video/') ||
254
+ r.contentType.startsWith('model/')
255
+ );
256
+ if (mediaResources.length === 0) {
257
+ warnings.push(ValidationUtils.warning(
258
+ 'NO_MEDIA_RESOURCES',
259
+ 'No media resources found. Ensure resources have appropriate types',
260
+ 'resources',
261
+ ));
262
+ }
263
+
264
+ return errors.length > 0
265
+ ? ValidationUtils.failure(errors, warnings)
266
+ : ValidationUtils.success(warnings);
267
+ }
268
+ }
269
+