deps: make node-gyp work with io.js
[platform/upstream/nodejs.git] / deps / npm / node_modules / node-gyp / lib / build.js
index 3ffc4c7..b35ca01 100644 (file)
@@ -5,30 +5,33 @@ module.exports = exports = build
  * Module dependencies.
  */
 
-var fs = require('fs')
+var fs = require('graceful-fs')
+  , rm = require('rimraf')
   , path = require('path')
   , glob = require('glob')
+  , log = require('npmlog')
   , which = require('which')
-  , asyncEmit = require('./util/asyncEmit')
-  , createHook = require('./util/hook')
+  , mkdirp = require('mkdirp')
+  , exec = require('child_process').exec
   , win = process.platform == 'win32'
 
 exports.usage = 'Invokes `' + (win ? 'msbuild' : 'make') + '` and builds the module'
 
 function build (gyp, argv, callback) {
 
-  gyp.verbose('build args', argv)
-  var command = win ? 'msbuild' : 'make'
+  var makeCommand = gyp.opts.make || process.env.MAKE
+      || (process.platform.indexOf('bsd') != -1 && process.platform.indexOf('kfreebsd') == -1 ? 'gmake' : 'make')
+    , command = win ? 'msbuild' : makeCommand
     , buildDir = path.resolve('build')
     , configPath = path.resolve(buildDir, 'config.gypi')
+    , jobs = gyp.opts.jobs || process.env.JOBS
+    , buildType
     , config
-    , emitter
+    , arch
+    , nodeDir
+    , copyDevLib
 
-  createHook('gyp-build.js', function (err, _e) {
-    if (err) return callback(err)
-    emitter = _e
-    loadConfigGypi()
-  })
+  loadConfigGypi()
 
   /**
    * Load the "config.gypi" file that was generated during "configure".
@@ -45,6 +48,24 @@ function build (gyp, argv, callback) {
         return
       }
       config = JSON.parse(data.replace(/\#.+\n/, ''))
+
+      // get the 'arch', 'buildType', and 'nodeDir' vars from the config
+      buildType = config.target_defaults.default_configuration
+      arch = config.variables.target_arch
+      nodeDir = config.variables.nodedir
+      copyDevLib = config.variables.copy_dev_lib == 'true'
+
+      if ('debug' in gyp.opts) {
+        buildType = gyp.opts.debug ? 'Debug' : 'Release'
+      }
+      if (!buildType) {
+        buildType = 'Release'
+      }
+
+      log.verbose('build type', buildType)
+      log.verbose('architecture', arch)
+      log.verbose('node dev dir', nodeDir)
+
       if (win) {
         findSolutionFile()
       } else {
@@ -54,7 +75,7 @@ function build (gyp, argv, callback) {
   }
 
   /**
-   * On Windows, find first build/*.sln file.
+   * On Windows, find the first build/*.sln file.
    */
 
   function findSolutionFile () {
@@ -64,88 +85,165 @@ function build (gyp, argv, callback) {
         return callback(new Error('Could not find *.sln file. Did you run "configure"?'))
       }
       guessedSolution = files[0]
-      gyp.verbose('found first Solution file', guessedSolution)
+      log.verbose('found first Solution file', guessedSolution)
       doWhich()
     })
   }
 
+  /**
+   * Uses node-which to locate the msbuild / make executable.
+   */
+
   function doWhich () {
     // First make sure we have the build command in the PATH
     which(command, function (err, execPath) {
       if (err) {
         if (win && /not found/.test(err.message)) {
           // On windows and no 'msbuild' found. Let's guess where it is
-          guessMsbuild()
+          findMsbuild()
         } else {
           // Some other error or 'make' not found on Unix, report that to the user
           callback(err)
         }
         return
       }
-      gyp.verbose('`which` succeeded for `' + command + '`', execPath)
-      build()
+      log.verbose('`which` succeeded for `' + command + '`', execPath)
+      copyNodeLib()
     })
   }
 
   /**
-   * Guess the location of the "msbuild.exe" file on Windows.
+   * Search for the location of "msbuild.exe" file on Windows.
    */
 
