mirror of
https://github.com/status-im/react-native.git
synced 2025-01-15 03:56:03 +00:00
854689dcfa
Summary: Fetch includes a cloning function that was not included in the React Native fetch.js. This adds it in so that cloning is available on responses from fetch calls in React Native. The code is taken directly from [fetch](https://github.com/github/fetch/blob/master/fetch.js) on Github. <img width="601" alt="screen shot 2015-10-22 at 11 50 53 am" src="https://cloud.githubusercontent.com/assets/11564650/10675101/304ebe5a-78b3-11e5-9d6b-24ea6d9fb998.png"> <img width="596" alt="screen shot 2015-10-22 at 12 20 08 pm" src="https://cloud.githubusercontent.com/assets/11564650/10675834/4abaf4ee-78b7-11e5-9d34-436290b64b30.png"> Closes https://github.com/facebook/react-native/pull/3614 Reviewed By: svcscm Differential Revision: D2600517 Pulled By: sahrens fb-gh-sync-id: d70c5c58e923d997f93bb5aa2d1d90e95d5a75f2
410 lines
11 KiB
JavaScript
410 lines
11 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* This is a third-party polyfill grabbed from:
|
|
* https://github.com/github/fetch
|
|
*
|
|
* @providesModule fetch
|
|
* @nolint
|
|
*/
|
|
/* eslint-disable */
|
|
'use strict';
|
|
|
|
var self = {};
|
|
|
|
/**
|
|
* Copyright (c) 2014 GitHub, Inc.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
* @preserve-header
|
|
*/
|
|
(function() {
|
|
'use strict';
|
|
|
|
if (self.fetch) {
|
|
return
|
|
}
|
|
|
|
function normalizeName(name) {
|
|
if (typeof name !== 'string') {
|
|
name = String(name)
|
|
}
|
|
if (/[^a-z0-9\-#$%&'*+.\^_`|~]/i.test(name)) {
|
|
throw new TypeError('Invalid character in header field name')
|
|
}
|
|
return name.toLowerCase()
|
|
}
|
|
|
|
function normalizeValue(value) {
|
|
if (typeof value !== 'string') {
|
|
value = String(value)
|
|
}
|
|
return value
|
|
}
|
|
|
|
function Headers(headers) {
|
|
this.map = {}
|
|
|
|
if (headers instanceof Headers) {
|
|
headers.forEach(function(value, name) {
|
|
this.append(name, value)
|
|
}, this)
|
|
|
|
} else if (headers) {
|
|
Object.getOwnPropertyNames(headers).forEach(function(name) {
|
|
this.append(name, headers[name])
|
|
}, this)
|
|
}
|
|
}
|
|
|
|
Headers.prototype.append = function(name, value) {
|
|
name = normalizeName(name)
|
|
value = normalizeValue(value)
|
|
var list = this.map[name]
|
|
if (!list) {
|
|
list = []
|
|
this.map[name] = list
|
|
}
|
|
list.push(value)
|
|
}
|
|
|
|
Headers.prototype['delete'] = function(name) {
|
|
delete this.map[normalizeName(name)]
|
|
}
|
|
|
|
Headers.prototype.get = function(name) {
|
|
var values = this.map[normalizeName(name)]
|
|
return values ? values[0] : null
|
|
}
|
|
|
|
Headers.prototype.getAll = function(name) {
|
|
return this.map[normalizeName(name)] || []
|
|
}
|
|
|
|
Headers.prototype.has = function(name) {
|
|
return this.map.hasOwnProperty(normalizeName(name))
|
|
}
|
|
|
|
Headers.prototype.set = function(name, value) {
|
|
this.map[normalizeName(name)] = [normalizeValue(value)]
|
|
}
|
|
|
|
Headers.prototype.forEach = function(callback, thisArg) {
|
|
Object.getOwnPropertyNames(this.map).forEach(function(name) {
|
|
this.map[name].forEach(function(value) {
|
|
callback.call(thisArg, value, name, this)
|
|
}, this)
|
|
}, this)
|
|
}
|
|
|
|
function consumed(body) {
|
|
if (body.bodyUsed) {
|
|
return Promise.reject(new TypeError('Already read'))
|
|
}
|
|
body.bodyUsed = true
|
|
}
|
|
|
|
function fileReaderReady(reader) {
|
|
return new Promise(function(resolve, reject) {
|
|
reader.onload = function() {
|
|
resolve(reader.result)
|
|
}
|
|
reader.onerror = function() {
|
|
reject(reader.error)
|
|
}
|
|
})
|
|
}
|
|
|
|
function readBlobAsArrayBuffer(blob) {
|
|
var reader = new FileReader()
|
|
reader.readAsArrayBuffer(blob)
|
|
return fileReaderReady(reader)
|
|
}
|
|
|
|
function readBlobAsText(blob) {
|
|
var reader = new FileReader()
|
|
reader.readAsText(blob)
|
|
return fileReaderReady(reader)
|
|
}
|
|
|
|
var support = {
|
|
blob: typeof FileReader === 'function' && typeof Blob === 'function' && (function() {
|
|
try {
|
|
new Blob();
|
|
return true
|
|
} catch(e) {
|
|
return false
|
|
}
|
|
})(),
|
|
formData: typeof FormData === 'function'
|
|
}
|
|
|
|
function Body() {
|
|
this.bodyUsed = false
|
|
|
|
|
|
this._initBody = function(body) {
|
|
this._bodyInit = body
|
|
if (typeof body === 'string') {
|
|
this._bodyText = body
|
|
} else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
|
|
this._bodyBlob = body
|
|
} else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
|
|
this._bodyFormData = body
|
|
} else if (!body) {
|
|
this._bodyText = ''
|
|
} else {
|
|
throw new Error('unsupported BodyInit type')
|
|
}
|
|
}
|
|
|
|
if (support.blob) {
|
|
this.blob = function() {
|
|
var rejected = consumed(this)
|
|
if (rejected) {
|
|
return rejected
|
|
}
|
|
|
|
if (this._bodyBlob) {
|
|
return Promise.resolve(this._bodyBlob)
|
|
} else if (this._bodyFormData) {
|
|
throw new Error('could not read FormData body as blob')
|
|
} else {
|
|
return Promise.resolve(new Blob([this._bodyText]))
|
|
}
|
|
}
|
|
|
|
this.arrayBuffer = function() {
|
|
return this.blob().then(readBlobAsArrayBuffer)
|
|
}
|
|
|
|
this.text = function() {
|
|
var rejected = consumed(this)
|
|
if (rejected) {
|
|
return rejected
|
|
}
|
|
|
|
if (this._bodyBlob) {
|
|
return readBlobAsText(this._bodyBlob)
|
|
} else if (this._bodyFormData) {
|
|
throw new Error('could not read FormData body as text')
|
|
} else {
|
|
return Promise.resolve(this._bodyText)
|
|
}
|
|
}
|
|
} else {
|
|
this.text = function() {
|
|
var rejected = consumed(this)
|
|
return rejected ? rejected : Promise.resolve(this._bodyText)
|
|
}
|
|
}
|
|
|
|
if (support.formData) {
|
|
this.formData = function() {
|
|
return this.text().then(decode)
|
|
}
|
|
}
|
|
|
|
this.json = function() {
|
|
return this.text().then(JSON.parse)
|
|
}
|
|
|
|
return this
|
|
}
|
|
|
|
// HTTP methods whose capitalization should be normalized
|
|
var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']
|
|
|
|
function normalizeMethod(method) {
|
|
var upcased = method.toUpperCase()
|
|
return (methods.indexOf(upcased) > -1) ? upcased : method
|
|
}
|
|
|
|
function Request(input, options) {
|
|
options = options || {}
|
|
var body = options.body
|
|
if (Request.prototype.isPrototypeOf(input)) {
|
|
if (input.bodyUsed) {
|
|
throw new TypeError('Already read')
|
|
}
|
|
this.url = input.url
|
|
this.credentials = input.credentials
|
|
if (!options.headers) {
|
|
this.headers = new Headers(input.headers)
|
|
}
|
|
this.method = input.method
|
|
this.mode = input.mode
|
|
if (!body) {
|
|
body = input._bodyInit
|
|
input.bodyUsed = true
|
|
}
|
|
} else {
|
|
this.url = input
|
|
}
|
|
|
|
this.credentials = options.credentials || this.credentials || 'omit'
|
|
if (options.headers || !this.headers) {
|
|
this.headers = new Headers(options.headers)
|
|
}
|
|
this.method = normalizeMethod(options.method || this.method || 'GET')
|
|
this.mode = options.mode || this.mode || null
|
|
this.referrer = null
|
|
|
|
if ((this.method === 'GET' || this.method === 'HEAD') && body) {
|
|
throw new TypeError('Body not allowed for GET or HEAD requests')
|
|
}
|
|
this._initBody(body)
|
|
}
|
|
|
|
Request.prototype.clone = function() {
|
|
return new Request(this)
|
|
}
|
|
|
|
function decode(body) {
|
|
var form = new FormData()
|
|
body.trim().split('&').forEach(function(bytes) {
|
|
if (bytes) {
|
|
var split = bytes.split('=')
|
|
var name = split.shift().replace(/\+/g, ' ')
|
|
var value = split.join('=').replace(/\+/g, ' ')
|
|
form.append(decodeURIComponent(name), decodeURIComponent(value))
|
|
}
|
|
})
|
|
return form
|
|
}
|
|
|
|
function headers(xhr) {
|
|
var head = new Headers()
|
|
var pairs = xhr.getAllResponseHeaders().trim().split('\n')
|
|
pairs.forEach(function(header) {
|
|
var split = header.trim().split(':')
|
|
var key = split.shift().trim()
|
|
var value = split.join(':').trim()
|
|
head.append(key, value)
|
|
})
|
|
return head
|
|
}
|
|
|
|
Body.call(Request.prototype)
|
|
|
|
function Response(bodyInit, options) {
|
|
if (!options) {
|
|
options = {}
|
|
}
|
|
|
|
this._initBody(bodyInit)
|
|
this.type = 'default'
|
|
this.url = null
|
|
this.status = options.status
|
|
this.ok = this.status >= 200 && this.status < 300
|
|
this.statusText = options.statusText
|
|
this.headers = options.headers instanceof Headers ? options.headers : new Headers(options.headers)
|
|
this.url = options.url || ''
|
|
}
|
|
|
|
Response.prototype.clone = function() {
|
|
return new Response(this._bodyInit, {
|
|
status: this.status,
|
|
statusText: this.statusText,
|
|
headers: new Headers(this.headers),
|
|
url: this.url
|
|
})
|
|
}
|
|
|
|
Body.call(Response.prototype)
|
|
|
|
self.Headers = Headers;
|
|
self.Request = Request;
|
|
self.Response = Response;
|
|
|
|
self.fetch = function(input, init) {
|
|
var request
|
|
if (Request.prototype.isPrototypeOf(input) && !init) {
|
|
request = input
|
|
} else {
|
|
request = new Request(input, init)
|
|
}
|
|
|
|
return new Promise(function(resolve, reject) {
|
|
var xhr = new XMLHttpRequest()
|
|
|
|
function responseURL() {
|
|
if ('responseURL' in xhr) {
|
|
return xhr.responseURL
|
|
}
|
|
|
|
// Avoid security warnings on getResponseHeader when not allowed by CORS
|
|
if (/^X-Request-URL:/m.test(xhr.getAllResponseHeaders())) {
|
|
return xhr.getResponseHeader('X-Request-URL')
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
xhr.onload = function() {
|
|
var status = (xhr.status === 1223) ? 204 : xhr.status
|
|
if (status < 100 || status > 599) {
|
|
reject(new TypeError('Network request failed'))
|
|
return
|
|
}
|
|
var options = {
|
|
status: status,
|
|
statusText: xhr.statusText,
|
|
headers: headers(xhr),
|
|
url: responseURL()
|
|
}
|
|
var body = 'response' in xhr ? xhr.response : xhr.responseText;
|
|
resolve(new Response(body, options))
|
|
}
|
|
|
|
xhr.onerror = function() {
|
|
reject(new TypeError('Network request failed'))
|
|
}
|
|
|
|
xhr.open(request.method, request.url, true)
|
|
|
|
if (request.credentials === 'include') {
|
|
xhr.withCredentials = true
|
|
}
|
|
|
|
if ('responseType' in xhr && support.blob) {
|
|
xhr.responseType = 'blob'
|
|
}
|
|
|
|
request.headers.forEach(function(value, name) {
|
|
xhr.setRequestHeader(name, value)
|
|
})
|
|
|
|
xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)
|
|
})
|
|
}
|
|
self.fetch.polyfill = true
|
|
})();
|
|
|
|
/** End of the third-party code */
|
|
|
|
module.exports = self;
|