You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					236 lines
				
				7.7 KiB
			
		
		
			
		
	
	
					236 lines
				
				7.7 KiB
			| 
											3 years ago
										 | 'use strict'; | ||
|  | 
 | ||
|  | module.exports = exports = install; | ||
|  | 
 | ||
|  | exports.usage = 'Attempts to install pre-built binary for module'; | ||
|  | 
 | ||
|  | const fs = require('fs'); | ||
|  | const path = require('path'); | ||
|  | const log = require('npmlog'); | ||
|  | const existsAsync = fs.exists || path.exists; | ||
|  | const versioning = require('./util/versioning.js'); | ||
|  | const napi = require('./util/napi.js'); | ||
|  | const makeDir = require('make-dir'); | ||
|  | // for fetching binaries
 | ||
|  | const fetch = require('node-fetch'); | ||
|  | const tar = require('tar'); | ||
|  | 
 | ||
|  | let npgVersion = 'unknown'; | ||
|  | try { | ||
|  |   // Read own package.json to get the current node-pre-pyp version.
 | ||
|  |   const ownPackageJSON = fs.readFileSync(path.join(__dirname, '..', 'package.json'), 'utf8'); | ||
|  |   npgVersion = JSON.parse(ownPackageJSON).version; | ||
|  | } catch (e) { | ||
|  |   // do nothing
 | ||
|  | } | ||
|  | 
 | ||
|  | function place_binary(uri, targetDir, opts, callback) { | ||
|  |   log.http('GET', uri); | ||
|  | 
 | ||
|  |   // Try getting version info from the currently running npm.
 | ||
|  |   const envVersionInfo = process.env.npm_config_user_agent || | ||
|  |         'node ' + process.version; | ||
|  | 
 | ||
|  |   const sanitized = uri.replace('+', '%2B'); | ||
|  |   const requestOpts = { | ||
|  |     uri: sanitized, | ||
|  |     headers: { | ||
|  |       'User-Agent': 'node-pre-gyp (v' + npgVersion + ', ' + envVersionInfo + ')' | ||
|  |     }, | ||
|  |     follow_max: 10 | ||
|  |   }; | ||
|  | 
 | ||
|  |   if (opts.cafile) { | ||
|  |     try { | ||
|  |       requestOpts.ca = fs.readFileSync(opts.cafile); | ||
|  |     } catch (e) { | ||
|  |       return callback(e); | ||
|  |     } | ||
|  |   } else if (opts.ca) { | ||
|  |     requestOpts.ca = opts.ca; | ||
|  |   } | ||
|  | 
 | ||
|  |   const proxyUrl = opts.proxy || | ||
|  |                     process.env.http_proxy || | ||
|  |                     process.env.HTTP_PROXY || | ||
|  |                     process.env.npm_config_proxy; | ||
|  |   let agent; | ||
|  |   if (proxyUrl) { | ||
|  |     const ProxyAgent = require('https-proxy-agent'); | ||
|  |     agent = new ProxyAgent(proxyUrl); | ||
|  |     log.http('download', 'proxy agent configured using: "%s"', proxyUrl); | ||
|  |   } | ||
|  | 
 | ||
|  |   fetch(sanitized, { agent }) | ||
|  |     .then((res) => { | ||
|  |       if (!res.ok) { | ||
|  |         throw new Error(`response status ${res.status} ${res.statusText} on ${sanitized}`); | ||
|  |       } | ||
|  |       const dataStream = res.body; | ||
|  | 
 | ||
|  |       return new Promise((resolve, reject) => { | ||
|  |         let extractions = 0; | ||
|  |         const countExtractions = (entry) => { | ||
|  |           extractions += 1; | ||
|  |           log.info('install', 'unpacking %s', entry.path); | ||
|  |         }; | ||
|  | 
 | ||
|  |         dataStream.pipe(extract(targetDir, countExtractions)) | ||
|  |           .on('error', (e) => { | ||
|  |             reject(e); | ||
|  |           }); | ||
|  |         dataStream.on('end', () => { | ||
|  |           resolve(`extracted file count: ${extractions}`); | ||
|  |         }); | ||
|  |         dataStream.on('error', (e) => { | ||
|  |           reject(e); | ||
|  |         }); | ||
|  |       }); | ||
|  |     }) | ||
|  |     .then((text) => { | ||
|  |       log.info(text); | ||
|  |       callback(); | ||
|  |     }) | ||
|  |     .catch((e) => { | ||
|  |       log.error(`install ${e.message}`); | ||
|  |       callback(e); | ||
|  |     }); | ||
|  | } | ||
|  | 
 | ||
