@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,211 @@
1
+ /**
2
+ * App Kind Validator
3
+ *
4
+ * Validates manifests for executable applications with runtime and entrypoint.
5
+ */
6
+
7
+ import { OriginalKind, type OriginalManifest, type ValidationResult, type AppMetadata } from '../types';
8
+ import { BaseKindValidator, ValidationUtils } from './base';
9
+
10
+ /**
11
+ * Valid runtime environments
12
+ */
13
+ const VALID_RUNTIMES = [
14
+ 'node', 'deno', 'bun', 'browser', 'electron',
15
+ 'react-native', 'python', 'ruby', 'go', 'rust',
16
+ 'java', 'dotnet', 'wasm',
17
+ ];
18
+
19
+ /**
20
+ * Valid platforms
21
+ */
22
+ const VALID_PLATFORMS = ['linux', 'darwin', 'windows', 'web'];
23
+
24
+ /**
25
+ * Validator for App Originals
26
+ */
27
+ export class AppValidator extends BaseKindValidator<OriginalKind.App> {
28
+ readonly kind = OriginalKind.App;
29
+
30
+ protected validateKind(manifest: OriginalManifest<OriginalKind.App>): ValidationResult {
31
+ const errors: ValidationResult['errors'] = [];
32
+ const warnings: ValidationResult['warnings'] = [];
33
+ const metadata = manifest.metadata as AppMetadata;
34
+
35
+ // Validate metadata exists
36
+ if (!metadata || typeof metadata !== 'object') {
37
+ return ValidationUtils.failure([
38
+ ValidationUtils.error('MISSING_METADATA', 'App manifest must have metadata', 'metadata'),
39
+ ]);
40
+ }
41
+
42
+ // Validate runtime (required)
43
+ if (!metadata.runtime || typeof metadata.runtime !== 'string') {
44
+ errors.push(ValidationUtils.error(
45
+ 'MISSING_RUNTIME',
46
+ 'App must specify a runtime environment',
47
+ 'metadata.runtime',
48
+ ));
49
+ } else if (!VALID_RUNTIMES.includes(metadata.runtime.toLowerCase())) {
50
+ warnings.push(ValidationUtils.warning(
51
+ 'UNKNOWN_RUNTIME',
52
+ `Runtime "${metadata.runtime}" is not a commonly recognized runtime`,
53
+ 'metadata.runtime',
54
+ `Consider using one of: ${VALID_RUNTIMES.join(', ')}`,
55
+ ));
56
+ }
57
+
58
+ // Validate entrypoint (required)
59
+ if (!metadata.entrypoint || typeof metadata.entrypoint !== 'string') {
60
+ errors.push(ValidationUtils.error(
61
+ 'MISSING_ENTRYPOINT',
62
+ 'App must specify an entrypoint',
63
+ 'metadata.entrypoint',
64
+ ));
65
+ } else {
66
+ // Check if entrypoint references an existing resource
67
+ if (!ValidationUtils.resourceExists(metadata.entrypoint, manifest.resources)) {
68
+ // It might be a path within a resource, so just warn
69
+ warnings.push(ValidationUtils.warning(
70
+ 'ENTRYPOINT_NOT_RESOURCE',
71
+ `Entrypoint "${metadata.entrypoint}" does not match a resource ID`,
72
+ 'metadata.entrypoint',
73
+ 'Ensure the entrypoint is a valid resource ID or path within a resource',
74
+ ));
75
+ }
76
+ }
77
+
78
+ // Validate platforms if specified
79
+ if (metadata.platforms) {
80
+ if (!Array.isArray(metadata.platforms)) {
81
+ errors.push(ValidationUtils.error(
82
+ 'INVALID_PLATFORMS',
83
+ 'Platforms must be an array',
84
+ 'metadata.platforms',
85
+ ));
86
+ } else {
87
+ for (const platform of metadata.platforms) {
88
+ if (!VALID_PLATFORMS.includes(platform)) {
89
+ errors.push(ValidationUtils.error(
90
+ 'INVALID_PLATFORM',
91
+ `Invalid platform: "${platform}"`,
92
+ 'metadata.platforms',
93
+ platform,
94
+ ));
95
+ }
96
+ }
97
+ }
98
+ }
99
+
100
+ // Validate permissions if specified
101
+ if (metadata.permissions) {
102
+ if (!Array.isArray(metadata.permissions)) {
103
+ errors.push(ValidationUtils.error(
104
+ 'INVALID_PERMISSIONS',
105
+ 'Permissions must be an array of strings',
106
+ 'metadata.permissions',
107
+ ));
108
+ } else {
109
+ for (let i = 0; i < metadata.permissions.length; i++) {
110
+ if (typeof metadata.permissions[i] !== 'string') {
111
+ errors.push(ValidationUtils.error(
112
+ 'INVALID_PERMISSION',
113
+ `Permission at index ${i} must be a string`,
114
+ `metadata.permissions[${i}]`,
115
+ ));
116
+ }
117
+ }
118
+ }
119
+ }
120
+
121
+ // Validate env if specified
122
+ if (metadata.env) {
123
+ if (typeof metadata.env !== 'object' || Array.isArray(metadata.env)) {
124
+ errors.push(ValidationUtils.error(
125
+ 'INVALID_ENV',
126
+ 'Env must be an object',
127
+ 'metadata.env',
128
+ ));
129
+ } else {
130
+ for (const [key, value] of Object.entries(metadata.env)) {
131
+ if (!/^[A-Z][A-Z0-9_]*$/.test(key)) {
132
+ warnings.push(ValidationUtils.warning(
133
+ 'ENV_VAR_NAMING',
134
+ `Environment variable "${key}" should use SCREAMING_SNAKE_CASE`,
135
+ `metadata.env.${key}`,
136
+ ));
137
+ }
138
+ if (typeof value !== 'object' || value === null) {
139
+ errors.push(ValidationUtils.error(
140
+ 'INVALID_ENV_VALUE',
141
+ `Env variable "${key}" must have an object value`,
142
+ `metadata.env.${key}`,
143
+ ));
144
+ }
145
+ }
146
+ }
147
+ }
148
+
149
+ // Validate icons if specified
150
+ if (metadata.icons) {
151
+ if (typeof metadata.icons !== 'object' || Array.isArray(metadata.icons)) {
152
+ errors.push(ValidationUtils.error(
153
+ 'INVALID_ICONS',
154
+ 'Icons must be an object mapping sizes to resource IDs',
155
+ 'metadata.icons',
156
+ ));
157
+ } else {
158
+ for (const [size, resourceId] of Object.entries(metadata.icons)) {
159
+ if (!/^\d+x\d+$/.test(size) && !/^\d+$/.test(size)) {
160
+ warnings.push(ValidationUtils.warning(
161
+ 'ICON_SIZE_FORMAT',
162
+ `Icon size "${size}" should be in format "WxH" or just "N"`,
163
+ `metadata.icons.${size}`,
164
+ ));
165
+ }
166
+ }
167
+ }
168
+ }
169
+
170
+ // Validate commands if specified
171
+ if (metadata.commands) {
172
+ if (typeof metadata.commands !== 'object' || Array.isArray(metadata.commands)) {
173
+ errors.push(ValidationUtils.error(
174
+ 'INVALID_COMMANDS',
175
+ 'Commands must be an object',
176
+ 'metadata.commands',
177
+ ));
178
+ } else {
179
+ for (const [name, cmd] of Object.entries(metadata.commands)) {
180
+ if (!cmd || typeof cmd !== 'object') {
181
+ errors.push(ValidationUtils.error(
182
+ 'INVALID_COMMAND',
183
+ `Command "${name}" must be an object`,
184
+ `metadata.commands.${name}`,
185
+ ));
186
+ } else if (!cmd.description || typeof cmd.description !== 'string') {
187
+ warnings.push(ValidationUtils.warning(
188
+ 'MISSING_COMMAND_DESC',
189
+ `Command "${name}" should have a description`,
190
+ `metadata.commands.${name}.description`,
191
+ ));
192
+ }
193
+ }
194
+ }
195
+ }
196
+
197
+ // Suggest adding runtime version
198
+ if (!metadata.runtimeVersion && !metadata.minRuntimeVersion) {
199
+ warnings.push(ValidationUtils.warning(
200
+ 'MISSING_RUNTIME_VERSION',
201
+ 'Consider specifying runtimeVersion or minRuntimeVersion for compatibility',
202
+ 'metadata.runtimeVersion',
203
+ ));
204
+ }
205
+
206
+ return errors.length > 0
207
+ ? ValidationUtils.failure(errors, warnings)
208
+ : ValidationUtils.success(warnings);
209
+ }
210
+ }
211
+
@@ -0,0 +1,242 @@
1
+ /**
2
+ * Dataset Kind Validator
3
+ *
4
+ * Validates manifests for structured data collections with schema definitions.
5
+ */
6
+
7
+ import { OriginalKind, type OriginalManifest, type ValidationResult, type DatasetMetadata } from '../types';
8
+ import { BaseKindValidator, ValidationUtils } from './base';
9
+
10
+ /**
11
+ * Common data formats
12
+ */
13
+ const KNOWN_FORMATS = [
14
+ 'csv', 'json', 'jsonl', 'ndjson', 'parquet', 'avro', 'orc',
15
+ 'xml', 'yaml', 'toml', 'tsv', 'excel', 'sqlite', 'arrow',
16
+ ];
17
+
18
+ /**
19
+ * Valid privacy classifications
20
+ */
21
+ const VALID_PRIVACY = ['public', 'internal', 'confidential', 'restricted'];
22
+
23
+ /**
24
+ * Valid update frequencies
25
+ */
26
+ const VALID_UPDATE_FREQUENCIES = ['realtime', 'hourly', 'daily', 'weekly', 'monthly', 'static'];
27
+
28
+ /**
29
+ * Validator for Dataset Originals
30
+ */
31
+ export class DatasetValidator extends BaseKindValidator<OriginalKind.Dataset> {
32
+ readonly kind = OriginalKind.Dataset;
33
+
34
+ protected validateKind(manifest: OriginalManifest<OriginalKind.Dataset>): ValidationResult {
35
+ const errors: ValidationResult['errors'] = [];
36
+ const warnings: ValidationResult['warnings'] = [];
37
+ const metadata = manifest.metadata as DatasetMetadata;
38
+
39
+ // Validate metadata exists
40
+ if (!metadata || typeof metadata !== 'object') {
41
+ return ValidationUtils.failure([
42
+ ValidationUtils.error('MISSING_METADATA', 'Dataset manifest must have metadata', 'metadata'),
43
+ ]);
44
+ }
45
+
46
+ // Validate schema (required)
47
+ if (!metadata.schema) {
48
+ errors.push(ValidationUtils.error(
49
+ 'MISSING_SCHEMA',
50
+ 'Dataset must have a schema definition',
51
+ 'metadata.schema',
52
+ ));
53
+ } else if (typeof metadata.schema !== 'object' && typeof metadata.schema !== 'string') {
54
+ errors.push(ValidationUtils.error(
55
+ 'INVALID_SCHEMA',
56
+ 'Schema must be an object (JSON Schema) or string (URL)',
57
+ 'metadata.schema',
58
+ ));
59
+ } else if (typeof metadata.schema === 'string') {
60
+ // If it's a URL, validate it
61
+ if (!ValidationUtils.isValidURL(metadata.schema)) {
62
+ errors.push(ValidationUtils.error(
63
+ 'INVALID_SCHEMA_URL',
64
+ 'Schema URL is not a valid URL',
65
+ 'metadata.schema',
66
+ metadata.schema,
67
+ ));
68
+ }
69
+ }
70
+
71
+ // Validate format (required)
72
+ if (!metadata.format || typeof metadata.format !== 'string') {
73
+ errors.push(ValidationUtils.error(
74
+ 'MISSING_FORMAT',
75
+ 'Dataset must specify a data format',
76
+ 'metadata.format',
77
+ ));
78
+ } else {
79
+ const normalizedFormat = metadata.format.toLowerCase();
80
+ if (!KNOWN_FORMATS.includes(normalizedFormat)) {
81
+ warnings.push(ValidationUtils.warning(
82
+ 'UNKNOWN_FORMAT',
83
+ `Data format "${metadata.format}" is not a commonly recognized format`,
84
+ 'metadata.format',
85
+ `Consider using one of: ${KNOWN_FORMATS.join(', ')}`,
86
+ ));
87
+ }
88
+ }
89
+
90
+ // Validate recordCount if specified
91
+ if (metadata.recordCount !== undefined) {
92
+ if (typeof metadata.recordCount !== 'number' || metadata.recordCount < 0 || !Number.isInteger(metadata.recordCount)) {
93
+ errors.push(ValidationUtils.error(
94
+ 'INVALID_RECORD_COUNT',
95
+ 'Record count must be a non-negative integer',
96
+ 'metadata.recordCount',
97
+ metadata.recordCount,
98
+ ));
99
+ }
100
+ } else {
101
+ warnings.push(ValidationUtils.warning(
102
+ 'MISSING_RECORD_COUNT',
103
+ 'Consider specifying recordCount for better discoverability',
104
+ 'metadata.recordCount',
105
+ ));
106
+ }
107
+
108
+ // Validate columns if specified
109
+ if (metadata.columns) {
110
+ if (!Array.isArray(metadata.columns)) {
111
+ errors.push(ValidationUtils.error(
112
+ 'INVALID_COLUMNS',
113
+ 'Columns must be an array',
114
+ 'metadata.columns',
115
+ ));
116
+ } else {
117
+ const columnNames = new Set<string>();
118
+
119
+ for (let i = 0; i < metadata.columns.length; i++) {
120
+ const column = metadata.columns[i];
121
+ const columnPath = `metadata.columns[${i}]`;
122
+
123
+ if (!column || typeof column !== 'object') {
124
+ errors.push(ValidationUtils.error(
125
+ 'INVALID_COLUMN',
126
+ `Column at index ${i} must be an object`,
127
+ columnPath,
128
+ ));
129
+ continue;
130
+ }
131
+
132
+ if (!column.name || typeof column.name !== 'string') {
133
+ errors.push(ValidationUtils.error(
134
+ 'MISSING_COLUMN_NAME',
135
+ `Column at index ${i} must have a name`,
136
+ `${columnPath}.name`,
137
+ ));
138
+ } else {
139
+ if (columnNames.has(column.name)) {
140
+ errors.push(ValidationUtils.error(
141
+ 'DUPLICATE_COLUMN',
142
+ `Duplicate column name: "${column.name}"`,
143
+ `${columnPath}.name`,
144
+ ));
145
+ }
146
+ columnNames.add(column.name);
147
+ }
148
+
149
+ if (!column.type || typeof column.type !== 'string') {
150
+ errors.push(ValidationUtils.error(
151
+ 'MISSING_COLUMN_TYPE',
152
+ `Column "${column.name || i}" must have a type`,
153
+ `${columnPath}.type`,
154
+ ));
155
+ }
156
+ }
157
+ }
158
+ }
159
+
160
+ // Validate source if specified
161
+ if (metadata.source) {
162
+ if (typeof metadata.source !== 'object') {
163
+ errors.push(ValidationUtils.error(
164
+ 'INVALID_SOURCE',
165
+ 'Source must be an object',
166
+ 'metadata.source',
167
+ ));
168
+ }
169
+ }
170
+
171
+ // Validate statistics if specified
172
+ if (metadata.statistics) {
173
+ if (typeof metadata.statistics !== 'object') {
174
+ errors.push(ValidationUtils.error(
175
+ 'INVALID_STATISTICS',
176
+ 'Statistics must be an object',
177
+ 'metadata.statistics',
178
+ ));
179
+ } else {
180
+ if (metadata.statistics.sizeBytes !== undefined &&
181
+ (typeof metadata.statistics.sizeBytes !== 'number' || metadata.statistics.sizeBytes < 0)) {
182
+ errors.push(ValidationUtils.error(
183
+ 'INVALID_SIZE_BYTES',
184
+ 'sizeBytes must be a non-negative number',
185
+ 'metadata.statistics.sizeBytes',
186
+ ));
187
+ }
188
+ }
189
+ }
190
+
191
+ // Validate privacy if specified
192
+ if (metadata.privacy) {
193
+ if (!VALID_PRIVACY.includes(metadata.privacy)) {
194
+ errors.push(ValidationUtils.error(
195
+ 'INVALID_PRIVACY',
196
+ `Privacy must be one of: ${VALID_PRIVACY.join(', ')}`,
197
+ 'metadata.privacy',
198
+ metadata.privacy,
199
+ ));
200
+ }
201
+ } else {
202
+ warnings.push(ValidationUtils.warning(
203
+ 'MISSING_PRIVACY',
204
+ 'Consider specifying a privacy classification for the dataset',
205
+ 'metadata.privacy',
206
+ ));
207
+ }
208
+
209
+ // Validate updateFrequency if specified
210
+ if (metadata.updateFrequency) {
211
+ if (!VALID_UPDATE_FREQUENCIES.includes(metadata.updateFrequency)) {
212
+ errors.push(ValidationUtils.error(
213
+ 'INVALID_UPDATE_FREQUENCY',
214
+ `Update frequency must be one of: ${VALID_UPDATE_FREQUENCIES.join(', ')}`,
215
+ 'metadata.updateFrequency',
216
+ metadata.updateFrequency,
217
+ ));
218
+ }
219
+ }
220
+
221
+ // Check that at least one data resource exists
222
+ const dataResources = manifest.resources.filter(r =>
223
+ r.type === 'data' ||
224
+ r.contentType.includes('csv') ||
225
+ r.contentType.includes('json') ||
226
+ r.contentType.includes('parquet') ||
227
+ r.contentType.includes('octet-stream')
228
+ );
229
+ if (dataResources.length === 0) {
230
+ warnings.push(ValidationUtils.warning(
231
+ 'NO_DATA_RESOURCES',
232
+ 'No data resources found. Ensure resources have appropriate types',
233
+ 'resources',
234
+ ));
235
+ }
236
+
237
+ return errors.length > 0
238
+ ? ValidationUtils.failure(errors, warnings)
239
+ : ValidationUtils.success(warnings);
240
+ }
241
+ }
242
+