@arkade-os/sdk 0.3.12 → 0.4.0-next.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/README.md +483 -54
- package/dist/cjs/adapters/expo-db.js +35 -0
- package/dist/cjs/asset/assetGroup.js +141 -0
- package/dist/cjs/asset/assetId.js +88 -0
- package/dist/cjs/asset/assetInput.js +204 -0
- package/dist/cjs/asset/assetOutput.js +159 -0
- package/dist/cjs/asset/assetRef.js +82 -0
- package/dist/cjs/asset/index.js +24 -0
- package/dist/cjs/asset/metadata.js +172 -0
- package/dist/cjs/asset/packet.js +164 -0
- package/dist/cjs/asset/types.js +25 -0
- package/dist/cjs/asset/utils.js +105 -0
- package/dist/cjs/contracts/arkcontract.js +148 -0
- package/dist/cjs/contracts/contractManager.js +436 -0
- package/dist/cjs/contracts/contractWatcher.js +567 -0
- package/dist/cjs/contracts/handlers/default.js +85 -0
- package/dist/cjs/contracts/handlers/delegate.js +89 -0
- package/dist/cjs/contracts/handlers/helpers.js +105 -0
- package/dist/cjs/contracts/handlers/index.js +19 -0
- package/dist/cjs/contracts/handlers/registry.js +89 -0
- package/dist/cjs/contracts/handlers/vhtlc.js +193 -0
- package/dist/cjs/contracts/index.js +41 -0
- package/dist/cjs/contracts/types.js +2 -0
- package/dist/cjs/db/manager.js +97 -0
- package/dist/cjs/forfeit.js +12 -8
- package/dist/cjs/identity/index.js +1 -0
- package/dist/cjs/identity/seedIdentity.js +255 -0
- package/dist/cjs/index.js +70 -14
- package/dist/cjs/intent/index.js +28 -2
- package/dist/cjs/providers/ark.js +7 -0
- package/dist/cjs/providers/delegator.js +66 -0
- package/dist/cjs/providers/expoIndexer.js +5 -0
- package/dist/cjs/providers/indexer.js +68 -1
- package/dist/cjs/providers/onchain.js +2 -2
- package/dist/cjs/providers/utils.js +1 -0
- package/dist/cjs/repositories/contractRepository.js +0 -103
- package/dist/cjs/repositories/inMemory/contractRepository.js +55 -0
- package/dist/cjs/repositories/inMemory/walletRepository.js +80 -0
- package/dist/cjs/repositories/index.js +16 -0
- package/dist/cjs/repositories/indexedDB/contractRepository.js +187 -0
- package/dist/cjs/repositories/indexedDB/db.js +57 -0
- package/dist/cjs/repositories/indexedDB/schema.js +159 -0
- package/dist/cjs/repositories/indexedDB/walletRepository.js +338 -0
- package/dist/cjs/repositories/indexedDB/websqlAdapter.js +144 -0
- package/dist/cjs/repositories/migrations/contractRepositoryImpl.js +127 -0
- package/dist/cjs/repositories/migrations/fromStorageAdapter.js +66 -0
- package/dist/cjs/repositories/migrations/walletRepositoryImpl.js +180 -0
- package/dist/cjs/repositories/walletRepository.js +0 -169
- package/dist/cjs/script/base.js +54 -0
- package/dist/cjs/script/delegate.js +49 -0
- package/dist/cjs/storage/asyncStorage.js +4 -1
- package/dist/cjs/storage/fileSystem.js +3 -0
- package/dist/cjs/storage/inMemory.js +3 -0
- package/dist/cjs/storage/indexedDB.js +5 -1
- package/dist/cjs/storage/localStorage.js +3 -0
- package/dist/cjs/utils/arkTransaction.js +16 -0
- package/dist/cjs/utils/transactionHistory.js +50 -0
- package/dist/cjs/utils/txSizeEstimator.js +39 -14
- package/dist/cjs/wallet/asset-manager.js +338 -0
- package/dist/cjs/wallet/asset.js +117 -0
- package/dist/cjs/wallet/batch.js +1 -1
- package/dist/cjs/wallet/delegator.js +235 -0
- package/dist/cjs/wallet/expo/background.js +133 -0
- package/dist/cjs/wallet/expo/index.js +9 -0
- package/dist/cjs/wallet/expo/wallet.js +231 -0
- package/dist/cjs/wallet/onchain.js +57 -12
- package/dist/cjs/wallet/serviceWorker/wallet-message-handler.js +568 -0
- package/dist/cjs/wallet/serviceWorker/wallet.js +383 -102
- package/dist/cjs/wallet/unroll.js +7 -2
- package/dist/cjs/wallet/utils.js +60 -0
- package/dist/cjs/wallet/validation.js +151 -0
- package/dist/cjs/wallet/vtxo-manager.js +1 -1
- package/dist/cjs/wallet/wallet.js +702 -260
- package/dist/cjs/worker/browser/service-worker-manager.js +82 -0
- package/dist/cjs/{wallet/serviceWorker → worker/browser}/utils.js +2 -1
- package/dist/cjs/worker/expo/asyncStorageTaskQueue.js +78 -0
- package/dist/cjs/worker/expo/index.js +12 -0
- package/dist/cjs/worker/expo/processors/contractPollProcessor.js +61 -0
- package/dist/cjs/worker/expo/processors/index.js +6 -0
- package/dist/cjs/worker/expo/taskQueue.js +41 -0
- package/dist/cjs/worker/expo/taskRunner.js +57 -0
- package/dist/cjs/worker/messageBus.js +252 -0
- package/dist/esm/adapters/expo-db.js +27 -0
- package/dist/esm/asset/assetGroup.js +137 -0
- package/dist/esm/asset/assetId.js +84 -0
- package/dist/esm/asset/assetInput.js +199 -0
- package/dist/esm/asset/assetOutput.js +154 -0
- package/dist/esm/asset/assetRef.js +78 -0
- package/dist/esm/asset/index.js +8 -0
- package/dist/esm/asset/metadata.js +167 -0
- package/dist/esm/asset/packet.js +159 -0
- package/dist/esm/asset/types.js +22 -0
- package/dist/esm/asset/utils.js +99 -0
- package/dist/esm/contracts/arkcontract.js +141 -0
- package/dist/esm/contracts/contractManager.js +432 -0
- package/dist/esm/contracts/contractWatcher.js +563 -0
- package/dist/esm/contracts/handlers/default.js +82 -0
- package/dist/esm/contracts/handlers/delegate.js +86 -0
- package/dist/esm/contracts/handlers/helpers.js +66 -0
- package/dist/esm/contracts/handlers/index.js +12 -0
- package/dist/esm/contracts/handlers/registry.js +86 -0
- package/dist/esm/contracts/handlers/vhtlc.js +190 -0
- package/dist/esm/contracts/index.js +13 -0
- package/dist/esm/contracts/types.js +1 -0
- package/dist/esm/db/manager.js +92 -0
- package/dist/esm/forfeit.js +11 -8
- package/dist/esm/identity/index.js +1 -0
- package/dist/esm/identity/seedIdentity.js +249 -0
- package/dist/esm/index.js +25 -15
- package/dist/esm/intent/index.js +28 -2
- package/dist/esm/providers/ark.js +7 -0
- package/dist/esm/providers/delegator.js +62 -0
- package/dist/esm/providers/expoIndexer.js +5 -0
- package/dist/esm/providers/indexer.js +68 -1
- package/dist/esm/providers/onchain.js +2 -2
- package/dist/esm/providers/utils.js +1 -0
- package/dist/esm/repositories/contractRepository.js +1 -101
- package/dist/esm/repositories/inMemory/contractRepository.js +51 -0
- package/dist/esm/repositories/inMemory/walletRepository.js +76 -0
- package/dist/esm/repositories/index.js +8 -0
- package/dist/esm/repositories/indexedDB/contractRepository.js +183 -0
- package/dist/esm/repositories/indexedDB/db.js +42 -0
- package/dist/esm/repositories/indexedDB/schema.js +155 -0
- package/dist/esm/repositories/indexedDB/walletRepository.js +334 -0
- package/dist/esm/repositories/indexedDB/websqlAdapter.js +138 -0
- package/dist/esm/repositories/migrations/contractRepositoryImpl.js +121 -0
- package/dist/esm/repositories/migrations/fromStorageAdapter.js +58 -0
- package/dist/esm/repositories/migrations/walletRepositoryImpl.js +176 -0
- package/dist/esm/repositories/walletRepository.js +1 -167
- package/dist/esm/script/base.js +21 -1
- package/dist/esm/script/delegate.js +46 -0
- package/dist/esm/storage/asyncStorage.js +4 -1
- package/dist/esm/storage/fileSystem.js +3 -0
- package/dist/esm/storage/inMemory.js +3 -0
- package/dist/esm/storage/indexedDB.js +5 -1
- package/dist/esm/storage/localStorage.js +3 -0
- package/dist/esm/utils/arkTransaction.js +15 -0
- package/dist/esm/utils/transactionHistory.js +50 -0
- package/dist/esm/utils/txSizeEstimator.js +39 -14
- package/dist/esm/wallet/asset-manager.js +333 -0
- package/dist/esm/wallet/asset.js +111 -0
- package/dist/esm/wallet/batch.js +1 -1
- package/dist/esm/wallet/delegator.js +231 -0
- package/dist/esm/wallet/expo/background.js +128 -0
- package/dist/esm/wallet/expo/index.js +2 -0
- package/dist/esm/wallet/expo/wallet.js +194 -0
- package/dist/esm/wallet/onchain.js +57 -12
- package/dist/esm/wallet/serviceWorker/wallet-message-handler.js +564 -0
- package/dist/esm/wallet/serviceWorker/wallet.js +382 -101
- package/dist/esm/wallet/unroll.js +7 -2
- package/dist/esm/wallet/utils.js +55 -0
- package/dist/esm/wallet/validation.js +139 -0
- package/dist/esm/wallet/vtxo-manager.js +1 -1
- package/dist/esm/wallet/wallet.js +704 -229
- package/dist/esm/worker/browser/service-worker-manager.js +76 -0
- package/dist/esm/{wallet/serviceWorker → worker/browser}/utils.js +2 -1
- package/dist/esm/worker/expo/asyncStorageTaskQueue.js +74 -0
- package/dist/esm/worker/expo/index.js +4 -0
- package/dist/esm/worker/expo/processors/contractPollProcessor.js +58 -0
- package/dist/esm/worker/expo/processors/index.js +1 -0
- package/dist/esm/worker/expo/taskQueue.js +37 -0
- package/dist/esm/worker/expo/taskRunner.js +54 -0
- package/dist/esm/worker/messageBus.js +248 -0
- package/dist/types/adapters/expo-db.d.ts +7 -0
- package/dist/types/asset/assetGroup.d.ts +28 -0
- package/dist/types/asset/assetId.d.ts +19 -0
- package/dist/types/asset/assetInput.d.ts +46 -0
- package/dist/types/asset/assetOutput.d.ts +39 -0
- package/dist/types/asset/assetRef.d.ts +25 -0
- package/dist/types/asset/index.d.ts +8 -0
- package/dist/types/asset/metadata.d.ts +37 -0
- package/dist/types/asset/packet.d.ts +27 -0
- package/dist/types/asset/types.d.ts +18 -0
- package/dist/types/asset/utils.d.ts +21 -0
- package/dist/types/contracts/arkcontract.d.ts +101 -0
- package/dist/types/contracts/contractManager.d.ts +331 -0
- package/dist/types/contracts/contractWatcher.d.ts +192 -0
- package/dist/types/contracts/handlers/default.d.ts +19 -0
- package/dist/types/contracts/handlers/delegate.d.ts +21 -0
- package/dist/types/contracts/handlers/helpers.d.ts +18 -0
- package/dist/types/contracts/handlers/index.d.ts +7 -0
- package/dist/types/contracts/handlers/registry.d.ts +65 -0
- package/dist/types/contracts/handlers/vhtlc.d.ts +32 -0
- package/dist/types/contracts/index.d.ts +14 -0
- package/dist/types/contracts/types.d.ts +222 -0
- package/dist/types/db/manager.d.ts +22 -0
- package/dist/types/forfeit.d.ts +2 -1
- package/dist/types/identity/index.d.ts +1 -0
- package/dist/types/identity/seedIdentity.d.ts +128 -0
- package/dist/types/index.d.ts +21 -12
- package/dist/types/intent/index.d.ts +2 -1
- package/dist/types/providers/ark.d.ts +11 -2
- package/dist/types/providers/delegator.d.ts +29 -0
- package/dist/types/providers/indexer.d.ts +11 -1
- package/dist/types/repositories/contractRepository.d.ts +30 -19
- package/dist/types/repositories/inMemory/contractRepository.d.ts +17 -0
- package/dist/types/repositories/inMemory/walletRepository.d.ts +26 -0
- package/dist/types/repositories/index.d.ts +7 -0
- package/dist/types/repositories/indexedDB/contractRepository.d.ts +21 -0
- package/dist/types/repositories/indexedDB/db.d.ts +56 -0
- package/dist/types/repositories/indexedDB/schema.d.ts +8 -0
- package/dist/types/repositories/indexedDB/walletRepository.d.ts +25 -0
- package/dist/types/repositories/indexedDB/websqlAdapter.d.ts +49 -0
- package/dist/types/repositories/migrations/contractRepositoryImpl.d.ts +24 -0
- package/dist/types/repositories/migrations/fromStorageAdapter.d.ts +19 -0
- package/dist/types/repositories/migrations/walletRepositoryImpl.d.ts +27 -0
- package/dist/types/repositories/walletRepository.d.ts +13 -24
- package/dist/types/script/base.d.ts +1 -0
- package/dist/types/script/delegate.d.ts +36 -0
- package/dist/types/storage/asyncStorage.d.ts +4 -0
- package/dist/types/storage/fileSystem.d.ts +3 -0
- package/dist/types/storage/inMemory.d.ts +3 -0
- package/dist/types/storage/index.d.ts +3 -0
- package/dist/types/storage/indexedDB.d.ts +3 -0
- package/dist/types/storage/localStorage.d.ts +3 -0
- package/dist/types/utils/arkTransaction.d.ts +6 -0
- package/dist/types/utils/txSizeEstimator.d.ts +12 -2
- package/dist/types/wallet/asset-manager.d.ts +78 -0
- package/dist/types/wallet/asset.d.ts +21 -0
- package/dist/types/wallet/batch.d.ts +1 -1
- package/dist/types/wallet/delegator.d.ts +24 -0
- package/dist/types/wallet/expo/background.d.ts +66 -0
- package/dist/types/wallet/expo/index.d.ts +4 -0
- package/dist/types/wallet/expo/wallet.d.ts +97 -0
- package/dist/types/wallet/index.d.ts +75 -2
- package/dist/types/wallet/onchain.d.ts +22 -1
- package/dist/types/wallet/serviceWorker/wallet-message-handler.d.ts +366 -0
- package/dist/types/wallet/serviceWorker/wallet.d.ts +20 -11
- package/dist/types/wallet/utils.d.ts +13 -1
- package/dist/types/wallet/validation.d.ts +24 -0
- package/dist/types/wallet/wallet.d.ts +111 -17
- package/dist/types/worker/browser/service-worker-manager.d.ts +21 -0
- package/dist/types/{wallet/serviceWorker → worker/browser}/utils.d.ts +2 -1
- package/dist/types/worker/expo/asyncStorageTaskQueue.d.ts +46 -0
- package/dist/types/worker/expo/index.d.ts +7 -0
- package/dist/types/worker/expo/processors/contractPollProcessor.d.ts +14 -0
- package/dist/types/worker/expo/processors/index.d.ts +1 -0
- package/dist/types/worker/expo/taskQueue.d.ts +50 -0
- package/dist/types/worker/expo/taskRunner.d.ts +42 -0
- package/dist/types/worker/messageBus.d.ts +109 -0
- package/package.json +71 -17
- package/dist/cjs/wallet/serviceWorker/request.js +0 -78
- package/dist/cjs/wallet/serviceWorker/response.js +0 -222
- package/dist/cjs/wallet/serviceWorker/worker.js +0 -655
- package/dist/esm/wallet/serviceWorker/request.js +0 -75
- package/dist/esm/wallet/serviceWorker/response.js +0 -219
- package/dist/esm/wallet/serviceWorker/worker.js +0 -651
- package/dist/types/wallet/serviceWorker/request.d.ts +0 -74
- package/dist/types/wallet/serviceWorker/response.d.ts +0 -123
- package/dist/types/wallet/serviceWorker/worker.d.ts +0 -53
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import { hex } from "@scure/base";
|
|
2
|
+
import { DelegateVtxo } from '../../script/delegate.js';
|
|
3
|
+
import { DefaultVtxo } from '../../script/default.js';
|
|
4
|
+
import { isCsvSpendable, sequenceToTimelock, timelockToSequence, } from './helpers.js';
|
|
5
|
+
/**
|
|
6
|
+
* Handler for delegate wallet VTXOs.
|
|
7
|
+
*
|
|
8
|
+
* Delegate contracts extend the default tapscript with an additional delegate path:
|
|
9
|
+
* - forfeit: (Alice + Server) multisig for collaborative spending
|
|
10
|
+
* - exit: (Alice) + CSV timelock for unilateral exit
|
|
11
|
+
* - delegate: (Alice + Delegate + Server) multisig for delegated renewal
|
|
12
|
+
*/
|
|
13
|
+
export const DelegateContractHandler = {
|
|
14
|
+
type: "delegate",
|
|
15
|
+
createScript(params) {
|
|
16
|
+
const typed = this.deserializeParams(params);
|
|
17
|
+
return new DelegateVtxo.Script(typed);
|
|
18
|
+
},
|
|
19
|
+
serializeParams(params) {
|
|
20
|
+
return {
|
|
21
|
+
pubKey: hex.encode(params.pubKey),
|
|
22
|
+
serverPubKey: hex.encode(params.serverPubKey),
|
|
23
|
+
delegatePubKey: hex.encode(params.delegatePubKey),
|
|
24
|
+
csvTimelock: timelockToSequence(params.csvTimelock).toString(),
|
|
25
|
+
};
|
|
26
|
+
},
|
|
27
|
+
deserializeParams(params) {
|
|
28
|
+
const csvTimelock = params.csvTimelock
|
|
29
|
+
? sequenceToTimelock(Number(params.csvTimelock))
|
|
30
|
+
: DefaultVtxo.Script.DEFAULT_TIMELOCK;
|
|
31
|
+
return {
|
|
32
|
+
pubKey: hex.decode(params.pubKey),
|
|
33
|
+
serverPubKey: hex.decode(params.serverPubKey),
|
|
34
|
+
delegatePubKey: hex.decode(params.delegatePubKey),
|
|
35
|
+
csvTimelock,
|
|
36
|
+
};
|
|
37
|
+
},
|
|
38
|
+
selectPath(script, contract, context) {
|
|
39
|
+
if (context.collaborative) {
|
|
40
|
+
return { leaf: script.forfeit() };
|
|
41
|
+
}
|
|
42
|
+
const sequence = contract.params.csvTimelock
|
|
43
|
+
? Number(contract.params.csvTimelock)
|
|
44
|
+
: undefined;
|
|
45
|
+
if (!isCsvSpendable(context, sequence)) {
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
return {
|
|
49
|
+
leaf: script.exit(),
|
|
50
|
+
sequence,
|
|
51
|
+
};
|
|
52
|
+
},
|
|
53
|
+
getAllSpendingPaths(script, contract, context) {
|
|
54
|
+
const paths = [];
|
|
55
|
+
if (context.collaborative) {
|
|
56
|
+
paths.push({ leaf: script.forfeit() });
|
|
57
|
+
}
|
|
58
|
+
const exitPath = { leaf: script.exit() };
|
|
59
|
+
if (contract.params.csvTimelock) {
|
|
60
|
+
exitPath.sequence = Number(contract.params.csvTimelock);
|
|
61
|
+
}
|
|
62
|
+
paths.push(exitPath);
|
|
63
|
+
// Delegate path (Alice + Delegate + Server) — collaborative only
|
|
64
|
+
if (context.collaborative) {
|
|
65
|
+
paths.push({ leaf: script.delegate() });
|
|
66
|
+
}
|
|
67
|
+
return paths;
|
|
68
|
+
},
|
|
69
|
+
getSpendablePaths(script, contract, context) {
|
|
70
|
+
const paths = [];
|
|
71
|
+
if (context.collaborative) {
|
|
72
|
+
paths.push({ leaf: script.forfeit() });
|
|
73
|
+
}
|
|
74
|
+
const exitSequence = contract.params.csvTimelock
|
|
75
|
+
? Number(contract.params.csvTimelock)
|
|
76
|
+
: undefined;
|
|
77
|
+
if (isCsvSpendable(context, exitSequence)) {
|
|
78
|
+
const exitPath = { leaf: script.exit() };
|
|
79
|
+
if (exitSequence !== undefined) {
|
|
80
|
+
exitPath.sequence = exitSequence;
|
|
81
|
+
}
|
|
82
|
+
paths.push(exitPath);
|
|
83
|
+
}
|
|
84
|
+
return paths;
|
|
85
|
+
},
|
|
86
|
+
};
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import * as bip68 from "bip68";
|
|
2
|
+
/**
|
|
3
|
+
* Convert RelativeTimelock to BIP68 sequence number.
|
|
4
|
+
*/
|
|
5
|
+
export function timelockToSequence(timelock) {
|
|
6
|
+
return bip68.encode(timelock.type === "blocks"
|
|
7
|
+
? { blocks: Number(timelock.value) }
|
|
8
|
+
: { seconds: Number(timelock.value) });
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Convert BIP68 sequence number back to RelativeTimelock.
|
|
12
|
+
*/
|
|
13
|
+
export function sequenceToTimelock(sequence) {
|
|
14
|
+
const decoded = bip68.decode(sequence);
|
|
15
|
+
if ("blocks" in decoded && decoded.blocks !== undefined) {
|
|
16
|
+
return { type: "blocks", value: BigInt(decoded.blocks) };
|
|
17
|
+
}
|
|
18
|
+
if ("seconds" in decoded && decoded.seconds !== undefined) {
|
|
19
|
+
return { type: "seconds", value: BigInt(decoded.seconds) };
|
|
20
|
+
}
|
|
21
|
+
throw new Error(`Invalid BIP68 sequence: ${sequence}`);
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Resolve wallet's role from explicit role or by matching pubkey.
|
|
25
|
+
*/
|
|
26
|
+
export function resolveRole(contract, context) {
|
|
27
|
+
// Explicit role takes precedence
|
|
28
|
+
if (context.role === "sender" || context.role === "receiver") {
|
|
29
|
+
return context.role;
|
|
30
|
+
}
|
|
31
|
+
// Try to match wallet pubkey against contract params
|
|
32
|
+
if (context.walletPubKey) {
|
|
33
|
+
if (context.walletPubKey === contract.params.sender) {
|
|
34
|
+
return "sender";
|
|
35
|
+
}
|
|
36
|
+
if (context.walletPubKey === contract.params.receiver) {
|
|
37
|
+
return "receiver";
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
return undefined;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Check if a CSV timelock is currently satisfied for the given context/VTXO.
|
|
44
|
+
*/
|
|
45
|
+
export function isCsvSpendable(context, sequence) {
|
|
46
|
+
if (sequence === undefined)
|
|
47
|
+
return true;
|
|
48
|
+
if (!context.vtxo)
|
|
49
|
+
return false;
|
|
50
|
+
const timelock = sequenceToTimelock(sequence);
|
|
51
|
+
if (timelock.type === "blocks") {
|
|
52
|
+
if (context.blockHeight === undefined ||
|
|
53
|
+
context.vtxo.status.block_height === undefined) {
|
|
54
|
+
return false;
|
|
55
|
+
}
|
|
56
|
+
return (context.blockHeight - context.vtxo.status.block_height >=
|
|
57
|
+
Number(timelock.value));
|
|
58
|
+
}
|
|
59
|
+
if (timelock.type === "seconds") {
|
|
60
|
+
const blockTime = context.vtxo.status.block_time;
|
|
61
|
+
if (blockTime === undefined)
|
|
62
|
+
return false;
|
|
63
|
+
return context.currentTime / 1000 - blockTime >= Number(timelock.value);
|
|
64
|
+
}
|
|
65
|
+
return false;
|
|
66
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
export { contractHandlers } from './registry.js';
|
|
2
|
+
export { DefaultContractHandler } from './default.js';
|
|
3
|
+
export { DelegateContractHandler } from './delegate.js';
|
|
4
|
+
export { VHTLCContractHandler } from './vhtlc.js';
|
|
5
|
+
// Register built-in handlers
|
|
6
|
+
import { contractHandlers } from './registry.js';
|
|
7
|
+
import { DefaultContractHandler } from './default.js';
|
|
8
|
+
import { DelegateContractHandler } from './delegate.js';
|
|
9
|
+
import { VHTLCContractHandler } from './vhtlc.js';
|
|
10
|
+
contractHandlers.register(DefaultContractHandler);
|
|
11
|
+
contractHandlers.register(DelegateContractHandler);
|
|
12
|
+
contractHandlers.register(VHTLCContractHandler);
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Registry for contract handlers.
|
|
3
|
+
*
|
|
4
|
+
* Each contract type ("default", "vhtlc", etc.) has a handler that knows
|
|
5
|
+
* how to create VtxoScripts, serialize params, and select spending paths.
|
|
6
|
+
*
|
|
7
|
+
* @example
|
|
8
|
+
* ```typescript
|
|
9
|
+
* // Register a custom handler
|
|
10
|
+
* contractHandlers.register(myCustomHandler);
|
|
11
|
+
*
|
|
12
|
+
* // Get handler for a type
|
|
13
|
+
* const handler = contractHandlers.get("vhtlc");
|
|
14
|
+
* const script = handler.createScript(contract.params);
|
|
15
|
+
* ```
|
|
16
|
+
*/
|
|
17
|
+
class ContractHandlerRegistry {
|
|
18
|
+
constructor() {
|
|
19
|
+
this.handlers = new Map();
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Register a contract handler.
|
|
23
|
+
*
|
|
24
|
+
* @param handler - The handler to register
|
|
25
|
+
* @throws If a handler for this type is already registered
|
|
26
|
+
*/
|
|
27
|
+
register(handler) {
|
|
28
|
+
if (this.handlers.has(handler.type)) {
|
|
29
|
+
throw new Error(`Contract handler for type '${handler.type}' is already registered`);
|
|
30
|
+
}
|
|
31
|
+
this.handlers.set(handler.type, handler);
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Get a handler by type.
|
|
35
|
+
*
|
|
36
|
+
* @param type - The contract type
|
|
37
|
+
* @returns The handler, or undefined if not found
|
|
38
|
+
*/
|
|
39
|
+
get(type) {
|
|
40
|
+
return this.handlers.get(type);
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Get a handler by type, throwing if not found.
|
|
44
|
+
*
|
|
45
|
+
* @param type - The contract type
|
|
46
|
+
* @returns The handler
|
|
47
|
+
* @throws If no handler is registered for this type
|
|
48
|
+
*/
|
|
49
|
+
getOrThrow(type) {
|
|
50
|
+
const handler = this.get(type);
|
|
51
|
+
if (!handler) {
|
|
52
|
+
throw new Error(`No contract handler registered for type '${type}'`);
|
|
53
|
+
}
|
|
54
|
+
return handler;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Check if a handler is registered.
|
|
58
|
+
*
|
|
59
|
+
* @param type - The contract type
|
|
60
|
+
*/
|
|
61
|
+
has(type) {
|
|
62
|
+
return this.handlers.has(type);
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Get all registered types.
|
|
66
|
+
*/
|
|
67
|
+
getRegisteredTypes() {
|
|
68
|
+
return Array.from(this.handlers.keys());
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Unregister a handler (mainly for testing).
|
|
72
|
+
*/
|
|
73
|
+
unregister(type) {
|
|
74
|
+
return this.handlers.delete(type);
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Clear all handlers (mainly for testing).
|
|
78
|
+
*/
|
|
79
|
+
clear() {
|
|
80
|
+
this.handlers.clear();
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Global registry of contract handlers.
|
|
85
|
+
*/
|
|
86
|
+
export const contractHandlers = new ContractHandlerRegistry();
|
|
@@ -0,0 +1,190 @@
|
|
|
1
|
+
import { hex } from "@scure/base";
|
|
2
|
+
import { VHTLC } from '../../script/vhtlc.js';
|
|
3
|
+
import { isCsvSpendable, resolveRole, sequenceToTimelock, timelockToSequence, } from './helpers.js';
|
|
4
|
+
/**
|
|
5
|
+
* Handler for Virtual Hash Time Lock Contract (VHTLC).
|
|
6
|
+
*
|
|
7
|
+
* VHTLC supports multiple spending paths:
|
|
8
|
+
*
|
|
9
|
+
* Collaborative paths (with server):
|
|
10
|
+
* - claim: Receiver + Server with preimage
|
|
11
|
+
* - refund: Sender + Receiver + Server
|
|
12
|
+
* - refundWithoutReceiver: Sender + Server after CLTV locktime
|
|
13
|
+
*
|
|
14
|
+
* Unilateral paths (without server):
|
|
15
|
+
* - unilateralClaim: Receiver with preimage after CSV delay
|
|
16
|
+
* - unilateralRefund: Sender + Receiver after CSV delay
|
|
17
|
+
* - unilateralRefundWithoutReceiver: Sender after CSV delay
|
|
18
|
+
*/
|
|
19
|
+
export const VHTLCContractHandler = {
|
|
20
|
+
type: "vhtlc",
|
|
21
|
+
createScript(params) {
|
|
22
|
+
const typed = this.deserializeParams(params);
|
|
23
|
+
return new VHTLC.Script(typed);
|
|
24
|
+
},
|
|
25
|
+
serializeParams(params) {
|
|
26
|
+
return {
|
|
27
|
+
sender: hex.encode(params.sender),
|
|
28
|
+
receiver: hex.encode(params.receiver),
|
|
29
|
+
server: hex.encode(params.server),
|
|
30
|
+
hash: hex.encode(params.preimageHash),
|
|
31
|
+
refundLocktime: params.refundLocktime.toString(),
|
|
32
|
+
claimDelay: timelockToSequence(params.unilateralClaimDelay).toString(),
|
|
33
|
+
refundDelay: timelockToSequence(params.unilateralRefundDelay).toString(),
|
|
34
|
+
refundNoReceiverDelay: timelockToSequence(params.unilateralRefundWithoutReceiverDelay).toString(),
|
|
35
|
+
};
|
|
36
|
+
},
|
|
37
|
+
deserializeParams(params) {
|
|
38
|
+
return {
|
|
39
|
+
sender: hex.decode(params.sender),
|
|
40
|
+
receiver: hex.decode(params.receiver),
|
|
41
|
+
server: hex.decode(params.server),
|
|
42
|
+
preimageHash: hex.decode(params.hash),
|
|
43
|
+
refundLocktime: BigInt(params.refundLocktime),
|
|
44
|
+
unilateralClaimDelay: sequenceToTimelock(Number(params.claimDelay)),
|
|
45
|
+
unilateralRefundDelay: sequenceToTimelock(Number(params.refundDelay)),
|
|
46
|
+
unilateralRefundWithoutReceiverDelay: sequenceToTimelock(Number(params.refundNoReceiverDelay)),
|
|
47
|
+
};
|
|
48
|
+
},
|
|
49
|
+
/**
|
|
50
|
+
* Select spending path based on context.
|
|
51
|
+
*
|
|
52
|
+
* Role is determined from `context.role` or by matching `context.walletPubKey`
|
|
53
|
+
* against sender/receiver in contract params.
|
|
54
|
+
*/
|
|
55
|
+
selectPath(script, contract, context) {
|
|
56
|
+
const role = resolveRole(contract, context);
|
|
57
|
+
const preimage = contract.params?.preimage;
|
|
58
|
+
const refundLocktime = BigInt(contract.params.refundLocktime);
|
|
59
|
+
const currentTimeSec = Math.floor(context.currentTime / 1000);
|
|
60
|
+
if (!role) {
|
|
61
|
+
return null;
|
|
62
|
+
}
|
|
63
|
+
if (context.collaborative) {
|
|
64
|
+
if (role === "receiver" && preimage) {
|
|
65
|
+
return {
|
|
66
|
+
leaf: script.claim(),
|
|
67
|
+
extraWitness: [hex.decode(preimage)],
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
if (role === "sender" && BigInt(currentTimeSec) >= refundLocktime) {
|
|
71
|
+
return {
|
|
72
|
+
leaf: script.refundWithoutReceiver(),
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
return null;
|
|
76
|
+
}
|
|
77
|
+
// Unilateral paths
|
|
78
|
+
if (role === "receiver" && preimage) {
|
|
79
|
+
const sequence = Number(contract.params.claimDelay);
|
|
80
|
+
if (!isCsvSpendable(context, sequence))
|
|
81
|
+
return null;
|
|
82
|
+
return {
|
|
83
|
+
leaf: script.unilateralClaim(),
|
|
84
|
+
extraWitness: [hex.decode(preimage)],
|
|
85
|
+
sequence,
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
if (role === "sender") {
|
|
89
|
+
const sequence = Number(contract.params.refundNoReceiverDelay);
|
|
90
|
+
if (!isCsvSpendable(context, sequence))
|
|
91
|
+
return null;
|
|
92
|
+
return {
|
|
93
|
+
leaf: script.unilateralRefundWithoutReceiver(),
|
|
94
|
+
sequence,
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
return null;
|
|
98
|
+
},
|
|
99
|
+
/**
|
|
100
|
+
* Get all possible spending paths (no timelock checks).
|
|
101
|
+
*
|
|
102
|
+
* Role is determined from `context.role` or by matching `context.walletPubKey`
|
|
103
|
+
* against sender/receiver in contract params.
|
|
104
|
+
*/
|
|
105
|
+
getAllSpendingPaths(script, contract, context) {
|
|
106
|
+
const role = resolveRole(contract, context);
|
|
107
|
+
const paths = [];
|
|
108
|
+
if (!role) {
|
|
109
|
+
return paths;
|
|
110
|
+
}
|
|
111
|
+
const preimage = contract.params?.preimage;
|
|
112
|
+
if (context.collaborative) {
|
|
113
|
+
// Collaborative paths (no timelock checks)
|
|
114
|
+
if (role === "receiver" && preimage) {
|
|
115
|
+
paths.push({
|
|
116
|
+
leaf: script.claim(),
|
|
117
|
+
extraWitness: [hex.decode(preimage)],
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
if (role === "sender") {
|
|
121
|
+
paths.push({
|
|
122
|
+
leaf: script.refundWithoutReceiver(),
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
else {
|
|
127
|
+
// Unilateral paths (no timelock checks)
|
|
128
|
+
if (role === "receiver" && preimage) {
|
|
129
|
+
const sequence = Number(contract.params.claimDelay);
|
|
130
|
+
paths.push({
|
|
131
|
+
leaf: script.unilateralClaim(),
|
|
132
|
+
extraWitness: [hex.decode(preimage)],
|
|
133
|
+
sequence,
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
if (role === "sender") {
|
|
137
|
+
const sequence = Number(contract.params.refundNoReceiverDelay);
|
|
138
|
+
paths.push({
|
|
139
|
+
leaf: script.unilateralRefundWithoutReceiver(),
|
|
140
|
+
sequence,
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
return paths;
|
|
145
|
+
},
|
|
146
|
+
getSpendablePaths(script, contract, context) {
|
|
147
|
+
const role = resolveRole(contract, context);
|
|
148
|
+
const paths = [];
|
|
149
|
+
if (!role) {
|
|
150
|
+
return paths;
|
|
151
|
+
}
|
|
152
|
+
const preimage = contract.params?.preimage;
|
|
153
|
+
const refundLocktime = BigInt(contract.params.refundLocktime);
|
|
154
|
+
const currentTimeSec = Math.floor(context.currentTime / 1000);
|
|
155
|
+
if (context.collaborative) {
|
|
156
|
+
if (role === "receiver" && preimage) {
|
|
157
|
+
paths.push({
|
|
158
|
+
leaf: script.claim(),
|
|
159
|
+
extraWitness: [hex.decode(preimage)],
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
if (role === "sender" && BigInt(currentTimeSec) >= refundLocktime) {
|
|
163
|
+
paths.push({
|
|
164
|
+
leaf: script.refundWithoutReceiver(),
|
|
165
|
+
});
|
|
166
|
+
}
|
|
167
|
+
return paths;
|
|
168
|
+
}
|
|
169
|
+
if (role === "receiver" && preimage) {
|
|
170
|
+
const sequence = Number(contract.params.claimDelay);
|
|
171
|
+
if (isCsvSpendable(context, sequence)) {
|
|
172
|
+
paths.push({
|
|
173
|
+
leaf: script.unilateralClaim(),
|
|
174
|
+
extraWitness: [hex.decode(preimage)],
|
|
175
|
+
sequence,
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
if (role === "sender") {
|
|
180
|
+
const sequence = Number(contract.params.refundNoReceiverDelay);
|
|
181
|
+
if (isCsvSpendable(context, sequence)) {
|
|
182
|
+
paths.push({
|
|
183
|
+
leaf: script.unilateralRefundWithoutReceiver(),
|
|
184
|
+
sequence,
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
return paths;
|
|
189
|
+
},
|
|
190
|
+
};
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
// Types
|
|
2
|
+
export * from './types.js';
|
|
3
|
+
// Contract handlers
|
|
4
|
+
export { contractHandlers } from './handlers/index.js';
|
|
5
|
+
export { DefaultContractHandler } from './handlers/index.js';
|
|
6
|
+
export { DelegateContractHandler } from './handlers/index.js';
|
|
7
|
+
export { VHTLCContractHandler } from './handlers/index.js';
|
|
8
|
+
// arkcontract string codec
|
|
9
|
+
export { encodeArkContract, decodeArkContract, contractFromArkContract, contractFromArkContractWithAddress, isArkContract, } from './arkcontract.js';
|
|
10
|
+
// Contract watcher
|
|
11
|
+
export { ContractWatcher } from './contractWatcher.js';
|
|
12
|
+
// Contract manager
|
|
13
|
+
export { ContractManager } from './contractManager.js';
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
export function getGlobalObject() {
|
|
2
|
+
if (typeof globalThis !== "undefined") {
|
|
3
|
+
if (typeof globalThis.self === "object" && globalThis.self !== null) {
|
|
4
|
+
return { globalObject: globalThis.self };
|
|
5
|
+
}
|
|
6
|
+
if (typeof globalThis.window === "object" &&
|
|
7
|
+
globalThis.window !== null) {
|
|
8
|
+
return { globalObject: globalThis.window };
|
|
9
|
+
}
|
|
10
|
+
return { globalObject: globalThis };
|
|
11
|
+
}
|
|
12
|
+
throw new Error("Global object not found");
|
|
13
|
+
}
|
|
14
|
+
// database instance cache, avoiding multiple open requests
|
|
15
|
+
const dbCache = new Map();
|
|
16
|
+
// track reference counts for each database to avoid closing it prematurely
|
|
17
|
+
const refCounts = new Map();
|
|
18
|
+
/**
|
|
19
|
+
* Opens an IndexedDB database and increments the reference count.
|
|
20
|
+
* Handles global object detection and callbacks.
|
|
21
|
+
*
|
|
22
|
+
* @param dbName The name of the database to open.
|
|
23
|
+
* @param dbVersion The database version to open.
|
|
24
|
+
* @param initDatabase A function that migrates the database schema, called on `onupgradeneeded` only.
|
|
25
|
+
*
|
|
26
|
+
* @returns A promise that resolves to the database instance.
|
|
27
|
+
*/
|
|
28
|
+
export async function openDatabase(dbName, dbVersion, initDatabase) {
|
|
29
|
+
const { globalObject } = getGlobalObject();
|
|
30
|
+
if (!globalObject.indexedDB) {
|
|
31
|
+
throw new Error("IndexedDB is not available in this environment");
|
|
32
|
+
}
|
|
33
|
+
// Return cached promise if available (handles concurrent calls)
|
|
34
|
+
const cached = dbCache.get(dbName);
|
|
35
|
+
if (cached) {
|
|
36
|
+
if (cached.version !== dbVersion) {
|
|
37
|
+
throw new Error(`Database "${dbName}" already opened with version ${cached.version}; requested ${dbVersion}`);
|
|
38
|
+
}
|
|
39
|
+
refCounts.set(dbName, (refCounts.get(dbName) ?? 0) + 1);
|
|
40
|
+
return cached.promise;
|
|
41
|
+
}
|
|
42
|
+
const dbPromise = new Promise((resolve, reject) => {
|
|
43
|
+
const request = globalObject.indexedDB.open(dbName, dbVersion);
|
|
44
|
+
request.onerror = () => {
|
|
45
|
+
dbCache.delete(dbName); // Clean up on failure
|
|
46
|
+
refCounts.delete(dbName);
|
|
47
|
+
reject(request.error);
|
|
48
|
+
};
|
|
49
|
+
request.onsuccess = () => {
|
|
50
|
+
resolve(request.result);
|
|
51
|
+
};
|
|
52
|
+
request.onupgradeneeded = () => {
|
|
53
|
+
const db = request.result;
|
|
54
|
+
initDatabase(db);
|
|
55
|
+
};
|
|
56
|
+
request.onblocked = () => {
|
|
57
|
+
console.warn("Database upgrade blocked - close other tabs/connections");
|
|
58
|
+
};
|
|
59
|
+
});
|
|
60
|
+
// Cache immediately before awaiting
|
|
61
|
+
dbCache.set(dbName, { version: dbVersion, promise: dbPromise });
|
|
62
|
+
refCounts.set(dbName, 1);
|
|
63
|
+
return dbPromise;
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Decrements the reference count and closes the database when no references remain.
|
|
67
|
+
*
|
|
68
|
+
* @param dbName The name of the database to close.
|
|
69
|
+
*
|
|
70
|
+
* @returns True if the database was closed, false otherwise.
|
|
71
|
+
*/
|
|
72
|
+
export async function closeDatabase(dbName) {
|
|
73
|
+
const cachedEntry = dbCache.get(dbName);
|
|
74
|
+
if (!cachedEntry)
|
|
75
|
+
return false;
|
|
76
|
+
const count = (refCounts.get(dbName) ?? 1) - 1;
|
|
77
|
+
if (count > 0) {
|
|
78
|
+
refCounts.set(dbName, count);
|
|
79
|
+
return false;
|
|
80
|
+
}
|
|
81
|
+
// Last reference — actually close
|
|
82
|
+
refCounts.delete(dbName);
|
|
83
|
+
dbCache.delete(dbName);
|
|
84
|
+
try {
|
|
85
|
+
const db = await cachedEntry.promise;
|
|
86
|
+
db.close();
|
|
87
|
+
}
|
|
88
|
+
catch {
|
|
89
|
+
// DB failed to open, nothing to close
|
|
90
|
+
}
|
|
91
|
+
return true;
|
|
92
|
+
}
|
package/dist/esm/forfeit.js
CHANGED
|
@@ -1,24 +1,27 @@
|
|
|
1
1
|
import { Transaction } from './utils/transaction.js';
|
|
2
2
|
import { P2A } from './utils/anchor.js';
|
|
3
3
|
export function buildForfeitTx(inputs, forfeitPkScript, txLocktime) {
|
|
4
|
-
const tx = new Transaction({
|
|
5
|
-
version: 3,
|
|
6
|
-
lockTime: txLocktime,
|
|
7
|
-
});
|
|
8
4
|
let amount = 0n;
|
|
9
5
|
for (const input of inputs) {
|
|
10
6
|
if (!input.witnessUtxo) {
|
|
11
7
|
throw new Error("input needs witness utxo");
|
|
12
8
|
}
|
|
13
9
|
amount += input.witnessUtxo.amount;
|
|
14
|
-
tx.addInput(input);
|
|
15
10
|
}
|
|
16
|
-
|
|
17
|
-
tx.addOutput({
|
|
11
|
+
return buildForfeitTxWithOutput(inputs, {
|
|
18
12
|
script: forfeitPkScript,
|
|
19
13
|
amount,
|
|
14
|
+
}, txLocktime);
|
|
15
|
+
}
|
|
16
|
+
export function buildForfeitTxWithOutput(inputs, output, txLocktime) {
|
|
17
|
+
const tx = new Transaction({
|
|
18
|
+
version: 3,
|
|
19
|
+
lockTime: txLocktime,
|
|
20
20
|
});
|
|
21
|
-
|
|
21
|
+
for (const input of inputs) {
|
|
22
|
+
tx.addInput(input);
|
|
23
|
+
}
|
|
24
|
+
tx.addOutput(output);
|
|
22
25
|
tx.addOutput(P2A);
|
|
23
26
|
return tx;
|
|
24
27
|
}
|