Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add cip-1854 support to keyagent #1301

Closed
wants to merge 15 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import { validateFuzzyOptions, withTextFilter } from '../../../src/StakePool/Typ
describe('TypeormStakePoolProvider utils', () => {
describe('validateFuzzyOptions', () => {
it('throws if value is not a valid JSON encoded string', () =>
expect(() => validateFuzzyOptions('test')).toThrow('Unexpected token e in JSON at position 1'));
expect(() => validateFuzzyOptions('test')).toThrow(/Unexpected token/));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is an odd change for this PR, was this test flaky?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes it was. It fails locally as it receives this error message "Unexpected token 'e', "test" is not valid JSON". Maybe updating the expected error message to match the received message would work better than regex? wdyt?

it('throws if value is not an object', () =>
expect(() => validateFuzzyOptions('"test"')).toThrow('must be an object'));
it('throws without threshold', () =>
Expand Down
7 changes: 5 additions & 2 deletions packages/e2e/src/factories.ts
Original file line number Diff line number Diff line change
Expand Up @@ -204,7 +204,8 @@ keyManagementFactory.register('inMemory', async (params: any): Promise<CreateKey
accountIndex: params.accountIndex,
chainId: params.chainId,
getPassphrase: async () => Buffer.from(params.passphrase),
mnemonicWords
mnemonicWords,
purpose: params.purpose
},
dependencies
)
Expand All @@ -220,7 +221,8 @@ keyManagementFactory.register('ledger', async (params: any): Promise<CreateKeyAg
accountIndex: params.accountIndex,
chainId: params.chainId,
communicationType: CommunicationType.Node,
deviceConnection
deviceConnection,
purpose: params.purpose
},
dependencies
);
Expand All @@ -238,6 +240,7 @@ keyManagementFactory.register(
{
accountIndex: params.accountIndex,
chainId: params.chainId,
purpose: params.purpose,
trezorConfig: {
communicationType: CommunicationType.Node,
manifest: {
Expand Down
5 changes: 3 additions & 2 deletions packages/e2e/src/scripts/mnemonic.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
/* eslint-disable no-console */
/* eslint-disable @typescript-eslint/no-floating-promises */
import * as Crypto from '@cardano-sdk/crypto';
import { AddressType, InMemoryKeyAgent, util } from '@cardano-sdk/key-management';
import { AddressType, InMemoryKeyAgent, KeyPurpose, util } from '@cardano-sdk/key-management';
import { localNetworkChainId } from '../util';

/** Generates a new set of Mnemonic words and prints them to the console. */
Expand All @@ -14,7 +14,8 @@ import { localNetworkChainId } from '../util';
{
chainId: localNetworkChainId,
getPassphrase: async () => Buffer.from(''),
mnemonicWords: mnemonicArray
mnemonicWords: mnemonicArray,
purpose: KeyPurpose.STANDARD
},
{
bip32Ed25519: new Crypto.SodiumBip32Ed25519(),
Expand Down
8 changes: 6 additions & 2 deletions packages/e2e/src/util/createMockKeyAgent.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { Bip32PublicKeyHex, SodiumBip32Ed25519 } from '@cardano-sdk/crypto';
import { Cardano } from '@cardano-sdk/core';
import { GroupedAddress, KeyAgent, KeyAgentType } from '@cardano-sdk/key-management';
import { GroupedAddress, KeyAgent, KeyAgentType, KeyPurpose } from '@cardano-sdk/key-management';

const accountIndex = 0;
const chainId = Cardano.ChainIds.Preview;
Expand All @@ -18,12 +18,16 @@ export const createMockKeyAgent = (deriveAddressesReturn: GroupedAddress[] = [])
derivePublicKey: jest.fn(),
exportRootPrivateKey: jest.fn(),
extendedAccountPublicKey,
get purpose(): KeyPurpose {
return KeyPurpose.STANDARD;
},
serializableData: {
__typename: KeyAgentType.InMemory,
accountIndex,
chainId,
encryptedRootPrivateKeyBytes: [],
extendedAccountPublicKey
extendedAccountPublicKey,
purpose: KeyPurpose.STANDARD
},
signBlob: jest.fn(),
signTransaction: jest.fn()
Expand Down
9 changes: 6 additions & 3 deletions packages/e2e/src/util/util.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ import {
timeout
} from 'rxjs';
import { FAST_OPERATION_TIMEOUT_DEFAULT, SYNC_TIMEOUT_DEFAULT } from '../defaults';
import { InMemoryKeyAgent, TransactionSigner } from '@cardano-sdk/key-management';
import { InMemoryKeyAgent, KeyPurpose, TransactionSigner } from '@cardano-sdk/key-management';
import { InitializeTxProps } from '@cardano-sdk/tx-construction';
import { TestWallet, networkInfoProviderFactory } from '../factories';
import { getEnv, walletVariables } from '../environment';
Expand Down Expand Up @@ -228,17 +228,20 @@ export const submitCertificate = async (certificate: Cardano.Certificate, wallet
* @param mnemonics The random set of mnemonics.
* @param genesis Network genesis parameters
* @param bip32Ed25519 The Ed25519 cryptography implementation.
* @param purpose they key derivation purpose of either 1852 or 1854
*/
export const createStandaloneKeyAgent = async (
mnemonics: string[],
genesis: Cardano.CompactGenesis,
bip32Ed25519: Crypto.Bip32Ed25519
bip32Ed25519: Crypto.Bip32Ed25519,
purpose: KeyPurpose
) =>
await InMemoryKeyAgent.fromBip39MnemonicWords(
{
chainId: genesis,
getPassphrase: async () => Buffer.from(''),
mnemonicWords: mnemonics
mnemonicWords: mnemonics,
purpose
},
{ bip32Ed25519, logger }
);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import {
AddressType,
GroupedAddress,
InMemoryKeyAgent,
KeyPurpose,
KeyRole
} from '@cardano-sdk/key-management';
import {
Expand Down Expand Up @@ -215,7 +216,7 @@ export class MultiSigWallet {
body: multiSigTx.getTransaction().body,
hash: multiSigTx.getTransaction().id
},
{ knownAddresses: [this.#address], txInKeyPathMap: {} },
{ knownAddresses: [this.#address], purpose: KeyPurpose.MULTI_SIG, txInKeyPathMap: {} },
{ additionalKeyPaths: [DERIVATION_PATH] }
);

Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import * as Crypto from '@cardano-sdk/crypto';
import { BaseWallet } from '@cardano-sdk/wallet';
import { Cardano, EraSummary, StakePoolProvider, createSlotEpochCalc } from '@cardano-sdk/core';
import { InMemoryKeyAgent, KeyRole } from '@cardano-sdk/key-management';
import { InMemoryKeyAgent, KeyPurpose, KeyRole } from '@cardano-sdk/key-management';
import { MultiSigTx } from './MultiSigTx';
import { MultiSigWallet } from './MultiSigWallet';
import { Observable, filter, firstValueFrom, map, take } from 'rxjs';
Expand Down Expand Up @@ -63,7 +63,7 @@ const fundMultiSigWallet = async (sendingWallet: BaseWallet, address: Cardano.Pa
const getKeyAgent = async (mnemonics: string, faucetWallet: BaseWallet, bip32Ed25519: Crypto.Bip32Ed25519) => {
const genesis = await firstValueFrom(faucetWallet.genesisParameters$);

const keyAgent = await createStandaloneKeyAgent(mnemonics.split(' '), genesis, bip32Ed25519);
const keyAgent = await createStandaloneKeyAgent(mnemonics.split(' '), genesis, bip32Ed25519, KeyPurpose.STANDARD);

const pubKey = await keyAgent.derivePublicKey(DERIVATION_PATH);

Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
/* eslint-disable sonarjs/no-duplicate-string */
import { BaseWallet } from '@cardano-sdk/wallet';
import { Cardano, metadatum } from '@cardano-sdk/core';
import { KeyAgent, TransactionSigner } from '@cardano-sdk/key-management';
import { KeyAgent, KeyPurpose, TransactionSigner } from '@cardano-sdk/key-management';
import {
bip32Ed25519Factory,
burnTokens,
Expand Down Expand Up @@ -50,7 +50,8 @@ describe('Ada handle', () => {
keyAgent = await createStandaloneKeyAgent(
env.KEY_MANAGEMENT_PARAMS.mnemonic.split(' '),
await firstValueFrom(wallet.genesisParameters$),
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger)
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger),
KeyPurpose.STANDARD
);
({ policyScript, policySigner, policyId } = await createHandlePolicy(keyAgent));
const handleProviderPolicyId = await getHandlePolicyId(
Expand Down
5 changes: 3 additions & 2 deletions packages/e2e/test/wallet_epoch_0/PersonalWallet/mint.test.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { BaseWallet, FinalizeTxProps } from '@cardano-sdk/wallet';
import { Cardano, nativeScriptPolicyId } from '@cardano-sdk/core';
import { InitializeTxProps } from '@cardano-sdk/tx-construction';
import { KeyRole, util } from '@cardano-sdk/key-management';
import { KeyPurpose, KeyRole, util } from '@cardano-sdk/key-management';
import {
bip32Ed25519Factory,
burnTokens,
Expand Down Expand Up @@ -40,7 +40,8 @@ describe('PersonalWallet/mint', () => {
const aliceKeyAgent = await createStandaloneKeyAgent(
env.KEY_MANAGEMENT_PARAMS.mnemonic.split(' '),
genesis,
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger)
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger),
KeyPurpose.STANDARD
);

const derivationPath = {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/* eslint-disable sonarjs/no-duplicate-string */
import { AddressType, GroupedAddress, util } from '@cardano-sdk/key-management';
import { AddressType, GroupedAddress, KeyPurpose, util } from '@cardano-sdk/key-management';
import { BaseWallet } from '@cardano-sdk/wallet';
import { Cardano } from '@cardano-sdk/core';
import {
Expand Down Expand Up @@ -43,7 +43,8 @@ describe('PersonalWallet/multiAddress', () => {
const multiAddressKeyAgent = await createStandaloneKeyAgent(
mnemonics,
genesis,
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger)
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger),
KeyPurpose.STANDARD
);

let txBuilder = wallet.createTxBuilder();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
import { BaseWallet, FinalizeTxProps } from '@cardano-sdk/wallet';
import { Cardano, nativeScriptPolicyId } from '@cardano-sdk/core';
import { InitializeTxProps } from '@cardano-sdk/tx-construction';
import { KeyRole, util } from '@cardano-sdk/key-management';
import { KeyPurpose, KeyRole, util } from '@cardano-sdk/key-management';
import {
bip32Ed25519Factory,
burnTokens,
Expand Down Expand Up @@ -48,12 +48,14 @@ describe('PersonalWallet/multisignature', () => {
const aliceKeyAgent = await createStandaloneKeyAgent(
env.KEY_MANAGEMENT_PARAMS.mnemonic.split(' '),
genesis,
bip32Ed25519
bip32Ed25519,
KeyPurpose.MULTI_SIG
);
const bobKeyAgent = await createStandaloneKeyAgent(
env.KEY_MANAGEMENT_PARAMS.mnemonic.split(' '),
genesis,
bip32Ed25519
bip32Ed25519,
KeyPurpose.MULTI_SIG
);

const aliceDerivationPath = {
Expand Down
5 changes: 3 additions & 2 deletions packages/e2e/test/wallet_epoch_0/PersonalWallet/nft.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
import { Asset, Cardano, metadatum, nativeScriptPolicyId } from '@cardano-sdk/core';
import { Assets, BaseWallet, FinalizeTxProps } from '@cardano-sdk/wallet';
import { InitializeTxProps } from '@cardano-sdk/tx-construction';
import { KeyRole, TransactionSigner, util } from '@cardano-sdk/key-management';
import { KeyPurpose, KeyRole, TransactionSigner, util } from '@cardano-sdk/key-management';
import {
bip32Ed25519Factory,
burnTokens,
Expand Down Expand Up @@ -61,7 +61,8 @@ describe('PersonalWallet.assets/nft', () => {
const keyAgent = await createStandaloneKeyAgent(
env.KEY_MANAGEMENT_PARAMS.mnemonic.split(' '),
genesis,
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger)
await bip32Ed25519Factory.create(env.KEY_MANAGEMENT_PARAMS.bip32Ed25519, null, logger),
KeyPurpose.STANDARD
);

const derivationPath = {
Expand Down
8 changes: 7 additions & 1 deletion packages/e2e/test/wallet_epoch_0/SharedWallet/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import * as Crypto from '@cardano-sdk/crypto';
import {
AccountKeyDerivationPath,
KeyAgent,
KeyPurpose,
KeyRole,
SignBlobResult,
SignDataContext,
Expand Down Expand Up @@ -30,7 +31,12 @@ const getKeyAgent = async (
genesisParameters: Cardano.CompactGenesis,
bip32Ed25519: Crypto.Bip32Ed25519
) => {
const keyAgent = await createStandaloneKeyAgent(mnemonics.split(' '), genesisParameters, bip32Ed25519);
const keyAgent = await createStandaloneKeyAgent(
mnemonics.split(' '),
genesisParameters,
bip32Ed25519,
KeyPurpose.STANDARD
);

const pubKey = await keyAgent.derivePublicKey(DERIVATION_PATH);

Expand Down
4 changes: 3 additions & 1 deletion packages/e2e/test/web-extension/extension/ui.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import { Cardano } from '@cardano-sdk/core';
import {
CommunicationType,
InMemoryKeyAgent,
KeyPurpose,
SerializableInMemoryKeyAgentData,
emip3encrypt
} from '@cardano-sdk/key-management';
Expand Down Expand Up @@ -304,7 +305,8 @@ const createWalletIfNotExistsAndActivate = async (accountIndex: number) => {
accountIndex,
chainId: env.KEY_MANAGEMENT_PARAMS.chainId,
getPassphrase: async () => passphrase,
mnemonicWords
mnemonicWords,
purpose: KeyPurpose.STANDARD
},
{ bip32Ed25519, logger }
);
Expand Down
19 changes: 13 additions & 6 deletions packages/hardware-ledger/src/LedgerKeyAgent.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import {
KeyAgentBase,
KeyAgentDependencies,
KeyAgentType,
KeyPurpose,
SerializableLedgerKeyAgentData,
SignBlobResult,
SignTransactionContext,
Expand Down Expand Up @@ -67,12 +68,14 @@ export interface CreateLedgerKeyAgentProps {
accountIndex?: number;
communicationType: CommunicationType;
deviceConnection?: LedgerConnection | null;
purpose: KeyPurpose;
}

export interface GetLedgerXpubProps {
deviceConnection?: LedgerConnection;
communicationType: CommunicationType;
accountIndex: number;
purpose: KeyPurpose;
}

export interface CreateLedgerTransportProps {
Expand Down Expand Up @@ -434,11 +437,12 @@ export class LedgerKeyAgent extends KeyAgentBase {
static async getXpub({
deviceConnection,
communicationType,
accountIndex
accountIndex,
purpose
}: GetLedgerXpubProps): Promise<Crypto.Bip32PublicKeyHex> {
try {
const recoveredDeviceConnection = await LedgerKeyAgent.checkDeviceConnection(communicationType, deviceConnection);
const derivationPath = `${CardanoKeyConst.PURPOSE}'/${CardanoKeyConst.COIN_TYPE}'/${accountIndex}'`;
const derivationPath = `${purpose}'/${CardanoKeyConst.COIN_TYPE}'/${accountIndex}'`;
const extendedPublicKey = await recoveredDeviceConnection.getExtendedPublicKey({
path: str_to_path(derivationPath) // BIP32Path
});
Expand Down Expand Up @@ -468,7 +472,7 @@ export class LedgerKeyAgent extends KeyAgentBase {
* @throws TransportError
*/
static async createWithDevice(
{ chainId, accountIndex = 0, communicationType, deviceConnection }: CreateLedgerKeyAgentProps,
{ chainId, accountIndex = 0, communicationType, deviceConnection, purpose }: CreateLedgerKeyAgentProps,
dependencies: KeyAgentDependencies
) {
const deviceListPaths = await LedgerKeyAgent.getHidDeviceList(communicationType);
Expand All @@ -479,7 +483,8 @@ export class LedgerKeyAgent extends KeyAgentBase {
const extendedAccountPublicKey = await LedgerKeyAgent.getXpub({
accountIndex,
communicationType,
deviceConnection: activeDeviceConnection
deviceConnection: activeDeviceConnection,
purpose
});

return new LedgerKeyAgent(
Expand All @@ -488,7 +493,8 @@ export class LedgerKeyAgent extends KeyAgentBase {
chainId,
communicationType,
deviceConnection: activeDeviceConnection,
extendedAccountPublicKey
extendedAccountPublicKey,
purpose
},
dependencies
);
Expand Down Expand Up @@ -539,14 +545,15 @@ export class LedgerKeyAgent extends KeyAgentBase {
// TODO: Allow additional key paths
async signTransaction(
{ body, hash }: Cardano.TxBodyWithHash,
{ knownAddresses, txInKeyPathMap }: SignTransactionContext
{ knownAddresses, txInKeyPathMap, purpose }: SignTransactionContext
): Promise<Cardano.Signatures> {
try {
const ledgerTxData = await toLedgerTx(body, {
accountIndex: this.accountIndex,
chainId: this.chainId,
dRepPublicKey: await this.derivePublicKey(util.DREP_KEY_DERIVATION_PATH),
knownAddresses,
purpose,
txInKeyPathMap
});

Expand Down
16 changes: 10 additions & 6 deletions packages/hardware-ledger/src/transformers/txIn.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,18 @@ import { TxInId, util } from '@cardano-sdk/key-management';

const resolveKeyPath = (
txIn: Cardano.TxIn,
{ accountIndex, txInKeyPathMap }: LedgerTxTransformerContext
{ accountIndex, txInKeyPathMap, purpose }: LedgerTxTransformerContext
): Ledger.BIP32Path | null => {
const utxoKeyPath = txInKeyPathMap[TxInId(txIn)];
if (utxoKeyPath) {
return util.accountKeyDerivationPathToBip32Path(accountIndex, utxoKeyPath);
}
const txInKeyPath = txInKeyPathMap[TxInId(txIn)];

return null;
if (!txInKeyPath || txInKeyPath.role === undefined || txInKeyPath.index === undefined) return null;

const utxoKeyPath = {
...txInKeyPath,
purpose
};

return util.accountKeyDerivationPathToBip32Path(accountIndex, utxoKeyPath);
};

export const toTxIn: Transform<Cardano.TxIn, Ledger.TxInput, LedgerTxTransformerContext> = (txIn, context) => ({
Expand Down
Loading
Loading