From 9134ea744469175645b46620c9b53266a9b557a3 Mon Sep 17 00:00:00 2001 From: Steven Luscher Date: Thu, 13 Jul 2023 04:07:40 +0000 Subject: [PATCH] refactor(experimental): a function to convert a public key into a base 58 encoded address ## Summary This is useful in cases where you, yourself, generated a keypair, and now you need to display the public key in your UI. Otherwise, you will typically already have the `Base58EncodedAddress` object. ## Test Plan ``` cd packages/keys/ pnpm test:unit:browser pnpm test:unit:node ``` --- packages/keys/README.md | 20 ++++- packages/keys/src/__tests__/pubkey-test.ts | 90 ++++++++++++++++++++++ packages/keys/src/index.ts | 1 + packages/keys/src/pubkey.ts | 13 ++++ 4 files changed, 122 insertions(+), 2 deletions(-) create mode 100644 packages/keys/src/__tests__/pubkey-test.ts create mode 100644 packages/keys/src/pubkey.ts diff --git a/packages/keys/README.md b/packages/keys/README.md index d4ad7476ed6f..9b07da9bb04d 100644 --- a/packages/keys/README.md +++ b/packages/keys/README.md @@ -33,7 +33,7 @@ Client applications primarily deal with addresses and public keys in the form of From time to time you might acquire a string, that you expect to validate as an address, from an untrusted network API or user input. To assert that such an arbitrary string is a base58-encoded address, use the `assertIsBase58EncodedAddress` function. ```ts -import { assertIsBase58EncodedAddress } from '@solana/web3.js`; +import { assertIsBase58EncodedAddress } from '@solana/keys`; // Imagine a function that fetches an account's balance when a user submits a form. function handleSubmit() { @@ -51,6 +51,22 @@ function handleSubmit() { } ``` -### `generateKeyPair()` +### `generateKeypair()` Generates an Ed25519 public/private keypair for use with other methods in this package that accept `CryptoKey` objects. + +```ts +import { generateKeypair } from '@solana/keys'; + +const { privateKey, publicKey } = await generateKeypair(); +``` + +### `getBase58EncodedAddressFromPublicKey()` + +Given a public `CryptoKey`, this method will return its associated `Base58EncodedAddress`. + +```ts +import { getBase58EncodedAddressFromPublicKey } from '@solana/keys'; + +const address = await getBase58EncodedAddressFromPublicKey(publicKey); +``` diff --git a/packages/keys/src/__tests__/pubkey-test.ts b/packages/keys/src/__tests__/pubkey-test.ts new file mode 100644 index 000000000000..c9b14460a421 --- /dev/null +++ b/packages/keys/src/__tests__/pubkey-test.ts @@ -0,0 +1,90 @@ +import { getBase58EncodedAddressFromPublicKey } from '../pubkey'; + +// Corresponds to address `DcESq8KFcdTdpjWtr2DoGcvu5McM3VJoBetgM1X1vVct` +const MOCK_PUBLIC_KEY_BYTES = new Uint8Array([ + 0xbb, 0x52, 0xc6, 0x2d, 0x52, 0x4f, 0x7f, 0xea, 0x4f, 0x2c, 0x27, 0x13, 0xd6, 0x20, 0x80, 0xad, 0x6a, 0x36, 0x9a, + 0x0e, 0x36, 0x71, 0x74, 0x32, 0x8d, 0x1a, 0xf7, 0xee, 0x7e, 0x04, 0x76, 0x19, +]); + +describe('getBase58EncodedAddressFromPublicKey', () => { + let oldIsSecureContext: boolean; + beforeEach(() => { + oldIsSecureContext = globalThis.isSecureContext; + globalThis.isSecureContext = true; + }); + afterEach(() => { + globalThis.isSecureContext = oldIsSecureContext; + }); + it('returns the public key that corresponds to a given secret key', async () => { + expect.assertions(1); + const publicKey = await crypto.subtle.importKey( + 'raw', + MOCK_PUBLIC_KEY_BYTES, + 'Ed25519', + /* extractable */ true, + ['verify'] + ); + await expect(getBase58EncodedAddressFromPublicKey(publicKey)).resolves.toBe( + 'DcESq8KFcdTdpjWtr2DoGcvu5McM3VJoBetgM1X1vVct' + ); + }); + it('throws when the public key is non-extractable', async () => { + expect.assertions(1); + const publicKey = await crypto.subtle.importKey( + 'raw', + MOCK_PUBLIC_KEY_BYTES, + 'Ed25519', + /* extractable */ false, + ['verify'] + ); + await expect(() => getBase58EncodedAddressFromPublicKey(publicKey)).rejects.toThrow(); + }); + it('throws when called with a secret', async () => { + expect.assertions(1); + const publicKey = await crypto.subtle.generateKey( + { + length: 256, + name: 'AES-GCM', + }, + true, + ['encrypt', 'decrypt'] + ); + await expect(() => getBase58EncodedAddressFromPublicKey(publicKey)).rejects.toThrow(); + }); + it.each([ + { __variant: 'P256', name: 'ECDSA', namedCurve: 'P-256' }, + { __variant: 'P384', name: 'ECDSA', namedCurve: 'P-384' } as EcKeyGenParams, + { __variant: 'P521', name: 'ECDSA', namedCurve: 'P-521' } as EcKeyGenParams, + ...['RSASSA-PKCS1-v1_5', 'RSA-PSS'].flatMap(rsaAlgoName => + ['SHA-1', 'SHA-256', 'SHA-384', 'SHA-512'].map( + hashName => + ({ + __variant: hashName, + hash: { name: hashName }, + modulusLength: 2048, + name: rsaAlgoName, + publicExponent: new Uint8Array([0x01, 0x00, 0x01]), + } as RsaHashedKeyGenParams) + ) + ), + ])('throws when called with a $name/$__variant public key', async algorithm => { + expect.assertions(1); + const { publicKey } = await crypto.subtle.generateKey(algorithm, true, ['sign', 'verify']); + await expect(() => getBase58EncodedAddressFromPublicKey(publicKey)).rejects.toThrow(); + }); + it('throws when called with a private key', async () => { + expect.assertions(1); + const mockPrivateKey = await crypto.subtle.importKey( + 'pkcs8', + new Uint8Array([ + 0x30, 0x2e, 0x02, 0x01, 0x00, 0x30, 0x05, 0x06, 0x03, 0x2b, 0x65, 0x70, 0x04, 0x22, 0x04, 0x20, 0xf2, + 0x29, 0xe0, 0x33, 0x09, 0x44, 0x10, 0xd9, 0x64, 0x80, 0x42, 0x85, 0x9a, 0x18, 0x5c, 0x4a, 0x45, 0x45, + 0xd9, 0xd1, 0x75, 0xeb, 0x30, 0x89, 0xb4, 0x2b, 0x7b, 0xe3, 0xca, 0xbf, 0x63, 0xc9, + ]), + 'Ed25519', + /* extractable */ false, + ['sign'] + ); + await expect(() => getBase58EncodedAddressFromPublicKey(mockPrivateKey)).rejects.toThrow(); + }); +}); diff --git a/packages/keys/src/index.ts b/packages/keys/src/index.ts index 281bd9217d03..be00804daa9e 100644 --- a/packages/keys/src/index.ts +++ b/packages/keys/src/index.ts @@ -1,2 +1,3 @@ export * from './base58'; export * from './keypair'; +export * from './pubkey'; diff --git a/packages/keys/src/pubkey.ts b/packages/keys/src/pubkey.ts new file mode 100644 index 000000000000..c773788caa0f --- /dev/null +++ b/packages/keys/src/pubkey.ts @@ -0,0 +1,13 @@ +import { Base58EncodedAddress, getBase58EncodedAddressCodec } from './base58'; +import { assertKeyExporterIsAvailable } from './guard'; + +export async function getBase58EncodedAddressFromPublicKey(publicKey: CryptoKey): Promise { + await assertKeyExporterIsAvailable(); + if (publicKey.type !== 'public' || publicKey.algorithm.name !== 'Ed25519') { + // TODO: Coded error. + throw new Error('The `CryptoKey` must be an `Ed25519` public key'); + } + const publicKeyBytes = await crypto.subtle.exportKey('raw', publicKey); + const [base58EncodedAddress] = getBase58EncodedAddressCodec().deserialize(new Uint8Array(publicKeyBytes)); + return base58EncodedAddress; +}