-  function guessMsbuild () {
-    gyp.verbose('could not find "msbuild.exe". guessing location')
-    // This is basically just hard-coded. If this causes problems
-    // then we'll think of something more clever.
-    var windir = process.env.WINDIR || process.env.SYSTEMROOT || 'C:\\WINDOWS'
-      , frameworkDir = path.resolve(windir, 'Microsoft.NET', 'Framework')
-      , versionDir = path.resolve(frameworkDir, 'v4.0.30319') // This is probably the most brittle part...
-      , msbuild = path.resolve(versionDir, 'msbuild.exe')
-    fs.stat(msbuild, function (err, stat) {
+  function findMsbuild () {
+    log.verbose('could not find "msbuild.exe" in PATH - finding location in registry')
+    var notfoundErr = new Error('Can\'t find "msbuild.exe". Do you have Microsoft Visual Studio C++ 2008+ installed?')
+    var cmd = 'reg query "HKLM\\Software\\Microsoft\\MSBuild\\ToolsVersions" /s'
+    if (process.arch !== 'ia32')
+      cmd += ' /reg:32'
+    exec(cmd, function (err, stdout, stderr) {
+      var reVers = /ToolsVersions\\([^\\]+)$/i
+        , rePath = /\r\n[ \t]+MSBuildToolsPath[ \t]+REG_SZ[ \t]+([^\r]+)/i
+        , msbuilds = []
+        , r
+        , msbuildPath
       if (err) {
-        if (err.code == 'ENOENT') {
-          callback(new Error('Can\'t find "msbuild.exe". Do you have Microsoft Visual Studio C++ 2010 installed?'))
-        } else {
-          callback(err)
-        }
-        return
+        return callback(notfoundErr)
       }
-      command = msbuild
-      build()
+      stdout.split('\r\n\r\n').forEach(function(l) {
+        if (!l) return
+        l = l.trim()
+        if (r = reVers.exec(l.substring(0, l.indexOf('\r\n')))) {
+          var ver = parseFloat(r[1], 10)
+          if (ver >= 3.5) {
+            if (r = rePath.exec(l)) {
+              msbuilds.push({
+                version: ver,
+                path: r[1]
+              })
+            }
+          }
+        }
+      })
+      msbuilds.sort(function (x, y) {
+        return (x.version < y.version ? -1 : 1)
+      })
+      ;(function verifyMsbuild () {
+        if (!msbuilds.length) return callback(notfoundErr)
+        msbuildPath = path.resolve(msbuilds.pop().path, 'msbuild.exe')
+        fs.stat(msbuildPath, function (err, stat) {
+          if (err) {
+            if (err.code == 'ENOENT') {
+              if (msbuilds.length) {
+                return verifyMsbuild()
+              } else {
+                callback(notfoundErr)
+              }
+            } else {
+              callback(err)
+            }
+            return
+          }
+          command = msbuildPath
+          copyNodeLib()
+        })
+      })()
     })
   }
 
   /**
-   * Actually spawn the process and compile the module.
+   * Copies the iojs.lib file for the current target architecture into the
+   * current proper dev dir location.
    */
 
-  function build () {
-    var buildType = config.target_defaults.default_configuration
-      , platform = config.variables.target_arch == 'x64' ? '64' : '32'
+  function copyNodeLib () {
+    if (!win || !copyDevLib) return doBuild()
 
-    if (gyp.opts.debug) {
-      buildType = 'Debug'
-    }
+    var buildDir = path.resolve(nodeDir, buildType)
+      , archNodeLibPath = path.resolve(nodeDir, arch, 'iojs.lib')
+      , buildNodeLibPath = path.resolve(buildDir, 'iojs.lib')
+
+    mkdirp(buildDir, function (err, isNew) {
+      if (err) return callback(err)
+      log.verbose('"' + buildType + '" dir needed to be created?', isNew)
+      var rs = fs.createReadStream(archNodeLibPath)
+        , ws = fs.createWriteStream(buildNodeLibPath)
+      log.verbose('copying "iojs.lib" for ' + arch, buildNodeLibPath)
+      rs.pipe(ws)
+      rs.on('error', callback)
+      ws.on('error', callback)
+      rs.on('end', doBuild)
+    })
+  }
+
+  /**
+   * Actually spawn the process and compile the module.
+   */
+
+  function doBuild () {
 
     // Enable Verbose build
-    if (!win && gyp.opts.verbose) {
+    var verbose = log.levels[log.level] <= log.levels.verbose
+    if (!win && verbose) {
       argv.push('V=1')
     }
-    if (win && !gyp.opts.verbose) {
+    if (win && !verbose) {
       argv.push('/clp:Verbosity=minimal')
     }
 
-    // Turn off the Microsoft logo on Windows
     if (win) {
+      // Turn off the Microsoft logo on Windows
       argv.push('/nologo')
     }
 
     // Specify the build type, Release by default
     if (win) {
-      argv.push('/p:Configuration=' + buildType + ';Platform=Win' + platform)
+      var p = arch === 'x64' ? 'x64' : 'Win32'
+      argv.push('/p:Configuration=' + buildType + ';Platform=' + p)
+      if (jobs) {
+        var j = parseInt(jobs, 10)
+        if (!isNaN(j) && j > 0) {
+          argv.push('/m:' + j)
+        } else if (jobs.toUpperCase() === 'MAX') {
+          argv.push('/m:' + require('os').cpus().length)
+        }
+      }
     } else {
       argv.push('BUILDTYPE=' + buildType)
       // Invoke the Makefile in the 'build' dir.
       argv.push('-C')
       argv.push('build')
+      if (jobs) {
+        var j = parseInt(jobs, 10)
+        if (!isNaN(j) && j > 0) {
+          argv.push('--jobs')
+          argv.push(j)
+        } else if (jobs.toUpperCase() === 'MAX') {
+          argv.push('--jobs')
+          argv.push(require('os').cpus().length)
+        }
+      }
     }
 
     if (win) {
@@ -158,11 +256,8 @@ function build (gyp, argv, callback) {
       }
     }
 
-    asyncEmit(emitter, 'before', function (err) {
-      if (err) return callback(err)
-      var proc = gyp.spawn(command, argv)
-      proc.on('exit', onExit)
-    })
+    var proc = gyp.spawn(command, argv)
+    proc.on('exit', onExit)
   }
 
   /**
@@ -170,16 +265,13 @@ function build (gyp, argv, callback) {
    */
 
   function onExit (code, signal) {
-    asyncEmit(emitter, 'after', function (err) {
-      if (err) return callback(err)
-      if (code !== 0) {
-        return callback(new Error('`' + command + '` failed with exit code: ' + code))
-      }
-      if (signal) {
-        return callback(new Error('`' + command + '` got signal: ' + signal))
-      }
-      callback()
-    })
+    if (code !== 0) {
+      return callback(new Error('`' + command + '` failed with exit code: ' + code))
+    }
+    if (signal) {
+      return callback(new Error('`' + command + '` got signal: ' + signal))
+    }
+    callback()
   }
 
 }