UNPKG

i18next-http-backend

Version:

i18next-http-backend is a backend layer for i18next using in Node.js, in the browser and for Deno.

175 lines (159 loc) 6.45 kB
import { hasXMLHttpRequest } from './utils.js' // `globalThis` is the standard cross-environment global (Node ≥ 12, all // modern browsers, Deno, Bun). The `global` / `window` fallbacks remain only // for embedded JS runtimes that predate the ES2020 spec. const g = typeof globalThis !== 'undefined' ? globalThis : typeof global !== 'undefined' ? global : typeof window !== 'undefined' ? window : undefined let fetchApi if (typeof fetch === 'function') { fetchApi = fetch } else if (g && typeof g.fetch === 'function') { fetchApi = g.fetch } // XHR / ActiveXObject are picked up if present in the host, but no longer // polyfilled — v4 dropped the `cross-fetch` ponyfill. Runtimes without // native `fetch` (or without an XHR-like API) should inject one via // `options.alternateFetch`. const XmlHttpRequestApi = hasXMLHttpRequest() && g ? g.XMLHttpRequest : undefined const ActiveXObjectApi = typeof ActiveXObject === 'function' && g ? g.ActiveXObject : undefined const UNSAFE_KEYS = ['__proto__', 'constructor', 'prototype'] const addQueryString = (url, params) => { if (params && typeof params === 'object') { let queryString = '' // Must encode data. Iterate own enumerable keys only (Object.keys) and // drop prototype-chain names so a polluted Object.prototype cannot leak // into the URL. for (const paramName of Object.keys(params)) { if (UNSAFE_KEYS.indexOf(paramName) > -1) continue queryString += '&' + encodeURIComponent(paramName) + '=' + encodeURIComponent(params[paramName]) } if (!queryString) return url url = url + (url.indexOf('?') !== -1 ? '&' : '?') + queryString.slice(1) } return url } const fetchIt = (url, fetchOptions, callback, altFetch) => { const resolver = (response) => { if (!response.ok) return callback(response.statusText || 'Error', { status: response.status }) response.text().then((data) => { callback(null, { status: response.status, data }) }).catch(callback) } if (altFetch) { // already checked to have the proper signature const altResponse = altFetch(url, fetchOptions) if (altResponse instanceof Promise) { altResponse.then(resolver).catch(callback) return } // fall through } if (typeof fetch === 'function') { // react-native debug mode needs the fetch function to be called directly (no alias) fetch(url, fetchOptions).then(resolver).catch(callback) } else { fetchApi(url, fetchOptions).then(resolver).catch(callback) } } // fetch api stuff const requestWithFetch = (options, url, payload, callback) => { if (options.queryStringParams) { url = addQueryString(url, options.queryStringParams) } const headers = { ...(typeof options.customHeaders === 'function' ? options.customHeaders() : options.customHeaders) } if (typeof window === 'undefined' && typeof global !== 'undefined' && typeof global.process !== 'undefined' && global.process.versions && global.process.versions.node) { headers['User-Agent'] = `i18next-http-backend (node/${global.process.version}; ${global.process.platform} ${global.process.arch})` } if (payload) headers['Content-Type'] = 'application/json' const reqOptions = typeof options.requestOptions === 'function' ? options.requestOptions(payload) : options.requestOptions // `omitFetchOptions` is stored on the backend `options` object (per-instance) // instead of on a module-level flag. This prevents cross-instance // pollution: one instance hitting a "not implemented" fetch no longer // silently strips requestOptions from every other instance in the process. const fetchOptions = { method: payload ? 'POST' : 'GET', body: payload ? options.stringify(payload) : undefined, headers, ...(options._omitFetchOptions ? {} : reqOptions) } const altFetch = typeof options.alternateFetch === 'function' && options.alternateFetch.length >= 1 ? options.alternateFetch : undefined try { fetchIt(url, fetchOptions, callback, altFetch) } catch (e) { if (!reqOptions || Object.keys(reqOptions).length === 0 || !e.message || e.message.indexOf('not implemented') < 0) { return callback(e) } try { Object.keys(reqOptions).forEach((opt) => { delete fetchOptions[opt] }) fetchIt(url, fetchOptions, callback, altFetch) // Per-instance flag: only affects this backend, not siblings. options._omitFetchOptions = true } catch (err) { callback(err) } } } // xml http request stuff const requestWithXmlHttpRequest = (options, url, payload, callback) => { if (payload && typeof payload === 'object') { // if (!cache) payload._t = Date.now() // URL encoded form payload must be in querystring format payload = addQueryString('', payload).slice(1) } if (options.queryStringParams) { url = addQueryString(url, options.queryStringParams) } try { const x = XmlHttpRequestApi ? new XmlHttpRequestApi() : new ActiveXObjectApi('MSXML2.XMLHTTP.3.0') x.open(payload ? 'POST' : 'GET', url, 1) if (!options.crossDomain) { x.setRequestHeader('X-Requested-With', 'XMLHttpRequest') } x.withCredentials = !!options.withCredentials if (payload) { x.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded') } if (x.overrideMimeType) { x.overrideMimeType('application/json') } let h = options.customHeaders h = typeof h === 'function' ? h() : h if (h) { for (const i of Object.keys(h)) { if (UNSAFE_KEYS.indexOf(i) > -1) continue x.setRequestHeader(i, h[i]) } } x.onreadystatechange = () => { x.readyState > 3 && callback(x.status >= 400 ? x.statusText : null, { status: x.status, data: x.responseText }) } x.send(payload) } catch (e) { console && console.log(e) } } const request = (options, url, payload, callback) => { if (typeof payload === 'function') { callback = payload payload = undefined } callback = callback || (() => {}) if (fetchApi && url.indexOf('file:') !== 0) { // use fetch api return requestWithFetch(options, url, payload, callback) } if (hasXMLHttpRequest() || typeof ActiveXObject === 'function') { // use xml http request return requestWithXmlHttpRequest(options, url, payload, callback) } callback(new Error('No fetch and no xhr implementation found!')) } export default request