diff --git a/.env.example b/.env.example
index 0e20491..799d0b8 100644
--- a/.env.example
+++ b/.env.example
@@ -1,4 +1,4 @@
-# EVM CHAINS
+# EVM Chains
EVM_RPC_URL='https://sepolia.infura.io/v3/9aa3d95b3bc440fa88ea12eaa4456161'
EVM_WS_URL='wss://sepolia.infura.io/ws/v3/9aa3d95b3bc440fa88ea12eaa4456161'
@@ -13,7 +13,7 @@ EVM_TRANSACTION_LISTENER_TEST_IS_ACTIVE=false
EVM_COIN_BALANCE_TEST_AMOUNT=0.01
EVM_TOKEN_BALANCE_TEST_AMOUNT=1000
EVM_NFT_BALANCE_TEST_AMOUNT=2
-EVM_TRANSFER_TEST_AMOUNT=0.0001
+EVM_TRANSFER_TEST_AMOUNT=0.01
EVM_TOKEN_TRANSFER_TEST_AMOUNT=1
EVM_TOKEN_APPROVE_TEST_AMOUNT=100
EVM_NFT_TRANSFER_ID=7
@@ -43,9 +43,9 @@ EVM_ETHER_TRANSFER_TX='0x566002399664e92f82ed654c181095bdd7ff3d3f1921d9632575858
EVM_TOKEN_TRANSFER_TX='0xdabda3905e585db91768f2ef877f7fbef7c0e8612c0a09c7b379981bdbc48975'
EVM_NFT_TRANSFER_TX='0x272a4698cd2062f2463481cf9eb78b68b35d59938383679b7642e6d669ac87eb'
# Models
-# EVM CHAINS
+# EVM Chains
-#TRON
+#Tron
# Assets
TRON_COIN_TRANSFER_TEST_IS_ACTIVE=false
TRON_TOKEN_TRANSFER_TEST_IS_ACTIVE=false
@@ -82,4 +82,58 @@ TRON_MODEL_TEST_RECEIVER='TS1WYZNoNw32hog68m5GyhwZnkhf6HNzhi'
TRON_TRX_TRANSFER_TX='8697ad2c4e1713227c16a65a5845636458df2d3db3adf526e07e17699bc6b3c4'
TRON_TOKEN_TRANSFER_TX='bd0ba6ebb8d2f910b27de1565c66cc89337b792dfdb6484847c817ccbd240760'
TRON_NFT_TRANSFER_TX='d5dd97c09efdb93f36808a9f8e14642ef226880aa91a846583d4ce98c0084637'
-#TRON
\ No newline at end of file
+#Tron
+
+#Bitcoin
+BTC_TRANSFER_TEST_IS_ACTIVE=false
+BTC_LISTENER_TEST_IS_ACTIVE=false
+BTC_BLOCKCYPHER_TOKEN='49d43a59a4f24d31a9731eb067ab971c'
+BTC_SENDER_PRIVATE_KEY='cNHUtnWqwAwGajUGjyHLNbUcfHaDC3ujqjc6qcZik5Xa58Hj46vG'
+BTC_SENDER_ADDRESS='tb1q8juz7c302wdcpfz83zvvyf4jxc8sfq4wyth3pr'
+BTC_RECEIVER_ADDRESS='tb1q9uxj8p043sjkm0qzlsys7677mv98j76k8cvgtg'
+BTC_TRANSFER_AMOUNT=0.00001
+#Bitcoin
+
+#Solana
+# Assets
+SOL_COIN_TRANSFER_TEST_IS_ACTIVE=false
+SOL_TOKEN_TRANSFER_TEST_IS_ACTIVE=false
+SOL_TOKEN_APPROVE_TEST_IS_ACTIVE=false
+SOL_TOKEN_TRANSFER_FROM_TEST_IS_ACTIVE=false
+SOL_NFT_TRANSACTION_TEST_IS_ACTIVE=false
+SOL_TRANSACTION_LISTENER_TEST_IS_ACTIVE=false
+
+SOL_COIN_BALANCE_TEST_AMOUNT=1
+SOL_TOKEN_BALANCE_TEST_AMOUNT=20
+SOL_NFT_BALANCE_TEST_AMOUNT=2
+SOL_TRANSFER_TEST_AMOUNT=0.001
+SOL_TOKEN_TRANSFER_TEST_AMOUNT=1
+SOL_TOKEN_APPROVE_TEST_AMOUNT=10
+#NFT ID
+SOL_NFT_TRANSFER_ID='FxN19KB5UeZJFwxLFgT57WvYYXYhBFKxVumfq37xU4Ck'
+SOL_NFT_TRANSFER_ID2='EBQjkpi6VdY3WF9ihi1sVhjkNDjfCorg1LJGd3Vnn9Ec'
+SOL_TOKEN_PROGRAM='TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA'
+SOL_BALANCE_TEST_ADDRESS='2i6NifwwBGJKUEhYhqbGwu66kDhbfyjgp7zPN5dQsKsE'
+SOL_SENDER_PRIVATE_KEY='2hh51RHd1cEgiuEj2azZJFNphh27S8578otEHw8WvUia1yEBaW7o7XNuxdV1qQYbvxL5FHrcHpbZMCCxULTkCHRq'
+SOL_RECEIVER_PRIVATE_KEY='57XnG9wtVMVV5BZnZbmaN3JTQ7Mzkmbyh4ooMC45YypJz3gF16nGfCJw3hWGQjLFbKmruaF3N3BH3fJAGD6stGCb'
+SOL_SENDER_TEST_ADDRESS='7bn9So6CKmdn2vHYZU5oG6gaVEewW3N7U8vJoNmSv2vB'
+SOL_RECEIVER_TEST_ADDRESS='GHVMV3zZscR8V4K5GEgQTUjuV4jEixB9a4QX3KhgvVKy'
+SOL_TOKEN_TEST_ADDRESS='2ZHwL3dXk3szRgiBLZi244NmKs2VmoBx764AYMY2tQfx'
+SOL_TOKEN_2022_TEST_ADDRESS='FQPbc46pp1b3QWHFJFfsvQrv7YedX8XEum5c18mLnipE'
+#Collection Mint Address
+SOL_NFT_TEST_ADDRESS='9B8bxQWovwm5na85xtDPuAyUVingcVdqWk3d9yQV2zsA'
+# Assets
+
+# Models
+SOL_NFT_ID='BFqd1SZFWMci42bUrhxzWhnvodCnG32xKYZkSp2rXVAf'
+SOL_TOKEN_AMOUNT=50000
+SOL_COIN_AMOUNT=10
+
+SOL_MODEL_TEST_SENDER='37p742pby4ACHiGcT3d58gsjmC3Kd9bH2L89E3hY8FHZ'
+SOL_MODEL_TEST_RECEIVER='7bn9So6CKmdn2vHYZU5oG6gaVEewW3N7U8vJoNmSv2vB'
+
+SOL_TRANSFER_TX='2RDU1otuPR6UtevwYCQWnngvvjPiTFuHFdyCnzwQVR8wyZ7niqACt2QBmfuyD5aXJbEXSEUcqFquiCEtcQZzkWif'
+SOL_TOKEN_TRANSFER_TX='4XLpHmpiKXXDM7pAg8CXeSLjw7SYKZaSzJjXHP2E1vL2ndvrJ6GnuHUvaQpY3LHQeJww8fzFLJ9MiLnvgsyyyt3i'
+SOL_TOKEN_2022_TRANSFER_TX='3c2Myd3k4Pw1NbsCjuskkZCtbD9HRTjoyoxh2u7qsVLgFN1RbRYAXXRKzBRzwTAmv2pXDjArbotzVL6AVehBMeyg'
+SOL_NFT_TRANSFER_TX='3vrCoNVmeNgGG4LB1qvvdx21TYm6dnPBmhFqXChsusuLn5ZEjFZNFG3BwQQ8fodBYiPXG8QokdBLWjRtxgi7tnRD'
+#Solana
\ No newline at end of file
diff --git a/package.json b/package.json
index 22035f9..47d94e0 100644
--- a/package.json
+++ b/package.json
@@ -4,6 +4,7 @@
"type": "module",
"scripts": {
"test": "vitest",
+ "test-ui": "vitest watch --ui",
"coverage": "vitest run --coverage",
"format": "prettier --write packages/",
"lint": "eslint . --ext .ts --ignore-path .gitignore"
@@ -12,6 +13,7 @@
"@types/node": "^20.11.20",
"@typescript-eslint/eslint-plugin": "^6.21.0",
"@vitest/coverage-istanbul": "^1.5.3",
+ "@vitest/ui": "^1.6.0",
"esbuild": "^0.20.2",
"eslint": "^8.57.0",
"eslint-config-prettier": "^9.1.0",
diff --git a/packages/networks/bitcoin/.eslintrc.json b/packages/networks/bitcoin/.eslintrc.json
new file mode 100644
index 0000000..b62472f
--- /dev/null
+++ b/packages/networks/bitcoin/.eslintrc.json
@@ -0,0 +1,7 @@
+{
+ "extends": [
+ "plugin:require-extensions/recommended",
+ "../../../.eslintrc.json"
+ ],
+ "plugins": ["require-extensions"]
+}
\ No newline at end of file
diff --git a/packages/networks/bitcoin/.npmignore b/packages/networks/bitcoin/.npmignore
new file mode 100644
index 0000000..4f4a3d7
--- /dev/null
+++ b/packages/networks/bitcoin/.npmignore
@@ -0,0 +1,6 @@
+src
+tests
+node_modules
+.eslintrc.cjs
+pnpm-lock.yaml
+tsconfig.json
\ No newline at end of file
diff --git a/packages/networks/bitcoin/README.md b/packages/networks/bitcoin/README.md
new file mode 100644
index 0000000..e69de29
diff --git a/packages/networks/bitcoin/esbuild.ts b/packages/networks/bitcoin/esbuild.ts
new file mode 100644
index 0000000..5714b9c
--- /dev/null
+++ b/packages/networks/bitcoin/esbuild.ts
@@ -0,0 +1,3 @@
+void import('../../../esbuild.ts').then((module) => {
+ module.default()
+})
diff --git a/packages/networks/bitcoin/index.html b/packages/networks/bitcoin/index.html
new file mode 100644
index 0000000..2324a82
--- /dev/null
+++ b/packages/networks/bitcoin/index.html
@@ -0,0 +1,324 @@
+
+
+
+
+
+
+ Browser Tests
+
+
+
+
+
+
+
+
+ Adapter id:
+
+
+ Adapter name:
+
+
+
Adapter icon: ![icon]()
+
+
+ Platforms:
+
+
+ Download link:
+
+
+ Deep link:
+
+
+ Connected address:
+
+
+
+ Result:
+
+
+
+ Result:
+
+
+
+
+
+
\ No newline at end of file
diff --git a/packages/networks/bitcoin/package-lock.json b/packages/networks/bitcoin/package-lock.json
new file mode 100644
index 0000000..312fc86
--- /dev/null
+++ b/packages/networks/bitcoin/package-lock.json
@@ -0,0 +1,772 @@
+{
+ "name": "@multiplechain/bitcoin",
+ "version": "0.1.0",
+ "lockfileVersion": 3,
+ "requires": true,
+ "packages": {
+ "": {
+ "name": "@multiplechain/bitcoin",
+ "version": "0.1.0",
+ "license": "MIT",
+ "dependencies": {
+ "@multiplechain/types": "^0.1.55",
+ "@multiplechain/utils": "^0.1.18",
+ "axios": "^1.6.8",
+ "bitcore-lib": "^10.0.28",
+ "ws": "^8.17.0"
+ },
+ "devDependencies": {
+ "@types/bitcore-lib": "^0.15.6"
+ }
+ },
+ "node_modules/@multiplechain/types": {
+ "version": "0.1.55",
+ "resolved": "https://registry.npmjs.org/@multiplechain/types/-/types-0.1.55.tgz",
+ "integrity": "sha512-9fYrLaDxX2pj9zfIbmvk1hPF3FH/bK+Q3uTF+k3zdsitP/HzQ1S9zImNz20Tvoqkk1ns+/8ecE0Y6GNowsDOQA=="
+ },
+ "node_modules/@multiplechain/utils": {
+ "version": "0.1.18",
+ "resolved": "https://registry.npmjs.org/@multiplechain/utils/-/utils-0.1.18.tgz",
+ "integrity": "sha512-UCoOOBJrawp/lInepxuoEiYwId0wMPg7rX8p78FLqzGl8f/b93sAtv7sP87/VVKwhdoEuNZ/uQjckycmNeks/w==",
+ "dependencies": {
+ "@types/ws": "^8.5.10",
+ "bignumber.js": "^9.1.2",
+ "web3-utils": "^4.2.0",
+ "ws": "^8.16.0"
+ }
+ },
+ "node_modules/@noble/curves": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/@noble/curves/-/curves-1.3.0.tgz",
+ "integrity": "sha512-t01iSXPuN+Eqzb4eBX0S5oubSqXbK/xXa1Ne18Hj8f9pStxztHCE2gfboSp/dZRLSqfuLpRK2nDXDK+W9puocA==",
+ "dependencies": {
+ "@noble/hashes": "1.3.3"
+ },
+ "funding": {
+ "url": "https://paulmillr.com/funding/"
+ }
+ },
+ "node_modules/@noble/hashes": {
+ "version": "1.3.3",
+ "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.3.3.tgz",
+ "integrity": "sha512-V7/fPHgl+jsVPXqqeOzT8egNj2iBIVt+ECeMMG8TdcnTikP3oaBtUVqpT/gYCR68aEBJSF+XbYUxStjbFMqIIA==",
+ "engines": {
+ "node": ">= 16"
+ },
+ "funding": {
+ "url": "https://paulmillr.com/funding/"
+ }
+ },
+ "node_modules/@scure/base": {
+ "version": "1.1.6",
+ "resolved": "https://registry.npmjs.org/@scure/base/-/base-1.1.6.tgz",
+ "integrity": "sha512-ok9AWwhcgYuGG3Zfhyqg+zwl+Wn5uE+dwC0NV/2qQkx4dABbb/bx96vWu8NSj+BNjjSjno+JRYRjle1jV08k3g==",
+ "funding": {
+ "url": "https://paulmillr.com/funding/"
+ }
+ },
+ "node_modules/@scure/bip32": {
+ "version": "1.3.3",
+ "resolved": "https://registry.npmjs.org/@scure/bip32/-/bip32-1.3.3.tgz",
+ "integrity": "sha512-LJaN3HwRbfQK0X1xFSi0Q9amqOgzQnnDngIt+ZlsBC3Bm7/nE7K0kwshZHyaru79yIVRv/e1mQAjZyuZG6jOFQ==",
+ "dependencies": {
+ "@noble/curves": "~1.3.0",
+ "@noble/hashes": "~1.3.2",
+ "@scure/base": "~1.1.4"
+ },
+ "funding": {
+ "url": "https://paulmillr.com/funding/"
+ }
+ },
+ "node_modules/@scure/bip39": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/@scure/bip39/-/bip39-1.2.2.tgz",
+ "integrity": "sha512-HYf9TUXG80beW+hGAt3TRM8wU6pQoYur9iNypTROm42dorCGmLnFe3eWjz3gOq6G62H2WRh0FCzAR1PI+29zIA==",
+ "dependencies": {
+ "@noble/hashes": "~1.3.2",
+ "@scure/base": "~1.1.4"
+ },
+ "funding": {
+ "url": "https://paulmillr.com/funding/"
+ }
+ },
+ "node_modules/@types/bitcore-lib": {
+ "version": "0.15.6",
+ "resolved": "https://registry.npmjs.org/@types/bitcore-lib/-/bitcore-lib-0.15.6.tgz",
+ "integrity": "sha512-CtKDBgSBubPXZ0wFeCiUCSdzH+cuy6nFya3FboOqf44evi+OmkQPqEg3ASMpmPDYE8vkcxV302Iu8lZqCjYieg==",
+ "dev": true,
+ "dependencies": {
+ "@types/node": "*"
+ }
+ },
+ "node_modules/@types/node": {
+ "version": "20.12.11",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-20.12.11.tgz",
+ "integrity": "sha512-vDg9PZ/zi+Nqp6boSOT7plNuthRugEKixDv5sFTIpkE89MmNtEArAShI4mxuX2+UrLEe9pxC1vm2cjm9YlWbJw==",
+ "dependencies": {
+ "undici-types": "~5.26.4"
+ }
+ },
+ "node_modules/@types/ws": {
+ "version": "8.5.10",
+ "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.10.tgz",
+ "integrity": "sha512-vmQSUcfalpIq0R9q7uTo2lXs6eGIpt9wtnLdMv9LVpIjCA/+ufZRozlVoVelIYixx1ugCBKDhn89vnsEGOCx9A==",
+ "dependencies": {
+ "@types/node": "*"
+ }
+ },
+ "node_modules/asynckit": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q=="
+ },
+ "node_modules/available-typed-arrays": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz",
+ "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==",
+ "dependencies": {
+ "possible-typed-array-names": "^1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/axios": {
+ "version": "1.6.8",
+ "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.8.tgz",
+ "integrity": "sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==",
+ "dependencies": {
+ "follow-redirects": "^1.15.6",
+ "form-data": "^4.0.0",
+ "proxy-from-env": "^1.1.0"
+ }
+ },
+ "node_modules/base-x": {
+ "version": "3.0.9",
+ "resolved": "https://registry.npmjs.org/base-x/-/base-x-3.0.9.tgz",
+ "integrity": "sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ==",
+ "dependencies": {
+ "safe-buffer": "^5.0.1"
+ }
+ },
+ "node_modules/bech32": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/bech32/-/bech32-2.0.0.tgz",
+ "integrity": "sha512-LcknSilhIGatDAsY1ak2I8VtGaHNhgMSYVxFrGLXv+xLHytaKZKcaUJJUE7qmBr7h33o5YQwP55pMI0xmkpJwg=="
+ },
+ "node_modules/bigi": {
+ "version": "1.4.2",
+ "resolved": "https://registry.npmjs.org/bigi/-/bigi-1.4.2.tgz",
+ "integrity": "sha512-ddkU+dFIuEIW8lE7ZwdIAf2UPoM90eaprg5m3YXAVVTmKlqV/9BX4A2M8BOK2yOq6/VgZFVhK6QAxJebhlbhzw=="
+ },
+ "node_modules/bignumber.js": {
+ "version": "9.1.2",
+ "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.1.2.tgz",
+ "integrity": "sha512-2/mKyZH9K85bzOEfhXDBFZTGd1CTs+5IHpeFQo9luiBG7hghdC851Pj2WAhb6E3R6b9tZj/XKhbg4fum+Kepug==",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/bip-schnorr": {
+ "version": "0.6.4",
+ "resolved": "https://registry.npmjs.org/bip-schnorr/-/bip-schnorr-0.6.4.tgz",
+ "integrity": "sha512-dNKw7Lea8B0wMIN4OjEmOk/Z5qUGqoPDY0P2QttLqGk1hmDPytLWW8PR5Pb6Vxy6CprcdEgfJpOjUu+ONQveyg==",
+ "dependencies": {
+ "bigi": "^1.4.2",
+ "ecurve": "^1.0.6",
+ "js-sha256": "^0.9.0",
+ "randombytes": "^2.1.0",
+ "safe-buffer": "^5.2.1"
+ },
+ "engines": {
+ "node": ">=8.0.0"
+ }
+ },
+ "node_modules/bitcore-lib": {
+ "version": "10.0.28",
+ "resolved": "https://registry.npmjs.org/bitcore-lib/-/bitcore-lib-10.0.28.tgz",
+ "integrity": "sha512-uOWHpWbUxEj411p+tp6DCb9NfZdsCAHl6Z/rs96mquQMsef29fOqWUyk4Bl7yLf+KwzU5ZKy0HIPnjGFlmpXpg==",
+ "dependencies": {
+ "bech32": "=2.0.0",
+ "bip-schnorr": "=0.6.4",
+ "bn.js": "=4.11.8",
+ "bs58": "^4.0.1",
+ "buffer-compare": "=1.1.1",
+ "elliptic": "^6.5.3",
+ "inherits": "=2.0.1",
+ "lodash": "^4.17.20"
+ }
+ },
+ "node_modules/bitcore-lib/node_modules/inherits": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz",
+ "integrity": "sha512-8nWq2nLTAwd02jTqJExUYFSD/fKq6VH9Y/oG2accc/kdI0V98Bag8d5a4gi3XHz73rDWa2PvTtvcWYquKqSENA=="
+ },
+ "node_modules/bn.js": {
+ "version": "4.11.8",
+ "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz",
+ "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA=="
+ },
+ "node_modules/brorand": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz",
+ "integrity": "sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w=="
+ },
+ "node_modules/bs58": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/bs58/-/bs58-4.0.1.tgz",
+ "integrity": "sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==",
+ "dependencies": {
+ "base-x": "^3.0.2"
+ }
+ },
+ "node_modules/buffer-compare": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/buffer-compare/-/buffer-compare-1.1.1.tgz",
+ "integrity": "sha512-O6NvNiHZMd3mlIeMDjP6t/gPG75OqGPeiRZXoMQZJ6iy9GofCls4Ijs5YkPZZwoysizLiedhticmdyx/GyHghA=="
+ },
+ "node_modules/call-bind": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz",
+ "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==",
+ "dependencies": {
+ "es-define-property": "^1.0.0",
+ "es-errors": "^1.3.0",
+ "function-bind": "^1.1.2",
+ "get-intrinsic": "^1.2.4",
+ "set-function-length": "^1.2.1"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/combined-stream": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+ "dependencies": {
+ "delayed-stream": "~1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/define-data-property": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz",
+ "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==",
+ "dependencies": {
+ "es-define-property": "^1.0.0",
+ "es-errors": "^1.3.0",
+ "gopd": "^1.0.1"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/delayed-stream": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+ "engines": {
+ "node": ">=0.4.0"
+ }
+ },
+ "node_modules/ecurve": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/ecurve/-/ecurve-1.0.6.tgz",
+ "integrity": "sha512-/BzEjNfiSuB7jIWKcS/z8FK9jNjmEWvUV2YZ4RLSmcDtP7Lq0m6FvDuSnJpBlDpGRpfRQeTLGLBI8H+kEv0r+w==",
+ "dependencies": {
+ "bigi": "^1.1.0",
+ "safe-buffer": "^5.0.1"
+ }
+ },
+ "node_modules/elliptic": {
+ "version": "6.5.5",
+ "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.5.tgz",
+ "integrity": "sha512-7EjbcmUm17NQFu4Pmgmq2olYMj8nwMnpcddByChSUjArp8F5DQWcIcpriwO4ZToLNAJig0yiyjswfyGNje/ixw==",
+ "dependencies": {
+ "bn.js": "^4.11.9",
+ "brorand": "^1.1.0",
+ "hash.js": "^1.0.0",
+ "hmac-drbg": "^1.0.1",
+ "inherits": "^2.0.4",
+ "minimalistic-assert": "^1.0.1",
+ "minimalistic-crypto-utils": "^1.0.1"
+ }
+ },
+ "node_modules/elliptic/node_modules/bn.js": {
+ "version": "4.12.0",
+ "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+ "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA=="
+ },
+ "node_modules/es-define-property": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz",
+ "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==",
+ "dependencies": {
+ "get-intrinsic": "^1.2.4"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-errors": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/ethereum-cryptography": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ethereum-cryptography/-/ethereum-cryptography-2.1.3.tgz",
+ "integrity": "sha512-BlwbIL7/P45W8FGW2r7LGuvoEZ+7PWsniMvQ4p5s2xCyw9tmaDlpfsN9HjAucbF+t/qpVHwZUisgfK24TCW8aA==",
+ "dependencies": {
+ "@noble/curves": "1.3.0",
+ "@noble/hashes": "1.3.3",
+ "@scure/bip32": "1.3.3",
+ "@scure/bip39": "1.2.2"
+ }
+ },
+ "node_modules/eventemitter3": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-5.0.1.tgz",
+ "integrity": "sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA=="
+ },
+ "node_modules/follow-redirects": {
+ "version": "1.15.6",
+ "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz",
+ "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==",
+ "funding": [
+ {
+ "type": "individual",
+ "url": "https://github.com/sponsors/RubenVerborgh"
+ }
+ ],
+ "engines": {
+ "node": ">=4.0"
+ },
+ "peerDependenciesMeta": {
+ "debug": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/for-each": {
+ "version": "0.3.3",
+ "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz",
+ "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==",
+ "dependencies": {
+ "is-callable": "^1.1.3"
+ }
+ },
+ "node_modules/form-data": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+ "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+ "dependencies": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.8",
+ "mime-types": "^2.1.12"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/function-bind": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/get-intrinsic": {
+ "version": "1.2.4",
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz",
+ "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "function-bind": "^1.1.2",
+ "has-proto": "^1.0.1",
+ "has-symbols": "^1.0.3",
+ "hasown": "^2.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/gopd": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz",
+ "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==",
+ "dependencies": {
+ "get-intrinsic": "^1.1.3"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-property-descriptors": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz",
+ "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==",
+ "dependencies": {
+ "es-define-property": "^1.0.0"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-proto": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz",
+ "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-symbols": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz",
+ "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-tostringtag": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz",
+ "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==",
+ "dependencies": {
+ "has-symbols": "^1.0.3"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/hash.js": {
+ "version": "1.1.7",
+ "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz",
+ "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==",
+ "dependencies": {
+ "inherits": "^2.0.3",
+ "minimalistic-assert": "^1.0.1"
+ }
+ },
+ "node_modules/hasown": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
+ "dependencies": {
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/hmac-drbg": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz",
+ "integrity": "sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==",
+ "dependencies": {
+ "hash.js": "^1.0.3",
+ "minimalistic-assert": "^1.0.0",
+ "minimalistic-crypto-utils": "^1.0.1"
+ }
+ },
+ "node_modules/inherits": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
+ },
+ "node_modules/is-arguments": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz",
+ "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==",
+ "dependencies": {
+ "call-bind": "^1.0.2",
+ "has-tostringtag": "^1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/is-callable": {
+ "version": "1.2.7",
+ "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz",
+ "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/is-generator-function": {
+ "version": "1.0.10",
+ "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz",
+ "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==",
+ "dependencies": {
+ "has-tostringtag": "^1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/is-typed-array": {
+ "version": "1.1.13",
+ "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.13.tgz",
+ "integrity": "sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==",
+ "dependencies": {
+ "which-typed-array": "^1.1.14"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/js-sha256": {
+ "version": "0.9.0",
+ "resolved": "https://registry.npmjs.org/js-sha256/-/js-sha256-0.9.0.tgz",
+ "integrity": "sha512-sga3MHh9sgQN2+pJ9VYZ+1LPwXOxuBJBA5nrR5/ofPfuiJBE2hnjsaN8se8JznOmGLN2p49Pe5U/ttafcs/apA=="
+ },
+ "node_modules/lodash": {
+ "version": "4.17.21",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
+ },
+ "node_modules/mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/mime-types": {
+ "version": "2.1.35",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+ "dependencies": {
+ "mime-db": "1.52.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/minimalistic-assert": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz",
+ "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A=="
+ },
+ "node_modules/minimalistic-crypto-utils": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz",
+ "integrity": "sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg=="
+ },
+ "node_modules/possible-typed-array-names": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz",
+ "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/proxy-from-env": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+ "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
+ },
+ "node_modules/randombytes": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz",
+ "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==",
+ "dependencies": {
+ "safe-buffer": "^5.1.0"
+ }
+ },
+ "node_modules/safe-buffer": {
+ "version": "5.2.1",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
+ "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ]
+ },
+ "node_modules/set-function-length": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz",
+ "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==",
+ "dependencies": {
+ "define-data-property": "^1.1.4",
+ "es-errors": "^1.3.0",
+ "function-bind": "^1.1.2",
+ "get-intrinsic": "^1.2.4",
+ "gopd": "^1.0.1",
+ "has-property-descriptors": "^1.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/undici-types": {
+ "version": "5.26.5",
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz",
+ "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA=="
+ },
+ "node_modules/util": {
+ "version": "0.12.5",
+ "resolved": "https://registry.npmjs.org/util/-/util-0.12.5.tgz",
+ "integrity": "sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==",
+ "dependencies": {
+ "inherits": "^2.0.3",
+ "is-arguments": "^1.0.4",
+ "is-generator-function": "^1.0.7",
+ "is-typed-array": "^1.1.3",
+ "which-typed-array": "^1.1.2"
+ }
+ },
+ "node_modules/web3-errors": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/web3-errors/-/web3-errors-1.1.4.tgz",
+ "integrity": "sha512-WahtszSqILez+83AxGecVroyZsMuuRT+KmQp4Si5P4Rnqbczno1k748PCrZTS1J4UCPmXMG2/Vt+0Bz2zwXkwQ==",
+ "dependencies": {
+ "web3-types": "^1.3.1"
+ },
+ "engines": {
+ "node": ">=14",
+ "npm": ">=6.12.0"
+ }
+ },
+ "node_modules/web3-types": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/web3-types/-/web3-types-1.6.0.tgz",
+ "integrity": "sha512-qgOtADqlD5hw+KPKBUGaXAcdNLL0oh6qTeVgXwewCfbL/lG9R+/GrgMQB1gbTJ3cit8hMwtH8KX2Em6OwO0HRw==",
+ "engines": {
+ "node": ">=14",
+ "npm": ">=6.12.0"
+ }
+ },
+ "node_modules/web3-utils": {
+ "version": "4.2.3",
+ "resolved": "https://registry.npmjs.org/web3-utils/-/web3-utils-4.2.3.tgz",
+ "integrity": "sha512-m5plKTC2YtQntHITQRyIePw52UVP1IrShhmA2FACtn4zmc5ADmrXOlQWiPzxFP/18eRJsAaUAw2+CQn1u4WPxQ==",
+ "dependencies": {
+ "ethereum-cryptography": "^2.0.0",
+ "eventemitter3": "^5.0.1",
+ "web3-errors": "^1.1.4",
+ "web3-types": "^1.6.0",
+ "web3-validator": "^2.0.5"
+ },
+ "engines": {
+ "node": ">=14",
+ "npm": ">=6.12.0"
+ }
+ },
+ "node_modules/web3-validator": {
+ "version": "2.0.5",
+ "resolved": "https://registry.npmjs.org/web3-validator/-/web3-validator-2.0.5.tgz",
+ "integrity": "sha512-2gLOSW8XqEN5pw5jVUm20EB7A8SbQiekpAtiI0JBmCIV0a2rp97v8FgWY5E3UEqnw5WFfEqvcDVW92EyynDTyQ==",
+ "dependencies": {
+ "ethereum-cryptography": "^2.0.0",
+ "util": "^0.12.5",
+ "web3-errors": "^1.1.4",
+ "web3-types": "^1.5.0",
+ "zod": "^3.21.4"
+ },
+ "engines": {
+ "node": ">=14",
+ "npm": ">=6.12.0"
+ }
+ },
+ "node_modules/which-typed-array": {
+ "version": "1.1.15",
+ "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.15.tgz",
+ "integrity": "sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==",
+ "dependencies": {
+ "available-typed-arrays": "^1.0.7",
+ "call-bind": "^1.0.7",
+ "for-each": "^0.3.3",
+ "gopd": "^1.0.1",
+ "has-tostringtag": "^1.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/ws": {
+ "version": "8.17.0",
+ "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.0.tgz",
+ "integrity": "sha512-uJq6108EgZMAl20KagGkzCKfMEjxmKvZHG7Tlq0Z6nOky7YF7aq4mOx6xK8TJ/i1LeK4Qus7INktacctDgY8Ow==",
+ "engines": {
+ "node": ">=10.0.0"
+ },
+ "peerDependencies": {
+ "bufferutil": "^4.0.1",
+ "utf-8-validate": ">=5.0.2"
+ },
+ "peerDependenciesMeta": {
+ "bufferutil": {
+ "optional": true
+ },
+ "utf-8-validate": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/zod": {
+ "version": "3.23.8",
+ "resolved": "https://registry.npmjs.org/zod/-/zod-3.23.8.tgz",
+ "integrity": "sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==",
+ "funding": {
+ "url": "https://github.com/sponsors/colinhacks"
+ }
+ }
+ }
+}
diff --git a/packages/networks/bitcoin/package.json b/packages/networks/bitcoin/package.json
new file mode 100644
index 0000000..7040f81
--- /dev/null
+++ b/packages/networks/bitcoin/package.json
@@ -0,0 +1,75 @@
+{
+ "name": "@multiplechain/bitcoin",
+ "version": "0.4.0",
+ "type": "module",
+ "main": "dist/index.cjs",
+ "module": "dist/index.es.js",
+ "unpkg": "dist/index.umd.js",
+ "browser": "dist/index.umd.js",
+ "jsdelivr": "dist/index.umd.js",
+ "exports": {
+ ".": {
+ "import": {
+ "default": "./dist/index.es.js",
+ "types": "./dist/browser/index.d.ts"
+ },
+ "require": {
+ "default": "./dist/index.cjs",
+ "types": "./dist/index.d.ts"
+ }
+ },
+ "./node": {
+ "default": "./dist/index.cjs",
+ "types": "./dist/index.d.ts"
+ },
+ "./browser": {
+ "default": "./dist/index.es.js",
+ "types": "./dist/browser/index.d.ts"
+ }
+ },
+ "files": [
+ "dist",
+ "README.md",
+ "!tsconfig.tsbuildinfo"
+ ],
+ "scripts": {
+ "dev": "vite",
+ "clean": "rm -rf dist",
+ "watch": "tsc --watch",
+ "build:vite": "vite build",
+ "build:node": "tsx esbuild.ts",
+ "typecheck": "tsc --noEmit",
+ "lint": "eslint . --ext .ts",
+ "test": "vitest run --dir tests",
+ "test-ui": "vitest watch --ui",
+ "prepublishOnly": "pnpm run build",
+ "build": "pnpm run build:vite && pnpm run build:node"
+ },
+ "keywords": [
+ "web3",
+ "crypto",
+ "blockchain",
+ "multiple-chain"
+ ],
+ "author": "MultipleChain",
+ "license": "MIT",
+ "homepage": "https://github.com/MultipleChain/js/tree/master/packages/networks/network-name",
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/MultipleChain/js.git"
+ },
+ "bugs": {
+ "url": "https://github.com/MultipleChain/js/issues"
+ },
+ "dependencies": {
+ "@multiplechain/types": "^0.1.56",
+ "@multiplechain/utils": "^0.1.20",
+ "axios": "^1.6.8",
+ "bitcore-lib": "^10.0.28",
+ "sats-connect": "^2.3.1",
+ "ws": "^8.17.0"
+ },
+ "devDependencies": {
+ "@types/bitcore-lib": "^0.15.6"
+ }
+}
diff --git a/packages/networks/bitcoin/pnpm-lock.yaml b/packages/networks/bitcoin/pnpm-lock.yaml
new file mode 100644
index 0000000..3cc25a4
--- /dev/null
+++ b/packages/networks/bitcoin/pnpm-lock.yaml
@@ -0,0 +1,630 @@
+lockfileVersion: '6.0'
+
+settings:
+ autoInstallPeers: true
+ excludeLinksFromLockfile: false
+
+dependencies:
+ '@multiplechain/types':
+ specifier: ^0.1.56
+ version: 0.1.56
+ '@multiplechain/utils':
+ specifier: ^0.1.20
+ version: 0.1.20
+ axios:
+ specifier: ^1.6.8
+ version: 1.6.8
+ bitcore-lib:
+ specifier: ^10.0.28
+ version: 10.0.28
+ sats-connect:
+ specifier: ^2.3.1
+ version: 2.3.1(typescript@5.4.4)
+ ws:
+ specifier: ^8.17.0
+ version: 8.17.0
+
+devDependencies:
+ '@types/bitcore-lib':
+ specifier: ^0.15.6
+ version: 0.15.6
+
+packages:
+
+ /@multiplechain/types@0.1.56:
+ resolution: {integrity: sha512-tnkgDwW0AWxEfhE/392LR6ZPE7Pxz5/Ei1XYsQISGkOfYYZrsvJXr69fXTjDOHCq7RCuQqytKmhnKs5K4xHd7w==}
+ dev: false
+
+ /@multiplechain/utils@0.1.20:
+ resolution: {integrity: sha512-4eEs4YR/V4F2Qnkl5KTZvPpewRQiX7C193d0tjpQuIfyumTEBJtdHYm1CW9CoJ6EjQUaB0oqSRKXomhzfK1qkw==}
+ dependencies:
+ '@types/ws': 8.5.10
+ bignumber.js: 9.1.2
+ web3-utils: 4.2.1
+ ws: 8.17.0
+ transitivePeerDependencies:
+ - bufferutil
+ - utf-8-validate
+ dev: false
+
+ /@noble/curves@1.3.0:
+ resolution: {integrity: sha512-t01iSXPuN+Eqzb4eBX0S5oubSqXbK/xXa1Ne18Hj8f9pStxztHCE2gfboSp/dZRLSqfuLpRK2nDXDK+W9puocA==}
+ dependencies:
+ '@noble/hashes': 1.3.3
+ dev: false
+
+ /@noble/hashes@1.3.3:
+ resolution: {integrity: sha512-V7/fPHgl+jsVPXqqeOzT8egNj2iBIVt+ECeMMG8TdcnTikP3oaBtUVqpT/gYCR68aEBJSF+XbYUxStjbFMqIIA==}
+ engines: {node: '>= 16'}
+ dev: false
+
+ /@noble/secp256k1@1.7.1:
+ resolution: {integrity: sha512-hOUk6AyBFmqVrv7k5WAw/LpszxVbj9gGN4JRkIX52fdFAj1UA61KXmZDvqVEm+pOyec3+fIeZB02LYa/pWOArw==}
+ dev: false
+
+ /@sats-connect/core@0.0.7:
+ resolution: {integrity: sha512-4m5amq+orHDbqLqCRWojvDQigKAys33Ntwc7U5xNtFeib4j+DpYz6lVAL/s3cay1kq03WUZ+Gil3l5rv+5bQWQ==}
+ dependencies:
+ axios: 1.6.8
+ bitcoin-address-validation: 2.2.3
+ buffer: 6.0.3
+ jsontokens: 4.0.1
+ lodash.omit: 4.5.0
+ transitivePeerDependencies:
+ - debug
+ dev: false
+
+ /@sats-connect/core@0.0.8:
+ resolution: {integrity: sha512-vb7drnd8lFfO4ahCzaVAFkX1eHF1J7jheJl2V/JuuJd5f1sy6nHeNzKMp1zmiuql8uNwe0Sx1WrK1I+4tUmDHg==}
+ dependencies:
+ axios: 1.6.8
+ bitcoin-address-validation: 2.2.3
+ buffer: 6.0.3
+ jsontokens: 4.0.1
+ lodash.omit: 4.5.0
+ transitivePeerDependencies:
+ - debug
+ dev: false
+
+ /@sats-connect/make-default-provider-config@0.0.4(typescript@5.4.4):
+ resolution: {integrity: sha512-PsLzg1hV3FxMXUp9XrOUmDJgbuyR4VDHq/7mh1O1CtC3dDZQnJFa+Ue43duPMmUaRGinuVKtS2hnMhPLyURdGA==}
+ peerDependencies:
+ typescript: 5.4.4
+ dependencies:
+ '@sats-connect/core': 0.0.7
+ '@sats-connect/ui': 0.0.5-c661c02
+ bowser: 2.11.0
+ typescript: 5.4.4
+ transitivePeerDependencies:
+ - debug
+ dev: false
+
+ /@sats-connect/ui@0.0.5-c661c02:
+ resolution: {integrity: sha512-6MUXFDGTapBhZAxb6deAdqKuB64GOe6k927gGww5JYwVnOUCaHGDcfaZ/lwexzYL45u8RJof12I4np7MgS+Bwg==}
+ dev: false
+
+ /@sats-connect/ui@0.0.6:
+ resolution: {integrity: sha512-H3bFFhr9CcY1oNosNi/QJszmMHSht4U19bUWfM3vzayAKgV4ebY6iUnRK5g3p2rVLLWVzlpaw1J9m+7JWwyBfA==}
+ dev: false
+
+ /@scure/base@1.1.6:
+ resolution: {integrity: sha512-ok9AWwhcgYuGG3Zfhyqg+zwl+Wn5uE+dwC0NV/2qQkx4dABbb/bx96vWu8NSj+BNjjSjno+JRYRjle1jV08k3g==}
+ dev: false
+
+ /@scure/bip32@1.3.3:
+ resolution: {integrity: sha512-LJaN3HwRbfQK0X1xFSi0Q9amqOgzQnnDngIt+ZlsBC3Bm7/nE7K0kwshZHyaru79yIVRv/e1mQAjZyuZG6jOFQ==}
+ dependencies:
+ '@noble/curves': 1.3.0
+ '@noble/hashes': 1.3.3
+ '@scure/base': 1.1.6
+ dev: false
+
+ /@scure/bip39@1.2.2:
+ resolution: {integrity: sha512-HYf9TUXG80beW+hGAt3TRM8wU6pQoYur9iNypTROm42dorCGmLnFe3eWjz3gOq6G62H2WRh0FCzAR1PI+29zIA==}
+ dependencies:
+ '@noble/hashes': 1.3.3
+ '@scure/base': 1.1.6
+ dev: false
+
+ /@types/bitcore-lib@0.15.6:
+ resolution: {integrity: sha512-CtKDBgSBubPXZ0wFeCiUCSdzH+cuy6nFya3FboOqf44evi+OmkQPqEg3ASMpmPDYE8vkcxV302Iu8lZqCjYieg==}
+ dependencies:
+ '@types/node': 20.12.7
+ dev: true
+
+ /@types/node@20.12.7:
+ resolution: {integrity: sha512-wq0cICSkRLVaf3UGLMGItu/PtdY7oaXaI/RVU+xliKVOtRna3PRY57ZDfztpDL0n11vfymMUnXv8QwYCO7L1wg==}
+ dependencies:
+ undici-types: 5.26.5
+
+ /@types/ws@8.5.10:
+ resolution: {integrity: sha512-vmQSUcfalpIq0R9q7uTo2lXs6eGIpt9wtnLdMv9LVpIjCA/+ufZRozlVoVelIYixx1ugCBKDhn89vnsEGOCx9A==}
+ dependencies:
+ '@types/node': 20.12.7
+ dev: false
+
+ /asynckit@0.4.0:
+ resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==}
+ dev: false
+
+ /available-typed-arrays@1.0.7:
+ resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ possible-typed-array-names: 1.0.0
+ dev: false
+
+ /axios@1.6.8:
+ resolution: {integrity: sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==}
+ dependencies:
+ follow-redirects: 1.15.6
+ form-data: 4.0.0
+ proxy-from-env: 1.1.0
+ transitivePeerDependencies:
+ - debug
+ dev: false
+
+ /base-x@3.0.9:
+ resolution: {integrity: sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ==}
+ dependencies:
+ safe-buffer: 5.2.1
+ dev: false
+
+ /base58-js@1.0.5:
+ resolution: {integrity: sha512-LkkAPP8Zu+c0SVNRTRVDyMfKVORThX+rCViget00xdgLRrKkClCTz1T7cIrpr69ShwV5XJuuoZvMvJ43yURwkA==}
+ engines: {node: '>= 8'}
+ dev: false
+
+ /base64-js@1.5.1:
+ resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==}
+ dev: false
+
+ /bech32@2.0.0:
+ resolution: {integrity: sha512-LcknSilhIGatDAsY1ak2I8VtGaHNhgMSYVxFrGLXv+xLHytaKZKcaUJJUE7qmBr7h33o5YQwP55pMI0xmkpJwg==}
+ dev: false
+
+ /bigi@1.4.2:
+ resolution: {integrity: sha512-ddkU+dFIuEIW8lE7ZwdIAf2UPoM90eaprg5m3YXAVVTmKlqV/9BX4A2M8BOK2yOq6/VgZFVhK6QAxJebhlbhzw==}
+ dev: false
+
+ /bignumber.js@9.1.2:
+ resolution: {integrity: sha512-2/mKyZH9K85bzOEfhXDBFZTGd1CTs+5IHpeFQo9luiBG7hghdC851Pj2WAhb6E3R6b9tZj/XKhbg4fum+Kepug==}
+ dev: false
+
+ /bip-schnorr@0.6.4:
+ resolution: {integrity: sha512-dNKw7Lea8B0wMIN4OjEmOk/Z5qUGqoPDY0P2QttLqGk1hmDPytLWW8PR5Pb6Vxy6CprcdEgfJpOjUu+ONQveyg==}
+ engines: {node: '>=8.0.0'}
+ dependencies:
+ bigi: 1.4.2
+ ecurve: 1.0.6
+ js-sha256: 0.9.0
+ randombytes: 2.1.0
+ safe-buffer: 5.2.1
+ dev: false
+
+ /bitcoin-address-validation@2.2.3:
+ resolution: {integrity: sha512-1uGCGl26Ye8JG5qcExtFLQfuib6qEZWNDo1ZlLlwp/z7ygUFby3IxolgEfgMGaC+LG9csbVASLcH8fRLv7DIOg==}
+ dependencies:
+ base58-js: 1.0.5
+ bech32: 2.0.0
+ sha256-uint8array: 0.10.7
+ dev: false
+
+ /bitcore-lib@10.0.28:
+ resolution: {integrity: sha512-uOWHpWbUxEj411p+tp6DCb9NfZdsCAHl6Z/rs96mquQMsef29fOqWUyk4Bl7yLf+KwzU5ZKy0HIPnjGFlmpXpg==}
+ dependencies:
+ bech32: 2.0.0
+ bip-schnorr: 0.6.4
+ bn.js: 4.11.8
+ bs58: 4.0.1
+ buffer-compare: 1.1.1
+ elliptic: 6.5.5
+ inherits: 2.0.1
+ lodash: 4.17.21
+ dev: false
+
+ /bn.js@4.11.8:
+ resolution: {integrity: sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==}
+ dev: false
+
+ /bn.js@4.12.0:
+ resolution: {integrity: sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==}
+ dev: false
+
+ /bowser@2.11.0:
+ resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==}
+ dev: false
+
+ /brorand@1.1.0:
+ resolution: {integrity: sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==}
+ dev: false
+
+ /bs58@4.0.1:
+ resolution: {integrity: sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==}
+ dependencies:
+ base-x: 3.0.9
+ dev: false
+
+ /buffer-compare@1.1.1:
+ resolution: {integrity: sha512-O6NvNiHZMd3mlIeMDjP6t/gPG75OqGPeiRZXoMQZJ6iy9GofCls4Ijs5YkPZZwoysizLiedhticmdyx/GyHghA==}
+ dev: false
+
+ /buffer@6.0.3:
+ resolution: {integrity: sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==}
+ dependencies:
+ base64-js: 1.5.1
+ ieee754: 1.2.1
+ dev: false
+
+ /call-bind@1.0.7:
+ resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ es-define-property: 1.0.0
+ es-errors: 1.3.0
+ function-bind: 1.1.2
+ get-intrinsic: 1.2.4
+ set-function-length: 1.2.2
+ dev: false
+
+ /combined-stream@1.0.8:
+ resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==}
+ engines: {node: '>= 0.8'}
+ dependencies:
+ delayed-stream: 1.0.0
+ dev: false
+
+ /define-data-property@1.1.4:
+ resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ es-define-property: 1.0.0
+ es-errors: 1.3.0
+ gopd: 1.0.1
+ dev: false
+
+ /delayed-stream@1.0.0:
+ resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==}
+ engines: {node: '>=0.4.0'}
+ dev: false
+
+ /ecurve@1.0.6:
+ resolution: {integrity: sha512-/BzEjNfiSuB7jIWKcS/z8FK9jNjmEWvUV2YZ4RLSmcDtP7Lq0m6FvDuSnJpBlDpGRpfRQeTLGLBI8H+kEv0r+w==}
+ dependencies:
+ bigi: 1.4.2
+ safe-buffer: 5.2.1
+ dev: false
+
+ /elliptic@6.5.5:
+ resolution: {integrity: sha512-7EjbcmUm17NQFu4Pmgmq2olYMj8nwMnpcddByChSUjArp8F5DQWcIcpriwO4ZToLNAJig0yiyjswfyGNje/ixw==}
+ dependencies:
+ bn.js: 4.12.0
+ brorand: 1.1.0
+ hash.js: 1.1.7
+ hmac-drbg: 1.0.1
+ inherits: 2.0.4
+ minimalistic-assert: 1.0.1
+ minimalistic-crypto-utils: 1.0.1
+ dev: false
+
+ /es-define-property@1.0.0:
+ resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ get-intrinsic: 1.2.4
+ dev: false
+
+ /es-errors@1.3.0:
+ resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==}
+ engines: {node: '>= 0.4'}
+ dev: false
+
+ /ethereum-cryptography@2.1.3:
+ resolution: {integrity: sha512-BlwbIL7/P45W8FGW2r7LGuvoEZ+7PWsniMvQ4p5s2xCyw9tmaDlpfsN9HjAucbF+t/qpVHwZUisgfK24TCW8aA==}
+ dependencies:
+ '@noble/curves': 1.3.0
+ '@noble/hashes': 1.3.3
+ '@scure/bip32': 1.3.3
+ '@scure/bip39': 1.2.2
+ dev: false
+
+ /eventemitter3@5.0.1:
+ resolution: {integrity: sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA==}
+ dev: false
+
+ /follow-redirects@1.15.6:
+ resolution: {integrity: sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==}
+ engines: {node: '>=4.0'}
+ peerDependencies:
+ debug: '*'
+ peerDependenciesMeta:
+ debug:
+ optional: true
+ dev: false
+
+ /for-each@0.3.3:
+ resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==}
+ dependencies:
+ is-callable: 1.2.7
+ dev: false
+
+ /form-data@4.0.0:
+ resolution: {integrity: sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==}
+ engines: {node: '>= 6'}
+ dependencies:
+ asynckit: 0.4.0
+ combined-stream: 1.0.8
+ mime-types: 2.1.35
+ dev: false
+
+ /function-bind@1.1.2:
+ resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==}
+ dev: false
+
+ /get-intrinsic@1.2.4:
+ resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ es-errors: 1.3.0
+ function-bind: 1.1.2
+ has-proto: 1.0.3
+ has-symbols: 1.0.3
+ hasown: 2.0.2
+ dev: false
+
+ /gopd@1.0.1:
+ resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==}
+ dependencies:
+ get-intrinsic: 1.2.4
+ dev: false
+
+ /has-property-descriptors@1.0.2:
+ resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==}
+ dependencies:
+ es-define-property: 1.0.0
+ dev: false
+
+ /has-proto@1.0.3:
+ resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==}
+ engines: {node: '>= 0.4'}
+ dev: false
+
+ /has-symbols@1.0.3:
+ resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==}
+ engines: {node: '>= 0.4'}
+ dev: false
+
+ /has-tostringtag@1.0.2:
+ resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-symbols: 1.0.3
+ dev: false
+
+ /hash.js@1.1.7:
+ resolution: {integrity: sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==}
+ dependencies:
+ inherits: 2.0.4
+ minimalistic-assert: 1.0.1
+ dev: false
+
+ /hasown@2.0.2:
+ resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ function-bind: 1.1.2
+ dev: false
+
+ /hmac-drbg@1.0.1:
+ resolution: {integrity: sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==}
+ dependencies:
+ hash.js: 1.1.7
+ minimalistic-assert: 1.0.1
+ minimalistic-crypto-utils: 1.0.1
+ dev: false
+
+ /ieee754@1.2.1:
+ resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==}
+ dev: false
+
+ /inherits@2.0.1:
+ resolution: {integrity: sha512-8nWq2nLTAwd02jTqJExUYFSD/fKq6VH9Y/oG2accc/kdI0V98Bag8d5a4gi3XHz73rDWa2PvTtvcWYquKqSENA==}
+ dev: false
+
+ /inherits@2.0.4:
+ resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==}
+ dev: false
+
+ /is-arguments@1.1.1:
+ resolution: {integrity: sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ call-bind: 1.0.7
+ has-tostringtag: 1.0.2
+ dev: false
+
+ /is-callable@1.2.7:
+ resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==}
+ engines: {node: '>= 0.4'}
+ dev: false
+
+ /is-generator-function@1.0.10:
+ resolution: {integrity: sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ has-tostringtag: 1.0.2
+ dev: false
+
+ /is-typed-array@1.1.13:
+ resolution: {integrity: sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ which-typed-array: 1.1.15
+ dev: false
+
+ /js-sha256@0.9.0:
+ resolution: {integrity: sha512-sga3MHh9sgQN2+pJ9VYZ+1LPwXOxuBJBA5nrR5/ofPfuiJBE2hnjsaN8se8JznOmGLN2p49Pe5U/ttafcs/apA==}
+ dev: false
+
+ /jsontokens@4.0.1:
+ resolution: {integrity: sha512-+MO415LEN6M+3FGsRz4wU20g7N2JA+2j9d9+pGaNJHviG4L8N0qzavGyENw6fJqsq9CcrHOIL6iWX5yeTZ86+Q==}
+ dependencies:
+ '@noble/hashes': 1.3.3
+ '@noble/secp256k1': 1.7.1
+ base64-js: 1.5.1
+ dev: false
+
+ /lodash.omit@4.5.0:
+ resolution: {integrity: sha512-XeqSp49hNGmlkj2EJlfrQFIzQ6lXdNro9sddtQzcJY8QaoC2GO0DT7xaIokHeyM+mIT0mPMlPvkYzg2xCuHdZg==}
+ dev: false
+
+ /lodash@4.17.21:
+ resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==}
+ dev: false
+
+ /mime-db@1.52.0:
+ resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==}
+ engines: {node: '>= 0.6'}
+ dev: false
+
+ /mime-types@2.1.35:
+ resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==}
+ engines: {node: '>= 0.6'}
+ dependencies:
+ mime-db: 1.52.0
+ dev: false
+
+ /minimalistic-assert@1.0.1:
+ resolution: {integrity: sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==}
+ dev: false
+
+ /minimalistic-crypto-utils@1.0.1:
+ resolution: {integrity: sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==}
+ dev: false
+
+ /possible-typed-array-names@1.0.0:
+ resolution: {integrity: sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==}
+ engines: {node: '>= 0.4'}
+ dev: false
+
+ /proxy-from-env@1.1.0:
+ resolution: {integrity: sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==}
+ dev: false
+
+ /randombytes@2.1.0:
+ resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==}
+ dependencies:
+ safe-buffer: 5.2.1
+ dev: false
+
+ /safe-buffer@5.2.1:
+ resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==}
+ dev: false
+
+ /sats-connect@2.3.1(typescript@5.4.4):
+ resolution: {integrity: sha512-3KzqRO5KVBlge7Q4a/L828SfCkFD+M4MVdtgJZS+L+oHiDYoXlLkvnu3almh9Ynhcm0HnsGmVH1pKVL0lonjyQ==}
+ dependencies:
+ '@sats-connect/core': 0.0.8
+ '@sats-connect/make-default-provider-config': 0.0.4(typescript@5.4.4)
+ '@sats-connect/ui': 0.0.6
+ transitivePeerDependencies:
+ - debug
+ - typescript
+ dev: false
+
+ /set-function-length@1.2.2:
+ resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ define-data-property: 1.1.4
+ es-errors: 1.3.0
+ function-bind: 1.1.2
+ get-intrinsic: 1.2.4
+ gopd: 1.0.1
+ has-property-descriptors: 1.0.2
+ dev: false
+
+ /sha256-uint8array@0.10.7:
+ resolution: {integrity: sha512-1Q6JQU4tX9NqsDGodej6pkrUVQVNapLZnvkwIhddH/JqzBZF1fSaxSWNY6sziXBE8aEa2twtGkXUrwzGeZCMpQ==}
+ dev: false
+
+ /typescript@5.4.4:
+ resolution: {integrity: sha512-dGE2Vv8cpVvw28v8HCPqyb08EzbBURxDpuhJvTrusShUfGnhHBafDsLdS1EhhxyL6BJQE+2cT3dDPAv+MQ6oLw==}
+ engines: {node: '>=14.17'}
+ hasBin: true
+ dev: false
+
+ /undici-types@5.26.5:
+ resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==}
+
+ /util@0.12.5:
+ resolution: {integrity: sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==}
+ dependencies:
+ inherits: 2.0.4
+ is-arguments: 1.1.1
+ is-generator-function: 1.0.10
+ is-typed-array: 1.1.13
+ which-typed-array: 1.1.15
+ dev: false
+
+ /web3-errors@1.1.4:
+ resolution: {integrity: sha512-WahtszSqILez+83AxGecVroyZsMuuRT+KmQp4Si5P4Rnqbczno1k748PCrZTS1J4UCPmXMG2/Vt+0Bz2zwXkwQ==}
+ engines: {node: '>=14', npm: '>=6.12.0'}
+ dependencies:
+ web3-types: 1.5.0
+ dev: false
+
+ /web3-types@1.5.0:
+ resolution: {integrity: sha512-geWuMIeegQ8AedKAO6wO4G4j1gyQ1F/AyKLMw2vud4bsfZayyzWJgCMDZtjYMm5uo2a7i8j1W3/4QFmzlSy5cw==}
+ engines: {node: '>=14', npm: '>=6.12.0'}
+ dev: false
+
+ /web3-utils@4.2.1:
+ resolution: {integrity: sha512-Fk29BlEqD9Q9Cnw4pBkKw7czcXiRpsSco/BzEUl4ye0ZTSHANQFfjsfQmNm4t7uY11u6Ah+8F3tNjBeU4CA80A==}
+ engines: {node: '>=14', npm: '>=6.12.0'}
+ dependencies:
+ ethereum-cryptography: 2.1.3
+ eventemitter3: 5.0.1
+ web3-errors: 1.1.4
+ web3-types: 1.5.0
+ web3-validator: 2.0.4
+ dev: false
+
+ /web3-validator@2.0.4:
+ resolution: {integrity: sha512-qRxVePwdW+SByOmTpDZFWHIUAa7PswvxNszrOua6BoGqAhERo5oJZBN+EbWtK/+O+ApNxt5FR3nCPmiZldiOQA==}
+ engines: {node: '>=14', npm: '>=6.12.0'}
+ dependencies:
+ ethereum-cryptography: 2.1.3
+ util: 0.12.5
+ web3-errors: 1.1.4
+ web3-types: 1.5.0
+ zod: 3.22.4
+ dev: false
+
+ /which-typed-array@1.1.15:
+ resolution: {integrity: sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==}
+ engines: {node: '>= 0.4'}
+ dependencies:
+ available-typed-arrays: 1.0.7
+ call-bind: 1.0.7
+ for-each: 0.3.3
+ gopd: 1.0.1
+ has-tostringtag: 1.0.2
+ dev: false
+
+ /ws@8.17.0:
+ resolution: {integrity: sha512-uJq6108EgZMAl20KagGkzCKfMEjxmKvZHG7Tlq0Z6nOky7YF7aq4mOx6xK8TJ/i1LeK4Qus7INktacctDgY8Ow==}
+ engines: {node: '>=10.0.0'}
+ peerDependencies:
+ bufferutil: ^4.0.1
+ utf-8-validate: '>=5.0.2'
+ peerDependenciesMeta:
+ bufferutil:
+ optional: true
+ utf-8-validate:
+ optional: true
+ dev: false
+
+ /zod@3.22.4:
+ resolution: {integrity: sha512-iC+8Io04lddc+mVqQ9AZ7OQ2MrUKGN+oIQyq1vemgt46jwCwLfhq7/pwnBnNXXXZb8VTVLKwp9EDkx+ryxIWmg==}
+ dev: false
diff --git a/packages/networks/bitcoin/src/assets/Coin.ts b/packages/networks/bitcoin/src/assets/Coin.ts
new file mode 100644
index 0000000..a50d1ea
--- /dev/null
+++ b/packages/networks/bitcoin/src/assets/Coin.ts
@@ -0,0 +1,109 @@
+import axios from 'axios'
+import { Provider } from '../services/Provider.ts'
+import { fromSatoshi, toSatoshi } from '../utils.ts'
+import { Transaction, Script, Address } from 'bitcore-lib'
+import { ErrorTypeEnum, type CoinInterface } from '@multiplechain/types'
+import { CoinTransactionSigner } from '../services/TransactionSigner.ts'
+
+export class Coin implements CoinInterface {
+ /**
+ * Blockchain network provider
+ */
+ provider: Provider
+
+ /**
+ * @param {Provider} provider network provider
+ */
+ constructor(provider?: Provider) {
+ this.provider = provider ?? Provider.instance
+ }
+
+ /**
+ * @returns {string} Coin name
+ */
+ getName(): string {
+ return 'Bitcoin'
+ }
+
+ /**
+ * @returns {string} Coin symbol
+ */
+ getSymbol(): string {
+ return 'BTC'
+ }
+
+ /**
+ * @returns {number} Decimal value of the coin
+ */
+ getDecimals(): number {
+ return 8
+ }
+
+ /**
+ * @param {string} owner Wallet address
+ * @returns {Promise} Wallet balance as currency of COIN
+ */
+ async getBalance(owner: string): Promise {
+ const addressStatsApi = this.provider.createEndpoint('address/' + owner)
+ const addressStats = await axios.get(addressStatsApi).then((res) => res.data)
+ const balanceSat =
+ addressStats.chain_stats.funded_txo_sum - addressStats.chain_stats.spent_txo_sum
+ return fromSatoshi(balanceSat)
+ }
+
+ /**
+ * @param {string} sender Sender wallet address
+ * @param {string} receiver Receiver wallet address
+ * @param {number} amount Amount of assets that will be transferred
+ * @returns {Promise} Transaction signer
+ */
+ async transfer(
+ sender: string,
+ receiver: string,
+ amount: number
+ ): Promise {
+ if (amount < 0) {
+ throw new Error(ErrorTypeEnum.INVALID_AMOUNT)
+ }
+
+ if (amount > (await this.getBalance(sender))) {
+ throw new Error(ErrorTypeEnum.INSUFFICIENT_BALANCE)
+ }
+
+ if (sender === receiver) {
+ throw new Error(ErrorTypeEnum.INVALID_ADDRESS)
+ }
+
+ const inputs = []
+ const transaction = new Transaction()
+ const senderAddress = new Address(sender)
+ const satoshiToSend = toSatoshi(amount)
+
+ const utxos = await axios
+ .get(this.provider.createEndpoint('address/' + sender + '/utxo'))
+ .then((res) => res.data)
+
+ for (const utxo of utxos) {
+ inputs.push(
+ new Transaction.UnspentOutput({
+ txId: utxo.txid,
+ satoshis: utxo.value,
+ address: senderAddress,
+ outputIndex: utxo.vout,
+ script: Script.fromAddress(senderAddress)
+ })
+ )
+ }
+
+ transaction.from(inputs)
+ transaction.change(sender)
+ transaction.to(receiver, satoshiToSend)
+
+ return new CoinTransactionSigner({
+ sender,
+ receiver,
+ amount: satoshiToSend,
+ bitcoreLib: transaction
+ })
+ }
+}
diff --git a/packages/networks/bitcoin/src/assets/Contract.ts b/packages/networks/bitcoin/src/assets/Contract.ts
new file mode 100644
index 0000000..14044c0
--- /dev/null
+++ b/packages/networks/bitcoin/src/assets/Contract.ts
@@ -0,0 +1,59 @@
+import { Provider } from '../services/Provider.ts'
+import type { ContractInterface } from '@multiplechain/types'
+
+export class Contract implements ContractInterface {
+ /**
+ * Contract address
+ */
+ address: string
+
+ /**
+ * Blockchain network provider
+ */
+ provider: Provider
+
+ /**
+ * @param {string} address Contract address
+ * @param {Provider} provider Blockchain network provider
+ */
+ constructor(address: string, provider?: Provider) {
+ this.address = address
+ this.provider = provider ?? Provider.instance
+ throw new Error('This class is not implemented for Bitcoin.')
+ }
+
+ /**
+ * @returns {string} Contract address
+ */
+ getAddress(): string {
+ return this.address
+ }
+
+ /**
+ * @param {string} method Method name
+ * @param {any[]} args Method parameters
+ * @returns {Promise} Method result
+ */
+ async callMethod(method: string, ...args: any[]): Promise {
+ return {}
+ }
+
+ /**
+ * @param {string} method Method name
+ * @param {any[]} args Sender wallet address
+ * @returns {Promise} Encoded method data
+ */
+ async getMethodData(method: string, ...args: any[]): Promise {
+ return {}
+ }
+
+ /**
+ * @param {string} method Method name
+ * @param {string} from Sender wallet address
+ * @param {any[]} args Method parameters
+ * @returns {Promise} Encoded method data
+ */
+ async createTransactionData(method: string, from: string, ...args: any[]): Promise {
+ return ''
+ }
+}
diff --git a/packages/networks/bitcoin/src/assets/NFT.ts b/packages/networks/bitcoin/src/assets/NFT.ts
new file mode 100644
index 0000000..0a89a51
--- /dev/null
+++ b/packages/networks/bitcoin/src/assets/NFT.ts
@@ -0,0 +1,96 @@
+import { Contract } from './Contract.ts'
+import type { NftInterface } from '@multiplechain/types'
+import { NftTransactionSigner } from '../services/TransactionSigner.ts'
+
+export class NFT extends Contract implements NftInterface {
+ /**
+ * @returns {Promise} NFT name
+ */
+ async getName(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @returns {Promise} NFT symbol
+ */
+ async getSymbol(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @param {string} owner Wallet address
+ * @returns {Promise} Wallet balance as currency of NFT
+ */
+ async getBalance(owner: string): Promise {
+ return 0
+ }
+
+ /**
+ * @param {number | string} nftId NFT ID
+ * @returns {Promise} Wallet address of the owner of the NFT
+ */
+ async getOwner(nftId: number | string): Promise {
+ return 'example'
+ }
+
+ /**
+ * @param {number | string} nftId NFT ID
+ * @returns {Promise} URI of the NFT
+ */
+ async getTokenURI(nftId: number | string): Promise {
+ return 'example'
+ }
+
+ /**
+ * @param {number | string} nftId ID of the NFT that will be transferred
+ * @returns {Promise} Wallet address of the approved spender
+ */
+ async getApproved(nftId: number | string): Promise {
+ return 'example'
+ }
+
+ /**
+ * @param {string} sender Sender address
+ * @param {string} receiver Receiver address
+ * @param {number | string} nftId NFT ID
+ * @returns {Promise} Transaction signer
+ */
+ async transfer(
+ sender: string,
+ receiver: string,
+ nftId: number | string
+ ): Promise {
+ return new NftTransactionSigner('example')
+ }
+
+ /**
+ * @param {string} spender Spender address
+ * @param {string} owner Owner address
+ * @param {string} receiver Receiver address
+ * @param {number | string} nftId NFT ID
+ * @returns {Promise} Transaction signer
+ */
+ async transferFrom(
+ spender: string,
+ owner: string,
+ receiver: string,
+ nftId: number | string
+ ): Promise {
+ return new NftTransactionSigner('example')
+ }
+
+ /**
+ * Gives permission to the spender to spend owner's tokens
+ * @param {string} owner Address of owner of the tokens that will be used
+ * @param {string} spender Address of the spender that will use the tokens of owner
+ * @param {number | string} nftId ID of the NFT that will be transferred
+ * @returns {Promise} Transaction signer
+ */
+ async approve(
+ owner: string,
+ spender: string,
+ nftId: number | string
+ ): Promise {
+ return new NftTransactionSigner('example')
+ }
+}
diff --git a/packages/networks/bitcoin/src/assets/Token.ts b/packages/networks/bitcoin/src/assets/Token.ts
new file mode 100644
index 0000000..a090488
--- /dev/null
+++ b/packages/networks/bitcoin/src/assets/Token.ts
@@ -0,0 +1,92 @@
+import { Contract } from './Contract.ts'
+import type { TokenInterface } from '@multiplechain/types'
+import { TokenTransactionSigner } from '../services/TransactionSigner.ts'
+
+export class Token extends Contract implements TokenInterface {
+ /**
+ * @returns {Promise} Token name
+ */
+ async getName(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @returns {Promise} Token symbol
+ */
+ async getSymbol(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @returns {Promise} Decimal value of the token
+ */
+ async getDecimals(): Promise {
+ return 18
+ }
+
+ /**
+ * @param {string} owner Wallet address
+ * @returns {Promise} Wallet balance as currency of TOKEN
+ */
+ async getBalance(owner: string): Promise {
+ return 0
+ }
+
+ /**
+ * @returns {Promise} Total supply of the token
+ */
+ async getTotalSupply(): Promise {
+ return 0
+ }
+
+ /**
+ * @param {string} owner Address of owner of the tokens that is being used
+ * @param {string} spender Address of the spender that is using the tokens of owner
+ * @returns {Promise} Amount of tokens that the spender is allowed to spend
+ */
+ async getAllowance(owner: string, spender: string): Promise {
+ return 0
+ }
+
+ /**
+ * transfer() method is the main method for processing transfers for fungible assets (TOKEN, COIN)
+ * @param {string} sender Sender wallet address
+ * @param {string} receiver Receiver wallet address
+ * @param {number} amount Amount of assets that will be transferred
+ * @returns {Promise} Transaction signer
+ */
+ async transfer(
+ sender: string,
+ receiver: string,
+ amount: number
+ ): Promise {
+ return new TokenTransactionSigner('example')
+ }
+
+ /**
+ * @param {string} spender Address of the spender of transaction
+ * @param {string} owner Sender wallet address
+ * @param {string} receiver Receiver wallet address
+ * @param {number} amount Amount of tokens that will be transferred
+ * @returns {Promise} Transaction signer
+ */
+ async transferFrom(
+ spender: string,
+ owner: string,
+ receiver: string,
+ amount: number
+ ): Promise {
+ return new TokenTransactionSigner('example')
+ }
+
+ /**
+ * Gives permission to the spender to spend owner's tokens
+ * @param {string} owner Address of owner of the tokens that will be used
+ * @param {string} spender Address of the spender that will use the tokens of owner
+ * @param {number} amount Amount of the tokens that will be used
+ * @returns {Promise} Transaction signer
+ */
+ async approve(owner: string, spender: string, amount: number): Promise {
+ return new TokenTransactionSigner('example')
+ }
+}
diff --git a/packages/networks/bitcoin/src/assets/index.ts b/packages/networks/bitcoin/src/assets/index.ts
new file mode 100644
index 0000000..7ea522f
--- /dev/null
+++ b/packages/networks/bitcoin/src/assets/index.ts
@@ -0,0 +1,4 @@
+export * from './NFT.ts'
+export * from './Coin.ts'
+export * from './Token.ts'
+export * from './Contract.ts'
diff --git a/packages/networks/bitcoin/src/browser/Wallet.ts b/packages/networks/bitcoin/src/browser/Wallet.ts
new file mode 100644
index 0000000..cb8e8d0
--- /dev/null
+++ b/packages/networks/bitcoin/src/browser/Wallet.ts
@@ -0,0 +1,168 @@
+import {
+ type WalletInterface,
+ type WalletAdapterInterface,
+ type WalletPlatformEnum,
+ type TransactionSignerInterface,
+ type ProviderInterface,
+ ErrorTypeEnum
+} from '@multiplechain/types'
+import { Provider } from '../services/Provider.ts'
+import type { TransactionData } from '../services/TransactionSigner.ts'
+
+export interface BitcoinWalletAdapter {
+ getAddress: () => Promise
+ signMessage: (message: string) => Promise
+ sendBitcoin: (to: string, amount: number) => Promise
+ on: (event: string, callback: (data: any) => void) => void
+}
+
+const rejectMap = (error: any, reject: (a: any) => any): any => {
+ console.error('MultipleChain Bitcoin Wallet Error:', error)
+
+ if (typeof error === 'object') {
+ if (error.code === 4001 || String(error.message).includes('User rejected the request')) {
+ return reject(ErrorTypeEnum.WALLET_REQUEST_REJECTED)
+ } else if (String(error).includes('is not valid JSON')) {
+ return reject(ErrorTypeEnum.UNACCEPTED_CHAIN)
+ }
+ }
+
+ return reject(error)
+}
+
+export class Wallet implements WalletInterface {
+ adapter: WalletAdapterInterface
+
+ walletProvider: BitcoinWalletAdapter
+
+ networkProvider: Provider
+
+ /**
+ * @param {WalletAdapterInterface} adapter
+ * @param {Provider} provider
+ */
+ constructor(adapter: WalletAdapterInterface, provider?: Provider) {
+ this.adapter = adapter
+ this.networkProvider = provider ?? Provider.instance
+ }
+
+ /**
+ * @returns {string}
+ */
+ getId(): string {
+ return this.adapter.id
+ }
+
+ /**
+ * @returns {string}
+ */
+ getName(): string {
+ return this.adapter.name
+ }
+
+ /**
+ * @returns {string}
+ */
+ getIcon(): string {
+ return this.adapter.icon
+ }
+
+ /**
+ * @returns {WalletPlatformEnum[]}
+ */
+ getPlatforms(): WalletPlatformEnum[] {
+ return this.adapter.platforms
+ }
+
+ /**
+ * @returns {string}
+ */
+ getDownloadLink(): string | undefined {
+ return this.adapter.downloadLink
+ }
+
+ /**
+ * @param {string} url
+ * @param {object} ops
+ * @returns {string}
+ */
+ createDeepLink(url: string, ops?: object): string | null {
+ if (this.adapter.createDeepLink === undefined) {
+ return null
+ }
+
+ return this.adapter.createDeepLink(url, ops)
+ }
+
+ /**
+ * @param {ProviderInterface} provider
+ * @param {Object} ops
+ * @returns {Promise}
+ */
+ async connect(provider?: ProviderInterface, ops?: object): Promise {
+ return await new Promise((resolve, reject) => {
+ this.adapter
+ .connect(provider, ops)
+ .then(async (provider) => {
+ this.walletProvider = provider as BitcoinWalletAdapter
+ resolve(await this.getAddress())
+ })
+ .catch((error) => {
+ const customReject = (error: any): void => {
+ if (error.message === ErrorTypeEnum.WALLET_REQUEST_REJECTED) {
+ reject(new Error(ErrorTypeEnum.WALLET_CONNECT_REJECTED))
+ } else {
+ reject(error)
+ }
+ }
+ rejectMap(error, customReject)
+ })
+ })
+ }
+
+ /**
+ * @returns {boolean}
+ */
+ async isDetected(): Promise {
+ return await this.adapter.isDetected()
+ }
+
+ /**
+ * @returns {boolean}
+ */
+ async isConnected(): Promise {
+ return await this.adapter.isConnected()
+ }
+
+ /**
+ * @returns {Promise}
+ */
+ async getAddress(): Promise {
+ return await this.walletProvider.getAddress()
+ }
+
+ /**
+ * @param {string} message
+ */
+ async signMessage(message: string): Promise {
+ return await this.walletProvider.signMessage(message)
+ }
+
+ /**
+ * @param {TransactionSignerInterface} transactionSigner
+ * @returns {Promise}
+ */
+ async sendTransaction(transactionSigner: TransactionSignerInterface): Promise {
+ const data = (await transactionSigner.getRawData()) as TransactionData
+ return await this.walletProvider.sendBitcoin(data.receiver, data.amount)
+ }
+
+ /**
+ * @param {string} eventName
+ * @param {Function} callback
+ * @returns {void}
+ */
+ on(eventName: string, callback: (...args: any[]) => void): void {
+ this.walletProvider.on(eventName, callback)
+ }
+}
diff --git a/packages/networks/bitcoin/src/browser/adapters/Leather.ts b/packages/networks/bitcoin/src/browser/adapters/Leather.ts
new file mode 100644
index 0000000..3f363fa
--- /dev/null
+++ b/packages/networks/bitcoin/src/browser/adapters/Leather.ts
@@ -0,0 +1,105 @@
+import icons from './icons.ts'
+import { WalletPlatformEnum } from '@multiplechain/types'
+import type { ProviderInterface, WalletAdapterInterface } from '@multiplechain/types'
+import type { BitcoinWalletAdapter } from '../Wallet.ts'
+
+declare global {
+ interface Window {
+ btc: {
+ listen?: (event: string, callback: (data: any) => void) => void
+ }
+ LeatherProvider: {
+ request: (method: string, params: any) => Promise
+ }
+ }
+}
+
+let connected = false
+
+const Leather: WalletAdapterInterface = {
+ id: 'leather',
+ name: 'Leather',
+ icon: icons.Leather,
+ platforms: [WalletPlatformEnum.BROWSER],
+ downloadLink: 'https://leather.io/install-extension',
+ isDetected: () => Boolean(window.LeatherProvider),
+ isConnected: async () => connected,
+ connect: async (provider?: ProviderInterface): Promise => {
+ return await new Promise((resolve, reject) => {
+ const leather = window.LeatherProvider
+
+ const network = provider !== undefined && provider?.isTestnet() ? 'testnet' : 'mainnet'
+
+ const walletAdapter: BitcoinWalletAdapter = {
+ on: (event, callback) => {
+ if (window.btc?.listen !== undefined) {
+ window.btc.listen(event, callback)
+ }
+ },
+ signMessage: async (message: string) => {
+ const response = await leather.request('signMessage', {
+ message,
+ network,
+ account: 0,
+ paymentType: 'p2wpkh'
+ })
+
+ return response.result.signature as string
+ },
+ sendBitcoin: async (to: string, amount: number) => {
+ return await new Promise((resolve, reject) => {
+ try {
+ leather
+ .request('sendTransfer', {
+ address: to,
+ amount,
+ network
+ })
+ .then((response) => {
+ resolve(response.result.txid as string)
+ })
+ .catch(({ error }) => {
+ reject(error)
+ })
+ } catch (error) {
+ reject(error)
+ }
+ })
+ },
+ getAddress: async () => ''
+ }
+
+ const connect = async (): Promise => {
+ const addresses = (
+ await leather.request('getAddresses', {
+ network
+ })
+ ).result.addresses
+
+ const bitcoin = addresses.find((address: any) => address.type === 'p2wpkh')
+
+ // for ordinals & BRC-20 integrations
+ // const ordinals = addresses.find(address => address.type == 'p2tr');
+
+ walletAdapter.getAddress = async () => {
+ return bitcoin.address
+ }
+
+ return walletAdapter
+ }
+
+ try {
+ connect()
+ .then((walletAdapter) => {
+ connected = true
+ resolve(walletAdapter)
+ })
+ .catch(reject)
+ } catch (error) {
+ reject(error)
+ }
+ })
+ }
+}
+
+export default Leather
diff --git a/packages/networks/bitcoin/src/browser/adapters/UniSat.ts b/packages/networks/bitcoin/src/browser/adapters/UniSat.ts
new file mode 100644
index 0000000..b089c8c
--- /dev/null
+++ b/packages/networks/bitcoin/src/browser/adapters/UniSat.ts
@@ -0,0 +1,60 @@
+import icons from './icons.ts'
+import { WalletPlatformEnum } from '@multiplechain/types'
+import type { ProviderInterface, WalletAdapterInterface } from '@multiplechain/types'
+import type { BitcoinWalletAdapter } from '../Wallet.ts'
+
+declare global {
+ interface Window {
+ unisat: {
+ _isConnected: boolean
+ getAccounts: () => Promise
+ requestAccounts: () => Promise
+ signMessage: (message: string) => Promise
+ switchNetwork: (network: string) => Promise
+ on: (event: string, callback: (data: any) => void) => void
+ sendBitcoin: (to: string, amount: number) => Promise
+ }
+ }
+}
+
+const UniSat: WalletAdapterInterface = {
+ id: 'unisat',
+ name: 'UniSat',
+ icon: icons.UniSat,
+ platforms: [WalletPlatformEnum.BROWSER],
+ downloadLink: 'https://unisat.io/download',
+ isDetected: () => Boolean(window.unisat?.requestAccounts),
+ isConnected: async () => window?.unisat?._isConnected ?? false,
+ connect: async (provider?: ProviderInterface): Promise => {
+ return await new Promise((resolve, reject) => {
+ const network = provider !== undefined && provider?.isTestnet() ? 'testnet' : 'livenet'
+
+ const walletAdapter: BitcoinWalletAdapter = {
+ on: window.unisat.on,
+ signMessage: window.unisat.signMessage,
+ sendBitcoin: window.unisat.sendBitcoin,
+ getAddress: async () => {
+ return (await window.unisat.getAccounts())[0]
+ }
+ }
+
+ try {
+ window.unisat
+ .requestAccounts()
+ .then(() => {
+ window.unisat
+ .switchNetwork(network)
+ .then(() => {
+ resolve(walletAdapter)
+ })
+ .catch(reject)
+ })
+ .catch(reject)
+ } catch (error) {
+ reject(error)
+ }
+ })
+ }
+}
+
+export default UniSat
diff --git a/packages/networks/bitcoin/src/browser/adapters/Xverse.ts b/packages/networks/bitcoin/src/browser/adapters/Xverse.ts
new file mode 100644
index 0000000..ef50592
--- /dev/null
+++ b/packages/networks/bitcoin/src/browser/adapters/Xverse.ts
@@ -0,0 +1,139 @@
+import icons from './icons.ts'
+import type { BitcoinWalletAdapter } from '../Wallet.ts'
+import {
+ ErrorTypeEnum,
+ WalletPlatformEnum,
+ type ProviderInterface,
+ type WalletAdapterInterface
+} from '@multiplechain/types'
+import {
+ getAddress,
+ sendBtcTransaction,
+ BitcoinNetworkType,
+ signMessage,
+ AddressPurpose
+} from 'sats-connect'
+
+let connected = false
+
+const Xverse: WalletAdapterInterface = {
+ id: 'xverse',
+ name: 'Xverse',
+ icon: icons.Xverse,
+ platforms: [WalletPlatformEnum.BROWSER, WalletPlatformEnum.MOBILE],
+ downloadLink: 'https://www.xverse.app/download',
+ isDetected: () => Boolean(window.XverseProviders?.BitcoinProvider),
+ isConnected: async () => connected,
+ connect: async (provider?: ProviderInterface): Promise => {
+ return await new Promise((resolve, reject) => {
+ const type =
+ provider !== undefined && provider?.isTestnet()
+ ? BitcoinNetworkType.Testnet
+ : BitcoinNetworkType.Mainnet
+
+ const walletAdapter: BitcoinWalletAdapter = {
+ on: (_event: string, _callback: (data: any) => void) => {},
+ signMessage: async (message: string) => {
+ const address = await walletAdapter.getAddress()
+ return await new Promise((resolve, reject) => {
+ signMessage({
+ payload: {
+ network: {
+ type
+ },
+ address,
+ message
+ },
+ onFinish: (signature) => {
+ resolve(signature)
+ },
+ onCancel: () => {
+ reject(ErrorTypeEnum.WALLET_REQUEST_REJECTED)
+ }
+ }).catch(reject)
+ })
+ },
+ sendBitcoin: async (to: string, amount: number) => {
+ const senderAddress = await walletAdapter.getAddress()
+ return await new Promise((resolve, reject) => {
+ sendBtcTransaction({
+ payload: {
+ network: {
+ type
+ },
+ recipients: [
+ {
+ address: to,
+ amountSats: BigInt(amount)
+ }
+ ],
+ senderAddress
+ },
+ onFinish: (txId) => {
+ resolve(txId)
+ },
+ onCancel: () => {
+ reject(ErrorTypeEnum.WALLET_REQUEST_REJECTED)
+ }
+ }).catch(reject)
+ })
+ },
+ getAddress: async () => ''
+ }
+
+ const connect = async (): Promise => {
+ return await new Promise((resolve, reject) => {
+ try {
+ getAddress({
+ payload: {
+ purposes: [AddressPurpose.Payment, AddressPurpose.Ordinals],
+ message: 'Address for receiving Ordinals and payments',
+ network: {
+ type
+ }
+ },
+ onFinish: (response) => {
+ const addresses = Object.values(response.addresses)
+ const bitcoin = addresses.find(
+ (address) => address.purpose === AddressPurpose.Payment
+ )
+
+ // for ordinals & BRC-20 integrations
+ // const ordinals = addresses.find(address => address.purpose == 'ordinals');
+
+ if (bitcoin === undefined) {
+ reject(ErrorTypeEnum.WALLET_CONNECTION_FAILED)
+ return
+ }
+
+ walletAdapter.getAddress = async () => {
+ return bitcoin.address
+ }
+
+ resolve(walletAdapter)
+ },
+ onCancel: () => {
+ reject(ErrorTypeEnum.WALLET_REQUEST_REJECTED)
+ }
+ }).catch(reject)
+ } catch (error) {
+ reject(error)
+ }
+ })
+ }
+
+ try {
+ connect()
+ .then((walletAdapter) => {
+ connected = true
+ resolve(walletAdapter)
+ })
+ .catch(reject)
+ } catch (error) {
+ reject(error)
+ }
+ })
+ }
+}
+
+export default Xverse
diff --git a/packages/networks/bitcoin/src/browser/adapters/icons.ts b/packages/networks/bitcoin/src/browser/adapters/icons.ts
new file mode 100644
index 0000000..5417b1d
--- /dev/null
+++ b/packages/networks/bitcoin/src/browser/adapters/icons.ts
@@ -0,0 +1,6 @@
+export default {
+ UniSat: '',
+ Xverse: '',
+ Leather:
+ ''
+}
diff --git a/packages/networks/bitcoin/src/browser/adapters/index.ts b/packages/networks/bitcoin/src/browser/adapters/index.ts
new file mode 100644
index 0000000..6408790
--- /dev/null
+++ b/packages/networks/bitcoin/src/browser/adapters/index.ts
@@ -0,0 +1,3 @@
+export { default as UniSat } from './UniSat.ts'
+export { default as Xverse } from './Xverse.ts'
+export { default as Leather } from './Leather.ts'
diff --git a/packages/networks/bitcoin/src/browser/index.ts b/packages/networks/bitcoin/src/browser/index.ts
new file mode 100644
index 0000000..787e9bd
--- /dev/null
+++ b/packages/networks/bitcoin/src/browser/index.ts
@@ -0,0 +1,25 @@
+import { Wallet } from './Wallet.ts'
+import * as adapterList from './adapters/index.ts'
+import type {
+ WalletAdapterListType,
+ WalletAdapterInterface,
+ RegisterWalletAdapterType
+} from '@multiplechain/types'
+
+const adapters: WalletAdapterListType = {}
+
+const registerAdapter: RegisterWalletAdapterType = (adapter: WalletAdapterInterface): void => {
+ if (Object.values(adapters).find((a) => a.id === adapter.id) !== undefined) {
+ throw new Error(`Adapter with id ${adapter.id} already exists`)
+ }
+
+ adapters[adapter.id] = adapter
+}
+
+export * from '../index.ts'
+
+export const browser = {
+ Wallet,
+ registerAdapter,
+ adapters: Object.assign(adapters, adapterList)
+}
diff --git a/packages/networks/bitcoin/src/index.ts b/packages/networks/bitcoin/src/index.ts
new file mode 100644
index 0000000..7b0e1cd
--- /dev/null
+++ b/packages/networks/bitcoin/src/index.ts
@@ -0,0 +1,8 @@
+export * from './services/Provider.ts'
+
+export * as assets from './assets/index.ts'
+export * as models from './models/index.ts'
+export * as services from './services/index.ts'
+
+export * as utils from './utils.ts'
+export * as types from '@multiplechain/types'
diff --git a/packages/networks/bitcoin/src/models/CoinTransaction.ts b/packages/networks/bitcoin/src/models/CoinTransaction.ts
new file mode 100644
index 0000000..d681e05
--- /dev/null
+++ b/packages/networks/bitcoin/src/models/CoinTransaction.ts
@@ -0,0 +1,63 @@
+import { fromSatoshi } from '../utils.ts'
+import { Transaction } from './Transaction.ts'
+import { TransactionStatusEnum } from '@multiplechain/types'
+import { AssetDirectionEnum, type CoinTransactionInterface } from '@multiplechain/types'
+
+export class CoinTransaction extends Transaction implements CoinTransactionInterface {
+ /**
+ * @returns {Promise} Wallet address of the receiver of transaction
+ */
+ async getReceiver(): Promise {
+ const data = await this.getData()
+ return data?.vout[0].scriptpubkey_address ?? ''
+ }
+
+ /**
+ * @returns {Promise} Wallet address of the sender of transaction
+ */
+ async getSender(): Promise {
+ return await this.getSigner()
+ }
+
+ /**
+ * @returns {Promise} Amount of coin that will be transferred
+ */
+ async getAmount(): Promise {
+ const data = await this.getData()
+ return fromSatoshi(data?.vout[0].value ?? 0)
+ }
+
+ /**
+ * @param {AssetDirectionEnum} direction - Direction of the transaction (asset)
+ * @param {string} address - Wallet address of the receiver or sender of the transaction, dependant on direction
+ * @param {number} amount Amount of assets that will be transferred
+ * @returns {Promise} Status of the transaction
+ */
+ async verifyTransfer(
+ direction: AssetDirectionEnum,
+ address: string,
+ amount: number
+ ): Promise {
+ const status = await this.getStatus()
+
+ if (status === TransactionStatusEnum.PENDING) {
+ return TransactionStatusEnum.PENDING
+ }
+
+ if ((await this.getAmount()) !== amount) {
+ return TransactionStatusEnum.FAILED
+ }
+
+ if (direction === AssetDirectionEnum.INCOMING) {
+ if ((await this.getReceiver()).toLowerCase() !== address.toLowerCase()) {
+ return TransactionStatusEnum.FAILED
+ }
+ } else {
+ if ((await this.getSender()).toLowerCase() !== address.toLowerCase()) {
+ return TransactionStatusEnum.FAILED
+ }
+ }
+
+ return TransactionStatusEnum.CONFIRMED
+ }
+}
diff --git a/packages/networks/bitcoin/src/models/ContractTransaction.ts b/packages/networks/bitcoin/src/models/ContractTransaction.ts
new file mode 100644
index 0000000..4813a5a
--- /dev/null
+++ b/packages/networks/bitcoin/src/models/ContractTransaction.ts
@@ -0,0 +1,21 @@
+import { Transaction } from './Transaction.ts'
+import type { Provider } from '../services/Provider.ts'
+import type { ContractTransactionInterface } from '@multiplechain/types'
+
+export class ContractTransaction extends Transaction implements ContractTransactionInterface {
+ /**
+ * @param {string} id Transaction id
+ * @param {Provider} provider Blockchain network provider
+ */
+ constructor(id: string, provider?: Provider) {
+ super(id, provider)
+ throw new Error('This class is not implemented for Bitcoin.')
+ }
+
+ /**
+ * @returns {Promise} Contract address of the transaction
+ */
+ async getAddress(): Promise {
+ return 'example'
+ }
+}
diff --git a/packages/networks/bitcoin/src/models/NftTransaction.ts b/packages/networks/bitcoin/src/models/NftTransaction.ts
new file mode 100644
index 0000000..09b8d80
--- /dev/null
+++ b/packages/networks/bitcoin/src/models/NftTransaction.ts
@@ -0,0 +1,41 @@
+import { ContractTransaction } from './ContractTransaction.ts'
+import { TransactionStatusEnum } from '@multiplechain/types'
+import type { NftTransactionInterface, AssetDirectionEnum } from '@multiplechain/types'
+
+export class NftTransaction extends ContractTransaction implements NftTransactionInterface {
+ /**
+ * @returns {Promise} Receiver wallet address
+ */
+ async getReceiver(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @returns {Promise} Wallet address of the sender of transaction
+ */
+ async getSender(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @returns {Promise} NFT ID
+ */
+ async getNftId(): Promise {
+ return 0
+ }
+
+ /**
+ * @param {AssetDirectionEnum} direction - Direction of the transaction (nft)
+ * @param {string} address - Wallet address of the receiver or sender of the transaction, dependant on direction
+ * @param {number} nftId ID of the NFT that will be transferred
+ * @override verifyTransfer() in AssetTransactionInterface
+ * @returns {Promise} Status of the transaction
+ */
+ async verifyTransfer(
+ direction: AssetDirectionEnum,
+ address: string,
+ nftId: number | string
+ ): Promise {
+ return TransactionStatusEnum.PENDING
+ }
+}
diff --git a/packages/networks/bitcoin/src/models/TokenTransaction.ts b/packages/networks/bitcoin/src/models/TokenTransaction.ts
new file mode 100644
index 0000000..b1ea29e
--- /dev/null
+++ b/packages/networks/bitcoin/src/models/TokenTransaction.ts
@@ -0,0 +1,40 @@
+import { ContractTransaction } from './ContractTransaction.ts'
+import { TransactionStatusEnum } from '@multiplechain/types'
+import type { AssetDirectionEnum, TokenTransactionInterface } from '@multiplechain/types'
+
+export class TokenTransaction extends ContractTransaction implements TokenTransactionInterface {
+ /**
+ * @returns {Promise} Wallet address of the receiver of transaction
+ */
+ async getReceiver(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @returns {Promise} Wallet address of the sender of transaction
+ */
+ async getSender(): Promise {
+ return 'example'
+ }
+
+ /**
+ * @returns {Promise} Amount of tokens that will be transferred
+ */
+ async getAmount(): Promise {
+ return 0
+ }
+
+ /**
+ * @param {AssetDirectionEnum} direction - Direction of the transaction (token)
+ * @param {string} address - Wallet address of the owner or spender of the transaction, dependant on direction
+ * @param {number} amount Amount of tokens that will be approved
+ * @returns {Promise} Status of the transaction
+ */
+ async verifyTransfer(
+ direction: AssetDirectionEnum,
+ address: string,
+ amount: number
+ ): Promise {
+ return TransactionStatusEnum.PENDING
+ }
+}
diff --git a/packages/networks/bitcoin/src/models/Transaction.ts b/packages/networks/bitcoin/src/models/Transaction.ts
new file mode 100644
index 0000000..4cdeed8
--- /dev/null
+++ b/packages/networks/bitcoin/src/models/Transaction.ts
@@ -0,0 +1,205 @@
+import { fromSatoshi, sleep } from '../utils.ts'
+import axios, { type AxiosError } from 'axios'
+import { Provider } from '../services/Provider.ts'
+import type { TransactionInterface } from '@multiplechain/types'
+import { ErrorTypeEnum, TransactionStatusEnum } from '@multiplechain/types'
+
+export interface VinObject {
+ txid: string
+ vout: number
+ prevout: {
+ scriptpubkey: string
+ scriptpubkey_asm: string
+ scriptpubkey_type: string
+ scriptpubkey_address: string
+ value: number
+ }
+ scriptsig: string
+ scriptsig_asm: string
+ witness: string[]
+ is_coinbase: boolean
+ sequence: number
+}
+
+export interface VoutObject {
+ scriptpubkey: string
+ scriptpubkey_asm: string
+ scriptpubkey_type: string
+ scriptpubkey_address: string
+ value: number
+}
+
+export interface TransactionData {
+ txid: string
+ version: number
+ locktime: number
+ vin: VinObject[]
+ vout: VoutObject[]
+ size: number
+ weight: number
+ fee: number
+ status: {
+ confirmed: boolean
+ block_height: number
+ block_hash: string
+ block_time: number
+ }
+}
+
+let counter = 0
+
+export class Transaction implements TransactionInterface {
+ /**
+ * Each transaction has its own unique ID defined by the user
+ */
+ id: string
+
+ /**
+ * Blockchain network provider
+ */
+ provider: Provider
+
+ /**
+ * Transaction data
+ */
+ data: TransactionData | null = null
+
+ /**
+ * @param {string} id Transaction id
+ * @param {Provider} provider Blockchain network provider
+ */
+ constructor(id: string, provider?: Provider) {
+ this.id = id
+ this.provider = provider ?? Provider.instance
+ }
+
+ /**
+ * @returns {Promise} Transaction data
+ */
+ async getData(): Promise {
+ if (this.data !== null) {
+ return this.data
+ }
+ try {
+ const data = (await axios.get(this.provider.createEndpoint('tx/' + this.id))).data
+
+ if (data?.txid !== this.id) {
+ return (this.data = null)
+ }
+
+ return (this.data = data as TransactionData)
+ } catch (error) {
+ const axiosError = error as AxiosError
+ // Returns empty data when the transaction is first created. For this reason, it would be better to check it intermittently and give an error if it still does not exist. Average 10 seconds.
+ if (String(axiosError?.response?.data).includes('Transaction not found')) {
+ if (counter > 5) {
+ throw new Error(ErrorTypeEnum.TRANSACTION_NOT_FOUND)
+ }
+ counter++
+ await sleep(2000)
+ return await this.getData()
+ }
+ throw new Error(ErrorTypeEnum.RPC_REQUEST_ERROR)
+ }
+ }
+
+ /**
+ * @param {number} ms - Milliseconds to wait for the transaction to be confirmed. Default is 4000ms
+ * @returns {Promise} Status of the transaction
+ */
+ async wait(ms: number = 4000): Promise {
+ return await new Promise((resolve, reject) => {
+ const check = async (): Promise => {
+ try {
+ const status = await this.getStatus()
+ if (status === TransactionStatusEnum.CONFIRMED) {
+ resolve(TransactionStatusEnum.CONFIRMED)
+ return
+ } else if (status === TransactionStatusEnum.FAILED) {
+ reject(TransactionStatusEnum.FAILED)
+ return
+ }
+ setTimeout(check, ms)
+ } catch (error) {
+ reject(TransactionStatusEnum.FAILED)
+ }
+ }
+ void check()
+ })
+ }
+
+ /**
+ * @returns {string} Transaction ID
+ */
+ getId(): string {
+ return this.id
+ }
+
+ /**
+ * @returns {string} Transaction URL
+ */
+ getUrl(): string {
+ return this.provider.explorer + 'tx/' + this.id
+ }
+
+ /**
+ * @returns {Promise} Wallet address of the sender of transaction
+ */
+ async getSigner(): Promise {
+ const data = await this.getData()
+ return data?.vin[0].prevout.scriptpubkey_address ?? ''
+ }
+
+ /**
+ * @returns {Promise} Transaction fee
+ */
+ async getFee(): Promise {
+ const data = await this.getData()
+ return fromSatoshi(data?.fee ?? 0)
+ }
+
+ /**
+ * @returns {Promise} Block number that transaction
+ */
+ async getBlockNumber(): Promise {
+ const data = await this.getData()
+ return data?.status?.block_height ?? 0
+ }
+
+ /**
+ * @returns {Promise} Block timestamp that transaction
+ */
+ async getBlockTimestamp(): Promise {
+ const data = await this.getData()
+ return data?.status?.block_time ?? 0
+ }
+
+ /**
+ * @returns {Promise} Confirmation count of the block
+ */
+ async getBlockConfirmationCount(): Promise {
+ const data = await this.getData()
+ if (data === null) {
+ return 0
+ }
+ const latestBlock = await axios.get(this.provider.createEndpoint('blocks/tip/height'))
+ return (latestBlock.data as number) - data?.status?.block_height
+ }
+
+ /**
+ * @returns {Promise} Status of the transaction
+ */
+ async getStatus(): Promise {
+ const data = await this.getData()
+ if (data === null) {
+ return TransactionStatusEnum.PENDING
+ } else if (data.status?.block_height !== undefined) {
+ if (data.status.confirmed) {
+ return TransactionStatusEnum.CONFIRMED
+ } else {
+ return TransactionStatusEnum.FAILED
+ }
+ }
+ return TransactionStatusEnum.PENDING
+ }
+}
diff --git a/packages/networks/bitcoin/src/models/index.ts b/packages/networks/bitcoin/src/models/index.ts
new file mode 100644
index 0000000..cc0a27c
--- /dev/null
+++ b/packages/networks/bitcoin/src/models/index.ts
@@ -0,0 +1,5 @@
+export * from './Transaction.ts'
+export * from './NftTransaction.ts'
+export * from './CoinTransaction.ts'
+export * from './TokenTransaction.ts'
+export * from './ContractTransaction.ts'
diff --git a/packages/networks/bitcoin/src/services/Provider.ts b/packages/networks/bitcoin/src/services/Provider.ts
new file mode 100644
index 0000000..30086e2
--- /dev/null
+++ b/packages/networks/bitcoin/src/services/Provider.ts
@@ -0,0 +1,156 @@
+import axios from 'axios'
+import { checkWebSocket } from '@multiplechain/utils'
+import { ErrorTypeEnum, type ProviderInterface } from '@multiplechain/types'
+
+export interface BitcoinNetworkConfigInterface {
+ testnet: boolean
+ blockCypherToken?: string
+}
+
+export class Provider implements Omit {
+ /**
+ * Network configuration of the provider
+ */
+ network: BitcoinNetworkConfigInterface
+
+ /**
+ * API URL
+ */
+ api: string
+
+ /**
+ * Explorer URL
+ */
+ explorer: string
+
+ /**
+ * Websocket URL
+ */
+ wsUrl: string
+
+ /**
+ * BlockCypher token
+ */
+ blockCypherToken?: string
+
+ /**
+ * Default BlockCypher token
+ */
+ defaultBlockCypherToken = '49d43a59a4f24d31a9731eb067ab971c'
+
+ /**
+ * Static instance of the provider
+ */
+ private static _instance: Provider
+
+ /**
+ * @param network - Network configuration of the provider
+ */
+ constructor(network: BitcoinNetworkConfigInterface) {
+ this.update(network)
+ }
+
+ /**
+ * Get the static instance of the provider
+ * @returns {Provider} Provider
+ */
+ static get instance(): Provider {
+ if (Provider._instance === undefined) {
+ throw new Error(ErrorTypeEnum.PROVIDER_IS_NOT_INITIALIZED)
+ }
+ return Provider._instance
+ }
+
+ /**
+ * Initialize the static instance of the provider
+ * @param {BitcoinNetworkConfigInterface} network - Network configuration of the provider
+ * @returns {void}
+ */
+ static initialize(network: BitcoinNetworkConfigInterface): void {
+ if (Provider._instance !== undefined) {
+ throw new Error(ErrorTypeEnum.PROVIDER_IS_ALREADY_INITIALIZED)
+ }
+ Provider._instance = new Provider(network)
+ }
+
+ /**
+ * Check RPC connection
+ * @param {string} url - RPC URL
+ * @returns {Promise}
+ */
+ async checkRpcConnection(url?: string): Promise {
+ try {
+ const response = await axios.get(url ?? this.createEndpoint('blocks/tip/height'))
+
+ if (response.status !== 200) {
+ return new Error(response.statusText + ': ' + JSON.stringify(response.data))
+ }
+
+ return true
+ } catch (error) {
+ return error as Error
+ }
+ }
+
+ /**
+ * Check WS connection
+ * @param {string} url - Websocket URL
+ * @returns {Promise}
+ */
+ async checkWsConnection(url?: string): Promise {
+ try {
+ const result: any = await checkWebSocket(url ?? this.wsUrl)
+
+ if (result instanceof Error) {
+ return result
+ }
+
+ return true
+ } catch (error) {
+ return error as Error
+ }
+ }
+
+ /**
+ * Update network configuration of the provider
+ * @param network - Network configuration of the provider
+ */
+ update(network: BitcoinNetworkConfigInterface): void {
+ this.network = network
+ Provider._instance = this
+ this.blockCypherToken = this.network.blockCypherToken
+ if (this.network.testnet) {
+ this.api = 'https://blockstream.info/testnet/api/'
+ this.explorer = 'https://blockstream.info/testnet/'
+ const token = this.network.blockCypherToken ?? this.defaultBlockCypherToken
+ this.wsUrl = 'wss://socket.blockcypher.com/v1/btc/test3?token=' + token
+ } else {
+ this.api = 'https://blockstream.info/api/'
+ this.explorer = 'https://blockstream.info/'
+ if (this.network.blockCypherToken !== undefined) {
+ this.wsUrl =
+ 'wss://socket.blockcypher.com/v1/btc/main?token=' +
+ this.network.blockCypherToken
+ } else {
+ this.wsUrl = 'wss://ws.blockchain.info/inv'
+ }
+ }
+ }
+
+ /**
+ * Create a new endpoint
+ * @param {string} endpoint - Endpoint
+ * @returns {string} Endpoint
+ */
+ createEndpoint(endpoint: string): string {
+ return this.api + endpoint
+ }
+
+ /**
+ * Get the current network configuration is testnet or not
+ * @returns boolean
+ */
+ isTestnet(): boolean {
+ return this.network?.testnet ?? false
+ }
+}
diff --git a/packages/networks/bitcoin/src/services/TransactionListener.ts b/packages/networks/bitcoin/src/services/TransactionListener.ts
new file mode 100644
index 0000000..04fa972
--- /dev/null
+++ b/packages/networks/bitcoin/src/services/TransactionListener.ts
@@ -0,0 +1,352 @@
+import type {
+ TransactionTypeEnum,
+ DynamicTransactionType,
+ TransactionListenerInterface,
+ DynamicTransactionListenerFilterType
+} from '@multiplechain/types'
+import WebSocket from 'ws'
+import { Provider } from './Provider.ts'
+import { fromSatoshi } from '../utils.ts'
+import { Transaction } from '../models/Transaction.ts'
+import type { NftTransaction } from '../models/NftTransaction.ts'
+import { CoinTransaction } from '../models/CoinTransaction.ts'
+import type { TokenTransaction } from '../models/TokenTransaction.ts'
+import type { ContractTransaction } from '../models/ContractTransaction.ts'
+import { checkWebSocket, objectsEqual } from '@multiplechain/utils'
+import { TransactionListenerProcessIndex } from '@multiplechain/types'
+
+interface Values {
+ txId: string
+ amount?: number
+ sender?: string
+ receiver?: string
+}
+
+type TransactionListenerTriggerType = DynamicTransactionType<
+ T,
+ Transaction,
+ ContractTransaction,
+ CoinTransaction,
+ TokenTransaction,
+ NftTransaction
+>
+
+type TransactionListenerCallbackType<
+ T extends TransactionTypeEnum,
+ Transaction = TransactionListenerTriggerType
+> = (transaction: Transaction) => void
+
+export class TransactionListener<
+ T extends TransactionTypeEnum,
+ DTransaction extends TransactionListenerTriggerType,
+ CallBackType extends TransactionListenerCallbackType
+> implements TransactionListenerInterface
+{
+ /**
+ * Transaction type
+ */
+ type: T
+
+ /**
+ * Transaction listener callback
+ */
+ callbacks: CallBackType[] = []
+
+ /**
+ * Transaction listener filter
+ */
+ filter?: DynamicTransactionListenerFilterType | Record
+
+ /**
+ * Provider
+ */
+ provider: Provider
+
+ /**
+ * Listener status
+ */
+ status: boolean = false
+
+ /**
+ * Triggered transactions
+ */
+ triggeredTransactions: string[] = []
+
+ /**
+ * WebSocket
+ */
+ webSocket: WebSocket
+
+ /**
+ * Dynamic stop method
+ */
+ dynamicStop: () => void = () => {}
+
+ /**
+ * @param {T} type - Transaction type
+ * @param {DynamicTransactionListenerFilterType} filter - Transaction listener filter
+ * @param {Provider} provider - Provider
+ */
+ constructor(type: T, filter?: DynamicTransactionListenerFilterType, provider?: Provider) {
+ this.type = type
+ this.filter = filter ?? {}
+ this.provider = provider ?? Provider.instance
+ }
+
+ /**
+ * Close the listener
+ * @returns {void}
+ */
+ stop(): void {
+ if (this.status) {
+ this.status = false
+ this.dynamicStop()
+ }
+ }
+
+ /**
+ * Start the listener
+ * @returns {void}
+ */
+ start(): void {
+ if (!this.status) {
+ this.status = true
+ // @ts-expect-error allow dynamic access
+ this[TransactionListenerProcessIndex[this.type]]()
+ }
+ }
+
+ /**
+ * Get the listener status
+ * @returns {boolean} Listener status
+ */
+ getStatus(): boolean {
+ return this.status
+ }
+
+ /**
+ * Listen to the transaction events
+ * @param {CallBackType} callback - Transaction listener callback
+ * @returns {Promise}
+ */
+ async on(callback: CallBackType): Promise {
+ if (this.webSocket === undefined) {
+ try {
+ await checkWebSocket(this.provider.wsUrl)
+ this.webSocket = new WebSocket(this.provider.wsUrl)
+ } catch (error) {
+ throw new Error(
+ 'WebSocket connection is not available' +
+ (error instanceof Error ? ': ' + error.message : '')
+ )
+ }
+ }
+
+ this.start()
+ this.callbacks.push(callback)
+
+ return true
+ }
+
+ /**
+ * Trigger the event when a transaction is detected
+ * @param {TransactionListenerTriggerType} transaction - Transaction data
+ * @returns {void}
+ */
+ trigger(transaction: TransactionListenerTriggerType): void {
+ if (!this.triggeredTransactions.includes(transaction.id)) {
+ this.triggeredTransactions.push(transaction.id)
+ this.callbacks.forEach((callback) => {
+ callback(transaction as unknown as DTransaction)
+ })
+ }
+ }
+
+ isBlockCypherProcess(): boolean {
+ return this.provider.isTestnet() || this.provider.blockCypherToken !== undefined
+ }
+
+ /**
+ * Create message for the listener
+ * @param {string} receiver - Receiver address
+ * @returns {string} Message
+ */
+ createMessage(receiver?: string): string {
+ let message
+ if (this.isBlockCypherProcess()) {
+ interface Config {
+ event: string
+ token: string
+ address?: string
+ }
+
+ const config: Config = {
+ event: 'unconfirmed-tx',
+ token: this.provider.blockCypherToken ?? this.provider.defaultBlockCypherToken
+ }
+
+ if (receiver !== undefined) {
+ config.address = receiver
+ }
+
+ message = JSON.stringify(config)
+ } else {
+ message = JSON.stringify({
+ op: 'unconfirmed_sub'
+ })
+ }
+
+ this.dynamicStop = () => {
+ if (!this.isBlockCypherProcess()) {
+ this.webSocket.send(
+ JSON.stringify({
+ op: 'unconfirmed_unsub'
+ })
+ )
+ }
+ this.webSocket.close()
+ }
+
+ return message
+ }
+
+ /**
+ * Parse the data
+ * @param {any} data - Data
+ * @returns {Values} Parsed data
+ */
+ getValues(data: any): Values {
+ const values: Values = {
+ txId: ''
+ }
+
+ if (this.isBlockCypherProcess()) {
+ values.txId = data.hash
+ values.sender = data.inputs[0].addresses[0]
+ values.receiver = data.outputs[0].addresses[0]
+ values.amount = fromSatoshi(data.outputs[0].value as number)
+ } else {
+ values.txId = data.x.hash
+ values.receiver = data.x.out[0].addr
+ values.sender = data.x.inputs[0].prev_out.addr
+ values.amount = fromSatoshi(data.x.out[0].value as number)
+ }
+
+ return values
+ }
+
+ /**
+ * General transaction process
+ * @returns {void}
+ */
+ generalProcess(): void {
+ const message = this.createMessage()
+
+ this.webSocket.addEventListener('open', () => {
+ this.webSocket.send(message)
+ })
+
+ this.webSocket.addEventListener('message', async (res) => {
+ const values = this.getValues(JSON.parse(res.data as string))
+
+ if (
+ this.filter?.signer !== undefined &&
+ values.sender !== this.filter.signer.toLowerCase()
+ ) {
+ return
+ }
+
+ this.trigger(new Transaction(values.txId))
+ })
+ }
+
+ /**
+ * Contract transaction process
+ * @returns {void}
+ */
+ contractProcess(): void {
+ throw new Error('This method is not implemented for Bitcoin.')
+ }
+
+ /**
+ * Coin transaction process
+ * @returns {void}
+ */
+ coinProcess(): void {
+ const filter = this.filter as DynamicTransactionListenerFilterType
+
+ if (
+ filter.signer !== undefined &&
+ filter.sender !== undefined &&
+ filter.signer !== filter.sender
+ ) {
+ throw new Error(
+ 'Sender and signer must be the same in coin transactions. Or only one of them can be defined.'
+ )
+ }
+
+ const sender = filter.sender ?? filter.signer
+
+ const message = this.createMessage(filter.receiver)
+
+ this.webSocket.addEventListener('open', () => {
+ this.webSocket.send(message)
+ })
+
+ this.webSocket.addEventListener('message', async (res) => {
+ const data = JSON.parse(res.data as string)
+
+ interface ParamsType {
+ sender?: string
+ receiver?: string
+ }
+
+ const expectedParams: ParamsType = {}
+ const receivedParams: ParamsType = {}
+
+ if (String(data.event).includes('events limit reached')) {
+ throw new Error('BlockCypher events limit reached.')
+ }
+
+ const values = this.getValues(data)
+
+ if (sender !== undefined) {
+ expectedParams.sender = sender.toLowerCase()
+ receivedParams.sender = values.sender?.toLowerCase()
+ }
+
+ if (filter.receiver !== undefined) {
+ expectedParams.receiver = filter.receiver.toLowerCase()
+ receivedParams.receiver = values.receiver?.toLowerCase()
+ }
+
+ if (!objectsEqual(expectedParams, receivedParams)) {
+ return
+ }
+
+ const transaction = new CoinTransaction(values.txId)
+
+ if (filter.amount !== undefined && values.amount !== filter.amount) {
+ return
+ }
+
+ this.trigger(transaction)
+ })
+ }
+
+ /**
+ * Token transaction process
+ * @returns {void}
+ */
+ tokenProcess(): void {
+ throw new Error('This method is not implemented for Bitcoin.')
+ }
+
+ /**
+ * NFT transaction process
+ * @returns {void}
+ */
+ nftProcess(): void {
+ throw new Error('This method is not implemented for Bitcoin.')
+ }
+}
diff --git a/packages/networks/bitcoin/src/services/TransactionSigner.ts b/packages/networks/bitcoin/src/services/TransactionSigner.ts
new file mode 100644
index 0000000..15ae13a
--- /dev/null
+++ b/packages/networks/bitcoin/src/services/TransactionSigner.ts
@@ -0,0 +1,115 @@
+import axios from 'axios'
+import type { AxiosError } from 'axios'
+import { Provider } from '../services/Provider.ts'
+import { Transaction } from '../models/Transaction.ts'
+import { NftTransaction } from '../models/NftTransaction.ts'
+import { CoinTransaction } from '../models/CoinTransaction.ts'
+import { TokenTransaction } from '../models/TokenTransaction.ts'
+import { type TransactionSignerInterface } from '@multiplechain/types'
+import type { Transaction as BitcoreLibTransactionData } from 'bitcore-lib'
+
+export interface TransactionData {
+ sender: string
+ receiver: string
+ amount: number
+ bitcoreLib: BitcoreLibTransactionData
+}
+
+export class TransactionSigner implements TransactionSignerInterface {
+ /**
+ * Transaction data from the blockchain network
+ */
+ rawData: TransactionData
+
+ /**
+ * Signed transaction data
+ */
+ signedData?: string
+
+ /**
+ * Blockchain network provider
+ */
+ provider: Provider
+
+ /**
+ * @param {TransactionData} rawData - Transaction data
+ */
+ constructor(rawData: TransactionData, provider?: Provider) {
+ this.rawData = rawData
+ this.provider = provider ?? Provider.instance
+ }
+
+ /**
+ * Sign the transaction
+ * @param {string} privateKey - Transaction data
+ * @returns {Promise} Signed transaction data
+ */
+ async sign(privateKey: string): Promise {
+ this.rawData.bitcoreLib.sign(privateKey)
+ this.signedData = this.rawData.bitcoreLib.serialize()
+ return this
+ }
+
+ /**
+ * Send the transaction to the blockchain network
+ * @returns {Promise}
+ */
+ async send(): Promise {
+ try {
+ const result = await axios({
+ method: 'POST',
+ url: `https://blockstream.info/testnet/api/tx`,
+ data: this.signedData
+ })
+ return new Transaction(result.data as string)
+ } catch (error: any) {
+ throw new Error(JSON.stringify((error as AxiosError).response?.data))
+ }
+ }
+
+ /**
+ * Get the raw transaction data
+ * @returns Transaction data
+ */
+ getRawData(): TransactionData {
+ return this.rawData
+ }
+
+ /**
+ * Get the signed transaction data
+ * @returns Signed transaction data
+ */
+ getSignedData(): string {
+ return this.signedData ?? ''
+ }
+}
+
+export class CoinTransactionSigner extends TransactionSigner {
+ /**
+ * Send the transaction to the blockchain network
+ * @returns {Promise} Transaction data
+ */
+ async send(): Promise {
+ return new CoinTransaction((await super.send()).getId())
+ }
+}
+
+export class TokenTransactionSigner extends TransactionSigner {
+ /**
+ * Send the transaction to the blockchain network
+ * @returns {Promise} Transaction data
+ */
+ async send(): Promise {
+ return new TokenTransaction((await super.send()).getId())
+ }
+}
+
+export class NftTransactionSigner extends TransactionSigner {
+ /**
+ * Send the transaction to the blockchain network
+ * @returns {Promise} Transaction data
+ */
+ async send(): Promise {
+ return new NftTransaction((await super.send()).getId())
+ }
+}
diff --git a/packages/networks/bitcoin/src/services/index.ts b/packages/networks/bitcoin/src/services/index.ts
new file mode 100644
index 0000000..549a382
--- /dev/null
+++ b/packages/networks/bitcoin/src/services/index.ts
@@ -0,0 +1,2 @@
+export * from './TransactionSigner.ts'
+export * from './TransactionListener.ts'
diff --git a/packages/networks/bitcoin/src/utils.ts b/packages/networks/bitcoin/src/utils.ts
new file mode 100644
index 0000000..14bd3ec
--- /dev/null
+++ b/packages/networks/bitcoin/src/utils.ts
@@ -0,0 +1,11 @@
+import { math } from '@multiplechain/utils'
+
+export * from '@multiplechain/utils'
+
+export const fromSatoshi = (amount: number): number => {
+ return math.div(amount, 100000000, 8)
+}
+
+export const toSatoshi = (amount: number): number => {
+ return math.mul(amount, 100000000, 8)
+}
diff --git a/packages/networks/bitcoin/tests/assets.spec.ts b/packages/networks/bitcoin/tests/assets.spec.ts
new file mode 100644
index 0000000..36147df
--- /dev/null
+++ b/packages/networks/bitcoin/tests/assets.spec.ts
@@ -0,0 +1,63 @@
+import { describe, it, expect, assert } from 'vitest'
+
+import { Coin } from '../src/assets/Coin.ts'
+import { math } from '@multiplechain/utils'
+import { Transaction } from '../src/models/Transaction.ts'
+import { TransactionStatusEnum } from '@multiplechain/types'
+import { TransactionSigner } from '../src/services/TransactionSigner.ts'
+
+const testAmount = Number(process.env.BTC_TRANSFER_AMOUNT)
+const senderTestAddress = String(process.env.BTC_SENDER_ADDRESS)
+const receiverTestAddress = String(process.env.BTC_RECEIVER_ADDRESS)
+const senderPrivateKey = String(process.env.BTC_SENDER_PRIVATE_KEY)
+const transferTestIsActive = Boolean(process.env.BTC_TRANSFER_TEST_IS_ACTIVE !== 'false')
+
+const checkSigner = async (signer: TransactionSigner, privateKey?: string): Promise => {
+ expect(signer).toBeInstanceOf(TransactionSigner)
+
+ const rawData = signer.getRawData()
+
+ assert.isObject(rawData)
+
+ await signer.sign(privateKey ?? senderPrivateKey)
+
+ assert.isString(signer.getSignedData())
+}
+
+const checkTx = async (transaction: Transaction): Promise => {
+ expect(transaction).toBeInstanceOf(Transaction)
+ const status = await transaction.wait(10 * 1000)
+ expect(status).toBe(TransactionStatusEnum.CONFIRMED)
+}
+
+describe('Coin', () => {
+ const coin = new Coin()
+ it('Name and symbol', () => {
+ expect(coin.getName()).toBe('Bitcoin')
+ expect(coin.getSymbol()).toBe('BTC')
+ })
+
+ it('Decimals', () => {
+ expect(coin.getDecimals()).toBe(8)
+ })
+
+ it('Balance', async () => {
+ const balance = await coin.getBalance('tb1qc240vx54n08hnhx8l4rqxjzcxf4f0ssq5asawm')
+ expect(balance).toBe(0.00003)
+ })
+
+ it('Transfer', async () => {
+ const signer = await coin.transfer(senderTestAddress, receiverTestAddress, testAmount)
+
+ await checkSigner(signer)
+
+ if (!transferTestIsActive) return
+
+ const beforeBalance = await coin.getBalance(receiverTestAddress)
+
+ await checkTx(await signer.send())
+
+ const afterBalance = await coin.getBalance(receiverTestAddress)
+ expect(afterBalance).toBe(math.add(beforeBalance, testAmount))
+ })
+})
diff --git a/packages/networks/bitcoin/tests/models.spec.ts b/packages/networks/bitcoin/tests/models.spec.ts
new file mode 100644
index 0000000..d4265de
--- /dev/null
+++ b/packages/networks/bitcoin/tests/models.spec.ts
@@ -0,0 +1,78 @@
+import { describe, it, expect } from 'vitest'
+import { Transaction } from '../src/models/Transaction.ts'
+import { CoinTransaction } from '../src/models/CoinTransaction.ts'
+import { AssetDirectionEnum, TransactionStatusEnum } from '@multiplechain/types'
+
+const testAmount = Number(process.env.BTC_TRANSFER_AMOUNT)
+const senderTestAddress = String(process.env.BTC_SENDER_ADDRESS)
+const receiverTestAddress = String(process.env.BTC_RECEIVER_ADDRESS)
+const txId = '335c8a251e5f18121977c3159f46983d5943325abccc19e4718c49089553d60c'
+
+describe('Transaction', () => {
+ const tx = new Transaction(txId)
+ it('Id', async () => {
+ expect(tx.getId()).toBe(txId)
+ })
+
+ it('Data', async () => {
+ expect(await tx.getData()).toBeTypeOf('object')
+ })
+
+ it('Wait', async () => {
+ expect(await tx.wait()).toBe(TransactionStatusEnum.CONFIRMED)
+ })
+
+ it('URL', async () => {
+ expect(tx.getUrl()).toBe('https://blockstream.info/testnet/tx/' + txId)
+ })
+
+ it('Sender', async () => {
+ expect((await tx.getSigner()).toLowerCase()).toBe(senderTestAddress.toLowerCase())
+ })
+
+ it('Fee', async () => {
+ expect(await tx.getFee()).toBe(0.00014)
+ })
+
+ it('Block Number', async () => {
+ expect(await tx.getBlockNumber()).toBe(2814543)
+ })
+
+ it('Block Timestamp', async () => {
+ expect(await tx.getBlockTimestamp()).toBe(1715328679)
+ })
+
+ it('Block Confirmation Count', async () => {
+ expect(await tx.getBlockConfirmationCount()).toBeGreaterThan(13)
+ })
+
+ it('Status', async () => {
+ expect(await tx.getStatus()).toBe(TransactionStatusEnum.CONFIRMED)
+ })
+})
+
+describe('Coin Transaction', () => {
+ const tx = new CoinTransaction(txId)
+
+ it('Receiver', async () => {
+ expect((await tx.getReceiver()).toLowerCase()).toBe(receiverTestAddress.toLowerCase())
+ })
+
+ it('Amount', async () => {
+ expect(await tx.getAmount()).toBe(testAmount)
+ })
+
+ it('Verify Transfer', async () => {
+ expect(
+ await tx.verifyTransfer(AssetDirectionEnum.INCOMING, receiverTestAddress, testAmount)
+ ).toBe(TransactionStatusEnum.CONFIRMED)
+
+ expect(
+ await tx.verifyTransfer(AssetDirectionEnum.OUTGOING, senderTestAddress, testAmount)
+ ).toBe(TransactionStatusEnum.CONFIRMED)
+
+ expect(
+ await tx.verifyTransfer(AssetDirectionEnum.OUTGOING, receiverTestAddress, testAmount)
+ ).toBe(TransactionStatusEnum.FAILED)
+ })
+})
diff --git a/packages/networks/bitcoin/tests/services.spec.ts b/packages/networks/bitcoin/tests/services.spec.ts
new file mode 100644
index 0000000..d5e59cb
--- /dev/null
+++ b/packages/networks/bitcoin/tests/services.spec.ts
@@ -0,0 +1,67 @@
+import { describe, it, expect } from 'vitest'
+
+import { provider } from './setup.ts'
+import { Provider } from '../src/services/Provider.ts'
+import { TransactionListener } from '../src/services/TransactionListener.ts'
+import { TransactionTypeEnum } from '@multiplechain/types'
+import { CoinTransaction } from '../src/models/CoinTransaction.ts'
+import { Coin } from '../src/assets/Coin.ts'
+import { sleep } from '@multiplechain/utils'
+
+const senderTestAddress = String(process.env.BTC_SENDER_ADDRESS)
+const receiverTestAddress = String(process.env.BTC_RECEIVER_ADDRESS)
+const senderPrivateKey = String(process.env.BTC_SENDER_PRIVATE_KEY)
+const listenerTestIsActive = Boolean(process.env.BTC_LISTENER_TEST_IS_ACTIVE !== 'false')
+
+describe('Provider', () => {
+ it('isTestnet', () => {
+ expect(provider.isTestnet()).toBe(true)
+ })
+
+ it('instance', () => {
+ expect(Provider.instance).toBe(provider)
+ })
+
+ it('checkRpcConnection', async () => {
+ expect(await provider.checkRpcConnection()).toBe(true)
+ })
+
+ it('checkWsConnection', async () => {
+ expect(await provider.checkWsConnection()).toBe(true)
+ })
+})
+
+describe('Transaction Listener', () => {
+ if (!listenerTestIsActive) {
+ it('No test is active', () => {
+ expect(true).toBe(true)
+ })
+ return
+ }
+
+ it('Coin', async () => {
+ const listener = new TransactionListener(TransactionTypeEnum.COIN, {
+ signer: senderTestAddress,
+ receiver: receiverTestAddress
+ })
+
+ const signer = await new Coin().transfer(senderTestAddress, receiverTestAddress, 0.0001)
+
+ const waitListenerEvent = async (): Promise => {
+ return await new Promise((resolve, reject) => {
+ void listener
+ .on((transaction) => {
+ listener.stop()
+ resolve(transaction)
+ })
+ .then(async () => {
+ await sleep(2000)
+ void (await signer.sign(senderPrivateKey)).send()
+ })
+ .catch(reject)
+ })
+ }
+
+ expect(await waitListenerEvent()).toBeInstanceOf(CoinTransaction)
+ })
+})
diff --git a/packages/networks/bitcoin/tests/setup.ts b/packages/networks/bitcoin/tests/setup.ts
new file mode 100644
index 0000000..6c04b6b
--- /dev/null
+++ b/packages/networks/bitcoin/tests/setup.ts
@@ -0,0 +1,13 @@
+import { Provider } from '../src/services/Provider.ts'
+
+let provider: Provider
+
+try {
+ provider = Provider.instance
+} catch (e) {
+ provider = new Provider({
+ testnet: true
+ })
+}
+
+export { provider }
diff --git a/packages/networks/bitcoin/tsconfig.json b/packages/networks/bitcoin/tsconfig.json
new file mode 100644
index 0000000..e40c547
--- /dev/null
+++ b/packages/networks/bitcoin/tsconfig.json
@@ -0,0 +1,21 @@
+{
+ "compilerOptions": {
+ "noEmit": true,
+ "composite": true,
+ "declaration": true,
+ "outDir": "./dist/esm",
+ "declarationDir": "./dist/types"
+ },
+ "extends": "../../../tsconfig.json",
+ "include": [
+ "src",
+ ".eslintrc.json",
+ "tests",
+ "vite.config.ts",
+ "esbuild.ts",
+ "vitest.config.ts",
+ "../../../esbuild.ts",
+ "../../../vite.config.ts",
+ "../../../vitest.config.ts",
+ ]
+}
diff --git a/packages/networks/bitcoin/vite.config.ts b/packages/networks/bitcoin/vite.config.ts
new file mode 100644
index 0000000..87de654
--- /dev/null
+++ b/packages/networks/bitcoin/vite.config.ts
@@ -0,0 +1,10 @@
+import { mergeConfig } from 'vite'
+import mainConfig from '../../../vite.config.ts'
+
+export default mergeConfig(mainConfig, {
+ build: {
+ lib: {
+ name: 'Bitcoin'
+ }
+ }
+})
diff --git a/packages/networks/bitcoin/vite.svg b/packages/networks/bitcoin/vite.svg
new file mode 100644
index 0000000..e7b8dfb
--- /dev/null
+++ b/packages/networks/bitcoin/vite.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/packages/networks/bitcoin/vitest.config.ts b/packages/networks/bitcoin/vitest.config.ts
new file mode 100644
index 0000000..c96edcb
--- /dev/null
+++ b/packages/networks/bitcoin/vitest.config.ts
@@ -0,0 +1,12 @@
+import { mergeConfig, defineConfig } from 'vitest/config'
+import mainConfig from '../../../vite.config.ts'
+
+export default mergeConfig(
+ mainConfig,
+ defineConfig({
+ test: {
+ testTimeout: 600000,
+ setupFiles: ['./tests/setup.ts']
+ }
+ })
+)
diff --git a/packages/networks/boilerplate/index.html b/packages/networks/boilerplate/index.html
index 236da6d..4e508e4 100644
--- a/packages/networks/boilerplate/index.html
+++ b/packages/networks/boilerplate/index.html
@@ -187,9 +187,6 @@
Deep link:
-
- Chain id:
-
Connected address:
@@ -215,7 +212,7 @@
+