Bump fetch and add exports/globals for Headers, Request and Response

Summary:
Now on fetch 0.8.1, the latest tagged release. Previous version used was 0.7.0. See #1162 cc @vjeux @jtremback
Closes https://github.com/facebook/react-native/pull/1192
Github Author: Brent Vatne <brent.vatne@madriska.com>

Test Plan: I arc patched and ran movies demo and storyline, they work fine
This commit is contained in:
Brent Vatne 2015-05-07 12:29:36 -07:00
parent 1ef4e00fba
commit 7141948a05
5 changed files with 133 additions and 65 deletions

View File

@ -29,8 +29,6 @@ var TimerMixin = require('react-timer-mixin');
var MovieCell = require('./MovieCell'); var MovieCell = require('./MovieCell');
var MovieScreen = require('./MovieScreen'); var MovieScreen = require('./MovieScreen');
var fetch = require('fetch');
/** /**
* This is for demo purposes only, and rate limited. * This is for demo purposes only, and rate limited.
* In case you want to use the Rotten Tomatoes' API on a real app you should * In case you want to use the Rotten Tomatoes' API on a real app you should

View File

@ -47,6 +47,23 @@ var self = {};
return return
} }
function normalizeName(name) {
if (typeof name !== 'string') {
name = name.toString();
}
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 = value.toString();
}
return value
}
function Headers(headers) { function Headers(headers) {
this.map = {} this.map = {}
@ -66,7 +83,8 @@ var self = {};
} }
Headers.prototype.append = function(name, value) { Headers.prototype.append = function(name, value) {
name = name.toLowerCase() name = normalizeName(name)
value = normalizeValue(value)
var list = this.map[name] var list = this.map[name]
if (!list) { if (!list) {
list = [] list = []
@ -76,24 +94,24 @@ var self = {};
} }
Headers.prototype['delete'] = function(name) { Headers.prototype['delete'] = function(name) {
delete this.map[name.toLowerCase()] delete this.map[normalizeName(name)]
} }
Headers.prototype.get = function(name) { Headers.prototype.get = function(name) {
var values = this.map[name.toLowerCase()] var values = this.map[normalizeName(name)]
return values ? values[0] : null return values ? values[0] : null
} }
Headers.prototype.getAll = function(name) { Headers.prototype.getAll = function(name) {
return this.map[name.toLowerCase()] || [] return this.map[normalizeName(name)] || []
} }
Headers.prototype.has = function(name) { Headers.prototype.has = function(name) {
return this.map.hasOwnProperty(name.toLowerCase()) return this.map.hasOwnProperty(normalizeName(name))
} }
Headers.prototype.set = function(name, value) { Headers.prototype.set = function(name, value) {
this.map[name.toLowerCase()] = [value] this.map[normalizeName(name)] = [normalizeValue(value)]
} }
// Instead of iterable for now. // Instead of iterable for now.
@ -134,22 +152,51 @@ var self = {};
return fileReaderReady(reader) return fileReaderReady(reader)
} }
var blobSupport = 'FileReader' in self && 'Blob' in self && (function() { var support = {
blob: 'FileReader' in self && 'Blob' in self && (function() {
try { try {
new Blob(); new Blob();
return true return true
} catch(e) { } catch(e) {
return false return false
} }
})(); })(),
formData: 'FormData' in self
}
function Body() { function Body() {
this.bodyUsed = false this.bodyUsed = false
if (blobSupport) {
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() { this.blob = function() {
var rejected = consumed(this) var rejected = consumed(this)
return rejected ? rejected : Promise.resolve(this._bodyBlob) 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() { this.arrayBuffer = function() {
@ -157,7 +204,18 @@ var self = {};
} }
this.text = function() { this.text = function() {
return this.blob().then(readBlobAsText) 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 { } else {
this.text = function() { this.text = function() {
@ -166,7 +224,7 @@ var self = {};
} }
} }
if ('FormData' in self) { if (support.formData) {
this.formData = function() { this.formData = function() {
return this.text().then(decode) return this.text().then(decode)
} }
@ -190,12 +248,17 @@ var self = {};
function Request(url, options) { function Request(url, options) {
options = options || {} options = options || {}
this.url = url this.url = url
this._body = options.body
this.credentials = options.credentials || 'omit' this.credentials = options.credentials || 'omit'
this.headers = new Headers(options.headers) this.headers = new Headers(options.headers)
this.method = normalizeMethod(options.method || 'GET') this.method = normalizeMethod(options.method || 'GET')
this.mode = options.mode || null this.mode = options.mode || null
this.referrer = null this.referrer = null
if ((this.method === 'GET' || this.method === 'HEAD') && options.body) {
throw new TypeError('Body not allowed for GET or HEAD requests')
}
this._initBody(options.body)
} }
function decode(body) { function decode(body) {
@ -223,11 +286,43 @@ var self = {};
return head return head
} }
Request.prototype.fetch = function() { Body.call(Request.prototype)
var self = this
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 || ''
}
Body.call(Response.prototype)
self.Headers = Headers;
self.Request = Request;
self.Response = Response;
self.fetch = function(input, init) {
// TODO: Request constructor should accept input, init
var request
if (Request.prototype.isPrototypeOf(input) && !init) {
request = input
} else {
request = new Request(input, init)
}
return new Promise(function(resolve, reject) { return new Promise(function(resolve, reject) {
var xhr = new XMLHttpRequest() var xhr = new XMLHttpRequest()
if (request.credentials === 'cors') {
xhr.withCredentials = true;
}
function responseURL() { function responseURL() {
if ('responseURL' in xhr) { if ('responseURL' in xhr) {
@ -262,57 +357,24 @@ var self = {};
reject(new TypeError('Network request failed')) reject(new TypeError('Network request failed'))
} }
xhr.open(self.method, self.url) xhr.open(request.method, request.url, true)
if ('responseType' in xhr && blobSupport) {
if ('responseType' in xhr && support.blob) {
xhr.responseType = 'blob' xhr.responseType = 'blob'
} }
self.headers.forEach(function(name, values) { request.headers.forEach(function(name, values) {
values.forEach(function(value) { values.forEach(function(value) {
xhr.setRequestHeader(name, value) xhr.setRequestHeader(name, value)
}) })
}) })
xhr.send((self._body === undefined) ? null : self._body) xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)
}) })
} }
Body.call(Request.prototype)
function Response(bodyInit, options) {
if (!options) {
options = {}
}
if (blobSupport) {
if (typeof bodyInit === 'string') {
this._bodyBlob = new Blob([bodyInit])
} else {
this._bodyBlob = bodyInit
}
} else {
this._bodyText = bodyInit
}
this.type = 'default'
this.url = null
this.status = options.status
this.statusText = options.statusText
this.headers = options.headers
this.url = options.url || ''
}
Body.call(Response.prototype)
self.Headers = Headers;
self.Request = Request;
self.Response = Response;
self.fetch = function (url, options) {
return new Request(url, options).fetch()
}
self.fetch.polyfill = true self.fetch.polyfill = true
})(); })();
/** End of the third-party code */ /** End of the third-party code */
module.exports = self.fetch; module.exports = self;

