Skip to content

Commit 618e36f

Browse files
authored
Merge pull request #530 from fmorency/fix-ledger
fix: ledger wallet
2 parents 4c801e6 + 2873855 commit 618e36f

File tree

8 files changed

+94
-30
lines changed

8 files changed

+94
-30
lines changed

wallets/ledger/package.json

+7-5
Original file line numberDiff line numberDiff line change
@@ -66,12 +66,14 @@
6666
},
6767
"dependencies": {
6868
"@cosmos-kit/core": "^2.15.1",
69-
"@ledgerhq/hw-app-cosmos": "^6.28.1",
70-
"@ledgerhq/hw-transport-webhid": "^6.27.15",
71-
"@ledgerhq/hw-transport-webusb": "^6.27.15"
69+
"@ledgerhq/hw-app-cosmos": "^6.30.4",
70+
"@ledgerhq/hw-transport-webhid": "^6.30.0",
71+
"@ledgerhq/hw-transport-webusb": "^6.29.4"
7272
},
7373
"peerDependencies": {
74-
"@cosmjs/amino": ">=0.32.3",
75-
"@cosmjs/proto-signing": ">=0.32.3"
74+
"@cosmjs/amino": ">=0.32.4",
75+
"@cosmjs/crypto": ">=0.32.4",
76+
"@cosmjs/encoding": ">=0.32.4",
77+
"@cosmjs/proto-signing": ">=0.32.4"
7678
}
7779
}

wallets/ledger/src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
1+
export * from './constant';
12
export * from './ledger';
23
export * from './web-usb-hid/registry';
3-
export * from './constant';
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import { ChainRecord, ChainWalletBase, Wallet } from '@cosmos-kit/core';
22

3-
export class LedgerChianWallet extends ChainWalletBase {
3+
export class LedgerChainWallet extends ChainWalletBase {
44
constructor(walletInfo: Wallet, chainInfo: ChainRecord) {
55
super(walletInfo, chainInfo);
66
}
7-
}
7+
}

wallets/ledger/src/web-usb-hid/client.ts

+40-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,17 @@
1-
import { StdSignDoc } from '@cosmjs/amino';
1+
import {
2+
encodeSecp256k1Signature,
3+
OfflineAminoSigner,
4+
StdSignDoc,
5+
} from '@cosmjs/amino';
6+
import { sortedJsonStringify } from '@cosmjs/amino/build/signdoc';
7+
import { Secp256k1Signature } from '@cosmjs/crypto';
8+
import { fromHex } from '@cosmjs/encoding';
29
import { Algo } from '@cosmjs/proto-signing';
3-
import { WalletClient } from '@cosmos-kit/core';
10+
import { SignType, WalletClient } from '@cosmos-kit/core';
411
import Cosmos from '@ledgerhq/hw-app-cosmos';
512

613
import { ChainIdToBech32Prefix, getCosmosApp, getCosmosPath } from './utils';
14+
715
export class LedgerClient implements WalletClient {
816
client: Cosmos;
917

@@ -39,16 +47,44 @@ export class LedgerClient implements WalletClient {
3947
username: username ?? path,
4048
address,
4149
algo: 'secp256k1' as Algo,
42-
pubkey: new TextEncoder().encode(publicKey),
50+
pubkey: fromHex(publicKey),
4351
isNanoLedger: true,
4452
};
4553
}
4654

55+
getOfflineSigner(chainId: string, preferredSignType?: SignType) {
56+
// Ledger doesn't support direct sign, only Amino sign
57+
if (preferredSignType === 'direct') {
58+
throw new Error('Unsupported sign type: direct');
59+
}
60+
return this.getOfflineSignerAmino(chainId);
61+
}
62+
63+
getOfflineSignerAmino(chainId: string): OfflineAminoSigner {
64+
return {
65+
getAccounts: async () => {
66+
return [await this.getAccount(chainId)];
67+
},
68+
signAmino: async (_signerAddress, signDoc) => {
69+
const { pubkey } = await this.getAccount(chainId);
70+
const { signature: derSignature } = await this.sign(signDoc); // The signature is in DER format
71+
const signature = Secp256k1Signature.fromDer(derSignature); // Convert the DER signature to fixed length (64 bytes)
72+
return {
73+
signed: signDoc,
74+
signature: encodeSecp256k1Signature(
75+
pubkey,
76+
signature.toFixedLength()
77+
),
78+
};
79+
},
80+
};
81+
}
82+
4783
async sign(signDoc: StdSignDoc, accountIndex = 0) {
4884
if (!this.client) await this.initClient();
4985
return await this.client.sign(
5086
getCosmosPath(accountIndex),
51-
JSON.stringify(signDoc)
87+
sortedJsonStringify(signDoc) // signDoc MUST be serialized in lexicographical key order
5288
);
5389
}
5490
}

wallets/ledger/src/web-usb-hid/main-wallet.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
1-
import { EndpointOptions, Wallet } from '@cosmos-kit/core';
2-
import { MainWalletBase } from '@cosmos-kit/core';
3-
import { LedgerChianWallet } from './chain-wallet';
1+
import { EndpointOptions, MainWalletBase, Wallet } from '@cosmos-kit/core';
2+
3+
import { LedgerChainWallet } from './chain-wallet';
44
import { LedgerClient } from './client';
55
import { TransportType } from './utils';
66

