1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 | exports.httpReq = httpReq;
|
7 | exports.httpReq2 = httpReq2;
|
8 |
|
9 | var _request = _interopRequireDefault(require("request"));
|
10 |
|
11 | var _lodash = _interopRequireDefault(require("lodash"));
|
12 |
|
13 | var _asyncUtils = require("./asyncUtils");
|
14 |
|
15 | var _errorUtils = require("./errorUtils");
|
16 |
|
17 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
18 |
|
19 | const promisifiedRequest = async options => await new Promise((resp, rej) => (0, _request.default)(options, (err, data) => err ? rej(err) : resp(data)));
|
20 |
|
21 | const DEFAULT_REQ_HEADERS = {};
|
22 | const DEFAULT_REQ_REJECT_UNAUTHORIZED = true;
|
23 |
|
24 | async function httpReq(url, options = {}) {
|
25 | let body = null;
|
26 |
|
27 | try {
|
28 | options = options || {};
|
29 | options.url = url;
|
30 | options.method = options.method ? options.method.toUpperCase() : 'GET';
|
31 | options.timeout = typeof options.timeout !== 'undefined' ? options.timeout : 5000;
|
32 | options.headers = options.headers || DEFAULT_REQ_HEADERS;
|
33 | options.body = typeof options.body === 'string' ? options.body : JSON.stringify(options.body);
|
34 | options.rejectUnauthorized = typeof options.rejectUnauthorized !== 'undefined' ? options.rejectUnauthorized : DEFAULT_REQ_REJECT_UNAUTHORIZED;
|
35 | options.maxRetries = options.maxRetries || 0;
|
36 | options.expoBackoffMs = options.expoBackoffMs || 100;
|
37 | return await (0, _asyncUtils.asyncRetry)(async triesExecuted => {
|
38 | body = null;
|
39 | const resp = await promisifiedRequest(options);
|
40 |
|
41 | if (options.parseJsonResponse === true) {
|
42 | try {
|
43 | body = resp.body;
|
44 | resp.bodyObj = JSON.parse(resp.body);
|
45 | resp.triesExecuted = triesExecuted;
|
46 | } catch (err) {
|
47 | // NOTE: don't change this string without searching the code for it
|
48 | throw new _errorUtils.ExtendedError('Failed to parse HTTP response as JSON', {
|
49 | url: url,
|
50 | bodyStart: resp.body.substr(0, 1000)
|
51 | });
|
52 | }
|
53 | }
|
54 |
|
55 | return resp;
|
56 | }, {
|
57 | opName: `${options.url}`,
|
58 | maxRetries: options.maxRetries,
|
59 | expoBackoffMs: options.expoBackoffMs,
|
60 | asyncErrorHandler: options.asyncErrorHandler
|
61 | });
|
62 | } catch (err) {
|
63 | throw new _errorUtils.ExtendedError('HTTP request failed', {
|
64 | url,
|
65 | err,
|
66 | body
|
67 | });
|
68 | }
|
69 | }
|
70 |
|
71 | async function httpReq2(url, options = {}) {
|
72 | let body = null;
|
73 |
|
74 | try {
|
75 | options = options || {};
|
76 | options.url = url;
|
77 | options.method = options.method ? options.method.toUpperCase() : 'GET';
|
78 | options.timeout = typeof options.timeout !== 'undefined' ? options.timeout : 5000;
|
79 | options.headers = options.headers || DEFAULT_REQ_HEADERS;
|
80 | options.body = typeof options.body === 'string' ? options.body : JSON.stringify(options.body);
|
81 | options.rejectUnauthorized = typeof options.rejectUnauthorized !== 'undefined' ? options.rejectUnauthorized : DEFAULT_REQ_REJECT_UNAUTHORIZED;
|
82 | options.resultHandlers = options.resultHandlers ? _lodash.default.clone(options.resultHandlers) : options.resultHandlers;
|
83 | let triesLeft = 1;
|
84 | let triesExecuted = 0;
|
85 | let resp = null;
|
86 | let error = null;
|
87 |
|
88 | while (triesLeft > 0) {
|
89 | try {
|
90 | triesExecuted++;
|
91 | triesLeft--;
|
92 | resp = await promisifiedRequest(options);
|
93 | } catch (err) {
|
94 | error = err;
|
95 | }
|
96 |
|
97 | if (options.resultHandlers) {
|
98 | for (const resultHandler of options.resultHandlers) {
|
99 | if (resultHandler.onStatus == null && resultHandler.onError == null || resp && typeof resultHandler.onStatus === 'number' && resultHandler.onStatus === resp.statusCode || resp && resultHandler.onStatus && typeof resultHandler.onStatus.test === 'function' && // $FlowIgnore
|
100 | resultHandler.onStatus.test(resp.statusCode.toString()) || error && resultHandler.onError && typeof resultHandler.onError.test === 'function' && resultHandler.onError.test(error.message)) {
|
101 | if (resultHandler.retry != null) {
|
102 | if (resultHandler.retry > 0) {
|
103 | triesLeft++;
|
104 | resultHandler.retry--; // $FlowIgnore
|
105 |
|
106 | resultHandler.nextBackoffMs = resultHandler.nextBackoffMs == null ? 100 : resultHandler.nextBackoffMs;
|
107 | await (0, _asyncUtils.sleep)(resultHandler.nextBackoffMs); // $FlowIgnore
|
108 |
|
109 | resultHandler.nextBackoffMs *= 1.5;
|
110 | error = undefined;
|
111 | resp = undefined;
|
112 | }
|
113 | } else if (resultHandler.throw != null) {
|
114 | throw resultHandler.throw;
|
115 | } else if (resultHandler.return != null) {
|
116 | return resultHandler.return;
|
117 | } else {
|
118 | throw new _errorUtils.ExtendedError('Invalid result handler in httpReq', resultHandler);
|
119 | }
|
120 | }
|
121 | }
|
122 | }
|
123 | }
|
124 |
|
125 | if (resp) {
|
126 | if (options.parseJsonResponse === true) {
|
127 | try {
|
128 | body = resp.body;
|
129 | resp.bodyObj = JSON.parse(resp.body);
|
130 | resp.triesExecuted = triesExecuted;
|
131 | } catch (err) {
|
132 | // NOTE: don't change this string without searching the code for it
|
133 | throw new _errorUtils.ExtendedError('Failed to parse HTTP response as JSON', {
|
134 | url: url,
|
135 | bodyStart: resp.body.substr(0, 1000)
|
136 | });
|
137 | }
|
138 | }
|
139 |
|
140 | return resp;
|
141 | }
|
142 |
|
143 | throw error;
|
144 | } catch (err) {
|
145 | throw new _errorUtils.ExtendedError('HTTP request failed', {
|
146 | url,
|
147 | err,
|
148 | body
|
149 | });
|
150 | }
|
151 | }
|
152 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/uninstrumentedHttpUtils.js"],"names":["promisifiedRequest","options","Promise","resp","rej","err","data","DEFAULT_REQ_HEADERS","DEFAULT_REQ_REJECT_UNAUTHORIZED","httpReq","url","body","method","toUpperCase","timeout","headers","JSON","stringify","rejectUnauthorized","maxRetries","expoBackoffMs","triesExecuted","parseJsonResponse","bodyObj","parse","ExtendedError","bodyStart","substr","opName","asyncErrorHandler","httpReq2","resultHandlers","_","clone","triesLeft","error","resultHandler","onStatus","onError","statusCode","test","toString","message","retry","nextBackoffMs","undefined","throw","return"],"mappings":"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","sourcesContent":["// @flow\n\nimport request from 'request'\nimport _ from 'lodash'\n\nimport { asyncRetry, sleep } from './asyncUtils'\nimport { ExtendedError } from './errorUtils'\n\nconst promisifiedRequest = async (options: Object) => await new Promise((resp, rej) => request(options, (err, data) => err ? rej(err) : resp(data)))\n\nconst DEFAULT_REQ_HEADERS = {}\nconst DEFAULT_REQ_REJECT_UNAUTHORIZED = true\n\nexport type HttpReqResponse = {|\n  bodyObj: ?Object, \n  body: string, \n  statusCode: number, \n  triesExecuted: number,\n|}\n\nexport type HttpReqOptions = {\n  url?: ?string,\n  method?: string,\n  timeout?: ?number,\n  headers?: { [id: string]: string },\n  body?: ?string | Object,\n  maxRetries?: number,\n  expoBackoffMs?: ?number,\n  asyncErrorHandler?: ?(Object) => Promise<void>,\n  rejectUnauthorized?: boolean,\n  trackingName?: ?string,\n  parseJsonResponse?: ?boolean,\n}\n\nexport async function httpReq(url: string, options: HttpReqOptions = {}): Promise<HttpReqResponse> {\n  let body: ?string = null\n  try {\n    options = options || {}\n    options.url = url\n    options.method = options.method ? options.method.toUpperCase() : 'GET'\n    options.timeout = typeof options.timeout !== 'undefined' ? options.timeout : 5000\n    options.headers = options.headers || DEFAULT_REQ_HEADERS\n    options.body = typeof options.body === 'string' ? options.body : JSON.stringify(options.body)\n    options.rejectUnauthorized = typeof options.rejectUnauthorized !== 'undefined' ? options.rejectUnauthorized : DEFAULT_REQ_REJECT_UNAUTHORIZED\n\n    options.maxRetries = options.maxRetries || 0\n    options.expoBackoffMs = options.expoBackoffMs || 100\n\n    return await asyncRetry(async (triesExecuted) => {\n      body = null\n      const resp = await promisifiedRequest(options)\n      if (options.parseJsonResponse === true) {\n        try {\n          body = resp.body\n          resp.bodyObj = JSON.parse(resp.body)\n          resp.triesExecuted = triesExecuted\n        } catch (err) {\n          // NOTE: don't change this string without searching the code for it\n          throw new ExtendedError('Failed to parse HTTP response as JSON', { url: url, bodyStart: resp.body.substr(0, 1000) })\n        } \n      }\n      return resp\n    }, {\n      opName: `${options.url}`,\n      maxRetries: options.maxRetries,\n      expoBackoffMs: options.expoBackoffMs,\n      asyncErrorHandler: options.asyncErrorHandler,\n    })\n  } catch (err) {\n    throw new ExtendedError('HTTP request failed', { url, err, body })\n  }\n}\n\n\ntype ResultHandlerTrigger = {|\n  onError: RegExp,\n|} | {|\n  onStatus: number | RegExp,\n|} | {||}\ntype ResultHandlerAction = {|\n  retry: number,\n  nextBackoffMs?: ?number,\n|} | {|\n  throw: Error,\n|} | {|\n  return: HttpReqResponse,\n|}\ntype ResultHandler = {|\n  ...ResultHandlerTrigger,\n  ...ResultHandlerAction,\n|}\n\nexport type HttpReqOptions2 = {\n  url?: ?string,\n  method?: string,\n  timeout?: ?number,\n  headers?: { [id: string]: string },\n  body?: ?string | Object,\n  maxRetries?: number,\n  rejectUnauthorized?: boolean,\n  trackingName?: ?string,\n  parseJsonResponse?: ?boolean,\n  resultHandlers?: ?Array<ResultHandler>\n}\n\nexport async function httpReq2(url: string, options: HttpReqOptions2 = {}): Promise<HttpReqResponse> {\n  let body: ?string = null\n  try {\n    options = options || {}\n    options.url = url\n    options.method = options.method ? options.method.toUpperCase() : 'GET'\n    options.timeout = typeof options.timeout !== 'undefined' ? options.timeout : 5000\n    options.headers = options.headers || DEFAULT_REQ_HEADERS\n    options.body = typeof options.body === 'string' ? options.body : JSON.stringify(options.body)\n    options.rejectUnauthorized = typeof options.rejectUnauthorized !== 'undefined' ? options.rejectUnauthorized : DEFAULT_REQ_REJECT_UNAUTHORIZED\n    options.resultHandlers = options.resultHandlers ? _.clone(options.resultHandlers) : options.resultHandlers\n\n    let triesLeft = 1\n    let triesExecuted = 0\n\n    let resp: ?HttpReqResponse = null\n    let error: ?Error = null\n\n    while (triesLeft > 0) {\n\n      try {\n        triesExecuted++\n        triesLeft--\n        resp = await promisifiedRequest(options)\n      } catch (err) {\n        error = err\n      }\n      \n      if (options.resultHandlers) {\n        for (const resultHandler of options.resultHandlers) {\n          if (\n            (\n              resultHandler.onStatus == null && \n              resultHandler.onError == null\n            ) || (\n              resp && \n              typeof resultHandler.onStatus === 'number' &&\n              resultHandler.onStatus === resp.statusCode\n            ) || (\n              resp && \n              resultHandler.onStatus && \n              typeof resultHandler.onStatus.test === 'function' && \n              // $FlowIgnore\n              resultHandler.onStatus.test(resp.statusCode.toString())\n            ) || (\n              error && \n              resultHandler.onError && \n              typeof resultHandler.onError.test === 'function' && \n              resultHandler.onError.test(error.message)\n            )\n          ) {\n            if (resultHandler.retry != null) {\n              if (resultHandler.retry > 0) {\n                triesLeft++\n                resultHandler.retry--\n                // $FlowIgnore\n                resultHandler.nextBackoffMs = resultHandler.nextBackoffMs == null ? 100 : resultHandler.nextBackoffMs\n                await sleep(resultHandler.nextBackoffMs)\n                // $FlowIgnore\n                resultHandler.nextBackoffMs *= 1.5\n\n                error = undefined\n                resp = undefined\n              }\n            } else if (resultHandler.throw != null) {\n              throw resultHandler.throw\n            } else if (resultHandler.return != null) {\n              return resultHandler.return\n            } else {\n              throw new ExtendedError('Invalid result handler in httpReq', resultHandler)\n            }\n          }\n        }\n      }\n    }\n\n    if (resp) {\n      if (options.parseJsonResponse === true) {\n        try {\n          body = resp.body\n          resp.bodyObj = JSON.parse(resp.body)\n          resp.triesExecuted = triesExecuted\n        } catch (err) {\n          // NOTE: don't change this string without searching the code for it\n          throw new ExtendedError('Failed to parse HTTP response as JSON', { url: url, bodyStart: resp.body.substr(0, 1000) })\n        }\n      }\n      return resp\n    }\n\n    throw error\n\n  } catch (err) {\n    throw new ExtendedError('HTTP request failed', { url, err, body })\n  }\n}"]} |
\ | No newline at end of file |