mirror of
https://github.com/NixOS/nixpkgs.git
synced 2025-01-23 21:33:49 +00:00
9c7ff7277c
`importNpmLock.buildNodeModules` returns a derivation with a pre-built `node_modules` directory, as imported by `importNpmLock`. This is to be used together with `importNpmLock.hooks.linkNodeModulesHook` to facilitate `nix-shell`/`nix develop` based development workflows: ```nix pkgs.mkShell { packages = [ importNpmLock.hooks.linkNodeModulesHook nodejs ]; npmDeps = importNpmLock.buildNodeModules { npmRoot = ./.; inherit nodejs; }; } ``` will create a development shell where a `node_modules` directory is created & packages symlinked to the Nix store when activated. This code is adapted from https://github.com/adisbladis/buildNodeModules
97 lines
2.8 KiB
JavaScript
97 lines
2.8 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 to a temporary dummy path and rename.
|
|
// This is to get some degree of atomicity.
|
|
try {
|
|
await fs.promises.symlink(sourcePath, targetPath + "-nix-hook-temp");
|
|
} catch (err) {
|
|
if (err.code !== "EEXIST") {
|
|
throw err;
|
|
}
|
|
|
|
await fs.promises.unlink(targetPath + "-nix-hook-temp");
|
|
await fs.promises.symlink(sourcePath, targetPath + "-nix-hook-temp");
|
|
}
|
|
await fs.promises.rename(targetPath + "-nix-hook-temp", 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();
|