Clone of https://github.com/NixOS/nixpkgs.git (to stress-test knotserver)
1# This file originates from node2nix 2 3{lib, stdenv, nodejs, python2, pkgs, libtool, runCommand, writeTextFile, writeShellScript}: 4 5let 6 # Workaround to cope with utillinux in Nixpkgs 20.09 and util-linux in Nixpkgs master 7 utillinux = if pkgs ? utillinux then pkgs.utillinux else pkgs.util-linux; 8 9 python = if nodejs ? python then nodejs.python else python2; 10 11 # Create a tar wrapper that filters all the 'Ignoring unknown extended header keyword' noise 12 tarWrapper = runCommand "tarWrapper" {} '' 13 mkdir -p $out/bin 14 15 cat > $out/bin/tar <<EOF 16 #! ${stdenv.shell} -e 17 $(type -p tar) "\$@" --warning=no-unknown-keyword --delay-directory-restore 18 EOF 19 20 chmod +x $out/bin/tar 21 ''; 22 23 # Function that generates a TGZ file from a NPM project 24 buildNodeSourceDist = 25 { name, version, src, ... }: 26 27 stdenv.mkDerivation { 28 name = "node-tarball-${name}-${version}"; 29 inherit src; 30 buildInputs = [ nodejs ]; 31 buildPhase = '' 32 export HOME=$TMPDIR 33 tgzFile=$(npm pack | tail -n 1) # Hooks to the pack command will add output (https://docs.npmjs.com/misc/scripts) 34 ''; 35 installPhase = '' 36 mkdir -p $out/tarballs 37 mv $tgzFile $out/tarballs 38 mkdir -p $out/nix-support 39 echo "file source-dist $out/tarballs/$tgzFile" >> $out/nix-support/hydra-build-products 40 ''; 41 }; 42 43 # Common shell logic 44 installPackage = writeShellScript "install-package" '' 45 installPackage() { 46 local packageName=$1 src=$2 47 48 local strippedName 49 50 local DIR=$PWD 51 cd $TMPDIR 52 53 unpackFile $src 54 55 # Make the base dir in which the target dependency resides first 56 mkdir -p "$(dirname "$DIR/$packageName")" 57 58 if [ -f "$src" ] 59 then 60 # Figure out what directory has been unpacked 61 packageDir="$(find . -maxdepth 1 -type d | tail -1)" 62 63 # Restore write permissions to make building work 64 find "$packageDir" -type d -exec chmod u+x {} \; 65 chmod -R u+w "$packageDir" 66 67 # Move the extracted tarball into the output folder 68 mv "$packageDir" "$DIR/$packageName" 69 elif [ -d "$src" ] 70 then 71 # Get a stripped name (without hash) of the source directory. 72 # On old nixpkgs it's already set internally. 73 if [ -z "$strippedName" ] 74 then 75 strippedName="$(stripHash $src)" 76 fi 77 78 # Restore write permissions to make building work 79 chmod -R u+w "$strippedName" 80 81 # Move the extracted directory into the output folder 82 mv "$strippedName" "$DIR/$packageName" 83 fi 84 85 # Change to the package directory to install dependencies 86 cd "$DIR/$packageName" 87 } 88 ''; 89 90 # Bundle the dependencies of the package 91 # 92 # Only include dependencies if they don't exist. They may also be bundled in the package. 93 includeDependencies = {dependencies}: 94 lib.optionalString (dependencies != []) ( 95 '' 96 mkdir -p node_modules 97 cd node_modules 98 '' 99 + (lib.concatMapStrings (dependency: 100 '' 101 if [ ! -e "${dependency.packageName}" ]; then 102 ${composePackage dependency} 103 fi 104 '' 105 ) dependencies) 106 + '' 107 cd .. 108 '' 109 ); 110 111 # Recursively composes the dependencies of a package 112 composePackage = { name, packageName, src, dependencies ? [], ... }@args: 113 builtins.addErrorContext "while evaluating node package '${packageName}'" '' 114 installPackage "${packageName}" "${src}" 115 ${includeDependencies { inherit dependencies; }} 116 cd .. 117 ${lib.optionalString (builtins.substring 0 1 packageName == "@") "cd .."} 118 ''; 119 120 pinpointDependencies = {dependencies, production}: 121 let 122 pinpointDependenciesFromPackageJSON = writeTextFile { 123 name = "pinpointDependencies.js"; 124 text = '' 125 var fs = require('fs'); 126 var path = require('path'); 127 128 function resolveDependencyVersion(location, name) { 129 if(location == process.env['NIX_STORE']) { 130 return null; 131 } else { 132 var dependencyPackageJSON = path.join(location, "node_modules", name, "package.json"); 133 134 if(fs.existsSync(dependencyPackageJSON)) { 135 var dependencyPackageObj = JSON.parse(fs.readFileSync(dependencyPackageJSON)); 136 137 if(dependencyPackageObj.name == name) { 138 return dependencyPackageObj.version; 139 } 140 } else { 141 return resolveDependencyVersion(path.resolve(location, ".."), name); 142 } 143 } 144 } 145 146 function replaceDependencies(dependencies) { 147 if(typeof dependencies == "object" && dependencies !== null) { 148 for(var dependency in dependencies) { 149 var resolvedVersion = resolveDependencyVersion(process.cwd(), dependency); 150 151 if(resolvedVersion === null) { 152 process.stderr.write("WARNING: cannot pinpoint dependency: "+dependency+", context: "+process.cwd()+"\n"); 153 } else { 154 dependencies[dependency] = resolvedVersion; 155 } 156 } 157 } 158 } 159 160 /* Read the package.json configuration */ 161 var packageObj = JSON.parse(fs.readFileSync('./package.json')); 162 163 /* Pinpoint all dependencies */ 164 replaceDependencies(packageObj.dependencies); 165 if(process.argv[2] == "development") { 166 replaceDependencies(packageObj.devDependencies); 167 } 168 else { 169 packageObj.devDependencies = {}; 170 } 171 replaceDependencies(packageObj.optionalDependencies); 172 replaceDependencies(packageObj.peerDependencies); 173 174 /* Write the fixed package.json file */ 175 fs.writeFileSync("package.json", JSON.stringify(packageObj, null, 2)); 176 ''; 177 }; 178 in 179 '' 180 node ${pinpointDependenciesFromPackageJSON} ${if production then "production" else "development"} 181 182 ${lib.optionalString (dependencies != []) 183 '' 184 if [ -d node_modules ] 185 then 186 cd node_modules 187 ${lib.concatMapStrings (dependency: pinpointDependenciesOfPackage dependency) dependencies} 188 cd .. 189 fi 190 ''} 191 ''; 192 193 # Recursively traverses all dependencies of a package and pinpoints all 194 # dependencies in the package.json file to the versions that are actually 195 # being used. 196 197 pinpointDependenciesOfPackage = { packageName, dependencies ? [], production ? true, ... }@args: 198 '' 199 if [ -d "${packageName}" ] 200 then 201 cd "${packageName}" 202 ${pinpointDependencies { inherit dependencies production; }} 203 cd .. 204 ${lib.optionalString (builtins.substring 0 1 packageName == "@") "cd .."} 205 fi 206 ''; 207 208 # Extract the Node.js source code which is used to compile packages with 209 # native bindings 210 nodeSources = runCommand "node-sources" {} '' 211 tar --no-same-owner --no-same-permissions -xf ${nodejs.src} 212 mv node-* $out 213 ''; 214 215 # Script that adds _integrity fields to all package.json files to prevent NPM from consulting the cache (that is empty) 216 addIntegrityFieldsScript = writeTextFile { 217 name = "addintegrityfields.js"; 218 text = '' 219 var fs = require('fs'); 220 var path = require('path'); 221 222 function augmentDependencies(baseDir, dependencies) { 223 for(var dependencyName in dependencies) { 224 var dependency = dependencies[dependencyName]; 225 226 // Open package.json and augment metadata fields 227 var packageJSONDir = path.join(baseDir, "node_modules", dependencyName); 228 var packageJSONPath = path.join(packageJSONDir, "package.json"); 229 230 if(fs.existsSync(packageJSONPath)) { // Only augment packages that exist. Sometimes we may have production installs in which development dependencies can be ignored 231 console.log("Adding metadata fields to: "+packageJSONPath); 232 var packageObj = JSON.parse(fs.readFileSync(packageJSONPath)); 233 234 if(dependency.integrity) { 235 packageObj["_integrity"] = dependency.integrity; 236 } else { 237 packageObj["_integrity"] = "sha1-000000000000000000000000000="; // When no _integrity string has been provided (e.g. by Git dependencies), add a dummy one. It does not seem to harm and it bypasses downloads. 238 } 239 240 if(dependency.resolved) { 241 packageObj["_resolved"] = dependency.resolved; // Adopt the resolved property if one has been provided 242 } else { 243 packageObj["_resolved"] = dependency.version; // Set the resolved version to the version identifier. This prevents NPM from cloning Git repositories. 244 } 245 246 if(dependency.from !== undefined) { // Adopt from property if one has been provided 247 packageObj["_from"] = dependency.from; 248 } 249 250 fs.writeFileSync(packageJSONPath, JSON.stringify(packageObj, null, 2)); 251 } 252 253 // Augment transitive dependencies 254 if(dependency.dependencies !== undefined) { 255 augmentDependencies(packageJSONDir, dependency.dependencies); 256 } 257 } 258 } 259 260 if(fs.existsSync("./package-lock.json")) { 261 var packageLock = JSON.parse(fs.readFileSync("./package-lock.json")); 262 263 if(![1, 2].includes(packageLock.lockfileVersion)) { 264 process.stderr.write("Sorry, I only understand lock file versions 1 and 2!\n"); 265 process.exit(1); 266 } 267 268 if(packageLock.dependencies !== undefined) { 269 augmentDependencies(".", packageLock.dependencies); 270 } 271 } 272 ''; 273 }; 274 275 # Reconstructs a package-lock file from the node_modules/ folder structure and package.json files with dummy sha1 hashes 276 reconstructPackageLock = writeTextFile { 277 name = "reconstructpackagelock.js"; 278 text = '' 279 var fs = require('fs'); 280 var path = require('path'); 281 282 var packageObj = JSON.parse(fs.readFileSync("package.json")); 283 284 var lockObj = { 285 name: packageObj.name, 286 version: packageObj.version, 287 lockfileVersion: 2, 288 requires: true, 289 packages: { 290 "": { 291 name: packageObj.name, 292 version: packageObj.version, 293 license: packageObj.license, 294 bin: packageObj.bin, 295 dependencies: packageObj.dependencies, 296 engines: packageObj.engines, 297 optionalDependencies: packageObj.optionalDependencies 298 } 299 }, 300 dependencies: {} 301 }; 302 303 function augmentPackageJSON(filePath, packages, dependencies) { 304 var packageJSON = path.join(filePath, "package.json"); 305 if(fs.existsSync(packageJSON)) { 306 var packageObj = JSON.parse(fs.readFileSync(packageJSON)); 307 packages[filePath] = { 308 version: packageObj.version, 309 integrity: "sha1-000000000000000000000000000=", 310 dependencies: packageObj.dependencies, 311 engines: packageObj.engines, 312 optionalDependencies: packageObj.optionalDependencies 313 }; 314 dependencies[packageObj.name] = { 315 version: packageObj.version, 316 integrity: "sha1-000000000000000000000000000=", 317 dependencies: {} 318 }; 319 processDependencies(path.join(filePath, "node_modules"), packages, dependencies[packageObj.name].dependencies); 320 } 321 } 322 323 function processDependencies(dir, packages, dependencies) { 324 if(fs.existsSync(dir)) { 325 var files = fs.readdirSync(dir); 326 327 files.forEach(function(entry) { 328 var filePath = path.join(dir, entry); 329 var stats = fs.statSync(filePath); 330 331 if(stats.isDirectory()) { 332 if(entry.substr(0, 1) == "@") { 333 // When we encounter a namespace folder, augment all packages belonging to the scope 334 var pkgFiles = fs.readdirSync(filePath); 335 336 pkgFiles.forEach(function(entry) { 337 if(stats.isDirectory()) { 338 var pkgFilePath = path.join(filePath, entry); 339 augmentPackageJSON(pkgFilePath, packages, dependencies); 340 } 341 }); 342 } else { 343 augmentPackageJSON(filePath, packages, dependencies); 344 } 345 } 346 }); 347 } 348 } 349 350 processDependencies("node_modules", lockObj.packages, lockObj.dependencies); 351 352 fs.writeFileSync("package-lock.json", JSON.stringify(lockObj, null, 2)); 353 ''; 354 }; 355 356 # Script that links bins defined in package.json to the node_modules bin directory 357 # NPM does not do this for top-level packages itself anymore as of v7 358 linkBinsScript = writeTextFile { 359 name = "linkbins.js"; 360 text = '' 361 var fs = require('fs'); 362 var path = require('path'); 363 364 var packageObj = JSON.parse(fs.readFileSync("package.json")); 365 366 var nodeModules = Array(packageObj.name.split("/").length).fill("..").join(path.sep); 367 368 if(packageObj.bin !== undefined) { 369 fs.mkdirSync(path.join(nodeModules, ".bin")) 370 371 if(typeof packageObj.bin == "object") { 372 Object.keys(packageObj.bin).forEach(function(exe) { 373 if(fs.existsSync(packageObj.bin[exe])) { 374 console.log("linking bin '" + exe + "'"); 375 fs.symlinkSync( 376 path.join("..", packageObj.name, packageObj.bin[exe]), 377 path.join(nodeModules, ".bin", exe) 378 ); 379 } 380 else { 381 console.log("skipping non-existent bin '" + exe + "'"); 382 } 383 }) 384 } 385 else { 386 if(fs.existsSync(packageObj.bin)) { 387 console.log("linking bin '" + packageObj.bin + "'"); 388 fs.symlinkSync( 389 path.join("..", packageObj.name, packageObj.bin), 390 path.join(nodeModules, ".bin", packageObj.name.split("/").pop()) 391 ); 392 } 393 else { 394 console.log("skipping non-existent bin '" + packageObj.bin + "'"); 395 } 396 } 397 } 398 else if(packageObj.directories !== undefined && packageObj.directories.bin !== undefined) { 399 fs.mkdirSync(path.join(nodeModules, ".bin")) 400 401 fs.readdirSync(packageObj.directories.bin).forEach(function(exe) { 402 if(fs.existsSync(path.join(packageObj.directories.bin, exe))) { 403 console.log("linking bin '" + exe + "'"); 404 fs.symlinkSync( 405 path.join("..", packageObj.name, packageObj.directories.bin, exe), 406 path.join(nodeModules, ".bin", exe) 407 ); 408 } 409 else { 410 console.log("skipping non-existent bin '" + exe + "'"); 411 } 412 }) 413 } 414 ''; 415 }; 416 417 prepareAndInvokeNPM = {packageName, bypassCache, reconstructLock, npmFlags, production}: 418 let 419 forceOfflineFlag = if bypassCache then "--offline" else "--registry http://www.example.com"; 420 in 421 '' 422 # Pinpoint the versions of all dependencies to the ones that are actually being used 423 echo "pinpointing versions of dependencies..." 424 source $pinpointDependenciesScriptPath 425 426 # Patch the shebangs of the bundled modules to prevent them from 427 # calling executables outside the Nix store as much as possible 428 patchShebangs . 429 430 # Deploy the Node.js package by running npm install. Since the 431 # dependencies have been provided already by ourselves, it should not 432 # attempt to install them again, which is good, because we want to make 433 # it Nix's responsibility. If it needs to install any dependencies 434 # anyway (e.g. because the dependency parameters are 435 # incomplete/incorrect), it fails. 436 # 437 # The other responsibilities of NPM are kept -- version checks, build 438 # steps, postprocessing etc. 439 440 export HOME=$TMPDIR 441 cd "${packageName}" 442 runHook preRebuild 443 444 ${lib.optionalString bypassCache '' 445 ${lib.optionalString reconstructLock '' 446 if [ -f package-lock.json ] 447 then 448 echo "WARNING: Reconstruct lock option enabled, but a lock file already exists!" 449 echo "This will most likely result in version mismatches! We will remove the lock file and regenerate it!" 450 rm package-lock.json 451 else 452 echo "No package-lock.json file found, reconstructing..." 453 fi 454 455 node ${reconstructPackageLock} 456 ''} 457 458 node ${addIntegrityFieldsScript} 459 ''} 460 461 npm ${forceOfflineFlag} --nodedir=${nodeSources} ${npmFlags} ${lib.optionalString production "--production"} rebuild 462 463 runHook postRebuild 464 465 if [ "''${dontNpmInstall-}" != "1" ] 466 then 467 # NPM tries to download packages even when they already exist if npm-shrinkwrap is used. 468 rm -f npm-shrinkwrap.json 469 470 npm ${forceOfflineFlag} --nodedir=${nodeSources} --no-bin-links --ignore-scripts ${npmFlags} ${lib.optionalString production "--production"} install 471 fi 472 473 # Link executables defined in package.json 474 node ${linkBinsScript} 475 ''; 476 477 # Builds and composes an NPM package including all its dependencies 478 buildNodePackage = 479 { name 480 , packageName 481 , version ? null 482 , dependencies ? [] 483 , buildInputs ? [] 484 , production ? true 485 , npmFlags ? "" 486 , dontNpmInstall ? false 487 , bypassCache ? false 488 , reconstructLock ? false 489 , preRebuild ? "" 490 , dontStrip ? true 491 , unpackPhase ? "true" 492 , buildPhase ? "true" 493 , meta ? {} 494 , ... }@args: 495 496 let 497 extraArgs = removeAttrs args [ "name" "dependencies" "buildInputs" "dontStrip" "dontNpmInstall" "preRebuild" "unpackPhase" "buildPhase" "meta" ]; 498 in 499 stdenv.mkDerivation ({ 500 name = "${name}${if version == null then "" else "-${version}"}"; 501 buildInputs = [ tarWrapper python nodejs ] 502 ++ lib.optional (stdenv.isLinux) utillinux 503 ++ lib.optional (stdenv.isDarwin) libtool 504 ++ buildInputs; 505 506 inherit nodejs; 507 508 inherit dontStrip; # Stripping may fail a build for some package deployments 509 inherit dontNpmInstall preRebuild unpackPhase buildPhase; 510 511 compositionScript = composePackage args; 512 pinpointDependenciesScript = pinpointDependenciesOfPackage args; 513 514 passAsFile = [ "compositionScript" "pinpointDependenciesScript" ]; 515 516 installPhase = '' 517 source ${installPackage} 518 519 # Create and enter a root node_modules/ folder 520 mkdir -p $out/lib/node_modules 521 cd $out/lib/node_modules 522 523 # Compose the package and all its dependencies 524 source $compositionScriptPath 525 526 ${prepareAndInvokeNPM { inherit packageName bypassCache reconstructLock npmFlags production; }} 527 528 # Create symlink to the deployed executable folder, if applicable 529 if [ -d "$out/lib/node_modules/.bin" ] 530 then 531 ln -s $out/lib/node_modules/.bin $out/bin 532 533 # Patch the shebang lines of all the executables 534 ls $out/bin/* | while read i 535 do 536 file="$(readlink -f "$i")" 537 chmod u+rwx "$file" 538 patchShebangs "$file" 539 done 540 fi 541 542 # Create symlinks to the deployed manual page folders, if applicable 543 if [ -d "$out/lib/node_modules/${packageName}/man" ] 544 then 545 mkdir -p $out/share 546 for dir in "$out/lib/node_modules/${packageName}/man/"* 547 do 548 mkdir -p $out/share/man/$(basename "$dir") 549 for page in "$dir"/* 550 do 551 ln -s $page $out/share/man/$(basename "$dir") 552 done 553 done 554 fi 555 556 # Run post install hook, if provided 557 runHook postInstall 558 ''; 559 560 meta = { 561 # default to Node.js' platforms 562 platforms = nodejs.meta.platforms; 563 } // meta; 564 } // extraArgs); 565 566 # Builds a node environment (a node_modules folder and a set of binaries) 567 buildNodeDependencies = 568 { name 569 , packageName 570 , version ? null 571 , src 572 , dependencies ? [] 573 , buildInputs ? [] 574 , production ? true 575 , npmFlags ? "" 576 , dontNpmInstall ? false 577 , bypassCache ? false 578 , reconstructLock ? false 579 , dontStrip ? true 580 , unpackPhase ? "true" 581 , buildPhase ? "true" 582 , ... }@args: 583 584 let 585 extraArgs = removeAttrs args [ "name" "dependencies" "buildInputs" ]; 586 in 587 stdenv.mkDerivation ({ 588 name = "node-dependencies-${name}${if version == null then "" else "-${version}"}"; 589 590 buildInputs = [ tarWrapper python nodejs ] 591 ++ lib.optional (stdenv.isLinux) utillinux 592 ++ lib.optional (stdenv.isDarwin) libtool 593 ++ buildInputs; 594 595 inherit dontStrip; # Stripping may fail a build for some package deployments 596 inherit dontNpmInstall unpackPhase buildPhase; 597 598 includeScript = includeDependencies { inherit dependencies; }; 599 pinpointDependenciesScript = pinpointDependenciesOfPackage args; 600 601 passAsFile = [ "includeScript" "pinpointDependenciesScript" ]; 602 603 installPhase = '' 604 source ${installPackage} 605 606 mkdir -p $out/${packageName} 607 cd $out/${packageName} 608 609 source $includeScriptPath 610 611 # Create fake package.json to make the npm commands work properly 612 cp ${src}/package.json . 613 chmod 644 package.json 614 ${lib.optionalString bypassCache '' 615 if [ -f ${src}/package-lock.json ] 616 then 617 cp ${src}/package-lock.json . 618 chmod 644 package-lock.json 619 fi 620 ''} 621 622 # Go to the parent folder to make sure that all packages are pinpointed 623 cd .. 624 ${lib.optionalString (builtins.substring 0 1 packageName == "@") "cd .."} 625 626 ${prepareAndInvokeNPM { inherit packageName bypassCache reconstructLock npmFlags production; }} 627 628 # Expose the executables that were installed 629 cd .. 630 ${lib.optionalString (builtins.substring 0 1 packageName == "@") "cd .."} 631 632 mv ${packageName} lib 633 ln -s $out/lib/node_modules/.bin $out/bin 634 ''; 635 } // extraArgs); 636 637 # Builds a development shell 638 buildNodeShell = 639 { name 640 , packageName 641 , version ? null 642 , src 643 , dependencies ? [] 644 , buildInputs ? [] 645 , production ? true 646 , npmFlags ? "" 647 , dontNpmInstall ? false 648 , bypassCache ? false 649 , reconstructLock ? false 650 , dontStrip ? true 651 , unpackPhase ? "true" 652 , buildPhase ? "true" 653 , ... }@args: 654 655 let 656 nodeDependencies = buildNodeDependencies args; 657 extraArgs = removeAttrs args [ "name" "dependencies" "buildInputs" "dontStrip" "dontNpmInstall" "unpackPhase" "buildPhase" ]; 658 in 659 stdenv.mkDerivation ({ 660 name = "node-shell-${name}${if version == null then "" else "-${version}"}"; 661 662 buildInputs = [ python nodejs ] ++ lib.optional (stdenv.isLinux) utillinux ++ buildInputs; 663 buildCommand = '' 664 mkdir -p $out/bin 665 cat > $out/bin/shell <<EOF 666 #! ${stdenv.shell} -e 667 $shellHook 668 exec ${stdenv.shell} 669 EOF 670 chmod +x $out/bin/shell 671 ''; 672 673 # Provide the dependencies in a development shell through the NODE_PATH environment variable 674 inherit nodeDependencies; 675 shellHook = lib.optionalString (dependencies != []) '' 676 export NODE_PATH=${nodeDependencies}/lib/node_modules 677 export PATH="${nodeDependencies}/bin:$PATH" 678 ''; 679 } // extraArgs); 680in 681{ 682 buildNodeSourceDist = lib.makeOverridable buildNodeSourceDist; 683 buildNodePackage = lib.makeOverridable buildNodePackage; 684 buildNodeDependencies = lib.makeOverridable buildNodeDependencies; 685 buildNodeShell = lib.makeOverridable buildNodeShell; 686}