|  | function extract(to, onentry) { | ||
|  |   return tar.extract({ | ||
|  |     cwd: to, | ||
|  |     strip: 1, | ||
|  |     onentry | ||
|  |   }); | ||
|  | } | ||
|  | 
 | ||
|  | function extract_from_local(from, targetDir, callback) { | ||
|  |   if (!fs.existsSync(from)) { | ||
|  |     return callback(new Error('Cannot find file ' + from)); | ||
|  |   } | ||
|  |   log.info('Found local file to extract from ' + from); | ||
|  | 
 | ||
|  |   // extract helpers
 | ||
|  |   let extractCount = 0; | ||
|  |   function countExtractions(entry) { | ||
|  |     extractCount += 1; | ||
|  |     log.info('install', 'unpacking ' + entry.path); | ||
|  |   } | ||
|  |   function afterExtract(err) { | ||
|  |     if (err) return callback(err); | ||
|  |     if (extractCount === 0) { | ||
|  |       return callback(new Error('There was a fatal problem while extracting the tarball')); | ||
|  |     } | ||
|  |     log.info('tarball', 'done parsing tarball'); | ||
|  |     callback(); | ||
|  |   } | ||
|  | 
 | ||
|  |   fs.createReadStream(from).pipe(extract(targetDir, countExtractions)) | ||
|  |     .on('close', afterExtract) | ||
|  |     .on('error', afterExtract); | ||
|  | } | ||
|  | 
 | ||
|  | function do_build(gyp, argv, callback) { | ||
|  |   const args = ['rebuild'].concat(argv); | ||
|  |   gyp.todo.push({ name: 'build', args: args }); | ||
|  |   process.nextTick(callback); | ||
|  | } | ||
|  | 
 | ||
