import type { Faker } from '.'; import { FakerError } from './errors/faker-error'; import { deprecated } from './internal/deprecated'; /** * Method to reduce array of characters. * * @param arr existing array of characters * @param values array of characters which should be removed * @returns new array without banned characters */ function arrayRemove(arr: T[], values: T[]): T[] { values.forEach((value) => { arr = arr.filter((ele) => ele !== value); }); return arr; } /** * Generates random values of different kinds. Some methods are deprecated and have been moved to dedicated modules. */ export class Random { constructor(private readonly faker: Faker) { // Bind `this` so namespaced is working correctly for (const name of Object.getOwnPropertyNames(Random.prototype)) { if (name === 'constructor' || typeof this[name] !== 'function') { continue; } this[name] = this[name].bind(this); } } /** * Returns a single random number between zero and the given max value or the given range with the specified precision. * The bounds are inclusive. * * @param options Maximum value or options object. * @param options.min Lower bound for generated number. Defaults to `0`. * @param options.max Upper bound for generated number. Defaults to `99999`. * @param options.precision Precision of the generated number. Defaults to `1`. * * @see faker.datatype.number() * * @example * faker.random.number() // 55422 * faker.random.number(100) // 52 * faker.random.number({ min: 1000000 }) // 431433 * faker.random.number({ max: 100 }) // 42 * faker.random.number({ precision: 0.01 }) // 64246.18 * faker.random.number({ min: 10, max: 100, precision: 0.01 }) // 36.94 * * @deprecated */ number( options?: number | { min?: number; max?: number; precision?: number } ): number { deprecated({ deprecated: 'faker.random.number()', proposed: 'faker.datatype.number()', // since: 'v5.0.0', (?) until: 'v7.0.0', }); return this.faker.datatype.number(options); } /** * Returns a single random floating-point number for the given precision or range and precision. * * @param options Precision or options object. * @param options.min Lower bound for generated number. Defaults to `0`. * @param options.max Upper bound for generated number. Defaults to `99999`. * @param options.precision Precision of the generated number. Defaults to `0.01`. * * @see faker.datatype.float() * * @example * faker.random.float() // 51696.36 * faker.random.float(0.1) // 52023.2 * faker.random.float({ min: 1000000 }) // 212859.76 * faker.random.float({ max: 100 }) // 28.11 * faker.random.float({ precision: 0.1 }) // 84055.3 * faker.random.float({ min: 10, max: 100, precision: 0.001 }) // 57.315 * * @deprecated */ float( options?: number | { min?: number; max?: number; precision?: number } ): number { deprecated({ deprecated: 'faker.random.float()', proposed: 'faker.datatype.float()', // since: 'v5.0.0', (?) until: 'v7.0.0', }); return this.faker.datatype.float(options); } /** * Returns random element from the given array. * * @template T The type of the entries to pick from. * @param array Array to pick the value from. Defaults to `['a', 'b', 'c']`. * * @example * faker.random.arrayElement() // 'b' * faker.random.arrayElement(['cat', 'dog', 'mouse']) // 'dog' */ arrayElement( array: ReadonlyArray = ['a', 'b', 'c'] as unknown as ReadonlyArray ): T { const index = array.length > 1 ? this.faker.datatype.number({ max: array.length - 1 }) : 0; return array[index]; } /** * Returns a subset with random elements of the given array in random order. * * @template T The type of the entries to pick from. * @param array Array to pick the value from. Defaults to `['a', 'b', 'c']`. * @param count Number of elements to pick. * When not provided, random number of elements will be picked. * When value exceeds array boundaries, it will be limited to stay inside. * * @example * faker.random.arrayElements() // ['b', 'c'] * faker.random.arrayElements(['cat', 'dog', 'mouse']) // ['mouse', 'cat'] * faker.random.arrayElements([1, 2, 3, 4, 5], 2) // [4, 2] */ arrayElements( array: ReadonlyArray = ['a', 'b', 'c'] as unknown as ReadonlyArray, count?: number ): T[] { if (typeof count !== 'number') { count = this.faker.datatype.number({ min: 1, max: array.length }); } else if (count > array.length) { count = array.length; } else if (count < 0) { count = 0; } const arrayCopy = array.slice(0); let i = array.length; const min = i - count; let temp: T; let index: number; while (i-- > min) { index = Math.floor( (i + 1) * this.faker.datatype.float({ min: 0, max: 0.99 }) ); temp = arrayCopy[index]; arrayCopy[index] = arrayCopy[i]; arrayCopy[i] = temp; } return arrayCopy.slice(min); } /** * Returns a random key or value from given object. * * @template T The type of `Record` to pick from. * @template K The keys of `T`. * @param object The object to get the keys or values from. * @param field If this is set to `'key'`, this method will a return a random key of the given instance. * If this is set to `'value'`, this method will a return a random value of the given instance. * Defaults to `'value'`. * * @example * const object = { keyA: 'valueA', keyB: 42 }; * faker.random.objectElement(object) // 42 * faker.random.objectElement(object, 'key') // 'keyB' * faker.random.objectElement(object, 'value') // 'valueA' */ objectElement, K extends keyof T>( object: T, field: 'key' ): K; objectElement, K extends keyof T>( object: T, field?: unknown ): T[K]; /** * Returns a random key or value from given object. * * @template T The type of `Record` to pick from. * @template K The keys of `T`. * @param object The object to get the keys or values from. * @param field If this is set to `'key'`, this method will a return a random key of the given instance. * If this is set to `'value'`, this method will a return a random value of the given instance. * Defaults to `'value'`. * * @example * const object = { keyA: 'valueA', keyB: 42 }; * faker.random.objectElement(object) // 42 * faker.random.objectElement(object, 'key') // 'keyB' * faker.random.objectElement(object, 'value') // 'valueA' */ objectElement, K extends keyof T>( object: T, field?: 'key' | 'value' ): K | T[K]; objectElement, K extends keyof T>( object = { foo: 'bar', too: 'car' } as unknown as T, field = 'value' ): K | T[K] { const array: Array = Object.keys(object); const key = this.faker.random.arrayElement(array); return field === 'key' ? (key as K) : (object[key] as T[K]); } /** * Returns a UUID v4 ([Universally Unique Identifier](https://en.wikipedia.org/wiki/Universally_unique_identifier)). * * @see faker.datatype.uuid() * * @example * faker.random.uuid() // '4136cd0b-d90b-4af7-b485-5d1ded8db252' * * @deprecated */ uuid(): string { deprecated({ deprecated: 'faker.random.uuid()', proposed: 'faker.datatype.uuid()', // since: 'v5.0.0', (?) until: 'v7.0.0', }); return this.faker.datatype.uuid(); } /** * Returns the boolean value `true` or `false`. * * @see faker.datatype.boolean() * * @example * faker.random.boolean() // false * * @deprecated */ boolean(): boolean { deprecated({ deprecated: 'faker.random.boolean()', proposed: 'faker.datatype.boolean()', // since: 'v5.0.0', (?) until: 'v7.0.0', }); return this.faker.datatype.boolean(); } /** * Returns random word. * * @example * faker.random.word() // 'Seamless' */ word(): string { const wordMethods = [ this.faker.commerce.department, this.faker.commerce.productName, this.faker.commerce.productAdjective, this.faker.commerce.productMaterial, this.faker.commerce.product, this.faker.commerce.color, this.faker.company.catchPhraseAdjective, this.faker.company.catchPhraseDescriptor, this.faker.company.catchPhraseNoun, this.faker.company.bsAdjective, this.faker.company.bsBuzz, this.faker.company.bsNoun, this.faker.address.streetSuffix, this.faker.address.county, this.faker.address.country, this.faker.address.state, this.faker.finance.accountName, this.faker.finance.transactionType, this.faker.finance.currencyName, this.faker.hacker.noun, this.faker.hacker.verb, this.faker.hacker.adjective, this.faker.hacker.ingverb, this.faker.hacker.abbreviation, this.faker.name.jobDescriptor, this.faker.name.jobArea, this.faker.name.jobType, ]; const bannedChars = [ '!', '#', '%', '&', '*', ')', '(', '+', '=', '.', '<', '>', '{', '}', '[', ']', ':', ';', "'", '"', '_', '-', ]; let result: string; do { // randomly pick from the many faker methods that can generate words const randomWordMethod = this.faker.random.arrayElement(wordMethods); result = randomWordMethod(); } while (!result || bannedChars.some((char) => result.includes(char))); return this.faker.random.arrayElement(result.split(' ')); } /** * Returns string with set of random words. * * @param count Number of words. Defaults to a random value between `1` and `3`. * * @example * faker.random.words() // 'neural' * faker.random.words(5) // 'copy Handcrafted bus client-server Point' */ words(count?: number): string { const words: string[] = []; if (count == null) { count = this.faker.datatype.number({ min: 1, max: 3 }); } for (let i = 0; i < count; i++) { words.push(this.faker.random.word()); } return words.join(' '); } /** * Returns a random image url. * * @see faker.random.image() * * @example * faker.random.image() // 'http://placeimg.com/640/480/animals' * * @deprecated */ image(): string { deprecated({ deprecated: 'faker.random.image()', proposed: 'faker.image.image()', // since: 'v5.0.0', (?) until: 'v7.0.0', }); return this.faker.image.image(); } /** * Returns a random locale, that is available in this faker instance. * You can use the returned locale with `faker.setLocale(result)`. * * @example * faker.random.locale() // 'el' */ locale(): string { return this.faker.random.arrayElement(Object.keys(this.faker.locales)); } /** * Generating a string consisting of lower/upper alpha characters based on count and upcase options. * * @param options Either the number of characters or an options instance. Defaults to `{ count: 1, upcase: false, bannedChars: [] }`. * @param options.count The number of characters to generate. Defaults to `1`. * @param options.upcase If true, the result will be uppercase. If false, it will be lowercase. Defaults to `false`. * @param options.bannedChars An array with characters to exclude. Defaults to `[]`. * * @example * faker.random.alpha() // 'b' * faker.random.alpha(10) // 'qccrabobaf' * faker.random.alpha({ count: 5, upcase: true, bannedChars: ['a'] }) // 'DTCIC' */ // TODO @Shinigami92 2022-02-14: Tests covered `(count, options)`, but they were never typed like that alpha( options?: | number | { count?: number; upcase?: boolean; bannedChars?: string[] } ): string { if (options == null) { options = { count: 1, }; } else if (typeof options === 'number') { options = { count: options, }; } else if (options.count == null) { options.count = 1; } if (options.upcase == null) { options.upcase = false; } if (options.bannedChars == null) { options.bannedChars = []; } let wholeString = ''; let charsArray = [ 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', ]; // TODO @Shinigami92 2022-01-11: A default empty array gets assigned above, we should check the length against 0 or not here if (options.bannedChars) { charsArray = arrayRemove(charsArray, options.bannedChars); } for (let i = 0; i < options.count; i++) { wholeString += this.faker.random.arrayElement(charsArray); } return options.upcase ? wholeString.toUpperCase() : wholeString; } /** * Generating a string consisting of lower/upper alpha characters and digits based on count and upcase options. * * @param count The number of characters and digits to generate. Defaults to `1`. * @param options The options to use. Defaults to `{ bannedChars: [] }`. * @param options.bannedChars An array of characters and digits which should be banned in the generated string. Defaults to `[]`. * * @example * faker.random.alphaNumeric() // '2' * faker.random.alphaNumeric(5) // '3e5v7' * faker.random.alphaNumeric(5, { bannedChars: ["a"] }) // 'xszlm' */ alphaNumeric( count: number = 1, options: { bannedChars?: string[] } = {} ): string { if (options.bannedChars == null) { options.bannedChars = []; } let wholeString = ''; let charsArray = [ '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', ]; if (options.bannedChars) { charsArray = arrayRemove(charsArray, options.bannedChars); } if (charsArray.length === 0) { throw new FakerError( 'Unable to generate string, because all possible characters are banned.' ); } for (let i = 0; i < count; i++) { wholeString += this.faker.random.arrayElement(charsArray); } return wholeString; } /** * Returns a hexadecimal number. * * @param count Length of the generated number. Defaults to `1`. * * @see faker.datatype.hexadecimal() * * @example * faker.random.hexaDecimal() // '0xb' * faker.random.hexaDecimal(10) // '0xaE13F044fb' * * @deprecated */ hexaDecimal(count?: number): string { deprecated({ deprecated: 'faker.random.hexaDecimal()', proposed: 'faker.datatype.hexadecimal()', // since: 'v5.0.0', (?) until: 'v7.0.0', }); return this.faker.datatype.hexadecimal(count); } }