| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224 | var capability = require('./capability')var inherits = require('inherits')var stream = require('readable-stream')var rStates = exports.readyStates = {	UNSENT: 0,	OPENED: 1,	HEADERS_RECEIVED: 2,	LOADING: 3,	DONE: 4}var IncomingMessage = exports.IncomingMessage = function (xhr, response, mode, fetchTimer) {	var self = this	stream.Readable.call(self)	self._mode = mode	self.headers = {}	self.rawHeaders = []	self.trailers = {}	self.rawTrailers = []	// Fake the 'close' event, but only once 'end' fires	self.on('end', function () {		// The nextTick is necessary to prevent the 'request' module from causing an infinite loop		process.nextTick(function () {			self.emit('close')		})	})	if (mode === 'fetch') {		self._fetchResponse = response		self.url = response.url		self.statusCode = response.status		self.statusMessage = response.statusText				response.headers.forEach(function (header, key){			self.headers[key.toLowerCase()] = header			self.rawHeaders.push(key, header)		})		if (capability.writableStream) {			var writable = new WritableStream({				write: function (chunk) {					return new Promise(function (resolve, reject) {						if (self._destroyed) {							reject()						} else if(self.push(new Buffer(chunk))) {							resolve()						} else {							self._resumeFetch = resolve						}					})				},				close: function () {					global.clearTimeout(fetchTimer)					if (!self._destroyed)						self.push(null)				},				abort: function (err) {					if (!self._destroyed)						self.emit('error', err)				}			})			try {				response.body.pipeTo(writable).catch(function (err) {					global.clearTimeout(fetchTimer)					if (!self._destroyed)						self.emit('error', err)				})				return			} catch (e) {} // pipeTo method isn't defined. Can't find a better way to feature test this		}		// fallback for when writableStream or pipeTo aren't available		var reader = response.body.getReader()		function read () {			reader.read().then(function (result) {				if (self._destroyed)					return				if (result.done) {					global.clearTimeout(fetchTimer)					self.push(null)					return				}				self.push(new Buffer(result.value))				read()			}).catch(function (err) {				global.clearTimeout(fetchTimer)				if (!self._destroyed)					self.emit('error', err)			})		}		read()	} else {		self._xhr = xhr		self._pos = 0		self.url = xhr.responseURL		self.statusCode = xhr.status		self.statusMessage = xhr.statusText		var headers = xhr.getAllResponseHeaders().split(/\r?\n/)		headers.forEach(function (header) {			var matches = header.match(/^([^:]+):\s*(.*)/)			if (matches) {				var key = matches[1].toLowerCase()				if (key === 'set-cookie') {					if (self.headers[key] === undefined) {						self.headers[key] = []					}					self.headers[key].push(matches[2])				} else if (self.headers[key] !== undefined) {					self.headers[key] += ', ' + matches[2]				} else {					self.headers[key] = matches[2]				}				self.rawHeaders.push(matches[1], matches[2])			}		})		self._charset = 'x-user-defined'		if (!capability.overrideMimeType) {			var mimeType = self.rawHeaders['mime-type']			if (mimeType) {				var charsetMatch = mimeType.match(/;\s*charset=([^;])(;|$)/)				if (charsetMatch) {					self._charset = charsetMatch[1].toLowerCase()				}			}			if (!self._charset)				self._charset = 'utf-8' // best guess		}	}}inherits(IncomingMessage, stream.Readable)IncomingMessage.prototype._read = function () {	var self = this	var resolve = self._resumeFetch	if (resolve) {		self._resumeFetch = null		resolve()	}}IncomingMessage.prototype._onXHRProgress = function () {	var self = this	var xhr = self._xhr	var response = null	switch (self._mode) {		case 'text:vbarray': // For IE9			if (xhr.readyState !== rStates.DONE)				break			try {				// This fails in IE8				response = new global.VBArray(xhr.responseBody).toArray()			} catch (e) {}			if (response !== null) {				self.push(new Buffer(response))				break			}			// Falls through in IE8			case 'text':			try { // This will fail when readyState = 3 in IE9. Switch mode and wait for readyState = 4				response = xhr.responseText			} catch (e) {				self._mode = 'text:vbarray'				break			}			if (response.length > self._pos) {				var newData = response.substr(self._pos)				if (self._charset === 'x-user-defined') {					var buffer = new Buffer(newData.length)					for (var i = 0; i < newData.length; i++)						buffer[i] = newData.charCodeAt(i) & 0xff					self.push(buffer)				} else {					self.push(newData, self._charset)				}				self._pos = response.length			}			break		case 'arraybuffer':			if (xhr.readyState !== rStates.DONE || !xhr.response)				break			response = xhr.response			self.push(new Buffer(new Uint8Array(response)))			break		case 'moz-chunked-arraybuffer': // take whole			response = xhr.response			if (xhr.readyState !== rStates.LOADING || !response)				break			self.push(new Buffer(new Uint8Array(response)))			break		case 'ms-stream':			response = xhr.response			if (xhr.readyState !== rStates.LOADING)				break			var reader = new global.MSStreamReader()			reader.onprogress = function () {				if (reader.result.byteLength > self._pos) {					self.push(new Buffer(new Uint8Array(reader.result.slice(self._pos))))					self._pos = reader.result.byteLength				}			}			reader.onload = function () {				self.push(null)			}			// reader.onerror = ??? // TODO: this			reader.readAsArrayBuffer(response)			break	}	// The ms-stream case handles end separately in reader.onload()	if (self._xhr.readyState === rStates.DONE && self._mode !== 'ms-stream') {		self.push(null)	}}
 |