@originals/sdk 1.4.2 → 1.4.5
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/{src/adapters/index.ts → dist/adapters/index.d.ts} +0 -1
- 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/{src/bitcoin/fee-calculation.ts → dist/bitcoin/fee-calculation.js} +5 -12
- 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/{src/bitcoin/transactions/index.ts → dist/bitcoin/transactions/index.js} +1 -6
- 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/{src/kinds/index.ts → dist/kinds/index.js} +6 -44
- 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/{src/kinds/validators/index.ts → dist/kinds/validators/index.d.ts} +0 -2
- 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/{src/migration/index.ts → dist/migration/index.js} +0 -6
- 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/{src/resources/index.ts → dist/resources/index.d.ts} +3 -13
- 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/{src/storage/index.ts → dist/storage/index.d.ts} +0 -1
- 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/{src/types/index.ts → dist/types/index.d.ts} +0 -2
- 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 +5 -1
- package/.eslintrc.json +0 -33
- package/src/adapters/FeeOracleMock.ts +0 -9
- 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 -330
- package/src/bitcoin/BroadcastClient.ts +0 -54
- package/src/bitcoin/OrdinalsClient.ts +0 -119
- package/src/bitcoin/PSBTBuilder.ts +0 -106
- 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/transfer.ts +0 -43
- package/src/bitcoin/utxo-selection.ts +0 -322
- package/src/bitcoin/utxo.ts +0 -113
- 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 -416
- package/src/crypto/Multikey.ts +0 -194
- package/src/crypto/Signer.ts +0 -254
- package/src/crypto/noble-init.ts +0 -121
- package/src/did/BtcoDidResolver.ts +0 -227
- package/src/did/DIDManager.ts +0 -694
- package/src/did/Ed25519Verifier.ts +0 -68
- package/src/did/KeyManager.ts +0 -236
- package/src/did/WebVHManager.ts +0 -498
- package/src/did/createBtcoDidDocument.ts +0 -59
- 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 -150
- package/src/kinds/KindRegistry.ts +0 -290
- 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/lifecycle/BatchOperations.ts +0 -373
- package/src/lifecycle/LifecycleManager.ts +0 -2126
- 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 -527
- 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/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 -344
- 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/types.ts +0 -202
- package/src/storage/LocalStorageAdapter.ts +0 -61
- package/src/storage/MemoryStorageAdapter.ts +0 -29
- package/src/storage/StorageAdapter.ts +0 -25
- package/src/types/bitcoin.ts +0 -98
- package/src/types/common.ts +0 -92
- package/src/types/credentials.ts +0 -88
- package/src/types/did.ts +0 -31
- package/src/types/external-shims.d.ts +0 -53
- package/src/types/network.ts +0 -175
- package/src/utils/EventLogger.ts +0 -298
- package/src/utils/Logger.ts +0 -322
- package/src/utils/MetricsCollector.ts +0 -358
- package/src/utils/bitcoin-address.ts +0 -130
- package/src/utils/cbor.ts +0 -12
- package/src/utils/encoding.ts +0 -127
- package/src/utils/hash.ts +0 -6
- package/src/utils/retry.ts +0 -46
- package/src/utils/satoshi-validation.ts +0 -196
- package/src/utils/serialization.ts +0 -96
- package/src/utils/telemetry.ts +0 -40
- package/src/utils/validation.ts +0 -119
- package/src/vc/CredentialManager.ts +0 -918
- package/src/vc/Issuer.ts +0 -100
- package/src/vc/Verifier.ts +0 -47
- 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 -67
- package/src/vc/proofs/data-integrity.ts +0 -33
- package/src/vc/utils/jsonld.ts +0 -18
- package/tests/__mocks__/bbs-signatures.js +0 -17
- package/tests/__mocks__/mf-base58.js +0 -24
- package/tests/fixtures/did-documents.ts +0 -247
- package/tests/index.test.ts +0 -21
- package/tests/integration/BatchOperations.test.ts +0 -531
- package/tests/integration/CompleteLifecycle.e2e.test.ts +0 -735
- package/tests/integration/CredentialManager.test.ts +0 -42
- package/tests/integration/DIDManager.test.ts +0 -41
- package/tests/integration/DidPeerToWebVhFlow.test.ts +0 -351
- package/tests/integration/Events.test.ts +0 -435
- package/tests/integration/Lifecycle.transfer.btco.integration.test.ts +0 -25
- package/tests/integration/LifecycleManager.test.ts +0 -21
- package/tests/integration/MultikeyFlow.test.ts +0 -52
- package/tests/integration/TelemetryIntegration.test.ts +0 -395
- package/tests/integration/WebVhPublish.test.ts +0 -48
- package/tests/integration/createTypedOriginal.test.ts +0 -379
- package/tests/integration/migration/peer-to-webvh.test.ts +0 -172
- package/tests/manual/test-commit-creation.ts +0 -323
- package/tests/mocks/MockKeyStore.ts +0 -38
- package/tests/mocks/adapters/MemoryStorageAdapter.ts +0 -24
- package/tests/mocks/adapters/MockFeeOracle.ts +0 -11
- package/tests/mocks/adapters/MockOrdinalsProvider.ts +0 -76
- package/tests/mocks/adapters/OrdMockProvider.test.ts +0 -176
- package/tests/mocks/adapters/index.ts +0 -6
- package/tests/performance/BatchOperations.perf.test.ts +0 -403
- package/tests/performance/logging.perf.test.ts +0 -336
- package/tests/sdk.test.ts +0 -43
- package/tests/security/bitcoin-penetration-tests.test.ts +0 -622
- package/tests/setup.bun.ts +0 -69
- package/tests/setup.jest.ts +0 -23
- package/tests/stress/batch-operations-stress.test.ts +0 -571
- package/tests/unit/adapters/FeeOracleMock.test.ts +0 -40
- package/tests/unit/bitcoin/BitcoinManager.test.ts +0 -293
- package/tests/unit/bitcoin/BroadcastClient.test.ts +0 -52
- package/tests/unit/bitcoin/OrdNodeProvider.test.ts +0 -53
- package/tests/unit/bitcoin/OrdinalsClient.test.ts +0 -381
- package/tests/unit/bitcoin/OrdinalsClientProvider.test.ts +0 -102
- package/tests/unit/bitcoin/PSBTBuilder.test.ts +0 -84
- package/tests/unit/bitcoin/fee-calculation.test.ts +0 -261
- package/tests/unit/bitcoin/transactions/commit.test.ts +0 -649
- package/tests/unit/bitcoin/transfer.test.ts +0 -31
- package/tests/unit/bitcoin/utxo-selection-new.test.ts +0 -502
- package/tests/unit/bitcoin/utxo.more.test.ts +0 -39
- package/tests/unit/bitcoin/utxo.selection.test.ts +0 -38
- package/tests/unit/core/OriginalsSDK.test.ts +0 -152
- package/tests/unit/crypto/Multikey.test.ts +0 -206
- package/tests/unit/crypto/Signer.test.ts +0 -408
- package/tests/unit/did/BtcoDidResolver.test.ts +0 -611
- package/tests/unit/did/DIDManager.more.test.ts +0 -43
- package/tests/unit/did/DIDManager.test.ts +0 -185
- package/tests/unit/did/Ed25519Verifier.test.ts +0 -160
- package/tests/unit/did/KeyManager.test.ts +0 -452
- package/tests/unit/did/OrdinalsClientProviderAdapter.test.ts +0 -45
- package/tests/unit/did/WebVHManager.test.ts +0 -435
- package/tests/unit/did/createBtcoDidDocument.test.ts +0 -67
- package/tests/unit/did/providers/OrdinalsClientProviderAdapter.test.ts +0 -159
- package/tests/unit/events/EventEmitter.test.ts +0 -407
- package/tests/unit/kinds/KindRegistry.test.ts +0 -329
- package/tests/unit/kinds/types.test.ts +0 -409
- package/tests/unit/kinds/validators.test.ts +0 -651
- package/tests/unit/lifecycle/BatchOperations.test.ts +0 -527
- package/tests/unit/lifecycle/LifecycleManager.cleanapi.test.ts +0 -441
- package/tests/unit/lifecycle/LifecycleManager.keymanagement.test.ts +0 -312
- package/tests/unit/lifecycle/LifecycleManager.prov.test.ts +0 -18
- package/tests/unit/lifecycle/LifecycleManager.test.ts +0 -213
- package/tests/unit/lifecycle/LifecycleManager.transfer.unit.test.ts +0 -30
- package/tests/unit/lifecycle/OriginalsAsset.test.ts +0 -176
- package/tests/unit/lifecycle/ProvenanceQuery.test.ts +0 -577
- package/tests/unit/lifecycle/ResourceVersioning.test.ts +0 -651
- package/tests/unit/resources/ResourceManager.test.ts +0 -740
- package/tests/unit/storage/MemoryStorageAdapter.test.ts +0 -93
- package/tests/unit/types/network.test.ts +0 -255
- package/tests/unit/utils/EventIntegration.test.ts +0 -384
- package/tests/unit/utils/Logger.test.ts +0 -473
- package/tests/unit/utils/MetricsCollector.test.ts +0 -358
- package/tests/unit/utils/bitcoin-address.test.ts +0 -250
- package/tests/unit/utils/cbor.test.ts +0 -35
- package/tests/unit/utils/encoding.test.ts +0 -318
- package/tests/unit/utils/hash.test.ts +0 -12
- package/tests/unit/utils/retry.test.ts +0 -100
- package/tests/unit/utils/satoshi-validation.test.ts +0 -354
- package/tests/unit/utils/serialization.test.ts +0 -124
- package/tests/unit/utils/telemetry.test.ts +0 -52
- package/tests/unit/utils/validation.test.ts +0 -141
- package/tests/unit/vc/CredentialManager.helpers.test.ts +0 -527
- package/tests/unit/vc/CredentialManager.test.ts +0 -487
- package/tests/unit/vc/Issuer.test.ts +0 -107
- package/tests/unit/vc/Verifier.test.ts +0 -525
- package/tests/unit/vc/bbs.test.ts +0 -282
- package/tests/unit/vc/cryptosuites/eddsa.test.ts +0 -398
- package/tests/unit/vc/documentLoader.test.ts +0 -121
- package/tests/unit/vc/proofs/data-integrity.test.ts +0 -24
- package/tsconfig.json +0 -31
- package/tsconfig.test.json +0 -15
|
@@ -0,0 +1,218 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Base validator interface and common validation utilities
|
|
3
|
+
*/
|
|
4
|
+
/**
|
|
5
|
+
* Common validation utilities
|
|
6
|
+
*/
|
|
7
|
+
export class ValidationUtils {
|
|
8
|
+
/**
|
|
9
|
+
* Check if a string is a valid semantic version
|
|
10
|
+
*/
|
|
11
|
+
static isValidSemver(version) {
|
|
12
|
+
const semverRegex = /^(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$/;
|
|
13
|
+
return semverRegex.test(version);
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Check if a string is a valid DID
|
|
17
|
+
*/
|
|
18
|
+
static isValidDID(did) {
|
|
19
|
+
// DID format: did:method:identifier (identifier can contain : for path segments)
|
|
20
|
+
// Examples: did:peer:123, did:webvh:example.com:user, did:btco:12345
|
|
21
|
+
const didRegex = /^did:[a-z0-9]+:[a-zA-Z0-9._:%-]+$/;
|
|
22
|
+
return didRegex.test(did);
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Check if a string is a valid SPDX license identifier
|
|
26
|
+
*/
|
|
27
|
+
static isValidSPDXLicense(license) {
|
|
28
|
+
// Common SPDX identifiers (not exhaustive)
|
|
29
|
+
const commonLicenses = [
|
|
30
|
+
'MIT', 'Apache-2.0', 'GPL-3.0', 'GPL-2.0', 'BSD-2-Clause', 'BSD-3-Clause',
|
|
31
|
+
'ISC', 'MPL-2.0', 'LGPL-3.0', 'AGPL-3.0', 'Unlicense', 'CC0-1.0',
|
|
32
|
+
'CC-BY-4.0', 'CC-BY-SA-4.0', 'CC-BY-NC-4.0', 'WTFPL', 'Zlib', 'BSL-1.0',
|
|
33
|
+
'MIT-0', 'Apache-1.0', 'Apache-1.1', 'EPL-2.0', 'EUPL-1.2',
|
|
34
|
+
];
|
|
35
|
+
return commonLicenses.includes(license) || /^[A-Z0-9][A-Z0-9._-]*$/i.test(license);
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Check if a string is a valid MIME type
|
|
39
|
+
*/
|
|
40
|
+
static isValidMimeType(mimeType) {
|
|
41
|
+
const mimeRegex = /^[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}\/[a-zA-Z0-9][a-zA-Z0-9!#$&^_.+-]{0,126}$/;
|
|
42
|
+
return mimeRegex.test(mimeType);
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Check if a string is a valid URL
|
|
46
|
+
*/
|
|
47
|
+
static isValidURL(url) {
|
|
48
|
+
try {
|
|
49
|
+
new URL(url);
|
|
50
|
+
return true;
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
return false;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Check if a string is a valid ISO 639-1 language code
|
|
58
|
+
*/
|
|
59
|
+
static isValidLanguageCode(code) {
|
|
60
|
+
// ISO 639-1 is 2 lowercase letters
|
|
61
|
+
return /^[a-z]{2}$/.test(code);
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Check if a resource ID exists in the resources array
|
|
65
|
+
*/
|
|
66
|
+
static resourceExists(resourceId, resources) {
|
|
67
|
+
return resources.some(r => r.id === resourceId);
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Get a resource by ID
|
|
71
|
+
*/
|
|
72
|
+
static getResource(resourceId, resources) {
|
|
73
|
+
return resources.find(r => r.id === resourceId);
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Create a validation error
|
|
77
|
+
*/
|
|
78
|
+
static error(code, message, path, value) {
|
|
79
|
+
return { code, message, path, value };
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Create a validation warning
|
|
83
|
+
*/
|
|
84
|
+
static warning(code, message, path, suggestion) {
|
|
85
|
+
return { code, message, path, suggestion };
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Create a successful validation result
|
|
89
|
+
*/
|
|
90
|
+
static success(warnings = []) {
|
|
91
|
+
return { isValid: true, errors: [], warnings };
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Create a failed validation result
|
|
95
|
+
*/
|
|
96
|
+
static failure(errors, warnings = []) {
|
|
97
|
+
return { isValid: false, errors, warnings };
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Merge multiple validation results
|
|
101
|
+
*/
|
|
102
|
+
static merge(...results) {
|
|
103
|
+
const errors = [];
|
|
104
|
+
const warnings = [];
|
|
105
|
+
for (const result of results) {
|
|
106
|
+
errors.push(...result.errors);
|
|
107
|
+
warnings.push(...result.warnings);
|
|
108
|
+
}
|
|
109
|
+
return {
|
|
110
|
+
isValid: errors.length === 0,
|
|
111
|
+
errors,
|
|
112
|
+
warnings,
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Base validator class with common validation logic
|
|
118
|
+
*/
|
|
119
|
+
export class BaseKindValidator {
|
|
120
|
+
/**
|
|
121
|
+
* Validate a manifest
|
|
122
|
+
* Combines base validation with kind-specific validation
|
|
123
|
+
*/
|
|
124
|
+
validate(manifest) {
|
|
125
|
+
const baseResult = this.validateBase(manifest);
|
|
126
|
+
const kindResult = this.validateKind(manifest);
|
|
127
|
+
return ValidationUtils.merge(baseResult, kindResult);
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* Validate base manifest fields common to all kinds
|
|
131
|
+
*/
|
|
132
|
+
validateBase(manifest) {
|
|
133
|
+
const errors = [];
|
|
134
|
+
const warnings = [];
|
|
135
|
+
// Validate kind
|
|
136
|
+
if (!manifest.kind) {
|
|
137
|
+
errors.push(ValidationUtils.error('MISSING_KIND', 'Manifest must specify a kind', 'kind'));
|
|
138
|
+
}
|
|
139
|
+
// Validate name
|
|
140
|
+
if (!manifest.name || typeof manifest.name !== 'string') {
|
|
141
|
+
errors.push(ValidationUtils.error('MISSING_NAME', 'Manifest must have a name', 'name'));
|
|
142
|
+
}
|
|
143
|
+
else if (manifest.name.length < 1 || manifest.name.length > 200) {
|
|
144
|
+
errors.push(ValidationUtils.error('INVALID_NAME_LENGTH', 'Name must be between 1 and 200 characters', 'name', manifest.name));
|
|
145
|
+
}
|
|
146
|
+
// Validate version
|
|
147
|
+
if (!manifest.version || typeof manifest.version !== 'string') {
|
|
148
|
+
errors.push(ValidationUtils.error('MISSING_VERSION', 'Manifest must have a version', 'version'));
|
|
149
|
+
}
|
|
150
|
+
else if (!ValidationUtils.isValidSemver(manifest.version)) {
|
|
151
|
+
errors.push(ValidationUtils.error('INVALID_VERSION', 'Version must be a valid semantic version (e.g., 1.0.0)', 'version', manifest.version));
|
|
152
|
+
}
|
|
153
|
+
// Validate resources
|
|
154
|
+
if (!manifest.resources || !Array.isArray(manifest.resources)) {
|
|
155
|
+
errors.push(ValidationUtils.error('MISSING_RESOURCES', 'Manifest must have resources array', 'resources'));
|
|
156
|
+
}
|
|
157
|
+
else if (manifest.resources.length === 0) {
|
|
158
|
+
errors.push(ValidationUtils.error('EMPTY_RESOURCES', 'Manifest must have at least one resource', 'resources'));
|
|
159
|
+
}
|
|
160
|
+
else {
|
|
161
|
+
// Validate each resource
|
|
162
|
+
for (let i = 0; i < manifest.resources.length; i++) {
|
|
163
|
+
const resource = manifest.resources[i];
|
|
164
|
+
const resourcePath = `resources[${i}]`;
|
|
165
|
+
if (!resource.id || typeof resource.id !== 'string') {
|
|
166
|
+
errors.push(ValidationUtils.error('INVALID_RESOURCE_ID', `Resource at index ${i} must have an id`, `${resourcePath}.id`));
|
|
167
|
+
}
|
|
168
|
+
if (!resource.type || typeof resource.type !== 'string') {
|
|
169
|
+
errors.push(ValidationUtils.error('INVALID_RESOURCE_TYPE', `Resource at index ${i} must have a type`, `${resourcePath}.type`));
|
|
170
|
+
}
|
|
171
|
+
if (!resource.contentType || !ValidationUtils.isValidMimeType(resource.contentType)) {
|
|
172
|
+
errors.push(ValidationUtils.error('INVALID_CONTENT_TYPE', `Resource at index ${i} must have a valid MIME contentType`, `${resourcePath}.contentType`, resource.contentType));
|
|
173
|
+
}
|
|
174
|
+
if (!resource.hash || typeof resource.hash !== 'string' || !/^[0-9a-fA-F]+$/.test(resource.hash)) {
|
|
175
|
+
errors.push(ValidationUtils.error('INVALID_RESOURCE_HASH', `Resource at index ${i} must have a valid hex hash`, `${resourcePath}.hash`));
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
// Check for duplicate resource IDs
|
|
179
|
+
const resourceIds = manifest.resources.map(r => r.id);
|
|
180
|
+
const duplicates = resourceIds.filter((id, index) => resourceIds.indexOf(id) !== index);
|
|
181
|
+
if (duplicates.length > 0) {
|
|
182
|
+
errors.push(ValidationUtils.error('DUPLICATE_RESOURCE_IDS', `Duplicate resource IDs found: ${[...new Set(duplicates)].join(', ')}`, 'resources'));
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
// Validate dependencies if present
|
|
186
|
+
if (manifest.dependencies) {
|
|
187
|
+
if (!Array.isArray(manifest.dependencies)) {
|
|
188
|
+
errors.push(ValidationUtils.error('INVALID_DEPENDENCIES', 'Dependencies must be an array', 'dependencies'));
|
|
189
|
+
}
|
|
190
|
+
else {
|
|
191
|
+
for (let i = 0; i < manifest.dependencies.length; i++) {
|
|
192
|
+
const dep = manifest.dependencies[i];
|
|
193
|
+
const depPath = `dependencies[${i}]`;
|
|
194
|
+
if (!dep.did || !ValidationUtils.isValidDID(dep.did)) {
|
|
195
|
+
errors.push(ValidationUtils.error('INVALID_DEPENDENCY_DID', `Dependency at index ${i} must have a valid DID`, `${depPath}.did`, dep.did));
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
// Validate optional fields
|
|
201
|
+
if (manifest.license && !ValidationUtils.isValidSPDXLicense(manifest.license)) {
|
|
202
|
+
warnings.push(ValidationUtils.warning('UNKNOWN_LICENSE', `License "${manifest.license}" is not a recognized SPDX identifier`, 'license', 'Use a valid SPDX license identifier like MIT, Apache-2.0, etc.'));
|
|
203
|
+
}
|
|
204
|
+
if (manifest.homepage && !ValidationUtils.isValidURL(manifest.homepage)) {
|
|
205
|
+
errors.push(ValidationUtils.error('INVALID_HOMEPAGE', 'Homepage must be a valid URL', 'homepage', manifest.homepage));
|
|
206
|
+
}
|
|
207
|
+
if (manifest.repository && !ValidationUtils.isValidURL(manifest.repository)) {
|
|
208
|
+
errors.push(ValidationUtils.error('INVALID_REPOSITORY', 'Repository must be a valid URL', 'repository', manifest.repository));
|
|
209
|
+
}
|
|
210
|
+
// Suggest adding description if missing
|
|
211
|
+
if (!manifest.description) {
|
|
212
|
+
warnings.push(ValidationUtils.warning('MISSING_DESCRIPTION', 'Consider adding a description for better discoverability', 'description', 'Add a brief description of this Original'));
|
|
213
|
+
}
|
|
214
|
+
return errors.length > 0
|
|
215
|
+
? ValidationUtils.failure(errors, warnings)
|
|
216
|
+
: ValidationUtils.success(warnings);
|
|
217
|
+
}
|
|
218
|
+
}
|
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Kind validators exports
|
|
3
3
|
*/
|
|
4
|
-
|
|
5
4
|
export { BaseKindValidator, ValidationUtils, type KindValidator } from './base';
|
|
6
5
|
export { AppValidator } from './AppValidator';
|
|
7
6
|
export { AgentValidator } from './AgentValidator';
|
|
@@ -9,4 +8,3 @@ export { ModuleValidator } from './ModuleValidator';
|
|
|
9
8
|
export { DatasetValidator } from './DatasetValidator';
|
|
10
9
|
export { MediaValidator } from './MediaValidator';
|
|
11
10
|
export { DocumentValidator } from './DocumentValidator';
|
|
12
|
-
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Kind validators exports
|
|
3
|
+
*/
|
|
4
|
+
export { BaseKindValidator, ValidationUtils } from './base';
|
|
5
|
+
export { AppValidator } from './AppValidator';
|
|
6
|
+
export { AgentValidator } from './AgentValidator';
|
|
7
|
+
export { ModuleValidator } from './ModuleValidator';
|
|
8
|
+
export { DatasetValidator } from './DatasetValidator';
|
|
9
|
+
export { MediaValidator } from './MediaValidator';
|
|
10
|
+
export { DocumentValidator } from './DocumentValidator';
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Batch Operations for Originals SDK
|
|
3
|
+
*
|
|
4
|
+
* Enables efficient processing of multiple assets with:
|
|
5
|
+
* - Configurable concurrency
|
|
6
|
+
* - Retry logic with exponential backoff
|
|
7
|
+
* - Fail-fast vs continue-on-error modes
|
|
8
|
+
* - Pre-validation of all items
|
|
9
|
+
* - Detailed timing and error tracking
|
|
10
|
+
*/
|
|
11
|
+
/**
|
|
12
|
+
* Result of a batch operation containing successful and failed items
|
|
13
|
+
*/
|
|
14
|
+
export interface BatchResult<T> {
|
|
15
|
+
successful: Array<{
|
|
16
|
+
index: number;
|
|
17
|
+
result: T;
|
|
18
|
+
duration: number;
|
|
19
|
+
}>;
|
|
20
|
+
failed: Array<{
|
|
21
|
+
index: number;
|
|
22
|
+
error: Error;
|
|
23
|
+
duration: number;
|
|
24
|
+
retryAttempts?: number;
|
|
25
|
+
}>;
|
|
26
|
+
totalProcessed: number;
|
|
27
|
+
totalDuration: number;
|
|
28
|
+
batchId: string;
|
|
29
|
+
startedAt: string;
|
|
30
|
+
completedAt: string;
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Options for configuring batch operation execution
|
|
34
|
+
*/
|
|
35
|
+
export interface BatchOperationOptions {
|
|
36
|
+
continueOnError?: boolean;
|
|
37
|
+
maxConcurrent?: number;
|
|
38
|
+
retryCount?: number;
|
|
39
|
+
retryDelay?: number;
|
|
40
|
+
timeoutMs?: number;
|
|
41
|
+
validateFirst?: boolean;
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Options for batch inscription operations with Bitcoin-specific settings
|
|
45
|
+
*/
|
|
46
|
+
export interface BatchInscriptionOptions extends BatchOperationOptions {
|
|
47
|
+
singleTransaction?: boolean;
|
|
48
|
+
feeRate?: number;
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Error thrown when a batch operation fails
|
|
52
|
+
*/
|
|
53
|
+
export declare class BatchError extends Error {
|
|
54
|
+
batchId: string;
|
|
55
|
+
operation: string;
|
|
56
|
+
partialResults: {
|
|
57
|
+
successful: number;
|
|
58
|
+
failed: number;
|
|
59
|
+
};
|
|
60
|
+
constructor(batchId: string, operation: string, partialResults: {
|
|
61
|
+
successful: number;
|
|
62
|
+
failed: number;
|
|
63
|
+
}, message: string);
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Result of a single-transaction batch inscription with cost analysis
|
|
67
|
+
*/
|
|
68
|
+
export interface BatchInscriptionResult<T> {
|
|
69
|
+
batchTransactionId: string;
|
|
70
|
+
individualInscriptionIds: string[];
|
|
71
|
+
assets: T[];
|
|
72
|
+
totalFee: number;
|
|
73
|
+
feePerAsset: number[];
|
|
74
|
+
feeSavings: {
|
|
75
|
+
batchFee: number;
|
|
76
|
+
individualFees: number;
|
|
77
|
+
savings: number;
|
|
78
|
+
savingsPercentage: number;
|
|
79
|
+
};
|
|
80
|
+
batchId: string;
|
|
81
|
+
processingTime: number;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Executor for batch operations with configurable options
|
|
85
|
+
*/
|
|
86
|
+
export declare class BatchOperationExecutor {
|
|
87
|
+
private defaultOptions;
|
|
88
|
+
constructor(defaultOptions?: BatchOperationOptions);
|
|
89
|
+
/**
|
|
90
|
+
* Execute a batch operation on multiple items
|
|
91
|
+
*
|
|
92
|
+
* @param items - Array of items to process
|
|
93
|
+
* @param operation - Function to execute on each item
|
|
94
|
+
* @param options - Batch operation options
|
|
95
|
+
* @param predeterminedBatchId - Optional pre-generated batch ID for event correlation
|
|
96
|
+
* @returns BatchResult with successful and failed operations
|
|
97
|
+
*/
|
|
98
|
+
execute<T, R>(items: T[], operation: (item: T, index: number) => Promise<R>, options?: BatchOperationOptions, predeterminedBatchId?: string): Promise<BatchResult<R>>;
|
|
99
|
+
/**
|
|
100
|
+
* Generate unique batch ID
|
|
101
|
+
*/
|
|
102
|
+
generateBatchId(): string;
|
|
103
|
+
/**
|
|
104
|
+
* Calculate retry delay with exponential backoff
|
|
105
|
+
*/
|
|
106
|
+
private calculateRetryDelay;
|
|
107
|
+
/**
|
|
108
|
+
* Execute operation with timeout
|
|
109
|
+
*/
|
|
110
|
+
private executeWithTimeout;
|
|
111
|
+
/**
|
|
112
|
+
* Sleep for specified milliseconds
|
|
113
|
+
*/
|
|
114
|
+
private sleep;
|
|
115
|
+
/**
|
|
116
|
+
* Chunk array into smaller arrays of specified size
|
|
117
|
+
*/
|
|
118
|
+
private chunkArray;
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Validation result for batch operations
|
|
122
|
+
*/
|
|
123
|
+
export interface ValidationResult {
|
|
124
|
+
isValid: boolean;
|
|
125
|
+
errors: string[];
|
|
126
|
+
warnings?: string[];
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Validator for batch operations
|
|
130
|
+
*/
|
|
131
|
+
export declare class BatchValidator {
|
|
132
|
+
/**
|
|
133
|
+
* Validate batch of resources for asset creation
|
|
134
|
+
*/
|
|
135
|
+
validateBatchCreate(resourcesList: any[][]): ValidationResult[];
|
|
136
|
+
/**
|
|
137
|
+
* Validate batch of assets for inscription
|
|
138
|
+
*/
|
|
139
|
+
validateBatchInscription(assets: any[]): ValidationResult[];
|
|
140
|
+
/**
|
|
141
|
+
* Validate batch of transfer operations
|
|
142
|
+
*/
|
|
143
|
+
validateBatchTransfer(transfers: Array<{
|
|
144
|
+
asset: any;
|
|
145
|
+
to: string;
|
|
146
|
+
}>): ValidationResult[];
|
|
147
|
+
}
|
|
@@ -0,0 +1,251 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Batch Operations for Originals SDK
|
|
3
|
+
*
|
|
4
|
+
* Enables efficient processing of multiple assets with:
|
|
5
|
+
* - Configurable concurrency
|
|
6
|
+
* - Retry logic with exponential backoff
|
|
7
|
+
* - Fail-fast vs continue-on-error modes
|
|
8
|
+
* - Pre-validation of all items
|
|
9
|
+
* - Detailed timing and error tracking
|
|
10
|
+
*/
|
|
11
|
+
import { randomBytes, bytesToHex } from '@noble/hashes/utils.js';
|
|
12
|
+
/**
|
|
13
|
+
* Error thrown when a batch operation fails
|
|
14
|
+
*/
|
|
15
|
+
export class BatchError extends Error {
|
|
16
|
+
constructor(batchId, operation, partialResults, message) {
|
|
17
|
+
super(message);
|
|
18
|
+
this.batchId = batchId;
|
|
19
|
+
this.operation = operation;
|
|
20
|
+
this.partialResults = partialResults;
|
|
21
|
+
this.name = 'BatchError';
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Executor for batch operations with configurable options
|
|
26
|
+
*/
|
|
27
|
+
export class BatchOperationExecutor {
|
|
28
|
+
constructor(defaultOptions = {}) {
|
|
29
|
+
this.defaultOptions = defaultOptions;
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Execute a batch operation on multiple items
|
|
33
|
+
*
|
|
34
|
+
* @param items - Array of items to process
|
|
35
|
+
* @param operation - Function to execute on each item
|
|
36
|
+
* @param options - Batch operation options
|
|
37
|
+
* @param predeterminedBatchId - Optional pre-generated batch ID for event correlation
|
|
38
|
+
* @returns BatchResult with successful and failed operations
|
|
39
|
+
*/
|
|
40
|
+
async execute(items, operation, options, predeterminedBatchId) {
|
|
41
|
+
const opts = { ...this.defaultOptions, ...options };
|
|
42
|
+
const { continueOnError = false, maxConcurrent = 1, retryCount = 0, retryDelay = 1000, timeoutMs = 30000 } = opts;
|
|
43
|
+
const batchId = predeterminedBatchId || this.generateBatchId();
|
|
44
|
+
const startedAt = new Date().toISOString();
|
|
45
|
+
const startTime = Date.now();
|
|
46
|
+
const successful = [];
|
|
47
|
+
const failed = [];
|
|
48
|
+
// Process items with concurrency control
|
|
49
|
+
const processItem = async (item, index) => {
|
|
50
|
+
const itemStartTime = Date.now();
|
|
51
|
+
let lastError = null;
|
|
52
|
+
let attempts = 0;
|
|
53
|
+
for (let attempt = 0; attempt <= retryCount; attempt++) {
|
|
54
|
+
attempts = attempt + 1;
|
|
55
|
+
try {
|
|
56
|
+
// Execute with timeout
|
|
57
|
+
const result = await this.executeWithTimeout(() => operation(item, index), timeoutMs);
|
|
58
|
+
const duration = Date.now() - itemStartTime;
|
|
59
|
+
successful.push({ index, result, duration });
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
catch (error) {
|
|
63
|
+
lastError = error instanceof Error ? error : new Error(String(error));
|
|
64
|
+
// If not last attempt, wait with exponential backoff
|
|
65
|
+
if (attempt < retryCount) {
|
|
66
|
+
const delay = this.calculateRetryDelay(attempt, retryDelay);
|
|
67
|
+
await this.sleep(delay);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
// All retries failed
|
|
72
|
+
const duration = Date.now() - itemStartTime;
|
|
73
|
+
failed.push({
|
|
74
|
+
index,
|
|
75
|
+
error: lastError,
|
|
76
|
+
duration,
|
|
77
|
+
retryAttempts: attempts - 1
|
|
78
|
+
});
|
|
79
|
+
// If fail-fast mode, throw error
|
|
80
|
+
if (!continueOnError) {
|
|
81
|
+
throw lastError;
|
|
82
|
+
}
|
|
83
|
+
};
|
|
84
|
+
// Process items in batches based on maxConcurrent
|
|
85
|
+
try {
|
|
86
|
+
if (maxConcurrent === 1) {
|
|
87
|
+
// Sequential processing
|
|
88
|
+
for (let i = 0; i < items.length; i++) {
|
|
89
|
+
await processItem(items[i], i);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
else {
|
|
93
|
+
// Concurrent processing with limit
|
|
94
|
+
const chunks = this.chunkArray(items, maxConcurrent);
|
|
95
|
+
for (const chunk of chunks) {
|
|
96
|
+
await Promise.all(chunk.map((item, chunkIndex) => {
|
|
97
|
+
const globalIndex = chunks.slice(0, chunks.indexOf(chunk))
|
|
98
|
+
.reduce((acc, c) => acc + c.length, 0) + chunkIndex;
|
|
99
|
+
return processItem(item, globalIndex);
|
|
100
|
+
}));
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
catch (error) {
|
|
105
|
+
// In fail-fast mode, re-throw the error so callers can handle it
|
|
106
|
+
if (!continueOnError) {
|
|
107
|
+
throw error instanceof Error ? error : new Error(String(error));
|
|
108
|
+
}
|
|
109
|
+
// In continue-on-error mode, the error was already logged in processItem
|
|
110
|
+
// and we'll return the partial results below
|
|
111
|
+
}
|
|
112
|
+
const totalDuration = Date.now() - startTime;
|
|
113
|
+
const completedAt = new Date().toISOString();
|
|
114
|
+
return {
|
|
115
|
+
successful,
|
|
116
|
+
failed,
|
|
117
|
+
totalProcessed: successful.length + failed.length,
|
|
118
|
+
totalDuration,
|
|
119
|
+
batchId,
|
|
120
|
+
startedAt,
|
|
121
|
+
completedAt
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Generate unique batch ID
|
|
126
|
+
*/
|
|
127
|
+
generateBatchId() {
|
|
128
|
+
return `batch_${Date.now()}_${bytesToHex(randomBytes(8))}`;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Calculate retry delay with exponential backoff
|
|
132
|
+
*/
|
|
133
|
+
calculateRetryDelay(attempt, baseDelay) {
|
|
134
|
+
// Exponential backoff: baseDelay * 2^attempt
|
|
135
|
+
return baseDelay * Math.pow(2, attempt);
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Execute operation with timeout
|
|
139
|
+
*/
|
|
140
|
+
async executeWithTimeout(operation, timeoutMs) {
|
|
141
|
+
return Promise.race([
|
|
142
|
+
operation(),
|
|
143
|
+
new Promise((_, reject) => setTimeout(() => reject(new Error(`Operation timeout after ${timeoutMs}ms`)), timeoutMs))
|
|
144
|
+
]);
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Sleep for specified milliseconds
|
|
148
|
+
*/
|
|
149
|
+
sleep(ms) {
|
|
150
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Chunk array into smaller arrays of specified size
|
|
154
|
+
*/
|
|
155
|
+
chunkArray(array, size) {
|
|
156
|
+
const chunks = [];
|
|
157
|
+
for (let i = 0; i < array.length; i += size) {
|
|
158
|
+
chunks.push(array.slice(i, i + size));
|
|
159
|
+
}
|
|
160
|
+
return chunks;
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Validator for batch operations
|
|
165
|
+
*/
|
|
166
|
+
export class BatchValidator {
|
|
167
|
+
/**
|
|
168
|
+
* Validate batch of resources for asset creation
|
|
169
|
+
*/
|
|
170
|
+
validateBatchCreate(resourcesList) {
|
|
171
|
+
return resourcesList.map((resources, index) => {
|
|
172
|
+
const errors = [];
|
|
173
|
+
if (!Array.isArray(resources)) {
|
|
174
|
+
errors.push(`Item ${index}: Resources must be an array`);
|
|
175
|
+
return { isValid: false, errors };
|
|
176
|
+
}
|
|
177
|
+
if (resources.length === 0) {
|
|
178
|
+
errors.push(`Item ${index}: At least one resource is required`);
|
|
179
|
+
return { isValid: false, errors };
|
|
180
|
+
}
|
|
181
|
+
// Validate each resource
|
|
182
|
+
for (let i = 0; i < resources.length; i++) {
|
|
183
|
+
const resource = resources[i];
|
|
184
|
+
if (!resource || typeof resource !== 'object') {
|
|
185
|
+
errors.push(`Item ${index}, resource ${i}: Invalid resource object`);
|
|
186
|
+
continue;
|
|
187
|
+
}
|
|
188
|
+
if (!resource.id || typeof resource.id !== 'string') {
|
|
189
|
+
errors.push(`Item ${index}, resource ${i}: Missing or invalid id`);
|
|
190
|
+
}
|
|
191
|
+
if (!resource.type || typeof resource.type !== 'string') {
|
|
192
|
+
errors.push(`Item ${index}, resource ${i}: Missing or invalid type`);
|
|
193
|
+
}
|
|
194
|
+
if (!resource.contentType || typeof resource.contentType !== 'string') {
|
|
195
|
+
errors.push(`Item ${index}, resource ${i}: Missing or invalid contentType`);
|
|
196
|
+
}
|
|
197
|
+
if (!resource.hash || typeof resource.hash !== 'string' || !/^[0-9a-fA-F]+$/.test(resource.hash)) {
|
|
198
|
+
errors.push(`Item ${index}, resource ${i}: Missing or invalid hash`);
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
return { isValid: errors.length === 0, errors };
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* Validate batch of assets for inscription
|
|
206
|
+
*/
|
|
207
|
+
validateBatchInscription(assets) {
|
|
208
|
+
return assets.map((asset, index) => {
|
|
209
|
+
const errors = [];
|
|
210
|
+
if (!asset || typeof asset !== 'object') {
|
|
211
|
+
errors.push(`Item ${index}: Invalid asset object`);
|
|
212
|
+
return { isValid: false, errors };
|
|
213
|
+
}
|
|
214
|
+
if (!asset.id || typeof asset.id !== 'string') {
|
|
215
|
+
errors.push(`Item ${index}: Missing or invalid asset id`);
|
|
216
|
+
}
|
|
217
|
+
if (!asset.currentLayer) {
|
|
218
|
+
errors.push(`Item ${index}: Missing currentLayer`);
|
|
219
|
+
}
|
|
220
|
+
else if (asset.currentLayer === 'did:btco') {
|
|
221
|
+
errors.push(`Item ${index}: Asset already inscribed on Bitcoin`);
|
|
222
|
+
}
|
|
223
|
+
if (!asset.resources || !Array.isArray(asset.resources) || asset.resources.length === 0) {
|
|
224
|
+
errors.push(`Item ${index}: Asset must have at least one resource`);
|
|
225
|
+
}
|
|
226
|
+
return { isValid: errors.length === 0, errors };
|
|
227
|
+
});
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Validate batch of transfer operations
|
|
231
|
+
*/
|
|
232
|
+
validateBatchTransfer(transfers) {
|
|
233
|
+
return transfers.map((transfer, index) => {
|
|
234
|
+
const errors = [];
|
|
235
|
+
if (!transfer || typeof transfer !== 'object') {
|
|
236
|
+
errors.push(`Item ${index}: Invalid transfer object`);
|
|
237
|
+
return { isValid: false, errors };
|
|
238
|
+
}
|
|
239
|
+
if (!transfer.asset || typeof transfer.asset !== 'object') {
|
|
240
|
+
errors.push(`Item ${index}: Invalid asset`);
|
|
241
|
+
}
|
|
242
|
+
else if (transfer.asset.currentLayer !== 'did:btco') {
|
|
243
|
+
errors.push(`Item ${index}: Asset must be inscribed on Bitcoin before transfer`);
|
|
244
|
+
}
|
|
245
|
+
if (!transfer.to || typeof transfer.to !== 'string') {
|
|
246
|
+
errors.push(`Item ${index}: Invalid destination address`);
|
|
247
|
+
}
|
|
248
|
+
return { isValid: errors.length === 0, errors };
|
|
249
|
+
});
|
|
250
|
+
}
|
|
251
|
+
}
|