
* FIX: Some security improved. * NEW: Protocol changes for exchanging questions between UI and Bot Server.
207 lines
8.6 KiB
TypeScript
207 lines
8.6 KiB
TypeScript
/*****************************************************************************\
|
|
| ( )_ _ |
|
|
| _ _ _ __ _ _ __ ___ ___ _ _ | ,_)(_) ___ ___ _ |
|
|
| ( '_`\ ( '__)/'_` ) /'_ `\/' _ ` _ `\ /'_` )| | | |/',__)/' _ `\ /'_`\ |
|
|
| | (_) )| | ( (_| |( (_) || ( ) ( ) |( (_| || |_ | |\__, \| ( ) |( (_) ) |
|
|
| | ,__/'(_) `\__,_)`\__ |(_) (_) (_)`\__,_)`\__)(_)(____/(_) (_)`\___/' |
|
|
| | | ( )_) | |
|
|
| (_) \___/' |
|
|
| |
|
|
| General Bots Copyright (c) Pragmatismo.io. All rights reserved. |
|
|
| Licensed under the AGPL-3.0. |
|
|
| |
|
|
| 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, |
|
|
| but WITHOUT ANY WARRANTY without even the implied warranty of |
|
|
| 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. |
|
|
| |
|
|
\*****************************************************************************/
|
|
|
|
"use strict";
|
|
|
|
const UrlJoin = require("url-join");
|
|
import { AzureSearch } from "pragmatismo-io-framework";
|
|
import { GBMinInstance } from "botlib";
|
|
import { IGBDialog } from "botlib";
|
|
import { GBDeployer } from "../../core.gbapp/services/GBDeployer";
|
|
import { GBImporter } from "../../core.gbapp/services/GBImporter";
|
|
import { GBConfigService } from "../../core.gbapp/services/GBConfigService";
|
|
import { KBService } from "./../../kb.gbapp/services/KBService";
|
|
import { BotAdapter } from "botbuilder";
|
|
import { GBAdminService } from "../services/GBAdminService";
|
|
|
|
/**
|
|
* Dialogs for administration tasks.
|
|
*/
|
|
export class AdminDialog extends IGBDialog {
|
|
static async undeployPackageCommand(text: any, min: GBMinInstance, dc) {
|
|
let packageName = text.split(" ")[1];
|
|
let importer = new GBImporter(min.core);
|
|
let deployer = new GBDeployer(min.core, importer);
|
|
dc.context.sendActivity(`Undeploying package ${packageName}...`);
|
|
await deployer.undeployPackageFromLocalPath(
|
|
min.instance,
|
|
UrlJoin("deploy", packageName)
|
|
);
|
|
dc.context.sendActivity(`Package ${packageName} undeployed...`);
|
|
}
|
|
|
|
static async deployPackageCommand(
|
|
text: string,
|
|
dc,
|
|
deployer: GBDeployer,
|
|
min: GBMinInstance
|
|
) {
|
|
let packageName = text.split(" ")[1];
|
|
await dc.context.sendActivity(
|
|
`Deploying package ${packageName}... (It may take a few seconds)`
|
|
);
|
|
let additionalPath = GBConfigService.get("ADDITIONAL_DEPLOY_PATH");
|
|
await deployer.deployPackageFromLocalPath(
|
|
UrlJoin(additionalPath, packageName)
|
|
);
|
|
await dc.context.sendActivity(
|
|
`Package ${packageName} deployed... Please run rebuildIndex command.`
|
|
);
|
|
}
|
|
|
|
static async rebuildIndexCommand(min: GBMinInstance, dc) {
|
|
let search = new AzureSearch(
|
|
min.instance.searchKey,
|
|
min.instance.searchHost,
|
|
min.instance.searchIndex,
|
|
min.instance.searchIndexer
|
|
);
|
|
dc.context.sendActivity("Rebuilding index...");
|
|
await search.deleteIndex();
|
|
let kbService = new KBService(min.core.sequelize);
|
|
await search.createIndex(
|
|
kbService.getSearchSchema(min.instance.searchIndex),
|
|
"gb"
|
|
);
|
|
await dc.context.sendActivity("Index rebuilt.");
|
|
}
|
|
|
|
/**
|
|
* Setup dialogs flows and define services call.
|
|
*
|
|
* @param bot The bot adapter.
|
|
* @param min The minimal bot instance data.
|
|
*/
|
|
static setup(bot: BotAdapter, min: GBMinInstance) {
|
|
// Setup services.
|
|
|
|
let importer = new GBImporter(min.core);
|
|
let deployer = new GBDeployer(min.core, importer);
|
|
|
|
min.dialogs.add("/adminRat", [
|
|
async dc => {
|
|
await AdminDialog.refreshAdminToken(min, dc);
|
|
// await dc.context.sendActivity(
|
|
// `Deploying package ... (It may take a few seconds)`
|
|
// );
|
|
// await AdminDialog.deployPackageCommand(
|
|
// "deployPackage ProjectOnline.gbkb",
|
|
// dc,
|
|
// deployer,
|
|
// min
|
|
// );
|
|
await dc.endAll();
|
|
}
|
|
]);
|
|
|
|
min.dialogs.add("/adminUpdateToken", [
|
|
async (dc, args, next) => {
|
|
await dc.endAll();
|
|
let service = new GBAdminService();
|
|
await service.saveValue("authenticatorToken", args.token)
|
|
await dc.context.sendActivities([
|
|
{ type: 'typing' },
|
|
{ type: 'message', text: "Token has been updated." },
|
|
{ type: 'message', text: "Please, log out now from the administration work account on next screen." },
|
|
{ type: 'delay', value: 4000 },
|
|
])
|
|
}
|
|
]);
|
|
|
|
min.dialogs.add("/admin", [
|
|
async (dc, args) => {
|
|
const prompt = "Please, authenticate:";
|
|
await dc.prompt("textPrompt", prompt);
|
|
},
|
|
async (dc, value) => {
|
|
let text = value;
|
|
const user = min.userState.get(dc.context);
|
|
|
|
if (!user.authenticated || text === GBConfigService.get("ADMIN_PASS")) {
|
|
user.authenticated = true;
|
|
await dc.context.sendActivity(
|
|
"Welcome to Pragmatismo.io GeneralBots Administration."
|
|
);
|
|
await dc.prompt("textPrompt", "Which task do you wanna run now?");
|
|
} else {
|
|
await dc.endAll();
|
|
}
|
|
},
|
|
async (dc, value) => {
|
|
var text = value;
|
|
const user = min.userState.get(dc.context);
|
|
|
|
if (text === "quit") {
|
|
user.authenticated = false;
|
|
await dc.replace("/");
|
|
} else if (text === "sync") {
|
|
await min.core.syncDatabaseStructure();
|
|
await dc.context.sendActivity("Sync started...");
|
|
await dc.replace("/admin", { firstRun: false });
|
|
} else if (text.split(" ")[0] === "rebuildIndex") {
|
|
await AdminDialog.rebuildIndexCommand(min, dc);
|
|
await dc.replace("/admin", { firstRun: false });
|
|
} else if (text.split(" ")[0] === "deployPackage") {
|
|
await AdminDialog.deployPackageCommand(text, dc, deployer, min);
|
|
await dc.replace("/admin", { firstRun: false });
|
|
} else if (text.split(" ")[0] === "redeployPackage") {
|
|
await AdminDialog.undeployPackageCommand(text, min, dc);
|
|
await AdminDialog.deployPackageCommand(text, dc, deployer, min);
|
|
await dc.context.sendActivity("Redeploy done.");
|
|
await dc.replace("/admin", { firstRun: false });
|
|
} else if (text.split(" ")[0] === "undeployPackage") {
|
|
await AdminDialog.undeployPackageCommand(text, min, dc);
|
|
await dc.replace("/admin", { firstRun: false });
|
|
} else if (text.split(" ")[0] === "applyPackage") {
|
|
await dc.context.sendActivity("Applying in progress...");
|
|
await min.core.loadInstance(text.split(" ")[1]);
|
|
await dc.context.sendActivity("Applying done...");
|
|
await dc.replace("/admin", { firstRun: false });
|
|
} else if (text.split(" ")[0] === "rat") {
|
|
await AdminDialog.refreshAdminToken(min, dc);
|
|
}
|
|
}
|
|
]);
|
|
}
|
|
|
|
private static async refreshAdminToken(min: any, dc: any) {
|
|
let config = {
|
|
authenticatorTenant: min.instance.authenticatorTenant,
|
|
authenticatorClientID: min.instance.authenticatorClientID
|
|
};
|
|
await min.conversationalService.sendEvent(dc, "play", {
|
|
playerType: "login",
|
|
data: config
|
|
});
|
|
await dc.context.sendActivity("Update your Administrative token by Login...");
|
|
}
|
|
}
|