mirror of
https://github.com/danbulant/jose
synced 2026-05-25 13:01:49 +00:00
61 lines
1.9 KiB
TypeScript
61 lines
1.9 KiB
TypeScript
import { promisify } from 'util'
|
|
import { pbkdf2 as pbkdf2cb } from 'crypto'
|
|
import type { Pbes2KWDecryptFunction, Pbes2KWEncryptFunction } from '../interfaces'
|
|
import random from './random.js'
|
|
import { p2s as concatSalt } from '../../lib/buffer_utils.js'
|
|
import { encode as base64url } from './base64url.js'
|
|
import { wrap, unwrap } from './aeskw.js'
|
|
import checkP2s from '../../lib/check_p2s.js'
|
|
import { isCryptoKey, getKeyObject } from './webcrypto.js'
|
|
import isKeyObject from './is_key_object.js'
|
|
import invalidKeyInput from './invalid_key_input.js'
|
|
|
|
const pbkdf2 = promisify(pbkdf2cb)
|
|
|
|
function getPassword(key: unknown, alg: string) {
|
|
if (isKeyObject(key)) {
|
|
return key.export()
|
|
}
|
|
if (key instanceof Uint8Array) {
|
|
return key
|
|
}
|
|
if (isCryptoKey(key)) {
|
|
return getKeyObject(key, alg, new Set(['deriveBits', 'deriveKey'])).export()
|
|
}
|
|
throw new TypeError(invalidKeyInput(key, 'KeyObject', 'CryptoKey', 'Uint8Array'))
|
|
}
|
|
|
|
export const encrypt: Pbes2KWEncryptFunction = async (
|
|
alg: string,
|
|
key: unknown,
|
|
cek: Uint8Array,
|
|
p2c: number = Math.floor(Math.random() * 2049) + 2048,
|
|
p2s: Uint8Array = random(new Uint8Array(16)),
|
|
) => {
|
|
checkP2s(p2s)
|
|
const salt = concatSalt(alg, p2s)
|
|
const keylen = parseInt(alg.substr(13, 3), 10) >> 3
|
|
const password = getPassword(key, alg)
|
|
|
|
const derivedKey = await pbkdf2(password, salt, p2c, keylen, `sha${alg.substr(8, 3)}`)
|
|
const encryptedKey = await wrap(alg.substr(-6), derivedKey, cek)
|
|
|
|
return { encryptedKey, p2c, p2s: base64url(p2s) }
|
|
}
|
|
|
|
export const decrypt: Pbes2KWDecryptFunction = async (
|
|
alg: string,
|
|
key: unknown,
|
|
encryptedKey: Uint8Array,
|
|
p2c: number,
|
|
p2s: Uint8Array,
|
|
) => {
|
|
checkP2s(p2s)
|
|
const salt = concatSalt(alg, p2s)
|
|
const keylen = parseInt(alg.substr(13, 3), 10) >> 3
|
|
const password = getPassword(key, alg)
|
|
|
|
const derivedKey = await pbkdf2(password, salt, p2c, keylen, `sha${alg.substr(8, 3)}`)
|
|
|
|
return unwrap(alg.substr(-6), derivedKey, encryptedKey)
|
|
}
|