|  | function print_fallback_error(err, opts, package_json) { | ||
|  |   const fallback_message = ' (falling back to source compile with node-gyp)'; | ||
|  |   let full_message = ''; | ||
|  |   if (err.statusCode !== undefined) { | ||
|  |     // If we got a network response it but failed to download
 | ||
|  |     // it means remote binaries are not available, so let's try to help
 | ||
|  |     // the user/developer with the info to debug why
 | ||
|  |     full_message = 'Pre-built binaries not found for ' + package_json.name + '@' + package_json.version; | ||
|  |     full_message += ' and ' + opts.runtime + '@' + (opts.target || process.versions.node) + ' (' + opts.node_abi + ' ABI, ' + opts.libc + ')'; | ||
|  |     full_message += fallback_message; | ||
|  |     log.warn('Tried to download(' + err.statusCode + '): ' + opts.hosted_tarball); | ||
|  |     log.warn(full_message); | ||
|  |     log.http(err.message); | ||
|  |   } else { | ||
|  |     // If we do not have a statusCode that means an unexpected error
 | ||
|  |     // happened and prevented an http response, so we output the exact error
 | ||
|  |     full_message = 'Pre-built binaries not installable for ' + package_json.name + '@' + package_json.version; | ||
|  |     full_message += ' and ' + opts.runtime + '@' + (opts.target || process.versions.node) + ' (' + opts.node_abi + ' ABI, ' + opts.libc + ')'; | ||
|  |     full_message += fallback_message; | ||
|  |     log.warn(full_message); | ||
|  |     log.warn('Hit error ' + err.message); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | //
 | ||
|  | // install
 | ||
|  | //
 | ||
|  | function install(gyp, argv, callback) { | ||
|  |   const package_json = gyp.package_json; | ||
|  |   const napi_build_version = napi.get_napi_build_version_from_command_args(argv); | ||
|  |   const source_build = gyp.opts['build-from-source'] || gyp.opts.build_from_source; | ||
|  |   const update_binary = gyp.opts['update-binary'] || gyp.opts.update_binary; | ||
|  |   const should_do_source_build = source_build === package_json.name || (source_build === true || source_build === 'true'); | ||
|  |   if (should_do_source_build) { | ||
|  |     log.info('build', 'requesting source compile'); | ||
|  |     return do_build(gyp, argv, callback); | ||
|  |   } else { | ||
|  |     const fallback_to_build = gyp.opts['fallback-to-build'] || gyp.opts.fallback_to_build; | ||
|  |     let should_do_fallback_build = fallback_to_build === package_json.name || (fallback_to_build === true || fallback_to_build === 'true'); | ||
|  |     // but allow override from npm
 | ||
|  |     if (process.env.npm_config_argv) { | ||
|  |       const cooked = JSON.parse(process.env.npm_config_argv).cooked; | ||
|  |       const match = cooked.indexOf('--fallback-to-build'); | ||
|  |       if (match > -1 && cooked.length > match && cooked[match + 1] === 'false') { | ||
|  |         should_do_fallback_build = false; | ||
|  |         log.info('install', 'Build fallback disabled via npm flag: --fallback-to-build=false'); | ||
|  |       } | ||
|  |     } | ||
|  |     let opts; | ||
|  |     try { | ||
|  |       opts = versioning.evaluate(package_json, gyp.opts, napi_build_version); | ||
|  |     } catch (err) { | ||
|  |       return callback(err); | ||
|  |     } | ||
|  | 
 | ||
|  |     opts.ca = gyp.opts.ca; | ||
|  |     opts.cafile = gyp.opts.cafile; | ||
|  | 
 | ||
|  |     const from = opts.hosted_tarball; | ||
|  |     const to = opts.module_path; | ||
|  |     const binary_module = path.join(to, opts.module_name + '.node'); | ||
|  |     existsAsync(binary_module, (found) => { | ||
|  |       if (!update_binary) { | ||
|  |         if (found) { | ||
|  |           console.log('[' + package_json.name + '] Success: "' + binary_module + '" already installed'); | ||
|  |           console.log('Pass --update-binary to reinstall or --build-from-source to recompile'); | ||
|  |           return callback(); | ||
|  |         } | ||
|  |         log.info('check', 'checked for "' + binary_module + '" (not found)'); | ||
|  |       } | ||
|  | 
 | ||
|  |       makeDir(to).then(() => { | ||
|  |         const fileName = from.startsWith('file://') && from.slice('file://'.length); | ||
|  |         if (fileName) { | ||
|  |           extract_from_local(fileName, to, after_place); | ||
|  |         } else { | ||
|  |           place_binary(from, to, opts, after_place); | ||
|  |         } | ||
|  |       }).catch((err) => { | ||
|  |         after_place(err); | ||
|  |       }); | ||
|  | 
 | ||
|  |       function after_place(err) { | ||
|  |         if (err && should_do_fallback_build) { | ||
|  |           print_fallback_error(err, opts, package_json); | ||
|  |           return do_build(gyp, argv, callback); | ||
|  |         } else if (err) { | ||
|  |           return callback(err); | ||
|  |         } else { | ||
|  |           console.log('[' + package_json.name + '] Success: "' + binary_module + '" is installed via remote'); | ||
|  |           return callback(); | ||
|  |         } | ||
|  |       } | ||
|  |     }); | ||
|  |   } | ||
|  | } |