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.6KB

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