You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Admin.ts 6.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. 'use strict'
  2. import { getLogger } from 'frontblock-generic/Types';
  3. import { promises as fs, mkdirSync } from "fs"
  4. import { RPCServer } from 'rpclibrary'
  5. import * as Path from 'path'
  6. import * as Knex from 'knex';
  7. import * as http from 'http';
  8. import * as express from 'express';
  9. import { GuildManager } from '../Components/Guild/GuildManager';
  10. import { ItemManager } from '../Components/Item/ItemManager';
  11. import { RaidManager } from '../Components/Raid/RaidManager';
  12. import { CharacterManager } from '../Components/Character/CharacterManager';
  13. import { UserManager } from '../Components/User/UserManager';
  14. import { RootComponent } from '../Injector/ServiceDecorator';
  15. import { TableDefinitionExporter } from '../Types/Interfaces';
  16. import { AdminConf, TableDefiniton } from '../Types/Types';
  17. import { RPCConfigLoader } from '../Components/RPCConfigLoader';
  18. import { FrontworkComponent } from '../Types/FrontworkComponent';
  19. import { IAdmin } from './Interface';
  20. import { Injector } from '../Injector/Injector';
  21. import { Shoutbox } from '../Components/Shoutbox/Shoutbox';
  22. const logger = getLogger("admin", 'debug')
  23. @RootComponent({
  24. injectable: IAdmin,
  25. injects: [
  26. GuildManager,
  27. ItemManager,
  28. RaidManager,
  29. CharacterManager,
  30. UserManager,
  31. Shoutbox
  32. ]
  33. })
  34. export class FrontworkAdmin
  35. implements TableDefinitionExporter, IAdmin {
  36. knex:Knex
  37. config: RPCConfigLoader<AdminConf>
  38. rpcServer: RPCServer
  39. private express
  40. private httpServer
  41. constructor(private frontworkComponents: FrontworkComponent[] = []){
  42. this.config = new RPCConfigLoader<AdminConf>({
  43. name: "FrontworkAdminConf",
  44. getDefaultConfig: () => {
  45. return {
  46. httpPort: 8080,
  47. eventBusConf: {},
  48. dbConf: {
  49. client: 'sqlite3',
  50. connection: {
  51. filename: Path.join(__dirname, "data/frontworkAdmin.sqlite")
  52. },
  53. useNullAsDefault: true,
  54. }
  55. }
  56. }
  57. }, './config', this.configChangeHandler)
  58. }
  59. async start(){
  60. await this.makeKnex()
  61. this.startWebsocket()
  62. await Promise.all( this.frontworkComponents.map(c => c.initialize?c.initialize():undefined ))
  63. logger.debug(this.frontworkComponents.length+" components initialized")
  64. this.startWebserver()
  65. }
  66. stop(){
  67. Promise.race([
  68. Promise.all( this.frontworkComponents.map(c => c.stop?c.stop():undefined )),
  69. new Promise((res, rej) => { setTimeout(res, 250);})
  70. ]).catch(e => logger.warn(e))
  71. .finally(() => { process.exit(0) })
  72. }
  73. protected configChangeHandler = (conf:AdminConf, key?:string) => {
  74. if(key === 'dbConf'){
  75. this.makeKnex()
  76. }
  77. }
  78. getConfigKey(key:string){
  79. return this.config.getConfigKey(key)
  80. }
  81. setConfigKey(key:string, value:any){
  82. return this.config.setConfigKey(key, value)
  83. }
  84. getTableDefinitions(): TableDefiniton[]{
  85. return [
  86. ...this.frontworkComponents
  87. ].flatMap(exporter => exporter.getTableDefinitions())
  88. }
  89. private startWebsocket(){
  90. this.rpcServer = new RPCServer(20000, [
  91. ...this.frontworkComponents,
  92. {
  93. name: "debug",
  94. exportRPCs: () => [{
  95. name: 'dumpDb',
  96. call: async (table) => await this.knex(table).select('*')
  97. }]
  98. }
  99. ], {
  100. visibility: '0.0.0.0'
  101. })
  102. logger.debug("Websocket up on", 20000)
  103. }
  104. private startWebserver(){
  105. if(this.httpServer != null || this.express != null){
  106. logger.warn("Webserver is already running")
  107. return
  108. }
  109. let port:number = this.config.getConfig().httpPort
  110. this.express = express()
  111. this.express.use('/', express.static('dist/static'))
  112. /**
  113. * get the compiled FrontendPlugins.js
  114. */
  115. this.express.get('/plugins/:id'+".js", async (request, response) => {
  116. const pth = Path.resolve("plugins/"+request.params.id, "FrontendPlugin.js");
  117. const file = await fs.readFile(pth)
  118. const frontend = file.toString()
  119. response.status(200)
  120. response.set('Content-Type', 'application/javascript')
  121. response.send(frontend)
  122. })
  123. /**
  124. * serve the index.html from the static folder
  125. */
  126. this.express.get("/", (request, response) => {
  127. response.status(200)
  128. response.sendFile('index.html');
  129. })
  130. /**
  131. * redirect all the other traffic to the single page app
  132. */
  133. this.express.get("*", (request, response) => {
  134. response.status(301)
  135. response.redirect('/')
  136. })
  137. this.httpServer = new http.Server(this.express)
  138. this.httpServer.listen(port, () => {
  139. logger.info('Admin panel listening for HTTP on *'+port)
  140. })
  141. }
  142. private stopWebserver(){
  143. if(this.httpServer == null || this.express == null){
  144. logger.warn("Webserver is not running")
  145. return
  146. }
  147. this.httpServer.close()
  148. this.httpServer = null
  149. this.express = null
  150. logger.info("Webserver stopped")
  151. }
  152. async makeKnex():Promise<Knex>{
  153. const conf:Knex.Config = this.config.getConfigKey("dbConf")
  154. logger.debug("Making new knex:", conf)
  155. if(conf.client === 'sqlite3'){
  156. mkdirSync(Path.dirname((<any>conf.connection).filename), {recursive: true})
  157. }
  158. this.knex = Knex(conf)
  159. if(conf.client === 'sqlite3'){
  160. await this.knex.raw('PRAGMA foreign_keys = ON');
  161. }
  162. await Promise.all(
  163. this.getTableDefinitions()
  164. //make unique by name
  165. .filter((other, index, self) => index === self.findIndex(
  166. (self) => self.name === other.name)
  167. )
  168. //create table if not exists
  169. .map(async (def)=> {
  170. const hasTable = await this.knex.schema.hasTable(def.name)
  171. if(!hasTable)
  172. return await this.knex.schema.createTable(def.name, def.tableBuilder)
  173. })
  174. )
  175. return this.knex
  176. }
  177. }
  178. process.on( 'SIGINT', function() {
  179. logger.info("Shutting down from SIGINT (Ctrl-C)" );
  180. Injector.resolve<FrontworkAdmin>(FrontworkAdmin).stop()
  181. })