electron/lib/browser/api/net.js

360 lines
9.3 KiB
JavaScript
Raw Normal View History

'use strict'
2016-09-29 13:24:28 +00:00
const url = require('url')
const {EventEmitter} = require('events')
const util = require('util')
2016-10-25 10:41:01 +00:00
const {Readable} = require('stream')
const {app} = require('electron')
const {Session} = process.atomBinding('session')
2016-10-25 10:41:01 +00:00
const {net, Net} = process.atomBinding('net')
const {URLRequest} = net
Object.setPrototypeOf(Net.prototype, EventEmitter.prototype)
Object.setPrototypeOf(URLRequest.prototype, EventEmitter.prototype)
2016-10-25 10:41:01 +00:00
const kSupportedProtocols = new Set(['http:', 'https:'])
class IncomingMessage extends Readable {
2016-09-29 13:24:28 +00:00
constructor (urlRequest) {
super()
this.urlRequest = urlRequest
this.shouldPush = false
this.data = []
this.urlRequest.on('data', (event, chunk) => {
this._storeInternalData(chunk)
this._pushInternalData()
})
this.urlRequest.on('end', () => {
this._storeInternalData(null)
this._pushInternalData()
})
}
2016-09-29 13:24:28 +00:00
get statusCode () {
return this.urlRequest.statusCode
}
2016-09-29 13:24:28 +00:00
get statusMessage () {
return this.urlRequest.statusMessage
}
2016-09-29 13:24:28 +00:00
get headers () {
return this.urlRequest.rawResponseHeaders
}
2016-09-29 13:24:28 +00:00
get httpVersion () {
return `${this.httpVersionMajor}.${this.httpVersionMinor}`
}
2016-09-29 13:24:28 +00:00
get httpVersionMajor () {
return this.urlRequest.httpVersionMajor
}
2016-09-29 13:24:28 +00:00
get httpVersionMinor () {
return this.urlRequest.httpVersionMinor
}
2016-10-12 10:29:25 +00:00
get rawTrailers () {
throw new Error('HTTP trailers are not supported.')
}
2016-10-12 10:29:25 +00:00
get trailers () {
throw new Error('HTTP trailers are not supported.')
}
2016-10-12 10:29:25 +00:00
_storeInternalData (chunk) {
this.data.push(chunk)
}
2016-10-12 10:29:25 +00:00
_pushInternalData () {
while (this.shouldPush && this.data.length > 0) {
const chunk = this.data.shift()
this.shouldPush = this.push(chunk)
}
}
2016-10-12 10:29:25 +00:00
_read () {
this.shouldPush = true
this._pushInternalData()
}
}
2016-10-25 10:41:01 +00:00
URLRequest.prototype._emitRequestEvent = function (isAsync, ...rest) {
if (isAsync) {
process.nextTick(() => {
this.clientRequest.emit.apply(this.clientRequest, rest)
})
} else {
this.clientRequest.emit.apply(this.clientRequest, rest)
}
}
2016-10-25 10:41:01 +00:00
URLRequest.prototype._emitResponseEvent = function (isAsync, ...rest) {
if (isAsync) {
process.nextTick(() => {
this._response.emit.apply(this._response, rest)
})
} else {
this._response.emit.apply(this._response, rest)
}
}
class ClientRequest extends EventEmitter {
2016-09-29 13:24:28 +00:00
constructor (options, callback) {
super()
if (!app.isReady()) {
throw new Error('net module can only be used after app is ready')
}
if (typeof options === 'string') {
2016-09-29 13:24:28 +00:00
options = url.parse(options)
} else {
2016-09-29 13:24:28 +00:00
options = util._extend({}, options)
}
2016-09-29 13:24:28 +00:00
const method = (options.method || 'GET').toUpperCase()
let urlStr = options.url
2016-09-29 13:24:28 +00:00
if (!urlStr) {
let urlObj = {}
const protocol = options.protocol || 'http:'
if (!kSupportedProtocols.has(protocol)) {
2016-09-29 13:24:28 +00:00
throw new Error('Protocol "' + protocol + '" not supported. ')
}
2016-09-29 13:24:28 +00:00
urlObj.protocol = protocol
if (options.host) {
2016-09-29 13:24:28 +00:00
urlObj.host = options.host
} else {
if (options.hostname) {
2016-09-29 13:24:28 +00:00
urlObj.hostname = options.hostname
} else {
2016-09-29 13:24:28 +00:00
urlObj.hostname = 'localhost'
}
if (options.port) {
2016-09-29 13:24:28 +00:00
urlObj.port = options.port
}
}
if (options.path && / /.test(options.path)) {
// The actual regex is more like /[^A-Za-z0-9\-._~!$&'()*+,;=/:@]/
// with an additional rule for ignoring percentage-escaped characters
// but that's a) hard to capture in a regular expression that performs
// well, and b) possibly too restrictive for real-world usage. That's
// why it only scans for spaces because those are guaranteed to create
// an invalid request.
2016-09-29 13:24:28 +00:00
throw new TypeError('Request path contains unescaped characters.')
2016-10-12 10:29:25 +00:00
}
let pathObj = url.parse(options.path || '/')
2016-10-12 10:29:25 +00:00
urlObj.pathname = pathObj.pathname
urlObj.search = pathObj.search
urlObj.hash = pathObj.hash
2016-09-29 13:24:28 +00:00
urlStr = url.format(urlObj)
}
let urlRequestOptions = {
method: method,
url: urlStr
}
if (options.session) {
if (options.session instanceof Session) {
urlRequestOptions.session = options.session
} else {
throw new TypeError('`session` should be an instance of the Session class.')
}
} else if (options.partition) {
if (typeof options.partition === 'string') {
urlRequestOptions.partition = options.partition
} else {
throw new TypeError('`partition` should be an a string.')
}
}
let urlRequest = new URLRequest(urlRequestOptions)
// Set back and forward links.
this.urlRequest = urlRequest
urlRequest.clientRequest = this
// This is a copy of the extra headers structure held by the native
// net::URLRequest. The main reason is to keep the getHeader API synchronous
// after the request starts.
this.extraHeaders = {}
if (options.headers) {
2016-10-25 10:41:01 +00:00
for (let key in options.headers) {
2016-09-29 13:24:28 +00:00
this.setHeader(key, options.headers[key])
}
}
// Set when the request uses chunked encoding. Can be switched
// to true only once and never set back to false.
this.chunkedEncodingEnabled = false
2016-09-26 12:03:49 +00:00
2016-09-29 13:24:28 +00:00
urlRequest.on('response', () => {
const response = new IncomingMessage(urlRequest)
urlRequest._response = response
this.emit('response', response)
})
2016-09-21 15:35:03 +00:00
urlRequest.on('login', (event, authInfo, callback) => {
this.emit('login', authInfo, (username, password) => {
// If null or undefined usrename/password, force to empty string.
if (username === null || username === undefined) {
username = ''
}
if (typeof username !== 'string') {
throw new Error('username must be a string')
}
if (password === null || password === undefined) {
password = ''
}
if (typeof password !== 'string') {
throw new Error('password must be a string')
}
callback(username, password)
})
})
if (callback) {
this.once('response', callback)
}
2016-09-21 15:35:03 +00:00
}
2016-09-29 13:24:28 +00:00
get chunkedEncoding () {
return this.chunkedEncodingEnabled
}
2016-09-29 13:24:28 +00:00
set chunkedEncoding (value) {
if (!this.urlRequest.notStarted) {
2016-09-29 13:24:28 +00:00
throw new Error('Can\'t set the transfer encoding, headers have been sent.')
}
this.chunkedEncodingEnabled = value
}
2016-09-29 13:24:28 +00:00
setHeader (name, value) {
if (typeof name !== 'string') {
throw new TypeError('`name` should be a string in setHeader(name, value).')
}
if (value === undefined) {
throw new Error('`value` required in setHeader("' + name + '", value).')
}
if (!this.urlRequest.notStarted) {
2016-09-29 13:24:28 +00:00
throw new Error('Can\'t set headers after they are sent.')
}
2016-09-21 15:35:03 +00:00
2016-09-29 13:24:28 +00:00
const key = name.toLowerCase()
this.extraHeaders[key] = value
this.urlRequest.setExtraHeader(name, value)
2016-09-21 15:35:03 +00:00
}
2016-09-29 13:24:28 +00:00
getHeader (name) {
if (arguments.length < 1) {
2016-09-29 13:24:28 +00:00
throw new Error('`name` is required for getHeader(name).')
}
2016-09-21 15:35:03 +00:00
if (!this.extraHeaders) {
2016-09-29 13:24:28 +00:00
return
}
2016-09-21 15:35:03 +00:00
2016-09-29 13:24:28 +00:00
const key = name.toLowerCase()
return this.extraHeaders[key]
}
2016-09-29 13:24:28 +00:00
removeHeader (name) {
if (arguments.length < 1) {
2016-09-29 13:24:28 +00:00
throw new Error('`name` is required for removeHeader(name).')
}
if (!this.urlRequest.notStarted) {
2016-09-29 13:24:28 +00:00
throw new Error('Can\'t remove headers after they are sent.')
}
2016-09-21 15:35:03 +00:00
2016-09-29 13:24:28 +00:00
const key = name.toLowerCase()
delete this.extraHeaders[key]
this.urlRequest.removeExtraHeader(name)
2016-09-21 15:35:03 +00:00
}
2016-09-29 13:24:28 +00:00
_write (chunk, encoding, callback, isLast) {
let chunkIsString = typeof chunk === 'string'
let chunkIsBuffer = chunk instanceof Buffer
if (!chunkIsString && !chunkIsBuffer) {
throw new TypeError('First argument must be a string or Buffer.')
}
2016-09-29 13:24:28 +00:00
if (chunkIsString) {
// We convert all strings into binary buffers.
2016-09-29 13:24:28 +00:00
chunk = Buffer.from(chunk, encoding)
}
2016-09-29 13:24:28 +00:00
// Since writing to the network is asynchronous, we conservatively
// assume that request headers are written after delivering the first
// buffer to the network IO thread.
if (this.urlRequest.notStarted) {
this.urlRequest.setChunkedUpload(this.chunkedEncoding)
}
// Headers are assumed to be sent on first call to _writeBuffer,
// i.e. after the first call to write or end.
let result = this.urlRequest.write(chunk, isLast)
// The write callback is fired asynchronously to mimic Node.js.
if (callback) {
2016-09-29 13:24:28 +00:00
process.nextTick(callback)
}
2016-09-29 13:24:28 +00:00
return result
}
2016-09-29 13:24:28 +00:00
write (data, encoding, callback) {
if (this.urlRequest.finished) {
2016-09-29 13:24:28 +00:00
let error = new Error('Write after end.')
process.nextTick(writeAfterEndNT, this, error, callback)
return true
}
2016-09-26 12:03:49 +00:00
2016-09-29 13:24:28 +00:00
return this._write(data, encoding, callback, false)
2016-09-21 15:35:03 +00:00
}
2016-09-29 13:24:28 +00:00
end (data, encoding, callback) {
if (this.urlRequest.finished) {
2016-09-29 13:24:28 +00:00
return false
}
2016-09-29 13:24:28 +00:00
if (typeof data === 'function') {
2016-09-29 13:24:28 +00:00
callback = data
encoding = null
data = null
} else if (typeof encoding === 'function') {
2016-09-29 13:24:28 +00:00
callback = encoding
encoding = null
}
2016-09-21 15:35:03 +00:00
2016-09-29 13:24:28 +00:00
data = data || ''
2016-09-21 15:35:03 +00:00
2016-09-29 13:24:28 +00:00
return this._write(data, encoding, callback, true)
2016-09-26 12:03:49 +00:00
}
2016-09-29 13:24:28 +00:00
abort () {
this.urlRequest.cancel()
}
}
2016-09-21 15:35:03 +00:00
2016-09-29 13:24:28 +00:00
function writeAfterEndNT (self, error, callback) {
self.emit('error', error)
if (callback) callback(error)
2016-09-21 15:35:03 +00:00
}
2016-09-29 13:24:28 +00:00
Net.prototype.request = function (options, callback) {
return new ClientRequest(options, callback)
}
2016-09-29 13:24:28 +00:00
net.ClientRequest = ClientRequest
module.exports = net