@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,211 +0,0 @@
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
-
@@ -1,242 +0,0 @@
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
-