deps: make node-gyp work with io.js
[platform/upstream/nodejs.git] / deps / npm / node_modules / node-gyp / lib / build.js
index 26f5c4b..b35ca01 100644 (file)
@@ -12,6 +12,7 @@ var fs = require('graceful-fs')
   , log = require('npmlog')
   , which = require('which')
   , mkdirp = require('mkdirp')
+  , exec = require('child_process').exec
   , win = process.platform == 'win32'
 
 exports.usage = 'Invokes `' + (win ? 'msbuild' : 'make') + '` and builds the module'
@@ -19,7 +20,7 @@ exports.usage = 'Invokes `' + (win ? 'msbuild' : 'make') + '` and builds the mod
 function build (gyp, argv, callback) {
 
   var makeCommand = gyp.opts.make || process.env.MAKE
-      || (process.platform.indexOf('bsd') != -1 ? 'gmake' : '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')
@@ -99,7 +100,7 @@ function build (gyp, argv, callback) {
       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)
@@ -112,33 +113,67 @@ function build (gyp, argv, callback) {
   }
 
   /**
-   * Guess the location of the "msbuild.exe" file on Windows.
+   * Search for the location of "msbuild.exe" file on Windows.
    */
 
-  function guessMsbuild () {
-    log.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
-      copyNodeLib()
+      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()
+        })
+      })()
     })
   }
 
   /**
-   * Copies the node.lib file for the current target architecture into the
+   * Copies the iojs.lib file for the current target architecture into the
    * current proper dev dir location.
    */
 
@@ -146,15 +181,15 @@ function build (gyp, argv, callback) {
     if (!win || !copyDevLib) return doBuild()
 
     var buildDir = path.resolve(nodeDir, buildType)
-      , archNodeLibPath = path.resolve(nodeDir, arch, 'node.lib')
-      , buildNodeLibPath = path.resolve(buildDir, 'node.lib')
+      , 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 "node.lib" for ' + arch, buildNodeLibPath)
+      log.verbose('copying "iojs.lib" for ' + arch, buildNodeLibPath)
       rs.pipe(ws)
       rs.on('error', callback)
       ws.on('error', callback)
@@ -180,10 +215,6 @@ function build (gyp, argv, callback) {
     if (win) {
       // Turn off the Microsoft logo on Windows
       argv.push('/nologo')
-
-      // Fix "warning MSB8012: TargetExt(.dll) does not match
-      // the Linker's OutputFile property value (.node)"
-      argv.push('/property:TargetExt=.node')
     }
 
     // Specify the build type, Release by default
@@ -191,8 +222,9 @@ function build (gyp, argv, callback) {
       var p = arch === 'x64' ? 'x64' : 'Win32'
       argv.push('/p:Configuration=' + buildType + ';Platform=' + p)
       if (jobs) {
-        if (!isNaN(parseInt(jobs, 10))) {
-          argv.push('/m:' + parseInt(jobs, 10))
+        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)
         }
@@ -203,9 +235,10 @@ function build (gyp, argv, callback) {
       argv.push('-C')
       argv.push('build')
       if (jobs) {
-        if (!isNaN(parseInt(jobs, 10))) {
+        var j = parseInt(jobs, 10)
+        if (!isNaN(j) && j > 0) {
           argv.push('--jobs')
-          argv.push(parseInt(jobs, 10))
+          argv.push(j)
         } else if (jobs.toUpperCase() === 'MAX') {
           argv.push('--jobs')
           argv.push(require('os').cpus().length)
@@ -238,55 +271,7 @@ function build (gyp, argv, callback) {
     if (signal) {
       return callback(new Error('`' + command + '` got signal: ' + signal))
     }
-    //symlinkNodeBinding()
     callback()
   }
 
-  function symlinkNodeBinding () {
-    var source, target
-    var buildDir = 'build/' + buildType + '/*.node'
-    log.verbose('globbing for files', buildDir)
-    glob(buildDir, function (err, nodeFiles) {
-      if (err) return callback(err)
-      log.silly('symlink', 'linking files', nodeFiles)
-      function link () {
-        var file = nodeFiles.shift()
-        if (!file) {
-          // no more files to link... done!
-          return callback()
-        }
-        if (win) {
-          // windows requires absolute paths for junctions
-          source = path.resolve('build', path.basename(file))
-          target = path.resolve(file)
-        } else {
-          // on unix, use only relative paths since they're nice
-          source = path.join('build', path.basename(file))
-          target = path.relative('build', file)
-        }
-        log.info('symlink', 'creating %s "%s" pointing to "%s"', win ? 'junction' : 'symlink', source, target)
-        fs.symlink(target, source, 'junction', function (err) {
-          if (err) {
-            if (err.code === 'EEXIST') {
-              log.verbose('destination already exists; deleting', dest)
-              rm(dest, function (err) {
-                if (err) return callback(err)
-                log.verbose('delete successful; trying symlink again')
-                nodeFiles.unshift(file)
-                link()
-              })
-            } else {
-              callback(err)
-            }
-            return
-          }
-          // process the next file, if any
-          link()
-        })
-      }
-      // start linking
-      link()
-    })
-  }
-
 }