'use strict' const fs = require('graceful-fs').promises const path = require('path') const { glob } = require('glob') const log = require('./log') const which = require('which') const win = process.platform === 'win32' async function build (gyp, argv) { let platformMake = 'make' if (process.platform === 'aix') { platformMake = 'gmake' } else if (process.platform === 'os400') { platformMake = 'gmake' } else if (process.platform.indexOf('bsd') !== -1) { platformMake = 'gmake' } else if (win && argv.length > 0) { argv = argv.map(function (target) { return '/t:' + target }) } const makeCommand = gyp.opts.make || process.env.MAKE || platformMake let command = win ? 'msbuild' : makeCommand const jobs = gyp.opts.jobs || process.env.JOBS let buildType let config let arch let nodeDir let guessedSolution let python let buildBinsDir await loadConfigGypi() /** * Load the "config.gypi" file that was generated during "configure". */ async function loadConfigGypi () { let data try { const configPath = path.resolve('build', 'config.gypi') data = await fs.readFile(configPath, 'utf8') } catch (err) { if (err.code === 'ENOENT') { throw new Error('You must run `node-gyp configure` first!') } else { throw err } } 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 python = config.variables.python 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) log.verbose('python', python) if (win) { await findSolutionFile() } else { await doWhich() } } /** * On Windows, find the first build/*.sln file. */ async function findSolutionFile () { const files = await glob('build/*.sln') if (files.length === 0) { throw new Error('Could not find *.sln file. Did you run "configure"?') } guessedSolution = files[0] log.verbose('found first Solution file', guessedSolution) await doWhich() } /** * Uses node-which to locate the msbuild / make executable. */ async function doWhich () { // On Windows use msbuild provided by node-gyp configure if (win) { if (!config.variables.msbuild_path) { throw new Error('MSBuild is not set, please run `node-gyp configure`.') } command = config.variables.msbuild_path log.verbose('using MSBuild:', command) await doBuild() return } // First make sure we have the build command in the PATH const execPath = await which(command) log.verbose('`which` succeeded for `' + command + '`', execPath) await doBuild() } /** * Actually spawn the process and compile the module. */ async function doBuild () { // Enable Verbose build const verbose = log.logger.isVisible('verbose') let j if (!win && verbose) { argv.push('V=1') } if (win && !verbose) { argv.push('/clp:Verbosity=minimal') } if (win) { // Turn off the Microsoft logo on Windows argv.push('/nologo') } // Specify the build type, Release by default if (win) { // Convert .gypi config target_arch to MSBuild /Platform // Since there are many ways to state '32-bit Intel', default to it. // N.B. msbuild's Condition string equality tests are case-insensitive. const archLower = arch.toLowerCase() const p = archLower === 'x64' ? 'x64' : (archLower === 'arm' ? 'ARM' : (archLower === 'arm64' ? 'ARM64' : 'Win32')) argv.push('/p:Configuration=' + buildType + ';Platform=' + p) if (jobs) { 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) { 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) { // did the user specify their own .sln file? const hasSln = argv.some(function (arg) { return path.extname(arg) === '.sln' }) if (!hasSln) { argv.unshift(gyp.opts.solution || guessedSolution) } } if (!win) { // Add build-time dependency symlinks (such as Python) to PATH buildBinsDir = path.resolve('build', 'node_gyp_bins') process.env.PATH = `${buildBinsDir}:${process.env.PATH}` await fs.mkdir(buildBinsDir, { recursive: true }) const symlinkDestination = path.join(buildBinsDir, 'python3') try { await fs.unlink(symlinkDestination) } catch (err) { if (err.code !== 'ENOENT') throw err } await fs.symlink(python, symlinkDestination) log.verbose('bin symlinks', `created symlink to "${python}" in "${buildBinsDir}" and added to PATH`) } const proc = gyp.spawn(command, argv) await new Promise((resolve, reject) => proc.on('exit', async (code, signal) => { if (buildBinsDir) { // Clean up the build-time dependency symlinks: await fs.rm(buildBinsDir, { recursive: true }) } if (code !== 0) { return reject(new Error('`' + command + '` failed with exit code: ' + code)) } if (signal) { return reject(new Error('`' + command + '` got signal: ' + signal)) } resolve() })) } } module.exports = build module.exports.usage = 'Invokes `' + (win ? 'msbuild' : 'make') + '` and builds the module'