• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 'use strict'
2 
3 const fs = require('graceful-fs')
4 const path = require('path')
5 const glob = require('glob')
6 const log = require('npmlog')
7 const which = require('which')
8 const win = process.platform === 'win32'
9 
10 function build (gyp, argv, callback) {
11   var platformMake = 'make'
12   if (process.platform === 'aix') {
13     platformMake = 'gmake'
14   } else if (process.platform.indexOf('bsd') !== -1) {
15     platformMake = 'gmake'
16   } else if (win && argv.length > 0) {
17     argv = argv.map(function (target) {
18       return '/t:' + target
19     })
20   }
21 
22   var makeCommand = gyp.opts.make || process.env.MAKE || platformMake
23   var command = win ? 'msbuild' : makeCommand
24   var jobs = gyp.opts.jobs || process.env.JOBS
25   var buildType
26   var config
27   var arch
28   var nodeDir
29   var guessedSolution
30 
31   loadConfigGypi()
32 
33   /**
34    * Load the "config.gypi" file that was generated during "configure".
35    */
36 
37   function loadConfigGypi () {
38     var configPath = path.resolve('build', 'config.gypi')
39 
40     fs.readFile(configPath, 'utf8', function (err, data) {
41       if (err) {
42         if (err.code === 'ENOENT') {
43           callback(new Error('You must run `node-gyp configure` first!'))
44         } else {
45           callback(err)
46         }
47         return
48       }
49       config = JSON.parse(data.replace(/#.+\n/, ''))
50 
51       // get the 'arch', 'buildType', and 'nodeDir' vars from the config
52       buildType = config.target_defaults.default_configuration
53       arch = config.variables.target_arch
54       nodeDir = config.variables.nodedir
55 
56       if ('debug' in gyp.opts) {
57         buildType = gyp.opts.debug ? 'Debug' : 'Release'
58       }
59       if (!buildType) {
60         buildType = 'Release'
61       }
62 
63       log.verbose('build type', buildType)
64       log.verbose('architecture', arch)
65       log.verbose('node dev dir', nodeDir)
66 
67       if (win) {
68         findSolutionFile()
69       } else {
70         doWhich()
71       }
72     })
73   }
74 
75   /**
76    * On Windows, find the first build/*.sln file.
77    */
78 
79   function findSolutionFile () {
80     glob('build/*.sln', function (err, files) {
81       if (err) {
82         return callback(err)
83       }
84       if (files.length === 0) {
85         return callback(new Error('Could not find *.sln file. Did you run "configure"?'))
86       }
87       guessedSolution = files[0]
88       log.verbose('found first Solution file', guessedSolution)
89       doWhich()
90     })
91   }
92 
93   /**
94    * Uses node-which to locate the msbuild / make executable.
95    */
96 
97   function doWhich () {
98     // On Windows use msbuild provided by node-gyp configure
99     if (win) {
100       if (!config.variables.msbuild_path) {
101         return callback(new Error(
102           'MSBuild is not set, please run `node-gyp configure`.'))
103       }
104       command = config.variables.msbuild_path
105       log.verbose('using MSBuild:', command)
106       doBuild()
107       return
108     }
109     // First make sure we have the build command in the PATH
110     which(command, function (err, execPath) {
111       if (err) {
112         // Some other error or 'make' not found on Unix, report that to the user
113         callback(err)
114         return
115       }
116       log.verbose('`which` succeeded for `' + command + '`', execPath)
117       doBuild()
118     })
119   }
120 
121   /**
122    * Actually spawn the process and compile the module.
123    */
124 
125   function doBuild () {
126     // Enable Verbose build
127     var verbose = log.levels[log.level] <= log.levels.verbose
128     var j
129 
130     if (!win && verbose) {
131       argv.push('V=1')
132     }
133 
134     if (win && !verbose) {
135       argv.push('/clp:Verbosity=minimal')
136     }
137 
138     if (win) {
139       // Turn off the Microsoft logo on Windows
140       argv.push('/nologo')
141     }
142 
143     // Specify the build type, Release by default
144     if (win) {
145       // Convert .gypi config target_arch to MSBuild /Platform
146       // Since there are many ways to state '32-bit Intel', default to it.
147       // N.B. msbuild's Condition string equality tests are case-insensitive.
148       var archLower = arch.toLowerCase()
149       var p = archLower === 'x64' ? 'x64'
150         : (archLower === 'arm' ? 'ARM'
151           : (archLower === 'arm64' ? 'ARM64' : 'Win32'))
152       argv.push('/p:Configuration=' + buildType + ';Platform=' + p)
153       if (jobs) {
154         j = parseInt(jobs, 10)
155         if (!isNaN(j) && j > 0) {
156           argv.push('/m:' + j)
157         } else if (jobs.toUpperCase() === 'MAX') {
158           argv.push('/m:' + require('os').cpus().length)
159         }
160       }
161     } else {
162       argv.push('BUILDTYPE=' + buildType)
163       // Invoke the Makefile in the 'build' dir.
164       argv.push('-C')
165       argv.push('build')
166       if (jobs) {
167         j = parseInt(jobs, 10)
168         if (!isNaN(j) && j > 0) {
169           argv.push('--jobs')
170           argv.push(j)
171         } else if (jobs.toUpperCase() === 'MAX') {
172           argv.push('--jobs')
173           argv.push(require('os').cpus().length)
174         }
175       }
176     }
177 
178     if (win) {
179       // did the user specify their own .sln file?
180       var hasSln = argv.some(function (arg) {
181         return path.extname(arg) === '.sln'
182       })
183       if (!hasSln) {
184         argv.unshift(gyp.opts.solution || guessedSolution)
185       }
186     }
187 
188     var proc = gyp.spawn(command, argv)
189     proc.on('exit', onExit)
190   }
191 
192   function onExit (code, signal) {
193     if (code !== 0) {
194       return callback(new Error('`' + command + '` failed with exit code: ' + code))
195     }
196     if (signal) {
197       return callback(new Error('`' + command + '` got signal: ' + signal))
198     }
199     callback()
200   }
201 }
202 
203 module.exports = build
204 module.exports.usage = 'Invokes `' + (win ? 'msbuild' : 'make') + '` and builds the module'
205