filestack-js
Version:
Official JavaScript library for Filestack
170 lines (168 loc) • 24.5 kB
JavaScript
import { __awaiter, __generator } from "tslib";
/*
* Copyright (c) 2018 by Filestack
* Some rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import Debug from 'debug';
import * as utils from '../utils';
import { FsHttpMethod } from '../types';
import { FsRequestError, FsRequestErrorCode } from '../error';
import { prepareData, parseResponse, parse as parseHeaders, combineURL } from './../helpers';
var debug = Debug('fs:request:xhr');
var CANCEL_CLEAR = "FsCleanMemory";
var XhrAdapter = /** @class */ (function () {
function XhrAdapter() {
}
XhrAdapter.prototype.request = function (config) {
var _this = this;
// if this option is unspecified set it by default
if (typeof config.filestackHeaders === 'undefined') {
config.filestackHeaders = true;
}
config = prepareData(config);
config.headers = config.headers || {};
var data = config.data, headers = config.headers;
// if data is type of form let browser to set proper content type
if (utils.isFormData(data)) {
delete headers['Content-Type'];
}
var request = new XMLHttpRequest();
if (config.blobResponse) {
request.responseType = 'blob';
}
// HTTP basic authentication
if (config.auth) {
if (!config.auth.username || config.auth.username.length === 0 || !config.auth.password || config.auth.password.length === 0) {
return Promise.reject(new FsRequestError("Basic auth: username and password are required ".concat(config.auth), config));
}
headers.Authorization = 'Basic ' + btoa(unescape(encodeURIComponent("".concat(config.auth.username, ":").concat(config.auth.password))));
debug('Set request authorization to %s', config.auth.username + config.auth.password);
}
var url = config.url.trim();
if (!/^http(s)?:\/\//.test(url)) {
url = "https://".concat(url);
}
url = combineURL(url, config.params);
debug('Starting request to %s with options %O', url, config);
request.open(config.method.toUpperCase(), url, true);
request.timeout = config.timeout;
return new Promise(function (resolve, reject) {
var cancelListener;
if (config.cancelToken) {
cancelListener = function (reason) {
/* istanbul ignore next: if request is done cancel token should not throw any error */
if (request) {
request.abort();
request = null;
}
debug('Request canceled by user %s, config: %O', reason, config);
return reject(new FsRequestError("Request aborted. Reason: ".concat(reason), config, null, FsRequestErrorCode.ABORTED));
};
config.cancelToken.once('cancel', cancelListener);
}
request.onreadystatechange = function () { return __awaiter(_this, void 0, void 0, function () {
var responseHeaders, responseData, response;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
if (!request || request.readyState !== 4) {
return [2 /*return*/];
}
if (request.status === 0 && !request.responseURL) {
return [2 /*return*/];
}
responseHeaders = parseHeaders(request.getAllResponseHeaders());
responseData = request.response;
response = {
data: responseData,
status: request.status,
statusText: request.statusText,
headers: responseHeaders,
config: config,
};
request = null;
return [4 /*yield*/, parseResponse(response)];
case 1:
response = _a.sent();
if (500 <= response.status && response.status <= 599) {
// server error throw
debug('Server error(5xx) - %O', response);
return [2 /*return*/, reject(new FsRequestError("Server error ".concat(url), config, response, FsRequestErrorCode.SERVER))];
}
else if (400 <= response.status && response.status <= 499) {
debug('Request error(4xx) - %O', response);
return [2 /*return*/, reject(new FsRequestError("Request error ".concat(url), config, response, FsRequestErrorCode.REQUEST))];
}
// clear cancel token to avoid memory leak
if (config.cancelToken) {
config.cancelToken.removeListener('cancel', cancelListener);
cancelListener = null;
}
return [2 /*return*/, resolve(response)];
}
});
}); };
// Handle browser request cancellation (as opposed to a manual cancellation)
request.onabort = function handleAbort() {
/* istanbul ignore next: just to be sure that abort was not called twice */
if (!request) {
return;
}
request = null;
reject(new FsRequestError('Request aborted', config, null, FsRequestErrorCode.ABORTED));
};
// Handle low level network errors
request.onerror = function handleError(err) {
request = null;
debug('Request error! %O', err);
reject(new FsRequestError('Network Error', config, null, FsRequestErrorCode.NETWORK));
};
// Handle timeout
request.ontimeout = function handleTimeout() {
request = null;
debug('Request timed out. %O', config);
reject(new FsRequestError('Request timeout', config, null, FsRequestErrorCode.TIMEOUT));
};
// Add headers to the request
if ('setRequestHeader' in request && headers && Object.keys(headers).length) {
for (var key in headers) {
if (headers[key] === undefined) {
continue;
}
debug('Set request header %s to %s', key, headers[key]);
request.setRequestHeader(key, headers[key]);
}
}
if (typeof config.onProgress === 'function' && [FsHttpMethod.POST, FsHttpMethod.PUT].indexOf(config.method) > -1) {
/* istanbul ignore else: else path is just fallback to normal progress event */
if (request.upload) {
debug('Bind to upload progress event');
request.upload.addEventListener('progress', config.onProgress);
}
else {
debug('Bind to progress event');
request.addEventListener('progress', config.onProgress);
}
}
if (data === undefined) {
data = null;
}
request.send(data);
});
};
return XhrAdapter;
}());
export { XhrAdapter };
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["../../src/lib/request/adapters/xhr.ts"],"names":[],"mappings":"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","file":"lib/request/adapters/xhr.js","sourcesContent":["/*\n * Copyright (c) 2018 by Filestack\n * Some rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n *     http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport Debug from 'debug';\nimport * as utils from '../utils';\nimport { AdapterInterface } from './interface';\nimport { FsRequestOptions, FsResponse, FsHttpMethod } from '../types';\nimport { FsRequestError, FsRequestErrorCode } from '../error';\nimport { prepareData, parseResponse, parse as parseHeaders, combineURL } from './../helpers';\n\nconst debug = Debug('fs:request:xhr');\nconst CANCEL_CLEAR = `FsCleanMemory`;\n\nexport class XhrAdapter implements AdapterInterface {\n\n  request(config: FsRequestOptions) {\n    // if this option is unspecified set it by default\n    if (typeof config.filestackHeaders === 'undefined') {\n      config.filestackHeaders = true;\n    }\n\n    config = prepareData(config);\n    config.headers = config.headers || {};\n\n    let { data, headers } = config;\n\n    // if data is type of form let browser to set proper content type\n    if (utils.isFormData(data)) {\n      delete headers['Content-Type'];\n    }\n\n    let request = new XMLHttpRequest();\n\n    if (config.blobResponse) {\n      request.responseType = 'blob';\n    }\n\n    // HTTP basic authentication\n    if (config.auth) {\n      if (!config.auth.username || config.auth.username.length === 0 || !config.auth.password || config.auth.password.length === 0) {\n        return Promise.reject(new FsRequestError(`Basic auth: username and password are required ${config.auth}`, config));\n      }\n\n      headers.Authorization = 'Basic ' + btoa(unescape(encodeURIComponent(`${config.auth.username}:${config.auth.password}`)));\n      debug('Set request authorization to %s', config.auth.username + config.auth.password);\n    }\n\n    let url = config.url.trim();\n\n    if (!/^http(s)?:\\/\\//.test(url)) {\n      url = `https://${url}`;\n    }\n\n    url = combineURL(url, config.params);\n\n    debug('Starting request to %s with options %O', url, config);\n\n    request.open(config.method.toUpperCase(), url, true);\n\n    request.timeout = config.timeout;\n\n    return new Promise<FsResponse>((resolve, reject) => {\n      let cancelListener;\n\n      if (config.cancelToken) {\n        cancelListener = (reason) => {\n          /* istanbul ignore next: if request is done cancel token should not throw any error */\n          if (request) {\n            request.abort();\n            request = null;\n          }\n\n          debug('Request canceled by user %s, config: %O', reason, config);\n          return reject(new FsRequestError(`Request aborted. Reason: ${reason}`, config, null, FsRequestErrorCode.ABORTED));\n        };\n\n        config.cancelToken.once('cancel', cancelListener);\n      }\n\n      request.onreadystatechange = async () => {\n        if (!request || request.readyState !== 4) {\n          return;\n        }\n\n        if (request.status === 0 && !request.responseURL) {\n          return;\n        }\n\n        // Prepare the response\n        const responseHeaders = parseHeaders(request.getAllResponseHeaders());\n        const responseData = request.response;\n\n        let response: FsResponse = {\n          data: responseData,\n          status: request.status,\n          statusText: request.statusText,\n          headers: responseHeaders,\n          config: config,\n        };\n\n        request = null;\n        response = await parseResponse(response);\n\n        if (500 <= response.status && response.status <= 599) {\n          // server error throw\n          debug('Server error(5xx) - %O', response);\n          return reject(new FsRequestError(`Server error ${url}`, config, response, FsRequestErrorCode.SERVER));\n        } else if (400 <= response.status && response.status <= 499) {\n          debug('Request error(4xx) - %O', response);\n          return reject(new FsRequestError(`Request error ${url}`, config, response, FsRequestErrorCode.REQUEST));\n        }\n\n        // clear cancel token to avoid memory leak\n        if (config.cancelToken) {\n          config.cancelToken.removeListener('cancel', cancelListener);\n          cancelListener = null;\n        }\n\n        return resolve(response);\n      };\n\n      // Handle browser request cancellation (as opposed to a manual cancellation)\n      request.onabort = function handleAbort() {\n        /* istanbul ignore next: just to be sure that abort was not called twice */\n        if (!request) {\n          return;\n        }\n\n        request = null;\n        reject(new FsRequestError('Request aborted', config, null, FsRequestErrorCode.ABORTED));\n      };\n\n      // Handle low level network errors\n      request.onerror = function handleError(err) {\n        request = null;\n        debug('Request error! %O', err);\n        reject(new FsRequestError('Network Error', config, null, FsRequestErrorCode.NETWORK));\n      };\n\n      // Handle timeout\n      request.ontimeout = function handleTimeout() {\n        request = null;\n        debug('Request timed out. %O', config);\n        reject(new FsRequestError('Request timeout', config, null, FsRequestErrorCode.TIMEOUT));\n      };\n\n      // Add headers to the request\n      if ('setRequestHeader' in request && headers && Object.keys(headers).length) {\n        for (let key in headers) {\n          if (headers[key] === undefined) {\n            continue;\n          }\n\n          debug('Set request header %s to %s', key, headers[key]);\n          request.setRequestHeader(key, headers[key]);\n        }\n      }\n\n      if (typeof config.onProgress === 'function' && [FsHttpMethod.POST, FsHttpMethod.PUT].indexOf(config.method) > -1) {\n        /* istanbul ignore else: else path is just fallback to normal progress event */\n        if (request.upload) {\n          debug('Bind to upload progress event');\n          request.upload.addEventListener('progress', config.onProgress);\n        } else {\n          debug('Bind to progress event');\n          request.addEventListener('progress', config.onProgress);\n        }\n      }\n\n      if (data === undefined) {\n        data = null;\n      }\n\n      request.send(data);\n    });\n  }\n}\n"]}