From e8faf7252db932f5aaf39a19110065211c9e7bf7 Mon Sep 17 00:00:00 2001 From: Shinigami92 Date: Tue, 11 Jan 2022 09:59:20 +0100 Subject: [PATCH 1/4] feat: migrate fake --- src/fake.ts | 102 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 102 insertions(+) create mode 100644 src/fake.ts diff --git a/src/fake.ts b/src/fake.ts new file mode 100644 index 00000000000..f7ca18ce114 --- /dev/null +++ b/src/fake.ts @@ -0,0 +1,102 @@ +import type { Faker } from '.'; + +/** + * Generator method for combining faker methods based on string input + */ +export class Fake { + constructor(private readonly faker: Faker) {} + + /** + * Generator method for combining faker methods based on string input + * + * __Example:__ + * + * ``` + * console.log(faker.fake('{{name.lastName}}, {{name.firstName}} {{name.suffix}}')); + * // outputs: "Marks, Dean Sr." + * ``` + * + * This will interpolate the format string with the value of methods + * [name.lastName]{@link faker.name.lastName}, [name.firstName]{@link faker.name.firstName}, + * and [name.suffix]{@link faker.name.suffix} + * + * @method faker.fake + * @param str + */ + fake(str: string) { + // setup default response as empty string + let res = ''; + + // if incoming str parameter is not provided, return error message + if (typeof str !== 'string' || str.length === 0) { + throw new Error('string parameter is required!'); + } + + // find first matching {{ and }} + const start = str.search('{{'); + const end = str.search('}}'); + + // if no {{ and }} is found, we are done + if (start === -1 || end === -1) { + return str; + } + + // console.log('attempting to parse', str); + + // extract method name from between the {{ }} that we found + // for example: {{name.firstName}} + const token = str.substr(start + 2, end - start - 2); + let method = token.replace('}}', '').replace('{{', ''); + + // console.log('method', method) + + // extract method parameters + const regExp = /\(([^)]+)\)/; + const matches = regExp.exec(method); + let parameters = ''; + if (matches) { + method = method.replace(regExp, ''); + parameters = matches[1]; + } + + // split the method into module and function + const parts = method.split('.'); + + if (typeof this.faker[parts[0]] === 'undefined') { + throw new Error('Invalid module: ' + parts[0]); + } + + if (typeof this.faker[parts[0]][parts[1]] === 'undefined') { + throw new Error('Invalid method: ' + parts[0] + '.' + parts[1]); + } + + // assign the function from the module.function namespace + const fn = this.faker[parts[0]][parts[1]]; + + // If parameters are populated here, they are always going to be of string type + // since we might actually be dealing with an object or array, + // we always attempt to the parse the incoming parameters into JSON + let params: any; + // Note: we experience a small performance hit here due to JSON.parse try / catch + // If anyone actually needs to optimize this specific code path, please open a support issue on github + try { + params = JSON.parse(parameters); + } catch (err) { + // since JSON.parse threw an error, assume parameters was actually a string + params = parameters; + } + + let result: string; + if (typeof params === 'string' && params.length === 0) { + result = fn.call(this); + } else { + result = fn.call(this, params); + } + + // replace the found tag with the returned fake value + res = str.replace('{{' + token + '}}', result); + + // return the response recursively until we are done finding all tags + return this.fake(res); + } +} From 100a862ba705b40b8e115caa8bb5f5bbd29fecf0 Mon Sep 17 00:00:00 2001 From: Shinigami92 Date: Tue, 11 Jan 2022 10:06:56 +0100 Subject: [PATCH 2/4] chore: bind methods --- src/fake.ts | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/src/fake.ts b/src/fake.ts index f7ca18ce114..fbdfcad1c4e 100644 --- a/src/fake.ts +++ b/src/fake.ts @@ -4,7 +4,15 @@ import type { Faker } from '.'; * Generator method for combining faker methods based on string input */ export class Fake { - constructor(private readonly faker: Faker) {} + constructor(private readonly faker: Faker) { + // Bind `this` so namespaced is working correctly + for (const name of Object.getOwnPropertyNames(Fake.prototype)) { + if (name === 'constructor' || typeof this[name] !== 'function') { + continue; + } + this[name] = this[name].bind(this); + } + } /** * Generator method for combining faker methods based on string input From e37dbb1a1ca3cef904ad7b326315bf5b33690b39 Mon Sep 17 00:00:00 2001 From: Shinigami92 Date: Tue, 11 Jan 2022 10:07:13 +0100 Subject: [PATCH 3/4] chore: register the fake function --- src/index.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/index.ts b/src/index.ts index 768ea2efd23..b8fef5e8c96 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,5 +1,6 @@ import { Datatype } from './datatype'; import { _Date } from './date'; +import { Fake } from './fake'; import { Git } from './git'; import { Hacker } from './hacker'; import { Helpers } from './helpers'; @@ -155,7 +156,7 @@ export class Faker { seedValue?: any[] | any; - readonly fake = new (require('./fake'))(this).fake; + readonly fake: Fake['fake'] = new Fake(this).fake; readonly unique = new (require('./unique'))(this).unique; readonly mersenne: Mersenne = new Mersenne(); From 830fc0ed830be6810500ce0a8afaf3b123b36ef8 Mon Sep 17 00:00:00 2001 From: Shinigami Date: Tue, 11 Jan 2022 10:27:17 +0100 Subject: [PATCH 4/4] chore: add function return type Co-authored-by: Honza Machala --- src/fake.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/fake.ts b/src/fake.ts index fbdfcad1c4e..e435ce8966e 100644 --- a/src/fake.ts +++ b/src/fake.ts @@ -31,7 +31,7 @@ export class Fake { * @method faker.fake * @param str */ - fake(str: string) { + fake(str: string): string { // setup default response as empty string let res = '';