|
import fs from 'node:fs'; |
|
import path from 'node:path'; |
|
|
|
|
|
export function mkdirp(dir) { |
|
try { |
|
fs.mkdirSync(dir, { recursive: true }); |
|
} catch ( e) { |
|
if (e.code === 'EEXIST') { |
|
if (!fs.statSync(dir).isDirectory()) { |
|
throw new Error(`Cannot create directory ${dir}, a file already exists at this position`); |
|
} |
|
return; |
|
} |
|
throw e; |
|
} |
|
} |
|
|
|
|
|
export function rimraf(path) { |
|
fs.rmSync(path, { force: true, recursive: true }); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export function copy(source, target, opts = {}) { |
|
if (!fs.existsSync(source)) return []; |
|
|
|
|
|
const files = []; |
|
|
|
const prefix = posixify(target) + '/'; |
|
|
|
const regex = opts.replace |
|
? new RegExp(`\\b(${Object.keys(opts.replace).join('|')})\\b`, 'g') |
|
: null; |
|
|
|
|
|
|
|
|
|
|
|
function go(from, to) { |
|
if (opts.filter && !opts.filter(path.basename(from))) return; |
|
|
|
const stats = fs.statSync(from); |
|
|
|
if (stats.isDirectory()) { |
|
fs.readdirSync(from).forEach((file) => { |
|
go(path.join(from, file), path.join(to, file)); |
|
}); |
|
} else { |
|
mkdirp(path.dirname(to)); |
|
|
|
if (opts.replace) { |
|
const data = fs.readFileSync(from, 'utf-8'); |
|
fs.writeFileSync( |
|
to, |
|
data.replace( |
|
(regex), |
|
(_match, key) => (opts.replace)[key] |
|
) |
|
); |
|
} else { |
|
fs.copyFileSync(from, to); |
|
} |
|
|
|
files.push(to === target ? posixify(path.basename(to)) : posixify(to).replace(prefix, '')); |
|
} |
|
} |
|
|
|
go(source, target); |
|
|
|
return files; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export function walk(cwd, dirs = false) { |
|
|
|
const all_files = []; |
|
|
|
|
|
function walk_dir(dir) { |
|
const files = fs.readdirSync(path.join(cwd, dir)); |
|
|
|
for (const file of files) { |
|
const joined = path.join(dir, file); |
|
const stats = fs.statSync(path.join(cwd, joined)); |
|
if (stats.isDirectory()) { |
|
if (dirs) all_files.push(joined); |
|
walk_dir(joined); |
|
} else { |
|
all_files.push(joined); |
|
} |
|
} |
|
} |
|
|
|
return walk_dir(''), all_files; |
|
} |
|
|
|
|
|
export function posixify(str) { |
|
return str.replace(/\\/g, '/'); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
export function join_relative(...str) { |
|
let result = posixify(path.join(...str)); |
|
if (!result.startsWith('.')) { |
|
result = `./${result}`; |
|
} |
|
return result; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export function relative_path(from, to) { |
|
return join_relative(path.relative(from, to)); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
export function to_fs(str) { |
|
str = posixify(str); |
|
return `/@fs${ |
|
// Windows/Linux separation - Windows starts with a drive letter, we need a / in front there |
|
str.startsWith('/') ? '' : '/' |
|
}${str}`; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
export function from_fs(str) { |
|
str = posixify(str); |
|
if (!str.startsWith('/@fs')) return str; |
|
|
|
str = str.slice(4); |
|
|
|
return str[2] === ':' && /[A-Z]/.test(str[1]) ? str.slice(1) : str; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
export function resolve_entry(entry) { |
|
if (fs.existsSync(entry)) { |
|
const stats = fs.statSync(entry); |
|
if (stats.isDirectory()) { |
|
return resolve_entry(path.join(entry, 'index')); |
|
} |
|
|
|
return entry; |
|
} else { |
|
const dir = path.dirname(entry); |
|
|
|
if (fs.existsSync(dir)) { |
|
const base = path.basename(entry); |
|
const files = fs.readdirSync(dir); |
|
|
|
const found = files.find((file) => file.replace(/\.(js|ts)$/, '') === base); |
|
|
|
if (found) return path.join(dir, found); |
|
} |
|
} |
|
|
|
return null; |
|
} |
|
|
|
|
|
export function read(file) { |
|
return fs.readFileSync(file, 'utf-8'); |
|
} |
|
|