index.js 4.7 KB

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