Next.js website for Rocky Mountain Vending company featuring: - Product catalog with Stripe integration - Service areas and parts pages - Admin dashboard with Clerk authentication - SEO optimized pages with JSON-LD structured data Co-authored-by: Cursor <cursoragent@cursor.com>
95 lines
2.3 KiB
Text
95 lines
2.3 KiB
Text
/**
|
|
* @license
|
|
* Copyright 2023 Google Inc.
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
const createdFunctions = new Map<string, (...args: unknown[]) => unknown>();
|
|
|
|
/**
|
|
* Creates a function from a string.
|
|
*
|
|
* @internal
|
|
*/
|
|
export const createFunction = (
|
|
functionValue: string,
|
|
): ((...args: unknown[]) => unknown) => {
|
|
let fn = createdFunctions.get(functionValue);
|
|
if (fn) {
|
|
return fn;
|
|
}
|
|
fn = new Function(`return ${functionValue}`)() as (
|
|
...args: unknown[]
|
|
) => unknown;
|
|
createdFunctions.set(functionValue, fn);
|
|
return fn;
|
|
};
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
export function stringifyFunction(fn: (...args: never) => unknown): string {
|
|
let value = fn.toString();
|
|
if (
|
|
value.match(/^(async )*function(\(|\s)/) ||
|
|
value.match(/^(async )*function\s*\*\s*/)
|
|
) {
|
|
return value;
|
|
}
|
|
const isArrow =
|
|
value.startsWith('(') ||
|
|
value.match(/^async\s*\(/) ||
|
|
value.match(
|
|
/^(async)*\s*(?:[$_\p{ID_Start}])(?:[$\u200C\u200D\p{ID_Continue}])*\s*=>/u,
|
|
);
|
|
if (isArrow) {
|
|
return value;
|
|
}
|
|
// This means we might have a function shorthand (e.g. `test(){}`). Let's
|
|
// try prefixing.
|
|
let prefix = 'function ';
|
|
if (value.startsWith('async ')) {
|
|
prefix = `async ${prefix}`;
|
|
value = value.substring('async '.length);
|
|
}
|
|
return `${prefix}${value}`;
|
|
}
|
|
|
|
/**
|
|
* Replaces `PLACEHOLDER`s with the given replacements.
|
|
*
|
|
* All replacements must be valid JS code.
|
|
*
|
|
* @example
|
|
*
|
|
* ```ts
|
|
* interpolateFunction(() => PLACEHOLDER('test'), {test: 'void 0'});
|
|
* // Equivalent to () => void 0
|
|
* ```
|
|
*
|
|
* @internal
|
|
*/
|
|
export const interpolateFunction = <T extends (...args: never[]) => unknown>(
|
|
fn: T,
|
|
replacements: Record<string, string>,
|
|
): T => {
|
|
let value = stringifyFunction(fn);
|
|
for (const [name, jsValue] of Object.entries(replacements)) {
|
|
value = value.replace(
|
|
new RegExp(`PLACEHOLDER\\(\\s*(?:'${name}'|"${name}")\\s*\\)`, 'g'),
|
|
// Wrapping this ensures tersers that accidentally inline PLACEHOLDER calls
|
|
// are still valid. Without, we may get calls like ()=>{...}() which is
|
|
// not valid.
|
|
`(${jsValue})`,
|
|
);
|
|
}
|
|
return createFunction(value) as unknown as T;
|
|
};
|
|
|
|
declare global {
|
|
/**
|
|
* Used for interpolation with {@link interpolateFunction}.
|
|
*
|
|
* @internal
|
|
*/
|
|
function PLACEHOLDER<T>(name: string): T;
|
|
}
|