server.js 12 KB

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