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.
 
 

130 lines
3.6 KiB

var express = require('express');
var Promise = require('bluebird');
var config = require( __dirname + '/./config' );
var utils = require( __dirname + '/./utils');
var NPM_PATH = config.NPM_PATH;
var REGISTRY_NAME = config.REGISTRY_NAME;
var ENABLE_NPM_FAILOVER = config.ENABLE_NPM_FAILOVER;
var fetchAndCacheMetadata = utils.fetchAndCacheMetadata;
var fetchAndCacheTarball = utils.fetchAndCacheTarball;
var patchData = utils.patchData;
var fileExists = utils.fileExists;
var readFile = utils.readFile;
var app = express();
app.use( function(req, res, next ){
res._log = {
method: req.method,
path: req.path,
cacheHit: 'Hit',
cacheFile: '',
};
res.on( 'finish', function(){
var log = this._log;
console.log( log.cacheHit, this.statusCode, log.method, log.path, ' => ', log.cacheFile );
});
// setTimeout( next, 1000 );
next();
});
app.get( '/:package', function( req, res, next ){
var packageName = req.params.package;
var cacheFile = [ NPM_PATH, REGISTRY_NAME, packageName, '.cache.json' ].join( '/' );
return fileExists( cacheFile )
.tap( function( isExists ){
if( !isExists ){
if ( !ENABLE_NPM_FAILOVER ) {
res._log.cacheHit = '!!!';
return Promise.reject( { status: 404 });
}
res._log.cacheHit = '---';
return fetchAndCacheMetadata( packageName, cacheFile );
}
})
.then( function( ){
res._log.cacheFile = cacheFile;
return readFile( cacheFile, 'utf-8' );
})
.then( function( cachedData ){
cachedData = JSON.parse( cachedData );
patchData( cachedData );
return res.send( cachedData );
})
.catch( next );
});
app.get( '/:package/-/:tarball', function( req, res, next ){
var packageName = req.params.package;
var version = req.params.tarball.match( new RegExp( packageName + '-(.*).tgz') )[1];
var packagePath = [ NPM_PATH , packageName, version, 'package.tgz'].join( '/' );
fileExists( packagePath )
.tap( function( isExists ){
if( !isExists ){
if ( !ENABLE_NPM_FAILOVER ) {
res._log.cacheHit = '!!!';
return Promise.reject( { status: 404 });
}
res._log.cacheHit = '---';
return fetchAndCacheTarball( packageName, version, packagePath );
}
})
.then( function( ){
res._log.cacheFile = packagePath;
return res.sendFile( packagePath );
})
.catch( next );
});
app.get( '/:scope/:package/-/:tarball', function( req, res, next ){
var scopeName = req.params.scope;
var packageName = req.params.package;
var version = req.params.tarball.match( new RegExp( packageName + '-(.*).tgz') )[1];
var packagePath = [ NPM_PATH , scopeName + "/" + packageName, version, 'package.tgz'].join( '/' );
fileExists( packagePath )
.tap( function( isExists ){
if( !isExists ){
if ( !ENABLE_NPM_FAILOVER ) {
res._log.cacheHit = '!!!';
return Promise.reject( { status: 404 });
}
res._log.cacheHit = '---';
return fetchAndCacheTarball( packageName, version, packagePath, scopeName );
}
})
.then( function( ){
res._log.cacheFile = packagePath;
return res.sendFile( packagePath );
})
.catch( next );
});
// catch 404 and forward to error handler
app.use(function(req, res, next) {
next({ status: 404 });
});
// error handlers
app.use(function(err, req, res, next) {
var message = err.message || err;
err.stack && console.log( err.stack );
res.status(err.status || 500);
// NPM registry returns empty objects for unknown packages
if( err.status == 404 ){
message = {};
}
res.send( message );
if( next ) { next(); }
});
module.exports = app;