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';
|
2018-11-30 11:55:44 -02:00
|
|
|
import { GBMinInstance, IGBCoreService } from 'botlib';
|
2018-11-26 15:54:34 -02:00
|
|
|
import * as fs from 'fs';
|
2018-11-26 14:09:09 -02:00
|
|
|
import { GBDeployer } from './GBDeployer';
|
2018-11-30 17:30:48 -02:00
|
|
|
import { TSCompiler } from './TSCompiler';
|
2019-02-23 13:17:21 -03:00
|
|
|
import DialogClass from './GBAPIService';
|
2019-02-25 08:36:43 -03:00
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
const walkPromise = require('walk-promise');
|
2018-11-30 11:55:44 -02:00
|
|
|
const logger = require('../../../src/logger');
|
2018-11-26 14:09:09 -02:00
|
|
|
const vm = require('vm');
|
2018-11-26 15:54:34 -02:00
|
|
|
const UrlJoin = require('url-join');
|
2018-11-30 17:30:48 -02:00
|
|
|
const vb2ts = require('vbscript-to-typescript/dist/converter');
|
2019-02-23 13:17:21 -03:00
|
|
|
var beautify = require('js-beautify').js;
|
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
|
|
|
|
* translate BASIC to TSand string replacements to emulate await code.
|
|
|
|
* 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
|
|
|
*/
|
|
|
|
|
2018-11-26 14:09:09 -02:00
|
|
|
export class GBVMService implements IGBCoreService {
|
2019-01-31 11:32:33 -02:00
|
|
|
private readonly script = new vm.Script();
|
2018-04-21 02:59:30 -03:00
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
public async loadDialogPackage(folder: string, min: GBMinInstance, core: IGBCoreService, deployer: GBDeployer) {
|
2019-02-25 08:36:43 -03:00
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
const files = await walkPromise(folder);
|
2019-02-25 08:36:43 -03:00
|
|
|
this.addHearDialog(min);
|
2019-02-23 13:17:21 -03:00
|
|
|
|
|
|
|
return Promise.all(
|
|
|
|
files.map(async file => {
|
|
|
|
if (
|
|
|
|
file.name.endsWith('.vbs') ||
|
|
|
|
file.name.endsWith('.vb') ||
|
|
|
|
file.name.endsWith('.basic') ||
|
|
|
|
file.name.endsWith('.bas')
|
|
|
|
) {
|
|
|
|
const mainName = file.name.replace(/\-|\./g, '');
|
2019-02-25 08:36:43 -03:00
|
|
|
min.scriptMap[file.name] = mainName;
|
2019-02-23 13:17:21 -03:00
|
|
|
|
|
|
|
const filename = UrlJoin(folder, file.name);
|
|
|
|
fs.watchFile(filename, async () => {
|
|
|
|
await this.run(filename, min, deployer, mainName);
|
|
|
|
});
|
2019-02-25 08:36:43 -03:00
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
await this.run(filename, min, deployer, mainName);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts General Bots BASIC
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @param code General Bots BASIC
|
|
|
|
*/
|
|
|
|
public convertGBASICToVBS(code: string) {
|
|
|
|
// Start and End of VB2TS tags of processing.
|
|
|
|
|
|
|
|
code = `<%\n${code}`;
|
|
|
|
|
|
|
|
// Keywords from General Bots BASIC.
|
|
|
|
|
|
|
|
code = code.replace(/(hear)\s*(\w+)/g, ($0, $1, $2) => {
|
|
|
|
return `${$2} = hear()`;
|
2018-11-30 17:30:48 -02:00
|
|
|
});
|
2019-02-23 13:17:21 -03:00
|
|
|
|
|
|
|
code = code.replace(/(wait)\s*(\d+)/g, ($0, $1, $2) => {
|
|
|
|
return `sys().wait(${$2})`;
|
|
|
|
});
|
|
|
|
|
|
|
|
code = code.replace(/(generate a password)/g, ($0, $1) => {
|
|
|
|
return 'let password = sys().generatePassword()';
|
|
|
|
});
|
|
|
|
|
|
|
|
code = code.replace(/(create a bot farm using)(\s)(.*)/g, ($0, $1, $2, $3) => {
|
|
|
|
return `sys().createABotFarmUsing (${$3})`;
|
|
|
|
});
|
|
|
|
|
|
|
|
code = code.replace(/(talk)(\s)(.*)/g, ($0, $1, $2, $3) => {
|
|
|
|
return `talk (${$3})\n`;
|
|
|
|
});
|
|
|
|
|
|
|
|
code = `${code}\n%>`;
|
|
|
|
|
|
|
|
return code;
|
2018-12-01 17:31:57 -02:00
|
|
|
}
|
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
public async run(filename: any, min: GBMinInstance, deployer: GBDeployer, mainName: string) {
|
|
|
|
// 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');
|
|
|
|
const vbsCode = await this.convertGBASICToVBS(basicCode);
|
|
|
|
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');
|
|
|
|
tsCode = tsCode.replace(/export.*\n/g, `export function ${mainName}() {`);
|
|
|
|
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;
|
2019-01-31 11:32:33 -02:00
|
|
|
const hearExp = /(\w+).*hear.*\(\)/;
|
2018-12-01 20:48:08 -02:00
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
let match1;
|
|
|
|
|
|
|
|
while ((match1 = hearExp.exec(code))) {
|
|
|
|
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 ().
|
2018-12-01 20:48:08 -02:00
|
|
|
|
2018-12-02 19:59:27 -02:00
|
|
|
parsedCode = code.substring(pos, pos + match1.index);
|
|
|
|
parsedCode += `hear (async (${variable}) => {\n`;
|
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;
|
2018-12-02 19:59:27 -02:00
|
|
|
let match2;
|
|
|
|
while ((match2 = /\{|\}/.exec(tempCode))) {
|
|
|
|
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);
|
|
|
|
parsedCode += '});\n';
|
|
|
|
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
|
|
|
|
2019-02-23 13:17:21 -03:00
|
|
|
parsedCode = beautify(parsedCode, { indent_size: 2, space_in_empty_paren: true })
|
|
|
|
fs.writeFileSync(jsfile, parsedCode);
|
2018-12-01 20:48:08 -02:00
|
|
|
|
2018-12-01 23:01:42 -02:00
|
|
|
const sandbox: DialogClass = new DialogClass(min);
|
|
|
|
const context = vm.createContext(sandbox);
|
2018-12-02 19:59:27 -02:00
|
|
|
vm.runInContext(parsedCode, context);
|
2018-12-01 23:01:42 -02:00
|
|
|
min.sandbox = sandbox;
|
|
|
|
await deployer.deployScriptToStorage(min.instanceId, filename);
|
|
|
|
logger.info(`[GBVMService] Finished loading of ${filename}`);
|
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.
|
|
|
|
|
|
|
|
code = code.replace(/sys\(\)/g, 'this.sys()');
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\btalk\b/g, ($0, $1) => {
|
|
|
|
return $1 == undefined ? 'this.talk' : $1;
|
|
|
|
});
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bhear\b/g, ($0, $1) => {
|
|
|
|
return $1 == undefined ? 'this.hear' : $1;
|
|
|
|
});
|
|
|
|
code = code.replace(/("[^"]*"|'[^']*')|\bsendEmail\b/g, ($0, $1) => {
|
|
|
|
return $1 == undefined ? 'this.sendEmail' : $1;
|
|
|
|
});
|
|
|
|
|
|
|
|
// await insertion.
|
|
|
|
|
|
|
|
code = code.replace(/this\./gm, 'await this.');
|
|
|
|
code = code.replace(/function/gm, 'async function');
|
|
|
|
|
|
|
|
return code;
|
|
|
|
}
|
|
|
|
|
2018-12-01 23:01:42 -02:00
|
|
|
private addHearDialog(min) {
|
|
|
|
min.dialogs.add(
|
|
|
|
new WaterfallDialog('/hear', [
|
|
|
|
async step => {
|
|
|
|
step.activeDialog.state.cbId = step.options['id'];
|
|
|
|
|
|
|
|
return await step.prompt('textPrompt', {});
|
|
|
|
},
|
|
|
|
async step => {
|
|
|
|
min.sandbox.context = step.context;
|
|
|
|
min.sandbox.step = step;
|
|
|
|
|
|
|
|
const cbId = step.activeDialog.state.cbId;
|
|
|
|
const cb = min.cbMap[cbId];
|
2019-01-31 11:32:33 -02:00
|
|
|
cb.bind({ step: step, context: step.context }); // TODO: Necessary or min.sandbox?
|
|
|
|
|
2018-12-02 21:39:36 -02:00
|
|
|
await step.endDialog();
|
2018-12-01 23:01:42 -02:00
|
|
|
|
2018-12-02 21:39:36 -02:00
|
|
|
return await cb(step.result);
|
2018-12-01 23:01:42 -02:00
|
|
|
}
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
2018-11-12 12:20:44 -02:00
|
|
|
}
|