2018-04-21 02:59:30 -03:00
|
|
|
|
/*****************************************************************************\
|
|
|
|
|
| ( )_ _ |
|
|
|
|
|
| _ _ _ __ _ _ __ ___ ___ _ _ | ,_)(_) ___ ___ _ |
|
2020-07-01 15:00:40 -03:00
|
|
|
|
| ( '_`\ ( '__)/'_` ) /'_ `\/' _ ` _ `\ /'_` )| | | |/',__)/' v `\ /'_`\ |
|
2019-03-09 16:59:31 -03:00
|
|
|
|
| | (_) )| | ( (_| |( (_) || ( ) ( ) |( (_| || |_ | |\__, \| (˅) |( (_) ) |
|
2018-04-21 02:59:30 -03:00
|
|
|
|
| | ,__/'(_) `\__,_)`\__ |(_) (_) (_)`\__,_)`\__)(_)(____/(_) (_)`\___/' |
|
|
|
|
|
| | | ( )_) | |
|
|
|
|
|
| (_) \___/' |
|
|
|
|
|
| |
|
|
|
|
|
| General Bots Copyright (c) Pragmatismo.io. All rights reserved. |
|
|
|
|
|
| Licensed under the AGPL-3.0. |
|
2018-11-11 19:09:18 -02:00
|
|
|
|
| |
|
2018-04-21 02:59:30 -03:00
|
|
|
|
| According to our dual licensing model, this program can be used either |
|
|
|
|
|
| under the terms of the GNU Affero General Public License, version 3, |
|
|
|
|
|
| or under a proprietary license. |
|
|
|
|
|
| |
|
|
|
|
|
| The texts of the GNU Affero General Public License with an additional |
|
|
|
|
|
| permission and of our proprietary license can be found at and |
|
|
|
|
|
| in the LICENSE file you have received along with this program. |
|
|
|
|
|
| |
|
|
|
|
|
| This program is distributed in the hope that it will be useful, |
|
2018-09-11 19:40:53 -03:00
|
|
|
|
| but WITHOUT ANY WARRANTY, without even the implied warranty of |
|
2018-04-21 02:59:30 -03:00
|
|
|
|
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
|
|
|
|
| GNU Affero General Public License for more details. |
|
|
|
|
|
| |
|
|
|
|
|
| "General Bots" is a registered trademark of Pragmatismo.io. |
|
|
|
|
|
| The licensing of the program under the AGPLv3 does not imply a |
|
|
|
|
|
| trademark license. Therefore any rights, title and interest in |
|
|
|
|
|
| our trademarks remain entirely with us. |
|
|
|
|
|
| |
|
|
|
|
|
\*****************************************************************************/
|
|
|
|
|
|
2018-11-26 14:09:09 -02:00
|
|
|
|
'use strict';
|
|
|
|
|
|
2019-01-31 11:32:33 -02:00
|
|
|
|
import { WaterfallDialog } from 'botbuilder-dialogs';
|
2020-08-29 15:02:19 -03:00
|
|
|
|
import { GBLog, GBMinInstance, GBService, IGBCoreService, GBDialogStep } from 'botlib';
|
2018-11-26 15:54:34 -02:00
|
|
|
|
import * as fs from 'fs';
|
2020-12-27 13:30:56 -03:00
|
|
|
|
import { GBDeployer } from '../../core.gbapp/services/GBDeployer';
|
2018-11-30 17:30:48 -02:00
|
|
|
|
import { TSCompiler } from './TSCompiler';
|
2020-06-14 21:40:41 -03:00
|
|
|
|
import { CollectionUtil } from 'pragmatismo-io-framework';
|
2019-03-09 16:59:31 -03:00
|
|
|
|
import urlJoin = require('url-join');
|
2020-12-27 13:30:56 -03:00
|
|
|
|
import { DialogKeywords } from './DialogKeywords';
|
2020-05-15 14:07:30 -03:00
|
|
|
|
import { Messages } from '../strings';
|
2020-12-27 13:30:56 -03:00
|
|
|
|
import { GBConversationalService } from '../../core.gbapp/services/GBConversationalService';
|
2019-03-09 16:59:31 -03:00
|
|
|
|
//tslint:disable-next-line:no-submodule-imports
|
2020-08-29 15:02:19 -03:00
|
|
|
|
const vm = require('vm');
|
2020-12-27 13:30:56 -03:00
|
|
|
|
const vb2ts = require('./vbscript-to-typescript');
|
2019-03-08 06:49:22 -03:00
|
|
|
|
const beautify = require('js-beautify').js;
|
2020-12-31 15:36:19 -03:00
|
|
|
|
const textract = require('textract');
|
|
|
|
|
const walkPromise = require('walk-promise');
|
2020-12-07 23:08:52 -03:00
|
|
|
|
const phoneUtil = require('google-libphonenumber').PhoneNumberUtil.getInstance();
|
|
|
|
|
const phone = require('phone');
|
2018-11-26 14:09:09 -02:00
|
|
|
|
|
2018-11-11 19:09:18 -02:00
|
|
|
|
/**
|
2018-11-30 11:55:44 -02:00
|
|
|
|
* @fileoverview Virtualization services for emulation of BASIC.
|
2019-02-23 13:17:21 -03:00
|
|
|
|
* This alpha version is using a hack in form of converter to
|
2019-10-10 07:40:15 -03:00
|
|
|
|
* translate BASIC to TS and string replacements to emulate await code.
|
2019-02-23 13:17:21 -03:00
|
|
|
|
* See http://jsfiddle.net/roderick/dym05hsy for more info on vb2ts, so
|
2019-01-31 11:32:33 -02:00
|
|
|
|
* http://stevehanov.ca/blog/index.php?id=92 should be used to run it without
|
|
|
|
|
* translation and enhance classic BASIC experience.
|
2018-11-11 19:09:18 -02:00
|
|
|
|
*/
|
|
|
|
|
|
2019-03-09 16:59:31 -03:00
|
|
|
|
/**
|
|
|
|
|
* Basic services for BASIC manipulation.
|
|
|
|
|
*/
|
2019-03-08 06:37:13 -03:00
|
|
|
|
export class GBVMService extends GBService {
|
2019-02-23 13:17:21 -03:00
|
|
|
|
public async loadDialogPackage(folder: string, min: GBMinInstance, core: IGBCoreService, deployer: GBDeployer) {
|
|
|
|
|
const files = await walkPromise(folder);
|
2019-02-25 08:36:43 -03:00
|
|
|
|
this.addHearDialog(min);
|
2019-02-23 13:17:21 -03:00
|
|
|
|
|
2020-06-14 21:40:41 -03:00
|
|
|
|
await CollectionUtil.asyncForEach(files, async file => {
|
|
|
|
|
if (!file) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let filename: string = file.name;
|
|
|
|
|
|
|
|
|
|
if (filename.endsWith('.docx')) {
|
|
|
|
|
const wordFile = filename;
|
|
|
|
|
const vbsFile = filename.substr(0, filename.indexOf('docx')) + 'vbs';
|
|
|
|
|
const fullVbsFile = urlJoin(folder, vbsFile);
|
|
|
|
|
const docxStat = fs.statSync(urlJoin(folder, wordFile));
|
|
|
|
|
const interval = 30000; // If compiled is older 30 seconds, then recompile.
|
|
|
|
|
let writeVBS = true;
|
|
|
|
|
if (fs.existsSync(fullVbsFile)) {
|
|
|
|
|
const vbsStat = fs.statSync(fullVbsFile);
|
2020-08-29 15:02:19 -03:00
|
|
|
|
if (docxStat.mtimeMs < vbsStat.mtimeMs + interval) {
|
2020-06-14 21:40:41 -03:00
|
|
|
|
writeVBS = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (writeVBS) {
|
|
|
|
|
let text = await this.getTextFromWord(folder, wordFile);
|
|
|
|
|
fs.writeFileSync(urlJoin(folder, vbsFile), text);
|
|
|
|
|
}
|
2020-05-14 12:46:57 -03:00
|
|
|
|
|
2020-06-14 21:40:41 -03:00
|
|
|
|
filename = vbsFile;
|
2020-06-04 13:44:02 -03:00
|
|
|
|
|
2020-11-06 17:06:22 -03:00
|
|
|
|
let mainName = GBVMService.getMethodNameFromVBSFilename(filename);
|
|
|
|
|
min.scriptMap[filename] = mainName;
|
2020-06-04 13:44:02 -03:00
|
|
|
|
|
2020-06-14 21:40:41 -03:00
|
|
|
|
const fullFilename = urlJoin(folder, filename);
|
|
|
|
|
// TODO: Implement in development mode, how swap for .vbs files
|
|
|
|
|
// fs.watchFile(fullFilename, async () => {
|
|
|
|
|
// await this.run(fullFilename, min, deployer, mainName);
|
|
|
|
|
// });
|
|
|
|
|
|
|
|
|
|
const compiledAt = fs.statSync(fullFilename);
|
|
|
|
|
const jsfile = urlJoin(folder, `${filename}.js`);
|
|
|
|
|
|
|
|
|
|
if (fs.existsSync(jsfile)) {
|
|
|
|
|
const jsStat = fs.statSync(jsfile);
|
|
|
|
|
const interval = 30000; // If compiled is older 30 seconds, then recompile.
|
2020-08-29 15:02:19 -03:00
|
|
|
|
if (compiledAt.isFile() && compiledAt.mtimeMs > jsStat.mtimeMs + interval) {
|
2020-06-14 21:40:41 -03:00
|
|
|
|
await this.executeBASIC(fullFilename, min, deployer, mainName);
|
2020-08-29 15:02:19 -03:00
|
|
|
|
} else {
|
2020-06-14 21:40:41 -03:00
|
|
|
|
const parsedCode: string = fs.readFileSync(jsfile, 'utf8');
|
|
|
|
|
this.executeJS(min, deployer, parsedCode, mainName);
|
2020-06-04 13:44:02 -03:00
|
|
|
|
}
|
2020-08-29 15:02:19 -03:00
|
|
|
|
} else {
|
2020-06-14 21:40:41 -03:00
|
|
|
|
await this.executeBASIC(fullFilename, min, deployer, mainName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
2019-02-23 13:17:21 -03:00
|
|
|
|
}
|
|
|
|
|
|
2020-11-06 17:06:22 -03:00
|
|
|
|
public static getMethodNameFromVBSFilename(filename: string) {
|
|
|
|
|
let mainName = filename.replace(/\s|\-/gi, '').split('.')[0];
|
|
|
|
|
return mainName.toLowerCase();
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-15 14:07:30 -03:00
|
|
|
|
private async getTextFromWord(folder: string, filename: string) {
|
|
|
|
|
return new Promise<string>(async (resolve, reject) => {
|
2020-08-29 15:02:19 -03:00
|
|
|
|
textract.fromFileWithPath(urlJoin(folder, filename), { preserveLineBreaks: true }, (error, text) => {
|
|
|
|
|
if (error) {
|
|
|
|
|
reject(error);
|
|
|
|
|
} else {
|
|
|
|
|
text = text.replace('“', '"');
|
|
|
|
|
text = text.replace('”', '"');
|
|
|
|
|
text = text.replace('‘', "'");
|
|
|
|
|
text = text.replace('’', "'");
|
|
|
|
|
|
|
|
|
|
resolve(text);
|
|
|
|
|
}
|
|
|
|
|
});
|
2020-05-15 14:07:30 -03:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
/**
|
|
|
|
|
* Converts General Bots BASIC
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @param code General Bots BASIC
|
|
|
|
|
*/
|
|
|
|
|
public convertGBASICToVBS(code: string) {
|
|
|
|
|
// Start and End of VB2TS tags of processing.
|
|
|
|
|
|
2020-11-22 11:22:55 -03:00
|
|
|
|
|
2020-05-11 10:41:41 -03:00
|
|
|
|
code = `<%\n
|
|
|
|
|
|
|
|
|
|
id = sys().getRandomId()
|
2021-01-03 19:13:27 -03:00
|
|
|
|
username = this.userName(step);
|
|
|
|
|
mobile = this.userMobile(step);
|
2020-12-28 18:43:34 -03:00
|
|
|
|
from = mobile;
|
2020-11-11 16:03:05 -03:00
|
|
|
|
ubound = function(list){return list.length};
|
2020-11-08 13:39:18 -03:00
|
|
|
|
|
2020-05-11 10:41:41 -03:00
|
|
|
|
${code}
|
|
|
|
|
`;
|
2019-02-23 13:17:21 -03:00
|
|
|
|
|
|
|
|
|
// Keywords from General Bots BASIC.
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as email/gi, ($0, $1) => {
|
|
|
|
|
return `${$1} = hear("email")`;
|
2020-12-07 22:25:43 -03:00
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as integer/gi, ($0, $1, $2) => {
|
|
|
|
|
return `${$1} = hear("integer")`;
|
2020-12-07 22:25:43 -03:00
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:24:00 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as boolean/gi, ($0, $1, $2) => {
|
|
|
|
|
return `${$1} = hear("boolean")`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as name/gi, ($0, $1, $2) => {
|
2020-12-07 22:25:43 -03:00
|
|
|
|
return `${$1} = hear("name")`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as date/gi, ($0, $1, $2) => {
|
2020-12-07 22:25:43 -03:00
|
|
|
|
return `${$1} = hear("date")`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as hour/gi, ($0, $1, $2) => {
|
2020-12-07 22:25:43 -03:00
|
|
|
|
return `${$1} = hear("hour")`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as phone/gi, ($0, $1, $2) => {
|
2020-12-07 22:25:43 -03:00
|
|
|
|
return `${$1} = hear("phone")`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as money/gi, ($0, $1, $2) => {
|
2020-12-07 22:25:43 -03:00
|
|
|
|
return `${$1} = hear("money")`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-22 21:48:41 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as language/gi, ($0, $1, $2) => {
|
|
|
|
|
return `${$1} = hear("language")`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
code = code.replace(/hear (\w+) as zipcode/gi, ($0, $1, $2) => {
|
|
|
|
|
return `${$1} = hear("zipcode")`;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
code = code.replace(/hear (\w+) as (.*)/gi, ($0, $1, $2) => {
|
|
|
|
|
return `${$1} = hear("menu", ${$2})`;
|
2020-12-07 22:25:43 -03:00
|
|
|
|
});
|
2020-05-14 12:46:57 -03:00
|
|
|
|
|
2020-12-01 18:01:53 -03:00
|
|
|
|
code = code.replace(/(hear on)(\s)(.*)/gi, ($0, $1, $2, $3) => {
|
|
|
|
|
return `sys().gotoDialog(${$3})\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/(hear)\s*(\w+)/gi, ($0, $1, $2) => {
|
2019-02-23 13:17:21 -03:00
|
|
|
|
return `${$2} = hear()`;
|
2018-11-30 17:30:48 -02:00
|
|
|
|
});
|
2020-07-07 10:15:39 -03:00
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/(\w)\s*\=\s*find\s*(.*)/gi, ($0, $1, $2, $3) => {
|
2020-05-14 12:46:57 -03:00
|
|
|
|
return `${$1} = sys().find(${$2})\n`;
|
|
|
|
|
});
|
2019-02-23 13:17:21 -03:00
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/(wait)\s*(\d+)/gi, ($0, $1, $2) => {
|
2019-02-23 13:17:21 -03:00
|
|
|
|
return `sys().wait(${$2})`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/(get stock for )(.*)/gi, ($0, $1, $2) => {
|
2020-05-11 10:41:41 -03:00
|
|
|
|
return `let stock = sys().getStock(${$2})`;
|
2019-02-23 13:17:21 -03:00
|
|
|
|
});
|
|
|
|
|
|
2020-08-15 11:39:43 -03:00
|
|
|
|
code = code.replace(/(\w+)\s*\=\s*get\s(.*)/gi, ($0, $1, $2) => {
|
2020-11-06 15:23:56 -03:00
|
|
|
|
if ($2.indexOf('http') !== -1) {
|
|
|
|
|
return `let ${$1} = sys().httpGet (${$2})`;
|
|
|
|
|
} else {
|
|
|
|
|
return `let ${$1} = sys().get (${$2})`;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2020-12-22 21:48:41 -03:00
|
|
|
|
code = code.replace(/(set language)(\s*)(.*)/gi, ($0, $1, $2, $3) => {
|
|
|
|
|
return `setLanguage (step, ${$3})\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-11-06 15:23:56 -03:00
|
|
|
|
code = code.replace(/set\s(.*)/gi, ($0, $1, $2) => {
|
2020-11-09 17:40:34 -03:00
|
|
|
|
return `sys().set (${$1})`;
|
2020-08-15 11:39:43 -03:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
code = code.replace(/(\w+)\s*\=\s*post\s*(.*),\s*(.*)/gi, ($0, $1, $2, $3) => {
|
|
|
|
|
return `let ${$1} = sys().httpPost (${$2}, ${$3})`;
|
2019-03-08 06:37:13 -03:00
|
|
|
|
});
|
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/(create a bot farm using)(\s)(.*)/gi, ($0, $1, $2, $3) => {
|
2019-02-23 13:17:21 -03:00
|
|
|
|
return `sys().createABotFarmUsing (${$3})`;
|
|
|
|
|
});
|
|
|
|
|
|
2021-02-07 08:12:32 -03:00
|
|
|
|
code = code.replace(/(transfer)(?=(?:[^"]|"[^"]*")*$)/gi, () => {
|
2020-10-14 14:04:02 -03:00
|
|
|
|
return `transfer (step)\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2021-02-07 18:28:54 -03:00
|
|
|
|
code = code.replace(/(exit)/gi, () => {
|
2020-12-22 21:48:41 -03:00
|
|
|
|
return `resolve();\n`;
|
2020-12-02 17:20:38 -03:00
|
|
|
|
});
|
|
|
|
|
|
2021-02-07 18:28:54 -03:00
|
|
|
|
code = code.replace(/(show menu)/gi, () => {
|
2020-12-11 07:35:55 -03:00
|
|
|
|
return `showMenu (step)\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-11-08 13:39:18 -03:00
|
|
|
|
code = code.replace(/(talk to)(\s)(.*)/gi, ($0, $1, $2, $3) => {
|
|
|
|
|
return `sys().talkTo(${$3})\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/(talk)(\s)(.*)/gi, ($0, $1, $2, $3) => {
|
2019-08-29 19:59:58 -03:00
|
|
|
|
return `talk (step, ${$3})\n`;
|
2019-02-23 13:17:21 -03:00
|
|
|
|
});
|
2020-11-11 12:27:17 -03:00
|
|
|
|
|
2020-11-26 12:45:10 -03:00
|
|
|
|
code = code.replace(/(send sms to)(\s*)(.*)/gi, ($0, $1, $2, $3) => {
|
|
|
|
|
return `sys().sendSmsTo (${$3})\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2020-11-11 11:10:40 -03:00
|
|
|
|
code = code.replace(/(send file to)(\s*)(.*)/gi, ($0, $1, $2, $3) => {
|
|
|
|
|
return `sendFileTo (step, ${$3})\n`;
|
|
|
|
|
});
|
2019-02-23 13:17:21 -03:00
|
|
|
|
|
2020-11-11 11:38:19 -03:00
|
|
|
|
code = code.replace(/(send file)(\s*)(.*)/gi, ($0, $1, $2, $3) => {
|
|
|
|
|
return `sendFile (step, ${$3})\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2021-01-20 18:23:42 -03:00
|
|
|
|
code = code.replace(/(copy)(\s*)(.*)/gi, ($0, $1, $2, $3) => {
|
2021-01-28 08:42:06 -03:00
|
|
|
|
return `sys().copyFile(${$3})\n`;
|
2021-01-20 18:23:42 -03:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
code = code.replace(/(convert)(\s*)(.*)/gi, ($0, $1, $2, $3) => {
|
2021-01-28 08:42:06 -03:00
|
|
|
|
return `sys().convert(${$3})\n`;
|
2021-01-15 11:50:43 -03:00
|
|
|
|
});
|
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/(save)(\s)(.*)/gi, ($0, $1, $2, $3) => {
|
2020-05-11 10:41:41 -03:00
|
|
|
|
return `sys().save(${$3})\n`;
|
|
|
|
|
});
|
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
code = `${code}\n%>`;
|
|
|
|
|
|
|
|
|
|
return code;
|
2018-12-01 17:31:57 -02:00
|
|
|
|
}
|
|
|
|
|
|
2020-06-04 13:44:02 -03:00
|
|
|
|
public async executeBASIC(filename: any, min: GBMinInstance, deployer: GBDeployer, mainName: string) {
|
2019-02-23 13:17:21 -03:00
|
|
|
|
// Converts General Bots BASIC into regular VBS
|
2018-04-21 02:59:30 -03:00
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
const basicCode: string = fs.readFileSync(filename, 'utf8');
|
2019-03-09 16:59:31 -03:00
|
|
|
|
const vbsCode = this.convertGBASICToVBS(basicCode);
|
2019-02-23 13:17:21 -03:00
|
|
|
|
const vbsFile = `${filename}.compiled`;
|
|
|
|
|
fs.writeFileSync(vbsFile, vbsCode, 'utf8');
|
|
|
|
|
|
|
|
|
|
// Converts VBS into TS.
|
|
|
|
|
vb2ts.convertFile(vbsFile);
|
2018-11-30 11:55:44 -02:00
|
|
|
|
|
2018-11-30 17:30:48 -02:00
|
|
|
|
// Convert TS into JS.
|
2019-02-23 13:17:21 -03:00
|
|
|
|
const tsfile: string = `${filename}.ts`;
|
|
|
|
|
let tsCode: string = fs.readFileSync(tsfile, 'utf8');
|
2020-06-11 09:47:59 -03:00
|
|
|
|
tsCode = tsCode.replace(/export.*\n/gi, `export function ${mainName}(step:any) { let resolve;`);
|
2019-02-23 13:17:21 -03:00
|
|
|
|
fs.writeFileSync(tsfile, tsCode);
|
|
|
|
|
|
2018-11-30 17:30:48 -02:00
|
|
|
|
const tsc = new TSCompiler();
|
2019-02-23 13:17:21 -03:00
|
|
|
|
tsc.compile([tsfile]);
|
2018-12-02 19:59:27 -02:00
|
|
|
|
|
2018-11-30 17:30:48 -02:00
|
|
|
|
// Run JS into the GB context.
|
2019-02-23 13:17:21 -03:00
|
|
|
|
const jsfile = `${tsfile}.js`.replace('.ts', '');
|
|
|
|
|
|
|
|
|
|
if (fs.existsSync(jsfile)) {
|
|
|
|
|
let code: string = fs.readFileSync(jsfile, 'utf8');
|
2018-12-01 17:31:57 -02:00
|
|
|
|
|
2018-11-30 17:30:48 -02:00
|
|
|
|
code = code.replace(/^.*exports.*$/gm, '');
|
2018-12-02 19:59:27 -02:00
|
|
|
|
|
|
|
|
|
// Finds all hear calls.
|
|
|
|
|
|
|
|
|
|
let parsedCode = code;
|
2020-12-07 22:25:43 -03:00
|
|
|
|
const hearExp = /(\w+).*hear.*\((.*)\)/;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2019-05-15 22:30:14 -03:00
|
|
|
|
let match1;
|
2018-12-02 19:59:27 -02:00
|
|
|
|
|
2019-05-15 22:30:14 -03:00
|
|
|
|
while ((match1 = hearExp.exec(code))) {
|
2018-12-02 19:59:27 -02:00
|
|
|
|
let pos = 0;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
|
// Writes async body.
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
const variable = match1[1]; // Construct variable = hear ().
|
2020-12-07 22:25:43 -03:00
|
|
|
|
const args = match1[2]; // Construct variable = hear ("A", "B").
|
2019-08-29 19:59:58 -03:00
|
|
|
|
const promiseName = `promiseFor${variable}`;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
|
parsedCode = code.substring(pos, pos + match1.index);
|
2019-08-29 19:59:58 -03:00
|
|
|
|
parsedCode += ``;
|
2020-08-29 15:02:19 -03:00
|
|
|
|
parsedCode += `const ${promiseName}= async (step, ${variable}) => {`;
|
2020-12-13 10:02:49 -03:00
|
|
|
|
parsedCode += ` return new Promise(async (resolve, reject) => { try {`;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
|
// Skips old construction and point to the async block.
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2018-12-01 23:01:42 -02:00
|
|
|
|
pos = pos + match1.index;
|
2018-12-02 19:59:27 -02:00
|
|
|
|
let tempCode = code.substring(pos + match1[0].length + 1);
|
2019-01-31 11:32:33 -02:00
|
|
|
|
const start = pos;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
|
// Balances code blocks and checks for exits.
|
2018-12-01 23:01:42 -02:00
|
|
|
|
|
2018-12-01 20:48:08 -02:00
|
|
|
|
let right = 0;
|
|
|
|
|
let left = 1;
|
2019-05-15 22:30:14 -03:00
|
|
|
|
let match2;
|
|
|
|
|
while ((match2 = /\{|\}/.exec(tempCode))) {
|
2018-12-02 19:59:27 -02:00
|
|
|
|
const c = tempCode.substring(match2.index, match2.index + 1);
|
2018-12-01 23:01:42 -02:00
|
|
|
|
|
2018-12-01 20:48:08 -02:00
|
|
|
|
if (c === '}') {
|
|
|
|
|
right++;
|
|
|
|
|
} else if (c === '{') {
|
|
|
|
|
left++;
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
|
tempCode = tempCode.substring(match2.index + 1);
|
2018-12-01 23:01:42 -02:00
|
|
|
|
pos += match2.index + 1;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2018-12-01 23:01:42 -02:00
|
|
|
|
if (left === right) {
|
|
|
|
|
break;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-12-01 23:01:42 -02:00
|
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
|
parsedCode += code.substring(start + match1[0].length + 1, pos + match1[0].length);
|
2020-12-22 21:48:41 -03:00
|
|
|
|
|
2020-12-13 10:02:49 -03:00
|
|
|
|
parsedCode += '}catch(error){reject(error);}});\n';
|
2019-08-29 19:59:58 -03:00
|
|
|
|
parsedCode += '}\n';
|
2020-12-01 18:01:53 -03:00
|
|
|
|
|
|
|
|
|
|
2020-12-07 23:08:52 -03:00
|
|
|
|
parsedCode += `hear (step, ${promiseName}, resolve, ${args === '' ? null : args});\n`;
|
2018-12-02 19:59:27 -02:00
|
|
|
|
parsedCode += code.substring(pos + match1[0].length);
|
|
|
|
|
|
|
|
|
|
// A interaction will be made for each hear.
|
2018-12-01 23:01:42 -02:00
|
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
|
code = parsedCode;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
parsedCode = this.handleThisAndAwait(parsedCode);
|
2018-12-02 19:59:27 -02:00
|
|
|
|
|
2020-11-26 10:24:55 -03:00
|
|
|
|
parsedCode = parsedCode.replace(/(now)(?=(?:[^"]|"[^"]*")*$)/gi, 'await this.getNow(step)');
|
|
|
|
|
parsedCode = parsedCode.replace(/(today)(?=(?:[^"]|"[^"]*")*$)/gi, 'await this.getToday(step)');
|
2020-11-22 11:22:55 -03:00
|
|
|
|
|
2020-08-29 15:02:19 -03:00
|
|
|
|
parsedCode = beautify(parsedCode, { indent_size: 2, space_in_empty_paren: true });
|
2019-02-23 13:17:21 -03:00
|
|
|
|
fs.writeFileSync(jsfile, parsedCode);
|
2018-12-01 20:48:08 -02:00
|
|
|
|
|
2020-06-04 13:44:02 -03:00
|
|
|
|
this.executeJS(min, deployer, parsedCode, mainName);
|
2020-12-01 18:01:53 -03:00
|
|
|
|
GBLog.info(`[GBVMService] Finished loading of ${filename}, JavaScript from Word: \n ${parsedCode}`);
|
2020-06-04 13:44:02 -03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private executeJS(min: GBMinInstance, deployer: GBDeployer, parsedCode: string, mainName: string) {
|
|
|
|
|
try {
|
2020-08-29 15:02:19 -03:00
|
|
|
|
min.sandBoxMap[mainName.toLowerCase()] = parsedCode;
|
|
|
|
|
} catch (error) {
|
2020-06-04 13:44:02 -03:00
|
|
|
|
GBLog.error(`[GBVMService] ERROR loading ${error}`);
|
2018-12-01 20:48:08 -02:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-12-01 23:01:42 -02:00
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
private handleThisAndAwait(code: string) {
|
|
|
|
|
// this insertion.
|
|
|
|
|
|
2020-06-03 21:31:00 -03:00
|
|
|
|
code = code.replace(/sys\(\)/gi, 'this.sys()');
|
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\btalk\b/gi, ($0, $1) => {
|
2019-03-09 16:59:31 -03:00
|
|
|
|
return $1 === undefined ? 'this.talk' : $1;
|
2019-02-23 13:17:21 -03:00
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bhear\b/gi, ($0, $1) => {
|
2019-03-09 16:59:31 -03:00
|
|
|
|
return $1 === undefined ? 'this.hear' : $1;
|
2019-02-23 13:17:21 -03:00
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bsendEmail\b/gi, ($0, $1) => {
|
2019-03-09 16:59:31 -03:00
|
|
|
|
return $1 === undefined ? 'this.sendEmail' : $1;
|
2019-02-23 13:17:21 -03:00
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\baskEmail\b/gi, ($0, $1) => {
|
2020-05-11 10:41:41 -03:00
|
|
|
|
return $1 === undefined ? 'this.askEmail' : $1;
|
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bsendFileTo\b/gi, ($0, $1) => {
|
2020-11-11 12:27:17 -03:00
|
|
|
|
return $1 === undefined ? 'this.sendFileTo' : $1;
|
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bsendFile\b/gi, ($0, $1) => {
|
2020-11-11 12:43:08 -03:00
|
|
|
|
return $1 === undefined ? 'this.sendFile' : $1;
|
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bsetLanguage\b/gi, ($0, $1) => {
|
2020-12-22 21:48:41 -03:00
|
|
|
|
return $1 === undefined ? 'this.setLanguage' : $1;
|
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\btransfer\b/gi, ($0, $1) => {
|
2020-10-14 14:04:02 -03:00
|
|
|
|
return $1 === undefined ? 'this.transfer' : $1;
|
|
|
|
|
});
|
2020-12-11 07:35:55 -03:00
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bmenu\b/gi, ($0, $1) => {
|
|
|
|
|
return $1 === undefined ? 'this.menu' : $1;
|
|
|
|
|
});
|
2021-02-23 07:26:08 -03:00
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
// await insertion.
|
|
|
|
|
|
|
|
|
|
code = code.replace(/this\./gm, 'await this.');
|
|
|
|
|
code = code.replace(/function/gm, 'async function');
|
2020-11-11 16:03:05 -03:00
|
|
|
|
code = code.replace('ubound = async', 'ubound ='); // TODO: Improve this.
|
2020-11-22 11:22:55 -03:00
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-01 23:01:42 -02:00
|
|
|
|
private addHearDialog(min) {
|
|
|
|
|
min.dialogs.add(
|
|
|
|
|
new WaterfallDialog('/hear', [
|
|
|
|
|
async step => {
|
2020-12-01 18:01:53 -03:00
|
|
|
|
step.activeDialog.state.options = step.options;
|
|
|
|
|
step.activeDialog.state.options.id = (step.options as any).id;
|
2019-08-29 19:59:58 -03:00
|
|
|
|
step.activeDialog.state.options.previousResolve = (step.options as any).previousResolve;
|
2020-12-01 18:01:53 -03:00
|
|
|
|
|
2020-12-07 22:25:43 -03:00
|
|
|
|
if (step.options['args']) {
|
2020-12-01 18:01:53 -03:00
|
|
|
|
|
2020-12-07 22:25:43 -03:00
|
|
|
|
GBLog.info(`BASIC: Asking for input (HEAR with ${step.options['args'][0]}).`);
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
|
|
|
|
|
GBLog.info('BASIC: Asking for input (HEAR).');
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-04 13:44:02 -03:00
|
|
|
|
return await min.conversationalService.prompt(min, step, null);
|
2018-12-01 23:01:42 -02:00
|
|
|
|
},
|
|
|
|
|
async step => {
|
2020-05-15 14:07:30 -03:00
|
|
|
|
|
2020-12-01 18:01:53 -03:00
|
|
|
|
const isIntentYes = (locale, utterance) => {
|
|
|
|
|
return utterance.toLowerCase().match(Messages[locale].affirmative_sentences);
|
|
|
|
|
}
|
2020-12-06 16:22:34 -03:00
|
|
|
|
|
2020-12-01 18:01:53 -03:00
|
|
|
|
let result = step.result;
|
2020-12-07 22:25:43 -03:00
|
|
|
|
if (step.activeDialog.state.options['kind'] === "boolean") {
|
2020-12-07 23:24:00 -03:00
|
|
|
|
if (isIntentYes('pt-BR', step.result)) {
|
2020-12-01 18:01:53 -03:00
|
|
|
|
result = true;
|
|
|
|
|
}
|
|
|
|
|
else {
|
2020-12-07 23:24:00 -03:00
|
|
|
|
result = false;
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-12-07 22:25:43 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "email") {
|
2020-12-07 23:08:52 -03:00
|
|
|
|
|
|
|
|
|
const extractEntity = (text) => {
|
|
|
|
|
return text.match(/([a-zA-Z0-9._-]+@[a-zA-Z0-9._-]+\.[a-zA-Z0-9_-]+)/gi);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const value = extractEntity(step.result);
|
|
|
|
|
|
|
|
|
|
if (value === null) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite um e-mail válido.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = value;
|
|
|
|
|
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-07 22:25:43 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "name") {
|
|
|
|
|
const extractEntity = text => {
|
|
|
|
|
return text.match(/[_a-zA-Z][_a-zA-Z0-9]{0,16}/gi);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const value = extractEntity(step.result);
|
|
|
|
|
|
|
|
|
|
if (value === null || value.length != 1) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite um nome válido.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = value;
|
|
|
|
|
|
|
|
|
|
}
|
2020-12-07 23:08:52 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "integer") {
|
2020-12-07 22:25:43 -03:00
|
|
|
|
const extractEntity = text => {
|
|
|
|
|
return text.match(/\d+/gi);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const value = extractEntity(step.result);
|
|
|
|
|
|
|
|
|
|
if (value === null || value.length != 1) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite um número válido.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = value;
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-07 22:25:43 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "date") {
|
|
|
|
|
const extractEntity = text => {
|
|
|
|
|
return text.match(/(^(((0[1-9]|1[0-9]|2[0-8])[\/](0[1-9]|1[012]))|((29|30|31)[\/](0[13578]|1[02]))|((29|30)[\/](0[4,6,9]|11)))[\/](19|[2-9][0-9])\d\d$)|(^29[\/]02[\/](19|[2-9][0-9])(00|04|08|12|16|20|24|28|32|36|40|44|48|52|56|60|64|68|72|76|80|84|88|92|96)$)/gi);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const value = extractEntity(step.result);
|
|
|
|
|
|
|
|
|
|
if (value === null || value.length != 1) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite uma data no formato 12/12/2020.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = value;
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-07 22:25:43 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "hour") {
|
2020-12-07 23:08:52 -03:00
|
|
|
|
|
|
|
|
|
const extractEntity = text => {
|
|
|
|
|
return text.match(/^([0-1]?[0-9]|2[0-4]):([0-5][0-9])(:[0-5][0-9])?$/gi);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const value = extractEntity(step.result);
|
|
|
|
|
|
|
|
|
|
if (value === null || value.length != 1) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite um horário no formato hh:ss.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = value;
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-07 22:25:43 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "money") {
|
2020-12-07 23:08:52 -03:00
|
|
|
|
const extractEntity = text => {
|
|
|
|
|
|
|
|
|
|
if (step.context.locale === 'en') { // TODO: Change to user.
|
|
|
|
|
return text.match(/(?:\d{1,3},)*\d{1,3}(?:\.\d+)?/gi);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
return text.match(/(?:\d{1,3}.)*\d{1,3}(?:\,\d+)?/gi);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const value = extractEntity(step.result);
|
|
|
|
|
|
|
|
|
|
if (value === null || value.length != 1) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite um valor monetário.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = value;
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-07 23:08:52 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "mobile") {
|
|
|
|
|
const locale = step.context.activity.locale;
|
|
|
|
|
let phoneNumber;
|
|
|
|
|
try {
|
|
|
|
|
phoneNumber = phone(step.result, 'BRA')[0]; // TODO: Use accordingly to the person.
|
|
|
|
|
phoneNumber = phoneUtil.parse(phoneNumber);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
await step.context.sendActivity(Messages[locale].validation_enter_valid_mobile);
|
|
|
|
|
|
|
|
|
|
return await step.replaceDialog('/profile_mobile', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
if (!phoneUtil.isPossibleNumber(phoneNumber)) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite um número de telefone válido.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = phoneNumber;
|
|
|
|
|
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-07 22:25:43 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "zipcode") {
|
2020-12-07 23:08:52 -03:00
|
|
|
|
const extractEntity = text => {
|
|
|
|
|
|
|
|
|
|
text = text.replace(/\-/gi, '');
|
|
|
|
|
|
|
|
|
|
if (step.context.locale === 'en') { // TODO: Change to user.
|
|
|
|
|
return text.match(/\d{8}/gi);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
return text.match(/(?:\d{1,3}.)*\d{1,3}(?:\,\d+)?/gi);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const value = extractEntity(step.result);
|
|
|
|
|
|
|
|
|
|
if (value === null || value.length != 1) {
|
|
|
|
|
await step.context.sendActivity("Por favor, digite um valor monetário.");
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = value[0];
|
|
|
|
|
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-07 22:25:43 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "menu") {
|
|
|
|
|
|
|
|
|
|
const list = step.activeDialog.state.options['args'];
|
|
|
|
|
result = null;
|
|
|
|
|
await CollectionUtil.asyncForEach(list, async item => {
|
|
|
|
|
if (GBConversationalService.kmpSearch(step.result, item) != -1) {
|
|
|
|
|
result = item;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (result === null) {
|
2020-12-07 23:08:52 -03:00
|
|
|
|
await step.context.sendActivity(`Escolha por favor um dos itens sugeridos.`);
|
2020-12-07 22:25:43 -03:00
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
2020-12-01 18:01:53 -03:00
|
|
|
|
}
|
2020-12-22 21:48:41 -03:00
|
|
|
|
else if (step.activeDialog.state.options['kind'] === "language") {
|
|
|
|
|
|
|
|
|
|
result = null;
|
|
|
|
|
|
|
|
|
|
const list = [
|
|
|
|
|
{ name: 'english', code: 'en' },
|
|
|
|
|
{ name: 'inglês', code: 'en' },
|
|
|
|
|
{ name: 'portuguese', code: 'pt' },
|
|
|
|
|
{ name: 'português', code: 'pt' },
|
|
|
|
|
{ name: 'français', code: 'fr' },
|
|
|
|
|
{ name: 'francês', code: 'fr' },
|
|
|
|
|
{ name: 'french', code: 'fr' },
|
|
|
|
|
{ name: 'spanish', code: 'es' },
|
|
|
|
|
{ name: 'espanõl', code: 'es' },
|
|
|
|
|
{ name: 'espanhol', code: 'es' },
|
|
|
|
|
{ name: 'german', code: 'de' },
|
|
|
|
|
{ name: 'deutsch', code: 'de' },
|
|
|
|
|
{ name: 'alemão', code: 'de' }
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
const text = step.context.activity['originalText'];
|
|
|
|
|
|
|
|
|
|
await CollectionUtil.asyncForEach(list, async item => {
|
|
|
|
|
if (GBConversationalService.kmpSearch(text.toLowerCase(), item.name.toLowerCase()) != -1 ||
|
|
|
|
|
GBConversationalService.kmpSearch(text.toLowerCase(), item.code.toLowerCase()) != -1) {
|
|
|
|
|
result = item.code;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (result === null) {
|
|
|
|
|
await min.conversationalService.sendText(min, step, `Escolha por favor um dos idiomas sugeridos.`);
|
|
|
|
|
return await step.replaceDialog('/hear', step.activeDialog.state.options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
2020-12-01 18:01:53 -03:00
|
|
|
|
|
|
|
|
|
const id = step.activeDialog.state.options.id;
|
|
|
|
|
if (min.cbMap[id]) {
|
|
|
|
|
const promise = min.cbMap[id].promise;
|
|
|
|
|
delete min.cbMap[id];
|
2020-11-09 17:40:34 -03:00
|
|
|
|
try {
|
2020-12-31 15:36:19 -03:00
|
|
|
|
|
2020-12-22 21:48:41 -03:00
|
|
|
|
|
2021-01-13 13:36:44 -03:00
|
|
|
|
// if (step.activeDialog.state.options.previousResolve != undefined) {
|
|
|
|
|
// step.activeDialog.state.options.previousResolve();
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
const opts = await promise(step, result);
|
|
|
|
|
return await step.replaceDialog('/hear', opts);
|
2020-11-09 17:40:34 -03:00
|
|
|
|
} catch (error) {
|
2020-11-22 11:22:55 -03:00
|
|
|
|
GBLog.error(`Error in BASIC code: ${error}`);
|
2020-11-09 17:40:34 -03:00
|
|
|
|
const locale = step.context.activity.locale;
|
2020-12-13 10:02:49 -03:00
|
|
|
|
await min.conversationalService.sendText(min, step, Messages[locale].very_sorry_about_error);
|
2020-11-09 17:40:34 -03:00
|
|
|
|
}
|
2020-05-15 14:07:30 -03:00
|
|
|
|
}
|
2021-01-13 13:36:44 -03:00
|
|
|
|
return await step.endDialog();
|
2020-11-22 11:22:55 -03:00
|
|
|
|
}
|
2018-12-01 23:01:42 -02:00
|
|
|
|
])
|
|
|
|
|
);
|
|
|
|
|
}
|
2020-08-29 15:02:19 -03:00
|
|
|
|
|
2021-01-05 07:47:48 -03:00
|
|
|
|
/**
|
|
|
|
|
* Executes the converted JavaScript from BASIC code inside execution context.
|
|
|
|
|
*/
|
2020-08-29 15:02:19 -03:00
|
|
|
|
public static async callVM(text: string, min: GBMinInstance, step: GBDialogStep, deployer: GBDeployer) {
|
2021-01-13 13:36:44 -03:00
|
|
|
|
|
2021-01-05 07:47:48 -03:00
|
|
|
|
// Creates a class DialogKeywords which is the *this* pointer
|
|
|
|
|
// in BASIC.
|
|
|
|
|
|
2020-12-27 13:30:56 -03:00
|
|
|
|
const sandbox: DialogKeywords = new DialogKeywords(min, deployer);
|
2021-01-13 13:36:44 -03:00
|
|
|
|
|
2021-01-05 07:47:48 -03:00
|
|
|
|
// Injects the .gbdialog generated code into the VM.
|
2021-01-13 13:36:44 -03:00
|
|
|
|
|
2020-08-29 15:02:19 -03:00
|
|
|
|
const context = vm.createContext(sandbox);
|
|
|
|
|
const code = min.sandBoxMap[text];
|
|
|
|
|
vm.runInContext(code, context);
|
|
|
|
|
|
2021-01-05 07:47:48 -03:00
|
|
|
|
// Tries to find the method related to this call.
|
|
|
|
|
|
2020-08-29 15:02:19 -03:00
|
|
|
|
const mainMethod = text.toLowerCase();
|
2021-01-05 07:47:48 -03:00
|
|
|
|
if (!sandbox[mainMethod]) {
|
|
|
|
|
GBLog.error(`BASIC: Associated '${mainMethod}' dialog not found. Verify if .gbdialog is correctly published.`);
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2020-08-29 15:02:19 -03:00
|
|
|
|
sandbox[mainMethod].bind(sandbox);
|
2020-12-06 16:22:34 -03:00
|
|
|
|
|
2021-01-05 07:47:48 -03:00
|
|
|
|
// Calls the function.
|
|
|
|
|
|
2020-12-06 16:22:34 -03:00
|
|
|
|
let ret = null;
|
|
|
|
|
try {
|
|
|
|
|
ret = await sandbox[mainMethod](step);
|
|
|
|
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
GBLog.error(`BASIC ERROR: ${error.message} ${error.stack}`);
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
2020-08-29 15:02:19 -03:00
|
|
|
|
}
|
2018-11-12 12:20:44 -02:00
|
|
|
|
}
|