/** * 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 */ '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 Headers(headers) { this.map = {} var self = this if (headers instanceof Headers) { headers.forEach(function(name, values) { values.forEach(function(value) { self.append(name, value) }) }) } else if (headers) { Object.getOwnPropertyNames(headers).forEach(function(name) { self.append(name, headers[name]) }) } } Headers.prototype.append = function(name, value) { name = name.toLowerCase() var list = this.map[name] if (!list) { list = [] this.map[name] = list } list.push(value) } Headers.prototype['delete'] = function(name) { delete this.map[name.toLowerCase()] } Headers.prototype.get = function(name) { var values = this.map[name.toLowerCase()] return values ? values[0] : null } Headers.prototype.getAll = function(name) { return this.map[name.toLowerCase()] || [] } Headers.prototype.has = function(name) { return this.map.hasOwnProperty(name.toLowerCase()) } Headers.prototype.set = function(name, value) { this.map[name.toLowerCase()] = [value] } // Instead of iterable for now. Headers.prototype.forEach = function(callback) { var self = this Object.getOwnPropertyNames(this.map).forEach(function(name) { callback(name, self.map[name]) }) } 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 blobSupport = 'FileReader' in self && 'Blob' in self && (function() { try { new Blob(); return true } catch(e) { return false } })(); function Body() { this.bodyUsed = false if (blobSupport) { this.blob = function() { var rejected = consumed(this) return rejected ? rejected : Promise.resolve(this._bodyBlob) } this.arrayBuffer = function() { return this.blob().then(readBlobAsArrayBuffer) } this.text = function() { return this.blob().then(readBlobAsText) } } else { this.text = function() { var rejected = consumed(this) return rejected ? rejected : Promise.resolve(this._bodyText) } } if ('FormData' in self) { 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(url, options) { options = options || {} this.url = url this._body = options.body this.credentials = options.credentials || 'omit' this.headers = new Headers(options.headers) this.method = normalizeMethod(options.method || 'GET') this.mode = options.mode || null this.referrer = null } 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 } Request.prototype.fetch = function() { var self = this 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(self.method, self.url) if ('responseType' in xhr && blobSupport) { xhr.responseType = 'blob' } self.headers.forEach(function(name, values) { values.forEach(function(value) { xhr.setRequestHeader(name, value) }) }) xhr.send((self._body === undefined) ? null : self._body) }) } 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 })(); /** End of the third-party code */ module.exports = self.fetch;