@originals/sdk 1.8.2 → 1.8.3

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 (144) hide show
  1. package/package.json +5 -6
  2. package/src/adapters/FeeOracleMock.ts +0 -9
  3. package/src/adapters/index.ts +0 -5
  4. package/src/adapters/providers/OrdHttpProvider.ts +0 -126
  5. package/src/adapters/providers/OrdMockProvider.ts +0 -101
  6. package/src/adapters/types.ts +0 -66
  7. package/src/bitcoin/BitcoinManager.ts +0 -329
  8. package/src/bitcoin/BroadcastClient.ts +0 -54
  9. package/src/bitcoin/OrdinalsClient.ts +0 -120
  10. package/src/bitcoin/PSBTBuilder.ts +0 -106
  11. package/src/bitcoin/fee-calculation.ts +0 -38
  12. package/src/bitcoin/providers/OrdNodeProvider.ts +0 -92
  13. package/src/bitcoin/providers/OrdinalsProvider.ts +0 -56
  14. package/src/bitcoin/providers/types.ts +0 -59
  15. package/src/bitcoin/transactions/commit.ts +0 -465
  16. package/src/bitcoin/transactions/index.ts +0 -13
  17. package/src/bitcoin/transfer.ts +0 -43
  18. package/src/bitcoin/utxo-selection.ts +0 -322
  19. package/src/bitcoin/utxo.ts +0 -113
  20. package/src/cel/ExternalReferenceManager.ts +0 -87
  21. package/src/cel/OriginalsCel.ts +0 -460
  22. package/src/cel/algorithms/createEventLog.ts +0 -68
  23. package/src/cel/algorithms/deactivateEventLog.ts +0 -109
  24. package/src/cel/algorithms/index.ts +0 -11
  25. package/src/cel/algorithms/updateEventLog.ts +0 -99
  26. package/src/cel/algorithms/verifyEventLog.ts +0 -306
  27. package/src/cel/algorithms/witnessEvent.ts +0 -87
  28. package/src/cel/cli/create.ts +0 -330
  29. package/src/cel/cli/index.ts +0 -383
  30. package/src/cel/cli/inspect.ts +0 -549
  31. package/src/cel/cli/migrate.ts +0 -473
  32. package/src/cel/cli/verify.ts +0 -249
  33. package/src/cel/hash.ts +0 -71
  34. package/src/cel/index.ts +0 -16
  35. package/src/cel/layers/BtcoCelManager.ts +0 -408
  36. package/src/cel/layers/PeerCelManager.ts +0 -371
  37. package/src/cel/layers/WebVHCelManager.ts +0 -361
  38. package/src/cel/layers/index.ts +0 -27
  39. package/src/cel/serialization/cbor.ts +0 -189
  40. package/src/cel/serialization/index.ts +0 -10
  41. package/src/cel/serialization/json.ts +0 -209
  42. package/src/cel/types.ts +0 -160
  43. package/src/cel/witnesses/BitcoinWitness.ts +0 -184
  44. package/src/cel/witnesses/HttpWitness.ts +0 -241
  45. package/src/cel/witnesses/WitnessService.ts +0 -51
  46. package/src/cel/witnesses/index.ts +0 -11
  47. package/src/contexts/credentials-v1.json +0 -237
  48. package/src/contexts/credentials-v2-examples.json +0 -5
  49. package/src/contexts/credentials-v2.json +0 -340
  50. package/src/contexts/credentials.json +0 -237
  51. package/src/contexts/data-integrity-v2.json +0 -81
  52. package/src/contexts/dids.json +0 -58
  53. package/src/contexts/ed255192020.json +0 -93
  54. package/src/contexts/ordinals-plus.json +0 -23
  55. package/src/contexts/originals.json +0 -22
  56. package/src/core/OriginalsSDK.ts +0 -420
  57. package/src/crypto/Multikey.ts +0 -194
  58. package/src/crypto/Signer.ts +0 -262
  59. package/src/crypto/noble-init.ts +0 -138
  60. package/src/did/BtcoDidResolver.ts +0 -231
  61. package/src/did/DIDManager.ts +0 -705
  62. package/src/did/Ed25519Verifier.ts +0 -68
  63. package/src/did/KeyManager.ts +0 -239
  64. package/src/did/WebVHManager.ts +0 -499
  65. package/src/did/createBtcoDidDocument.ts +0 -60
  66. package/src/did/providers/OrdinalsClientProviderAdapter.ts +0 -68
  67. package/src/events/EventEmitter.ts +0 -222
  68. package/src/events/index.ts +0 -19
  69. package/src/events/types.ts +0 -331
  70. package/src/examples/basic-usage.ts +0 -78
  71. package/src/examples/create-module-original.ts +0 -435
  72. package/src/examples/full-lifecycle-flow.ts +0 -514
  73. package/src/examples/run.ts +0 -60
  74. package/src/index.ts +0 -204
  75. package/src/kinds/KindRegistry.ts +0 -320
  76. package/src/kinds/index.ts +0 -74
  77. package/src/kinds/types.ts +0 -470
  78. package/src/kinds/validators/AgentValidator.ts +0 -257
  79. package/src/kinds/validators/AppValidator.ts +0 -211
  80. package/src/kinds/validators/DatasetValidator.ts +0 -242
  81. package/src/kinds/validators/DocumentValidator.ts +0 -311
  82. package/src/kinds/validators/MediaValidator.ts +0 -269
  83. package/src/kinds/validators/ModuleValidator.ts +0 -225
  84. package/src/kinds/validators/base.ts +0 -276
  85. package/src/kinds/validators/index.ts +0 -12
  86. package/src/lifecycle/BatchOperations.ts +0 -381
  87. package/src/lifecycle/LifecycleManager.ts +0 -2156
  88. package/src/lifecycle/OriginalsAsset.ts +0 -524
  89. package/src/lifecycle/ProvenanceQuery.ts +0 -280
  90. package/src/lifecycle/ResourceVersioning.ts +0 -163
  91. package/src/migration/MigrationManager.ts +0 -587
  92. package/src/migration/audit/AuditLogger.ts +0 -176
  93. package/src/migration/checkpoint/CheckpointManager.ts +0 -112
  94. package/src/migration/checkpoint/CheckpointStorage.ts +0 -101
  95. package/src/migration/index.ts +0 -33
  96. package/src/migration/operations/BaseMigration.ts +0 -126
  97. package/src/migration/operations/PeerToBtcoMigration.ts +0 -105
  98. package/src/migration/operations/PeerToWebvhMigration.ts +0 -62
  99. package/src/migration/operations/WebvhToBtcoMigration.ts +0 -105
  100. package/src/migration/rollback/RollbackManager.ts +0 -170
  101. package/src/migration/state/StateMachine.ts +0 -92
  102. package/src/migration/state/StateTracker.ts +0 -156
  103. package/src/migration/types.ts +0 -356
  104. package/src/migration/validation/BitcoinValidator.ts +0 -107
  105. package/src/migration/validation/CredentialValidator.ts +0 -62
  106. package/src/migration/validation/DIDCompatibilityValidator.ts +0 -151
  107. package/src/migration/validation/LifecycleValidator.ts +0 -64
  108. package/src/migration/validation/StorageValidator.ts +0 -79
  109. package/src/migration/validation/ValidationPipeline.ts +0 -213
  110. package/src/resources/ResourceManager.ts +0 -655
  111. package/src/resources/index.ts +0 -21
  112. package/src/resources/types.ts +0 -202
  113. package/src/storage/LocalStorageAdapter.ts +0 -64
  114. package/src/storage/MemoryStorageAdapter.ts +0 -29
  115. package/src/storage/StorageAdapter.ts +0 -25
  116. package/src/storage/index.ts +0 -3
  117. package/src/types/bitcoin.ts +0 -98
  118. package/src/types/common.ts +0 -92
  119. package/src/types/credentials.ts +0 -89
  120. package/src/types/did.ts +0 -31
  121. package/src/types/external-shims.d.ts +0 -53
  122. package/src/types/index.ts +0 -7
  123. package/src/types/network.ts +0 -178
  124. package/src/utils/EventLogger.ts +0 -298
  125. package/src/utils/Logger.ts +0 -324
  126. package/src/utils/MetricsCollector.ts +0 -358
  127. package/src/utils/bitcoin-address.ts +0 -132
  128. package/src/utils/cbor.ts +0 -31
  129. package/src/utils/encoding.ts +0 -135
  130. package/src/utils/hash.ts +0 -12
  131. package/src/utils/retry.ts +0 -46
  132. package/src/utils/satoshi-validation.ts +0 -196
  133. package/src/utils/serialization.ts +0 -102
  134. package/src/utils/telemetry.ts +0 -44
  135. package/src/utils/validation.ts +0 -123
  136. package/src/vc/CredentialManager.ts +0 -955
  137. package/src/vc/Issuer.ts +0 -105
  138. package/src/vc/Verifier.ts +0 -54
  139. package/src/vc/cryptosuites/bbs.ts +0 -253
  140. package/src/vc/cryptosuites/bbsSimple.ts +0 -21
  141. package/src/vc/cryptosuites/eddsa.ts +0 -99
  142. package/src/vc/documentLoader.ts +0 -81
  143. package/src/vc/proofs/data-integrity.ts +0 -33
  144. package/src/vc/utils/jsonld.ts +0 -18
@@ -1,311 +0,0 @@
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
-
@@ -1,269 +0,0 @@
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
-