botserver/src/app.ts

246 lines
10 KiB
TypeScript
Raw Normal View History

#! /usr/bin / env node
2018-04-21 02:59:30 -03:00
/*****************************************************************************\
| ( )_ _ |
| _ _ _ __ _ _ __ ___ ___ _ _ | ,_)(_) ___ ___ _ |
| ( '_`\ ( '__)/'_` ) /'_ `\/' _ ` _ `\ /'_` )| | | |/',__)/' _ `\ /'_`\ |
| | (_) )| | ( (_| |( (_) || ( ) ( ) |( (_| || |_ | |\__, \| ( ) |( (_) ) |
| | ,__/'(_) `\__,_)`\__ |(_) (_) (_)`\__,_)`\__)(_)(____/(_) (_)`\___/' |
| | | ( )_) | |
| (_) \___/' |
| |
| 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, |
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. |
| |
\*****************************************************************************/
"use strict";
const logger = require("./logger");
const express = require("express");
const bodyParser = require("body-parser");
2018-11-04 17:26:29 -02:00
const opn = require('opn');
import { GBConfigService } from "../packages/core.gbapp/services/GBConfigService";
import { GBConversationalService } from "../packages/core.gbapp/services/GBConversationalService";
import { GBMinService } from "../packages/core.gbapp/services/GBMinService";
import { GBDeployer } from "../packages/core.gbapp/services/GBDeployer";
import { GBWhatsappPackage } from "./../packages/whatsapp.gblib/index";
import { GBCoreService } from "../packages/core.gbapp/services/GBCoreService";
import { GBImporter } from "../packages/core.gbapp/services/GBImporter";
import { GBAnalyticsPackage } from "../packages/analytics.gblib";
import { GBCorePackage } from "../packages/core.gbapp";
import { GBKBPackage } from "../packages/kb.gbapp";
import { GBSecurityPackage } from "../packages/security.gblib";
import { GBAdminPackage } from "../packages/admin.gbapp/index";
import { GBCustomerSatisfactionPackage } from "../packages/customer-satisfaction.gbapp";
import { GBAdminService } from "../packages/admin.gbapp/services/GBAdminService";
import { GuaribasInstance } from "../packages/core.gbapp/models/GBModel";
import { AzureDeployerService } from "../packages/azuredeployer.gbapp/services/AzureDeployerService";
2018-10-25 18:13:51 -03:00
import { IGBPackage, IGBInstance } from "botlib";
let appPackages = new Array<IGBPackage>();
2018-05-11 23:27:00 -03:00
2018-04-21 02:59:30 -03:00
/**
* General Bots open-core entry point.
*/
export class GBServer {
/**
* Program entry-point.
*/
2018-04-21 02:59:30 -03:00
static run() {
logger.info(`The Bot Server is in STARTING mode...`);
2018-04-21 02:59:30 -03:00
// Creates a basic HTTP server that will serve several URL, one for each
// bot instance. This allows the same server to attend multiple Bot on
// the Marketplace until GB get serverless.
let port = process.env.port || process.env.PORT || 4242;
let server = express();
2018-04-21 02:59:30 -03:00
server.use(bodyParser.json()); // to support JSON-encoded bodies
server.use(
bodyParser.urlencoded({
// to support URL-encoded bodies
extended: true
})
);
2018-05-11 23:27:00 -03:00
2018-10-25 18:13:51 -03:00
let bootInstance: IGBInstance;
2018-09-09 16:40:04 -03:00
server.listen(port, () => {
(async () => {
try {
logger.info(`Now accepting connections on ${port}...`);
2018-04-21 02:59:30 -03:00
2018-09-09 16:40:04 -03:00
// Reads basic configuration, initialize minimal services.
2018-09-09 14:39:37 -03:00
GBConfigService.init();
let core = new GBCoreService();
2018-10-25 18:13:51 -03:00
// Ensures cloud / on-premises infrastructure is setup.
logger.info(`Establishing a development local proxy (ngrok)...`);
let proxyAddress = await core.ensureProxy(port);
2018-11-04 17:26:29 -02:00
let deployer = new GBDeployer(core, new GBImporter(core));
let azureDeployer = new AzureDeployerService(deployer);
2018-10-25 18:13:51 -03:00
try {
await core.initDatabase();
} catch (error) {
logger.info(`Deploying cognitive infrastructure (on the cloud / on premises)...`);
try {
bootInstance = await azureDeployer.deployFarm(proxyAddress);
} catch (error) {
logger.warn(
2018-11-04 17:26:29 -02:00
"In case of error, please cleanup any infrastructure objects created during this procedure and .env before running again."
);
throw error;
}
2018-10-25 18:13:51 -03:00
core.writeEnv(bootInstance);
logger.info(`File .env written, starting General Bots...`);
2018-10-25 18:13:51 -03:00
GBConfigService.init();
await core.initDatabase();
}
// TODO: Get .gb* templates from GitHub and download do additional deploy folder.
2018-04-21 02:59:30 -03:00
// Check admin password.
2018-10-25 18:13:51 -03:00
let conversationalService = new GBConversationalService(core);
let adminService = new GBAdminService(core);
let password = GBConfigService.get("ADMIN_PASS");
if (!GBAdminService.StrongRegex.test(password)) {
throw new Error(
2018-11-04 17:26:29 -02:00
"Please, define a really strong password in ADMIN_PASS environment variable before running the server."
);
}
2018-04-21 02:59:30 -03:00
// NOTE: the semicolon is necessary before this line.
// Loads all system packages.
2018-04-21 02:59:30 -03:00
[
GBAdminPackage,
GBAnalyticsPackage,
GBCorePackage,
GBSecurityPackage,
GBKBPackage,
GBCustomerSatisfactionPackage,
GBWhatsappPackage
].forEach(e => {
logger.info(`Loading sys package: ${e.name}...`);
let p = Object.create(e.prototype) as IGBPackage;
p.loadPackage(core, core.sequelize);
});
// Loads all bot instances from object storage, if it's formatted.
2018-10-25 18:13:51 -03:00
logger.info(`Loading instances from storage...`);
let instances: GuaribasInstance[];
try {
instances = await core.loadInstances();
let instance = instances[0];
if (process.env.NODE_ENV === "development") {
logger.info(`Updating bot endpoint to local reverse proxy (ngrok)...`);
await azureDeployer.updateBotProxy(
instance.botId,
instance.botId,
`${proxyAddress}/api/messages/${instance.botId}`
);
}
} catch (error) {
2018-10-25 18:13:51 -03:00
if (error.parent.code === "ELOGIN") {
let group = GBConfigService.get("CLOUD_GROUP");
let serverName = GBConfigService.get("STORAGE_SERVER").split(
".database.windows.net"
)[0];
await azureDeployer.openStorageFirewall(group, serverName);
2018-10-25 18:13:51 -03:00
} else {
// Check if storage is empty and needs formatting.
let isInvalidObject =
error.parent.number == 208 || error.parent.errno == 1; // MSSQL or SQLITE.
if (isInvalidObject) {
if (GBConfigService.get("STORAGE_SYNC") != "true") {
throw `Operating storage is out of sync or there is a storage connection error. Try setting STORAGE_SYNC to true in .env file. Error: ${
error.message
}.`;
} else {
logger.info(
`Storage is empty. After collecting storage structure from all .gbapps it will get synced.`
);
}
} else {
2018-10-25 18:13:51 -03:00
throw `Cannot connect to operating storage: ${error.message}.`;
}
}
}
// Deploy packages and format object store according to .gbapp storage models.
logger.info(`Deploying packages...`);
await deployer.deployPackages(core, server, appPackages);
// If instances is undefined here it's because storage has been formatted.
// Load all instances from .gbot found on deploy package directory.
if (!instances) {
2018-10-25 18:13:51 -03:00
let saveInstance = new GuaribasInstance(bootInstance);
await saveInstance.save();
instances = await core.loadInstances();
}
// Setup server dynamic (per bot instance) resources and listeners.
2018-11-05 14:23:36 -02:00
logger.info(`Publishing instances...`);
let minService = new GBMinService(
core,
conversationalService,
adminService,
deployer
);
await minService.buildMin(server, appPackages, instances);
logger.info(`The Bot Server is in RUNNING mode...`);
2018-11-04 17:26:29 -02:00
if (process.env.NODE_ENV === "development") {
opn('http://localhost:4242');
}
return core;
2018-09-09 16:40:04 -03:00
} catch (err) {
logger.error(`STOP: ${err} ${err.stack ? err.stack : ""}`);
process.exit(1);
2018-09-09 16:40:04 -03:00
}
})();
});
2018-04-21 02:59:30 -03:00
}
}
// First line to run.
GBServer.run();