@tiledesk/tiledesk-tybot-connector 2.0.29-rc4 → 2.0.29

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.
@@ -0,0 +1,418 @@
1
+ let axios = require('axios');
2
+ let https = require("https");
3
+ const { Filler } = require('../Filler');
4
+ const { TiledeskChatbot } = require('../../engine/TiledeskChatbot');
5
+ const { DirIntent } = require('./DirIntent');
6
+
7
+ class DirWebRequestV2 {
8
+ constructor(context) {
9
+ if (!context) {
10
+ throw new Error('context object is mandatory.');
11
+ }
12
+ this.context = context;
13
+ this.tdcache = context.tdcache;
14
+ this.requestId = context.requestId;
15
+ this.chatbot = context.chatbot;
16
+ this.log = context.log;
17
+
18
+ this.intentDir = new DirIntent(context);
19
+ }
20
+
21
+ execute(directive, callback) {
22
+ let action;
23
+ if (directive.action) {
24
+ action = directive.action;
25
+ }
26
+ else {
27
+ console.error("Incorrect directive:", JSON.stringify(directive));
28
+ callback();
29
+ return;
30
+ }
31
+ this.go(action, (stop) => {
32
+ if (this.log) {console.log("(webrequestv2, stop?", stop); }
33
+ callback(stop);
34
+ });
35
+ }
36
+
37
+ async go(action, callback) {
38
+ if (this.log) {console.log("webRequest action:", JSON.stringify(action));}
39
+
40
+ // Condition branches
41
+ let trueIntent = action.trueIntent;
42
+ let falseIntent = action.falseIntent;
43
+ const trueIntentAttributes = action.trueIntentAttributes;
44
+ const falseIntentAttributes = action.falseIntentAttributes;
45
+ let stopOnConditionMet = action.stopOnConditionMet;
46
+ if (trueIntent && trueIntent.trim() === "") {
47
+ trueIntent = null;
48
+ }
49
+ if (falseIntent && falseIntent.trim() === "") {
50
+ falseIntent = null;
51
+ }
52
+
53
+ let requestAttributes = null;
54
+ if (this.tdcache) {
55
+ requestAttributes =
56
+ await TiledeskChatbot.allParametersStatic(
57
+ this.tdcache, this.requestId
58
+ );
59
+ }
60
+ const filler = new Filler();
61
+ const url = filler.fill(action.url, requestAttributes);
62
+
63
+ let headers = {};
64
+ if (action.headersString) {
65
+ let headersDict = action.headersString
66
+ for (const [key, value] of Object.entries(headersDict)) {
67
+ if (this.log) {console.log("header:", key, "value:", value)}
68
+ let filled_value = filler.fill(value, requestAttributes);
69
+ headers[key] = filled_value;
70
+ }
71
+ }
72
+
73
+ let json = null;
74
+ try {
75
+ if (action.jsonBody && action.bodyType == "json") {
76
+ if (this.log) {console.log("action.body is:", action.jsonBody);}
77
+ let jsonBody = filler.fill(action.jsonBody, requestAttributes);
78
+ try {
79
+ json = JSON.parse(jsonBody);
80
+ if (this.log) {console.log("json is:", json);}
81
+ }
82
+ catch(err) {
83
+ console.error("Error parsing webRequest jsonBody:", jsonBody);
84
+ if (callback) {
85
+ if (falseIntent) {
86
+ await this.chatbot.addParameter("flowError", "Error parsing jsonBody");
87
+ this.#executeCondition(false, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes, () => {
88
+ console.log('herrrrr 11111' )
89
+ callback(true); // stop the flow
90
+ return;
91
+ });
92
+ }
93
+ else {
94
+ console.log('herrrrr 2222' )
95
+ callback(false); // continue the flow
96
+ return;
97
+ }
98
+ }
99
+ }
100
+ }
101
+ else if (action.formData && action.bodyType == "form-data") {
102
+ let formData = filler.fill(action.formData, requestAttributes);
103
+ if (this.log) {console.log("action.body is form-data:", formData);}
104
+ // // fill
105
+ if (formData && formData.length > 0) {
106
+ for (let i = 0; i < formData.length; i++) {
107
+ let field = formData[i];
108
+ if (field.value) {
109
+ field.value = filler.fill(field.value, requestAttributes);
110
+ if (this.log) {console.log("field filled:", field.value);}
111
+ }
112
+ }
113
+ }
114
+ json = {};
115
+ for (let i = 0; i < formData.length; i++) {
116
+ let field = formData[i];
117
+ if (field.enabled && field.value && field.type === "URL") {
118
+ if (this.log) {console.log("Getting file:", field.value);}
119
+ let response = await axios.get(field.value,
120
+ {
121
+ responseType: 'stream'
122
+ }
123
+ );
124
+ let stream = response.data;
125
+ // if (this.log) {console.log("Stream data:", stream);}
126
+ json[field.name] = stream;
127
+ // process.exit(0);
128
+ }
129
+ else if (field.enabled && field.value && field.type === "Text") {
130
+ json[field.name] = field.value;
131
+ }
132
+ }
133
+ if (this.log) {console.log("final json:", json);}
134
+ }
135
+ else {
136
+ if (this.log) {console.log("no action upload parts");}
137
+ }
138
+
139
+ }
140
+ catch(error) {
141
+ console.error("Error", error);
142
+ if (callback) {
143
+ if (falseIntent) {
144
+ await this.chatbot.addParameter("flowError", "Error: " + error);
145
+ this.#executeCondition(false, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes, () => {
146
+ callback(true); // stop the flow
147
+ return;
148
+ });
149
+ }
150
+ else {
151
+ callback(false); // continue the flow
152
+ return;
153
+ }
154
+ }
155
+ }
156
+
157
+
158
+
159
+ let timeout = this.#webrequest_timeout(action, 20000, 1, 300000);
160
+
161
+ if (this.log) {console.log("webRequest URL", url);}
162
+
163
+ const HTTPREQUEST = {
164
+ url: url,
165
+ headers: headers,
166
+ json: json,
167
+ method: action.method,
168
+ timeout: timeout
169
+ };
170
+
171
+ if (this.log) {console.log("webRequest HTTPREQUEST", HTTPREQUEST);}
172
+ this.#myrequest(
173
+ HTTPREQUEST, async (err, res) => {
174
+ if (this.log && err) {
175
+ console.log("webRequest error:", err);
176
+ }
177
+ if (this.log) {console.log("got res:", res);}
178
+ let resbody = res.data;
179
+ let status = res.status;
180
+ let error = res.error;
181
+ await this.#assignAttributes(action, resbody, status, error)
182
+ if (this.log) {console.log("webRequest resbody:", resbody);}
183
+ if (err) {
184
+ if (this.log) {console.error("webRequest error:", err);}
185
+ if (callback) {
186
+ if (falseIntent) {
187
+ this.#executeCondition(false, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes, () => {
188
+ callback(true); // stop the flow
189
+ });
190
+ }
191
+ else {
192
+ callback(false); // continue the flow
193
+ }
194
+ }
195
+ }
196
+ else if(res.status >= 200 && res.status <= 299) {
197
+ if (trueIntent) {
198
+ await this.#executeCondition(true, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes, () => {
199
+ callback(true); // stop the flow
200
+ });
201
+ }
202
+ else {
203
+ callback(false); // continue the flow
204
+ }
205
+ }
206
+ else {
207
+ if (falseIntent) {
208
+ this.#executeCondition(false, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes, () => {
209
+ callback(true); // stop the flow
210
+ });
211
+ }
212
+ else {
213
+ callback(false); // continue the flow
214
+ }
215
+ }
216
+ }
217
+ );
218
+ }
219
+
220
+ async #executeCondition(result, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes, callback) {
221
+ let trueIntentDirective = null;
222
+ if (trueIntent) {
223
+ trueIntentDirective = DirIntent.intentDirectiveFor(trueIntent, trueIntentAttributes);
224
+ }
225
+ let falseIntentDirective = null;
226
+ if (falseIntent) {
227
+ falseIntentDirective = DirIntent.intentDirectiveFor(falseIntent, falseIntentAttributes);
228
+ }
229
+ if (result === true) {
230
+ if (trueIntentDirective) {
231
+ this.intentDir.execute(trueIntentDirective, () => {
232
+ callback();
233
+ });
234
+ }
235
+ else {
236
+ if (this.log) {console.log("No trueIntentDirective specified");}
237
+ callback();
238
+ }
239
+ }
240
+ else {
241
+ if (falseIntentDirective) {
242
+ this.intentDir.execute(falseIntentDirective, () => {
243
+ callback();
244
+ });
245
+ }
246
+ else {
247
+ if (this.log) {console.log("No falseIntentDirective specified");}
248
+ callback();
249
+ }
250
+ }
251
+ }
252
+
253
+ async #assignAttributes(action, resbody, status, error) {
254
+ if (this.log) {
255
+ console.log("assignAttributes resbody:", resbody)
256
+ console.log("assignAttributes error:", error)
257
+ console.log("assignAttributes status:", status)
258
+ console.log("assignAttributes action:", action)
259
+ }
260
+ if (this.context.tdcache) {
261
+ if (action.assignResultTo && resbody) {
262
+ if (this.log) {console.log("assign assignResultTo:", resbody);}
263
+ await TiledeskChatbot.addParameterStatic(this.context.tdcache, this.context.requestId, action.assignResultTo, resbody);
264
+ }
265
+ if (action.assignErrorTo && error) {
266
+ if (this.log) {console.log("assign assignResultTo:", error);}
267
+ await TiledeskChatbot.addParameterStatic(this.context.tdcache, this.context.requestId, action.assignErrorTo, error);
268
+ }
269
+ if (action.assignStatusTo && status) {
270
+ if (this.log) {console.log("assign assignStatusTo:", status);}
271
+ await TiledeskChatbot.addParameterStatic(this.context.tdcache, this.context.requestId, action.assignStatusTo, status);
272
+ }
273
+ // Debug log
274
+ if (this.log) {
275
+ const all_parameters = await TiledeskChatbot.allParametersStatic(this.context.tdcache, this.context.requestId);
276
+ for (const [key, value] of Object.entries(all_parameters)) {
277
+ if (this.log) {console.log("(webRequest) request parameter:", key, "value:", value, "type:", typeof value)}
278
+ }
279
+ }
280
+ }
281
+ }
282
+
283
+ #myrequest(options, callback) {
284
+ try {
285
+ if (this.log) {
286
+ console.log("API URL:", options.url);
287
+ //console.log("** Options:", JSON.stringify(options));
288
+ // Stringify "options". FIX THE STRINGIFY OF CIRCULAR STRUCTURE BUG - START
289
+ let cache = [];
290
+ let str_Options = JSON.stringify(options, function(key, value) { // try to use a separate function
291
+ if (typeof value === 'object' && value != null) {
292
+ if (cache.indexOf(value) !== -1) {
293
+ return;
294
+ }
295
+ cache.push(value);
296
+ }
297
+ return value;
298
+ });
299
+ console.log("** Options:", str_Options);
300
+ }
301
+ let axios_options = {
302
+ url: options.url,
303
+ method: options.method,
304
+ params: options.params,
305
+ headers: options.headers,
306
+ timeout: options.timeout,
307
+ maxContentLength: 10000000, // max 10mb response size
308
+ maxBodyLength: 10000000 // max 10mb request body size
309
+ }
310
+
311
+ if (options.json !== null) {
312
+ axios_options.data = options.json
313
+ }
314
+ // if (this.log) {
315
+ // console.log("axios_options:", JSON.stringify(axios_options));
316
+ // }
317
+ if (options.url.startsWith("https:")) {
318
+ const httpsAgent = new https.Agent({
319
+ rejectUnauthorized: false,
320
+ });
321
+ axios_options.httpsAgent = httpsAgent;
322
+ }
323
+
324
+ axios(axios_options)
325
+ .then((res) => {
326
+ if (this.log) {
327
+ console.log("Success Response:", res);
328
+ console.log("Response for url:", options.url);
329
+ console.log("Response headers:\n", JSON.stringify(res.headers));
330
+ }
331
+ if (callback) {
332
+ callback(null, res);
333
+ }
334
+ })
335
+ .catch( (err) => {
336
+ if (this.log) {
337
+ if (err.response) {
338
+ console.log("Error Response data:", err.response.data);
339
+ }
340
+ // FIX THE STRINGIFY OF CIRCULAR STRUCTURE BUG - START
341
+ let cache = [];
342
+ let error_log = JSON.stringify(err, function(key, value) { // try to use a separate function
343
+ if (typeof value === 'object' && value != null) {
344
+ if (cache.indexOf(value) !== -1) {
345
+ return;
346
+ }
347
+ cache.push(value);
348
+ }
349
+ return value;
350
+ });
351
+ console.error("An error occurred: ", error_log);
352
+ // FIX THE STRINGIFY OF CIRCULAR STRUCTURE BUG - END
353
+ // console.error("An error occurred:", JSON.stringify(err));
354
+ }
355
+ if (callback) {
356
+ let status = 1000;
357
+ let cache = [];
358
+ let str_error = JSON.stringify(err, function(key, value) { // try to use a separate function
359
+ if (typeof value === 'object' && value != null) {
360
+ if (cache.indexOf(value) !== -1) {
361
+ return;
362
+ }
363
+ cache.push(value);
364
+ }
365
+ return value;
366
+ });
367
+ let error = JSON.parse(str_error) // "status" disappears without this trick
368
+ let errorMessage = JSON.stringify(error);
369
+ if (error.status) {
370
+ status = error.status;
371
+ }
372
+ if (error.message) {
373
+ errorMessage = error.message;
374
+ }
375
+ let data = null;
376
+ if (err.response) {
377
+ data = err.response.data;
378
+ }
379
+ callback(
380
+ null, {
381
+ status: status,
382
+ data: data,
383
+ error: errorMessage
384
+ }
385
+ );
386
+ }
387
+ });
388
+ }
389
+ catch(error) {
390
+ console.error("Error:", error);
391
+ }
392
+ }
393
+
394
+ #webrequest_timeout(action, default_timeout, min, max) {
395
+ let timeout = default_timeout;
396
+ if (!action.settings) {
397
+ return timeout;
398
+ }
399
+ // console.log("default timeout:", timeout);
400
+ // console.log("action.settings:", action.settings);
401
+ // console.log("action.settings.timeout:", action.settings.timeout);
402
+ // console.log("typeof action.settings.timeout:", typeof action.settings.timeout);
403
+ // console.log("action.settings.timeout > min", action.settings.timeout > min)
404
+ // console.log("action.settings.timeout < max", action.settings.timeout < max)
405
+
406
+ if (action.settings.timeout) {
407
+ if ((typeof action.settings.timeout === "number") && action.settings.timeout > min && action.settings.timeout < max) {
408
+ timeout = Math.round(action.settings.timeout)
409
+ // console.log("new timeout:", timeout);
410
+ }
411
+ }
412
+ // console.log("returning timeout:", timeout);
413
+ return timeout
414
+ }
415
+
416
+ }
417
+
418
+ module.exports = { DirWebRequestV2 };
@@ -37,7 +37,6 @@ class Directives {
37
37
  static ASK_GPT_V2 = "askgptv2";
38
38
  static GPT_TASK = "gpt_task";
39
39
  static AI_PROMPT = "ai_prompt";
40
- static AI_CONDITION = "ai_condition";
41
40
  /**** INTEGRATIONS ****/
42
41
  static QAPLA = 'qapla';
43
42
  static MAKE = 'make';
@@ -59,7 +58,6 @@ class Directives {
59
58
  static MOVE_TO_UNASSIGNED = "move_to_unassigned";
60
59
  static CONNECT_BLOCK = "connect_block";
61
60
  static ADD_TAGS = 'add_tags'
62
- static WEBHOOK = 'webhook';
63
61
  static WEB_RESPONSE = "web_response";
64
62
  static FLOW_LOG = "flow_log";
65
63
  static ADD_KB_CONTENT = "add_kb_content";
@@ -844,25 +844,6 @@ class TiledeskChatbotUtil {
844
844
  await chatbot.addParameter(TiledeskChatbotConst.REQ_DEPARTMENT_NAME_KEY, message.attributes.departmentName);
845
845
  }
846
846
 
847
- if (message.attributes) {
848
- const attrFields = [
849
- { key: TiledeskChatbotConst.REQ_EMAIL_SUBJECT, attr: "email_subject" },
850
- { key: TiledeskChatbotConst.REQ_EMAIL_TO, attr: "email_toEmail" },
851
- { key: TiledeskChatbotConst.REQ_EMAIL_FROM, attr: "email_fromEmail" },
852
- { key: TiledeskChatbotConst.REQ_EMAIL_MESSAGE_ID, attr: "email_messageId" },
853
- { key: TiledeskChatbotConst.REQ_EMAIL_REPLY_TO, attr: "email_replyTo" },
854
- { key: TiledeskChatbotConst.REQ_EMAIL_EML, attr: "email_eml" },
855
- { key: TiledeskChatbotConst.REQ_EMAIL_ATTACHMENTS_LINK, attr: "link" },
856
- { key: TiledeskChatbotConst.REQ_EMAIL_ATTACHMENTS_FILES, attr: "attachments" },
857
- // aggiungi qui altri campi se necessario
858
- ];
859
- for (const field of attrFields) {
860
- if (message.attributes[field.attr] !== undefined && message.attributes[field.attr] !== null) {
861
- await chatbot.addParameter(field.key, message.attributes[field.attr]);
862
- }
863
- }
864
- }
865
-
866
847
  if (message && message.request && message.request.attributes && message.request.attributes.payload) {
867
848
  if (!message.attributes) {
868
849
  message.attributes = {}
@@ -1025,20 +1006,6 @@ class TiledeskChatbotUtil {
1025
1006
  return userParams;
1026
1007
  }
1027
1008
 
1028
- static AiConditionPromptBuilder(prompt_header, intents, instructions) {
1029
- let conditions = "";
1030
- intents.forEach( function(intent) {
1031
- conditions += `- label: ${intent.label} When: ${intent.prompt}\n`
1032
- });
1033
-
1034
- instructions = instructions;
1035
- let raw_condition_prompt = `${prompt_header}
1036
-
1037
- ${conditions}
1038
- ${instructions}`
1039
- return raw_condition_prompt;
1040
- }
1041
-
1042
1009
  /**
1043
1010
  * A stub to get the request parameters, hosted by tilebot on:
1044
1011
  * /${TILEBOT_ROUTE}/ext/parameters/requests/${requestId}?all
@@ -1107,8 +1074,6 @@ ${instructions}`
1107
1074
  }
1108
1075
  });
1109
1076
  }
1110
-
1111
-
1112
1077
 
1113
1078
 
1114
1079
  }