@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.
- package/package.json +5 -6
- package/src/adapters/FeeOracleMock.ts +0 -9
- package/src/adapters/index.ts +0 -5
- package/src/adapters/providers/OrdHttpProvider.ts +0 -126
- package/src/adapters/providers/OrdMockProvider.ts +0 -101
- package/src/adapters/types.ts +0 -66
- package/src/bitcoin/BitcoinManager.ts +0 -329
- package/src/bitcoin/BroadcastClient.ts +0 -54
- package/src/bitcoin/OrdinalsClient.ts +0 -120
- package/src/bitcoin/PSBTBuilder.ts +0 -106
- package/src/bitcoin/fee-calculation.ts +0 -38
- package/src/bitcoin/providers/OrdNodeProvider.ts +0 -92
- package/src/bitcoin/providers/OrdinalsProvider.ts +0 -56
- package/src/bitcoin/providers/types.ts +0 -59
- package/src/bitcoin/transactions/commit.ts +0 -465
- package/src/bitcoin/transactions/index.ts +0 -13
- package/src/bitcoin/transfer.ts +0 -43
- package/src/bitcoin/utxo-selection.ts +0 -322
- package/src/bitcoin/utxo.ts +0 -113
- package/src/cel/ExternalReferenceManager.ts +0 -87
- package/src/cel/OriginalsCel.ts +0 -460
- package/src/cel/algorithms/createEventLog.ts +0 -68
- package/src/cel/algorithms/deactivateEventLog.ts +0 -109
- package/src/cel/algorithms/index.ts +0 -11
- package/src/cel/algorithms/updateEventLog.ts +0 -99
- package/src/cel/algorithms/verifyEventLog.ts +0 -306
- package/src/cel/algorithms/witnessEvent.ts +0 -87
- package/src/cel/cli/create.ts +0 -330
- package/src/cel/cli/index.ts +0 -383
- package/src/cel/cli/inspect.ts +0 -549
- package/src/cel/cli/migrate.ts +0 -473
- package/src/cel/cli/verify.ts +0 -249
- package/src/cel/hash.ts +0 -71
- package/src/cel/index.ts +0 -16
- package/src/cel/layers/BtcoCelManager.ts +0 -408
- package/src/cel/layers/PeerCelManager.ts +0 -371
- package/src/cel/layers/WebVHCelManager.ts +0 -361
- package/src/cel/layers/index.ts +0 -27
- package/src/cel/serialization/cbor.ts +0 -189
- package/src/cel/serialization/index.ts +0 -10
- package/src/cel/serialization/json.ts +0 -209
- package/src/cel/types.ts +0 -160
- package/src/cel/witnesses/BitcoinWitness.ts +0 -184
- package/src/cel/witnesses/HttpWitness.ts +0 -241
- package/src/cel/witnesses/WitnessService.ts +0 -51
- package/src/cel/witnesses/index.ts +0 -11
- package/src/contexts/credentials-v1.json +0 -237
- package/src/contexts/credentials-v2-examples.json +0 -5
- package/src/contexts/credentials-v2.json +0 -340
- package/src/contexts/credentials.json +0 -237
- package/src/contexts/data-integrity-v2.json +0 -81
- package/src/contexts/dids.json +0 -58
- package/src/contexts/ed255192020.json +0 -93
- package/src/contexts/ordinals-plus.json +0 -23
- package/src/contexts/originals.json +0 -22
- package/src/core/OriginalsSDK.ts +0 -420
- package/src/crypto/Multikey.ts +0 -194
- package/src/crypto/Signer.ts +0 -262
- package/src/crypto/noble-init.ts +0 -138
- package/src/did/BtcoDidResolver.ts +0 -231
- package/src/did/DIDManager.ts +0 -705
- package/src/did/Ed25519Verifier.ts +0 -68
- package/src/did/KeyManager.ts +0 -239
- package/src/did/WebVHManager.ts +0 -499
- package/src/did/createBtcoDidDocument.ts +0 -60
- package/src/did/providers/OrdinalsClientProviderAdapter.ts +0 -68
- package/src/events/EventEmitter.ts +0 -222
- package/src/events/index.ts +0 -19
- package/src/events/types.ts +0 -331
- package/src/examples/basic-usage.ts +0 -78
- package/src/examples/create-module-original.ts +0 -435
- package/src/examples/full-lifecycle-flow.ts +0 -514
- package/src/examples/run.ts +0 -60
- package/src/index.ts +0 -204
- package/src/kinds/KindRegistry.ts +0 -320
- package/src/kinds/index.ts +0 -74
- package/src/kinds/types.ts +0 -470
- package/src/kinds/validators/AgentValidator.ts +0 -257
- package/src/kinds/validators/AppValidator.ts +0 -211
- package/src/kinds/validators/DatasetValidator.ts +0 -242
- package/src/kinds/validators/DocumentValidator.ts +0 -311
- package/src/kinds/validators/MediaValidator.ts +0 -269
- package/src/kinds/validators/ModuleValidator.ts +0 -225
- package/src/kinds/validators/base.ts +0 -276
- package/src/kinds/validators/index.ts +0 -12
- package/src/lifecycle/BatchOperations.ts +0 -381
- package/src/lifecycle/LifecycleManager.ts +0 -2156
- package/src/lifecycle/OriginalsAsset.ts +0 -524
- package/src/lifecycle/ProvenanceQuery.ts +0 -280
- package/src/lifecycle/ResourceVersioning.ts +0 -163
- package/src/migration/MigrationManager.ts +0 -587
- package/src/migration/audit/AuditLogger.ts +0 -176
- package/src/migration/checkpoint/CheckpointManager.ts +0 -112
- package/src/migration/checkpoint/CheckpointStorage.ts +0 -101
- package/src/migration/index.ts +0 -33
- package/src/migration/operations/BaseMigration.ts +0 -126
- package/src/migration/operations/PeerToBtcoMigration.ts +0 -105
- package/src/migration/operations/PeerToWebvhMigration.ts +0 -62
- package/src/migration/operations/WebvhToBtcoMigration.ts +0 -105
- package/src/migration/rollback/RollbackManager.ts +0 -170
- package/src/migration/state/StateMachine.ts +0 -92
- package/src/migration/state/StateTracker.ts +0 -156
- package/src/migration/types.ts +0 -356
- package/src/migration/validation/BitcoinValidator.ts +0 -107
- package/src/migration/validation/CredentialValidator.ts +0 -62
- package/src/migration/validation/DIDCompatibilityValidator.ts +0 -151
- package/src/migration/validation/LifecycleValidator.ts +0 -64
- package/src/migration/validation/StorageValidator.ts +0 -79
- package/src/migration/validation/ValidationPipeline.ts +0 -213
- package/src/resources/ResourceManager.ts +0 -655
- package/src/resources/index.ts +0 -21
- package/src/resources/types.ts +0 -202
- package/src/storage/LocalStorageAdapter.ts +0 -64
- package/src/storage/MemoryStorageAdapter.ts +0 -29
- package/src/storage/StorageAdapter.ts +0 -25
- package/src/storage/index.ts +0 -3
- package/src/types/bitcoin.ts +0 -98
- package/src/types/common.ts +0 -92
- package/src/types/credentials.ts +0 -89
- package/src/types/did.ts +0 -31
- package/src/types/external-shims.d.ts +0 -53
- package/src/types/index.ts +0 -7
- package/src/types/network.ts +0 -178
- package/src/utils/EventLogger.ts +0 -298
- package/src/utils/Logger.ts +0 -324
- package/src/utils/MetricsCollector.ts +0 -358
- package/src/utils/bitcoin-address.ts +0 -132
- package/src/utils/cbor.ts +0 -31
- package/src/utils/encoding.ts +0 -135
- package/src/utils/hash.ts +0 -12
- package/src/utils/retry.ts +0 -46
- package/src/utils/satoshi-validation.ts +0 -196
- package/src/utils/serialization.ts +0 -102
- package/src/utils/telemetry.ts +0 -44
- package/src/utils/validation.ts +0 -123
- package/src/vc/CredentialManager.ts +0 -955
- package/src/vc/Issuer.ts +0 -105
- package/src/vc/Verifier.ts +0 -54
- package/src/vc/cryptosuites/bbs.ts +0 -253
- package/src/vc/cryptosuites/bbsSimple.ts +0 -21
- package/src/vc/cryptosuites/eddsa.ts +0 -99
- package/src/vc/documentLoader.ts +0 -81
- package/src/vc/proofs/data-integrity.ts +0 -33
- package/src/vc/utils/jsonld.ts +0 -18
|
@@ -1,655 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* ResourceManager - CRUD operations for immutable, versioned resources.
|
|
3
|
-
*
|
|
4
|
-
* Resources in the Originals SDK are content-addressed and immutable. Each "update"
|
|
5
|
-
* creates a new version with a new content hash, linked to the previous version
|
|
6
|
-
* via previousVersionHash. This creates a verifiable provenance chain.
|
|
7
|
-
*
|
|
8
|
-
* @example
|
|
9
|
-
* ```typescript
|
|
10
|
-
* const manager = new ResourceManager();
|
|
11
|
-
*
|
|
12
|
-
* // Create a new resource
|
|
13
|
-
* const resource = manager.createResource('Hello, World!', {
|
|
14
|
-
* type: 'text',
|
|
15
|
-
* contentType: 'text/plain'
|
|
16
|
-
* });
|
|
17
|
-
*
|
|
18
|
-
* // Update creates a new version
|
|
19
|
-
* const updatedResource = manager.updateResource(resource, 'Hello, Updated World!', {
|
|
20
|
-
* changes: 'Updated greeting'
|
|
21
|
-
* });
|
|
22
|
-
*
|
|
23
|
-
* // Get version history
|
|
24
|
-
* const history = manager.getResourceHistory(resource.id);
|
|
25
|
-
* ```
|
|
26
|
-
*/
|
|
27
|
-
|
|
28
|
-
import { sha256 } from '@noble/hashes/sha2.js';
|
|
29
|
-
import { bytesToHex } from '@noble/hashes/utils.js';
|
|
30
|
-
import { v4 as uuidv4 } from 'uuid';
|
|
31
|
-
import type {
|
|
32
|
-
Resource,
|
|
33
|
-
ResourceOptions,
|
|
34
|
-
ResourceUpdateOptions,
|
|
35
|
-
ResourceVersionHistory,
|
|
36
|
-
ResourceManagerConfig,
|
|
37
|
-
ResourceValidationResult,
|
|
38
|
-
ResourceType,
|
|
39
|
-
} from './types.js';
|
|
40
|
-
import { MIME_TYPE_MAP, DEFAULT_RESOURCE_CONFIG } from './types.js';
|
|
41
|
-
|
|
42
|
-
/**
|
|
43
|
-
* Regular expression for validating MIME types according to RFC 6838.
|
|
44
|
-
* Format: type/subtype where type and subtype are restricted character sets.
|
|
45
|
-
*/
|
|
46
|
-
const MIME_TYPE_REGEX = /^[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}\/[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}$/;
|
|
47
|
-
|
|
48
|
-
/**
|
|
49
|
-
* ResourceManager provides CRUD operations for immutable, content-addressed resources
|
|
50
|
-
* with versioning support and validation.
|
|
51
|
-
*/
|
|
52
|
-
export class ResourceManager {
|
|
53
|
-
private resources: Map<string, Resource[]>;
|
|
54
|
-
private config: Required<ResourceManagerConfig>;
|
|
55
|
-
|
|
56
|
-
/**
|
|
57
|
-
* Create a new ResourceManager instance.
|
|
58
|
-
*
|
|
59
|
-
* @param config - Optional configuration for the manager
|
|
60
|
-
*/
|
|
61
|
-
constructor(config?: ResourceManagerConfig) {
|
|
62
|
-
this.resources = new Map();
|
|
63
|
-
this.config = { ...DEFAULT_RESOURCE_CONFIG, ...config };
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
/**
|
|
67
|
-
* Create a new resource from content.
|
|
68
|
-
*
|
|
69
|
-
* @param content - The resource content (string for text, Buffer for binary)
|
|
70
|
-
* @param options - Creation options including type and contentType
|
|
71
|
-
* @returns The created Resource
|
|
72
|
-
* @throws Error if content or options are invalid
|
|
73
|
-
*
|
|
74
|
-
* @example
|
|
75
|
-
* ```typescript
|
|
76
|
-
* // Create a text resource
|
|
77
|
-
* const textResource = manager.createResource('# README\nHello', {
|
|
78
|
-
* type: 'text',
|
|
79
|
-
* contentType: 'text/markdown'
|
|
80
|
-
* });
|
|
81
|
-
*
|
|
82
|
-
* // Create a binary resource (image)
|
|
83
|
-
* const imageBuffer = fs.readFileSync('image.png');
|
|
84
|
-
* const imageResource = manager.createResource(imageBuffer, {
|
|
85
|
-
* type: 'image',
|
|
86
|
-
* contentType: 'image/png'
|
|
87
|
-
* });
|
|
88
|
-
* ```
|
|
89
|
-
*/
|
|
90
|
-
createResource(content: Buffer | string, options: ResourceOptions): Resource {
|
|
91
|
-
// Validate inputs
|
|
92
|
-
if (content === null || content === undefined) {
|
|
93
|
-
throw new Error('Content is required');
|
|
94
|
-
}
|
|
95
|
-
if (!options) {
|
|
96
|
-
throw new Error('Options are required');
|
|
97
|
-
}
|
|
98
|
-
if (!options.type) {
|
|
99
|
-
throw new Error('Resource type is required');
|
|
100
|
-
}
|
|
101
|
-
if (!options.contentType) {
|
|
102
|
-
throw new Error('Content type is required');
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
// Validate MIME type format
|
|
106
|
-
if (!this.isValidMimeType(options.contentType)) {
|
|
107
|
-
throw new Error(`Invalid MIME type format: ${options.contentType}`);
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
// Check allowed content types
|
|
111
|
-
if (this.config.allowedContentTypes.length > 0 &&
|
|
112
|
-
!this.config.allowedContentTypes.includes(options.contentType)) {
|
|
113
|
-
throw new Error(`Content type not allowed: ${options.contentType}. Allowed types: ${this.config.allowedContentTypes.join(', ')}`);
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
// Convert content to buffer for consistent handling
|
|
117
|
-
const contentBuffer = this.toBuffer(content);
|
|
118
|
-
|
|
119
|
-
// Validate size
|
|
120
|
-
const maxSize = options.maxSize || this.config.defaultMaxSize;
|
|
121
|
-
if (contentBuffer.length > maxSize) {
|
|
122
|
-
throw new Error(`Resource size (${contentBuffer.length} bytes) exceeds maximum allowed size (${maxSize} bytes)`);
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
// Generate hash
|
|
126
|
-
const hash = this.hashContent(contentBuffer);
|
|
127
|
-
|
|
128
|
-
// Generate or use provided ID
|
|
129
|
-
const id = options.id || uuidv4();
|
|
130
|
-
|
|
131
|
-
// Create resource object
|
|
132
|
-
const resource: Resource = {
|
|
133
|
-
id,
|
|
134
|
-
type: options.type,
|
|
135
|
-
contentType: options.contentType,
|
|
136
|
-
hash,
|
|
137
|
-
size: contentBuffer.length,
|
|
138
|
-
version: 1,
|
|
139
|
-
createdAt: new Date().toISOString(),
|
|
140
|
-
url: options.url,
|
|
141
|
-
description: options.description,
|
|
142
|
-
};
|
|
143
|
-
|
|
144
|
-
// Store content if configured to do so
|
|
145
|
-
if (this.config.storeContent) {
|
|
146
|
-
if (this.isBinaryContent(content)) {
|
|
147
|
-
resource.contentBase64 = contentBuffer.toString('base64');
|
|
148
|
-
} else {
|
|
149
|
-
resource.content = typeof content === 'string' ? content : contentBuffer.toString('utf-8');
|
|
150
|
-
}
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
// Store in version history
|
|
154
|
-
this.resources.set(id, [resource]);
|
|
155
|
-
|
|
156
|
-
return resource;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
/**
|
|
160
|
-
* Update a resource by creating a new version.
|
|
161
|
-
* The original resource remains unchanged (immutable versioning).
|
|
162
|
-
*
|
|
163
|
-
* @param resource - The resource to update (or its ID)
|
|
164
|
-
* @param newContent - The new content
|
|
165
|
-
* @param options - Optional update options including change description
|
|
166
|
-
* @returns The new version of the resource
|
|
167
|
-
* @throws Error if resource not found or content unchanged
|
|
168
|
-
*
|
|
169
|
-
* @example
|
|
170
|
-
* ```typescript
|
|
171
|
-
* const v2 = manager.updateResource(originalResource, 'Updated content', {
|
|
172
|
-
* changes: 'Fixed typo in documentation'
|
|
173
|
-
* });
|
|
174
|
-
*
|
|
175
|
-
* console.log(v2.version); // 2
|
|
176
|
-
* console.log(v2.previousVersionHash); // hash of v1
|
|
177
|
-
* ```
|
|
178
|
-
*/
|
|
179
|
-
updateResource(
|
|
180
|
-
resource: Resource | string,
|
|
181
|
-
newContent: Buffer | string,
|
|
182
|
-
options?: ResourceUpdateOptions
|
|
183
|
-
): Resource {
|
|
184
|
-
const resourceId = typeof resource === 'string' ? resource : resource.id;
|
|
185
|
-
|
|
186
|
-
// Get version history
|
|
187
|
-
const versions = this.resources.get(resourceId);
|
|
188
|
-
if (!versions || versions.length === 0) {
|
|
189
|
-
throw new Error(`Resource not found: ${resourceId}`);
|
|
190
|
-
}
|
|
191
|
-
|
|
192
|
-
// Get current (latest) version
|
|
193
|
-
const currentVersion = versions[versions.length - 1];
|
|
194
|
-
|
|
195
|
-
// Convert content to buffer
|
|
196
|
-
const contentBuffer = this.toBuffer(newContent);
|
|
197
|
-
|
|
198
|
-
// Generate hash for new content
|
|
199
|
-
const newHash = this.hashContent(contentBuffer);
|
|
200
|
-
|
|
201
|
-
// Check if content has actually changed
|
|
202
|
-
if (newHash === currentVersion.hash) {
|
|
203
|
-
throw new Error('Content unchanged - new version would be identical to current version');
|
|
204
|
-
}
|
|
205
|
-
|
|
206
|
-
// Validate size
|
|
207
|
-
if (contentBuffer.length > this.config.defaultMaxSize) {
|
|
208
|
-
throw new Error(`Resource size (${contentBuffer.length} bytes) exceeds maximum allowed size (${this.config.defaultMaxSize} bytes)`);
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
// Determine content type (use provided or inherit from previous)
|
|
212
|
-
const contentType = options?.contentType || currentVersion.contentType;
|
|
213
|
-
|
|
214
|
-
// Validate new content type if changed
|
|
215
|
-
if (options?.contentType && !this.isValidMimeType(options.contentType)) {
|
|
216
|
-
throw new Error(`Invalid MIME type format: ${options.contentType}`);
|
|
217
|
-
}
|
|
218
|
-
|
|
219
|
-
// Create new version
|
|
220
|
-
const newVersion: Resource = {
|
|
221
|
-
id: resourceId,
|
|
222
|
-
type: currentVersion.type,
|
|
223
|
-
contentType,
|
|
224
|
-
hash: newHash,
|
|
225
|
-
size: contentBuffer.length,
|
|
226
|
-
version: (currentVersion.version || 1) + 1,
|
|
227
|
-
previousVersionHash: currentVersion.hash,
|
|
228
|
-
createdAt: new Date().toISOString(),
|
|
229
|
-
url: currentVersion.url,
|
|
230
|
-
description: currentVersion.description,
|
|
231
|
-
};
|
|
232
|
-
|
|
233
|
-
// Store content if configured
|
|
234
|
-
if (this.config.storeContent) {
|
|
235
|
-
if (this.isBinaryContent(newContent)) {
|
|
236
|
-
newVersion.contentBase64 = contentBuffer.toString('base64');
|
|
237
|
-
} else {
|
|
238
|
-
newVersion.content = typeof newContent === 'string' ? newContent : contentBuffer.toString('utf-8');
|
|
239
|
-
}
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
// Add to version history
|
|
243
|
-
versions.push(newVersion);
|
|
244
|
-
|
|
245
|
-
return newVersion;
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
/**
|
|
249
|
-
* Get the complete version history for a resource.
|
|
250
|
-
*
|
|
251
|
-
* @param resourceId - The logical resource ID
|
|
252
|
-
* @returns Array of all versions (oldest to newest), or empty array if not found
|
|
253
|
-
*
|
|
254
|
-
* @example
|
|
255
|
-
* ```typescript
|
|
256
|
-
* const history = manager.getResourceHistory('my-resource-id');
|
|
257
|
-
* console.log(`Found ${history.length} versions`);
|
|
258
|
-
* history.forEach((v, i) => console.log(`v${i + 1}: ${v.hash}`));
|
|
259
|
-
* ```
|
|
260
|
-
*/
|
|
261
|
-
getResourceHistory(resourceId: string): Resource[] {
|
|
262
|
-
const versions = this.resources.get(resourceId);
|
|
263
|
-
if (!versions) {
|
|
264
|
-
return [];
|
|
265
|
-
}
|
|
266
|
-
return [...versions]; // Return copy to prevent external mutation
|
|
267
|
-
}
|
|
268
|
-
|
|
269
|
-
/**
|
|
270
|
-
* Get detailed version history with metadata.
|
|
271
|
-
*
|
|
272
|
-
* @param resourceId - The logical resource ID
|
|
273
|
-
* @returns ResourceVersionHistory object or null if not found
|
|
274
|
-
*/
|
|
275
|
-
getResourceVersionHistory(resourceId: string): ResourceVersionHistory | null {
|
|
276
|
-
const versions = this.resources.get(resourceId);
|
|
277
|
-
if (!versions || versions.length === 0) {
|
|
278
|
-
return null;
|
|
279
|
-
}
|
|
280
|
-
|
|
281
|
-
return {
|
|
282
|
-
resourceId,
|
|
283
|
-
versions: [...versions],
|
|
284
|
-
currentVersion: versions[versions.length - 1],
|
|
285
|
-
versionCount: versions.length,
|
|
286
|
-
};
|
|
287
|
-
}
|
|
288
|
-
|
|
289
|
-
/**
|
|
290
|
-
* Get a specific version of a resource.
|
|
291
|
-
*
|
|
292
|
-
* @param resourceId - The logical resource ID
|
|
293
|
-
* @param version - Version number (1-indexed)
|
|
294
|
-
* @returns The resource at that version, or null if not found
|
|
295
|
-
*/
|
|
296
|
-
getResourceVersion(resourceId: string, version: number): Resource | null {
|
|
297
|
-
const versions = this.resources.get(resourceId);
|
|
298
|
-
if (!versions || version < 1 || version > versions.length) {
|
|
299
|
-
return null;
|
|
300
|
-
}
|
|
301
|
-
return versions[version - 1];
|
|
302
|
-
}
|
|
303
|
-
|
|
304
|
-
/**
|
|
305
|
-
* Get the current (latest) version of a resource.
|
|
306
|
-
*
|
|
307
|
-
* @param resourceId - The logical resource ID
|
|
308
|
-
* @returns The current version, or null if not found
|
|
309
|
-
*/
|
|
310
|
-
getCurrentVersion(resourceId: string): Resource | null {
|
|
311
|
-
const versions = this.resources.get(resourceId);
|
|
312
|
-
if (!versions || versions.length === 0) {
|
|
313
|
-
return null;
|
|
314
|
-
}
|
|
315
|
-
return versions[versions.length - 1];
|
|
316
|
-
}
|
|
317
|
-
|
|
318
|
-
/**
|
|
319
|
-
* Get a resource by its content hash.
|
|
320
|
-
*
|
|
321
|
-
* @param hash - The content hash to search for
|
|
322
|
-
* @returns The resource with that hash, or null if not found
|
|
323
|
-
*/
|
|
324
|
-
getResourceByHash(hash: string): Resource | null {
|
|
325
|
-
for (const versions of this.resources.values()) {
|
|
326
|
-
const found = versions.find(r => r.hash === hash);
|
|
327
|
-
if (found) {
|
|
328
|
-
return found;
|
|
329
|
-
}
|
|
330
|
-
}
|
|
331
|
-
return null;
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
/**
|
|
335
|
-
* Validate a resource object.
|
|
336
|
-
*
|
|
337
|
-
* @param resource - The resource to validate
|
|
338
|
-
* @returns ValidationResult with valid flag and any errors/warnings
|
|
339
|
-
*
|
|
340
|
-
* @example
|
|
341
|
-
* ```typescript
|
|
342
|
-
* const result = manager.validateResource(resource);
|
|
343
|
-
* if (!result.valid) {
|
|
344
|
-
* console.error('Validation errors:', result.errors);
|
|
345
|
-
* }
|
|
346
|
-
* if (result.warnings.length > 0) {
|
|
347
|
-
* console.warn('Warnings:', result.warnings);
|
|
348
|
-
* }
|
|
349
|
-
* ```
|
|
350
|
-
*/
|
|
351
|
-
validateResource(resource: Resource): ResourceValidationResult {
|
|
352
|
-
const errors: string[] = [];
|
|
353
|
-
const warnings: string[] = [];
|
|
354
|
-
|
|
355
|
-
// Required field validation
|
|
356
|
-
if (!resource) {
|
|
357
|
-
return { valid: false, errors: ['Resource is null or undefined'], warnings: [] };
|
|
358
|
-
}
|
|
359
|
-
|
|
360
|
-
if (!resource.id || typeof resource.id !== 'string') {
|
|
361
|
-
errors.push('Missing or invalid resource ID');
|
|
362
|
-
}
|
|
363
|
-
|
|
364
|
-
if (!resource.type || typeof resource.type !== 'string') {
|
|
365
|
-
errors.push('Missing or invalid resource type');
|
|
366
|
-
}
|
|
367
|
-
|
|
368
|
-
if (!resource.contentType || typeof resource.contentType !== 'string') {
|
|
369
|
-
errors.push('Missing or invalid content type');
|
|
370
|
-
} else if (this.config.strictMimeValidation && !this.isValidMimeType(resource.contentType)) {
|
|
371
|
-
errors.push(`Invalid MIME type format: ${resource.contentType}`);
|
|
372
|
-
}
|
|
373
|
-
|
|
374
|
-
if (!resource.hash || typeof resource.hash !== 'string') {
|
|
375
|
-
errors.push('Missing or invalid hash');
|
|
376
|
-
} else if (!/^[0-9a-fA-F]{64}$/.test(resource.hash)) {
|
|
377
|
-
errors.push('Invalid hash format (must be 64 character hex string)');
|
|
378
|
-
}
|
|
379
|
-
|
|
380
|
-
// Version chain validation
|
|
381
|
-
if (resource.version !== undefined) {
|
|
382
|
-
if (typeof resource.version !== 'number' || resource.version < 1) {
|
|
383
|
-
errors.push('Invalid version number (must be positive integer)');
|
|
384
|
-
}
|
|
385
|
-
|
|
386
|
-
// v1 should not have previousVersionHash
|
|
387
|
-
if (resource.version === 1 && resource.previousVersionHash) {
|
|
388
|
-
warnings.push('First version should not have previousVersionHash');
|
|
389
|
-
}
|
|
390
|
-
|
|
391
|
-
// v2+ should have previousVersionHash
|
|
392
|
-
if (resource.version > 1 && !resource.previousVersionHash) {
|
|
393
|
-
errors.push('Versions greater than 1 must have previousVersionHash');
|
|
394
|
-
}
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
// Size validation
|
|
398
|
-
if (resource.size !== undefined) {
|
|
399
|
-
if (typeof resource.size !== 'number' || resource.size < 0) {
|
|
400
|
-
errors.push('Invalid size (must be non-negative number)');
|
|
401
|
-
}
|
|
402
|
-
|
|
403
|
-
if (resource.size > this.config.defaultMaxSize) {
|
|
404
|
-
warnings.push(`Resource size (${resource.size} bytes) exceeds default maximum (${this.config.defaultMaxSize} bytes)`);
|
|
405
|
-
}
|
|
406
|
-
}
|
|
407
|
-
|
|
408
|
-
// Timestamp validation
|
|
409
|
-
if (resource.createdAt) {
|
|
410
|
-
const date = new Date(resource.createdAt);
|
|
411
|
-
if (isNaN(date.getTime())) {
|
|
412
|
-
errors.push('Invalid createdAt timestamp');
|
|
413
|
-
}
|
|
414
|
-
}
|
|
415
|
-
|
|
416
|
-
// Content hash verification (if content is present)
|
|
417
|
-
if (resource.content || resource.contentBase64) {
|
|
418
|
-
const content = resource.content
|
|
419
|
-
? Buffer.from(resource.content, 'utf-8')
|
|
420
|
-
: Buffer.from(resource.contentBase64 || '', 'base64');
|
|
421
|
-
const computedHash = this.hashContent(content);
|
|
422
|
-
|
|
423
|
-
if (computedHash !== resource.hash) {
|
|
424
|
-
errors.push(`Content hash mismatch: expected ${resource.hash}, computed ${computedHash}`);
|
|
425
|
-
}
|
|
426
|
-
}
|
|
427
|
-
|
|
428
|
-
// Check allowed content types if configured
|
|
429
|
-
if (this.config.allowedContentTypes.length > 0 &&
|
|
430
|
-
resource.contentType &&
|
|
431
|
-
!this.config.allowedContentTypes.includes(resource.contentType)) {
|
|
432
|
-
errors.push(`Content type not allowed: ${resource.contentType}`);
|
|
433
|
-
}
|
|
434
|
-
|
|
435
|
-
return {
|
|
436
|
-
valid: errors.length === 0,
|
|
437
|
-
errors,
|
|
438
|
-
warnings,
|
|
439
|
-
};
|
|
440
|
-
}
|
|
441
|
-
|
|
442
|
-
/**
|
|
443
|
-
* Verify the integrity of a resource's version chain.
|
|
444
|
-
* Ensures that previousVersionHash references form a valid chain.
|
|
445
|
-
*
|
|
446
|
-
* @param resourceId - The logical resource ID to verify
|
|
447
|
-
* @returns ResourceValidationResult indicating chain integrity
|
|
448
|
-
*/
|
|
449
|
-
verifyVersionChain(resourceId: string): ResourceValidationResult {
|
|
450
|
-
const errors: string[] = [];
|
|
451
|
-
const warnings: string[] = [];
|
|
452
|
-
|
|
453
|
-
const versions = this.resources.get(resourceId);
|
|
454
|
-
if (!versions || versions.length === 0) {
|
|
455
|
-
return { valid: false, errors: ['Resource not found'], warnings: [] };
|
|
456
|
-
}
|
|
457
|
-
|
|
458
|
-
for (let i = 0; i < versions.length; i++) {
|
|
459
|
-
const version = versions[i];
|
|
460
|
-
|
|
461
|
-
// Check version numbers are sequential
|
|
462
|
-
if ((version.version || i + 1) !== i + 1) {
|
|
463
|
-
errors.push(`Version number mismatch at index ${i}: expected ${i + 1}, got ${version.version}`);
|
|
464
|
-
}
|
|
465
|
-
|
|
466
|
-
// First version should not have previousVersionHash
|
|
467
|
-
if (i === 0) {
|
|
468
|
-
if (version.previousVersionHash) {
|
|
469
|
-
warnings.push('First version has previousVersionHash (should not exist for v1)');
|
|
470
|
-
}
|
|
471
|
-
} else {
|
|
472
|
-
// Subsequent versions must link to previous
|
|
473
|
-
const prevVersion = versions[i - 1];
|
|
474
|
-
if (!version.previousVersionHash) {
|
|
475
|
-
errors.push(`Version ${i + 1} missing previousVersionHash`);
|
|
476
|
-
} else if (version.previousVersionHash !== prevVersion.hash) {
|
|
477
|
-
errors.push(`Version ${i + 1} previousVersionHash mismatch: expected ${prevVersion.hash}, got ${version.previousVersionHash}`);
|
|
478
|
-
}
|
|
479
|
-
}
|
|
480
|
-
}
|
|
481
|
-
|
|
482
|
-
return {
|
|
483
|
-
valid: errors.length === 0,
|
|
484
|
-
errors,
|
|
485
|
-
warnings,
|
|
486
|
-
};
|
|
487
|
-
}
|
|
488
|
-
|
|
489
|
-
/**
|
|
490
|
-
* Hash content using SHA-256.
|
|
491
|
-
*
|
|
492
|
-
* @param content - Content to hash (string or Buffer)
|
|
493
|
-
* @returns Hex-encoded SHA-256 hash
|
|
494
|
-
*
|
|
495
|
-
* @example
|
|
496
|
-
* ```typescript
|
|
497
|
-
* const hash = manager.hashContent('Hello, World!');
|
|
498
|
-
* console.log(hash); // 64-character hex string
|
|
499
|
-
* ```
|
|
500
|
-
*/
|
|
501
|
-
hashContent(content: Buffer | string): string {
|
|
502
|
-
const buffer = this.toBuffer(content);
|
|
503
|
-
const hash = sha256(buffer);
|
|
504
|
-
return bytesToHex(hash);
|
|
505
|
-
}
|
|
506
|
-
|
|
507
|
-
/**
|
|
508
|
-
* Delete a resource and all its versions.
|
|
509
|
-
*
|
|
510
|
-
* @param resourceId - The resource ID to delete
|
|
511
|
-
* @returns true if deleted, false if not found
|
|
512
|
-
*/
|
|
513
|
-
deleteResource(resourceId: string): boolean {
|
|
514
|
-
return this.resources.delete(resourceId);
|
|
515
|
-
}
|
|
516
|
-
|
|
517
|
-
/**
|
|
518
|
-
* List all resource IDs managed by this instance.
|
|
519
|
-
*
|
|
520
|
-
* @returns Array of resource IDs
|
|
521
|
-
*/
|
|
522
|
-
listResourceIds(): string[] {
|
|
523
|
-
return Array.from(this.resources.keys());
|
|
524
|
-
}
|
|
525
|
-
|
|
526
|
-
/**
|
|
527
|
-
* Get the total number of resources (unique IDs) managed.
|
|
528
|
-
*
|
|
529
|
-
* @returns Number of resources
|
|
530
|
-
*/
|
|
531
|
-
getResourceCount(): number {
|
|
532
|
-
return this.resources.size;
|
|
533
|
-
}
|
|
534
|
-
|
|
535
|
-
/**
|
|
536
|
-
* Get the total number of versions across all resources.
|
|
537
|
-
*
|
|
538
|
-
* @returns Total version count
|
|
539
|
-
*/
|
|
540
|
-
getTotalVersionCount(): number {
|
|
541
|
-
let count = 0;
|
|
542
|
-
for (const versions of this.resources.values()) {
|
|
543
|
-
count += versions.length;
|
|
544
|
-
}
|
|
545
|
-
return count;
|
|
546
|
-
}
|
|
547
|
-
|
|
548
|
-
/**
|
|
549
|
-
* Import a resource from an existing AssetResource.
|
|
550
|
-
* Useful for loading resources from storage or external sources.
|
|
551
|
-
*
|
|
552
|
-
* @param assetResource - The AssetResource to import
|
|
553
|
-
* @returns The imported Resource
|
|
554
|
-
*/
|
|
555
|
-
importResource(assetResource: Resource): Resource {
|
|
556
|
-
const resourceId = assetResource.id;
|
|
557
|
-
|
|
558
|
-
// Get or create version array
|
|
559
|
-
let versions = this.resources.get(resourceId);
|
|
560
|
-
if (!versions) {
|
|
561
|
-
versions = [];
|
|
562
|
-
this.resources.set(resourceId, versions);
|
|
563
|
-
}
|
|
564
|
-
|
|
565
|
-
// Check if this version already exists (by hash)
|
|
566
|
-
const existingVersion = versions.find(v => v.hash === assetResource.hash);
|
|
567
|
-
if (existingVersion) {
|
|
568
|
-
return existingVersion;
|
|
569
|
-
}
|
|
570
|
-
|
|
571
|
-
// Add to version array (maintain order by version number)
|
|
572
|
-
const version = assetResource.version || 1;
|
|
573
|
-
const insertIndex = versions.findIndex(v => (v.version || 1) > version);
|
|
574
|
-
if (insertIndex === -1) {
|
|
575
|
-
versions.push(assetResource);
|
|
576
|
-
} else {
|
|
577
|
-
versions.splice(insertIndex, 0, assetResource);
|
|
578
|
-
}
|
|
579
|
-
|
|
580
|
-
return assetResource;
|
|
581
|
-
}
|
|
582
|
-
|
|
583
|
-
/**
|
|
584
|
-
* Export all resources as an array (for serialization).
|
|
585
|
-
*
|
|
586
|
-
* @returns Array of all resources (all versions)
|
|
587
|
-
*/
|
|
588
|
-
exportResources(): Resource[] {
|
|
589
|
-
const allResources: Resource[] = [];
|
|
590
|
-
for (const versions of this.resources.values()) {
|
|
591
|
-
allResources.push(...versions);
|
|
592
|
-
}
|
|
593
|
-
return allResources;
|
|
594
|
-
}
|
|
595
|
-
|
|
596
|
-
/**
|
|
597
|
-
* Clear all resources from this manager.
|
|
598
|
-
*/
|
|
599
|
-
clear(): void {
|
|
600
|
-
this.resources.clear();
|
|
601
|
-
}
|
|
602
|
-
|
|
603
|
-
/**
|
|
604
|
-
* Infer resource type from MIME type.
|
|
605
|
-
*
|
|
606
|
-
* @param contentType - The MIME content type
|
|
607
|
-
* @returns Inferred ResourceType
|
|
608
|
-
*/
|
|
609
|
-
static inferResourceType(contentType: string): ResourceType {
|
|
610
|
-
// Check exact match first
|
|
611
|
-
if (contentType in MIME_TYPE_MAP) {
|
|
612
|
-
return MIME_TYPE_MAP[contentType];
|
|
613
|
-
}
|
|
614
|
-
|
|
615
|
-
// Check by prefix
|
|
616
|
-
const prefix = contentType.split('/')[0];
|
|
617
|
-
switch (prefix) {
|
|
618
|
-
case 'image':
|
|
619
|
-
return 'image';
|
|
620
|
-
case 'audio':
|
|
621
|
-
return 'audio';
|
|
622
|
-
case 'video':
|
|
623
|
-
return 'video';
|
|
624
|
-
case 'text':
|
|
625
|
-
return 'text';
|
|
626
|
-
default:
|
|
627
|
-
return 'other';
|
|
628
|
-
}
|
|
629
|
-
}
|
|
630
|
-
|
|
631
|
-
/**
|
|
632
|
-
* Check if a string is a valid MIME type format.
|
|
633
|
-
*/
|
|
634
|
-
private isValidMimeType(mimeType: string): boolean {
|
|
635
|
-
return MIME_TYPE_REGEX.test(mimeType);
|
|
636
|
-
}
|
|
637
|
-
|
|
638
|
-
/**
|
|
639
|
-
* Convert content to Buffer.
|
|
640
|
-
*/
|
|
641
|
-
private toBuffer(content: Buffer | string): Buffer {
|
|
642
|
-
if (Buffer.isBuffer(content)) {
|
|
643
|
-
return content;
|
|
644
|
-
}
|
|
645
|
-
return Buffer.from(content, 'utf-8');
|
|
646
|
-
}
|
|
647
|
-
|
|
648
|
-
/**
|
|
649
|
-
* Check if content is binary (Buffer) rather than text.
|
|
650
|
-
*/
|
|
651
|
-
private isBinaryContent(content: Buffer | string): boolean {
|
|
652
|
-
return Buffer.isBuffer(content);
|
|
653
|
-
}
|
|
654
|
-
}
|
|
655
|
-
|
package/src/resources/index.ts
DELETED
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Resources module for the Originals SDK.
|
|
3
|
-
*
|
|
4
|
-
* This module provides resource management with immutable versioning,
|
|
5
|
-
* content hashing, and validation.
|
|
6
|
-
*
|
|
7
|
-
* @module resources
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
export { ResourceManager } from './ResourceManager.js';
|
|
11
|
-
export type {
|
|
12
|
-
Resource,
|
|
13
|
-
ResourceOptions,
|
|
14
|
-
ResourceUpdateOptions,
|
|
15
|
-
ResourceVersionHistory,
|
|
16
|
-
ResourceManagerConfig,
|
|
17
|
-
ResourceValidationResult,
|
|
18
|
-
ResourceType,
|
|
19
|
-
} from './types.js';
|
|
20
|
-
export { MIME_TYPE_MAP, DEFAULT_RESOURCE_CONFIG } from './types.js';
|
|
21
|
-
|