@hpcc-js/comms 2.102.2 → 2.102.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (103) hide show
  1. package/LICENSE +43 -43
  2. package/README.md +50 -50
  3. package/dist/index.es6.js +2 -2
  4. package/dist/index.es6.js.map +1 -1
  5. package/dist/index.js +2 -2
  6. package/dist/index.js.map +1 -1
  7. package/dist/index.min.js +1 -1
  8. package/dist/index.min.js.map +1 -1
  9. package/dist/index.node.js +2 -2
  10. package/dist/index.node.js.map +1 -1
  11. package/dist/index.node.min.js +1 -1
  12. package/dist/index.node.min.js.map +1 -1
  13. package/lib-es6/__package__.js +2 -2
  14. package/package.json +4 -4
  15. package/src/__package__.ts +3 -3
  16. package/src/__tests__/dfuXRef.ts +22 -22
  17. package/src/__tests__/https.ts +69 -69
  18. package/src/__tests__/workunit.ts +35 -35
  19. package/src/clienttools/eclMeta.ts +506 -506
  20. package/src/clienttools/eclcc.ts +628 -628
  21. package/src/connection.ts +295 -295
  22. package/src/ecl/activity.ts +82 -82
  23. package/src/ecl/dfuWorkunit.ts +363 -363
  24. package/src/ecl/graph.ts +196 -196
  25. package/src/ecl/logicalFile.ts +195 -195
  26. package/src/ecl/machine.ts +63 -63
  27. package/src/ecl/query.ts +252 -252
  28. package/src/ecl/queryGraph.ts +813 -813
  29. package/src/ecl/resource.ts +39 -39
  30. package/src/ecl/result.ts +236 -236
  31. package/src/ecl/scope.ts +192 -192
  32. package/src/ecl/sourceFile.ts +34 -34
  33. package/src/ecl/store.ts +154 -154
  34. package/src/ecl/targetCluster.ts +149 -149
  35. package/src/ecl/timer.ts +42 -42
  36. package/src/ecl/topology.ts +131 -131
  37. package/src/ecl/workunit.ts +1314 -1314
  38. package/src/ecl/xsdParser.ts +268 -268
  39. package/src/espConnection.ts +172 -172
  40. package/src/index-common.ts +41 -41
  41. package/src/index.node.ts +68 -68
  42. package/src/index.ts +3 -3
  43. package/src/pem/trustwave.ts +909 -909
  44. package/src/services/fileSpray.ts +48 -48
  45. package/src/services/wsAccess.ts +8 -8
  46. package/src/services/wsAccount.ts +27 -27
  47. package/src/services/wsCloud.ts +73 -73
  48. package/src/services/wsCodesign.ts +94 -94
  49. package/src/services/wsDFU.ts +34 -34
  50. package/src/services/wsDFUXRef.ts +308 -308
  51. package/src/services/wsDali.ts +40 -40
  52. package/src/services/wsEcl.ts +123 -123
  53. package/src/services/wsElk.ts +8 -8
  54. package/src/services/wsLogaccess.ts +263 -263
  55. package/src/services/wsMachine.ts +89 -89
  56. package/src/services/wsPackageProcess.ts +8 -8
  57. package/src/services/wsResources.ts +8 -8
  58. package/src/services/wsSMC.ts +24 -24
  59. package/src/services/wsSasha.ts +7 -7
  60. package/src/services/wsStore.ts +230 -230
  61. package/src/services/wsTopology.ts +45 -45
  62. package/src/services/wsWorkunits.ts +160 -160
  63. package/src/services/wsdl/FileSpray/v1.23/FileSpray.ts +1008 -1008
  64. package/src/services/wsdl/FileSpray/v1.25/FileSpray.ts +1040 -1040
  65. package/src/services/wsdl/FileSpray/v1.26/FileSpray.ts +929 -929
  66. package/src/services/wsdl/WsCloud/v1/WsCloud.ts +38 -38
  67. package/src/services/wsdl/WsCloud/v1.02/WsCloud.ts +77 -77
  68. package/src/services/wsdl/WsDFUXRef/v1.02/WsDFUXRef.ts +224 -224
  69. package/src/services/wsdl/WsDali/v1.04/WsDali.ts +216 -216
  70. package/src/services/wsdl/WsDfu/v1.62/WsDfu.ts +1455 -1455
  71. package/src/services/wsdl/WsDfu/v1.63/WsDfu.ts +1465 -1465
  72. package/src/services/wsdl/WsDfu/v1.65/WsDfu.ts +1244 -1244
  73. package/src/services/wsdl/WsFileIO/v1.01/WsFileIO.ts +107 -107
  74. package/src/services/wsdl/WsPackageProcess/v1.04/WsPackageProcess.ts +519 -519
  75. package/src/services/wsdl/WsResources/v1.01/WsResources.ts +119 -119
  76. package/src/services/wsdl/WsSMC/v1.24/WsSMC.ts +665 -665
  77. package/src/services/wsdl/WsSMC/v1.27/WsSMC.ts +670 -670
  78. package/src/services/wsdl/WsTopology/v1.31/WsTopology.ts +856 -856
  79. package/src/services/wsdl/WsTopology/v1.32/WsTopology.ts +885 -885
  80. package/src/services/wsdl/WsWorkunits/v1.88/WsWorkunits.ts +2944 -2944
  81. package/src/services/wsdl/WsWorkunits/v1.94/WsWorkunits.ts +3072 -3072
  82. package/src/services/wsdl/WsWorkunits/v1.95/WsWorkunits.ts +3073 -3073
  83. package/src/services/wsdl/WsWorkunits/v1.97/WsWorkunits.ts +3134 -3134
  84. package/src/services/wsdl/WsWorkunits/v1.98/WsWorkunits.ts +3182 -3182
  85. package/src/services/wsdl/WsWorkunits/v1.99/WsWorkunits.ts +3162 -3162
  86. package/src/services/wsdl/WsWorkunits/v2/WsWorkunits.ts +3153 -3153
  87. package/src/services/wsdl/WsWorkunits/v2.02/WsWorkunits.ts +3157 -3157
  88. package/src/services/wsdl/ws_access/v1.16/ws_access.ts +1086 -1086
  89. package/src/services/wsdl/ws_access/v1.17/ws_access.ts +1023 -1023
  90. package/src/services/wsdl/ws_account/v1.05/ws_account.ts +111 -111
  91. package/src/services/wsdl/ws_account/v1.06/ws_account.ts +109 -109
  92. package/src/services/wsdl/ws_codesign/v1.1/ws_codesign.ts +100 -100
  93. package/src/services/wsdl/ws_elk/v1/ws_elk.ts +47 -47
  94. package/src/services/wsdl/ws_logaccess/v1/ws_logaccess.ts +83 -83
  95. package/src/services/wsdl/ws_logaccess/v1.02/ws_logaccess.ts +161 -161
  96. package/src/services/wsdl/ws_logaccess/v1.03/ws_logaccess.ts +190 -190
  97. package/src/services/wsdl/ws_logaccess/v1.04/ws_logaccess.ts +215 -215
  98. package/src/services/wsdl/ws_logaccess/v1.05/ws_logaccess.ts +219 -219
  99. package/src/services/wsdl/ws_logaccess/v1.08/ws_logaccess.ts +267 -267
  100. package/src/services/wsdl/ws_machine/v1.17/ws_machine.ts +567 -567
  101. package/src/services/wsdl/wsstore/v1.02/wsstore.ts +250 -250
  102. package/types/__package__.d.ts +2 -2
  103. package/types-3.4/__package__.d.ts +2 -2
