mirror of
https://github.com/oven-sh/bun
synced 2026-02-11 11:29:02 +00:00
156 lines
4.2 KiB
TypeScript
156 lines
4.2 KiB
TypeScript
import { isAscii } from "buffer";
|
|
import fs from "fs";
|
|
import path from "path";
|
|
|
|
// MSVC has a max of 16k characters per string literal
|
|
// Combining string literals didn't support constexpr apparently
|
|
// so we have to do this the gigantic array way
|
|
export function fmtCPPCharArray(str: string, nullTerminated: boolean = true) {
|
|
const normalized = str + "\n";
|
|
|
|
var remain = normalized;
|
|
|
|
const chars =
|
|
"{" +
|
|
remain
|
|
.split("")
|
|
.map(a => a.charCodeAt(0))
|
|
.join(",") +
|
|
(nullTerminated ? ",0" : "") +
|
|
"}";
|
|
return [chars, normalized.length + (nullTerminated ? 1 : 0)] as const;
|
|
}
|
|
|
|
export function addCPPCharArray(str: string, nullTerminated: boolean = true) {
|
|
const normalized = str.trim() + "\n";
|
|
return (
|
|
normalized
|
|
.split("")
|
|
.map(a => a.charCodeAt(0))
|
|
.join(",") + (nullTerminated ? ",0" : "")
|
|
);
|
|
}
|
|
|
|
export function declareASCIILiteral(name: string, value: string) {
|
|
const [chars, count] = fmtCPPCharArray(value, true);
|
|
return `static constexpr const char ${name}Bytes[${count}] = ${chars};
|
|
static constexpr ASCIILiteral ${name} = ASCIILiteral::fromLiteralUnsafe(${name}Bytes);`;
|
|
}
|
|
|
|
export function cap(str: string) {
|
|
return str[0].toUpperCase() + str.slice(1);
|
|
}
|
|
|
|
export function low(str: string) {
|
|
if (str.startsWith("JS")) {
|
|
return "js" + str.slice(2);
|
|
}
|
|
|
|
return str[0].toLowerCase() + str.slice(1);
|
|
}
|
|
|
|
export function readdirRecursive(root: string): string[] {
|
|
const files = fs.readdirSync(root, { withFileTypes: true });
|
|
return files.flatMap(file => {
|
|
const fullPath = path.join(root, file.name);
|
|
return file.isDirectory() ? readdirRecursive(fullPath) : fullPath;
|
|
});
|
|
}
|
|
|
|
export function resolveSyncOrNull(specifier: string, from: string) {
|
|
try {
|
|
return Bun.resolveSync(specifier, from);
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export function checkAscii(str: string) {
|
|
if (!isAscii(Buffer.from(str))) {
|
|
throw new Error(`non-ascii character in string "${str}". this will not be a valid ASCIILiteral`);
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
export function writeIfNotChanged(file: string, contents: string) {
|
|
if (Array.isArray(contents)) contents = contents.join("");
|
|
contents = contents.replaceAll("\r\n", "\n").trim() + "\n";
|
|
|
|
try {
|
|
const oldContents = fs.readFileSync(file, "utf8");
|
|
if (oldContents === contents) {
|
|
return;
|
|
}
|
|
} catch (e) {}
|
|
|
|
try {
|
|
fs.writeFileSync(file, contents);
|
|
} catch (error) {
|
|
fs.mkdirSync(path.dirname(file), { recursive: true });
|
|
fs.writeFileSync(file, contents);
|
|
}
|
|
|
|
if (fs.readFileSync(file, "utf8") !== contents) {
|
|
throw new Error(`Failed to write file ${file}`);
|
|
}
|
|
}
|
|
|
|
export function readdirRecursiveWithExclusionsAndExtensionsSync(
|
|
dir: string,
|
|
exclusions: string[],
|
|
exts: string[],
|
|
): string[] {
|
|
const entries = fs.readdirSync(dir, { withFileTypes: true });
|
|
return entries.flatMap(entry => {
|
|
if (exclusions.includes(entry.name)) return [];
|
|
const fullPath = path.join(dir, entry.name);
|
|
return entry.isDirectory()
|
|
? readdirRecursiveWithExclusionsAndExtensionsSync(fullPath, exclusions, exts)
|
|
: exts.some(ext => fullPath.endsWith(ext))
|
|
? fullPath
|
|
: [];
|
|
});
|
|
}
|
|
|
|
export function pathToUpperSnakeCase(filepath: string) {
|
|
return filepath
|
|
.replace(/^.*?:/, "")
|
|
.split(/[-_./\\]/g)
|
|
.join("_")
|
|
.toUpperCase();
|
|
}
|
|
|
|
export function camelCase(string: string) {
|
|
return string
|
|
.split(/[\s_]/)
|
|
.map((e, i) => (i ? e.charAt(0).toUpperCase() + e.slice(1).toLowerCase() : e.toLowerCase()));
|
|
}
|
|
|
|
export function pascalCase(string: string) {
|
|
return string.split(/[\s_]/).map((e, i) => (i ? e.charAt(0).toUpperCase() + e.slice(1) : e.toLowerCase()));
|
|
}
|
|
|
|
export function argParse(keys: string[]): any {
|
|
const options = {};
|
|
for (const arg of process.argv.slice(2)) {
|
|
if (!arg.startsWith("--")) {
|
|
console.error("Unknown argument " + arg);
|
|
process.exit(1);
|
|
}
|
|
const split = arg.split("=");
|
|
const value = split[1] || "true";
|
|
options[split[0].slice(2)] = value;
|
|
}
|
|
|
|
const unknown = new Set(Object.keys(options));
|
|
for (const key of keys) {
|
|
unknown.delete(key);
|
|
}
|
|
for (const key of unknown) {
|
|
console.error("Unknown argument: --" + key);
|
|
}
|
|
if (unknown.size > 0) process.exit(1);
|
|
return options;
|
|
}
|