'use strict'; const Stream = require('stream'); const Readable = Stream.Readable; const binding = process.binding('http2'); const constants = binding.constants; const { ERR_HTTP2_HEADERS_SENT, ERR_HTTP2_INFO_STATUS_NOT_ALLOWED, ERR_HTTP2_INVALID_HEADER_VALUE, ERR_HTTP2_INVALID_STREAM, ERR_HTTP2_NO_SOCKET_MANIPULATION, ERR_HTTP2_PSEUDOHEADER_NOT_ALLOWED, ERR_HTTP2_STATUS_INVALID, ERR_INVALID_ARG_TYPE, ERR_INVALID_ARG_VALUE, ERR_INVALID_CALLBACK, ERR_INVALID_HTTP_TOKEN } = require('internal/errors').codes; const { kSocket } = require('internal/http2/util'); const kFinish = Symbol('finish'); const kBeginSend = Symbol('begin-send'); const kState = Symbol('state'); const kStream = Symbol('stream'); const kRequest = Symbol('request'); const kResponse = Symbol('response'); const kHeaders = Symbol('headers'); const kRawHeaders = Symbol('rawHeaders'); const kTrailers = Symbol('trailers'); const kRawTrailers = Symbol('rawTrailers'); const kProxySocket = Symbol('proxySocket'); const kSetHeader = Symbol('setHeader'); const { HTTP2_HEADER_AUTHORITY, HTTP2_HEADER_METHOD, HTTP2_HEADER_PATH, HTTP2_HEADER_SCHEME, HTTP2_HEADER_STATUS, HTTP_STATUS_CONTINUE, HTTP_STATUS_EXPECTATION_FAILED, HTTP_STATUS_METHOD_NOT_ALLOWED, HTTP_STATUS_OK } = constants; let statusMessageWarned = false; // Defines and implements an API compatibility layer on top of the core // HTTP/2 implementation, intended to provide an interface that is as // close as possible to the current require('http') API function assertValidHeader(name, value) { let err; if (name === '' || typeof name !== 'string') { err = new ERR_INVALID_HTTP_TOKEN('Header name', name); } else if (isPseudoHeader(name)) { err = new ERR_HTTP2_PSEUDOHEADER_NOT_ALLOWED(); } else if (value === undefined || value === null) { err = new ERR_HTTP2_INVALID_HEADER_VALUE(value, name); } if (err !== undefined) { Error.captureStackTrace(err, assertValidHeader); throw err; } } function isPseudoHeader(name) { switch (name) { case HTTP2_HEADER_STATUS: // :status case HTTP2_HEADER_METHOD: // :method case HTTP2_HEADER_PATH: // :path case HTTP2_HEADER_AUTHORITY: // :authority case HTTP2_HEADER_SCHEME: // :scheme return true; default: return false; } } function statusMessageWarn() { if (statusMessageWarned === false) { process.emitWarning( 'Status message is not supported by HTTP/2 (RFC7540 8.1.2.4)', 'UnsupportedWarning' ); statusMessageWarned = true; } } function onStreamData(chunk) { const request = this[kRequest]; if (request !== undefined && !request.push(chunk)) this.pause(); } function onStreamTrailers(trailers, flags, rawTrailers) { const request = this[kRequest]; if (request !== undefined) { Object.assign(request[kTrailers], trailers); request[kRawTrailers].push(...rawTrailers); } } function onStreamEnd() { // Cause the request stream to end as well. const request = this[kRequest]; if (request !== undefined) this[kRequest].push(null); } function onStreamError(error) { // this is purposefully left blank // // errors in compatibility mode are // not forwarded to the request // and response objects. However, // they are forwarded to 'streamError' // on the server by Http2Stream } function onRequestPause() { this[kStream].pause(); } function onRequestResume() { this[kStream].resume(); } function onStreamDrain() { const response = this[kResponse]; if (response !== undefined) response.emit('drain'); } function onStreamAbortedRequest() { const request = this[kRequest]; if (request !== undefined && request[kState].closed === false) { request.emit('aborted'); } } function onStreamAbortedResponse() { // non-op for now } function resumeStream(stream) { stream.resume(); } const proxySocketHandler = { get(stream, prop) { switch (prop) { case 'on': case 'once': case 'end': case 'emit': case 'destroy': return stream[prop].bind(stream); case 'writable': case 'destroyed': return stream[prop]; case 'readable': if (stream.destroyed) return false; const request = stream[kRequest]; return request ? request.readable : stream.readable; case 'setTimeout': const session = stream.session; if (session !== undefined) return session.setTimeout.bind(session); return stream.setTimeout.bind(stream); case 'write': case 'read': case 'pause': case 'resume': throw new ERR_HTTP2_NO_SOCKET_MANIPULATION(); default: const ref = stream.session !== undefined ? stream.session[kSocket] : stream; const value = ref[prop]; return typeof value === 'function' ? value.bind(ref) : value; } }, getPrototypeOf(stream) { if (stream.session !== undefined) return Reflect.getPrototypeOf(stream.session[kSocket]); return Reflect.getPrototypeOf(stream); }, set(stream, prop, value) { switch (prop) { case 'writable': case 'readable': case 'destroyed': case 'on': case 'once': case 'end': case 'emit': case 'destroy': stream[prop] = value; return true; case 'setTimeout': const session = stream.session; if (session !== undefined) session.setTimeout = value; else stream.setTimeout = value; return true; case 'write': case 'read': case 'pause': case 'resume': throw new ERR_HTTP2_NO_SOCKET_MANIPULATION(); default: const ref = stream.session !== undefined ? stream.session[kSocket] : stream; ref[prop] = value; return true; } } }; class Http2ServerRequest extends Readable { constructor(stream, headers, options, rawHeaders) { super(options); this[kState] = { closed: false, didRead: false, }; this[kHeaders] = headers; this[kRawHeaders] = rawHeaders; this[kTrailers] = {}; this[kRawTrailers] = []; this[kStream] = stream; stream[kProxySocket] = null; stream[kRequest] = this; // Pause the stream.. stream.pause(); stream.on('data', onStreamData); stream.on('trailers', onStreamTrailers); stream.on('end', onStreamEnd); stream.on('error', onStreamError); stream.on('aborted', onStreamAbortedRequest); stream.on('close', this[kFinish].bind(this)); this.on('pause', onRequestPause); this.on('resume', onRequestResume); } get complete() { return this._readableState.ended || this[kState].closed || this[kStream].destroyed; } get stream() { return this[kStream]; } get headers() { return this[kHeaders]; } get rawHeaders() { return this[kRawHeaders]; } get trailers() { return this[kTrailers]; } get rawTrailers() { return this[kRawTrailers]; } get httpVersionMajor() { return 2; } get httpVersionMinor() { return 0; } get httpVersion() { return '2.0'; } get socket() { const stream = this[kStream]; const proxySocket = stream[kProxySocket]; if (proxySocket === null) return stream[kProxySocket] = new Proxy(stream, proxySocketHandler); return proxySocket; } get connection() { return this.socket; } _read(nread) { const state = this[kState]; if (!state.closed) { state.didRead = true; process.nextTick(resumeStream, this[kStream]); } else { this.emit('error', new ERR_HTTP2_INVALID_STREAM()); } } get method() { return this[kHeaders][HTTP2_HEADER_METHOD]; } set method(method) { if (typeof method !== 'string') throw new ERR_INVALID_ARG_TYPE('method', 'string', method); if (method.trim() === '') throw new ERR_INVALID_ARG_VALUE('method', method); this[kHeaders][HTTP2_HEADER_METHOD] = method; } get authority() { return this[kHeaders][HTTP2_HEADER_AUTHORITY]; } get scheme() { return this[kHeaders][HTTP2_HEADER_SCHEME]; } get url() { return this[kHeaders][HTTP2_HEADER_PATH]; } set url(url) { this[kHeaders][HTTP2_HEADER_PATH] = url; } setTimeout(msecs, callback) { if (this[kState].closed) return; this[kStream].setTimeout(msecs, callback); } [kFinish]() { const state = this[kState]; if (state.closed) return; state.closed = true; this.push(null); this[kStream][kRequest] = undefined; // if the user didn't interact with incoming data and didn't pipe it, // dump it for compatibility with http1 if (!state.didRead && !this._readableState.resumeScheduled) this.resume(); this.emit('close'); } } class Http2ServerResponse extends Stream { constructor(stream, options) { super(options); this[kState] = { closed: false, ending: false, headRequest: false, sendDate: true, statusCode: HTTP_STATUS_OK, }; this[kHeaders] = Object.create(null); this[kTrailers] = Object.create(null); this[kStream] = stream; stream[kProxySocket] = null; stream[kResponse] = this; this.writable = true; stream.on('drain', onStreamDrain); stream.on('aborted', onStreamAbortedResponse); stream.on('close', this[kFinish].bind(this)); } // User land modules such as finalhandler just check truthiness of this // but if someone is actually trying to use this for more than that // then we simply can't support such use cases get _header() { return this.headersSent; } get finished() { const stream = this[kStream]; return stream.destroyed || stream._writableState.ended || this[kState].closed; } get socket() { // this is compatible with http1 which removes socket reference // only from ServerResponse but not IncomingMessage if (this[kState].closed) return; const stream = this[kStream]; const proxySocket = stream[kProxySocket]; if (proxySocket === null) return stream[kProxySocket] = new Proxy(stream, proxySocketHandler); return proxySocket; } get connection() { return this.socket; } get stream() { return this[kStream]; } get headersSent() { return this[kStream].headersSent; } get sendDate() { return this[kState].sendDate; } set sendDate(bool) { this[kState].sendDate = Boolean(bool); } get statusCode() { return this[kState].statusCode; } set statusCode(code) { code |= 0; if (code >= 100 && code < 200) throw new ERR_HTTP2_INFO_STATUS_NOT_ALLOWED(); if (code < 100 || code > 599) throw new ERR_HTTP2_STATUS_INVALID(code); this[kState].statusCode = code; } setTrailer(name, value) { if (typeof name !== 'string') throw new ERR_INVALID_ARG_TYPE('name', 'string', name); name = name.trim().toLowerCase(); assertValidHeader(name, value); this[kTrailers][name] = value; } addTrailers(headers) { const keys = Object.keys(headers); let key = ''; for (var i = 0; i < keys.length; i++) { key = keys[i]; this.setTrailer(key, headers[key]); } } getHeader(name) { if (typeof name !== 'string') throw new ERR_INVALID_ARG_TYPE('name', 'string', name); name = name.trim().toLowerCase(); return this[kHeaders][name]; } getHeaderNames() { return Object.keys(this[kHeaders]); } getHeaders() { return Object.assign({}, this[kHeaders]); } hasHeader(name) { if (typeof name !== 'string') throw new ERR_INVALID_ARG_TYPE('name', 'string', name); name = name.trim().toLowerCase(); return Object.prototype.hasOwnProperty.call(this[kHeaders], name); } removeHeader(name) { if (typeof name !== 'string') throw new ERR_INVALID_ARG_TYPE('name', 'string', name); if (this[kStream].headersSent) throw new ERR_HTTP2_HEADERS_SENT(); name = name.trim().toLowerCase(); delete this[kHeaders][name]; } setHeader(name, value) { if (typeof name !== 'string') throw new ERR_INVALID_ARG_TYPE('name', 'string', name); if (this[kStream].headersSent) throw new ERR_HTTP2_HEADERS_SENT(); this[kSetHeader](name, value); } [kSetHeader](name, value) { name = name.trim().toLowerCase(); assertValidHeader(name, value); this[kHeaders][name] = value; } get statusMessage() { statusMessageWarn(); return ''; } set statusMessage(msg) { statusMessageWarn(); } flushHeaders() { const state = this[kState]; if (!state.closed && !this[kStream].headersSent) this.writeHead(state.statusCode); } writeHead(statusCode, statusMessage, headers) { const state = this[kState]; if (state.closed) throw new ERR_HTTP2_INVALID_STREAM(); if (this[kStream].headersSent) throw new ERR_HTTP2_HEADERS_SENT(); if (typeof statusMessage === 'string') statusMessageWarn(); if (headers === undefined && typeof statusMessage === 'object') headers = statusMessage; if (typeof headers === 'object') { const keys = Object.keys(headers); let key = ''; for (var i = 0; i < keys.length; i++) { key = keys[i]; this[kSetHeader](key, headers[key]); } } state.statusCode = statusCode; this[kBeginSend](); } write(chunk, encoding, cb) { if (typeof encoding === 'function') { cb = encoding; encoding = 'utf8'; } if (this[kState].closed) { const err = new ERR_HTTP2_INVALID_STREAM(); if (typeof cb === 'function') process.nextTick(cb, err); else throw err; return; } const stream = this[kStream]; if (!stream.headersSent) this.writeHead(this[kState].statusCode); return stream.write(chunk, encoding, cb); } end(chunk, encoding, cb) { const stream = this[kStream]; const state = this[kState]; if ((state.closed || state.ending) && state.headRequest === stream.headRequest) { return false; } if (typeof chunk === 'function') { cb = chunk; chunk = null; } else if (typeof encoding === 'function') { cb = encoding; encoding = 'utf8'; } if (chunk !== null && chunk !== undefined) this.write(chunk, encoding); const isFinished = this.finished; state.headRequest = stream.headRequest; state.ending = true; if (typeof cb === 'function') { if (isFinished) this.once('finish', cb); else stream.once('finish', cb); } if (!stream.headersSent) this.writeHead(this[kState].statusCode); if (isFinished) this[kFinish](); else stream.end(); return this; } destroy(err) { if (this[kState].closed) return; this[kStream].destroy(err); } setTimeout(msecs, callback) { if (this[kState].closed) return; this[kStream].setTimeout(msecs, callback); } createPushResponse(headers, callback) { if (typeof callback !== 'function') throw new ERR_INVALID_CALLBACK(); if (this[kState].closed) { process.nextTick(callback, new ERR_HTTP2_INVALID_STREAM()); return; } this[kStream].pushStream(headers, {}, (err, stream, headers, options) => { if (err) { callback(err); return; } callback(null, new Http2ServerResponse(stream)); }); } [kBeginSend]() { const state = this[kState]; const headers = this[kHeaders]; headers[HTTP2_HEADER_STATUS] = state.statusCode; const options = { endStream: state.ending, getTrailers: (trailers) => Object.assign(trailers, this[kTrailers]) }; this[kStream].respond(headers, options); } [kFinish]() { const stream = this[kStream]; const state = this[kState]; if (state.closed || stream.headRequest !== state.headRequest) return; state.closed = true; this[kProxySocket] = null; stream[kResponse] = undefined; this.emit('finish'); this.emit('close'); } // TODO doesn't support callbacks writeContinue() { const stream = this[kStream]; if (stream.headersSent || this[kState].closed) return false; stream.additionalHeaders({ [HTTP2_HEADER_STATUS]: HTTP_STATUS_CONTINUE }); return true; } } function onServerStream(ServerRequest, ServerResponse, stream, headers, flags, rawHeaders) { const server = this; const request = new ServerRequest(stream, headers, undefined, rawHeaders); const response = new ServerResponse(stream); // Check for the CONNECT method const method = headers[HTTP2_HEADER_METHOD]; if (method === 'CONNECT') { if (!server.emit('connect', request, response)) { response.statusCode = HTTP_STATUS_METHOD_NOT_ALLOWED; response.end(); } return; } // Check for Expectations if (headers.expect !== undefined) { if (headers.expect === '100-continue') { if (server.listenerCount('checkContinue')) { server.emit('checkContinue', request, response); } else { response.writeContinue(); server.emit('request', request, response); } } else if (server.listenerCount('checkExpectation')) { server.emit('checkExpectation', request, response); } else { response.statusCode = HTTP_STATUS_EXPECTATION_FAILED; response.end(); } return; } server.emit('request', request, response); } module.exports = { onServerStream, Http2ServerRequest, Http2ServerResponse, };