node-env.nix
292 lines
| 10.9 KiB
| text/x-nix
|
NixLexer
/ pkgs / node-env.nix
r707 | # This file originates from node2nix | |||
{stdenv, python, nodejs, utillinux, runCommand, writeTextFile}: | ||||
let | ||||
# Create a tar wrapper that filters all the 'Ignoring unknown extended header keyword' noise | ||||
tarWrapper = runCommand "tarWrapper" {} '' | ||||
mkdir -p $out/bin | ||||
cat > $out/bin/tar <<EOF | ||||
#! ${stdenv.shell} -e | ||||
$(type -p tar) "\$@" --warning=no-unknown-keyword | ||||
EOF | ||||
chmod +x $out/bin/tar | ||||
''; | ||||
# Function that generates a TGZ file from a NPM project | ||||
buildNodeSourceDist = | ||||
{ name, version, src, ... }: | ||||
stdenv.mkDerivation { | ||||
name = "node-tarball-${name}-${version}"; | ||||
inherit src; | ||||
buildInputs = [ nodejs ]; | ||||
buildPhase = '' | ||||
export HOME=$TMPDIR | ||||
tgzFile=$(npm pack) | ||||
''; | ||||
installPhase = '' | ||||
mkdir -p $out/tarballs | ||||
mv $tgzFile $out/tarballs | ||||
mkdir -p $out/nix-support | ||||
echo "file source-dist $out/tarballs/$tgzFile" >> $out/nix-support/hydra-build-products | ||||
''; | ||||
}; | ||||
includeDependencies = {dependencies}: | ||||
stdenv.lib.optionalString (dependencies != []) | ||||
(stdenv.lib.concatMapStrings (dependency: | ||||
'' | ||||
# Bundle the dependencies of the package | ||||
mkdir -p node_modules | ||||
cd node_modules | ||||
# Only include dependencies if they don't exist. They may also be bundled in the package. | ||||
if [ ! -e "${dependency.name}" ] | ||||
then | ||||
${composePackage dependency} | ||||
fi | ||||
cd .. | ||||
'' | ||||
) dependencies); | ||||
# Recursively composes the dependencies of a package | ||||
composePackage = { name, packageName, src, dependencies ? [], ... }@args: | ||||
let | ||||
fixImpureDependencies = writeTextFile { | ||||
name = "fixDependencies.js"; | ||||
text = '' | ||||
var fs = require('fs'); | ||||
var url = require('url'); | ||||
/* | ||||
* Replaces an impure version specification by * | ||||
*/ | ||||
function replaceImpureVersionSpec(versionSpec) { | ||||
var parsedUrl = url.parse(versionSpec); | ||||
if(versionSpec == "latest" || versionSpec == "unstable" || | ||||
versionSpec.substr(0, 2) == ".." || dependency.substr(0, 2) == "./" || dependency.substr(0, 2) == "~/" || dependency.substr(0, 1) == '/') | ||||
return '*'; | ||||
else if(parsedUrl.protocol == "git:" || parsedUrl.protocol == "git+ssh:" || parsedUrl.protocol == "git+http:" || parsedUrl.protocol == "git+https:" || | ||||
parsedUrl.protocol == "http:" || parsedUrl.protocol == "https:") | ||||
return '*'; | ||||
else | ||||
return versionSpec; | ||||
} | ||||
var packageObj = JSON.parse(fs.readFileSync('./package.json')); | ||||
/* Replace dependencies */ | ||||
if(packageObj.dependencies !== undefined) { | ||||
for(var dependency in packageObj.dependencies) { | ||||
var versionSpec = packageObj.dependencies[dependency]; | ||||
packageObj.dependencies[dependency] = replaceImpureVersionSpec(versionSpec); | ||||
} | ||||
} | ||||
/* Replace development dependencies */ | ||||
if(packageObj.devDependencies !== undefined) { | ||||
for(var dependency in packageObj.devDependencies) { | ||||
var versionSpec = packageObj.devDependencies[dependency]; | ||||
packageObj.devDependencies[dependency] = replaceImpureVersionSpec(versionSpec); | ||||
} | ||||
} | ||||
/* Replace optional dependencies */ | ||||
if(packageObj.optionalDependencies !== undefined) { | ||||
for(var dependency in packageObj.optionalDependencies) { | ||||
var versionSpec = packageObj.optionalDependencies[dependency]; | ||||
packageObj.optionalDependencies[dependency] = replaceImpureVersionSpec(versionSpec); | ||||
} | ||||
} | ||||
/* Write the fixed JSON file */ | ||||
fs.writeFileSync("package.json", JSON.stringify(packageObj)); | ||||
''; | ||||
}; | ||||
in | ||||
'' | ||||
DIR=$(pwd) | ||||
cd $TMPDIR | ||||
unpackFile ${src} | ||||
# Make the base dir in which the target dependency resides first | ||||
mkdir -p "$(dirname "$DIR/${packageName}")" | ||||
if [ -f "${src}" ] | ||||
then | ||||
# Figure out what directory has been unpacked | ||||
packageDir=$(find . -type d -maxdepth 1 | tail -1) | ||||
# Restore write permissions to make building work | ||||
chmod -R u+w "$packageDir" | ||||
# Move the extracted tarball into the output folder | ||||
mv "$packageDir" "$DIR/${packageName}" | ||||
elif [ -d "${src}" ] | ||||
then | ||||
# Restore write permissions to make building work | ||||
chmod -R u+w $strippedName | ||||
# Move the extracted directory into the output folder | ||||
mv $strippedName "$DIR/${packageName}" | ||||
fi | ||||
# Unset the stripped name to not confuse the next unpack step | ||||
unset strippedName | ||||
# Some version specifiers (latest, unstable, URLs, file paths) force NPM to make remote connections or consult paths outside the Nix store. | ||||
# The following JavaScript replaces these by * to prevent that | ||||
cd "$DIR/${packageName}" | ||||
node ${fixImpureDependencies} | ||||
# Include the dependencies of the package | ||||
${includeDependencies { inherit dependencies; }} | ||||
cd .. | ||||
${stdenv.lib.optionalString (builtins.substring 0 1 packageName == "@") "cd .."} | ||||
''; | ||||
# Extract the Node.js source code which is used to compile packages with | ||||
# native bindings | ||||
nodeSources = runCommand "node-sources" {} '' | ||||
tar --no-same-owner --no-same-permissions -xf ${nodejs.src} | ||||
mv node-* $out | ||||
''; | ||||
# Builds and composes an NPM package including all its dependencies | ||||
buildNodePackage = { name, packageName, version, dependencies ? [], production ? true, npmFlags ? "", dontNpmInstall ? false, preRebuild ? "", ... }@args: | ||||
stdenv.lib.makeOverridable stdenv.mkDerivation (builtins.removeAttrs args [ "dependencies" ] // { | ||||
name = "node-${name}-${version}"; | ||||
buildInputs = [ tarWrapper python nodejs ] ++ stdenv.lib.optional (stdenv.isLinux) utillinux ++ args.buildInputs or []; | ||||
dontStrip = args.dontStrip or true; # Striping may fail a build for some package deployments | ||||
inherit dontNpmInstall preRebuild; | ||||
unpackPhase = args.unpackPhase or "true"; | ||||
buildPhase = args.buildPhase or "true"; | ||||
compositionScript = composePackage args; | ||||
passAsFile = [ "compositionScript" ]; | ||||
installPhase = args.installPhase or '' | ||||
# Create and enter a root node_modules/ folder | ||||
mkdir -p $out/lib/node_modules | ||||
cd $out/lib/node_modules | ||||
# Compose the package and all its dependencies | ||||
source $compositionScriptPath | ||||
# Patch the shebangs of the bundled modules to prevent them from | ||||
# calling executables outside the Nix store as much as possible | ||||
patchShebangs . | ||||
# Deploy the Node.js package by running npm install. Since the | ||||
# dependencies have been provided already by ourselves, it should not | ||||
# attempt to install them again, which is good, because we want to make | ||||
# it Nix's responsibility. If it needs to install any dependencies | ||||
# anyway (e.g. because the dependency parameters are | ||||
# incomplete/incorrect), it fails. | ||||
# | ||||
# The other responsibilities of NPM are kept -- version checks, build | ||||
# steps, postprocessing etc. | ||||
export HOME=$TMPDIR | ||||
cd "${packageName}" | ||||
runHook preRebuild | ||||
npm --registry http://www.example.com --nodedir=${nodeSources} ${npmFlags} ${stdenv.lib.optionalString production "--production"} rebuild | ||||
if [ "$dontNpmInstall" != "1" ] | ||||
then | ||||
npm --registry http://www.example.com --nodedir=${nodeSources} ${npmFlags} ${stdenv.lib.optionalString production "--production"} install | ||||
fi | ||||
# Create symlink to the deployed executable folder, if applicable | ||||
if [ -d "$out/lib/node_modules/.bin" ] | ||||
then | ||||
ln -s $out/lib/node_modules/.bin $out/bin | ||||
fi | ||||
# Create symlinks to the deployed manual page folders, if applicable | ||||
if [ -d "$out/lib/node_modules/${packageName}/man" ] | ||||
then | ||||
mkdir -p $out/share | ||||
for dir in "$out/lib/node_modules/${packageName}/man/"* | ||||
do | ||||
mkdir -p $out/share/man/$(basename "$dir") | ||||
for page in "$dir"/* | ||||
do | ||||
ln -s $page $out/share/man/$(basename "$dir") | ||||
done | ||||
done | ||||
fi | ||||
''; | ||||
}); | ||||
# Builds a development shell | ||||
buildNodeShell = { name, packageName, version, src, dependencies ? [], production ? true, npmFlags ? "", dontNpmInstall ? false, ... }@args: | ||||
let | ||||
nodeDependencies = stdenv.mkDerivation { | ||||
name = "node-dependencies-${name}-${version}"; | ||||
buildInputs = [ tarWrapper python nodejs ] ++ stdenv.lib.optional (stdenv.isLinux) utillinux ++ args.buildInputs or []; | ||||
includeScript = includeDependencies { inherit dependencies; }; | ||||
passAsFile = [ "includeScript" ]; | ||||
buildCommand = '' | ||||
mkdir -p $out/lib | ||||
cd $out/lib | ||||
source $includeScriptPath | ||||
# Create fake package.json to make the npm commands work properly | ||||
cat > package.json <<EOF | ||||
{ | ||||
"name": "${packageName}", | ||||
"version": "${version}" | ||||
} | ||||
EOF | ||||
# Patch the shebangs of the bundled modules to prevent them from | ||||
# calling executables outside the Nix store as much as possible | ||||
patchShebangs . | ||||
export HOME=$TMPDIR | ||||
npm --registry http://www.example.com --nodedir=${nodeSources} ${npmFlags} ${stdenv.lib.optionalString production "--production"} rebuild | ||||
${stdenv.lib.optionalString (!dontNpmInstall) '' | ||||
npm --registry http://www.example.com --nodedir=${nodeSources} ${npmFlags} ${stdenv.lib.optionalString production "--production"} install | ||||
''} | ||||
ln -s $out/lib/node_modules/.bin $out/bin | ||||
''; | ||||
}; | ||||
in | ||||
stdenv.lib.makeOverridable stdenv.mkDerivation { | ||||
name = "node-shell-${name}-${version}"; | ||||
buildInputs = [ python nodejs ] ++ stdenv.lib.optional (stdenv.isLinux) utillinux ++ args.buildInputs or []; | ||||
buildCommand = '' | ||||
mkdir -p $out/bin | ||||
cat > $out/bin/shell <<EOF | ||||
#! ${stdenv.shell} -e | ||||
$shellHook | ||||
exec ${stdenv.shell} | ||||
EOF | ||||
chmod +x $out/bin/shell | ||||
''; | ||||
# Provide the dependencies in a development shell through the NODE_PATH environment variable | ||||
inherit nodeDependencies; | ||||
shellHook = stdenv.lib.optionalString (dependencies != []) '' | ||||
export NODE_PATH=$nodeDependencies/lib/node_modules | ||||
''; | ||||
}; | ||||
in | ||||
{ inherit buildNodeSourceDist buildNodePackage buildNodeShell; } | ||||