package/src/connection.ts CHANGED
@@ -1,295 +1,295 @@
1
- import { join, promiseTimeout, scopedLogger } from "@hpcc-js/util";
2
-
3
- const logger = scopedLogger("comms/connection.ts");
4
-
5
- export type RequestType = "post" | "get" | "jsonp";
6
- export type ResponseType = "json" | "text";
7
-
8
- export type IOptionsSend = (options: IOptions, action: string, request: any, responseType: ResponseType, defaultSend: SendFunc, header?: any) => Promise<any>;
9
- export interface IOptions {
10
- baseUrl: string;
11
- type?: RequestType;
12
- userID?: string;
13
- password?: string;
14
- rejectUnauthorized?: boolean;
15
- timeoutSecs?: number;
16
- hookSend?: IOptionsSend;
17
- encodeRequest?: boolean; // defaults to true
18
- }
19
- export function instanceOfIOptions(object: any): object is IOptions {
20
- return "baseUrl" in object;
21
- }
22
-
23
- const DefaultOptions: IOptions = {
24
- type: "post",
25
- baseUrl: "",
26
- userID: "",
27
- password: "",
28
- rejectUnauthorized: true,
29
- timeoutSecs: 60
30
- };
31
-
32
- export interface IConnection {
33
- opts(_: Partial<IOptions>): this;
34
- opts(): IOptions;
35
- baseUrl: string;
36
-
37
- send(action: string, request: any, responseType?: ResponseType): Promise<any>;
38
- clone(): IConnection;
39
- }
40
- export function instanceOfIConnection(object: any): object is IConnection {
41
- return typeof object.opts === "function" &&
42
- typeof object.send === "function" &&
43
- typeof object.clone === "function";
44
- }
45
-
46
- // comms ---
47
-
48
- function encode(uriComponent: string | number | boolean, encodeRequest: boolean): string {
49
- return (encodeRequest === undefined || encodeRequest === true) ? encodeURIComponent(uriComponent) : "" + uriComponent;
50
- }
51
-
52
- export function serializeRequest(obj: any, encodeRequest: boolean = true, prefix: string = ""): string {
53
- if (prefix) {
54
- prefix += ".";
55
- }
56
- if (typeof obj !== "object") {
57
- return encode(obj, encodeRequest);
58
- }
59
-
60
- const str: string[] = [];
61
- for (const key in obj) {
62
- if (obj.hasOwnProperty(key)) {
63
- if (obj[key] instanceof Array) {
64
- // Specific to ESP - but no REST standard exists...
65
- let includeItemCount = false;
66
- obj[key].forEach((row: any, i: number) => {
67
- if (typeof row === "object") {
68
- includeItemCount = true;
69
- str.push(serializeRequest(row, encodeRequest, prefix + encode(`${key}.${i}`, encodeRequest)));
70
- } else {
71
- str.push(prefix + encode(`${key}_i${i}`, encodeRequest) + "=" + serializeRequest(row, encodeRequest));
72
- }
73
- });
74
- if (includeItemCount) {
75
- str.push(prefix + encode(`${key}.itemcount`, encodeRequest) + "=" + obj[key].length);
76
- }
77
- } else if (typeof obj[key] === "object") {
78
- if (obj[key] && obj[key]["Item"] instanceof Array) { // Specific to ws_machine.GetTargetClusterInfo?
79
- str.push(serializeRequest(obj[key]["Item"], encodeRequest, prefix + encode(key, encodeRequest)));
80
- str.push(prefix + encode(`${key}.itemcount`, encodeRequest) + "=" + obj[key]["Item"].length);
81
- } else {
82
- str.push(serializeRequest(obj[key], encodeRequest, prefix + encode(key, encodeRequest)));
83
- }
84
- } else if (obj[key] !== undefined) {
85
- str.push(prefix + encode(key, encodeRequest) + "=" + encode(obj[key], encodeRequest));
86
- } else {
87
- str.push(prefix + encode(key, encodeRequest));
88
- }
89
- }
90
- }
91
- return str.join("&");
92
- }
93
-
94
- export function deserializeResponse(body: string) {
95
- return JSON.parse(body);
96
- }
97
-
98
- export function jsonp(opts: IOptions, action: string, request: any = {}, responseType: ResponseType = "json", header?: any): Promise<any> {
99
- if (header) {
100
- console.warn("Header attributes ignored for JSONP connections");
101
- }
102
- return new Promise<any>((resolve, reject) => {
103
- let respondedTimeout = opts.timeoutSecs! * 1000;
104
- const respondedTick = 5000;
105
- const callbackName = "jsonp_callback_" + Math.round(Math.random() * 999999);
106
- (window as any)[callbackName] = function (response: any) {
107
- respondedTimeout = 0;
108
- doCallback();
109
- resolve(responseType === "json" && typeof response === "string" ? deserializeResponse(response) : response);
110
- };
111
- const script = document.createElement("script");
112
- let url = join(opts.baseUrl, action);
113
- url += url.indexOf("?") >= 0 ? "&" : "?";
114
- script.src = url + "jsonp=" + callbackName + "&" + serializeRequest(request, opts.encodeRequest);
115
- document.body.appendChild(script);
116
- const progress = setInterval(function () {
117
- if (respondedTimeout <= 0) {
118
- clearInterval(progress);
119
- } else {
120
- respondedTimeout -= respondedTick;
121
- if (respondedTimeout <= 0) {
122
- clearInterval(progress);
123
- logger.error("Request timeout: " + script.src);
124
- doCallback();
125
- reject(Error("Request timeout: " + script.src));
126
- } else {
127
- logger.debug("Request pending (" + respondedTimeout / 1000 + " sec): " + script.src);
128
- }
129
- }
130
- }, respondedTick);
131
-
132
- function doCallback() {
133
- delete (window as any)[callbackName];
134
- document.body.removeChild(script);
135
- }
136
- });
137
- }
138
-
139
- function authHeader(opts: IOptions): object {
140
- return opts.userID ? { Authorization: `Basic ${btoa(`${opts.userID}:${opts.password}`)}` } : {};
141
- }
142
-
143
- // _omitMap is a workaround for older HPCC-Platform instances without credentials ---
144
- const _omitMap: { [baseUrl: string]: boolean } = {};
145
- function doFetch(opts: IOptions, action: string, requestInit: RequestInit, headersInit: HeadersInit, responseType: string) {
146
- headersInit = {
147
- ...authHeader(opts),
148
- ...headersInit
149
- };
150
-
151
- requestInit = {
152
- credentials: _omitMap[opts.baseUrl] ? "omit" : "include",
153
- ...requestInit,
154
- headers: headersInit
155
- };
156
-
157
- if (fetch["__setGlobalDispatcher"]) {
158
- fetch["__setGlobalDispatcher"](fetch["__defaultAgent"]);
159
- }
160
-
161
- if (opts.baseUrl.indexOf("https:") === 0) {
162
- // NodeJS / node-fetch only ---
163
- if (opts.rejectUnauthorized === false && fetch["__rejectUnauthorizedAgent"]) {
164
- if (fetch["__setGlobalDispatcher"]) {
165
- fetch["__setGlobalDispatcher"](fetch["__rejectUnauthorizedAgent"]);
166
- } else {
167
- requestInit["agent"] = fetch["__rejectUnauthorizedAgent"];
168
- }
169
- } else if (fetch["__trustwaveAgent"]) {
170
- requestInit["agent"] = fetch["__trustwaveAgent"];
171
- }
172
- }
173
-
174
- function handleResponse(response: Response): Promise<any> {
175
- if (response.ok) {
176
- return responseType === "json" ? response.json() : response.text();
177
- }
178
- throw new Error(response.statusText);
179
- }
180
-
181
- return promiseTimeout(opts.timeoutSecs! * 1000, fetch(join(opts.baseUrl, action), requestInit)
182
- .then(handleResponse)
183
- .catch(e => {
184
- // Try again with the opposite credentials mode ---
185
- requestInit.credentials = !_omitMap[opts.baseUrl] ? "omit" : "include";
186
- return fetch(join(opts.baseUrl, action), requestInit)
187
- .then(handleResponse)
188
- .then(responseBody => {
189
- _omitMap[opts.baseUrl] = !_omitMap[opts.baseUrl]; // The "opposite" credentials mode is known to work ---
190
- return responseBody;
191
- });
192
- })
193
- );
194
- }
195
-
196
- export function post(opts: IOptions, action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
197
- if (request.upload_) {
198
- delete request.upload_;
199
- action += "?upload_";
200
- }
201
- let abortSignal;
202
- if (request.abortSignal_) {
203
- abortSignal = request.abortSignal_;
204
- delete request.abortSignal_;
205
- }
206
- return doFetch(opts, action, {
207
- method: "post",
208
- body: serializeRequest(request, opts.encodeRequest),
209
- signal: abortSignal
210
- }, {
211
- "Content-Type": "application/x-www-form-urlencoded",
212
- ...header
213
- } as any, responseType);
214
- }
215
-
216
- export function get(opts: IOptions, action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
217
- let abortSignal;
218
- if (request.abortSignal_) {
219
- abortSignal = request.abortSignal_;
220
- delete request.abortSignal_;
221
- }
222
- return doFetch(opts, `${action}?${serializeRequest(request, opts.encodeRequest)}`, {
223
- method: "get",
224
- signal: abortSignal
225
- }, {
226
- ...header
227
- } as any, responseType);
228
- }
229
-
230
- export type SendFunc = (opts: IOptions, action: string, request: any, responseType: ResponseType, header?: any) => Promise<any>;
231
- export function send(opts: IOptions, action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
232
- let retVal: Promise<any>;
233
- switch (opts.type) {
234
- case "jsonp":
235
- retVal = jsonp(opts, action, request, responseType, header);
236
- break;
237
- case "get":
238
- retVal = get(opts, action, request, responseType, header);
239
- break;
240
- case "post":
241
- default:
242
- retVal = post(opts, action, request, responseType, header);
243
- break;
244
- }
245
- return retVal;
246
- }
247
-
248
- let hookedSend: SendFunc = send;
249
- export function hookSend(newSend?: SendFunc): SendFunc {
250
- const retVal = hookedSend;
251
- if (newSend) {
252
- hookedSend = newSend;
253
- }
254
- return retVal;
255
- }
256
-
257
- export class Connection implements IConnection {
258
- protected _opts: IOptions;
259
- get baseUrl() { return this._opts.baseUrl; }
260
-
261
- constructor(opts: IOptions) {
262
- this.opts(opts);
263
- }
264
-
265
- // IConnection ---
266
- opts(_: Partial<IOptions>): this;
267
- opts(): IOptions;
268
- opts(_?: Partial<IOptions>): this | IOptions {
269
- if (arguments.length === 0) return this._opts;
270
- this._opts = { ...DefaultOptions, ..._ };
271
- return this;
272
- }
273
-
274
- send(action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
275
- if (this._opts.hookSend) {
276
- return this._opts.hookSend(this._opts, action, request, responseType, hookedSend, header);
277
- }
278
- return hookedSend(this._opts, action, request, responseType, header);
279
- }
280
-
281
- clone() {
282
- return new Connection(this.opts());
283
- }
284
- }
285
-
286
- export type IConnectionFactory = (opts: IOptions) => IConnection;
287
- export let createConnection: IConnectionFactory = function (opts: IOptions): IConnection {
288
- return new Connection(opts);
289
- };
290
-
291
- export function setTransportFactory(newFunc: IConnectionFactory): IConnectionFactory {
292
- const retVal = createConnection;
293
- createConnection = newFunc;
294
- return retVal;
295
- }
1
+ import { join, promiseTimeout, scopedLogger } from "@hpcc-js/util";
2
+
3
+ const logger = scopedLogger("comms/connection.ts");
4
+
5
+ export type RequestType = "post" | "get" | "jsonp";
6
+ export type ResponseType = "json" | "text";
7
+
8
+ export type IOptionsSend = (options: IOptions, action: string, request: any, responseType: ResponseType, defaultSend: SendFunc, header?: any) => Promise<any>;
9
+ export interface IOptions {
10
+ baseUrl: string;
11
+ type?: RequestType;
12
+ userID?: string;
13
+ password?: string;
14
+ rejectUnauthorized?: boolean;
15
+ timeoutSecs?: number;
16
+ hookSend?: IOptionsSend;
17
+ encodeRequest?: boolean; // defaults to true
18
+ }
19
+ export function instanceOfIOptions(object: any): object is IOptions {
20
+ return "baseUrl" in object;
21
+ }
22
+
23
+ const DefaultOptions: IOptions = {
24
+ type: "post",
25
+ baseUrl: "",
26
+ userID: "",
27
+ password: "",
28
+ rejectUnauthorized: true,
29
+ timeoutSecs: 60
30
+ };
31
+
32
+ export interface IConnection {
33
+ opts(_: Partial<IOptions>): this;
34
+ opts(): IOptions;
35
+ baseUrl: string;
36
+
37
+ send(action: string, request: any, responseType?: ResponseType): Promise<any>;
38
+ clone(): IConnection;
39
+ }
40
+ export function instanceOfIConnection(object: any): object is IConnection {
41
+ return typeof object.opts === "function" &&
42
+ typeof object.send === "function" &&
43
+ typeof object.clone === "function";
44
+ }
45
+
46
+ // comms ---
47
+
48
+ function encode(uriComponent: string | number | boolean, encodeRequest: boolean): string {
49
+ return (encodeRequest === undefined || encodeRequest === true) ? encodeURIComponent(uriComponent) : "" + uriComponent;
50
+ }
51
+
52
+ export function serializeRequest(obj: any, encodeRequest: boolean = true, prefix: string = ""): string {
53
+ if (prefix) {
54
+ prefix += ".";
55
+ }
56
+ if (typeof obj !== "object") {
57
+ return encode(obj, encodeRequest);
58
+ }
59
+
60
+ const str: string[] = [];
61
+ for (const key in obj) {
62
+ if (obj.hasOwnProperty(key)) {
63
+ if (obj[key] instanceof Array) {
64
+ // Specific to ESP - but no REST standard exists...
65
+ let includeItemCount = false;
66
+ obj[key].forEach((row: any, i: number) => {
67
+ if (typeof row === "object") {
68
+ includeItemCount = true;
69
+ str.push(serializeRequest(row, encodeRequest, prefix + encode(`${key}.${i}`, encodeRequest)));
70
+ } else {
71
+ str.push(prefix + encode(`${key}_i${i}`, encodeRequest) + "=" + serializeRequest(row, encodeRequest));
72
+ }
73
+ });
74
+ if (includeItemCount) {
75
+ str.push(prefix + encode(`${key}.itemcount`, encodeRequest) + "=" + obj[key].length);
76
+ }
77
+ } else if (typeof obj[key] === "object") {
78
+ if (obj[key] && obj[key]["Item"] instanceof Array) { // Specific to ws_machine.GetTargetClusterInfo?
79
+ str.push(serializeRequest(obj[key]["Item"], encodeRequest, prefix + encode(key, encodeRequest)));
80
+ str.push(prefix + encode(`${key}.itemcount`, encodeRequest) + "=" + obj[key]["Item"].length);
81
+ } else {
82
+ str.push(serializeRequest(obj[key], encodeRequest, prefix + encode(key, encodeRequest)));
83
+ }
84
+ } else if (obj[key] !== undefined) {
85
+ str.push(prefix + encode(key, encodeRequest) + "=" + encode(obj[key], encodeRequest));
86
+ } else {
87
+ str.push(prefix + encode(key, encodeRequest));
88
+ }
89
+ }
90
+ }
91
+ return str.join("&");
92
+ }
93
+
94
+ export function deserializeResponse(body: string) {
95
+ return JSON.parse(body);
96
+ }
97
+
98
+ export function jsonp(opts: IOptions, action: string, request: any = {}, responseType: ResponseType = "json", header?: any): Promise<any> {
99
+ if (header) {
100
+ console.warn("Header attributes ignored for JSONP connections");
101
+ }
102
+ return new Promise<any>((resolve, reject) => {
103
+ let respondedTimeout = opts.timeoutSecs! * 1000;
104
+ const respondedTick = 5000;
105
+ const callbackName = "jsonp_callback_" + Math.round(Math.random() * 999999);
106
+ (window as any)[callbackName] = function (response: any) {
107
+ respondedTimeout = 0;
108
+ doCallback();
109
+ resolve(responseType === "json" && typeof response === "string" ? deserializeResponse(response) : response);
110
+ };
111
+ const script = document.createElement("script");
112
+ let url = join(opts.baseUrl, action);
113
+ url += url.indexOf("?") >= 0 ? "&" : "?";
114
+ script.src = url + "jsonp=" + callbackName + "&" + serializeRequest(request, opts.encodeRequest);
115
+ document.body.appendChild(script);
116
+ const progress = setInterval(function () {
117
+ if (respondedTimeout <= 0) {
118
+ clearInterval(progress);
119
+ } else {
120
+ respondedTimeout -= respondedTick;
121
+ if (respondedTimeout <= 0) {
122
+ clearInterval(progress);
123
+ logger.error("Request timeout: " + script.src);
124
+ doCallback();
125
+ reject(Error("Request timeout: " + script.src));
126
+ } else {
127
+ logger.debug("Request pending (" + respondedTimeout / 1000 + " sec): " + script.src);
128
+ }
129
+ }
130
+ }, respondedTick);
131
+
132
+ function doCallback() {
133
+ delete (window as any)[callbackName];
134
+ document.body.removeChild(script);
135
+ }
136
+ });
137
+ }
138
+
139
+ function authHeader(opts: IOptions): object {
140
+ return opts.userID ? { Authorization: `Basic ${btoa(`${opts.userID}:${opts.password}`)}` } : {};
141
+ }
142
+
143
+ // _omitMap is a workaround for older HPCC-Platform instances without credentials ---
144
+ const _omitMap: { [baseUrl: string]: boolean } = {};
145
+ function doFetch(opts: IOptions, action: string, requestInit: RequestInit, headersInit: HeadersInit, responseType: string) {
146
+ headersInit = {
147
+ ...authHeader(opts),
148
+ ...headersInit
149
+ };
150
+
151
+ requestInit = {
152
+ credentials: _omitMap[opts.baseUrl] ? "omit" : "include",
153
+ ...requestInit,
154
+ headers: headersInit
155
+ };
156
+
157
+ if (fetch["__setGlobalDispatcher"]) {
158
+ fetch["__setGlobalDispatcher"](fetch["__defaultAgent"]);
159
+ }
160
+
161
+ if (opts.baseUrl.indexOf("https:") === 0) {
162
+ // NodeJS / node-fetch only ---
163
+ if (opts.rejectUnauthorized === false && fetch["__rejectUnauthorizedAgent"]) {
164
+ if (fetch["__setGlobalDispatcher"]) {
165
+ fetch["__setGlobalDispatcher"](fetch["__rejectUnauthorizedAgent"]);
166
+ } else {
167
+ requestInit["agent"] = fetch["__rejectUnauthorizedAgent"];
168
+ }
169
+ } else if (fetch["__trustwaveAgent"]) {
170
+ requestInit["agent"] = fetch["__trustwaveAgent"];
171
+ }
172
+ }
173
+
174
+ function handleResponse(response: Response): Promise<any> {
175
+ if (response.ok) {
176
+ return responseType === "json" ? response.json() : response.text();
177
+ }
178
+ throw new Error(response.statusText);
179
+ }
180
+
181
+ return promiseTimeout(opts.timeoutSecs! * 1000, fetch(join(opts.baseUrl, action), requestInit)
182
+ .then(handleResponse)
183
+ .catch(e => {
184
+ // Try again with the opposite credentials mode ---
185
+ requestInit.credentials = !_omitMap[opts.baseUrl] ? "omit" : "include";
186
+ return fetch(join(opts.baseUrl, action), requestInit)
187
+ .then(handleResponse)
188
+ .then(responseBody => {
189
+ _omitMap[opts.baseUrl] = !_omitMap[opts.baseUrl]; // The "opposite" credentials mode is known to work ---
190
+ return responseBody;
191
+ });
192
+ })
193
+ );
194
+ }
195
+
196
+ export function post(opts: IOptions, action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
197
+ if (request.upload_) {
198
+ delete request.upload_;
199
+ action += "?upload_";
200
+ }
201
+ let abortSignal;
202
+ if (request.abortSignal_) {
203
+ abortSignal = request.abortSignal_;
204
+ delete request.abortSignal_;
205
+ }
206
+ return doFetch(opts, action, {
207
+ method: "post",
208
+ body: serializeRequest(request, opts.encodeRequest),
209
+ signal: abortSignal
210
+ }, {
211
+ "Content-Type": "application/x-www-form-urlencoded",
212
+ ...header
213
+ } as any, responseType);
214
+ }
215
+
216
+ export function get(opts: IOptions, action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
217
+ let abortSignal;
218
+ if (request.abortSignal_) {
219
+ abortSignal = request.abortSignal_;
220
+ delete request.abortSignal_;
221
+ }
222
+ return doFetch(opts, `${action}?${serializeRequest(request, opts.encodeRequest)}`, {
223
+ method: "get",
224
+ signal: abortSignal
225
+ }, {
226
+ ...header
227
+ } as any, responseType);
228
+ }
229
+
230
+ export type SendFunc = (opts: IOptions, action: string, request: any, responseType: ResponseType, header?: any) => Promise<any>;
231
+ export function send(opts: IOptions, action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
232
+ let retVal: Promise<any>;
233
+ switch (opts.type) {
234
+ case "jsonp":
235
+ retVal = jsonp(opts, action, request, responseType, header);
236
+ break;
237
+ case "get":
238
+ retVal = get(opts, action, request, responseType, header);
239
+ break;
240
+ case "post":
241
+ default:
242
+ retVal = post(opts, action, request, responseType, header);
243
+ break;
244
+ }
245
+ return retVal;
246
+ }
247
+
248
+ let hookedSend: SendFunc = send;
249
+ export function hookSend(newSend?: SendFunc): SendFunc {
250
+ const retVal = hookedSend;
251
+ if (newSend) {
252
+ hookedSend = newSend;
253
+ }
254
+ return retVal;
255
+ }
256
+
257
+ export class Connection implements IConnection {
258
+ protected _opts: IOptions;
259
+ get baseUrl() { return this._opts.baseUrl; }
260
+
261
+ constructor(opts: IOptions) {
262
+ this.opts(opts);
263
+ }
264
+
265
+ // IConnection ---
266
+ opts(_: Partial<IOptions>): this;
267
+ opts(): IOptions;
268
+ opts(_?: Partial<IOptions>): this | IOptions {
269
+ if (arguments.length === 0) return this._opts;
270
+ this._opts = { ...DefaultOptions, ..._ };
271
+ return this;
272
+ }
273
+
274
+ send(action: string, request: any, responseType: ResponseType = "json", header?: any): Promise<any> {
275
+ if (this._opts.hookSend) {
276
+ return this._opts.hookSend(this._opts, action, request, responseType, hookedSend, header);
277
+ }
278
+ return hookedSend(this._opts, action, request, responseType, header);
279
+ }
280
+
281
+ clone() {
282
+ return new Connection(this.opts());
283
+ }
284
+ }
285
+
286
+ export type IConnectionFactory = (opts: IOptions) => IConnection;
287
+ export let createConnection: IConnectionFactory = function (opts: IOptions): IConnection {
288
+ return new Connection(opts);
289
+ };
290
+
291
+ export function setTransportFactory(newFunc: IConnectionFactory): IConnectionFactory {
292
+ const retVal = createConnection;
293
+ createConnection = newFunc;
294
+ return retVal;
295
+ }