mirror of
				https://github.com/therootcompany/request.js.git
				synced 2024-11-16 17:28:58 +00:00 
			
		
		
		
	feat(browser): make script-tag friendly
This commit is contained in:
		
							parent
							
								
									91977b84e3
								
							
						
					
					
						commit
						fd6ed8722e
					
				
							
								
								
									
										394
									
								
								urequest.js
									
									
									
									
									
								
							
							
						
						
									
										394
									
								
								urequest.js
									
									
									
									
									
								
							| @ -1,223 +1,229 @@ | |||||||
| 'use strict'; | (function (exports) { | ||||||
|  |     'use strict'; | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * @typedef {import('./').Request} Request |  | ||||||
|  * @typedef {import('./').RequestOptions} RequestOptions |  | ||||||
|  * @typedef {import('./').Response} Response |  | ||||||
|  * @typedef {import('./').Headers} Headers |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| // `fetch` will be available for node and browsers as a global
 |  | ||||||
| //var fetch = window.fetch;
 |  | ||||||
| 
 |  | ||||||
| // https://developer.mozilla.org/en-US/docs/Web/API/fetch
 |  | ||||||
| // https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API/Using_Fetch
 |  | ||||||
| let _fetchDefaults = { |  | ||||||
|     method: 'GET', // *GET, POST, PATCH, PUT, DELETE, etc
 |  | ||||||
|     headers: {}, |  | ||||||
|     body: undefined, // String, ArrayBuffer, FormData, etc
 |  | ||||||
|     mode: 'cors', // no-cors, *cors, same-origin
 |  | ||||||
|     credentials: 'same-origin', // omit, *same-origin, include
 |  | ||||||
|     cache: 'default', // *default, no-store, reload, no-cache, force-cache, only-if-cached
 |  | ||||||
|     redirect: 'follow', // *follow, error, manual,
 |  | ||||||
|     referrer: undefined, |  | ||||||
|     referrerPolicy: 'no-referrer-when-downgrade', // no-referrer, *no-referrer-when-downgrade, same-origin, origin, strict-origin, origin-when-cross-origin, strict-origin-when-cross-origin, unsafe-url
 |  | ||||||
|     integrity: '', |  | ||||||
|     keepalive: false, |  | ||||||
|     signal: null //
 |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| let _optionKeys = Object.keys(_fetchDefaults).concat([ |  | ||||||
|     //'encoding', // N/A
 |  | ||||||
|     //'stream', // TODO via getReader
 |  | ||||||
|     //'json' // handled manually
 |  | ||||||
|     //'form', // TODO
 |  | ||||||
|     //'auth' // handled manually
 |  | ||||||
|     //'formData', // TODO
 |  | ||||||
|     //'FormData', // TODO
 |  | ||||||
|     //'userAgent' // not allowed, non-standard for request.js
 |  | ||||||
| ]); |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * @returns {Request>} |  | ||||||
|  */ |  | ||||||
| function setDefaults(_defs) { |  | ||||||
|     /** |     /** | ||||||
|      * @param {RequestOptions} opts |      * @typedef {import('./').Request} Request | ||||||
|      * @returns {Promise<Response>} |      * @typedef {import('./').RequestOptions} RequestOptions | ||||||
|      **/ |      * @typedef {import('./').Response} Response | ||||||
|     async function request(opts) { |      * @typedef {import('./').Headers} Headers | ||||||
|         if ('string' === typeof opts) { |      */ | ||||||
|             opts = { url: opts }; |  | ||||||
|         } |  | ||||||
|         let reqOpts = { headers: {} }; |  | ||||||
| 
 | 
 | ||||||
|         if ( |     // `fetch` will be available for node and browsers as a global
 | ||||||
|             opts.body || |     //var fetch = window.fetch;
 | ||||||
|             (opts.json && true !== opts.json) || |  | ||||||
|             opts.form || |  | ||||||
|             opts.formData |  | ||||||
|         ) { |  | ||||||
|             // TODO this is probably a deviation from request's API
 |  | ||||||
|             // need to check and probably eliminate it
 |  | ||||||
|             reqOpts.method = (reqOpts.method || 'POST').toUpperCase(); |  | ||||||
|         } else { |  | ||||||
|             reqOpts.method = (reqOpts.method || 'GET').toUpperCase(); |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         _optionKeys.forEach(function (key) { |     // https://developer.mozilla.org/en-US/docs/Web/API/fetch
 | ||||||
|             if (key in opts) { |     // https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API/Using_Fetch
 | ||||||
|                 if ('undefined' !== typeof opts[key]) { |     let _fetchDefaults = { | ||||||
|                     reqOpts[key] = opts[key]; |         method: 'GET', // *GET, POST, PATCH, PUT, DELETE, etc
 | ||||||
|  |         headers: {}, | ||||||
|  |         body: undefined, // String, ArrayBuffer, FormData, etc
 | ||||||
|  |         mode: 'cors', // no-cors, *cors, same-origin
 | ||||||
|  |         credentials: 'same-origin', // omit, *same-origin, include
 | ||||||
|  |         cache: 'default', // *default, no-store, reload, no-cache, force-cache, only-if-cached
 | ||||||
|  |         redirect: 'follow', // *follow, error, manual,
 | ||||||
|  |         referrer: undefined, | ||||||
|  |         referrerPolicy: 'no-referrer-when-downgrade', // no-referrer, *no-referrer-when-downgrade, same-origin, origin, strict-origin, origin-when-cross-origin, strict-origin-when-cross-origin, unsafe-url
 | ||||||
|  |         integrity: '', | ||||||
|  |         keepalive: false, | ||||||
|  |         signal: null //
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     let _optionKeys = Object.keys(_fetchDefaults).concat([ | ||||||
|  |         //'encoding', // N/A
 | ||||||
|  |         //'stream', // TODO via getReader
 | ||||||
|  |         //'json' // handled manually
 | ||||||
|  |         //'form', // TODO
 | ||||||
|  |         //'auth' // handled manually
 | ||||||
|  |         //'formData', // TODO
 | ||||||
|  |         //'FormData', // TODO
 | ||||||
|  |         //'userAgent' // not allowed, non-standard for request.js
 | ||||||
|  |     ]); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * @returns {Request>} | ||||||
|  |      */ | ||||||
|  |     function setDefaults(_defs) { | ||||||
|  |         /** | ||||||
|  |          * @param {RequestOptions} opts | ||||||
|  |          * @returns {Promise<Response>} | ||||||
|  |          **/ | ||||||
|  |         async function request(opts) { | ||||||
|  |             if ('string' === typeof opts) { | ||||||
|  |                 opts = { url: opts }; | ||||||
|  |             } | ||||||
|  |             let reqOpts = { headers: {} }; | ||||||
|  | 
 | ||||||
|  |             if ( | ||||||
|  |                 opts.body || | ||||||
|  |                 (opts.json && true !== opts.json) || | ||||||
|  |                 opts.form || | ||||||
|  |                 opts.formData | ||||||
|  |             ) { | ||||||
|  |                 // TODO this is probably a deviation from request's API
 | ||||||
|  |                 // need to check and probably eliminate it
 | ||||||
|  |                 reqOpts.method = (reqOpts.method || 'POST').toUpperCase(); | ||||||
|  |             } else { | ||||||
|  |                 reqOpts.method = (reqOpts.method || 'GET').toUpperCase(); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             _optionKeys.forEach(function (key) { | ||||||
|  |                 if (key in opts) { | ||||||
|  |                     if ('undefined' !== typeof opts[key]) { | ||||||
|  |                         reqOpts[key] = opts[key]; | ||||||
|  |                     } | ||||||
|  |                 } else if (key in _defs) { | ||||||
|  |                     reqOpts[key] = _defs[key]; | ||||||
|                 } |                 } | ||||||
|             } else if (key in _defs) { |             }); | ||||||
|                 reqOpts[key] = _defs[key]; |  | ||||||
|             } |  | ||||||
|         }); |  | ||||||
| 
 | 
 | ||||||
|         if (opts.auth) { |             if (opts.auth) { | ||||||
|             // if opts.uri specifies auth it will be parsed by url.parse and passed directly to the http module
 |                 // if opts.uri specifies auth it will be parsed by url.parse and passed directly to the http module
 | ||||||
|             if ('string' !== typeof opts.auth) { |                 if ('string' !== typeof opts.auth) { | ||||||
|                 let u = opts.auth.user || opts.auth.username || ''; |                     let u = opts.auth.user || opts.auth.username || ''; | ||||||
|                 let p = opts.auth.pass || opts.auth.password || ''; |                     let p = opts.auth.pass || opts.auth.password || ''; | ||||||
|                 reqOpts.headers.Authorization = encodeBasicAuth(`${u}:${p}`); |                     reqOpts.headers.Authorization = encodeBasicAuth( | ||||||
|             } else if ('string' === typeof opts.auth) { |                         `${u}:${p}` | ||||||
|                 reqOpts.headers.Authorization = encodeBasicAuth(`${opts.auth}`); |                     ); | ||||||
|  |                 } else if ('string' === typeof opts.auth) { | ||||||
|  |                     reqOpts.headers.Authorization = encodeBasicAuth( | ||||||
|  |                         `${opts.auth}` | ||||||
|  |                     ); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // [request-compat]
 | ||||||
|  |                 if (opts.auth.bearer) { | ||||||
|  |                     // having a shortcut for base64 encoding makes sense,
 | ||||||
|  |                     // but this? Eh, whatevs...
 | ||||||
|  |                     reqOpts.header.Authorization = `Bearer ${opts.auth.bearer}`; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             // [request-compat]
 |             let body; | ||||||
|             if (opts.auth.bearer) { |             if (opts.json && true !== opts.json) { | ||||||
|                 // having a shortcut for base64 encoding makes sense,
 |                 if (!opts.headers['content-type']) { | ||||||
|                 // but this? Eh, whatevs...
 |                     opts.headers['content-type'] = 'application/json'; | ||||||
|                 reqOpts.header.Authorization = `Bearer ${opts.auth.bearer}`; |                 } | ||||||
|  |                 body = JSON.stringify(opts.json); | ||||||
|  |                 if (!opts.method) { | ||||||
|  |                     opts.method = 'POST'; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         let body; |             // The node version will send HTTP Auth by default, but not Cookies.
 | ||||||
|         if (opts.json && true !== opts.json) { |             // We don't have an equivalent option for `fetch`. Furthermore,
 | ||||||
|             if (!opts.headers['content-type']) { |             // `fetch` caches HTTP Auth Basic across browser refreshes,
 | ||||||
|                 opts.headers['content-type'] = 'application/json'; |             // which is not analogous to the node behavior.
 | ||||||
|  |             //
 | ||||||
|  |             // "In the face of ambiguity, refuse the temptation to guess"
 | ||||||
|  |             //
 | ||||||
|  |             //if (!('credentials' in opts)) {
 | ||||||
|  |             //    opts.credentials = 'include';
 | ||||||
|  |             //}
 | ||||||
|  | 
 | ||||||
|  |             if (!('mode' in opts)) { | ||||||
|  |                 reqOpts.mode = 'cors'; | ||||||
|             } |             } | ||||||
|             body = JSON.stringify(opts.json); |             if (!('body' in opts)) { | ||||||
|             if (!opts.method) { |                 if (body) { | ||||||
|                 opts.method = 'POST'; |                     reqOpts.body = body; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         // The node version will send HTTP Auth by default, but not Cookies.
 |             let resp = await fetch(opts.url, reqOpts); | ||||||
|         // We don't have an equivalent option for `fetch`. Furthermore,
 |  | ||||||
|         // `fetch` caches HTTP Auth Basic across browser refreshes,
 |  | ||||||
|         // which is not analogous to the node behavior.
 |  | ||||||
|         //
 |  | ||||||
|         // "In the face of ambiguity, refuse the temptation to guess"
 |  | ||||||
|         //
 |  | ||||||
|         //if (!('credentials' in opts)) {
 |  | ||||||
|         //    opts.credentials = 'include';
 |  | ||||||
|         //}
 |  | ||||||
| 
 | 
 | ||||||
|         if (!('mode' in opts)) { |             let result = { | ||||||
|             reqOpts.mode = 'cors'; |                 ok: resp.ok, | ||||||
|         } |                 headers: headersToObj(resp.headers), | ||||||
|         if (!('body' in opts)) { |                 body: undefined, | ||||||
|             if (body) { |                 // swapped to match request.js
 | ||||||
|                 reqOpts.body = body; |                 statusCode: resp.status, | ||||||
|             } |                 status: resp.statusText, | ||||||
|         } |                 request: reqOpts, | ||||||
| 
 |                 response: resp | ||||||
|         let resp = await fetch(opts.url, reqOpts); |             }; | ||||||
| 
 |             result.toJSON = function () { | ||||||
|         let result = { |                 return { | ||||||
|             ok: resp.ok, |                     ok: result.ok, | ||||||
|             headers: headersToObj(resp.headers), |                     headers: result.headers, | ||||||
|             body: undefined, |                     body: result.body, | ||||||
|             // swapped to match request.js
 |                     statusCode: result.statusCode, | ||||||
|             statusCode: resp.status, |                     status: result.status | ||||||
|             status: resp.statusText, |                 }; | ||||||
|             request: reqOpts, |  | ||||||
|             response: resp |  | ||||||
|         }; |  | ||||||
|         result.toJSON = function () { |  | ||||||
|             return { |  | ||||||
|                 ok: result.ok, |  | ||||||
|                 headers: result.headers, |  | ||||||
|                 body: result.body, |  | ||||||
|                 statusCode: result.statusCode, |  | ||||||
|                 status: result.status |  | ||||||
|             }; |             }; | ||||||
|         }; |  | ||||||
| 
 | 
 | ||||||
|         // return early if there's no body
 |             // return early if there's no body
 | ||||||
|         if (!result.headers['content-type']) { |             if (!result.headers['content-type']) { | ||||||
|  |                 return result; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // TODO blob, formData ?
 | ||||||
|  |             if (null === opts.encoding) { | ||||||
|  |                 return await resp.arrayBuffer(); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (!opts.json) { | ||||||
|  |                 result.body = await resp.text(); | ||||||
|  |             } else { | ||||||
|  |                 result.body = await resp.json().catch(async function () { | ||||||
|  |                     return await resp.text(); | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|             return result; |             return result; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // TODO blob, formData ?
 |         return request; | ||||||
|         if (null === opts.encoding) { |  | ||||||
|             return await resp.arrayBuffer(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (!opts.json) { |  | ||||||
|             result.body = await resp.text(); |  | ||||||
|         } else { |  | ||||||
|             result.body = await resp.json().catch(async function () { |  | ||||||
|                 return await resp.text(); |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|         return result; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return request; |     /** | ||||||
| } |      * @param {Iterable.<*>} rheaders | ||||||
| 
 |      * @returns {Object.<String, String>} | ||||||
| /** |      */ | ||||||
|  * @param {Iterable.<*>} rheaders |     function headersToObj(rheaders) { | ||||||
|  * @returns {Object.<String, String>} |         /* | ||||||
|  */ |  | ||||||
| function headersToObj(rheaders) { |  | ||||||
|     /* |  | ||||||
|     Array.from(resp.headers.entries()).forEach(function (h) { |     Array.from(resp.headers.entries()).forEach(function (h) { | ||||||
|         headers[h[0]] = h[1]; |         headers[h[0]] = h[1]; | ||||||
|     }); |     }); | ||||||
|     */ |     */ | ||||||
|     let headerNames = Array.from(rheaders.keys()); |         let headerNames = Array.from(rheaders.keys()); | ||||||
|     let resHeaders = {}; |         let resHeaders = {}; | ||||||
|     headerNames.forEach(function (k) { |         headerNames.forEach(function (k) { | ||||||
|         resHeaders[k] = rheaders.get(k); |             resHeaders[k] = rheaders.get(k); | ||||||
|     }); |         }); | ||||||
|     return resHeaders; |         return resHeaders; | ||||||
| } |     } | ||||||
| 
 | 
 | ||||||
| /** |     /** | ||||||
|  * @param {String} utf8 |      * @param {String} utf8 | ||||||
|  * @returns {String} |      * @returns {String} | ||||||
|  */ |      */ | ||||||
| function encodeBasicAuth(utf8) { |     function encodeBasicAuth(utf8) { | ||||||
|     let b64 = unicodeToBase64(utf8); |         let b64 = unicodeToBase64(utf8); | ||||||
|     return `Basic ${b64}`; |         return `Basic ${b64}`; | ||||||
| } |     } | ||||||
| 
 | 
 | ||||||
| /** |     /** | ||||||
|  * @param {String} utf8 |      * @param {String} utf8 | ||||||
|  * @returns {String} |      * @returns {String} | ||||||
|  */ |      */ | ||||||
| function unicodeToBase64(utf8) { |     function unicodeToBase64(utf8) { | ||||||
|     let str = ''; |         let str = ''; | ||||||
|     let uint8 = new TextEncoder().encode(utf8); |         let uint8 = new TextEncoder().encode(utf8); | ||||||
|     uint8.forEach(function (b) { |         uint8.forEach(function (b) { | ||||||
|         str += String.fromCharCode(b); |             str += String.fromCharCode(b); | ||||||
|     }); |         }); | ||||||
|     let b64 = btoa(str); |         let b64 = btoa(str); | ||||||
|     return b64; |         return b64; | ||||||
| } |     } | ||||||
| 
 | 
 | ||||||
| let defaultRequest = setDefaults({ mode: 'cors' }); |     let defaultRequest = setDefaults({ mode: 'cors' }); | ||||||
| //@ts-ignore
 |     //@ts-ignore
 | ||||||
| exports.urequest = defaultRequest; |     exports.urequest = defaultRequest; | ||||||
| //@ts-ignore
 |     //@ts-ignore
 | ||||||
| exports.urequest.defaults = setDefaults; |     exports.urequest.defaults = setDefaults; | ||||||
| 
 | 
 | ||||||
| // for backwards compat
 |     // for backwards compat
 | ||||||
| if ('undefined' !== typeof module) { |     if ('undefined' !== typeof module) { | ||||||
|     module.exports = defaultRequest; |         module.exports = defaultRequest; | ||||||
|     module.exports.defaults = setDefaults; |         module.exports.defaults = setDefaults; | ||||||
| } |     } | ||||||
|  | })(('undefined' !== typeof module && module.exports) || window); | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user