-
Notifications
You must be signed in to change notification settings - Fork 917
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
[experimental] Validate that the public key generated from createKeyPairFromBytes() belongs to the private key #2329
Changes from 7 commits
3e8eb1d
1c3c7c0
4e9304b
0e7548f
62c7121
70611e0
99ff817
9292acf
0f1a0ea
6e50c49
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
import { SOLANA_ERROR__CRYPTO__RANDOM_VALUES_FUNCTION_UNIMPLEMENTED, SolanaError } from '@solana/errors'; | ||
|
||
import { assertPRNGIsAvailable } from '../crypto'; | ||
|
||
describe('assertPRNGIsAvailable()', () => { | ||
it('resolves to `undefined` without throwing', async () => { | ||
expect.assertions(1); | ||
await expect(assertPRNGIsAvailable()).resolves.toBeUndefined(); | ||
}); | ||
describe('when getRandomValues is not available', () => { | ||
let oldCrypto: InstanceType<typeof Crypto>['getRandomValues']; | ||
beforeEach(() => { | ||
oldCrypto = globalThis.crypto.getRandomValues; | ||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment | ||
// @ts-ignore | ||
globalThis.crypto.getRandomValues = undefined; | ||
}); | ||
afterEach(() => { | ||
globalThis.crypto.getRandomValues = oldCrypto; | ||
}); | ||
it('rejects', async () => { | ||
expect.assertions(1); | ||
await expect(() => assertPRNGIsAvailable()).rejects.toThrow( | ||
new SolanaError(SOLANA_ERROR__CRYPTO__RANDOM_VALUES_FUNCTION_UNIMPLEMENTED), | ||
); | ||
}); | ||
}); | ||
}); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
import { SOLANA_ERROR__CRYPTO__RANDOM_VALUES_FUNCTION_UNIMPLEMENTED, SolanaError } from '@solana/errors'; | ||
|
||
export async function assertPRNGIsAvailable() { | ||
if (typeof globalThis.crypto === 'undefined' || typeof globalThis.crypto.getRandomValues !== 'function') { | ||
throw new SolanaError(SOLANA_ERROR__CRYPTO__RANDOM_VALUES_FUNCTION_UNIMPLEMENTED); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,35 +1,88 @@ | ||
import { generateKeyPair } from '../key-pair'; | ||
import { | ||
SOLANA_ERROR__KEYS__INVALID_KEY_PAIR_BYTE_LENGTH, | ||
SOLANA_ERROR__KEYS__PUBLIC_KEY_MUST_MATCH_PRIVATE_KEY, | ||
SolanaError, | ||
} from '@solana/errors'; | ||
|
||
describe('generateKeyPair', () => { | ||
it.each(['private', 'public'])('generates an ed25519 %s `CryptoKey`', async type => { | ||
expect.assertions(1); | ||
const keyPair = await generateKeyPair(); | ||
expect(keyPair).toMatchObject({ | ||
[`${type}Key`]: expect.objectContaining({ | ||
[Symbol.toStringTag]: 'CryptoKey', | ||
algorithm: { name: 'Ed25519' }, | ||
type, | ||
}), | ||
import { createKeyPairFromBytes, generateKeyPair } from '../key-pair'; | ||
|
||
const MOCK_KEY_BYTES = new Uint8Array([ | ||
0xeb, 0xfa, 0x65, 0xeb, 0x93, 0xdc, 0x79, 0x15, 0x7a, 0xba, 0xde, 0xa2, 0xf7, 0x94, 0x37, 0x9d, 0xfc, 0x07, 0x1d, | ||
0x68, 0x86, 0x87, 0x37, 0x6d, 0xc5, 0xd5, 0xa0, 0x54, 0x12, 0x1d, 0x34, 0x4a, 0x1d, 0x0e, 0x93, 0x86, 0x4d, 0xcc, | ||
0x81, 0x5f, 0xc3, 0xf2, 0x86, 0x18, 0x09, 0x11, 0xd0, 0x0a, 0x3f, 0xd2, 0x06, 0xde, 0x31, 0xa1, 0xc9, 0x42, 0x87, | ||
0xcb, 0x43, 0xf0, 0x5f, 0xc9, 0xf2, 0xb5, | ||
]); | ||
|
||
const MOCK_INVALID_KEY_BYTES = new Uint8Array([ | ||
0xeb, 0xfa, 0x65, 0xeb, 0x93, 0xdc, 0x79, 0x15, 0x7a, 0xba, 0xde, 0xa2, 0xf7, 0x94, 0x37, 0x9d, 0xfc, 0x07, 0x1d, | ||
0x68, 0x86, 0x87, 0x37, 0x6d, 0xc5, 0xd5, 0xa0, 0x54, 0x12, 0x1d, 0x34, 0x4a, 0x1d, 0x0e, 0x93, 0x86, 0x4d, 0xcc, | ||
0x81, 0x5f, 0xc3, 0xf2, 0x86, 0x18, 0x09, 0x11, 0xd0, 0x0a, 0x3f, 0xd2, 0x06, 0xde, 0x31, 0xa1, 0xc9, 0x42, 0x87, | ||
0xcb, 0x43, 0xf0, 0x5f, 0xc9, 0xf2, 0xb1, | ||
]); | ||
|
||
describe('key-pair', () => { | ||
describe('generateKeyPair', () => { | ||
it.each(['private', 'public'])('generates an ed25519 %s `CryptoKey`', async type => { | ||
expect.assertions(1); | ||
const keyPair = await generateKeyPair(); | ||
expect(keyPair).toMatchObject({ | ||
[`${type}Key`]: expect.objectContaining({ | ||
[Symbol.toStringTag]: 'CryptoKey', | ||
algorithm: { name: 'Ed25519' }, | ||
type, | ||
}), | ||
}); | ||
}); | ||
it('generates a non-extractable private key', async () => { | ||
expect.assertions(1); | ||
const { privateKey } = await generateKeyPair(); | ||
expect(privateKey).toHaveProperty('extractable', false); | ||
}); | ||
it('generates a private key usable for signing operations', async () => { | ||
expect.assertions(1); | ||
const { privateKey } = await generateKeyPair(); | ||
expect(privateKey).toHaveProperty('usages', ['sign']); | ||
}); | ||
it('generates an extractable public key', async () => { | ||
expect.assertions(1); | ||
const { publicKey } = await generateKeyPair(); | ||
expect(publicKey).toHaveProperty('extractable', true); | ||
}); | ||
it('generates a public key usable for verifying signatures', async () => { | ||
expect.assertions(1); | ||
const { publicKey } = await generateKeyPair(); | ||
expect(publicKey).toHaveProperty('usages', ['verify']); | ||
}); | ||
}); | ||
it('generates a non-extractable private key', async () => { | ||
expect.assertions(1); | ||
const { privateKey } = await generateKeyPair(); | ||
expect(privateKey).toHaveProperty('extractable', false); | ||
}); | ||
it('generates a private key usable for signing operations', async () => { | ||
expect.assertions(1); | ||
const { privateKey } = await generateKeyPair(); | ||
expect(privateKey).toHaveProperty('usages', ['sign']); | ||
}); | ||
it('generates an extractable public key', async () => { | ||
expect.assertions(1); | ||
const { publicKey } = await generateKeyPair(); | ||
expect(publicKey).toHaveProperty('extractable', true); | ||
}); | ||
it('generates a public key usable for verifying signatures', async () => { | ||
expect.assertions(1); | ||
const { publicKey } = await generateKeyPair(); | ||
expect(publicKey).toHaveProperty('usages', ['verify']); | ||
|
||
describe('createKeyPairFromBytes', () => { | ||
it('creates a key pair from a 64-byte array', async () => { | ||
expect.assertions(1); | ||
const keyPair = await createKeyPairFromBytes(MOCK_KEY_BYTES); | ||
expect(keyPair).toMatchObject({ | ||
privateKey: expect.objectContaining({ | ||
[Symbol.toStringTag]: 'CryptoKey', | ||
algorithm: { name: 'Ed25519' }, | ||
type: 'private', | ||
}), | ||
publicKey: expect.objectContaining({ | ||
[Symbol.toStringTag]: 'CryptoKey', | ||
algorithm: { name: 'Ed25519' }, | ||
type: 'public', | ||
}), | ||
}); | ||
}); | ||
it('errors when the byte array is not 64 bytes', async () => { | ||
expect.assertions(1); | ||
await expect(createKeyPairFromBytes(MOCK_KEY_BYTES.slice(0, 31))).rejects.toThrow( | ||
new SolanaError(SOLANA_ERROR__KEYS__INVALID_KEY_PAIR_BYTE_LENGTH, { byteLength: 31 }), | ||
); | ||
}); | ||
it('errors when public key fails signature verification', async () => { | ||
expect.assertions(1); | ||
await expect(createKeyPairFromBytes(MOCK_INVALID_KEY_BYTES)).rejects.toThrow( | ||
new SolanaError(SOLANA_ERROR__KEYS__PUBLIC_KEY_MUST_MATCH_PRIVATE_KEY), | ||
); | ||
}); | ||
}); | ||
}); |
Original file line number | Diff line number | Diff line change | ||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -1,7 +1,12 @@ | ||||||||||||
import { assertKeyGenerationIsAvailable } from '@solana/assertions'; | ||||||||||||
import { SOLANA_ERROR__KEYS__INVALID_KEY_PAIR_BYTE_LENGTH, SolanaError } from '@solana/errors'; | ||||||||||||
import { assertKeyGenerationIsAvailable, assertPRNGIsAvailable, assert } from '@solana/assertions'; | ||||||||||||
Check failure on line 1 in packages/keys/src/key-pair.ts GitHub Actions / Build & Test on Node current
Check failure on line 1 in packages/keys/src/key-pair.ts GitHub Actions / Build & Test on Node current
Check failure on line 1 in packages/keys/src/key-pair.ts GitHub Actions / Build & Test on Node lts/*
|
||||||||||||
import { | ||||||||||||
SOLANA_ERROR__KEYS__INVALID_KEY_PAIR_BYTE_LENGTH, | ||||||||||||
SOLANA_ERROR__KEYS__PUBLIC_KEY_MUST_MATCH_PRIVATE_KEY, | ||||||||||||
SolanaError, | ||||||||||||
} from '@solana/errors'; | ||||||||||||
|
||||||||||||
import { createPrivateKeyFromBytes } from './private-key'; | ||||||||||||
import { signBytes, verifySignature } from './signatures'; | ||||||||||||
|
||||||||||||
export async function generateKeyPair(): Promise<CryptoKeyPair> { | ||||||||||||
await assertKeyGenerationIsAvailable(); | ||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. To generate a test message randomly each time, we'll need to assert that the PRNG is available (that
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think it would make more sense to put There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Oh, of course. Oops; I left my comment in the wrong spot! |
||||||||||||
|
@@ -14,12 +19,24 @@ | |||||||||||
} | ||||||||||||
|
||||||||||||
export async function createKeyPairFromBytes(bytes: Uint8Array, extractable?: boolean): Promise<CryptoKeyPair> { | ||||||||||||
await assertPRNGIsAvailable(); | ||||||||||||
|
||||||||||||
if (bytes.byteLength !== 64) { | ||||||||||||
throw new SolanaError(SOLANA_ERROR__KEYS__INVALID_KEY_PAIR_BYTE_LENGTH, { byteLength: bytes.byteLength }); | ||||||||||||
} | ||||||||||||
const [publicKey, privateKey] = await Promise.all([ | ||||||||||||
crypto.subtle.importKey('raw', bytes.slice(32), 'Ed25519', /* extractable */ true, ['verify']), | ||||||||||||
createPrivateKeyFromBytes(bytes.slice(0, 32), extractable), | ||||||||||||
]); | ||||||||||||
|
||||||||||||
// Verify the key pair | ||||||||||||
const randomBytes = new Uint8Array(32); | ||||||||||||
crypto.getRandomValues(randomBytes); | ||||||||||||
const signedData = await signBytes(privateKey, randomBytes); | ||||||||||||
const isValid = await verifySignature(publicKey, signedData, randomBytes); | ||||||||||||
if (!isValid) { | ||||||||||||
throw new SolanaError(SOLANA_ERROR__KEYS__PUBLIC_KEY_MUST_MATCH_PRIVATE_KEY); | ||||||||||||
} | ||||||||||||
|
||||||||||||
return { privateKey, publicKey } as CryptoKeyPair; | ||||||||||||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm… I wonder why some of these were made
async
. Seems like a mistake we shouldn't replicate here.