Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions lib/spdy/agent.js
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,7 @@ proto._getCreateSocket = function _getCreateSocket () {
}

proto._connect = function _connect (options, callback) {
var self = this
var state = this._spdyState

var protocols = state.options.protocols || [
Expand Down Expand Up @@ -158,6 +159,10 @@ proto._connect = function _connect (options, callback) {
return
}

connection.on('error', function (err) {
self.emit('error', err)
})

if (state.options['x-forwarded-for'] !== undefined) {
connection.sendXForwardedFor(state.options['x-forwarded-for'])
}
Expand Down
298 changes: 298 additions & 0 deletions lib/spdy/deceiver.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,298 @@
// Vendored from http-deceiver@1.2.7 (indutny/http-deceiver, MIT).
// Voxer changes from upstream:
// 1. `process.binding('http_parser')` was removed in Node 16+
// ("No such module: http_parser"). When that fails we now load
// `HTTPParser` from `node:_http_common` (Node 16+) and the
// methods list from `require('http').METHODS`, keeping the
// `mode === 'modern'` numeric kOn* path. The previous fallback
// to `mode = 'unsupported'` used string event names that the
// modern parser does not expose, so `parser[kOnHeadersComplete]`
// was undefined and any request crashed the process with
// `TypeError: this[kOnHeadersComplete] is not a function`.
// 2. `new Buffer(0)` -> `Buffer.alloc(0)` (the legacy form is removed in
// modern Node).

var assert = require('assert')

var Buffer = require('buffer').Buffer

// Node.js version
var mode = /^v0\.8\./.test(process.version) ? 'rusty'
: /^v0\.(9|10)\./.test(process.version) ? 'old'
: /^v0\.12\./.test(process.version) ? 'normal'
: 'modern'

var HTTPParser

var methods
var reverseMethods

var kOnHeaders
var kOnHeadersComplete
var kOnMessageComplete
var kOnBody

if (mode === 'normal' || mode === 'modern') {
try {
try {
HTTPParser = process.binding('http_parser').HTTPParser
} catch (e) {
// Node 16+: process.binding('http_parser') is gone. The same
// HTTPParser constructor (with kOn* slot indices) is exported
// from the internal _http_common module.
try {
HTTPParser = require('node:_http_common').HTTPParser
} catch (e2) {
HTTPParser = require('_http_common').HTTPParser
}
}
methods = HTTPParser.methods

// v6
if (!methods) {
try {
methods = process.binding('http_parser').methods
} catch (e) {
// Node 16+: HTTPParser does not carry .methods anymore.
// _http_common exports the parser-indexed list (DELETE, GET,
// HEAD, POST, ...). http.METHODS is alphabetically sorted, so
// using it here would shift indices and the deceiver would tell
// Node the wrong method (e.g. POST -> ACL).
try {
methods = require('node:_http_common').methods
} catch (e2) {
methods = require('_http_common').methods
}
}
}

reverseMethods = {}

methods.forEach(function (method, index) {
reverseMethods[method] = index
})

kOnHeaders = HTTPParser.kOnHeaders | 0
kOnHeadersComplete = HTTPParser.kOnHeadersComplete | 0
kOnMessageComplete = HTTPParser.kOnMessageComplete | 0
kOnBody = HTTPParser.kOnBody | 0
} catch (e) {
mode = 'unsupported'
}
}
if (mode !== 'normal' && mode !== 'modern') {
kOnHeaders = 'onHeaders'
kOnHeadersComplete = 'onHeadersComplete'
kOnMessageComplete = 'onMessageComplete'
kOnBody = 'onBody'
}

function Deceiver (socket, options) {
this.socket = socket
this.options = options || {}
this.isClient = this.options.isClient
}
module.exports = Deceiver

Deceiver.create = function create (stream, options) {
return new Deceiver(stream, options)
}

Deceiver.prototype._toHeaderList = function _toHeaderList (object) {
var out = []
var keys = Object.keys(object)

for (var i = 0; i < keys.length; i++) {
out.push(keys[i], object[keys[i]])
}

return out
}

Deceiver.prototype._isUpgrade = function _isUpgrade (request) {
return request.method === 'CONNECT' ||
request.headers.upgrade ||
(request.headers.connection &&
/(^|\W)upgrade(\W|$)/i.test(request.headers.connection))
}

// TODO(indutny): support CONNECT
if (mode === 'modern') {
/*
function parserOnHeadersComplete(versionMajor, versionMinor, headers, method,
url, statusCode, statusMessage, upgrade,
shouldKeepAlive) {
*/
Deceiver.prototype.emitRequest = function emitRequest (request) {
var parser = this.socket.parser
assert(parser, 'No parser present')

parser.execute = null

var self = this
var method = reverseMethods[request.method]
parser.execute = function execute () {
self._skipExecute(this)
this[kOnHeadersComplete](1,
1,
self._toHeaderList(request.headers),
method,
request.path,
0,
'',
self._isUpgrade(request),
true)
return 0
}

this._emitEmpty()
}

Deceiver.prototype.emitResponse = function emitResponse (response) {
var parser = this.socket.parser
assert(parser, 'No parser present')

parser.execute = null

var self = this
parser.execute = function execute () {
self._skipExecute(this)
this[kOnHeadersComplete](1,
1,
self._toHeaderList(response.headers),
response.path,
response.code,
response.status,
response.reason || '',
self._isUpgrade(response),
true)
return 0
}

this._emitEmpty()
}
} else {
/*
`function parserOnHeadersComplete(info) {`

info = { .versionMajor, .versionMinor, .url, .headers, .method,
.statusCode, .statusMessage, .upgrade, .shouldKeepAlive }
*/
Deceiver.prototype.emitRequest = function emitRequest (request) {
var parser = this.socket.parser
assert(parser, 'No parser present')

var method = request.method
if (reverseMethods) {
method = reverseMethods[method]
}

var info = {
versionMajor: 1,
versionMinor: 1,
url: request.path,
headers: this._toHeaderList(request.headers),
method: method,
statusCode: 0,
statusMessage: '',
upgrade: this._isUpgrade(request),
shouldKeepAlive: true
}

var self = this
parser.execute = function execute () {
self._skipExecute(this)
this[kOnHeadersComplete](info)
return 0
}

this._emitEmpty()
}

Deceiver.prototype.emitResponse = function emitResponse (response) {
var parser = this.socket.parser
assert(parser, 'No parser present')

var info = {
versionMajor: 1,
versionMinor: 1,
url: response.path,
headers: this._toHeaderList(response.headers),
method: false,
statusCode: response.status,
statusMessage: response.reason || '',
upgrade: this._isUpgrade(response),
shouldKeepAlive: true
}

var self = this
parser.execute = function execute () {
self._skipExecute(this)
this[kOnHeadersComplete](info)
return 0
}

this._emitEmpty()
}
}

Deceiver.prototype._skipExecute = function _skipExecute (parser) {
var self = this
var oldExecute = parser.constructor.prototype.execute
var oldFinish = parser.constructor.prototype.finish

parser.execute = null
parser.finish = null

parser.execute = function execute (buffer, start, len) {
// Parser reuse
if (this.socket !== self.socket) {
this.execute = oldExecute
this.finish = oldFinish
return this.execute(buffer, start, len)
}

if (start !== undefined) {
buffer = buffer.slice(start, start + len)
}
self.emitBody(buffer)
return len
}

parser.finish = function finish () {
// Parser reuse
if (this.socket !== self.socket) {
this.execute = oldExecute
this.finish = oldFinish
return this.finish()
}

this.execute = oldExecute
this.finish = oldFinish
self.emitMessageComplete()
}
}

Deceiver.prototype.emitBody = function emitBody (buffer) {
var parser = this.socket.parser
assert(parser, 'No parser present')

parser[kOnBody](buffer, 0, buffer.length)
}

Deceiver.prototype._emitEmpty = function _emitEmpty () {
// Emit data to force out handling of UPGRADE
var empty = Buffer.alloc(0)
if (this.socket.ondata) {
this.socket.ondata(empty, 0, 0)
} else {
this.socket.emit('data', empty)
}
}

Deceiver.prototype.emitMessageComplete = function emitMessageComplete () {
var parser = this.socket.parser
assert(parser, 'No parser present')

parser[kOnMessageComplete]()
}
5 changes: 4 additions & 1 deletion lib/spdy/handle.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@

var assert = require('assert')
var thing = require('handle-thing')
var httpDeceiver = require('http-deceiver')
var httpDeceiver = require('./deceiver')
var util = require('util')

function Handle (options, stream, socket) {
Expand Down Expand Up @@ -30,6 +30,9 @@ function Handle (options, stream, socket) {
if (!state.stream) {
this.on('stream', function (stream) {
state.stream = stream
stream.on('end', function () {
state.ending = true
})
})
}
}
Expand Down
5 changes: 2 additions & 3 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -35,11 +35,10 @@
],
"dependencies": {
"debug": "^2.6.8",
"handle-thing": "^1.2.5",
"http-deceiver": "^1.2.7",
"handle-thing": "^2.0.1",
"safe-buffer": "^5.0.1",
"select-hose": "^2.0.0",
"spdy-transport": "^2.0.18"
"spdy-transport": "Voxer/spdy-transport#voxer"
},
"devDependencies": {
"istanbul": "^0.4.5",
Expand Down