Some checks failed
Periodic Merges (6h) / master → staging-nixos (push) Failing after 12m50s
Periodic Merges (6h) / master → staging-next (push) Failing after 12m54s
Periodic Merges (24h) / merge-base(master,staging) → haskell-updates (push) Failing after 11m54s
Periodic Merges (6h) / staging-next → staging (push) Failing after 12m13s
Periodic Merges (24h) / staging-next-25.05 → staging-25.05 (push) Failing after 13m24s
Periodic Merges (24h) / release-25.05 → staging-next-25.05 (push) Failing after 14m28s
95 lines
2.6 KiB
JavaScript
95 lines
2.6 KiB
JavaScript
#!/usr/bin/env node
|
|
const fs = require("fs");
|
|
const path = require("path");
|
|
|
|
async function asyncFilter(arr, pred) {
|
|
const filtered = [];
|
|
for (const elem of arr) {
|
|
if (await pred(elem)) {
|
|
filtered.push(elem);
|
|
}
|
|
}
|
|
return filtered;
|
|
}
|
|
|
|
// Get a list of all _unmanaged_ files in node_modules.
|
|
// This means every file in node_modules that is _not_ a symlink to the Nix store.
|
|
async function getUnmanagedFiles(storePrefix, files) {
|
|
return await asyncFilter(files, async (file) => {
|
|
const filePath = path.join("node_modules", file);
|
|
|
|
// Is file a symlink
|
|
const stat = await fs.promises.lstat(filePath);
|
|
if (!stat.isSymbolicLink()) {
|
|
return true;
|
|
}
|
|
|
|
// Is file in the store
|
|
const linkTarget = await fs.promises.readlink(filePath);
|
|
return !linkTarget.startsWith(storePrefix);
|
|
});
|
|
}
|
|
|
|
async function main() {
|
|
const args = process.argv.slice(2);
|
|
const storePrefix = args[0];
|
|
const sourceModules = args[1];
|
|
|
|
// Ensure node_modules exists
|
|
try {
|
|
await fs.promises.mkdir("node_modules");
|
|
} catch (err) {
|
|
if (err.code !== "EEXIST") {
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
const files = await fs.promises.readdir("node_modules");
|
|
|
|
// Get deny list of files that we don't manage.
|
|
// We do manage nix store symlinks, but not other files.
|
|
// For example: If a .vite was present in both our
|
|
// source node_modules and the non-store node_modules we don't want to overwrite
|
|
// the non-store one.
|
|
const unmanaged = await getUnmanagedFiles(storePrefix, files);
|
|
const managed = new Set(files.filter((file) => ! unmanaged.includes(file)));
|
|
|
|
const sourceFiles = await fs.promises.readdir(sourceModules);
|
|
await Promise.all(
|
|
sourceFiles.map(async (file) => {
|
|
const sourcePath = path.join(sourceModules, file);
|
|
const targetPath = path.join("node_modules", file);
|
|
|
|
// Skip file if it's not a symlink to a store path
|
|
if (unmanaged.includes(file)) {
|
|
console.log(`'${targetPath}' exists, cowardly refusing to link.`);
|
|
return;
|
|
}
|
|
|
|
// Don't unlink this file, we just wrote it.
|
|
managed.delete(file);
|
|
|
|
// Link file
|
|
try {
|
|
await fs.promises.symlink(sourcePath, targetPath);
|
|
} catch (err) {
|
|
// If the target file already exists remove it and try again
|
|
if (err.code !== "EEXIST") {
|
|
throw err;
|
|
}
|
|
await fs.promises.unlink(targetPath);
|
|
await fs.promises.symlink(sourcePath, targetPath);
|
|
}
|
|
})
|
|
);
|
|
|
|
// Clean up store symlinks not included in this generation of node_modules
|
|
await Promise.all(
|
|
Array.from(managed).map((file) =>
|
|
fs.promises.unlink(path.join("node_modules", file)),
|
|
)
|
|
);
|
|
}
|
|
|
|
main();
|