@originals/sdk 1.4.3 → 1.5.0
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/dist/adapters/FeeOracleMock.d.ts +6 -0
- package/dist/adapters/FeeOracleMock.js +8 -0
- package/dist/adapters/index.d.ts +4 -0
- package/dist/adapters/index.js +4 -0
- package/dist/adapters/providers/OrdHttpProvider.d.ts +56 -0
- package/dist/adapters/providers/OrdHttpProvider.js +110 -0
- package/dist/adapters/providers/OrdMockProvider.d.ts +70 -0
- package/dist/adapters/providers/OrdMockProvider.js +75 -0
- package/dist/adapters/types.d.ts +71 -0
- package/dist/adapters/types.js +1 -0
- package/dist/bitcoin/BitcoinManager.d.ts +15 -0
- package/dist/bitcoin/BitcoinManager.js +262 -0
- package/dist/bitcoin/BroadcastClient.d.ts +30 -0
- package/dist/bitcoin/BroadcastClient.js +35 -0
- package/dist/bitcoin/OrdinalsClient.d.ts +21 -0
- package/dist/bitcoin/OrdinalsClient.js +105 -0
- package/dist/bitcoin/PSBTBuilder.d.ts +24 -0
- package/dist/bitcoin/PSBTBuilder.js +80 -0
- package/dist/bitcoin/fee-calculation.d.ts +14 -0
- package/dist/bitcoin/fee-calculation.js +31 -0
- package/dist/bitcoin/providers/OrdNodeProvider.d.ts +38 -0
- package/dist/bitcoin/providers/OrdNodeProvider.js +67 -0
- package/dist/bitcoin/providers/OrdinalsProvider.d.ts +33 -0
- package/dist/bitcoin/providers/OrdinalsProvider.js +50 -0
- package/dist/bitcoin/providers/types.d.ts +63 -0
- package/dist/bitcoin/providers/types.js +1 -0
- package/dist/bitcoin/transactions/commit.d.ts +89 -0
- package/dist/bitcoin/transactions/commit.js +311 -0
- package/dist/bitcoin/transactions/index.d.ts +7 -0
- package/dist/bitcoin/transactions/index.js +8 -0
- package/dist/bitcoin/transfer.d.ts +9 -0
- package/dist/bitcoin/transfer.js +26 -0
- package/dist/bitcoin/utxo-selection.d.ts +78 -0
- package/dist/bitcoin/utxo-selection.js +237 -0
- package/dist/bitcoin/utxo.d.ts +26 -0
- package/dist/bitcoin/utxo.js +78 -0
- package/dist/contexts/credentials-v1.json +195 -0
- package/dist/contexts/credentials-v2-examples.json +5 -0
- package/dist/contexts/credentials-v2.json +301 -0
- package/dist/contexts/credentials.json +195 -0
- package/dist/contexts/data-integrity-v2.json +81 -0
- package/dist/contexts/dids.json +57 -0
- package/dist/contexts/ed255192020.json +93 -0
- package/dist/contexts/ordinals-plus.json +23 -0
- package/dist/contexts/originals.json +22 -0
- package/dist/core/OriginalsSDK.d.ts +158 -0
- package/dist/core/OriginalsSDK.js +274 -0
- package/dist/crypto/Multikey.d.ts +30 -0
- package/dist/crypto/Multikey.js +149 -0
- package/dist/crypto/Signer.d.ts +21 -0
- package/dist/crypto/Signer.js +196 -0
- package/dist/crypto/noble-init.d.ts +18 -0
- package/dist/crypto/noble-init.js +106 -0
- package/dist/did/BtcoDidResolver.d.ts +57 -0
- package/dist/did/BtcoDidResolver.js +166 -0
- package/dist/did/DIDManager.d.ts +101 -0
- package/dist/did/DIDManager.js +493 -0
- package/dist/did/Ed25519Verifier.d.ts +30 -0
- package/dist/did/Ed25519Verifier.js +59 -0
- package/dist/did/KeyManager.d.ts +17 -0
- package/dist/did/KeyManager.js +207 -0
- package/dist/did/WebVHManager.d.ts +100 -0
- package/dist/did/WebVHManager.js +312 -0
- package/dist/did/createBtcoDidDocument.d.ts +10 -0
- package/dist/did/createBtcoDidDocument.js +42 -0
- package/dist/did/providers/OrdinalsClientProviderAdapter.d.ts +23 -0
- package/dist/did/providers/OrdinalsClientProviderAdapter.js +51 -0
- package/dist/events/EventEmitter.d.ts +115 -0
- package/dist/events/EventEmitter.js +198 -0
- package/dist/events/index.d.ts +7 -0
- package/dist/events/index.js +6 -0
- package/dist/events/types.d.ts +286 -0
- package/dist/events/types.js +9 -0
- package/dist/examples/basic-usage.d.ts +3 -0
- package/dist/examples/basic-usage.js +62 -0
- package/dist/examples/create-module-original.d.ts +32 -0
- package/dist/examples/create-module-original.js +376 -0
- package/dist/examples/full-lifecycle-flow.d.ts +56 -0
- package/dist/examples/full-lifecycle-flow.js +419 -0
- package/dist/examples/run.d.ts +12 -0
- package/dist/examples/run.js +51 -0
- package/dist/index.d.ts +43 -0
- package/dist/index.js +52 -0
- package/dist/kinds/KindRegistry.d.ts +76 -0
- package/dist/kinds/KindRegistry.js +216 -0
- package/dist/kinds/index.d.ts +33 -0
- package/dist/kinds/index.js +36 -0
- package/dist/kinds/types.d.ts +363 -0
- package/dist/kinds/types.js +25 -0
- package/dist/kinds/validators/AgentValidator.d.ts +14 -0
- package/dist/kinds/validators/AgentValidator.js +155 -0
- package/dist/kinds/validators/AppValidator.d.ts +14 -0
- package/dist/kinds/validators/AppValidator.js +135 -0
- package/dist/kinds/validators/DatasetValidator.d.ts +14 -0
- package/dist/kinds/validators/DatasetValidator.js +148 -0
- package/dist/kinds/validators/DocumentValidator.d.ts +14 -0
- package/dist/kinds/validators/DocumentValidator.js +180 -0
- package/dist/kinds/validators/MediaValidator.d.ts +14 -0
- package/dist/kinds/validators/MediaValidator.js +172 -0
- package/dist/kinds/validators/ModuleValidator.d.ts +14 -0
- package/dist/kinds/validators/ModuleValidator.js +140 -0
- package/dist/kinds/validators/base.d.ts +96 -0
- package/dist/kinds/validators/base.js +218 -0
- package/dist/kinds/validators/index.d.ts +10 -0
- package/dist/kinds/validators/index.js +10 -0
- package/dist/lifecycle/BatchOperations.d.ts +147 -0
- package/dist/lifecycle/BatchOperations.js +251 -0
- package/dist/lifecycle/LifecycleManager.d.ts +362 -0
- package/dist/lifecycle/LifecycleManager.js +1692 -0
- package/dist/lifecycle/OriginalsAsset.d.ts +164 -0
- package/dist/lifecycle/OriginalsAsset.js +380 -0
- package/dist/lifecycle/ProvenanceQuery.d.ts +126 -0
- package/dist/lifecycle/ProvenanceQuery.js +220 -0
- package/dist/lifecycle/ResourceVersioning.d.ts +73 -0
- package/dist/lifecycle/ResourceVersioning.js +127 -0
- package/dist/migration/MigrationManager.d.ts +86 -0
- package/dist/migration/MigrationManager.js +412 -0
- package/dist/migration/audit/AuditLogger.d.ts +51 -0
- package/dist/migration/audit/AuditLogger.js +156 -0
- package/dist/migration/checkpoint/CheckpointManager.d.ts +31 -0
- package/dist/migration/checkpoint/CheckpointManager.js +96 -0
- package/dist/migration/checkpoint/CheckpointStorage.d.ts +26 -0
- package/dist/migration/checkpoint/CheckpointStorage.js +89 -0
- package/dist/migration/index.d.ts +22 -0
- package/dist/migration/index.js +27 -0
- package/dist/migration/operations/BaseMigration.d.ts +48 -0
- package/dist/migration/operations/BaseMigration.js +83 -0
- package/dist/migration/operations/PeerToBtcoMigration.d.ts +25 -0
- package/dist/migration/operations/PeerToBtcoMigration.js +67 -0
- package/dist/migration/operations/PeerToWebvhMigration.d.ts +19 -0
- package/dist/migration/operations/PeerToWebvhMigration.js +46 -0
- package/dist/migration/operations/WebvhToBtcoMigration.d.ts +25 -0
- package/dist/migration/operations/WebvhToBtcoMigration.js +67 -0
- package/dist/migration/rollback/RollbackManager.d.ts +29 -0
- package/dist/migration/rollback/RollbackManager.js +146 -0
- package/dist/migration/state/StateMachine.d.ts +25 -0
- package/dist/migration/state/StateMachine.js +76 -0
- package/dist/migration/state/StateTracker.d.ts +36 -0
- package/dist/migration/state/StateTracker.js +123 -0
- package/dist/migration/types.d.ts +306 -0
- package/dist/migration/types.js +33 -0
- package/dist/migration/validation/BitcoinValidator.d.ts +13 -0
- package/dist/migration/validation/BitcoinValidator.js +83 -0
- package/dist/migration/validation/CredentialValidator.d.ts +13 -0
- package/dist/migration/validation/CredentialValidator.js +46 -0
- package/dist/migration/validation/DIDCompatibilityValidator.d.ts +16 -0
- package/dist/migration/validation/DIDCompatibilityValidator.js +127 -0
- package/dist/migration/validation/LifecycleValidator.d.ts +10 -0
- package/dist/migration/validation/LifecycleValidator.js +52 -0
- package/dist/migration/validation/StorageValidator.d.ts +10 -0
- package/dist/migration/validation/StorageValidator.js +65 -0
- package/dist/migration/validation/ValidationPipeline.d.ts +29 -0
- package/dist/migration/validation/ValidationPipeline.js +180 -0
- package/dist/resources/ResourceManager.d.ts +231 -0
- package/dist/resources/ResourceManager.js +573 -0
- package/dist/resources/index.d.ts +11 -0
- package/dist/resources/index.js +10 -0
- package/dist/resources/types.d.ts +93 -0
- package/dist/resources/types.js +80 -0
- package/dist/storage/LocalStorageAdapter.d.ts +11 -0
- package/dist/storage/LocalStorageAdapter.js +53 -0
- package/dist/storage/MemoryStorageAdapter.d.ts +6 -0
- package/dist/storage/MemoryStorageAdapter.js +21 -0
- package/dist/storage/StorageAdapter.d.ts +16 -0
- package/dist/storage/StorageAdapter.js +1 -0
- package/dist/storage/index.d.ts +2 -0
- package/dist/storage/index.js +2 -0
- package/dist/types/bitcoin.d.ts +84 -0
- package/dist/types/bitcoin.js +1 -0
- package/dist/types/common.d.ts +82 -0
- package/dist/types/common.js +1 -0
- package/dist/types/credentials.d.ts +75 -0
- package/dist/types/credentials.js +1 -0
- package/dist/types/did.d.ts +26 -0
- package/dist/types/did.js +1 -0
- package/dist/types/index.d.ts +5 -0
- package/dist/types/index.js +5 -0
- package/dist/types/network.d.ts +78 -0
- package/dist/types/network.js +145 -0
- package/dist/utils/EventLogger.d.ts +71 -0
- package/dist/utils/EventLogger.js +232 -0
- package/dist/utils/Logger.d.ts +106 -0
- package/dist/utils/Logger.js +257 -0
- package/dist/utils/MetricsCollector.d.ts +110 -0
- package/dist/utils/MetricsCollector.js +264 -0
- package/dist/utils/bitcoin-address.d.ts +38 -0
- package/dist/utils/bitcoin-address.js +113 -0
- package/dist/utils/cbor.d.ts +2 -0
- package/dist/utils/cbor.js +9 -0
- package/dist/utils/encoding.d.ts +37 -0
- package/dist/utils/encoding.js +120 -0
- package/dist/utils/hash.d.ts +1 -0
- package/dist/utils/hash.js +5 -0
- package/dist/utils/retry.d.ts +10 -0
- package/dist/utils/retry.js +35 -0
- package/dist/utils/satoshi-validation.d.ts +60 -0
- package/dist/utils/satoshi-validation.js +156 -0
- package/dist/utils/serialization.d.ts +14 -0
- package/dist/utils/serialization.js +76 -0
- package/dist/utils/telemetry.d.ts +17 -0
- package/dist/utils/telemetry.js +24 -0
- package/dist/utils/validation.d.ts +5 -0
- package/dist/utils/validation.js +98 -0
- package/dist/vc/CredentialManager.d.ts +329 -0
- package/dist/vc/CredentialManager.js +615 -0
- package/dist/vc/Issuer.d.ts +27 -0
- package/dist/vc/Issuer.js +70 -0
- package/dist/vc/Verifier.d.ts +16 -0
- package/dist/vc/Verifier.js +50 -0
- package/dist/vc/cryptosuites/bbs.d.ts +44 -0
- package/dist/vc/cryptosuites/bbs.js +213 -0
- package/dist/vc/cryptosuites/bbsSimple.d.ts +9 -0
- package/dist/vc/cryptosuites/bbsSimple.js +12 -0
- package/dist/vc/cryptosuites/eddsa.d.ts +30 -0
- package/dist/vc/cryptosuites/eddsa.js +81 -0
- package/dist/vc/documentLoader.d.ts +16 -0
- package/dist/vc/documentLoader.js +59 -0
- package/dist/vc/proofs/data-integrity.d.ts +21 -0
- package/dist/vc/proofs/data-integrity.js +15 -0
- package/dist/vc/utils/jsonld.d.ts +2 -0
- package/dist/vc/utils/jsonld.js +15 -0
- package/package.json +2 -1
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import type { ResourceProviderLike } from '../BtcoDidResolver';
|
|
2
|
+
import { OrdinalsClient } from '../../bitcoin/OrdinalsClient';
|
|
3
|
+
export interface OrdinalsClientProviderConfig {
|
|
4
|
+
baseUrl: string;
|
|
5
|
+
fetchFn?: (url: string, init?: RequestInit) => Promise<Response>;
|
|
6
|
+
timeout?: number;
|
|
7
|
+
}
|
|
8
|
+
export declare class OrdinalsClientProviderAdapter implements ResourceProviderLike {
|
|
9
|
+
private client;
|
|
10
|
+
private readonly config;
|
|
11
|
+
constructor(client: OrdinalsClient, configOrBaseUrl: string | OrdinalsClientProviderConfig);
|
|
12
|
+
getSatInfo(satNumber: string): Promise<{
|
|
13
|
+
inscription_ids: string[];
|
|
14
|
+
}>;
|
|
15
|
+
resolveInscription(inscriptionId: string): Promise<{
|
|
16
|
+
id: string;
|
|
17
|
+
sat: number;
|
|
18
|
+
content_type: string;
|
|
19
|
+
content_url: string;
|
|
20
|
+
}>;
|
|
21
|
+
getMetadata(inscriptionId: string): Promise<any>;
|
|
22
|
+
}
|
|
23
|
+
export default OrdinalsClientProviderAdapter;
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
export class OrdinalsClientProviderAdapter {
|
|
2
|
+
constructor(client, configOrBaseUrl) {
|
|
3
|
+
this.client = client;
|
|
4
|
+
if (typeof configOrBaseUrl === 'string') {
|
|
5
|
+
this.config = { baseUrl: configOrBaseUrl };
|
|
6
|
+
}
|
|
7
|
+
else {
|
|
8
|
+
this.config = configOrBaseUrl;
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
async getSatInfo(satNumber) {
|
|
12
|
+
return this.client.getSatInfo(satNumber);
|
|
13
|
+
}
|
|
14
|
+
async resolveInscription(inscriptionId) {
|
|
15
|
+
const base = (this.config.baseUrl || '').replace(/\/$/, '');
|
|
16
|
+
if (!base) {
|
|
17
|
+
throw new Error('OrdinalsClientProviderAdapter requires a baseUrl');
|
|
18
|
+
}
|
|
19
|
+
try {
|
|
20
|
+
// Use configurable fetch function or default to global fetch
|
|
21
|
+
const fetchFn = this.config.fetchFn || fetch;
|
|
22
|
+
const timeout = this.config.timeout || 10000; // 10 second default timeout
|
|
23
|
+
const fetchOptions = {
|
|
24
|
+
headers: { 'Accept': 'application/json' },
|
|
25
|
+
signal: AbortSignal.timeout(timeout)
|
|
26
|
+
};
|
|
27
|
+
const res = await fetchFn(`${base}/inscription/${inscriptionId}`, fetchOptions);
|
|
28
|
+
if (!res.ok) {
|
|
29
|
+
// Log warning but don't throw - allow graceful degradation
|
|
30
|
+
console.warn(`Failed to resolve inscription ${inscriptionId}: HTTP ${res.status}`);
|
|
31
|
+
throw new Error(`HTTP ${res.status}: ${res.statusText}`);
|
|
32
|
+
}
|
|
33
|
+
const info = await res.json();
|
|
34
|
+
return {
|
|
35
|
+
id: info.inscription_id || inscriptionId,
|
|
36
|
+
sat: typeof info.sat === 'number' ? info.sat : Number(info.sat || 0),
|
|
37
|
+
content_type: info.content_type || 'text/plain',
|
|
38
|
+
content_url: info.content_url || `${base}/content/${inscriptionId}`
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
catch (err) {
|
|
42
|
+
// Log error for debugging but re-throw for caller to handle
|
|
43
|
+
console.warn(`Failed to resolve inscription ${inscriptionId}:`, err.message || String(err));
|
|
44
|
+
throw new Error(`Failed to resolve inscription: ${inscriptionId}`);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
async getMetadata(inscriptionId) {
|
|
48
|
+
return this.client.getMetadata(inscriptionId);
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
export default OrdinalsClientProviderAdapter;
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Type-safe EventEmitter for Originals SDK
|
|
3
|
+
*
|
|
4
|
+
* Features:
|
|
5
|
+
* - Type-safe event emission and subscription
|
|
6
|
+
* - Support for both sync and async handlers
|
|
7
|
+
* - Error isolation (one failing handler doesn't affect others)
|
|
8
|
+
* - Event namespacing
|
|
9
|
+
* - Performance optimized (<1ms overhead per event)
|
|
10
|
+
*/
|
|
11
|
+
import type { EventHandler, EventTypeMap } from './types';
|
|
12
|
+
/**
|
|
13
|
+
* EventEmitter class for managing event subscriptions and emissions
|
|
14
|
+
*/
|
|
15
|
+
export declare class EventEmitter {
|
|
16
|
+
private handlers;
|
|
17
|
+
private onceHandlers;
|
|
18
|
+
/**
|
|
19
|
+
* Subscribe to an event
|
|
20
|
+
*
|
|
21
|
+
* @param eventType - The type of event to listen for
|
|
22
|
+
* @param handler - The handler function to call when the event is emitted
|
|
23
|
+
* @returns A function to unsubscribe the handler
|
|
24
|
+
*
|
|
25
|
+
* @example
|
|
26
|
+
* ```typescript
|
|
27
|
+
* const unsubscribe = emitter.on('asset:created', (event) => {
|
|
28
|
+
* console.log('Asset created:', event.asset.id);
|
|
29
|
+
* });
|
|
30
|
+
*
|
|
31
|
+
* // Later, to unsubscribe:
|
|
32
|
+
* unsubscribe();
|
|
33
|
+
* ```
|
|
34
|
+
*/
|
|
35
|
+
on<K extends keyof EventTypeMap>(eventType: K, handler: EventHandler<EventTypeMap[K]>): () => void;
|
|
36
|
+
/**
|
|
37
|
+
* Subscribe to an event for a single emission
|
|
38
|
+
*
|
|
39
|
+
* @param eventType - The type of event to listen for
|
|
40
|
+
* @param handler - The handler function to call when the event is emitted (only once)
|
|
41
|
+
* @returns A function to unsubscribe the handler
|
|
42
|
+
*
|
|
43
|
+
* @example
|
|
44
|
+
* ```typescript
|
|
45
|
+
* emitter.once('asset:migrated', (event) => {
|
|
46
|
+
* console.log('Asset migrated once:', event.asset.id);
|
|
47
|
+
* });
|
|
48
|
+
* ```
|
|
49
|
+
*/
|
|
50
|
+
once<K extends keyof EventTypeMap>(eventType: K, handler: EventHandler<EventTypeMap[K]>): () => void;
|
|
51
|
+
/**
|
|
52
|
+
* Unsubscribe from an event
|
|
53
|
+
*
|
|
54
|
+
* @param eventType - The type of event to stop listening for
|
|
55
|
+
* @param handler - The handler function to remove
|
|
56
|
+
*
|
|
57
|
+
* @example
|
|
58
|
+
* ```typescript
|
|
59
|
+
* const handler = (event) => console.log(event);
|
|
60
|
+
* emitter.on('asset:created', handler);
|
|
61
|
+
* emitter.off('asset:created', handler);
|
|
62
|
+
* ```
|
|
63
|
+
*/
|
|
64
|
+
off<K extends keyof EventTypeMap>(eventType: K, handler: EventHandler<EventTypeMap[K]>): void;
|
|
65
|
+
/**
|
|
66
|
+
* Emit an event to all subscribed handlers
|
|
67
|
+
*
|
|
68
|
+
* @param event - The event to emit
|
|
69
|
+
*
|
|
70
|
+
* Features:
|
|
71
|
+
* - Handlers are called in subscription order
|
|
72
|
+
* - Async handlers are awaited
|
|
73
|
+
* - Errors in handlers are isolated (logged but don't affect other handlers)
|
|
74
|
+
* - Once handlers are automatically removed after execution
|
|
75
|
+
*
|
|
76
|
+
* @example
|
|
77
|
+
* ```typescript
|
|
78
|
+
* emitter.emit({
|
|
79
|
+
* type: 'asset:created',
|
|
80
|
+
* timestamp: new Date().toISOString(),
|
|
81
|
+
* asset: { id: 'did:peer:123', layer: 'did:peer', resourceCount: 1 }
|
|
82
|
+
* });
|
|
83
|
+
* ```
|
|
84
|
+
*/
|
|
85
|
+
emit<K extends keyof EventTypeMap>(event: EventTypeMap[K]): Promise<void>;
|
|
86
|
+
/**
|
|
87
|
+
* Remove all handlers for a specific event type, or all handlers if no type specified
|
|
88
|
+
*
|
|
89
|
+
* @param eventType - Optional event type to clear handlers for
|
|
90
|
+
*
|
|
91
|
+
* @example
|
|
92
|
+
* ```typescript
|
|
93
|
+
* // Remove all handlers for 'asset:created'
|
|
94
|
+
* emitter.removeAllListeners('asset:created');
|
|
95
|
+
*
|
|
96
|
+
* // Remove all handlers for all events
|
|
97
|
+
* emitter.removeAllListeners();
|
|
98
|
+
* ```
|
|
99
|
+
*/
|
|
100
|
+
removeAllListeners<K extends keyof EventTypeMap>(eventType?: K): void;
|
|
101
|
+
/**
|
|
102
|
+
* Get the number of handlers for a specific event type
|
|
103
|
+
*
|
|
104
|
+
* @param eventType - The event type to check
|
|
105
|
+
* @returns The number of handlers subscribed to this event type
|
|
106
|
+
*/
|
|
107
|
+
listenerCount<K extends keyof EventTypeMap>(eventType: K): number;
|
|
108
|
+
/**
|
|
109
|
+
* Check if there are any handlers for a specific event type
|
|
110
|
+
*
|
|
111
|
+
* @param eventType - The event type to check
|
|
112
|
+
* @returns True if there are any handlers subscribed
|
|
113
|
+
*/
|
|
114
|
+
hasListeners<K extends keyof EventTypeMap>(eventType: K): boolean;
|
|
115
|
+
}
|
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Type-safe EventEmitter for Originals SDK
|
|
3
|
+
*
|
|
4
|
+
* Features:
|
|
5
|
+
* - Type-safe event emission and subscription
|
|
6
|
+
* - Support for both sync and async handlers
|
|
7
|
+
* - Error isolation (one failing handler doesn't affect others)
|
|
8
|
+
* - Event namespacing
|
|
9
|
+
* - Performance optimized (<1ms overhead per event)
|
|
10
|
+
*/
|
|
11
|
+
/**
|
|
12
|
+
* EventEmitter class for managing event subscriptions and emissions
|
|
13
|
+
*/
|
|
14
|
+
export class EventEmitter {
|
|
15
|
+
constructor() {
|
|
16
|
+
this.handlers = new Map();
|
|
17
|
+
this.onceHandlers = new Map();
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Subscribe to an event
|
|
21
|
+
*
|
|
22
|
+
* @param eventType - The type of event to listen for
|
|
23
|
+
* @param handler - The handler function to call when the event is emitted
|
|
24
|
+
* @returns A function to unsubscribe the handler
|
|
25
|
+
*
|
|
26
|
+
* @example
|
|
27
|
+
* ```typescript
|
|
28
|
+
* const unsubscribe = emitter.on('asset:created', (event) => {
|
|
29
|
+
* console.log('Asset created:', event.asset.id);
|
|
30
|
+
* });
|
|
31
|
+
*
|
|
32
|
+
* // Later, to unsubscribe:
|
|
33
|
+
* unsubscribe();
|
|
34
|
+
* ```
|
|
35
|
+
*/
|
|
36
|
+
on(eventType, handler) {
|
|
37
|
+
if (!this.handlers.has(eventType)) {
|
|
38
|
+
this.handlers.set(eventType, new Set());
|
|
39
|
+
}
|
|
40
|
+
this.handlers.get(eventType).add(handler);
|
|
41
|
+
// Return unsubscribe function
|
|
42
|
+
return () => this.off(eventType, handler);
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Subscribe to an event for a single emission
|
|
46
|
+
*
|
|
47
|
+
* @param eventType - The type of event to listen for
|
|
48
|
+
* @param handler - The handler function to call when the event is emitted (only once)
|
|
49
|
+
* @returns A function to unsubscribe the handler
|
|
50
|
+
*
|
|
51
|
+
* @example
|
|
52
|
+
* ```typescript
|
|
53
|
+
* emitter.once('asset:migrated', (event) => {
|
|
54
|
+
* console.log('Asset migrated once:', event.asset.id);
|
|
55
|
+
* });
|
|
56
|
+
* ```
|
|
57
|
+
*/
|
|
58
|
+
once(eventType, handler) {
|
|
59
|
+
if (!this.onceHandlers.has(eventType)) {
|
|
60
|
+
this.onceHandlers.set(eventType, new Set());
|
|
61
|
+
}
|
|
62
|
+
this.onceHandlers.get(eventType).add(handler);
|
|
63
|
+
// Return unsubscribe function
|
|
64
|
+
return () => {
|
|
65
|
+
const handlers = this.onceHandlers.get(eventType);
|
|
66
|
+
if (handlers) {
|
|
67
|
+
handlers.delete(handler);
|
|
68
|
+
}
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Unsubscribe from an event
|
|
73
|
+
*
|
|
74
|
+
* @param eventType - The type of event to stop listening for
|
|
75
|
+
* @param handler - The handler function to remove
|
|
76
|
+
*
|
|
77
|
+
* @example
|
|
78
|
+
* ```typescript
|
|
79
|
+
* const handler = (event) => console.log(event);
|
|
80
|
+
* emitter.on('asset:created', handler);
|
|
81
|
+
* emitter.off('asset:created', handler);
|
|
82
|
+
* ```
|
|
83
|
+
*/
|
|
84
|
+
off(eventType, handler) {
|
|
85
|
+
const handlers = this.handlers.get(eventType);
|
|
86
|
+
if (handlers) {
|
|
87
|
+
handlers.delete(handler);
|
|
88
|
+
}
|
|
89
|
+
const onceHandlers = this.onceHandlers.get(eventType);
|
|
90
|
+
if (onceHandlers) {
|
|
91
|
+
onceHandlers.delete(handler);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Emit an event to all subscribed handlers
|
|
96
|
+
*
|
|
97
|
+
* @param event - The event to emit
|
|
98
|
+
*
|
|
99
|
+
* Features:
|
|
100
|
+
* - Handlers are called in subscription order
|
|
101
|
+
* - Async handlers are awaited
|
|
102
|
+
* - Errors in handlers are isolated (logged but don't affect other handlers)
|
|
103
|
+
* - Once handlers are automatically removed after execution
|
|
104
|
+
*
|
|
105
|
+
* @example
|
|
106
|
+
* ```typescript
|
|
107
|
+
* emitter.emit({
|
|
108
|
+
* type: 'asset:created',
|
|
109
|
+
* timestamp: new Date().toISOString(),
|
|
110
|
+
* asset: { id: 'did:peer:123', layer: 'did:peer', resourceCount: 1 }
|
|
111
|
+
* });
|
|
112
|
+
* ```
|
|
113
|
+
*/
|
|
114
|
+
async emit(event) {
|
|
115
|
+
const eventType = event.type;
|
|
116
|
+
// Get regular handlers
|
|
117
|
+
const handlers = this.handlers.get(eventType);
|
|
118
|
+
if (handlers && handlers.size > 0) {
|
|
119
|
+
// Create array to avoid modification during iteration
|
|
120
|
+
const handlerArray = Array.from(handlers);
|
|
121
|
+
for (const handler of handlerArray) {
|
|
122
|
+
try {
|
|
123
|
+
await handler(event);
|
|
124
|
+
}
|
|
125
|
+
catch (error) {
|
|
126
|
+
// Error isolation: log but don't throw
|
|
127
|
+
// This ensures one failing handler doesn't affect others
|
|
128
|
+
if (typeof console !== 'undefined' && console.error) {
|
|
129
|
+
console.error(`Event handler error for ${eventType}:`, error);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
// Get and execute once handlers
|
|
135
|
+
const onceHandlers = this.onceHandlers.get(eventType);
|
|
136
|
+
if (onceHandlers && onceHandlers.size > 0) {
|
|
137
|
+
// Create array to avoid modification during iteration
|
|
138
|
+
const onceHandlerArray = Array.from(onceHandlers);
|
|
139
|
+
// Clear once handlers before execution
|
|
140
|
+
this.onceHandlers.delete(eventType);
|
|
141
|
+
for (const handler of onceHandlerArray) {
|
|
142
|
+
try {
|
|
143
|
+
await handler(event);
|
|
144
|
+
}
|
|
145
|
+
catch (error) {
|
|
146
|
+
// Error isolation: log but don't throw
|
|
147
|
+
if (typeof console !== 'undefined' && console.error) {
|
|
148
|
+
console.error(`Event handler error (once) for ${eventType}:`, error);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
/**
|
|
155
|
+
* Remove all handlers for a specific event type, or all handlers if no type specified
|
|
156
|
+
*
|
|
157
|
+
* @param eventType - Optional event type to clear handlers for
|
|
158
|
+
*
|
|
159
|
+
* @example
|
|
160
|
+
* ```typescript
|
|
161
|
+
* // Remove all handlers for 'asset:created'
|
|
162
|
+
* emitter.removeAllListeners('asset:created');
|
|
163
|
+
*
|
|
164
|
+
* // Remove all handlers for all events
|
|
165
|
+
* emitter.removeAllListeners();
|
|
166
|
+
* ```
|
|
167
|
+
*/
|
|
168
|
+
removeAllListeners(eventType) {
|
|
169
|
+
if (eventType) {
|
|
170
|
+
this.handlers.delete(eventType);
|
|
171
|
+
this.onceHandlers.delete(eventType);
|
|
172
|
+
}
|
|
173
|
+
else {
|
|
174
|
+
this.handlers.clear();
|
|
175
|
+
this.onceHandlers.clear();
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
/**
|
|
179
|
+
* Get the number of handlers for a specific event type
|
|
180
|
+
*
|
|
181
|
+
* @param eventType - The event type to check
|
|
182
|
+
* @returns The number of handlers subscribed to this event type
|
|
183
|
+
*/
|
|
184
|
+
listenerCount(eventType) {
|
|
185
|
+
const handlers = this.handlers.get(eventType);
|
|
186
|
+
const onceHandlers = this.onceHandlers.get(eventType);
|
|
187
|
+
return (handlers?.size || 0) + (onceHandlers?.size || 0);
|
|
188
|
+
}
|
|
189
|
+
/**
|
|
190
|
+
* Check if there are any handlers for a specific event type
|
|
191
|
+
*
|
|
192
|
+
* @param eventType - The event type to check
|
|
193
|
+
* @returns True if there are any handlers subscribed
|
|
194
|
+
*/
|
|
195
|
+
hasListeners(eventType) {
|
|
196
|
+
return this.listenerCount(eventType) > 0;
|
|
197
|
+
}
|
|
198
|
+
}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Events module for the Originals SDK
|
|
3
|
+
*
|
|
4
|
+
* Provides a type-safe event system for tracking asset lifecycle operations
|
|
5
|
+
*/
|
|
6
|
+
export { EventEmitter } from './EventEmitter';
|
|
7
|
+
export type { BaseEvent, AssetCreatedEvent, AssetMigratedEvent, AssetTransferredEvent, ResourcePublishedEvent, CredentialIssuedEvent, VerificationCompletedEvent, OriginalsEvent, EventHandler, EventTypeMap } from './types';
|
|
@@ -0,0 +1,286 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event types for the Originals SDK asset lifecycle operations
|
|
3
|
+
*
|
|
4
|
+
* All events follow a consistent structure:
|
|
5
|
+
* - type: Namespaced event name (e.g., 'asset:created')
|
|
6
|
+
* - timestamp: ISO 8601 timestamp of when the event occurred
|
|
7
|
+
* - Specific payload data relevant to the event
|
|
8
|
+
*/
|
|
9
|
+
import { LayerType } from '../types';
|
|
10
|
+
/**
|
|
11
|
+
* Base event interface that all events extend
|
|
12
|
+
*/
|
|
13
|
+
export interface BaseEvent {
|
|
14
|
+
type: string;
|
|
15
|
+
timestamp: string;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Emitted when a new asset is created
|
|
19
|
+
*/
|
|
20
|
+
export interface AssetCreatedEvent extends BaseEvent {
|
|
21
|
+
type: 'asset:created';
|
|
22
|
+
asset: {
|
|
23
|
+
id: string;
|
|
24
|
+
layer: LayerType;
|
|
25
|
+
resourceCount: number;
|
|
26
|
+
createdAt: string;
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Emitted when an asset migrates between layers
|
|
31
|
+
*
|
|
32
|
+
* Note: The `details` field is populated differently based on the target layer:
|
|
33
|
+
* - For did:webvh migrations: `details` is undefined (web publishing has no transaction details)
|
|
34
|
+
* - For did:btco migrations: `details` includes Bitcoin transaction information
|
|
35
|
+
* (transactionId, inscriptionId, satoshi, commitTxId, revealTxId, feeRate)
|
|
36
|
+
*/
|
|
37
|
+
export interface AssetMigratedEvent extends BaseEvent {
|
|
38
|
+
type: 'asset:migrated';
|
|
39
|
+
asset: {
|
|
40
|
+
id: string;
|
|
41
|
+
fromLayer: LayerType;
|
|
42
|
+
toLayer: LayerType;
|
|
43
|
+
};
|
|
44
|
+
details?: {
|
|
45
|
+
transactionId?: string;
|
|
46
|
+
inscriptionId?: string;
|
|
47
|
+
satoshi?: string;
|
|
48
|
+
commitTxId?: string;
|
|
49
|
+
revealTxId?: string;
|
|
50
|
+
feeRate?: number;
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Emitted when an asset's ownership is transferred
|
|
55
|
+
*/
|
|
56
|
+
export interface AssetTransferredEvent extends BaseEvent {
|
|
57
|
+
type: 'asset:transferred';
|
|
58
|
+
asset: {
|
|
59
|
+
id: string;
|
|
60
|
+
layer: LayerType;
|
|
61
|
+
};
|
|
62
|
+
from: string;
|
|
63
|
+
to: string;
|
|
64
|
+
transactionId: string;
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Emitted when a resource is published to web storage
|
|
68
|
+
*/
|
|
69
|
+
export interface ResourcePublishedEvent extends BaseEvent {
|
|
70
|
+
type: 'resource:published';
|
|
71
|
+
asset: {
|
|
72
|
+
id: string;
|
|
73
|
+
};
|
|
74
|
+
resource: {
|
|
75
|
+
id: string;
|
|
76
|
+
url: string;
|
|
77
|
+
contentType: string;
|
|
78
|
+
hash: string;
|
|
79
|
+
};
|
|
80
|
+
publisherDid: string;
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Emitted when a verifiable credential is issued for an asset
|
|
84
|
+
*/
|
|
85
|
+
export interface CredentialIssuedEvent extends BaseEvent {
|
|
86
|
+
type: 'credential:issued';
|
|
87
|
+
asset: {
|
|
88
|
+
id: string;
|
|
89
|
+
};
|
|
90
|
+
credential: {
|
|
91
|
+
type: string[];
|
|
92
|
+
issuer: string;
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Emitted when a new resource version is created
|
|
97
|
+
*/
|
|
98
|
+
export interface ResourceVersionCreatedEvent extends BaseEvent {
|
|
99
|
+
type: 'resource:version:created';
|
|
100
|
+
asset: {
|
|
101
|
+
id: string;
|
|
102
|
+
};
|
|
103
|
+
resource: {
|
|
104
|
+
id: string;
|
|
105
|
+
fromVersion: number;
|
|
106
|
+
toVersion: number;
|
|
107
|
+
fromHash: string;
|
|
108
|
+
toHash: string;
|
|
109
|
+
};
|
|
110
|
+
changes?: string;
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Emitted when asset verification is completed
|
|
114
|
+
*/
|
|
115
|
+
export interface VerificationCompletedEvent extends BaseEvent {
|
|
116
|
+
type: 'verification:completed';
|
|
117
|
+
asset: {
|
|
118
|
+
id: string;
|
|
119
|
+
};
|
|
120
|
+
result: boolean;
|
|
121
|
+
checks?: {
|
|
122
|
+
didDocument: boolean;
|
|
123
|
+
resources: boolean;
|
|
124
|
+
credentials: boolean;
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Emitted when a batch operation starts
|
|
129
|
+
*/
|
|
130
|
+
export interface BatchStartedEvent extends BaseEvent {
|
|
131
|
+
type: 'batch:started';
|
|
132
|
+
operation: 'create' | 'publish' | 'inscribe' | 'transfer';
|
|
133
|
+
batchId: string;
|
|
134
|
+
itemCount: number;
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* Emitted when a batch operation completes successfully
|
|
138
|
+
*/
|
|
139
|
+
export interface BatchCompletedEvent extends BaseEvent {
|
|
140
|
+
type: 'batch:completed';
|
|
141
|
+
batchId: string;
|
|
142
|
+
operation: string;
|
|
143
|
+
results: {
|
|
144
|
+
successful: number;
|
|
145
|
+
failed: number;
|
|
146
|
+
totalDuration: number;
|
|
147
|
+
costSavings?: {
|
|
148
|
+
amount: number;
|
|
149
|
+
percentage: number;
|
|
150
|
+
};
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Emitted when a batch operation fails
|
|
155
|
+
*/
|
|
156
|
+
export interface BatchFailedEvent extends BaseEvent {
|
|
157
|
+
type: 'batch:failed';
|
|
158
|
+
batchId: string;
|
|
159
|
+
operation: string;
|
|
160
|
+
error: string;
|
|
161
|
+
partialResults?: {
|
|
162
|
+
successful: number;
|
|
163
|
+
failed: number;
|
|
164
|
+
};
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Emitted when a migration starts
|
|
168
|
+
*/
|
|
169
|
+
export interface MigrationStartedEvent extends BaseEvent {
|
|
170
|
+
type: 'migration:started';
|
|
171
|
+
migrationId: string;
|
|
172
|
+
sourceDid: string;
|
|
173
|
+
targetLayer: string;
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* Emitted when migration validation completes
|
|
177
|
+
*/
|
|
178
|
+
export interface MigrationValidatedEvent extends BaseEvent {
|
|
179
|
+
type: 'migration:validated';
|
|
180
|
+
migrationId: string;
|
|
181
|
+
valid: boolean;
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Emitted when migration checkpoint is created
|
|
185
|
+
*/
|
|
186
|
+
export interface MigrationCheckpointedEvent extends BaseEvent {
|
|
187
|
+
type: 'migration:checkpointed';
|
|
188
|
+
migrationId: string;
|
|
189
|
+
checkpointId: string;
|
|
190
|
+
}
|
|
191
|
+
/**
|
|
192
|
+
* Emitted when migration enters in-progress state
|
|
193
|
+
*/
|
|
194
|
+
export interface MigrationInProgressEvent extends BaseEvent {
|
|
195
|
+
type: 'migration:in_progress';
|
|
196
|
+
migrationId: string;
|
|
197
|
+
currentOperation: string;
|
|
198
|
+
progress: number;
|
|
199
|
+
}
|
|
200
|
+
/**
|
|
201
|
+
* Emitted when migration enters anchoring state (Bitcoin)
|
|
202
|
+
*/
|
|
203
|
+
export interface MigrationAnchoringEvent extends BaseEvent {
|
|
204
|
+
type: 'migration:anchoring';
|
|
205
|
+
migrationId: string;
|
|
206
|
+
transactionId?: string;
|
|
207
|
+
}
|
|
208
|
+
/**
|
|
209
|
+
* Emitted when migration completes successfully
|
|
210
|
+
*/
|
|
211
|
+
export interface MigrationCompletedEvent extends BaseEvent {
|
|
212
|
+
type: 'migration:completed';
|
|
213
|
+
migrationId: string;
|
|
214
|
+
sourceDid: string;
|
|
215
|
+
targetDid: string;
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Emitted when migration fails
|
|
219
|
+
*/
|
|
220
|
+
export interface MigrationFailedEvent extends BaseEvent {
|
|
221
|
+
type: 'migration:failed';
|
|
222
|
+
migrationId: string;
|
|
223
|
+
error: any;
|
|
224
|
+
}
|
|
225
|
+
/**
|
|
226
|
+
* Emitted when migration is rolled back
|
|
227
|
+
*/
|
|
228
|
+
export interface MigrationRolledbackEvent extends BaseEvent {
|
|
229
|
+
type: 'migration:rolledback';
|
|
230
|
+
migrationId: string;
|
|
231
|
+
checkpointId: string;
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Emitted when migration enters quarantine state
|
|
235
|
+
*/
|
|
236
|
+
export interface MigrationQuarantineEvent extends BaseEvent {
|
|
237
|
+
type: 'migration:quarantine';
|
|
238
|
+
migrationId: string;
|
|
239
|
+
checkpointId: string;
|
|
240
|
+
reason: string;
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* Emitted during batch operations to report progress
|
|
244
|
+
*/
|
|
245
|
+
export interface BatchProgressEvent extends BaseEvent {
|
|
246
|
+
type: 'batch:progress';
|
|
247
|
+
batchId: string;
|
|
248
|
+
operation: string;
|
|
249
|
+
progress: number;
|
|
250
|
+
completed: number;
|
|
251
|
+
failed: number;
|
|
252
|
+
total: number;
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* Union type of all possible events
|
|
256
|
+
*/
|
|
257
|
+
export type OriginalsEvent = AssetCreatedEvent | AssetMigratedEvent | AssetTransferredEvent | ResourcePublishedEvent | CredentialIssuedEvent | VerificationCompletedEvent | BatchStartedEvent | BatchCompletedEvent | BatchFailedEvent | BatchProgressEvent | ResourceVersionCreatedEvent | MigrationStartedEvent | MigrationValidatedEvent | MigrationCheckpointedEvent | MigrationInProgressEvent | MigrationAnchoringEvent | MigrationCompletedEvent | MigrationFailedEvent | MigrationRolledbackEvent | MigrationQuarantineEvent;
|
|
258
|
+
/**
|
|
259
|
+
* Event handler function type
|
|
260
|
+
*/
|
|
261
|
+
export type EventHandler<T extends OriginalsEvent = OriginalsEvent> = (event: T) => void | Promise<void>;
|
|
262
|
+
/**
|
|
263
|
+
* Map of event types to their specific event interfaces
|
|
264
|
+
*/
|
|
265
|
+
export interface EventTypeMap {
|
|
266
|
+
'asset:created': AssetCreatedEvent;
|
|
267
|
+
'asset:migrated': AssetMigratedEvent;
|
|
268
|
+
'asset:transferred': AssetTransferredEvent;
|
|
269
|
+
'resource:published': ResourcePublishedEvent;
|
|
270
|
+
'credential:issued': CredentialIssuedEvent;
|
|
271
|
+
'verification:completed': VerificationCompletedEvent;
|
|
272
|
+
'batch:started': BatchStartedEvent;
|
|
273
|
+
'batch:completed': BatchCompletedEvent;
|
|
274
|
+
'batch:failed': BatchFailedEvent;
|
|
275
|
+
'batch:progress': BatchProgressEvent;
|
|
276
|
+
'resource:version:created': ResourceVersionCreatedEvent;
|
|
277
|
+
'migration:started': MigrationStartedEvent;
|
|
278
|
+
'migration:validated': MigrationValidatedEvent;
|
|
279
|
+
'migration:checkpointed': MigrationCheckpointedEvent;
|
|
280
|
+
'migration:in_progress': MigrationInProgressEvent;
|
|
281
|
+
'migration:anchoring': MigrationAnchoringEvent;
|
|
282
|
+
'migration:completed': MigrationCompletedEvent;
|
|
283
|
+
'migration:failed': MigrationFailedEvent;
|
|
284
|
+
'migration:rolledback': MigrationRolledbackEvent;
|
|
285
|
+
'migration:quarantine': MigrationQuarantineEvent;
|
|
286
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event types for the Originals SDK asset lifecycle operations
|
|
3
|
+
*
|
|
4
|
+
* All events follow a consistent structure:
|
|
5
|
+
* - type: Namespaced event name (e.g., 'asset:created')
|
|
6
|
+
* - timestamp: ISO 8601 timestamp of when the event occurred
|
|
7
|
+
* - Specific payload data relevant to the event
|
|
8
|
+
*/
|
|
9
|
+
export {};
|