2023-02-14 11:38:45 -05:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
|
|
// Copyright Joyent, Inc. and Node.js contributors. All rights reserved. MIT license.
|
|
|
|
|
2023-06-27 02:18:22 -04:00
|
|
|
// TODO(petamoriken): enable prefer-primordials for node polyfills
|
2023-09-07 09:09:16 -04:00
|
|
|
// deno-lint-ignore-file prefer-primordials
|
2023-05-01 11:40:00 -04:00
|
|
|
|
2023-03-08 06:44:54 -05:00
|
|
|
import { notImplemented } from "ext:deno_node/_utils.ts";
|
|
|
|
import randomBytes from "ext:deno_node/internal/crypto/_randomBytes.ts";
|
2023-02-14 11:38:45 -05:00
|
|
|
import randomFill, {
|
|
|
|
randomFillSync,
|
2023-09-23 04:04:55 -04:00
|
|
|
} from "ext:deno_node/internal/crypto/_randomFill.mjs";
|
2023-03-08 06:44:54 -05:00
|
|
|
import randomInt from "ext:deno_node/internal/crypto/_randomInt.ts";
|
2023-03-28 06:56:38 -04:00
|
|
|
import {
|
2023-04-27 10:10:59 -04:00
|
|
|
validateBoolean,
|
2023-03-28 06:56:38 -04:00
|
|
|
validateFunction,
|
|
|
|
validateInt32,
|
|
|
|
validateObject,
|
|
|
|
} from "ext:deno_node/internal/validators.mjs";
|
|
|
|
import {
|
|
|
|
isAnyArrayBuffer,
|
|
|
|
isArrayBufferView,
|
|
|
|
} from "ext:deno_node/internal/util/types.ts";
|
2023-04-27 10:10:59 -04:00
|
|
|
import {
|
|
|
|
ERR_INVALID_ARG_TYPE,
|
|
|
|
ERR_OUT_OF_RANGE,
|
|
|
|
} from "ext:deno_node/internal/errors.ts";
|
2023-02-14 11:38:45 -05:00
|
|
|
|
2023-03-08 06:44:54 -05:00
|
|
|
export { default as randomBytes } from "ext:deno_node/internal/crypto/_randomBytes.ts";
|
2023-02-14 11:38:45 -05:00
|
|
|
export {
|
|
|
|
default as randomFill,
|
|
|
|
randomFillSync,
|
2023-09-23 04:04:55 -04:00
|
|
|
} from "ext:deno_node/internal/crypto/_randomFill.mjs";
|
2023-03-08 06:44:54 -05:00
|
|
|
export { default as randomInt } from "ext:deno_node/internal/crypto/_randomInt.ts";
|
2023-02-14 11:38:45 -05:00
|
|
|
|
2023-12-07 08:21:01 -05:00
|
|
|
import { primordials } from "ext:core/mod.js";
|
2023-09-07 09:09:16 -04:00
|
|
|
const { StringPrototypePadStart, StringPrototypeToString } = primordials;
|
|
|
|
|
2023-03-28 06:56:38 -04:00
|
|
|
const { core } = globalThis.__bootstrap;
|
|
|
|
const { ops } = core;
|
2023-05-01 11:40:00 -04:00
|
|
|
const {
|
|
|
|
op_node_gen_prime_async,
|
|
|
|
op_node_check_prime_bytes_async,
|
|
|
|
op_node_check_prime_async,
|
2023-06-06 05:01:28 -04:00
|
|
|
} = Deno.core.ensureFastOps();
|
2023-03-28 06:56:38 -04:00
|
|
|
|
2023-02-14 11:38:45 -05:00
|
|
|
export type LargeNumberLike =
|
|
|
|
| ArrayBufferView
|
|
|
|
| SharedArrayBuffer
|
|
|
|
| ArrayBuffer
|
|
|
|
| bigint;
|
|
|
|
|
|
|
|
export interface CheckPrimeOptions {
|
|
|
|
/**
|
|
|
|
* The number of Miller-Rabin probabilistic primality iterations to perform.
|
|
|
|
* When the value is 0 (zero), a number of checks is used that yields a false positive rate of at most 2-64 for random input.
|
|
|
|
* Care must be used when selecting a number of checks.
|
|
|
|
* Refer to the OpenSSL documentation for the BN_is_prime_ex function nchecks options for more details.
|
|
|
|
*
|
|
|
|
* @default 0
|
|
|
|
*/
|
|
|
|
checks?: number | undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function checkPrime(
|
|
|
|
candidate: LargeNumberLike,
|
|
|
|
callback: (err: Error | null, result: boolean) => void,
|
|
|
|
): void;
|
|
|
|
export function checkPrime(
|
|
|
|
candidate: LargeNumberLike,
|
|
|
|
options: CheckPrimeOptions,
|
|
|
|
callback: (err: Error | null, result: boolean) => void,
|
|
|
|
): void;
|
|
|
|
export function checkPrime(
|
2023-03-28 06:56:38 -04:00
|
|
|
candidate: LargeNumberLike,
|
|
|
|
options: CheckPrimeOptions | ((err: Error | null, result: boolean) => void) =
|
|
|
|
{},
|
|
|
|
callback?: (err: Error | null, result: boolean) => void,
|
2023-02-14 11:38:45 -05:00
|
|
|
) {
|
2023-03-28 06:56:38 -04:00
|
|
|
if (typeof options === "function") {
|
|
|
|
callback = options;
|
|
|
|
options = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
validateFunction(callback, "callback");
|
|
|
|
validateObject(options, "options");
|
|
|
|
|
|
|
|
const {
|
|
|
|
checks = 0,
|
|
|
|
} = options!;
|
|
|
|
|
|
|
|
validateInt32(checks, "options.checks", 0);
|
|
|
|
|
2023-05-01 11:40:00 -04:00
|
|
|
let op = op_node_check_prime_bytes_async;
|
2023-03-28 06:56:38 -04:00
|
|
|
if (typeof candidate === "bigint") {
|
2023-05-01 11:40:00 -04:00
|
|
|
op = op_node_check_prime_async;
|
2023-03-28 06:56:38 -04:00
|
|
|
} else if (!isAnyArrayBuffer(candidate) && !isArrayBufferView(candidate)) {
|
|
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
|
|
"candidate",
|
|
|
|
[
|
|
|
|
"ArrayBuffer",
|
|
|
|
"TypedArray",
|
|
|
|
"Buffer",
|
|
|
|
"DataView",
|
|
|
|
"bigint",
|
|
|
|
],
|
|
|
|
candidate,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-05-01 11:40:00 -04:00
|
|
|
op(candidate, checks).then(
|
2023-03-28 06:56:38 -04:00
|
|
|
(result) => {
|
|
|
|
callback?.(null, result);
|
|
|
|
},
|
|
|
|
).catch((err) => {
|
|
|
|
callback?.(err, false);
|
|
|
|
});
|
2023-02-14 11:38:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export function checkPrimeSync(
|
2023-03-28 06:56:38 -04:00
|
|
|
candidate: LargeNumberLike,
|
|
|
|
options: CheckPrimeOptions = {},
|
2023-02-14 11:38:45 -05:00
|
|
|
): boolean {
|
2023-03-28 06:56:38 -04:00
|
|
|
validateObject(options, "options");
|
|
|
|
|
|
|
|
const {
|
|
|
|
checks = 0,
|
|
|
|
} = options!;
|
|
|
|
|
|
|
|
validateInt32(checks, "options.checks", 0);
|
|
|
|
|
|
|
|
if (typeof candidate === "bigint") {
|
|
|
|
return ops.op_node_check_prime(candidate, checks);
|
|
|
|
} else if (!isAnyArrayBuffer(candidate) && !isArrayBufferView(candidate)) {
|
|
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
|
|
"candidate",
|
|
|
|
[
|
|
|
|
"ArrayBuffer",
|
|
|
|
"TypedArray",
|
|
|
|
"Buffer",
|
|
|
|
"DataView",
|
|
|
|
"bigint",
|
|
|
|
],
|
|
|
|
candidate,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ops.op_node_check_prime_bytes(candidate, checks);
|
2023-02-14 11:38:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface GeneratePrimeOptions {
|
|
|
|
add?: LargeNumberLike | undefined;
|
|
|
|
rem?: LargeNumberLike | undefined;
|
|
|
|
/**
|
|
|
|
* @default false
|
|
|
|
*/
|
|
|
|
safe?: boolean | undefined;
|
|
|
|
bigint?: boolean | undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function generatePrime(
|
|
|
|
size: number,
|
2023-04-27 10:10:59 -04:00
|
|
|
options: GeneratePrimeOptions = {},
|
|
|
|
callback?: (err: Error | null, prime: ArrayBuffer | bigint) => void,
|
2023-02-14 11:38:45 -05:00
|
|
|
) {
|
2023-04-27 10:10:59 -04:00
|
|
|
validateInt32(size, "size", 1);
|
|
|
|
if (typeof options === "function") {
|
|
|
|
callback = options;
|
|
|
|
options = {};
|
|
|
|
}
|
|
|
|
validateFunction(callback, "callback");
|
|
|
|
const {
|
|
|
|
bigint,
|
|
|
|
} = validateRandomPrimeJob(size, options);
|
2023-05-01 11:40:00 -04:00
|
|
|
op_node_gen_prime_async(size).then((prime: Uint8Array) =>
|
2023-04-27 10:10:59 -04:00
|
|
|
bigint ? arrayBufferToUnsignedBigInt(prime.buffer) : prime.buffer
|
|
|
|
).then((prime: ArrayBuffer | bigint) => {
|
|
|
|
callback?.(null, prime);
|
|
|
|
});
|
2023-02-14 11:38:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export function generatePrimeSync(
|
|
|
|
size: number,
|
2023-04-27 10:10:59 -04:00
|
|
|
options: GeneratePrimeOptions = {},
|
|
|
|
): ArrayBuffer | bigint {
|
|
|
|
const {
|
|
|
|
bigint,
|
|
|
|
} = validateRandomPrimeJob(size, options);
|
|
|
|
|
|
|
|
const prime = ops.op_node_gen_prime(size);
|
|
|
|
if (bigint) return arrayBufferToUnsignedBigInt(prime.buffer);
|
|
|
|
return prime.buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
function validateRandomPrimeJob(
|
2023-02-14 11:38:45 -05:00
|
|
|
size: number,
|
|
|
|
options: GeneratePrimeOptions,
|
2023-04-27 10:10:59 -04:00
|
|
|
): GeneratePrimeOptions {
|
|
|
|
validateInt32(size, "size", 1);
|
|
|
|
validateObject(options, "options");
|
|
|
|
|
|
|
|
let {
|
|
|
|
safe = false,
|
|
|
|
bigint = false,
|
|
|
|
add,
|
|
|
|
rem,
|
|
|
|
} = options!;
|
|
|
|
|
|
|
|
validateBoolean(safe, "options.safe");
|
|
|
|
validateBoolean(bigint, "options.bigint");
|
|
|
|
|
|
|
|
if (add !== undefined) {
|
|
|
|
if (typeof add === "bigint") {
|
|
|
|
add = unsignedBigIntToBuffer(add, "options.add");
|
|
|
|
} else if (!isAnyArrayBuffer(add) && !isArrayBufferView(add)) {
|
|
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
|
|
"options.add",
|
|
|
|
[
|
|
|
|
"ArrayBuffer",
|
|
|
|
"TypedArray",
|
|
|
|
"Buffer",
|
|
|
|
"DataView",
|
|
|
|
"bigint",
|
|
|
|
],
|
|
|
|
add,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rem !== undefined) {
|
|
|
|
if (typeof rem === "bigint") {
|
|
|
|
rem = unsignedBigIntToBuffer(rem, "options.rem");
|
|
|
|
} else if (!isAnyArrayBuffer(rem) && !isArrayBufferView(rem)) {
|
|
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
|
|
"options.rem",
|
|
|
|
[
|
|
|
|
"ArrayBuffer",
|
|
|
|
"TypedArray",
|
|
|
|
"Buffer",
|
|
|
|
"DataView",
|
|
|
|
"bigint",
|
|
|
|
],
|
|
|
|
rem,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(@littledivy): safe, add and rem options are not implemented.
|
|
|
|
if (safe || add || rem) {
|
|
|
|
notImplemented("safe, add and rem options are not implemented.");
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
safe,
|
|
|
|
bigint,
|
|
|
|
add,
|
|
|
|
rem,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 48 is the ASCII code for '0', 97 is the ASCII code for 'a'.
|
|
|
|
* @param {number} number An integer between 0 and 15.
|
|
|
|
* @returns {number} corresponding to the ASCII code of the hex representation
|
|
|
|
* of the parameter.
|
|
|
|
*/
|
|
|
|
const numberToHexCharCode = (number: number): number =>
|
|
|
|
(number < 10 ? 48 : 87) + number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {ArrayBuffer} buf An ArrayBuffer.
|
|
|
|
* @return {bigint}
|
|
|
|
*/
|
|
|
|
function arrayBufferToUnsignedBigInt(buf: ArrayBuffer): bigint {
|
|
|
|
const length = buf.byteLength;
|
|
|
|
const chars: number[] = Array(length * 2);
|
|
|
|
const view = new DataView(buf);
|
|
|
|
|
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
const val = view.getUint8(i);
|
|
|
|
chars[2 * i] = numberToHexCharCode(val >> 4);
|
|
|
|
chars[2 * i + 1] = numberToHexCharCode(val & 0xf);
|
|
|
|
}
|
|
|
|
|
|
|
|
return BigInt(`0x${String.fromCharCode(...chars)}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
function unsignedBigIntToBuffer(bigint: bigint, name: string) {
|
|
|
|
if (bigint < 0) {
|
|
|
|
throw new ERR_OUT_OF_RANGE(name, ">= 0", bigint);
|
|
|
|
}
|
|
|
|
|
2023-09-07 09:09:16 -04:00
|
|
|
const hex = StringPrototypeToString(bigint, 16);
|
|
|
|
const padded = StringPrototypePadStart(hex, hex.length + (hex.length % 2), 0);
|
2023-04-27 10:10:59 -04:00
|
|
|
return Buffer.from(padded, "hex");
|
2023-02-14 11:38:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export const randomUUID = () => globalThis.crypto.randomUUID();
|
|
|
|
|
|
|
|
export default {
|
|
|
|
checkPrime,
|
|
|
|
checkPrimeSync,
|
|
|
|
generatePrime,
|
|
|
|
generatePrimeSync,
|
|
|
|
randomUUID,
|
|
|
|
randomInt,
|
|
|
|
randomBytes,
|
|
|
|
randomFill,
|
|
|
|
randomFillSync,
|
|
|
|
};
|