@tiledesk/tiledesk-tybot-connector 0.2.59 → 0.2.60-rc1
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.
package/CHANGELOG.md
CHANGED
|
@@ -5,6 +5,14 @@
|
|
|
5
5
|
available on:
|
|
6
6
|
▶️ https://www.npmjs.com/package/@tiledesk/tiledesk-tybot-connector
|
|
7
7
|
|
|
8
|
+
# v0.2.60 (next-release)
|
|
9
|
+
- Added support for DTMF Menu (Reply for voice channel)
|
|
10
|
+
- Added Brevo Action
|
|
11
|
+
|
|
12
|
+
# v0.2.60-rc1
|
|
13
|
+
- Added Brevo Action
|
|
14
|
+
- Added support for DTMF Menu (Reply for voice channel)
|
|
15
|
+
|
|
8
16
|
# v0.2.59
|
|
9
17
|
- added flow attribute "strongAuthentication" mapped on request.requester.isAuthenticated (true|false)
|
|
10
18
|
|
|
@@ -14,7 +22,11 @@ available on:
|
|
|
14
22
|
- added flow attribute "decodedCustomJWT"
|
|
15
23
|
|
|
16
24
|
# v0.2.57
|
|
25
|
+
- Changed env variables for AskGPT actions
|
|
26
|
+
- Added support for temperature, top_k, max_tokens and context for AskGPTV2 action
|
|
27
|
+
- Added support for context with variables for GptTask action
|
|
17
28
|
- fixed catching reply error in index.js: reply = await chatbot.replyToMessage(message);
|
|
29
|
+
- Added support for DTMF Form and Blind Transfer (Reply for voice channel)
|
|
18
30
|
- Added context and top_k parameter in DirAskGPTV2
|
|
19
31
|
|
|
20
32
|
# v0.2.56
|
package/package.json
CHANGED
|
@@ -44,6 +44,7 @@ const { DirMake } = require('./directives/DirMake');
|
|
|
44
44
|
const { DirReplaceBotV2 } = require('./directives/DirReplaceBotV2');
|
|
45
45
|
const { DirHubspot } = require('./directives/DirHubspot');
|
|
46
46
|
const { DirCustomerio } = require('./directives/DirCustomerio');
|
|
47
|
+
const { DirBrevo } = require('./directives/DirBrevo');
|
|
47
48
|
const { DirAskGPTV2 } = require('./directives/DirAskGPTV2');
|
|
48
49
|
|
|
49
50
|
class DirectivesChatbotPlug {
|
|
@@ -291,6 +292,27 @@ class DirectivesChatbotPlug {
|
|
|
291
292
|
this.process(next_dir);
|
|
292
293
|
});
|
|
293
294
|
}
|
|
295
|
+
else if (directive_name === Directives.DTMF_FORM) {
|
|
296
|
+
// console.log("...DirReply");
|
|
297
|
+
new DirReply(context).execute(directive, async () => {
|
|
298
|
+
let next_dir = await this.nextDirective(this.directives);
|
|
299
|
+
this.process(next_dir);
|
|
300
|
+
});
|
|
301
|
+
}
|
|
302
|
+
else if (directive_name === Directives.DTMF_MENU) {
|
|
303
|
+
// console.log("...DirReply");
|
|
304
|
+
new DirReply(context).execute(directive, async () => {
|
|
305
|
+
let next_dir = await this.nextDirective(this.directives);
|
|
306
|
+
this.process(next_dir);
|
|
307
|
+
});
|
|
308
|
+
}
|
|
309
|
+
else if (directive_name === Directives.BLIND_TRANSFER) {
|
|
310
|
+
// console.log("...DirReply");
|
|
311
|
+
new DirReply(context).execute(directive, async () => {
|
|
312
|
+
let next_dir = await this.nextDirective(this.directives);
|
|
313
|
+
this.process(next_dir);
|
|
314
|
+
});
|
|
315
|
+
}
|
|
294
316
|
else if (directive_name === Directives.RANDOM_REPLY) {
|
|
295
317
|
// console.log("...DirRandomReply");
|
|
296
318
|
new DirRandomReply(context).execute(directive, async () => {
|
|
@@ -685,6 +707,12 @@ class DirectivesChatbotPlug {
|
|
|
685
707
|
this.process(next_dir);
|
|
686
708
|
})
|
|
687
709
|
}
|
|
710
|
+
else if (directive_name === Directives.BREVO) {
|
|
711
|
+
new DirBrevo(context).execute(directive, async () => {
|
|
712
|
+
let next_dir = await this.nextDirective(this.directives);
|
|
713
|
+
this.process(next_dir);
|
|
714
|
+
})
|
|
715
|
+
}
|
|
688
716
|
else {
|
|
689
717
|
//console.log("Unhandled Post-message Directive:", directive_name);
|
|
690
718
|
let next_dir = await this.nextDirective(this.directives);
|
|
@@ -0,0 +1,356 @@
|
|
|
1
|
+
const axios = require("axios").default;
|
|
2
|
+
const { TiledeskChatbot } = require("../../models/TiledeskChatbot");
|
|
3
|
+
const { Filler } = require("../Filler");
|
|
4
|
+
const { DirIntent } = require("./DirIntent");
|
|
5
|
+
let https = require("https");
|
|
6
|
+
require('dotenv').config();
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
class DirBrevo {
|
|
10
|
+
|
|
11
|
+
constructor(context) {
|
|
12
|
+
if (!context) {
|
|
13
|
+
throw new Error('context object is mandatory');
|
|
14
|
+
}
|
|
15
|
+
this.context = context;
|
|
16
|
+
this.tdcache = this.context.tdcache;
|
|
17
|
+
this.requestId = this.context.requestId;
|
|
18
|
+
this.intentDir = new DirIntent(context);
|
|
19
|
+
this.log = context.log;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
execute(directive, callback) {
|
|
23
|
+
if (this.log) { console.log("DirBrevo directive: ", directive); }
|
|
24
|
+
let action;
|
|
25
|
+
if (directive.action) {
|
|
26
|
+
action = directive.action;
|
|
27
|
+
}
|
|
28
|
+
else {
|
|
29
|
+
console.error("DirBrevo Incorrect directive: ", JSON.stringify(directive));
|
|
30
|
+
callback();
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
this.go(action, (stop) => {
|
|
34
|
+
callback(stop);
|
|
35
|
+
})
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
async go(action, callback) {
|
|
39
|
+
if (this.log) { console.log("DirBrevo action:", JSON.stringify(action)); }
|
|
40
|
+
if (!this.tdcache) {
|
|
41
|
+
console.error("Error: DirBrevo tdcache is mandatory");
|
|
42
|
+
callback();
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
let trueIntent = action.trueIntent;
|
|
47
|
+
let falseIntent = action.falseIntent;
|
|
48
|
+
let trueIntentAttributes = action.trueIntentAttributes;
|
|
49
|
+
let falseIntentAttributes = action.falseIntentAttributes;
|
|
50
|
+
|
|
51
|
+
if (this.log) {
|
|
52
|
+
console.log("DirBrevo trueIntent", trueIntent)
|
|
53
|
+
console.log("DirBrevo falseIntent", falseIntent)
|
|
54
|
+
console.log("DirBrevo trueIntentAttributes", trueIntentAttributes)
|
|
55
|
+
console.log("DirBrevo falseIntentAttributes", falseIntentAttributes)
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
let requestVariables = null;
|
|
59
|
+
requestVariables =
|
|
60
|
+
await TiledeskChatbot.allParametersStatic(
|
|
61
|
+
this.tdcache, this.requestId
|
|
62
|
+
)
|
|
63
|
+
|
|
64
|
+
//let token = action.token;
|
|
65
|
+
let bodyParameters = action.bodyParameters;
|
|
66
|
+
if (this.log) { console.log("DirBrevo bodyParameters: ", bodyParameters); }
|
|
67
|
+
|
|
68
|
+
if (!bodyParameters || bodyParameters === '') {
|
|
69
|
+
if (this.log) { console.error("DirBrevo ERROR - bodyParameters is undefined or null or empty string") };
|
|
70
|
+
callback();
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
const server_base_url = process.env.API_ENDPOINT || process.env.API_URL;
|
|
75
|
+
const brevo_base_url = process.env.BREVO_ENDPOINT || "https://api.brevo.com/v3"
|
|
76
|
+
if (this.log) {
|
|
77
|
+
console.log("DirBrevo server_base_url ", server_base_url);
|
|
78
|
+
console.log("DirBrevo brevo_base_url ", brevo_base_url);
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
let key = await this.getKeyFromIntegrations(server_base_url);
|
|
82
|
+
if (this.log) { console.log('DirBrevo key Debug1: ', key) }
|
|
83
|
+
// ONLY FOR DEBUG CANCELLARE!!!!!
|
|
84
|
+
// if (process.env.BREVO_DEBUG == '1') {
|
|
85
|
+
// key = process.env.BREVO_TOKEN;
|
|
86
|
+
// }
|
|
87
|
+
if (!key) {
|
|
88
|
+
if (this.log) { console.log("DirBrevo - Key not found in Integrations."); }
|
|
89
|
+
if (falseIntent) {
|
|
90
|
+
await this.#executeCondition(false, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes);
|
|
91
|
+
callback(true);
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
const filler = new Filler();
|
|
97
|
+
for (const [key, value] of Object.entries(bodyParameters)) {
|
|
98
|
+
if (this.log) { console.log("bodyParam:", key, "value:", value) }
|
|
99
|
+
let filled_value = filler.fill(value, requestVariables);
|
|
100
|
+
bodyParameters[key] = filled_value;
|
|
101
|
+
}
|
|
102
|
+
if (this.log) { console.log('DirBrevo bodyParameters filler: ', bodyParameters) }
|
|
103
|
+
|
|
104
|
+
// CREATE THE JSON FOR BREVO
|
|
105
|
+
let brevo_email = '';
|
|
106
|
+
let brevo_bodyParameters = {};
|
|
107
|
+
for (const [key, value] of Object.entries(bodyParameters)) {
|
|
108
|
+
if (this.log) { console.log("bodyParam:", key, "value:", value) }
|
|
109
|
+
if (key === 'email') {brevo_email = value}
|
|
110
|
+
else { brevo_bodyParameters[key] = value;}
|
|
111
|
+
}
|
|
112
|
+
if (this.log) { console.log('DirBrevo brevo_email: ', brevo_email) }
|
|
113
|
+
if (this.log) { console.log('DirBrevo brevo_bodyParameters: ', brevo_bodyParameters) }
|
|
114
|
+
|
|
115
|
+
|
|
116
|
+
let json = {
|
|
117
|
+
email: brevo_email,
|
|
118
|
+
attributes: brevo_bodyParameters,
|
|
119
|
+
"emailBlacklisted": false,
|
|
120
|
+
"smsBlacklisted": false,
|
|
121
|
+
"listIds": [
|
|
122
|
+
0
|
|
123
|
+
],
|
|
124
|
+
"updateEnabled": false,
|
|
125
|
+
"smtpBlacklistSender": [
|
|
126
|
+
"info@mytest.com"
|
|
127
|
+
]
|
|
128
|
+
}
|
|
129
|
+
if (this.log) { console.log('DirBrevo key Debug2: ', key) }
|
|
130
|
+
//----------------
|
|
131
|
+
if (this.log) {console.log("DirBrevo brevo_base_url ",brevo_base_url);}
|
|
132
|
+
if (this.log) { console.log('DirBrevo json: ', json) }
|
|
133
|
+
const BREVO_HTTPREQUEST = {
|
|
134
|
+
url: brevo_base_url + '/contacts',
|
|
135
|
+
headers: {
|
|
136
|
+
'api-key': key,
|
|
137
|
+
'Content-Type': 'application/json',
|
|
138
|
+
'Accept': 'application/json'
|
|
139
|
+
},
|
|
140
|
+
json: json,
|
|
141
|
+
method: "POST"
|
|
142
|
+
}
|
|
143
|
+
if (this.log) { console.log("DirBrevo BREVO_HTTPREQUEST", JSON.stringify(BREVO_HTTPREQUEST)); }
|
|
144
|
+
|
|
145
|
+
this.#myrequest(
|
|
146
|
+
BREVO_HTTPREQUEST, async (err, resbody) => {
|
|
147
|
+
if (err) {
|
|
148
|
+
if (callback) {
|
|
149
|
+
if (this.log) {
|
|
150
|
+
console.error("(httprequest) DirBrevo err response:", err.response)
|
|
151
|
+
console.error("(httprequest) DirBrevo err data:", err.response.data)
|
|
152
|
+
};
|
|
153
|
+
|
|
154
|
+
let result = null;
|
|
155
|
+
let status = null;
|
|
156
|
+
let error;
|
|
157
|
+
|
|
158
|
+
if (err.response &&
|
|
159
|
+
err.response.status) {
|
|
160
|
+
status = err.response.status;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
if (err.response &&
|
|
164
|
+
err.response.data &&
|
|
165
|
+
err.response.data.message) {
|
|
166
|
+
error = err.response.data.message;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
if (this.log) {
|
|
170
|
+
console.error("(httprequest) DirBrevo err data result:", result); // CONTROLLA IL VALORE
|
|
171
|
+
console.error("(httprequest) DirBrevo err data status:", status);
|
|
172
|
+
console.error("(httprequest) DirBrevo err data error:", error);
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
await this.#assignAttributes(action, status, result, error);
|
|
176
|
+
if (falseIntent) {
|
|
177
|
+
await this.#executeCondition(false, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes);
|
|
178
|
+
callback(true);
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
callback();
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
} else if (callback) {
|
|
185
|
+
if (this.log) { console.log("DirBrevo resbody: ", JSON.stringify(resbody, null, 2).slice(2, -1)); }
|
|
186
|
+
|
|
187
|
+
let status = 201;
|
|
188
|
+
let error = null;
|
|
189
|
+
let result = JSON.stringify(resbody, null, 2).slice(2, -1);
|
|
190
|
+
await this.#assignAttributes(action, status, result, error);
|
|
191
|
+
if (trueIntent) {
|
|
192
|
+
await this.#executeCondition(true, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes)
|
|
193
|
+
callback(true);
|
|
194
|
+
return;
|
|
195
|
+
}
|
|
196
|
+
callback();
|
|
197
|
+
return;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
);
|
|
201
|
+
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
async #assignAttributes(action, status, result, error) {
|
|
205
|
+
if (this.log) {
|
|
206
|
+
console.log("DirBrevo assignAttributes action:", action)
|
|
207
|
+
console.log("DirBrevo assignAttributes status:", status)
|
|
208
|
+
console.log("DirBrevo assignAttributes result:", result)
|
|
209
|
+
console.log("DirBrevo assignAttributes error:", error)
|
|
210
|
+
}
|
|
211
|
+
if (this.context.tdcache) {
|
|
212
|
+
if (action.assignStatusTo) {
|
|
213
|
+
await TiledeskChatbot.addParameterStatic(this.context.tdcache, this.context.requestId, action.assignStatusTo, status);
|
|
214
|
+
}
|
|
215
|
+
if (action.assignResultTo) {
|
|
216
|
+
await TiledeskChatbot.addParameterStatic(this.context.tdcache, this.context.requestId, action.assignResultTo, result);
|
|
217
|
+
}
|
|
218
|
+
if (action.assignErrorTo) {
|
|
219
|
+
await TiledeskChatbot.addParameterStatic(this.context.tdcache, this.context.requestId, action.assignErrorTo, error);
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
// Debug log
|
|
223
|
+
if (this.log) {
|
|
224
|
+
const all_parameters = await TiledeskChatbot.allParametersStatic(this.context.tdcache, this.context.requestId);
|
|
225
|
+
for (const [key, value] of Object.entries(all_parameters)) {
|
|
226
|
+
if (this.log) { console.log("DirBrevo request parameter:", key, "value:", value, "type:", typeof value) }
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
#myrequest(options, callback) {
|
|
233
|
+
if (this.log) {
|
|
234
|
+
console.log("** API URL:", options.url);
|
|
235
|
+
console.log("** Options:", JSON.stringify(options));
|
|
236
|
+
}
|
|
237
|
+
let axios_options = {
|
|
238
|
+
url: options.url,
|
|
239
|
+
method: options.method,
|
|
240
|
+
params: options.params,
|
|
241
|
+
headers: options.headers
|
|
242
|
+
}
|
|
243
|
+
if (options.json !== null) {
|
|
244
|
+
axios_options.data = options.json
|
|
245
|
+
}
|
|
246
|
+
if (this.log) {
|
|
247
|
+
console.log("axios_options:", JSON.stringify(axios_options));
|
|
248
|
+
}
|
|
249
|
+
if (options.url.startsWith("https:")) {
|
|
250
|
+
const httpsAgent = new https.Agent({
|
|
251
|
+
rejectUnauthorized: false,
|
|
252
|
+
});
|
|
253
|
+
axios_options.httpsAgent = httpsAgent;
|
|
254
|
+
}
|
|
255
|
+
axios(axios_options)
|
|
256
|
+
.then((res) => {
|
|
257
|
+
if (this.log) {
|
|
258
|
+
console.log("Response for url:", options.url);
|
|
259
|
+
console.log("Response headers:\n", JSON.stringify(res.headers));
|
|
260
|
+
console.log("Response status:", JSON.stringify(res.status));
|
|
261
|
+
}
|
|
262
|
+
if (res && (res.status == 200 || res.status == 201) && res.data) {
|
|
263
|
+
if (callback) {
|
|
264
|
+
callback(null, res.data);
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
else {
|
|
268
|
+
if (callback) {
|
|
269
|
+
callback(new Error("Response status is not 200"), null);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
})
|
|
273
|
+
.catch((error) => {
|
|
274
|
+
if (callback) {
|
|
275
|
+
callback(error, null);
|
|
276
|
+
}
|
|
277
|
+
});
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
async #executeCondition(result, trueIntent, trueIntentAttributes, falseIntent, falseIntentAttributes, callback) {
|
|
281
|
+
let trueIntentDirective = null;
|
|
282
|
+
|
|
283
|
+
if (trueIntent) {
|
|
284
|
+
trueIntentDirective = DirIntent.intentDirectiveFor(trueIntent, trueIntentAttributes);
|
|
285
|
+
}
|
|
286
|
+
let falseIntentDirective = null;
|
|
287
|
+
if (falseIntent) {
|
|
288
|
+
falseIntentDirective = DirIntent.intentDirectiveFor(falseIntent, falseIntentAttributes);
|
|
289
|
+
}
|
|
290
|
+
if (this.log) { console.log('DirBrevo executeCondition/result', result) }
|
|
291
|
+
if (result === true) {
|
|
292
|
+
if (trueIntentDirective) {
|
|
293
|
+
this.intentDir.execute(trueIntentDirective, () => {
|
|
294
|
+
if (callback) {
|
|
295
|
+
callback();
|
|
296
|
+
}
|
|
297
|
+
});
|
|
298
|
+
}
|
|
299
|
+
else {
|
|
300
|
+
if (this.log) { console.log("No trueIntentDirective specified"); }
|
|
301
|
+
if (callback) {
|
|
302
|
+
callback();
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
else {
|
|
307
|
+
if (falseIntentDirective) {
|
|
308
|
+
this.intentDir.execute(falseIntentDirective, () => {
|
|
309
|
+
if (callback) {
|
|
310
|
+
callback();
|
|
311
|
+
}
|
|
312
|
+
});
|
|
313
|
+
}
|
|
314
|
+
else {
|
|
315
|
+
if (this.log) { console.log("No falseIntentDirective specified"); }
|
|
316
|
+
if (callback) {
|
|
317
|
+
callback();
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
async getKeyFromIntegrations(server_base_url) {
|
|
324
|
+
return new Promise((resolve) => {
|
|
325
|
+
|
|
326
|
+
const INTEGRATIONS_HTTPREQUEST = {
|
|
327
|
+
url: server_base_url + "/" + this.context.projectId + "/integration/name/Brevo",
|
|
328
|
+
headers: {
|
|
329
|
+
'Content-Type': 'application/json',
|
|
330
|
+
'Authorization': 'JWT ' + this.context.token
|
|
331
|
+
},
|
|
332
|
+
method: "GET"
|
|
333
|
+
}
|
|
334
|
+
if (this.log) { console.log("Brevo INTEGRATIONS_HTTPREQUEST ", INTEGRATIONS_HTTPREQUEST) }
|
|
335
|
+
|
|
336
|
+
this.#myrequest(
|
|
337
|
+
INTEGRATIONS_HTTPREQUEST, async (err, integration) => {
|
|
338
|
+
if (err) {
|
|
339
|
+
resolve(null);
|
|
340
|
+
} else {
|
|
341
|
+
|
|
342
|
+
if (integration &&
|
|
343
|
+
integration.value) {
|
|
344
|
+
resolve(integration.value.apikey)
|
|
345
|
+
}
|
|
346
|
+
else {
|
|
347
|
+
resolve(null)
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
})
|
|
351
|
+
})
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
module.exports = { DirBrevo }
|
|
@@ -27,16 +27,23 @@ class Directives {
|
|
|
27
27
|
static RANDOM_REPLY = 'randomreply';
|
|
28
28
|
static CODE = 'code';
|
|
29
29
|
static WHATSAPP_ATTRIBUTE = 'whatsapp_attribute';
|
|
30
|
+
static FORM = "form";
|
|
31
|
+
static CAPTURE_USER_REPLY = "capture_user_reply";
|
|
32
|
+
static REPLACE_BOT_V2 = "replacebotv2";
|
|
33
|
+
/**** AI ****/
|
|
30
34
|
static ASK_GPT = "askgpt";
|
|
31
35
|
static ASK_GPT_V2 = "askgptv2";
|
|
32
36
|
static GPT_TASK = "gpt_task";
|
|
33
|
-
|
|
34
|
-
static CAPTURE_USER_REPLY = "capture_user_reply";
|
|
37
|
+
/**** INTEGRATIONS ****/
|
|
35
38
|
static QAPLA = 'qapla';
|
|
36
39
|
static MAKE = 'make';
|
|
37
|
-
static REPLACE_BOT_V2 = "replacebotv2";
|
|
38
40
|
static HUBSPOT = 'hubspot';
|
|
39
41
|
static CUSTOMERIO = 'customerio';
|
|
42
|
+
static BREVO = 'brevo';
|
|
43
|
+
/**** VOICE CHANNEL ****/
|
|
44
|
+
static DTMF_FORM = 'dtmf_form';
|
|
45
|
+
static DTMF_MENU = 'dtmf_menu';
|
|
46
|
+
static BLIND_TRANSFER = 'blind_transfer';
|
|
40
47
|
|
|
41
48
|
// static WHEN_ONLINE_MOVE_TO_AGENT = "whenonlinemovetoagent"; // DEPRECATED?
|
|
42
49
|
// static WHEN_OFFLINE_HOURS = "whenofflinehours"; // DEPRECATED // adds a message on top of the original message when offline hours opts: --replace
|