|
|
|
|
|
import fs from 'fs' |
|
import path from 'path' |
|
|
|
export function indentRecursive(node, indent = 0) { |
|
node.each && |
|
node.each((child, i) => { |
|
if (!child.raws.before || !child.raws.before.trim() || child.raws.before.includes('\n')) { |
|
child.raws.before = `\n${node.type !== 'rule' && i > 0 ? '\n' : ''}${' '.repeat(indent)}` |
|
} |
|
child.raws.after = `\n${' '.repeat(indent)}` |
|
indentRecursive(child, indent + 1) |
|
}) |
|
} |
|
|
|
export function formatNodes(root) { |
|
indentRecursive(root) |
|
if (root.first) { |
|
root.first.raws.before = '' |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export async function readFileWithRetries(path, tries = 5) { |
|
for (let n = 0; n <= tries; n++) { |
|
try { |
|
return await fs.promises.readFile(path, 'utf8') |
|
} catch (err) { |
|
if (n !== tries) { |
|
if (err.code === 'ENOENT' || err.code === 'EBUSY') { |
|
await new Promise((resolve) => setTimeout(resolve, 10)) |
|
|
|
continue |
|
} |
|
} |
|
|
|
throw err |
|
} |
|
} |
|
} |
|
|
|
export function drainStdin() { |
|
return new Promise((resolve, reject) => { |
|
let result = '' |
|
process.stdin.on('data', (chunk) => { |
|
result += chunk |
|
}) |
|
process.stdin.on('end', () => resolve(result)) |
|
process.stdin.on('error', (err) => reject(err)) |
|
}) |
|
} |
|
|
|
export async function outputFile(file, newContents) { |
|
try { |
|
let currentContents = await fs.promises.readFile(file, 'utf8') |
|
if (currentContents === newContents) { |
|
return |
|
} |
|
} catch {} |
|
|
|
|
|
await fs.promises.mkdir(path.dirname(file), { recursive: true }) |
|
await fs.promises.writeFile(file, newContents, 'utf8') |
|
} |
|
|