"use strict"; var JSONStream = require('JSONStream') var Error = require('http-errors') var request = require('request') var Stream = require('readable-stream') var URL = require('url') var parse_interval = require('./config').parse_interval var Logger = require('./logger') var MyStreams = require('./streams') var Utils = require('./utils') var encode = function(thing) { return encodeURIComponent(thing).replace(/^%40/, '@'); }; module.exports = Storage // // Implements Storage interface // (same for storage.js, local-storage.js, up-storage.js) // function Storage(config, mainconfig) { var self = Object.create(Storage.prototype) self.config = config self.failed_requests = 0 self.userAgent = mainconfig.user_agent self.ca = config.ca self.logger = Logger.logger.child({sub: 'out'}) self.server_id = mainconfig.server_id self.url = URL.parse(self.config.url) _setupProxy.call(self, self.url.hostname, config, mainconfig, self.url.protocol === 'https:') self.config.url = self.config.url.replace(/\/$/, '') if (Number(self.config.timeout) >= 1000) { self.logger.warn([ 'Too big timeout value: ' + self.config.timeout, 'We changed time format to nginx-like one', '(see http://wiki.nginx.org/ConfigNotation)', 'so please update your config accordingly' ].join('\n')) } // a bunch of different configurable timers self.maxage = parse_interval(config_get('maxage' , '2m' )) self.timeout = parse_interval(config_get('timeout' , '30s')) self.max_fails = Number(config_get('max_fails' , 2 )) self.fail_timeout = parse_interval(config_get('fail_timeout', '5m' )) return self // just a helper (`config[key] || default` doesn't work because of zeroes) function config_get(key, def) { return config[key] != null ? config[key] : def } } function _setupProxy(hostname, config, mainconfig, isHTTPS) { var no_proxy var proxy_key = isHTTPS ? 'https_proxy' : 'http_proxy' // get http_proxy and no_proxy configs if (proxy_key in config) { this.proxy = config[proxy_key] } else if (proxy_key in mainconfig) { this.proxy = mainconfig[proxy_key] } if ('no_proxy' in config) { no_proxy = config.no_proxy } else if ('no_proxy' in mainconfig) { no_proxy = mainconfig.no_proxy } // use wget-like algorithm to determine if proxy shouldn't be used if (hostname[0] !== '.') { hostname = '.' + hostname; } if (typeof(no_proxy) === 'string' && no_proxy.length) { no_proxy = no_proxy.split(',') } if (Array.isArray(no_proxy)) { for (let i=0; i= this.max_fails && Math.abs(Date.now() - this.last_request_time) < this.fail_timeout) { return false } else { return true } } else { if (alive) { if (this.failed_requests >= this.max_fails) { this.logger.warn({ host: this.url.host }, 'host @{host} is back online') } this.failed_requests = 0 } else { this.failed_requests++ if (this.failed_requests === this.max_fails) { this.logger.warn({ host: this.url.host }, 'host @{host} is now offline') } } this.last_request_time = Date.now() } } Storage.prototype.can_fetch_url = function(url) { url = URL.parse(url) return url.protocol === this.url.protocol && url.host === this.url.host && url.path.indexOf(this.url.path) === 0 } Storage.prototype.get_package = function(name, options, callback) { if (typeof(options) === 'function') callback = options, options = {} var headers = {} if (options.etag) { headers['If-None-Match'] = options.etag headers['Accept'] = 'application/octet-stream' } this.request({ uri : '/' + encode(name), json : true, headers : headers, req : options.req, }, function(err, res, body) { if (err) return callback(err) if (res.statusCode === 404) { return callback( Error[404]("package doesn't exist on uplink") ) } if (!(res.statusCode >= 200 && res.statusCode < 300)) { var error = Error('bad status code: ' + res.statusCode) error.remoteStatus = res.statusCode return callback(error) } callback(null, body, res.headers.etag) }) } Storage.prototype.get_tarball = function(name, options, filename) { if (!options) options = {} return this.get_url(this.config.url + '/' + name + '/-/' + filename) } Storage.prototype.get_url = function(url) { var stream = MyStreams.ReadTarballStream() stream.abort = function() {} var current_length = 0, expected_length var rstream = this.request({ uri_full: url, encoding: null, headers: { Accept: 'application/octet-stream' }, }) rstream.on('response', function(res) { if (res.statusCode === 404) { return stream.emit('error', Error[404]("file doesn't exist on uplink")) } if (!(res.statusCode >= 200 && res.statusCode < 300)) { return stream.emit('error', Error('bad uplink status code: ' + res.statusCode)) } if (res.headers['content-length']) { expected_length = res.headers['content-length'] stream.emit('content-length', res.headers['content-length']) } rstream.pipe(stream) }) rstream.on('error', function(err) { stream.emit('error', err) }) rstream.on('data', function(d) { current_length += d.length }) rstream.on('end', function(d) { if (d) current_length += d.length if (expected_length && current_length != expected_length) stream.emit('error', Error('content length mismatch')) }) return stream } Storage.prototype.search = function(startkey, options) { var self = this var stream = new Stream.PassThrough({ objectMode: true }) var req = self.request({ uri: options.req.url, req: options.req, headers: { referer: options.req.headers.referer } }) req.on('response', function (res) { if (!String(res.statusCode).match(/^2\d\d$/)) { return stream.emit('error', Error('bad status code ' + res.statusCode + ' from uplink')) } res.pipe(JSONStream.parse('*')).on('data', function (pkg) { if (Utils.is_object(pkg)) { stream.emit('data', pkg) } }) res.on('end', function () { stream.emit('end') }) }) req.on('error', function (err) { stream.emit('error', err) }) stream.abort = function () { req.abort() stream.emit('end') } return stream } Storage.prototype._add_proxy_headers = function(req, headers) { if (req) { // Only submit X-Forwarded-For field if we don't have a proxy selected // in the config file. // // Otherwise misconfigured proxy could return 407: // https://github.com/rlidwka/sinopia/issues/254 // if (!this.proxy) { headers['X-Forwarded-For'] = ( req && req.headers['x-forwarded-for'] ? req.headers['x-forwarded-for'] + ', ' : '' ) + req.connection.remoteAddress } } // always attach Via header to avoid loops, even if we're not proxying headers['Via'] = req && req.headers['via'] ? req.headers['via'] + ', ' : '' headers['Via'] += '1.1 ' + this.server_id + ' (Verdaccio)' }