index.js 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. var httpProxy = module.exports,
  2. extend = require('util')._extend,
  3. parse_url = require('url').parse,
  4. EE3 = require('eventemitter3'),
  5. http = require('http'),
  6. https = require('https'),
  7. web = require('./passes/web-incoming'),
  8. ws = require('./passes/ws-incoming');
  9. httpProxy.Server = ProxyServer;
  10. /**
  11. * Returns a function that creates the loader for
  12. * either `ws` or `web`'s passes.
  13. *
  14. * Examples:
  15. *
  16. * httpProxy.createRightProxy('ws')
  17. * // => [Function]
  18. *
  19. * @param {String} Type Either 'ws' or 'web'
  20. * @return {Function} Loader Function that when called returns an iterator for the right passes
  21. *
  22. * @api private
  23. */
  24. function createRightProxy(type) {
  25. return function(options) {
  26. //console.log('options', options)
  27. return function(req, res /*, [head], [opts] */) {
  28. var passes = (type === 'ws') ? this.wsPasses : this.webPasses,
  29. args = [].slice.call(arguments),
  30. cntr = args.length - 1,
  31. head, cbl;
  32. /* optional args parse begin */
  33. if(typeof args[cntr] === 'function') {
  34. cbl = args[cntr];
  35. cntr--;
  36. }
  37. var requestOptions = options;
  38. if(
  39. !(args[cntr] instanceof Buffer) &&
  40. args[cntr] !== res
  41. ) {
  42. //Copy global options
  43. requestOptions = extend({}, options);
  44. //Overwrite with request options
  45. extend(requestOptions, args[cntr]);
  46. cntr--;
  47. }
  48. if(args[cntr] instanceof Buffer) {
  49. head = args[cntr];
  50. }
  51. /* optional args parse end */
  52. ['target', 'forward'].forEach(function(e) {
  53. if (typeof requestOptions[e] === 'string')
  54. requestOptions[e] = parse_url(requestOptions[e]);
  55. });
  56. if (!requestOptions.target && !requestOptions.forward) {
  57. return this.emit('error', new Error('Must provide a proper URL as target'));
  58. }
  59. for(var i=0; i < passes.length; i++) {
  60. /**
  61. * Call of passes functions
  62. * pass(req, res, options, head)
  63. *
  64. * In WebSockets case the `res` variable
  65. * refer to the connection socket
  66. * pass(req, socket, options, head)
  67. */
  68. if(passes[i](req, res, requestOptions, head, this, cbl)) { // passes can return a truthy value to halt the loop
  69. break;
  70. }
  71. }
  72. };
  73. };
  74. }
  75. httpProxy.createRightProxy = createRightProxy;
  76. function ProxyServer(options) {
  77. EE3.call(this);
  78. options = options || {};
  79. options.prependPath = options.prependPath === false ? false : true;
  80. this.web = this.proxyRequest = createRightProxy('web')(options);
  81. this.ws = this.proxyWebsocketRequest = createRightProxy('ws')(options);
  82. this.options = options;
  83. this.webPasses = Object.keys(web).map(function(pass) {
  84. return web[pass];
  85. });
  86. this.wsPasses = Object.keys(ws).map(function(pass) {
  87. return ws[pass];
  88. });
  89. this.on('error', this.onError, this);
  90. }
  91. require('util').inherits(ProxyServer, EE3);
  92. ProxyServer.prototype.onError = function (err) {
  93. //
  94. // Remark: Replicate node core behavior using EE3
  95. // so we force people to handle their own errors
  96. //
  97. if(this.listeners('error').length === 1) {
  98. throw err;
  99. }
  100. };
  101. ProxyServer.prototype.listen = function(port, hostname) {
  102. var self = this,
  103. closure = function(req, res) { self.web(req, res); };
  104. this._server = this.options.ssl ?
  105. https.createServer(this.options.ssl, closure) :
  106. http.createServer(closure);
  107. if(this.options.ws) {
  108. this._server.on('upgrade', function(req, socket, head) { self.ws(req, socket, head); });
  109. }
  110. this._server.listen(port, hostname);
  111. return this;
  112. };
  113. ProxyServer.prototype.close = function(callback) {
  114. var self = this;
  115. if (this._server) {
  116. this._server.close(done);
  117. }
  118. // Wrap callback to nullify server after all open connections are closed.
  119. function done() {
  120. self._server = null;
  121. if (callback) {
  122. callback.apply(null, arguments);
  123. }
  124. };
  125. };
  126. ProxyServer.prototype.before = function(type, passName, callback) {
  127. if (type !== 'ws' && type !== 'web') {
  128. throw new Error('type must be `web` or `ws`');
  129. }
  130. var passes = (type === 'ws') ? this.wsPasses : this.webPasses,
  131. i = false;
  132. passes.forEach(function(v, idx) {
  133. if(v.name === passName) i = idx;
  134. })
  135. if(i === false) throw new Error('No such pass');
  136. passes.splice(i, 0, callback);
  137. };
  138. ProxyServer.prototype.after = function(type, passName, callback) {
  139. if (type !== 'ws' && type !== 'web') {
  140. throw new Error('type must be `web` or `ws`');
  141. }
  142. var passes = (type === 'ws') ? this.wsPasses : this.webPasses,
  143. i = false;
  144. passes.forEach(function(v, idx) {
  145. if(v.name === passName) i = idx;
  146. })
  147. if(i === false) throw new Error('No such pass');
  148. passes.splice(i++, 0, callback);
  149. };