@@ -11,7 +11,7 @@ export class LedgerMainWallet extends MainWalletBase {
1111
preferredEndpoints?: EndpointOptions['endpoints'],
1212
transportType: TransportType = 'WebUSB'
1313
) {
14-
super(walletInfo, LedgerChianWallet);
14+
super(walletInfo, LedgerChainWallet);
1515
this.preferredEndpoints = preferredEndpoints;
1616
this.transportType = transportType;
1717
}

wallets/ledger/src/web-usb-hid/registry.ts

+1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import { Wallet } from '@cosmos-kit/core';
2+
23
import { ICON } from '../constant';
34

45
export const LedgerInfo: Wallet = {
+11-11
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,25 @@
1-
import { chains } from 'chain-registry'
2-
import Cosmos from "@ledgerhq/hw-app-cosmos";
1+
import Cosmos from '@ledgerhq/hw-app-cosmos';
2+
import TransportWebHID from '@ledgerhq/hw-transport-webhid';
33
import TransportWebUSB from '@ledgerhq/hw-transport-webusb';
4-
import TransportWebHID from '@ledgerhq/hw-transport-webhid'
4+
import { chains } from 'chain-registry';
55

6-
export type TransportType = 'WebUSB' | 'WebHID'
6+
export type TransportType = 'WebUSB' | 'WebHID';
77

88
export async function getCosmosApp(type: TransportType = 'WebUSB') {
99
if (type === 'WebUSB') {
10-
return new Cosmos(await TransportWebUSB.create())
10+
return new Cosmos(await TransportWebUSB.create());
1111
}
1212
if (type === 'WebHID') {
13-
return new Cosmos(await TransportWebHID.create())
13+
return new Cosmos(await TransportWebHID.create());
1414
}
15-
throw new Error(`Unknown transport type: ${type}`)
15+
throw new Error(`Unknown transport type: ${type}`);
1616
}
1717

1818
export function getCosmosPath(accountIndex = 0) {
19-
return `44'/118'/${accountIndex}'/0/0`
19+
return `44'/118'/${accountIndex}'/0/0`;
2020
}
2121

22-
export const ChainIdToBech32Prefix = {} as { [k: string]: string }
22+
export const ChainIdToBech32Prefix = {} as { [k: string]: string };
2323
for (const chain of chains) {
24-
ChainIdToBech32Prefix[chain.chain_id] = chain.bech32_prefix
25-
}
24+
ChainIdToBech32Prefix[chain.chain_id] = chain.bech32_prefix;
25+
}

yarn.lock

+28-3
Original file line numberDiff line numberDiff line change
@@ -18024,7 +18024,16 @@ string-length@^4.0.1:
1802418024
char-regex "^1.0.2"
1802518025
strip-ansi "^6.0.0"
1802618026

18027-
"string-width-cjs@npm:string-width@^4.2.0", "string-width@^1.0.2 || 2 || 3 || 4", string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3:
18027+
"string-width-cjs@npm:string-width@^4.2.0":
18028+
version "4.2.3"
18029+
resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010"
18030+
integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==
18031+
dependencies:
18032+
emoji-regex "^8.0.0"
18033+
is-fullwidth-code-point "^3.0.0"
18034+
strip-ansi "^6.0.1"
18035+
18036+
"string-width@^1.0.2 || 2 || 3 || 4", string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3:
1802818037
version "4.2.3"
1802918038
resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010"
1803018039
integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==
@@ -18140,7 +18149,7 @@ stringify-entities@^4.0.0:
1814018149
character-entities-html4 "^2.0.0"
1814118150
character-entities-legacy "^3.0.0"
1814218151

18143-
"strip-ansi-cjs@npm:strip-ansi@^6.0.1", strip-ansi@^6.0.0, strip-ansi@^6.0.1:
18152+
"strip-ansi-cjs@npm:strip-ansi@^6.0.1":
1814418153
version "6.0.1"
1814518154
resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9"
1814618155
integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==
@@ -18168,6 +18177,13 @@ strip-ansi@^5.1.0:
1816818177
dependencies:
1816918178
ansi-regex "^4.1.0"
1817018179

18180+
strip-ansi@^6.0.0, strip-ansi@^6.0.1:
18181+
version "6.0.1"
18182+
resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9"
18183+
integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==
18184+
dependencies:
18185+
ansi-regex "^5.0.1"
18186+
1817118187
strip-ansi@^7.0.1:
1817218188
version "7.1.0"
1817318189
resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-7.1.0.tgz#d5b6568ca689d8561370b0707685d22434faff45"
@@ -19723,7 +19739,7 @@ wordwrap@^1.0.0:
1972319739
resolved "https://registry.yarnpkg.com/wordwrap/-/wordwrap-1.0.0.tgz#27584810891456a4171c8d0226441ade90cbcaeb"
1972419740
integrity sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==
1972519741

19726-
"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0", wrap-ansi@^7.0.0:
19742+
"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0":
1972719743
version "7.0.0"
1972819744
resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43"
1972919745
integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==
@@ -19741,6 +19757,15 @@ wrap-ansi@^6.0.1, wrap-ansi@^6.2.0:
1974119757
string-width "^4.1.0"
1974219758
strip-ansi "^6.0.0"
1974319759

19760+
wrap-ansi@^7.0.0:
19761+
version "7.0.0"
19762+
resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43"
19763+
integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==
19764+
dependencies:
19765+
ansi-styles "^4.0.0"
19766+
string-width "^4.1.0"
19767+
strip-ansi "^6.0.0"
19768+
1974419769
wrap-ansi@^8.1.0:
1974519770
version "8.1.0"
1974619771
resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-8.1.0.tgz#56dc22368ee570face1b49819975d9b9a5ead214"

0 commit comments

Comments
 (0)