X-Git-Url: https://git.piment-noir.org/?a=blobdiff_plain;f=src%2Futils%2FUtils.ts;h=808ef380d170b2a5a11acfeaa96e7d23b49e1040;hb=c2a11566c6855de3b1f48f701be2c8779c0093a6;hp=1a93669dbc334f4b7c2b1d18bd7f049d0625cc4f;hpb=85cce27f97abb607fbc4a2c4c6b335680dd65c2b;p=e-mobility-charging-stations-simulator.git diff --git a/src/utils/Utils.ts b/src/utils/Utils.ts index 1a93669d..808ef380 100644 --- a/src/utils/Utils.ts +++ b/src/utils/Utils.ts @@ -1,4 +1,4 @@ -import { randomBytes, randomInt, randomUUID } from 'node:crypto'; +import { randomBytes, randomInt, randomUUID, webcrypto } from 'node:crypto'; import { inspect } from 'node:util'; import { @@ -12,7 +12,6 @@ import { minutesToSeconds, secondsToMilliseconds, } from 'date-fns'; -import deepClone from 'deep-clone'; import { Constants } from './Constants'; import { type TimestampedData, WebSocketCloseEventStatusString } from '../types'; @@ -215,8 +214,44 @@ type CloneableData = | CloneableData[] | { [key: string]: CloneableData }; +type FormatKey = (key: string) => string; + +const deepClone = ( + value: I, + formatKey?: FormatKey, + refs: Map = new Map(), +): O => { + const ref = refs.get(value); + if (ref !== undefined) { + return ref; + } + if (Array.isArray(value)) { + const clone: CloneableData[] = []; + refs.set(value, clone as O); + for (let i = 0; i < value.length; i++) { + clone[i] = deepClone(value[i], formatKey, refs); + } + return clone as O; + } + if (value instanceof Date) { + return new Date(value.valueOf()) as O; + } + if (typeof value !== 'object' || value === null) { + return value as unknown as O; + } + const clone: Record = {}; + refs.set(value, clone as O); + for (const key of Object.keys(value)) { + clone[typeof formatKey === 'function' ? formatKey(key) : key] = deepClone( + value[key], + formatKey, + refs, + ); + } + return clone as O; +}; + export const cloneObject = (object: T): T => { - // eslint-disable-next-line @typescript-eslint/no-unsafe-call return deepClone(object as CloneableData) as T; }; @@ -280,12 +315,12 @@ export const insertAt = (str: string, subStr: string, pos: number): string => * Computes the retry delay in milliseconds using an exponential backoff algorithm. * * @param retryNumber - the number of retries that have already been attempted - * @param maxDelayRatio - the maximum ratio of the delay that can be randomized + * @param delayFactor - the base delay factor in milliseconds * @returns delay in milliseconds */ -export const exponentialDelay = (retryNumber = 0, maxDelayRatio = 0.2): number => { - const delay = Math.pow(2, retryNumber) * 100; - const randomSum = delay * maxDelayRatio * secureRandom(); // 0-(maxDelayRatio*100)% of the delay +export const exponentialDelay = (retryNumber = 0, delayFactor = 100): number => { + const delay = Math.pow(2, retryNumber) * delayFactor; + const randomSum = delay * 0.2 * secureRandom(); // 0-20% of the delay return delay + randomSum; }; @@ -301,7 +336,7 @@ export const promiseWithTimeout = async ( /* This is intentional */ }, ): Promise => { - // Create a timeout promise that rejects in timeout milliseconds + // Creates a timeout promise that rejects in timeout milliseconds const timeoutPromise = new Promise((_, reject) => { setTimeout(() => { if (isPromisePending(promise)) { @@ -319,10 +354,10 @@ export const promiseWithTimeout = async ( /** * Generates a cryptographically secure random number in the [0,1[ range * - * @returns + * @returns A number in the [0,1[ range */ export const secureRandom = (): number => { - return randomBytes(4).readUInt32LE() / 0x100000000; + return webcrypto.getRandomValues(new Uint32Array(1))[0] / 0x100000000; }; export const JSONStringifyWithMapSupport = ( @@ -382,3 +417,24 @@ export const isArraySorted = (array: T[], compareFn: (a: T, b: T) => number): } return true; }; + +// eslint-disable-next-line @typescript-eslint/no-explicit-any +export const once = ( + fn: (...args: A) => R, + context: T, +): ((...args: A) => R) => { + let result: R; + return (...args: A) => { + if (fn) { + result = fn.apply(context, args); + (fn as unknown as undefined) = (context as unknown as undefined) = undefined; + } + return result; + }; +}; + +export const min = (...args: number[]): number => + args.reduce((minimum, num) => (minimum < num ? minimum : num), Infinity); + +export const max = (...args: number[]): number => + args.reduce((maximum, num) => (maximum > num ? maximum : num), -Infinity);