botserver/packages/basic.gblib/services/GBVMService.ts

1013 lines
32 KiB
TypeScript
Raw Normal View History

2018-04-21 02:59:30 -03:00
/*****************************************************************************\
2024-01-09 17:40:48 -03:00
| ® |
| |
| |
| |
| |
2018-04-21 02:59:30 -03:00
| |
| General Bots Copyright (c) pragmatismo.cloud. All rights reserved. |
2018-04-21 02:59:30 -03:00
| 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.cloud. |
2018-04-21 02:59:30 -03:00
| 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. |
| |
\*****************************************************************************/
'use strict';
import { GBMinInstance, GBService, IGBCoreService, GBLog } from 'botlib';
2024-09-15 14:41:56 -03:00
import fs from 'fs/promises';
2024-08-21 13:09:50 -03:00
import * as ji from 'just-indent';
import { GBServer } from '../../../src/app.js';
import { GBDeployer } from '../../core.gbapp/services/GBDeployer.js';
import { CollectionUtil } from 'pragmatismo-io-framework';
import { ScheduleServices } from './ScheduleServices.js';
import { GBConfigService } from '../../core.gbapp/services/GBConfigService.js';
import urlJoin from 'url-join';
import { NodeVM, VMScript } from 'vm2';
import { createVm2Pool } from './vm2-process/index.js';
2024-09-07 18:13:36 -03:00
import { watch } from 'fs';
import textract from 'textract';
import walkPromise from 'walk-promise';
import child_process from 'child_process';
2024-09-06 15:30:03 -03:00
import path from 'path';
import { GBAdminService } from '../../admin.gbapp/services/GBAdminService.js';
import { DialogKeywords } from './DialogKeywords.js';
import { KeywordsExpressions } from './KeywordsExpressions.js';
2023-02-12 14:31:21 -03:00
import { GBLogEx } from '../../core.gbapp/services/GBLogEx.js';
import { GuaribasUser } from '../../security.gbapp/models/index.js';
import { SystemKeywords } from './SystemKeywords.js';
import { Sequelize, QueryTypes } from '@sequelize/core';
2024-08-21 13:09:50 -03:00
import { z } from 'zod';
import { zodToJsonSchema } from 'zod-to-json-schema';
import { GBUtil } from '../../../src/util.js';
2018-11-11 19:09:18 -02:00
/**
2022-11-19 23:34:58 -03:00
* @fileoverview Decision was to priorize security(isolation) and debugging,
* over a beautiful BASIC transpiler (to be done).
2018-11-11 19:09:18 -02:00
*/
/**
* Basic services for BASIC manipulation.
*/
export class GBVMService extends GBService {
public static API_PORT = 1111;
public async loadDialogPackage(folder: string, min: GBMinInstance, core: IGBCoreService, deployer: GBDeployer) {
2024-09-06 15:30:03 -03:00
const ignore = path.join('work', GBUtil.getGBAIPath(min.botId, 'gbdialog'), 'node_modules');
2024-08-21 13:09:50 -03:00
const files = await walkPromise(folder, { ignore: [ignore] });
await CollectionUtil.asyncForEach(files, async file => {
if (!file) {
return;
}
let filename: string = file.name;
2024-08-21 13:09:50 -03:00
filename = await this.loadDialog(filename, folder, min);
});
}
public static compare(obj1, obj2) {
//check for obj2 overlapping props
if (!Object.keys(obj2).every(key => obj1.hasOwnProperty(key))) {
return false;
}
//check every key for being same
return Object.keys(obj1).every(function (key) {
//if object
2024-08-21 13:09:50 -03:00
if (typeof obj1[key] == 'object' && typeof obj2[key] == 'object') {
//recursively check
return GBVMService.compare(obj1[key], obj2[key]);
} else {
//do the normal compare
return obj1[key] === obj2[key];
}
});
}
public async loadDialog(filename: string, folder: string, min: GBMinInstance) {
2024-08-21 13:09:50 -03:00
const isWord = filename.endsWith('.docx');
if (
!(
isWord ||
filename.endsWith('.vbs') ||
filename.endsWith('.vb') ||
filename.endsWith('.vba') ||
filename.endsWith('.bas') ||
filename.endsWith('.basic')
)
) {
return;
}
2024-08-20 19:12:57 -03:00
const wordFile = filename;
2024-08-21 13:09:50 -03:00
const vbsFile = isWord ? filename.substr(0, filename.indexOf('docx')) + 'vbs' : filename;
const fullVbsFile = urlJoin(folder, vbsFile);
2024-09-15 14:41:56 -03:00
const docxStat = await fs.stat(urlJoin(folder, wordFile));
const interval = 3000; // If compiled is older 30 seconds, then recompile.
let writeVBS = true;
2024-04-01 13:00:47 -03:00
// TODO: #412.
// const subscription = {
// changeType: 'created,updated',
// notificationUrl: 'https://webhook.azurewebsites.net/notificationClient',
// lifecycleNotificationUrl: 'https://webhook.azurewebsites.net/api/lifecycleNotifications',
// resource: '/me/mailfolders(\'inbox\')/messages',
// expirationDateTime: '2016-03-20T11:00:00.0000000Z',
// clientState: 'SecretClientState'
// };
// let { baseUrl, client } = await GBDeployer.internalGetDriveClient(min);
2024-08-21 13:09:50 -03:00
2024-04-01 13:00:47 -03:00
// await client.api('/subscriptions')
// .post(subscription);
2024-04-01 11:55:45 -03:00
2024-09-07 18:13:36 -03:00
if (await GBUtil.exists(fullVbsFile)) {
2024-09-15 14:41:56 -03:00
const vbsStat = await fs.stat(fullVbsFile);
if (docxStat['mtimeMs'] < vbsStat['mtimeMs'] + interval) {
writeVBS = false;
}
}
filename = vbsFile;
let mainName = GBVMService.getMethodNameFromVBSFilename(filename);
min.scriptMap[filename] = mainName;
2024-08-20 19:12:57 -03:00
if (writeVBS && GBConfigService.get('STORAGE_NAME')) {
let text = await this.getTextFromWord(folder, wordFile);
2024-08-20 19:12:57 -03:00
// Write VBS file without pragma keywords.
2024-09-15 14:41:56 -03:00
await fs.writeFile(urlJoin(folder, vbsFile), text);
2024-08-20 19:12:57 -03:00
}
2024-03-16 21:36:03 -03:00
2024-08-20 19:12:57 -03:00
// Process node_modules install.
2024-03-16 21:36:03 -03:00
2024-09-07 18:13:36 -03:00
await this.processNodeModules(folder, min);
2024-03-16 21:36:03 -03:00
2024-08-20 19:12:57 -03:00
// Hot swap for .vbs files.
2024-03-16 21:36:03 -03:00
2024-08-20 19:12:57 -03:00
const fullFilename = urlJoin(folder, filename);
if (process.env.DEV_HOTSWAP) {
2024-09-07 18:13:36 -03:00
watch(fullFilename, async () => {
2024-08-20 19:12:57 -03:00
await this.translateBASIC(mainName, fullFilename, min);
2024-09-07 18:13:36 -03:00
const parsedCode: string = await fs.readFile(jsfile, 'utf8');
2024-08-20 19:12:57 -03:00
min.sandBoxMap[mainName.toLowerCase().trim()] = parsedCode;
});
}
2024-09-15 14:41:56 -03:00
const compiledAt = await fs.stat(fullFilename);
2024-08-20 19:12:57 -03:00
const jsfile = urlJoin(folder, `${filename}.js`);
2024-09-07 18:13:36 -03:00
if (await GBUtil.exists(jsfile)) {
2024-09-15 14:41:56 -03:00
const jsStat = await fs.stat(jsfile);
2024-08-21 13:26:40 -03:00
const interval = 1000; // If compiled is older 1 seconds, then recompile.
2024-08-20 19:12:57 -03:00
if (compiledAt.isFile() && compiledAt['mtimeMs'] > jsStat['mtimeMs'] + interval) {
await this.translateBASIC(mainName, fullFilename, min);
}
} else {
await this.translateBASIC(mainName, fullFilename, min);
}
2024-08-20 19:12:57 -03:00
// Syncronizes Database Objects with the ones returned from "Word".
2024-09-07 18:13:36 -03:00
await this.syncStorageFromTABLE(folder, filename, min, mainName);
2024-08-20 19:12:57 -03:00
2024-09-07 18:13:36 -03:00
const parsedCode: string = await fs.readFile(jsfile, 'utf8');
2024-08-20 19:12:57 -03:00
min.sandBoxMap[mainName.toLowerCase().trim()] = parsedCode;
return filename;
}
2024-09-07 18:13:36 -03:00
private async processNodeModules(folder: string, min: GBMinInstance) {
const node_modules = urlJoin(process.env.PWD, folder, 'node_modules');
2024-09-15 14:41:56 -03:00
if (!(await GBUtil.exists(node_modules))) {
const packageJson = `
{
"name": "${min.botId}.gbdialog",
"version": "1.0.0",
"description": "${min.botId} transpiled .gbdialog",
"author": "${min.botId} owner.",
"license": "ISC",
"dependencies": {
"yaml": "2.4.2",
"encoding": "0.1.13",
"isomorphic-fetch": "3.0.0",
"punycode": "2.1.1",
"@push-rpc/core": "1.8.2",
"@push-rpc/http": "1.8.2",
"vm2": "3.9.11",
"async-retry": "1.3.3"
}
}`;
2024-09-15 14:41:56 -03:00
await fs.writeFile(urlJoin(folder, 'package.json'), packageJson);
2024-09-12 15:05:32 -03:00
GBLogEx.info(min, `Installing node_modules...`);
const npmPath = urlJoin(process.env.PWD, 'node_modules', '.bin', 'npm');
2024-09-07 18:13:36 -03:00
child_process.exec(`${npmPath} install`, { cwd: folder });
}
2024-08-20 19:12:57 -03:00
}
2024-08-31 03:52:37 -03:00
public static async loadConnections(min) {
2024-08-29 19:53:56 -03:00
// Loads storage custom connections.
2024-09-07 00:08:23 -03:00
const packagePath = GBUtil.getGBAIPath(min.botId, null);
const filePath = path.join('work', packagePath, 'connections.json');
2024-08-29 19:53:56 -03:00
let connections = [];
2024-09-07 18:13:36 -03:00
if (await GBUtil.exists(filePath)) {
connections = JSON.parse(await fs.readFile(filePath, 'utf8'));
2024-08-29 19:53:56 -03:00
}
connections.forEach(async con => {
const connectionName = con['name'];
const dialect = con['storageDriver'];
const host = con['storageServer'];
const port = con['storagePort'];
const storageName = con['storageName'];
const username = con['storageUsername'];
const password = con['storagePassword'];
const logging: boolean | Function =
GBConfigService.get('STORAGE_LOGGING') === 'true'
? (str: string): void => {
GBLogEx.info(min, str);
}
: false;
const encrypt: boolean = GBConfigService.get('STORAGE_ENCRYPT') === 'true';
const acquire = parseInt(GBConfigService.get('STORAGE_ACQUIRE_TIMEOUT'));
const sequelizeOptions = {
define: {
charset: 'utf8',
collate: 'utf8_general_ci',
freezeTableName: true,
timestamps: false
},
host: host,
port: port,
logging: logging as boolean,
dialect: dialect,
quoteIdentifiers: false, // set case-insensitive
dialectOptions: {
options: {
trustServerCertificate: true,
encrypt: encrypt,
requestTimeout: 120 * 1000
}
},
pool: {
max: 5,
min: 0,
idle: 10000,
evict: 10000,
acquire: acquire
}
};
if (!min[connectionName]) {
2024-09-12 15:05:32 -03:00
GBLogEx.info(min, `Loading data connection ${connectionName}...`);
2024-08-29 19:53:56 -03:00
min[connectionName] = new Sequelize(storageName, username, password, sequelizeOptions);
min[connectionName]['gbconnection'] = con;
}
});
}
2024-09-07 18:13:36 -03:00
private async syncStorageFromTABLE(folder: string, filename: string, min: GBMinInstance, mainName: string) {
const tablesFile = urlJoin(folder, `${filename}.tables.json`);
let sync = false;
2024-09-07 18:13:36 -03:00
if (await GBUtil.exists(tablesFile)) {
2023-10-02 16:22:51 -03:00
const minBoot = GBServer.globals.minBoot;
2024-09-07 18:13:36 -03:00
const tableDef = JSON.parse(await fs.readFile(tablesFile, 'utf8')) as any;
const getTypeBasedOnCondition = (t, size) => {
if (1) {
switch (t) {
case 'string':
return `varchar(${size})`;
case 'guid':
return 'UUID';
case 'key':
return `varchar(${size})`;
case 'number':
return 'BIGINT';
case 'integer':
return 'INTEGER';
case 'double':
return 'FLOAT';
case 'float':
return 'FLOAT';
case 'date':
return 'DATE';
case 'boolean':
return 'BOOLEAN';
default:
return { type: 'TABLE', name: t };
}
} else {
switch (t) {
case 'string':
return { key: 'STRING' };
case 'guid':
return { key: 'UUID' };
case 'key':
return { key: 'STRING' }; // Assuming key is a string data type
case 'number':
return { key: 'BIGINT' };
case 'integer':
return { key: 'INTEGER' };
case 'double':
return { key: 'FLOAT' };
case 'float':
return { key: 'FLOAT' };
case 'date':
return { key: 'DATE' };
case 'boolean':
return { key: 'BOOLEAN' };
default:
return { key: 'TABLE', name: t };
}
}
};
const associations = [];
2024-08-21 13:09:50 -03:00
const shouldSync = min.core.getParam<boolean>(min.instance, 'Synchronize Database', false);
2024-08-21 13:09:50 -03:00
tableDef.forEach(async t => {
const tableName = t.name.trim();
2023-11-28 14:13:44 -03:00
// Determines autorelationship.
Object.keys(t.fields).forEach(key => {
let obj = t.fields[key];
obj.type = getTypeBasedOnCondition(obj.type, obj.size);
2024-08-21 13:09:50 -03:00
if (obj.type.key === 'TABLE') {
obj.type.key = 'BIGINT';
associations.push({ from: tableName, to: obj.type.name });
}
});
// Custom connection for TABLE.
const connectionName = t.connection;
2024-08-26 21:47:57 -03:00
let con = min[connectionName];
2023-11-28 14:13:44 -03:00
if (!con) {
2024-08-31 03:52:37 -03:00
GBLogEx.debug(min, `Invalid connection specified: ${min.bot} ${tableName} ${connectionName}.`);
} else {
2024-08-31 03:52:37 -03:00
// Field checking, syncs if there is any difference.
2024-08-31 03:52:37 -03:00
const seq = con ? con : minBoot.core.sequelize;
if (seq) {
const model = seq.models[tableName];
if (model) {
2024-08-31 03:52:37 -03:00
// Except Id, checks if has same number of fields.
2024-08-31 03:52:37 -03:00
let equals = 0;
Object.keys(t.fields).forEach(key => {
let obj1 = t.fields[key];
let obj2 = model['fieldRawAttributesMap'][key];
if (key !== 'id') {
if (obj1 && obj2) {
equals++;
}
}
2024-08-31 03:52:37 -03:00
});
2024-08-31 03:52:37 -03:00
if (equals != Object.keys(t.fields).length) {
sync = true;
}
}
seq.define(tableName, t.fields);
2024-08-31 03:52:37 -03:00
// New table checking, if needs sync.
let tables;
const dialect = con.dialect.name;
tables = await GBUtil.listTables(dialect, seq);
2023-10-02 16:22:51 -03:00
2024-08-31 03:52:37 -03:00
let found = false;
tables.forEach(storageTable => {
if (storageTable['table_name'] === tableName) {
found = true;
}
});
2024-08-31 03:52:37 -03:00
sync = sync ? sync : !found;
2024-10-17 14:00:42 -03:00
// Do not erase tables in case of an error in collection retrieval.
if (tables.length === 0){
sync = false;
}
2024-08-31 03:52:37 -03:00
associations.forEach(e => {
const from = seq.models[e.from];
const to = seq.models[e.to];
2024-08-31 03:52:37 -03:00
try {
to.hasMany(from);
} catch (error) {
throw new Error(
`Invalid relationship in ${mainName}: from ${e.from} to ${e.to} (${min.botId})... ${error.message}`
);
}
});
2024-08-31 03:52:37 -03:00
if (sync && shouldSync) {
GBLogEx.info(min, `Syncing changes for TABLE ${connectionName} ${tableName} keyword (${min.botId})...`);
await seq.sync({
alter: true,
force: false // Keep it false due to data loss danger.
});
GBLogEx.info(min, `Done sync for ${min.botId} ${connectionName} ${tableName} storage table...`);
}
}
}
});
2023-10-02 16:22:51 -03:00
}
}
public async translateBASIC(mainName, filename: any, min: GBMinInstance) {
// Converts General Bots BASIC into regular VBS
2024-09-07 18:13:36 -03:00
let basicCode: string = await fs.readFile(filename, 'utf8');
basicCode = GBVMService.normalizeQuotes(basicCode);
2024-09-15 14:41:56 -03:00
2024-08-20 19:12:57 -03:00
// Pre process SET SCHEDULE calls.
const schedules = GBVMService.getSetScheduleKeywordArgs(basicCode);
const s = new ScheduleServices();
await s.deleteScheduleIfAny(min, mainName);
let i = 1;
2024-08-21 13:09:50 -03:00
await CollectionUtil.asyncForEach(schedules, async syntax => {
2024-08-20 19:12:57 -03:00
if (s) {
2024-08-21 13:09:50 -03:00
await s.createOrUpdateSchedule(min, syntax, `${mainName};${i++}`);
2024-08-20 19:12:57 -03:00
}
});
basicCode = basicCode.replace(/^\s*SET SCHEDULE (.*)/gim, '');
// Process INCLUDE keyword to include another
// dialog inside the dialog.
let include = null;
do {
2022-11-19 23:34:58 -03:00
include = /^include\b(.*)$/gim.exec(basicCode);
if (include) {
let includeName = include[1].trim();
2024-09-06 15:30:03 -03:00
includeName = path.join(path.dirname(filename), includeName);
2022-11-19 23:34:58 -03:00
includeName = includeName.substr(0, includeName.lastIndexOf('.')) + '.vbs';
// To use include, two /publish will be necessary (for now)
// because of alphabet order may raise not found errors.
2024-09-07 18:13:36 -03:00
let includeCode: string = await fs.readFile(includeName, 'utf8');
2022-11-19 23:34:58 -03:00
basicCode = basicCode.replace(/^include\b.*$/gim, includeCode);
}
} while (include);
let { code, map, metadata, tasks } = await this.convert(filename, mainName, basicCode);
// Generates function JSON metadata to be used later.
const jsonFile = `${filename}.json`;
2024-09-15 14:41:56 -03:00
await fs.writeFile(jsonFile, JSON.stringify(metadata));
2022-11-12 17:17:14 -03:00
const mapFile = `${filename}.map`;
2024-09-15 14:41:56 -03:00
await fs.writeFile(mapFile, JSON.stringify(map));
2023-09-30 20:43:39 -03:00
// Execute off-line code tasks
await this.executeTasks(min, tasks);
2023-09-30 20:43:39 -03:00
// Run JS into the GB context.
const jsfile: string = `${filename}.js`;
2024-09-15 16:30:03 -03:00
const template = (await fs.readFile('./vm-inject.js')).toString();
code = template.replace('//##INJECTED_CODE_HERE', code );
2024-09-15 18:30:08 -03:00
code = code.replace('//##INJECTED_HEADER', `port=${GBVMService.API_PORT}; botId='${min.botId}';` );
code = ji.default(code, ' ');
2024-09-15 14:41:56 -03:00
await fs.writeFile(jsfile, code);
2023-09-30 20:43:39 -03:00
}
private async executeTasks(min, tasks) {
2023-09-30 20:43:39 -03:00
for (let i = 0; i < tasks.length; i++) {
const task = tasks[i];
2023-10-02 16:22:51 -03:00
if (task.kind === 'writeTableDefinition') {
// Creates an empty object that will receive Sequelize fields.
2023-09-30 20:43:39 -03:00
const tablesFile = `${task.file}.tables.json`;
2024-09-15 14:41:56 -03:00
await fs.writeFile(tablesFile, JSON.stringify(task.tables));
2023-09-30 20:43:39 -03:00
}
}
}
public static getMethodNameFromVBSFilename(filename: string) {
let mainName = filename.replace(/\s*|\-/gim, '').split('.')[0];
return mainName.toLowerCase();
}
2024-03-16 21:36:03 -03:00
public static getSetScheduleKeywordArgs(code) {
if (!code) return [];
const lines = code.split(/\n/);
const results = [];
lines.forEach(line => {
if (line.trim()) {
2024-08-21 13:09:50 -03:00
const keyword = /^\s*SET SCHEDULE (.*)/gi;
2024-03-16 21:36:03 -03:00
let result: any = keyword.exec(line);
if (result) {
2024-08-21 13:09:50 -03:00
result = result[1].replace(/\`|\"|\'/, '');
2024-03-16 21:36:03 -03:00
result = result.trim();
results.push(result);
}
}
});
2024-03-16 21:36:03 -03:00
return results;
}
2024-08-21 13:09:50 -03:00
private async getTextFromWord(folder: string, filename: string) {
return new Promise<string>(async (resolve, reject) => {
2024-09-07 00:08:23 -03:00
const filePath = urlJoin(folder, filename);
2024-09-07 18:13:36 -03:00
textract.fromFileWithPath(filePath, { preserveLineBreaks: true }, async (error, text) => {
if (error) {
if (error.message.startsWith('File not correctly recognized as zip file')) {
2024-09-07 18:13:36 -03:00
text = await fs.readFile(filePath, 'utf8');
} else {
reject(error);
}
}
resolve(text);
});
});
}
public static normalizeQuotes(text: any) {
text = text.replace(/\"/gm, '`');
text = text.replace(/\¨/gm, '`');
text = text.replace(/\“/gm, '`');
text = text.replace(/\”/gm, '`');
2023-03-19 20:09:54 -03:00
text = text.replace(/\/gm, "'");
text = text.replace(/\/gm, "'");
return text;
}
public static getMetadata(mainName: string, propertiesText: string[][], description: string) {
2024-03-06 14:38:37 -03:00
let properties = {};
if (!propertiesText || !description) {
2024-08-21 13:09:50 -03:00
return {};
2024-03-04 20:05:56 -03:00
}
2024-09-15 14:41:56 -03:00
const getType = (asClause: string) => {
2024-03-06 14:38:37 -03:00
asClause = asClause.trim().toUpperCase();
2024-09-15 14:41:56 -03:00
2024-03-06 14:38:37 -03:00
if (asClause.indexOf('STRING') !== -1) {
2024-03-04 20:05:56 -03:00
return 'string';
2024-08-21 13:09:50 -03:00
} else if (asClause.indexOf('OBJECT') !== -1) {
2024-03-04 20:05:56 -03:00
return 'object';
2024-08-21 13:09:50 -03:00
} else if (asClause.indexOf('INTEGER') !== -1 || asClause.indexOf('NUMBER') !== -1) {
2024-03-04 20:05:56 -03:00
return 'number';
} else {
return 'enum';
}
};
2024-09-15 14:41:56 -03:00
2024-03-04 20:05:56 -03:00
for (let i = 0; i < propertiesText.length; i++) {
const propertiesExp = propertiesText[i];
const t = getType(propertiesExp[2]);
let element;
2024-09-15 14:41:56 -03:00
const description = propertiesExp[4]?.trim();
2024-03-04 20:05:56 -03:00
if (t === 'enum') {
const list = propertiesExp[2] as any;
element = z.enum(list.split(','));
2024-03-04 20:05:56 -03:00
} else if (t === 'string') {
2024-09-15 14:41:56 -03:00
element = z.string({ description: description });
2024-03-04 20:05:56 -03:00
} else if (t === 'object') {
2024-09-15 14:41:56 -03:00
element = z.string({ description: description }); // Assuming 'object' is represented as a string here
2024-03-04 20:05:56 -03:00
} else if (t === 'number') {
2024-09-15 14:41:56 -03:00
element = z.number({ description: description });
2024-03-06 14:38:37 -03:00
} else {
GBLog.warn(`Element type invalid specified on .docx: ${propertiesExp[0]}`);
}
2024-09-04 00:18:19 -03:00
2024-03-04 20:05:56 -03:00
element['type'] = t;
2024-03-06 14:38:37 -03:00
properties[propertiesExp[1].trim()] = element;
}
2024-09-15 14:41:56 -03:00
const json = {
2024-08-21 13:09:50 -03:00
type: 'function',
2024-03-04 20:05:56 -03:00
function: {
name: mainName,
description: description ? description : '',
2024-09-04 00:18:19 -03:00
schema: zodToJsonSchema(z.object(properties))
},
arguments: propertiesText.reduce((acc, prop) => {
2024-09-15 14:41:56 -03:00
acc[prop[1].trim()] = prop[3]?.trim(); // Assuming value is in the 3rd index
return acc;
}, {})
2024-08-21 13:09:50 -03:00
};
2024-09-15 14:41:56 -03:00
return json;
}
2024-09-15 14:41:56 -03:00
2023-09-30 20:43:39 -03:00
public async parseField(line) {
let required = line.indexOf('*') !== -1;
let unique = /\bunique\b/gi.test(line);
2024-03-04 20:05:56 -03:00
let primaryKey = /\bkey\b/gi.test(line);
let autoIncrement = /\bauto\b/gi.test(line);
2024-03-04 20:05:56 -03:00
if (primaryKey) {
autoIncrement = true;
unique = true;
required = true;
}
2024-03-04 20:05:56 -03:00
2023-09-30 20:43:39 -03:00
line = line.replace('*', '');
const fieldRegExp = /^\s*(\w+)\s*(\w+)(?:\((.*)\))?/gim;
2023-09-30 20:43:39 -03:00
2023-10-02 16:22:51 -03:00
let reg = fieldRegExp.exec(line);
const name = reg[1];
const t = reg[2];
2024-03-04 20:05:56 -03:00
let definition = {
allowNull: !required,
2024-08-21 13:09:50 -03:00
unique: unique,
primaryKey: primaryKey,
2024-03-04 20:05:56 -03:00
autoIncrement: autoIncrement
};
definition['type'] = t;
if (reg[3]) {
definition['size'] = Number.parseInt(reg[3] === 'max' ? '4000' : reg[3]);
2023-09-30 20:43:39 -03:00
}
return { name, definition };
2023-09-30 20:43:39 -03:00
}
/**
* Converts General Bots BASIC
*
*
* @param code General Bots BASIC
*/
public async convert(filename: string, mainName: string, code: string) {
// Start and End of VB2TS tags of processing.
2023-02-12 14:31:21 -03:00
code = process.env.ENABLE_AUTH ? `hear GBLogExin as login\n${code}` : code;
var lines = code.split('\n');
const keywords = KeywordsExpressions.getKeywords();
let current = 41;
const map = {};
let properties = [];
let description;
2023-09-30 20:43:39 -03:00
let table = null; // Used for TABLE keyword.
2024-02-18 01:15:47 -03:00
let talk = null;
2024-03-11 13:30:11 -03:00
let systemPrompt = null;
let connection = null;
2023-09-30 20:43:39 -03:00
const tasks = [];
let fields = {};
let tables = [];
const outputLines = [];
let emmitIndex = 1;
for (let i = 1; i <= lines.length; i++) {
let line = lines[i - 1];
// Remove lines before statements.
line = line.replace(/^\s*\d+\s*/gi, '');
2024-03-11 13:30:11 -03:00
if (!table && !talk && !systemPrompt) {
for (let j = 0; j < keywords.length; j++) {
line = line.replace(keywords[j][0], keywords[j][1]); // TODO: Investigate delay here.
}
}
2023-09-30 20:43:39 -03:00
// Pre-process "off-line" static KEYWORDS.
2023-09-30 20:43:39 -03:00
let emmit = true;
const params = /^\s*PARAM\s*(.*)\s*AS\s*(.*)\s*LIKE\s*(.*)\s*DESCRIPTION\s*(.*)/gim;
const param = params.exec(line);
if (param) {
properties.push(param);
2023-09-30 20:43:39 -03:00
emmit = false;
}
2024-03-06 14:38:37 -03:00
const descriptionKeyword = /^\s*DESCRIPTION\s(.*)/gim;
let descriptionReg = descriptionKeyword.exec(line);
if (descriptionReg) {
description = descriptionReg[1];
2023-09-30 20:43:39 -03:00
emmit = false;
}
2024-03-11 13:30:11 -03:00
const endSystemPromptKeyword = /^\s*END SYSTEM PROMPT\s*/gim;
let endSystemPromptReg = endSystemPromptKeyword.exec(line);
if (endSystemPromptReg && systemPrompt) {
line = systemPrompt + '`})';
2024-03-11 13:30:11 -03:00
systemPrompt = null;
emmit = true;
}
2024-02-18 01:15:47 -03:00
const endTalkKeyword = /^\s*END TALK\s*/gim;
let endTalkReg = endTalkKeyword.exec(line);
if (endTalkReg && talk) {
line = talk + '`})';
2024-03-04 20:05:56 -03:00
2024-02-18 01:15:47 -03:00
talk = null;
emmit = true;
}
const endTableKeyword = /^\s*END TABLE\s*/gim;
2023-09-30 20:43:39 -03:00
let endTableReg = endTableKeyword.exec(line);
if (endTableReg && table) {
tables.push({
2024-08-21 13:09:50 -03:00
name: table,
fields: fields,
connection: connection
2023-09-30 20:43:39 -03:00
});
fields = {};
2023-09-30 20:43:39 -03:00
table = null;
connection = null;
2023-09-30 20:43:39 -03:00
emmit = false;
}
2024-02-18 01:15:47 -03:00
// Inside BEGIN TALK
if (talk) {
talk += line + '\\n';
emmit = false;
}
2024-03-11 13:30:11 -03:00
// Inside BEGIN SYSTEM PROMPT
if (systemPrompt) {
systemPrompt += line + '\\n';
emmit = false;
}
// Inside BEGIN/END table pair containing FIELDS.
2023-09-30 20:43:39 -03:00
if (table && line.trim() !== '') {
const field = await this.parseField(line);
fields[field['name']] = field.definition;
emmit = false;
}
2023-09-30 20:43:39 -03:00
const tableKeyword = /^\s*TABLE\s*(.*)\s*ON\s*(.*)/gim;
let tableReg = tableKeyword.exec(line);
if (tableReg && !table) {
table = tableReg[1];
connection = tableReg[2];
emmit = false;
}
2024-02-18 01:15:47 -03:00
const talkKeyword = /^\s*BEGIN TALK\s*/gim;
let talkReg = talkKeyword.exec(line);
if (talkReg && !talk) {
2024-08-21 13:09:50 -03:00
talk = 'await dk.talk ({pid: pid, text: `';
2024-02-18 01:15:47 -03:00
emmit = false;
}
2024-03-11 13:30:11 -03:00
const systemPromptKeyword = /^\s*BEGIN SYSTEM PROMPT\s*/gim;
let systemPromptReg = systemPromptKeyword.exec(line);
if (systemPromptReg && !systemPrompt) {
2024-08-21 13:09:50 -03:00
systemPrompt = 'await sys.setSystemPrompt ({pid: pid, text: `';
2024-03-11 13:30:11 -03:00
emmit = false;
}
// Add additional lines returned from replacement.
let add = emmit ? line.split(/\r\n|\r|\n/).length : 0;
current = current + (add ? add : 0);
2024-03-04 20:05:56 -03:00
if (emmit) {
emmitIndex++;
map[emmitIndex] = current;
outputLines[emmitIndex - 1] = line;
}
2022-11-12 17:17:14 -03:00
}
2022-11-11 21:35:05 -03:00
if (tables) {
tasks.push({
2024-08-21 13:09:50 -03:00
kind: 'writeTableDefinition',
file: filename,
tables
});
}
code = `${outputLines.join('\n')}\n`;
let metadata = GBVMService.getMetadata(mainName, properties, description);
2024-03-11 13:30:11 -03:00
return { code, map, metadata, tasks, systemPrompt };
2022-11-11 21:35:05 -03:00
}
/**
* Executes the converted JavaScript from BASIC code inside execution context.
*/
2024-08-21 13:09:50 -03:00
public static async callVM(text: string, min: GBMinInstance, step, pid, debug: boolean = false, params = []) {
// Creates a class DialogKeywords which is the *this* pointer
// in BASIC.
const sandbox = {};
const contentLocale = min.core.getParam<string>(
min.instance,
'Default Content Language',
GBConfigService.get('DEFAULT_CONTENT_LANGUAGE')
);
2024-09-04 00:18:19 -03:00
let variables = {};
// These variables will be automatically be available as normal BASIC variables.
try {
2024-08-21 13:09:50 -03:00
variables['aadToken'] = await (min.adminService as any)['acquireElevatedToken'](min.instance.instanceId, false);
} catch (error) {
variables['aadToken'] = 'ERROR: Configure /setupSecurity before using aadToken variable.';
}
// Adds all .gbot params as variables.
const gbotConfig = JSON.parse(min.instance.params);
let keys = Object.keys(gbotConfig);
for (let j = 0; j < keys.length; j++) {
const v = keys[j].replace(/\s/gi, '');
variables[v] = gbotConfig[keys[j]];
}
// Auto-NLP generates BASIC variables related to entities.
2024-09-25 16:54:52 -03:00
if (step?.context?.activity.originalText && min['nerEngine']) {
const result = await min['nerEngine'].process(step.context.activity.originalText);
2022-06-07 18:37:29 -03:00
for (let i = 0; i < result.entities.length; i++) {
const v = result.entities[i];
const variableName = `${v.entity}`;
variables[variableName] = v.option ? v.option : v.sourceText;
}
}
2023-12-15 11:59:24 -03:00
// Adds params as variables to be added later as global objects.
keys = Object.keys(params);
for (let j = 0; j < keys.length; j++) {
variables[keys[j]] = params[keys[j]];
}
const botId = min.botId;
2024-09-07 00:08:23 -03:00
const packagePath = GBUtil.getGBAIPath(min.botId, `gbdialog`);
const gbdialogPath = urlJoin(process.cwd(), 'work', packagePath);
2024-09-04 00:18:19 -03:00
const scriptFilePath = urlJoin(gbdialogPath, `${text}.js`);
let code = min.sandBoxMap[text];
2024-09-25 16:54:52 -03:00
const channel = step?.context ? step.context.activity.channelId : 'web';
2024-03-04 20:05:56 -03:00
const dk = new DialogKeywords();
const sys = new SystemKeywords();
await dk.setFilter({ pid: pid, value: null });
// Find all tokens in .gbot Config.
2024-03-04 20:05:56 -03:00
const strFind = ' Client ID';
const tokens = await min.core['findParam'](min.instance, strFind);
let tokensList = [];
await CollectionUtil.asyncForEach(tokens, async t => {
const tokenName = t.replace(strFind, '');
tokensList.push(tokenName);
});
sandbox['tokens'] = tokensList.join(',');
sandbox['variables'] = variables;
sandbox['id'] = sys.getRandomId();
2023-02-16 10:27:18 -03:00
sandbox['username'] = await dk.userName({ pid });
sandbox['mobile'] = await dk.userMobile({ pid });
sandbox['from'] = await dk.userMobile({ pid });
sandbox['ENTER'] = String.fromCharCode(13);
sandbox['headers'] = {};
sandbox['httpUsername'] = '';
sandbox['httpPs'] = '';
sandbox['pid'] = pid;
sandbox['contentLocale'] = contentLocale;
sandbox['callTimeout'] = 60 * 60 * 24 * 1000;
sandbox['channel'] = channel;
2023-03-27 17:38:31 -03:00
sandbox['today'] = await dk.getToday({ pid });
sandbox['now'] = await dk.getNow({ pid });
sandbox['returnValue'] = null;
2023-02-16 10:27:18 -03:00
let result;
try {
2024-08-28 19:42:12 -03:00
if (!GBConfigService.get('GBVM')) {
return await (async () => {
return await new Promise((resolve) => {
sandbox['resolve'] = resolve;
2024-04-01 11:55:45 -03:00
// TODO: #411 sandbox['reject'] = reject;
2024-08-21 13:09:50 -03:00
sandbox['reject'] = () => {};
2024-04-01 11:55:45 -03:00
const vm1 = new NodeVM({
allowAsync: true,
sandbox: sandbox,
console: 'inherit',
wrapper: 'commonjs',
require: {
builtin: ['stream', 'http', 'https', 'url', 'zlib', 'net', 'tls', 'crypto'],
root: ['./'],
external: true,
context: 'sandbox'
}
});
2024-09-04 00:18:19 -03:00
const s = new VMScript(code, { filename: scriptFilePath });
result = vm1.run(s);
});
})();
2023-02-16 10:27:18 -03:00
} else {
const runnerPath = urlJoin(
process.cwd(),
'dist',
'packages',
'basic.gblib',
'services',
'vm2-process',
'vm2ProcessRunner.js'
);
2022-11-12 21:33:45 -03:00
const { run } = createVm2Pool({
min: 0,
max: 0,
2022-11-13 22:56:09 -03:00
debug: debug,
2024-09-04 00:18:19 -03:00
// debuggerport: GBVMService.DEBUGGER_PORT,
2022-11-11 21:35:05 -03:00
botId: botId,
cpu: 100,
memory: 50000,
time: 60 * 60 * 24 * 14,
2024-09-04 00:18:19 -03:00
cwd: gbdialogPath,
script: runnerPath
});
2024-09-15 14:41:56 -03:00
result = await run(code, Object.assign(sandbox, { filename: scriptFilePath }));
2023-02-16 10:27:18 -03:00
}
} catch (error) {
throw new Error(`BASIC RUNTIME ERR: ${error.message ? error.message : error}\n Stack:${error.stack}`);
}
}
2023-08-21 13:21:49 -03:00
2024-08-21 13:09:50 -03:00
public static createProcessInfo(
user: GuaribasUser,
min: GBMinInstance,
channel: any,
executable: string,
step = null
) {
2023-08-21 13:21:49 -03:00
const pid = GBAdminService.getNumberIdentifier();
GBServer.globals.processes[pid] = {
pid: pid,
userId: user ? user.userId : 0,
instanceId: min.instance.instanceId,
2023-08-23 11:24:48 -03:00
channel: channel,
roles: 'everyone',
2024-03-16 21:36:03 -03:00
step: step,
executable: executable
2023-08-21 13:21:49 -03:00
};
return pid;
}
}