server.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. import express from 'express';
  2. import session from 'express-session';
  3. import memorystore from 'memorystore';
  4. import ejs from 'ejs';
  5. import Docker from 'dockerode';
  6. import { Readable } from 'stream';
  7. import { router } from './router/index.js';
  8. import { sequelize, Container, Permission } from './database/models.js';
  9. import { currentLoad, mem, networkStats, fsSize, dockerContainerStats, dockerImages, networkInterfaces } from 'systeminformation';
  10. import { containerCard } from './components/containerCard.js';
  11. import { modal } from './components/modal.js';
  12. import { permissionsModal } from './components/permissions_modal.js';
  13. export var docker = new Docker();
  14. const app = express();
  15. const MemoryStore = memorystore(session);
  16. const port = process.env.PORT || 8000;
  17. let [ hidden, activeEvent, cardList, clicked ] = ['', '', '', false];
  18. let sentList = '';
  19. let SSE = false;
  20. app.locals.installCard = '';
  21. let thanks = 0;
  22. // Session middleware
  23. const sessionMiddleware = session({
  24. store: new MemoryStore({ checkPeriod: 86400000 }), // Prune expired entries every 24h
  25. secret: "keyboard cat",
  26. resave: false,
  27. saveUninitialized: false,
  28. cookie:{
  29. secure:false, // Only set to true if you are using HTTPS.
  30. httpOnly:false, // Only set to true if you are using HTTPS.
  31. maxAge:3600000 * 8 // Session max age in milliseconds. 3600000 = 1 hour.
  32. }
  33. });
  34. // Express middleware
  35. app.set('view engine', 'html');
  36. app.engine('html', ejs.renderFile);
  37. app.use([
  38. express.static('public'),
  39. express.json(),
  40. express.urlencoded({ extended: true }),
  41. sessionMiddleware,
  42. router
  43. ]);
  44. // Initialize server
  45. app.listen(port, async () => {
  46. async function init() {
  47. try { await sequelize.authenticate().then(
  48. () => { console.log('DB Connection: ✔️') }); }
  49. catch { console.log('DB Connection: ❌'); }
  50. try { await sequelize.sync().then( // check out that formatting
  51. () => { console.log('Synced Models: ✔️') }); }
  52. catch { console.log('Synced Models: ❌'); } }
  53. await init().then(() => {
  54. console.log(`Listening on http://localhost:${port} ✔️`);
  55. });
  56. });
  57. // Get hidden containers
  58. async function getHidden() {
  59. hidden = await Container.findAll({ where: {visibility:false}});
  60. hidden = hidden.map((container) => container.name);
  61. }
  62. // Server metrics
  63. let [ cpu, ram, tx, rx, disk ] = [0, 0, 0, 0, 0];
  64. let serverMetrics = async () => {
  65. currentLoad().then(data => {
  66. cpu = Math.round(data.currentLoad);
  67. });
  68. mem().then(data => {
  69. ram = Math.round((data.active / data.total) * 100);
  70. });
  71. networkStats().then(data => {
  72. tx = data[0].tx_bytes / (1024 * 1024);
  73. rx = data[0].rx_bytes / (1024 * 1024);
  74. });
  75. fsSize().then(data => {
  76. disk = data[0].use;
  77. });
  78. }
  79. setInterval(serverMetrics, 1000);
  80. // Docker containers
  81. let containerCards = async () => {
  82. let list = '';
  83. const allContainers = await docker.listContainers({ all: true });
  84. for (const container of allContainers) {
  85. if (!hidden.includes(container.Names[0].slice(1))) {
  86. let imageVersion = container.Image.split('/');
  87. let service = imageVersion[imageVersion.length - 1].split(':')[0];
  88. let containerId = docker.getContainer(container.Id);
  89. let containerInfo = await containerId.inspect();
  90. let ports_list = [];
  91. try {
  92. for (const [key, value] of Object.entries(containerInfo.HostConfig.PortBindings)) {
  93. let ports = {
  94. check: 'checked',
  95. external: value[0].HostPort,
  96. internal: key.split('/')[0],
  97. protocol: key.split('/')[1]
  98. }
  99. ports_list.push(ports);
  100. }
  101. } catch {}
  102. let external_port = ports_list[0]?.external || 0;
  103. let internal_port = ports_list[0]?.internal || 0;
  104. let container_info = {
  105. name: container.Names[0].slice(1),
  106. service: service,
  107. id: container.Id,
  108. state: container.State,
  109. image: container.Image,
  110. external_port: external_port,
  111. internal_port: internal_port,
  112. ports: ports_list,
  113. link: 'localhost',
  114. }
  115. let card = containerCard(container_info);
  116. list += card;
  117. }
  118. }
  119. cardList = list;
  120. }
  121. // Store docker events
  122. docker.getEvents((err, stream) => {
  123. if (err) throw err;
  124. stream.on('data', (chunk) => {
  125. activeEvent += chunk.toString('utf8');
  126. });
  127. });
  128. // Check if the container cards need to be updated
  129. setInterval(async () => {
  130. if (activeEvent == '') { return; }
  131. activeEvent = '';
  132. await getHidden();
  133. await containerCards();
  134. if (cardList != sentList) {
  135. cardList = sentList;
  136. SSE = true;
  137. }
  138. }, 1000);
  139. /////////////////// HTMX routes //////////////////////////
  140. router.get('/stats', async (req, res) => {
  141. if (req.session.UUID == undefined) { return; }
  142. switch (req.header('HX-Trigger')) {
  143. case 'cpu':
  144. let info = '<div class="font-weight-medium">';
  145. info += '<label class="cpu-text mb-1" for="cpu">CPU ' + cpu + '%</label>';
  146. info += '</div>';
  147. info += '<div class="cpu-bar meter animate">';
  148. info += '<span style="width:' + cpu + '%"><span></span></span>';
  149. info += '</div>';
  150. res.send(info);
  151. break;
  152. case 'ram':
  153. let info2 = '<div class="font-weight-medium">';
  154. info2 += '<label class="ram-text mb-1" for="ram">RAM ' + ram + '%</label>';
  155. info2 += '</div>';
  156. info2 += '<div class="ram-bar meter animate orange">';
  157. info2 += '<span style="width:' + ram + '%"><span></span></span>';
  158. info2 += '</div>';
  159. res.send(info2);
  160. break;
  161. case 'tx':
  162. res.send('TX ' + tx.toFixed(2) + ' MB');
  163. break;
  164. case 'rx':
  165. res.send('RX ' + rx.toFixed(2) + ' MB');
  166. break;
  167. case 'disk':
  168. let info5 = '<div class="font-weight-medium">';
  169. info5 += '<label class="disk-text mb-1" for="disk">Disk ' + disk + '%</label>';
  170. info5 += '</div>';
  171. info5 += '<div class="disk-bar meter animate red">';
  172. info5 += '<span style="width:' + disk + '%"><span></span></span>';
  173. info5 += '</div>';
  174. res.send(info5);
  175. break;
  176. default:
  177. console.log('Unknown trigger');
  178. break;
  179. }
  180. });
  181. router.get('/containers', async (req, res) => {
  182. if (req.session.UUID == undefined) { return; }
  183. await getHidden();
  184. await containerCards();
  185. sentList = cardList;
  186. res.send(cardList);
  187. });
  188. router.get('/action', async (req, res) => {
  189. if (req.session.UUID == undefined) { return; }
  190. let name = req.header('hx-trigger-name');
  191. let id = req.header('hx-trigger');
  192. let value = req.query[name];
  193. var containerName = docker.getContainer(name);
  194. if ((id == 'start') && (value == 'stopped')) {
  195. containerName.start();
  196. } else if ((id == 'start') && (value == 'paused')) {
  197. containerName.unpause();
  198. } else if ((id == 'stop') && (value != 'stopped')) {
  199. containerName.stop();
  200. } else if ((id == 'pause') && (value == 'running')) {
  201. containerName.pause();
  202. } else if ((id == 'pause') && (value == 'paused')) {
  203. containerName.unpause();
  204. } else if (id == 'restart') {
  205. containerName.restart();
  206. }
  207. });
  208. router.get('/hide', async (req, res) => {
  209. if (req.session.UUID == undefined) { return; }
  210. let name = req.header('hx-trigger-name');
  211. let id = req.header('hx-trigger');
  212. if (id == 'hide') {
  213. let exists = await Container.findOne({ where: {name: name}});
  214. if (!exists) {
  215. const newContainer = await Container.create({ name: name, visibility: false, });
  216. SSE = true;
  217. return;
  218. } else {
  219. exists.update({ visibility: false });
  220. SSE = true;
  221. return;
  222. }
  223. }
  224. if (id == 'reset') {
  225. Container.update({ visibility: true }, { where: {} });
  226. SSE = true;
  227. }
  228. });
  229. router.get('/logs', async (req, res) => {
  230. if (req.session.UUID == undefined) { return; }
  231. let name = req.header('hx-trigger-name');
  232. function containerLogs (data) {
  233. return new Promise((resolve, reject) => {
  234. let logString = '';
  235. var options = {
  236. follow: false,
  237. stdout: true,
  238. stderr: false,
  239. timestamps: false
  240. };
  241. var containerName = docker.getContainer(data);
  242. containerName.logs(options, function (err, stream) {
  243. if (err) { reject(err); return; }
  244. const readableStream = Readable.from(stream);
  245. readableStream.on('data', function (chunk) {
  246. logString += chunk.toString('utf8');
  247. });
  248. readableStream.on('end', function () {
  249. resolve(logString);
  250. });
  251. });
  252. });
  253. };
  254. containerLogs(name).then((data) => {
  255. res.send(`<pre>${data}</pre> `)
  256. });
  257. });
  258. router.get('/sse_event', (req, res) => {
  259. res.writeHead(200, { 'Content-Type': 'text/event-stream', 'Cache-Control': 'no-cache', 'Connection': 'keep-alive', });
  260. let eventCheck = setInterval(async () => {
  261. if (SSE == true) {
  262. SSE = false;
  263. res.write(`event: docker\n`);
  264. res.write(`data: there was a docker event!\n\n`);
  265. }
  266. }, 1000);
  267. req.on('close', () => {
  268. clearInterval(eventCheck);
  269. });
  270. });
  271. router.get('/modal', async (req, res) => {
  272. if (req.session.UUID == undefined) { return; }
  273. let name = req.header('hx-trigger-name');
  274. let id = req.header('hx-trigger');
  275. if (id == 'permissions') {
  276. let containerPermissions = await Permission.findAll({ where: {containerName: name}});
  277. let form = permissionsModal();
  278. res.send(form);
  279. return;
  280. }
  281. let containerId = docker.getContainer(name);
  282. let containerInfo = await containerId.inspect();
  283. let ports_list = [];
  284. try {
  285. for (const [key, value] of Object.entries(containerInfo.HostConfig.PortBindings)) {
  286. let ports = {
  287. check: 'checked',
  288. external: value[0].HostPort,
  289. internal: key.split('/')[0],
  290. protocol: key.split('/')[1]
  291. }
  292. ports_list.push(ports);
  293. }
  294. } catch {}
  295. let external_port = ports_list[0]?.external || 0;
  296. let internal_port = ports_list[0]?.internal || 0;
  297. let container_info = {
  298. name: containerInfo.Name.slice(1),
  299. state: containerInfo.State.Status,
  300. image: containerInfo.Config.Image,
  301. external_port: external_port,
  302. internal_port: internal_port,
  303. ports: ports_list,
  304. link: 'localhost',
  305. }
  306. let form = modal(container_info);
  307. res.send(form);
  308. });
  309. let stats = {};
  310. router.get('/chart', async (req, res) => {
  311. if (req.session.UUID == undefined) { return; }
  312. let name = req.header('hx-trigger-name');
  313. // create an empty array if it doesn't exist
  314. if (!stats[name]) {
  315. stats[name] = { cpuArray: Array(15).fill(0), ramArray: Array(15).fill(0) };
  316. }
  317. // get the stats
  318. const info = await dockerContainerStats(name);
  319. // update the arrays
  320. stats[name].cpuArray.push(Math.round(info[0].cpuPercent));
  321. stats[name].ramArray.push(Math.round(info[0].memPercent));
  322. // slice them down to the last 15 values
  323. stats[name].cpuArray = stats[name].cpuArray.slice(-15);
  324. stats[name].ramArray = stats[name].ramArray.slice(-15);
  325. // replace the chart with the new data
  326. let chart = `
  327. <script>
  328. ${name}chart.updateSeries([{
  329. data: [${stats[name].cpuArray}]
  330. }, {
  331. data: [${stats[name].ramArray}]
  332. }])
  333. </script>`
  334. res.send(chart);
  335. });
  336. router.get('/thank', async (req, res) => {
  337. thanks++;
  338. let data = thanks.toString();
  339. if (thanks > 999) {
  340. data = 'Did you really click 1000 times?!';
  341. }
  342. res.send(data);
  343. });