2017-05-25 18:25:42 +00:00
|
|
|
require('isomorphic-fetch')
|
|
|
|
const gunzip = require('gunzip-maybe')
|
|
|
|
const mkdirp = require('mkdirp')
|
|
|
|
const tar = require('tar-fs')
|
|
|
|
const RegistryCache = require('./RegistryCache')
|
|
|
|
|
|
|
|
const getPackageInfoFromRegistry = (registryURL, packageName) => {
|
|
|
|
let encodedPackageName
|
|
|
|
if (packageName.charAt(0) === '@') {
|
|
|
|
encodedPackageName = `@${encodeURIComponent(packageName.substring(1))}`
|
|
|
|
} else {
|
|
|
|
encodedPackageName = encodeURIComponent(packageName)
|
|
|
|
}
|
|
|
|
|
|
|
|
const url = `${registryURL}/${encodedPackageName}`
|
|
|
|
|
|
|
|
return fetch(url, {
|
|
|
|
headers: { 'Accept': 'application/json' }
|
|
|
|
}).then(response => (
|
|
|
|
response.status === 404 ? null : response.json()
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
const PackageNotFound = 'PackageNotFound'
|
|
|
|
|
|
|
|
const getPackageInfo = (registryURL, packageName, callback) => {
|
2017-05-25 18:53:47 +00:00
|
|
|
RegistryCache.get(packageName, (error, value) => {
|
2017-05-25 18:25:42 +00:00
|
|
|
if (error) {
|
|
|
|
callback(error)
|
|
|
|
} else if (value) {
|
|
|
|
callback(null, value === PackageNotFound ? null : value)
|
|
|
|
} else {
|
|
|
|
getPackageInfoFromRegistry(registryURL, packageName).then(value => {
|
|
|
|
if (value == null) {
|
|
|
|
// Keep 404s in the cache for 5 minutes. This prevents us
|
|
|
|
// from making unnecessary requests to the registry for
|
|
|
|
// bad package names. In the worst case, a brand new
|
|
|
|
// package's info will be available within 5 minutes.
|
2017-05-25 18:53:47 +00:00
|
|
|
RegistryCache.set(packageName, PackageNotFound, 300)
|
2017-05-25 18:25:42 +00:00
|
|
|
} else {
|
2017-05-25 18:53:47 +00:00
|
|
|
// Keep package.json in the cache for a minute.
|
|
|
|
RegistryCache.set(packageName, value, 60)
|
2017-05-25 18:25:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
callback(null, value)
|
|
|
|
}, error => {
|
|
|
|
// Do not cache errors.
|
2017-05-25 18:53:47 +00:00
|
|
|
RegistryCache.del(packageName)
|
2017-05-25 18:25:42 +00:00
|
|
|
callback(error)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
const normalizeTarHeader = (header) => {
|
|
|
|
// Most packages have header names that look like "package/index.js"
|
|
|
|
// so we shorten that to just "index.js" here. A few packages use a
|
|
|
|
// prefix other than "package/". e.g. the firebase package uses the
|
|
|
|
// "firebase_npm/" prefix. So we just strip the first dir name.
|
|
|
|
header.name = header.name.replace(/^[^\/]+\//, '')
|
|
|
|
return header
|
|
|
|
}
|
|
|
|
|
|
|
|
const getPackage = (tarballURL, outputDir, callback) => {
|
|
|
|
mkdirp(outputDir, (error) => {
|
|
|
|
if (error) {
|
|
|
|
callback(error)
|
|
|
|
} else {
|
|
|
|
let callbackWasCalled = false
|
|
|
|
|
|
|
|
fetch(tarballURL).then(response => {
|
|
|
|
response.body
|
|
|
|
.pipe(gunzip())
|
|
|
|
.pipe(
|
|
|
|
tar.extract(outputDir, {
|
|
|
|
dmode: 0o666, // All dirs should be writable
|
|
|
|
fmode: 0o444, // All files should be readable
|
|
|
|
map: normalizeTarHeader
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.on('finish', callback)
|
|
|
|
.on('error', (error) => {
|
|
|
|
if (callbackWasCalled) // LOL node streams
|
|
|
|
return
|
|
|
|
|
|
|
|
callbackWasCalled = true
|
|
|
|
callback(error)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
getPackageInfo,
|
|
|
|
getPackage
|
|
|
|
}
|