View File

@ -135,7 +135,12 @@ function setupXHR() {
// The native XMLHttpRequest in Chrome dev tools is CORS aware and won't // The native XMLHttpRequest in Chrome dev tools is CORS aware and won't
// let you fetch anything from the internet // let you fetch anything from the internet
GLOBAL.XMLHttpRequest = require('XMLHttpRequest'); GLOBAL.XMLHttpRequest = require('XMLHttpRequest');
GLOBAL.fetch = require('fetch');
var fetchPolyfill = require('fetch');
GLOBAL.fetch = fetchPolyfill.fetch;
GLOBAL.Headers = fetchPolyfill.Headers;
GLOBAL.Request = fetchPolyfill.Request;
GLOBAL.Response = fetchPolyfill.Response;
} }
function setupGeolocation() { function setupGeolocation() {

View File

@ -17,8 +17,6 @@ var RCTSourceCode = require('NativeModules').SourceCode;
var SourceMapConsumer = require('SourceMap').SourceMapConsumer; var SourceMapConsumer = require('SourceMap').SourceMapConsumer;
var SourceMapURL = require('./source-map-url'); var SourceMapURL = require('./source-map-url');
var fetch = require('fetch');
function loadSourceMap(): Promise { function loadSourceMap(): Promise {
return fetchSourceMap() return fetchSourceMap()
.then(map => new SourceMapConsumer(map)); .then(map => new SourceMapConsumer(map));

View File

@ -16,3 +16,8 @@ declare var __DEV__: boolean;
declare var __REACT_DEVTOOLS_GLOBAL_HOOK__: any; /*?{ declare var __REACT_DEVTOOLS_GLOBAL_HOOK__: any; /*?{
inject: ?((stuff: Object) => void) inject: ?((stuff: Object) => void)
};*/ };*/
declare var fetch: any;
declare var Headers: any;
declare var Request: any;
declare var Response: any;