65 lines
2.4 KiB
JavaScript
65 lines
2.4 KiB
JavaScript
|
import { diffieHellman, generateKeyPair as generateKeyPairCb, KeyObject } from 'crypto';
|
||
|
import { promisify } from 'util';
|
||
|
import getNamedCurve from './get_named_curve.js';
|
||
|
import { encoder, concat, uint32be, lengthAndInput, concatKdf } from '../lib/buffer_utils.js';
|
||
|
import { JOSENotSupported } from '../util/errors.js';
|
||
|
import { isCryptoKey } from './webcrypto.js';
|
||
|
import { checkEncCryptoKey } from '../lib/crypto_key.js';
|
||
|
import isKeyObject from './is_key_object.js';
|
||
|
import invalidKeyInput from '../lib/invalid_key_input.js';
|
||
|
import { types } from './is_key_like.js';
|
||
|
const generateKeyPair = promisify(generateKeyPairCb);
|
||
|
export async function deriveKey(publicKee, privateKee, algorithm, keyLength, apu = new Uint8Array(0), apv = new Uint8Array(0)) {
|
||
|
let publicKey;
|
||
|
if (isCryptoKey(publicKee)) {
|
||
|
checkEncCryptoKey(publicKee, 'ECDH');
|
||
|
publicKey = KeyObject.from(publicKee);
|
||
|
}
|
||
|
else if (isKeyObject(publicKee)) {
|
||
|
publicKey = publicKee;
|
||
|
}
|
||
|
else {
|
||
|
throw new TypeError(invalidKeyInput(publicKee, ...types));
|
||
|
}
|
||
|
let privateKey;
|
||
|
if (isCryptoKey(privateKee)) {
|
||
|
checkEncCryptoKey(privateKee, 'ECDH', 'deriveBits');
|
||
|
privateKey = KeyObject.from(privateKee);
|
||
|
}
|
||
|
else if (isKeyObject(privateKee)) {
|
||
|
privateKey = privateKee;
|
||
|
}
|
||
|
else {
|
||
|
throw new TypeError(invalidKeyInput(privateKee, ...types));
|
||
|
}
|
||
|
const value = concat(lengthAndInput(encoder.encode(algorithm)), lengthAndInput(apu), lengthAndInput(apv), uint32be(keyLength));
|
||
|
const sharedSecret = diffieHellman({ privateKey, publicKey });
|
||
|
return concatKdf(sharedSecret, keyLength, value);
|
||
|
}
|
||
|
export async function generateEpk(kee) {
|
||
|
let key;
|
||
|
if (isCryptoKey(kee)) {
|
||
|
key = KeyObject.from(kee);
|
||
|
}
|
||
|
else if (isKeyObject(kee)) {
|
||
|
key = kee;
|
||
|
}
|
||
|
else {
|
||
|
throw new TypeError(invalidKeyInput(kee, ...types));
|
||
|
}
|
||
|
switch (key.asymmetricKeyType) {
|
||
|
case 'x25519':
|
||
|
return generateKeyPair('x25519');
|
||
|
case 'x448': {
|
||
|
return generateKeyPair('x448');
|
||
|
}
|
||
|
case 'ec': {
|
||
|
const namedCurve = getNamedCurve(key);
|
||
|
return generateKeyPair('ec', { namedCurve });
|
||
|
}
|
||
|
default:
|
||
|
throw new JOSENotSupported('Invalid or unsupported EPK');
|
||
|
}
|
||
|
}
|
||
|
export const ecdhAllowed = (key) => ['P-256', 'P-384', 'P-521', 'X25519', 'X448'].includes(getNamedCurve(key));
|