otx-btc-wallet-connectors 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +554 -0
- package/dist/base-IAFq7sd8.d.mts +53 -0
- package/dist/base-IAFq7sd8.d.ts +53 -0
- package/dist/binance/index.d.mts +81 -0
- package/dist/binance/index.d.ts +81 -0
- package/dist/binance/index.js +13 -0
- package/dist/binance/index.js.map +1 -0
- package/dist/binance/index.mjs +4 -0
- package/dist/binance/index.mjs.map +1 -0
- package/dist/bitget/index.d.mts +84 -0
- package/dist/bitget/index.d.ts +84 -0
- package/dist/bitget/index.js +13 -0
- package/dist/bitget/index.js.map +1 -0
- package/dist/bitget/index.mjs +4 -0
- package/dist/bitget/index.mjs.map +1 -0
- package/dist/chunk-5Z5Q2Y75.mjs +91 -0
- package/dist/chunk-5Z5Q2Y75.mjs.map +1 -0
- package/dist/chunk-7KK2LZLZ.mjs +208 -0
- package/dist/chunk-7KK2LZLZ.mjs.map +1 -0
- package/dist/chunk-AW2JZIHR.mjs +753 -0
- package/dist/chunk-AW2JZIHR.mjs.map +1 -0
- package/dist/chunk-EIJOSZXZ.js +331 -0
- package/dist/chunk-EIJOSZXZ.js.map +1 -0
- package/dist/chunk-EQHR7P7G.js +541 -0
- package/dist/chunk-EQHR7P7G.js.map +1 -0
- package/dist/chunk-EWRXLZO4.mjs +539 -0
- package/dist/chunk-EWRXLZO4.mjs.map +1 -0
- package/dist/chunk-FISNQZZ7.js +802 -0
- package/dist/chunk-FISNQZZ7.js.map +1 -0
- package/dist/chunk-HL4WDMGS.js +200 -0
- package/dist/chunk-HL4WDMGS.js.map +1 -0
- package/dist/chunk-IPYWR76I.js +314 -0
- package/dist/chunk-IPYWR76I.js.map +1 -0
- package/dist/chunk-JYYNWR5G.js +142 -0
- package/dist/chunk-JYYNWR5G.js.map +1 -0
- package/dist/chunk-LNKTYZJM.js +701 -0
- package/dist/chunk-LNKTYZJM.js.map +1 -0
- package/dist/chunk-LVZMONQL.mjs +699 -0
- package/dist/chunk-LVZMONQL.mjs.map +1 -0
- package/dist/chunk-MFXLQWOE.js +93 -0
- package/dist/chunk-MFXLQWOE.js.map +1 -0
- package/dist/chunk-NBIA4TTE.mjs +204 -0
- package/dist/chunk-NBIA4TTE.mjs.map +1 -0
- package/dist/chunk-O4DD2XJ2.js +206 -0
- package/dist/chunk-O4DD2XJ2.js.map +1 -0
- package/dist/chunk-P7HVBU2B.mjs +140 -0
- package/dist/chunk-P7HVBU2B.mjs.map +1 -0
- package/dist/chunk-Q7QVQYEB.js +210 -0
- package/dist/chunk-Q7QVQYEB.js.map +1 -0
- package/dist/chunk-RLZEG6KL.mjs +329 -0
- package/dist/chunk-RLZEG6KL.mjs.map +1 -0
- package/dist/chunk-SYLDBJ75.mjs +246 -0
- package/dist/chunk-SYLDBJ75.mjs.map +1 -0
- package/dist/chunk-TTEUU3CI.mjs +198 -0
- package/dist/chunk-TTEUU3CI.mjs.map +1 -0
- package/dist/chunk-V66BXDTR.mjs +292 -0
- package/dist/chunk-V66BXDTR.mjs.map +1 -0
- package/dist/chunk-X77ZT4OI.js +268 -0
- package/dist/chunk-X77ZT4OI.js.map +1 -0
- package/dist/imtoken/index.d.mts +116 -0
- package/dist/imtoken/index.d.ts +116 -0
- package/dist/imtoken/index.js +14 -0
- package/dist/imtoken/index.js.map +1 -0
- package/dist/imtoken/index.mjs +5 -0
- package/dist/imtoken/index.mjs.map +1 -0
- package/dist/index.d.mts +14 -0
- package/dist/index.d.ts +14 -0
- package/dist/index.js +170 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +13 -0
- package/dist/index.mjs.map +1 -0
- package/dist/ledger/index.d.mts +290 -0
- package/dist/ledger/index.d.ts +290 -0
- package/dist/ledger/index.js +14 -0
- package/dist/ledger/index.js.map +1 -0
- package/dist/ledger/index.mjs +5 -0
- package/dist/ledger/index.mjs.map +1 -0
- package/dist/okx/index.d.mts +88 -0
- package/dist/okx/index.d.ts +88 -0
- package/dist/okx/index.js +13 -0
- package/dist/okx/index.js.map +1 -0
- package/dist/okx/index.mjs +4 -0
- package/dist/okx/index.mjs.map +1 -0
- package/dist/phantom/index.d.mts +96 -0
- package/dist/phantom/index.d.ts +96 -0
- package/dist/phantom/index.js +14 -0
- package/dist/phantom/index.js.map +1 -0
- package/dist/phantom/index.mjs +5 -0
- package/dist/phantom/index.mjs.map +1 -0
- package/dist/psbt-builder-CFOs69Z5.d.mts +131 -0
- package/dist/psbt-builder-CFOs69Z5.d.ts +131 -0
- package/dist/trezor/index.d.mts +155 -0
- package/dist/trezor/index.d.ts +155 -0
- package/dist/trezor/index.js +14 -0
- package/dist/trezor/index.js.map +1 -0
- package/dist/trezor/index.mjs +5 -0
- package/dist/trezor/index.mjs.map +1 -0
- package/dist/unisat/index.d.mts +75 -0
- package/dist/unisat/index.d.ts +75 -0
- package/dist/unisat/index.js +13 -0
- package/dist/unisat/index.js.map +1 -0
- package/dist/unisat/index.mjs +4 -0
- package/dist/unisat/index.mjs.map +1 -0
- package/dist/utils/index.d.mts +398 -0
- package/dist/utils/index.d.ts +398 -0
- package/dist/utils/index.js +120 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/index.mjs +3 -0
- package/dist/utils/index.mjs.map +1 -0
- package/dist/xverse/index.d.mts +79 -0
- package/dist/xverse/index.d.ts +79 -0
- package/dist/xverse/index.js +13 -0
- package/dist/xverse/index.js.map +1 -0
- package/dist/xverse/index.mjs +4 -0
- package/dist/xverse/index.mjs.map +1 -0
- package/package.json +108 -0
- package/src/base.ts +132 -0
- package/src/binance/BinanceConnector.ts +307 -0
- package/src/binance/index.ts +1 -0
- package/src/bitget/BitgetConnector.ts +301 -0
- package/src/bitget/index.ts +1 -0
- package/src/imtoken/ImTokenConnector.ts +420 -0
- package/src/imtoken/index.ts +2 -0
- package/src/index.ts +78 -0
- package/src/ledger/LedgerConnector.ts +1019 -0
- package/src/ledger/index.ts +8 -0
- package/src/okx/OKXConnector.ts +230 -0
- package/src/okx/index.ts +1 -0
- package/src/phantom/PhantomConnector.ts +381 -0
- package/src/phantom/index.ts +2 -0
- package/src/trezor/TrezorConnector.ts +824 -0
- package/src/trezor/index.ts +6 -0
- package/src/unisat/UnisatConnector.ts +312 -0
- package/src/unisat/index.ts +1 -0
- package/src/utils/blockstream.ts +230 -0
- package/src/utils/btc-service.ts +364 -0
- package/src/utils/index.ts +56 -0
- package/src/utils/mempool.ts +232 -0
- package/src/utils/psbt-builder.ts +492 -0
- package/src/utils/types.ts +183 -0
- package/src/xverse/XverseConnector.ts +479 -0
- package/src/xverse/index.ts +1 -0
|
@@ -0,0 +1,301 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
WalletAccount,
|
|
3
|
+
BitcoinNetwork,
|
|
4
|
+
SignPsbtOptions,
|
|
5
|
+
} from 'otx-btc-wallet-core';
|
|
6
|
+
import { BaseConnector } from '../base';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Bitget sign PSBT options
|
|
10
|
+
* @see https://web3.bitget.com/en/docs/connect/mainnet/btc/
|
|
11
|
+
*/
|
|
12
|
+
interface BitgetSignPsbtOptions {
|
|
13
|
+
autoFinalized?: boolean;
|
|
14
|
+
toSignInputs?: Array<{
|
|
15
|
+
index: number;
|
|
16
|
+
address?: string;
|
|
17
|
+
publicKey?: string;
|
|
18
|
+
sighashTypes?: number[];
|
|
19
|
+
disableTweakSigner?: boolean;
|
|
20
|
+
}>;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Bitget Bitcoin provider interface
|
|
25
|
+
* Provider is accessible via window.bitkeep.unisat
|
|
26
|
+
* @see https://web3.bitget.com/en/docs/connect/mainnet/btc/
|
|
27
|
+
*/
|
|
28
|
+
interface BitgetBitcoinProvider {
|
|
29
|
+
requestAccounts(): Promise<string[]>;
|
|
30
|
+
getAccounts(): Promise<string[]>;
|
|
31
|
+
getPublicKey(): Promise<string>;
|
|
32
|
+
getBalance(): Promise<{ confirmed: number; unconfirmed: number; total: number }>;
|
|
33
|
+
getNetwork(): Promise<string>;
|
|
34
|
+
switchNetwork(network: 'livenet' | 'testnet' | 'signet'): Promise<void>;
|
|
35
|
+
signMessage(msg: string, type?: 'ecdsa' | 'bip322-simple'): Promise<string>;
|
|
36
|
+
signPsbt(psbtHex: string, options?: BitgetSignPsbtOptions): Promise<string>;
|
|
37
|
+
signPsbts(psbtHexs: string[], options?: BitgetSignPsbtOptions[]): Promise<string[]>;
|
|
38
|
+
pushPsbt(psbtHex: string): Promise<string>;
|
|
39
|
+
sendBitcoin(toAddress: string, satoshis: number, options?: { feeRate?: number }): Promise<string>;
|
|
40
|
+
pushTx(options: { rawtx: string }): Promise<string>;
|
|
41
|
+
on(event: 'accountsChanged', callback: (accounts: string[]) => void): void;
|
|
42
|
+
on(event: 'networkChanged', callback: (network: string) => void): void;
|
|
43
|
+
removeListener(event: 'accountsChanged', callback: (accounts: string[]) => void): void;
|
|
44
|
+
removeListener(event: 'networkChanged', callback: (network: string) => void): void;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// Extend window type
|
|
48
|
+
declare global {
|
|
49
|
+
interface Window {
|
|
50
|
+
bitkeep?: {
|
|
51
|
+
unisat?: BitgetBitcoinProvider;
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Bitget wallet icon
|
|
57
|
+
const BITGET_ICON =
|
|
58
|
+
'';
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Bitget network type
|
|
62
|
+
*/
|
|
63
|
+
type BitgetNetwork = 'livenet' | 'testnet' | 'signet';
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Bitget Wallet Connector
|
|
67
|
+
*
|
|
68
|
+
* @see https://web3.bitget.com/en/docs/connect/mainnet/btc/
|
|
69
|
+
*/
|
|
70
|
+
export class BitgetConnector extends BaseConnector {
|
|
71
|
+
readonly id = 'bitget';
|
|
72
|
+
readonly name = 'Bitget Wallet';
|
|
73
|
+
readonly icon = BITGET_ICON;
|
|
74
|
+
|
|
75
|
+
private _removeAccountChangeListener: (() => void) | null = null;
|
|
76
|
+
private _removeNetworkChangeListener: (() => void) | null = null;
|
|
77
|
+
|
|
78
|
+
protected getProvider(): BitgetBitcoinProvider | undefined {
|
|
79
|
+
if (typeof window === 'undefined') return undefined;
|
|
80
|
+
return window.bitkeep?.unisat;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
async connect(network: BitcoinNetwork = 'mainnet'): Promise<WalletAccount> {
|
|
84
|
+
this.ensureInstalled();
|
|
85
|
+
const provider = this.getProvider()!;
|
|
86
|
+
|
|
87
|
+
try {
|
|
88
|
+
const accounts = await provider.requestAccounts();
|
|
89
|
+
if (!accounts || accounts.length === 0) {
|
|
90
|
+
throw new Error('No accounts found');
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// Switch network if needed
|
|
94
|
+
await this.ensureNetwork(provider, network);
|
|
95
|
+
|
|
96
|
+
const publicKey = await provider.getPublicKey();
|
|
97
|
+
|
|
98
|
+
// Setup event listeners
|
|
99
|
+
this.setupEventListeners();
|
|
100
|
+
|
|
101
|
+
return {
|
|
102
|
+
address: accounts[0] ?? '',
|
|
103
|
+
publicKey,
|
|
104
|
+
type: this.inferAddressType(accounts[0] ?? ''),
|
|
105
|
+
};
|
|
106
|
+
} catch (error) {
|
|
107
|
+
this.handleError(error);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
private async ensureNetwork(provider: BitgetBitcoinProvider, network: BitcoinNetwork): Promise<void> {
|
|
112
|
+
const currentNetwork = await provider.getNetwork();
|
|
113
|
+
const targetNetwork = this.mapToBitgetNetwork(network);
|
|
114
|
+
|
|
115
|
+
if (currentNetwork !== targetNetwork) {
|
|
116
|
+
await provider.switchNetwork(targetNetwork);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
private setupEventListeners(): void {
|
|
121
|
+
const provider = this.getProvider();
|
|
122
|
+
if (!provider) return;
|
|
123
|
+
|
|
124
|
+
// Remove existing listeners
|
|
125
|
+
this.removeEventListeners();
|
|
126
|
+
|
|
127
|
+
// Account change listener
|
|
128
|
+
const handleAccountsChanged = (accounts: string[]) => {
|
|
129
|
+
if (!accounts || accounts.length === 0) {
|
|
130
|
+
this.emitAccountsChanged([]);
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
provider.getPublicKey().then((publicKey) => {
|
|
135
|
+
const walletAccounts: WalletAccount[] = accounts.map((address) => ({
|
|
136
|
+
address,
|
|
137
|
+
publicKey,
|
|
138
|
+
type: this.inferAddressType(address),
|
|
139
|
+
}));
|
|
140
|
+
this.emitAccountsChanged(walletAccounts);
|
|
141
|
+
}).catch(() => {
|
|
142
|
+
this.emitAccountsChanged([]);
|
|
143
|
+
});
|
|
144
|
+
};
|
|
145
|
+
|
|
146
|
+
// Network change listener
|
|
147
|
+
const handleNetworkChanged = (network: string) => {
|
|
148
|
+
const btcNetwork = this.mapNetwork(network);
|
|
149
|
+
this.emitNetworkChanged(btcNetwork);
|
|
150
|
+
};
|
|
151
|
+
|
|
152
|
+
provider.on('accountsChanged', handleAccountsChanged);
|
|
153
|
+
provider.on('networkChanged', handleNetworkChanged);
|
|
154
|
+
|
|
155
|
+
this._removeAccountChangeListener = () => {
|
|
156
|
+
provider.removeListener('accountsChanged', handleAccountsChanged);
|
|
157
|
+
};
|
|
158
|
+
this._removeNetworkChangeListener = () => {
|
|
159
|
+
provider.removeListener('networkChanged', handleNetworkChanged);
|
|
160
|
+
};
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
private removeEventListeners(): void {
|
|
164
|
+
this._removeAccountChangeListener?.();
|
|
165
|
+
this._removeAccountChangeListener = null;
|
|
166
|
+
|
|
167
|
+
this._removeNetworkChangeListener?.();
|
|
168
|
+
this._removeNetworkChangeListener = null;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
async disconnect(): Promise<void> {
|
|
172
|
+
this.removeEventListeners();
|
|
173
|
+
this.cleanup();
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
async getAccounts(): Promise<WalletAccount[]> {
|
|
177
|
+
this.ensureInstalled();
|
|
178
|
+
const provider = this.getProvider()!;
|
|
179
|
+
|
|
180
|
+
try {
|
|
181
|
+
const addresses = await provider.getAccounts();
|
|
182
|
+
const publicKey = await provider.getPublicKey();
|
|
183
|
+
|
|
184
|
+
return addresses.map((address) => ({
|
|
185
|
+
address,
|
|
186
|
+
publicKey,
|
|
187
|
+
type: this.inferAddressType(address),
|
|
188
|
+
}));
|
|
189
|
+
} catch (error) {
|
|
190
|
+
this.handleError(error);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
async signMessage(message: string): Promise<string> {
|
|
195
|
+
this.ensureInstalled();
|
|
196
|
+
const provider = this.getProvider()!;
|
|
197
|
+
|
|
198
|
+
try {
|
|
199
|
+
return await provider.signMessage(message);
|
|
200
|
+
} catch (error) {
|
|
201
|
+
this.handleError(error);
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
async signPsbt(psbtHex: string, options?: SignPsbtOptions): Promise<string> {
|
|
206
|
+
this.ensureInstalled();
|
|
207
|
+
const provider = this.getProvider()!;
|
|
208
|
+
|
|
209
|
+
try {
|
|
210
|
+
// Bitget uses same API as Unisat - pass options directly if provided
|
|
211
|
+
return await provider.signPsbt(psbtHex, options);
|
|
212
|
+
} catch (error) {
|
|
213
|
+
this.handleError(error);
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
async signPsbts(psbtHexs: string[], options?: SignPsbtOptions): Promise<string[]> {
|
|
218
|
+
this.ensureInstalled();
|
|
219
|
+
const provider = this.getProvider()!;
|
|
220
|
+
|
|
221
|
+
try {
|
|
222
|
+
const bitgetOptions: BitgetSignPsbtOptions = {
|
|
223
|
+
autoFinalized: options?.autoFinalize ?? true,
|
|
224
|
+
};
|
|
225
|
+
|
|
226
|
+
if (options?.toSignInputs) {
|
|
227
|
+
bitgetOptions.toSignInputs = options.toSignInputs;
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
// Create options array for each PSBT
|
|
231
|
+
const optionsArray = psbtHexs.map(() => bitgetOptions);
|
|
232
|
+
|
|
233
|
+
return await provider.signPsbts(psbtHexs, optionsArray);
|
|
234
|
+
} catch (error) {
|
|
235
|
+
this.handleError(error);
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
async sendTransaction(to: string, satoshis: number): Promise<string> {
|
|
240
|
+
this.ensureInstalled();
|
|
241
|
+
const provider = this.getProvider()!;
|
|
242
|
+
|
|
243
|
+
try {
|
|
244
|
+
return await provider.sendBitcoin(to, satoshis);
|
|
245
|
+
} catch (error) {
|
|
246
|
+
this.handleError(error);
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
async getNetwork(): Promise<BitcoinNetwork> {
|
|
251
|
+
this.ensureInstalled();
|
|
252
|
+
const provider = this.getProvider()!;
|
|
253
|
+
|
|
254
|
+
try {
|
|
255
|
+
const network = await provider.getNetwork();
|
|
256
|
+
return this.mapNetwork(network);
|
|
257
|
+
} catch (error) {
|
|
258
|
+
this.handleError(error);
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
async switchNetwork(network: BitcoinNetwork): Promise<void> {
|
|
263
|
+
this.ensureInstalled();
|
|
264
|
+
const provider = this.getProvider()!;
|
|
265
|
+
|
|
266
|
+
try {
|
|
267
|
+
const bitgetNetwork = this.mapToBitgetNetwork(network);
|
|
268
|
+
await provider.switchNetwork(bitgetNetwork);
|
|
269
|
+
} catch (error) {
|
|
270
|
+
this.handleError(error);
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
private mapNetwork(network: string): BitcoinNetwork {
|
|
275
|
+
switch (network.toLowerCase()) {
|
|
276
|
+
case 'livenet':
|
|
277
|
+
case 'mainnet':
|
|
278
|
+
return 'mainnet';
|
|
279
|
+
case 'testnet':
|
|
280
|
+
return 'testnet';
|
|
281
|
+
case 'signet':
|
|
282
|
+
return 'signet';
|
|
283
|
+
default:
|
|
284
|
+
return 'mainnet';
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
private mapToBitgetNetwork(network: BitcoinNetwork): BitgetNetwork {
|
|
289
|
+
switch (network) {
|
|
290
|
+
case 'mainnet':
|
|
291
|
+
return 'livenet';
|
|
292
|
+
case 'testnet':
|
|
293
|
+
case 'testnet4':
|
|
294
|
+
return 'testnet';
|
|
295
|
+
case 'signet':
|
|
296
|
+
return 'signet';
|
|
297
|
+
default:
|
|
298
|
+
return 'livenet';
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { BitgetConnector } from './BitgetConnector';
|