Browse Source

removed snake_case from app.js

renamed function system_information.js to system.js
lllllllillllllillll 1 year ago
parent
commit
ca8a40c065
7 changed files with 530 additions and 76 deletions
  1. 33 32
      app.js
  2. 23 38
      components/dashCard.js
  3. 1 1
      controllers/site_actions.js
  4. 205 0
      functions/compose.js
  5. 7 1
      functions/package_manager.js
  6. 256 0
      functions/system.js
  7. 5 4
      routes/index.js

+ 33 - 32
app.js

@@ -1,25 +1,25 @@
+// Express
 const express = require("express");
 const app = express();
 const session = require("express-session");
+const PORT = process.env.PORT || 8000;
+
+// Redis
 const redis = require('redis');
 const RedisStore = require("connect-redis").default;
-const redisClient = redis.createClient({
-    url: "redis://DweebCache:6379",
-    password: process.env.REDIS_PASS,
-});
+const redisClient = redis.createClient({ url: "redis://DweebCache:6379", password: process.env.REDIS_PASS, });
 redisClient.connect().catch(console.log);
-let redisStore = new RedisStore({
-    client: redisClient,
-});
+let redisStore = new RedisStore({ client: redisClient });
 
+// Routes
 const routes = require("./routes");
 
-const { serverStats, containerList, containerStats, containerAction } = require('./functions/system_information');
-const { RefreshSites } = require('./controllers/site_actions');
-
-let sent_list, clicked;
+// Functions and variables
+const { serverStats, containerList, containerStats, containerAction } = require('./functions/system');
+let sentList, clicked;
 app.locals.site_list = '';
 
+// Configure Session
 const sessionMiddleware = session({
     store: redisStore,
     secret: "keyboard cat", 
@@ -28,10 +28,11 @@ const sessionMiddleware = session({
     cookie:{
         secure:false, // Only set to true if you are using HTTPS.
         httpOnly:false, // Only set to true if you are using HTTPS.
-        maxAge:3600000 * 8// Session max age in milliseconds. 3600000 = 1 hour.
+        maxAge:3600000 * 8 // Session max age in milliseconds. 3600000 = 1 hour.
     } 
 })
 
+// Middleware
 app.set('view engine', 'ejs');
 app.use([
     express.static("public"),
@@ -41,49 +42,49 @@ app.use([
     routes
 ]);
 
-const server = app.listen(8000, async () => {
-    console.log(`App listening on port 8000`);   
+// Start Express server
+const server = app.listen(PORT, async () => {
+    console.log(`App listening on port ${PORT}`);   
 });
 
+// Start Socket.io
 const io = require('socket.io')(server);
 io.engine.use(sessionMiddleware);
 
-
 io.on('connection', (socket) => {
-    // set user session
+
+    // Set user session
     const user_session = socket.request.session;
     console.log(`${user_session.user} connected from ${socket.handshake.headers.host} ${socket.handshake.address}`);
 
-    // check if a list of containers needs to be sent
-    if (sent_list != null) { socket.emit('cards', sent_list); }
-
-    // check if an install card has to be sent
+    // Check if a list of containers or an install card needs to be sent
+    if (sentList != null) { socket.emit('cards', sentList); }
     if((app.locals.install != '') && (app.locals.install != null)){ socket.emit('install', app.locals.install); }
 
-    // send server metrics
+    // Send server metrics
     let ServerStats = setInterval(async () => {
         socket.emit('metrics', await serverStats());
     }, 1000);
 
-    // send container list
+    // Send list of containers
     let ContainerList = setInterval(async () => {
-        let card_list = await containerList();
-        if (sent_list !== card_list) {
-            sent_list = card_list;
+        let cardList = await containerList();
+        if (sentList !== cardList) {
+            sentList = cardList;
             app.locals.install = '';
-            socket.emit('cards', card_list);
+            socket.emit('cards', cardList);
         }
     }, 1000);
 
-    // send container metrics
+    // Send container metrics
     let ContainerStats = setInterval(async () => {
-        let container_stats = await containerStats();
-        for (let i = 0; i < container_stats.length; i++) {
-            socket.emit('container_stats', container_stats[i]);
+        let stats = await containerStats();
+        for (let i = 0; i < stats.length; i++) {
+            socket.emit('containerStats', stats[i]);
         }
     }, 1000);
 
-    // play/pause/stop/restart container
+    // Container controls
     socket.on('clicked', (data) => {
         if (clicked == true) { return; } clicked = true;
         let buttonPress = {
@@ -96,7 +97,7 @@ io.on('connection', (socket) => {
         containerAction(buttonPress);
         clicked = false;
     });
-    
+
     socket.on('disconnect', () => {                
         clearInterval(ServerStats);
         clearInterval(ContainerList);

+ 23 - 38
components/dashCard.js

@@ -270,50 +270,35 @@ module.exports.dashCard = function dashCard(data) {
       <div class="modal-dialog modal-dialog-centered modal-dialog-scrollable" role="document">
         <div class="modal-content">
           <div class="modal-header">
-            <h5 class="modal-title">Scrollable modal</h5>
+            <h5 class="modal-title">${name} Logs</h5>
             <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
           </div>
           <div class="modal-body">
-            <p>Cras mattis consectetur purus sit amet fermentum. Cras justo odio, dapibus ac facilisis in, egestas
-              eget quam. Morbi leo risus, porta ac consectetur ac, vestibulum at eros.</p>
-            <p>Praesent commodo cursus magna, vel scelerisque nisl consectetur et. Vivamus sagittis lacus vel augue
-              laoreet rutrum faucibus dolor auctor.</p>
-            <p>Aenean lacinia bibendum nulla sed consectetur. Praesent commodo cursus magna, vel scelerisque nisl
-              consectetur et. Donec sed odio dui. Donec ullamcorper nulla non metus auctor fringilla.</p>
-            <p>Cras mattis consectetur purus sit amet fermentum. Cras justo odio, dapibus ac facilisis in, egestas
-              eget quam. Morbi leo risus, porta ac consectetur ac, vestibulum at eros.</p>
-            <p>Praesent commodo cursus magna, vel scelerisque nisl consectetur et. Vivamus sagittis lacus vel augue
-              laoreet rutrum faucibus dolor auctor.</p>
-            <p>Aenean lacinia bibendum nulla sed consectetur. Praesent commodo cursus magna, vel scelerisque nisl
-              consectetur et. Donec sed odio dui. Donec ullamcorper nulla non metus auctor fringilla.</p>
-            <p>Cras mattis consectetur purus sit amet fermentum. Cras justo odio, dapibus ac facilisis in, egestas
-              eget quam. Morbi leo risus, porta ac consectetur ac, vestibulum at eros.</p>
-            <p>Praesent commodo cursus magna, vel scelerisque nisl consectetur et. Vivamus sagittis lacus vel augue
-              laoreet rutrum faucibus dolor auctor.</p>
-            <p>Aenean lacinia bibendum nulla sed consectetur. Praesent commodo cursus magna, vel scelerisque nisl
-              consectetur et. Donec sed odio dui. Donec ullamcorper nulla non metus auctor fringilla.</p>
-            <p>Cras mattis consectetur purus sit amet fermentum. Cras justo odio, dapibus ac facilisis in, egestas
-              eget quam. Morbi leo risus, porta ac consectetur ac, vestibulum at eros.</p>
-            <p>Praesent commodo cursus magna, vel scelerisque nisl consectetur et. Vivamus sagittis lacus vel augue
-              laoreet rutrum faucibus dolor auctor.</p>
-            <p>Aenean lacinia bibendum nulla sed consectetur. Praesent commodo cursus magna, vel scelerisque nisl
-              consectetur et. Donec sed odio dui. Donec ullamcorper nulla non metus auctor fringilla.</p>
-            <p>Cras mattis consectetur purus sit amet fermentum. Cras justo odio, dapibus ac facilisis in, egestas
-              eget quam. Morbi leo risus, porta ac consectetur ac, vestibulum at eros.</p>
-            <p>Praesent commodo cursus magna, vel scelerisque nisl consectetur et. Vivamus sagittis lacus vel augue
-              laoreet rutrum faucibus dolor auctor.</p>
-            <p>Aenean lacinia bibendum nulla sed consectetur. Praesent commodo cursus magna, vel scelerisque nisl
-              consectetur et. Donec sed odio dui. Donec ullamcorper nulla non metus auctor fringilla.</p>
-            <p>Cras mattis consectetur purus sit amet fermentum. Cras justo odio, dapibus ac facilisis in, egestas
-              eget quam. Morbi leo risus, porta ac consectetur ac, vestibulum at eros.</p>
-            <p>Praesent commodo cursus magna, vel scelerisque nisl consectetur et. Vivamus sagittis lacus vel augue
-              laoreet rutrum faucibus dolor auctor.</p>
-            <p>Aenean lacinia bibendum nulla sed consectetur. Praesent commodo cursus magna, vel scelerisque nisl
-              consectetur et. Donec sed odio dui. Donec ullamcorper nulla non metus auctor fringilla.</p>
+            
+          <div class="card-body">
+                <h4>
+                  Log File:
+                </h4>
+                <div>
+                  <pre><code>GET <a class="text-reset" target="_blank" href="https://tabler.io/demo">https://tabler.io/demo</a></code></pre>
+                </div>
+                <h4>Logs:</h4>
+                <div>
+                  <pre>Effective URL&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<a class="text-reset" target="_blank" href="https://tabler.io/demo">https://tabler.io/demo</a><br>Redirect count&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0<br>Name lookup time&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;3.4e-05<br>Connect time&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0.000521<br>Pre-transfer time&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0.0<br>Start-transfer time&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0.0<br>App connect time&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0.0<br>Redirect time&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0.0<br>Total time&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;28.000601<br>Response code&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0<br>Return keyword&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;operation_timedout</pre>
+                </div>
+                <h4>Response Headers</h4>
+                <div>
+                  <pre>HTTP/1.1 200 Connection established</pre>
+                </div>
+          </div>
+
           </div>
           <div class="modal-footer">
             <button type="button" class="btn me-auto" data-bs-dismiss="modal">Close</button>
-            <button type="button" class="btn btn-primary" data-bs-dismiss="modal">Save changes</button>
+            <button type="button" class="btn btn-info"> 
+              <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-refresh" width="24" height="24" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round"> <path stroke="none" d="M0 0h24v24H0z" fill="none"></path> <path d="M20 11a8.1 8.1 0 0 0 -15.5 -2m-.5 -4v4h4"></path> <path d="M4 13a8.1 8.1 0 0 0 15.5 2m.5 4v-4h-4"></path> </svg>
+                Refresh
+            </button>
           </div>
         </div>
       </div>

+ 1 - 1
controllers/site_actions.js

@@ -1,7 +1,7 @@
 const { readFileSync, writeFileSync, appendFileSync, readdirSync } = require('fs');
 const { execSync } = require("child_process");
 const { siteCard } = require('../components/siteCard');
-const { containerExec } = require('../functions/system_information')
+const { containerExec } = require('../functions/system')
 
 exports.AddSite = async function (req, res) {
 

+ 205 - 0
functions/compose.js

@@ -0,0 +1,205 @@
+const { writeFileSync, mkdirSync, readFileSync } = require("fs");
+const yaml = require('js-yaml');
+
+const { exec, execSync } = require("child_process");
+
+const { docker } = require('./system');
+
+var DockerodeCompose = require('dockerode-compose');
+
+
+module.exports.install = async function (data) {
+    
+        console.log(`[Start of install function]`);
+
+        let { service_name, name, image, command_check, command, net_mode, restart_policy } = data;        
+        let { port0, port1, port2, port3, port4, port5 } = data;
+        let { volume0, volume1, volume2, volume3, volume4, volume5 } = data;
+        let { env0, env1, env2, env3, env4, env5, env6, env7, env8, env9, env10, env11 } = data;
+        let { label0, label1, label2, label3, label4, label5, label6, label7, label8, label9, label10, label11 } = data;
+
+
+        if ((service_name.includes('caddy')) || (name.includes('caddy'))) {
+            req.app.locals.caddy = 'enabled';
+        }
+
+        let docker_volumes = [];
+
+        if (image.startsWith('https://')){
+            mkdirSync(`./appdata/${name}`, { recursive: true });
+            execSync(`curl -o ./appdata/${name}/${name}_stack.yml -L ${image}`);
+            console.log(`Downloaded stackfile: ${image}`);
+            let stackfile = yaml.load(readFileSync(`./appdata/${name}/${name}_stack.yml`, 'utf8'));
+            let services = Object.keys(stackfile.services);
+
+            for ( let i = 0; i < services.length; i++ ) {
+                try {
+                    console.log(stackfile.services[Object.keys(stackfile.services)[i]].environment);
+                } catch { console.log('no env') }
+            }
+            
+        } else {
+
+            let compose_file = `version: '3'`;
+                compose_file += `\nservices:`
+                compose_file += `\n  ${service_name}:`
+                compose_file += `\n    container_name: ${name}`;
+                compose_file += `\n    image: ${image}`;
+
+            // Command
+            if (command_check == 'on') {
+                compose_file += `\n    command: ${command}`
+            }
+
+            // Network mode
+            if (net_mode == 'host') {
+                compose_file += `\n    network_mode: 'host'`
+            }
+            else if (net_mode != 'host' && net_mode != 'docker') {
+                compose_file += `\n    network_mode: '${net_mode}'`
+            }
+            
+            // Restart policy
+            if (restart_policy != '') {
+                compose_file += `\n    restart: ${restart_policy}`
+            }
+
+            // Ports
+            if ((port0 == 'on' || port1 == 'on' || port2 == 'on' || port3 == 'on' || port4 == 'on' || port5 == 'on') && (net_mode != 'host')) {
+                compose_file += `\n    ports:`
+
+                    for (let i = 0; i < 6; i++) {
+                        if (data[`port${i}`] == 'on') {
+                            compose_file += `\n      - ${data[`port_${i}_external`]}:${data[`port_${i}_internal`]}/${data[`port_${i}_protocol`]}`
+                        }
+                    }
+            }
+
+            // Volumes
+            if (volume0 == 'on' || volume1 == 'on' || volume2 == 'on' || volume3 == 'on' || volume4 == 'on' || volume5 == 'on') {
+                compose_file += `\n    volumes:`
+
+                for (let i = 0; i < 6; i++) {
+
+                    // if volume is on and neither bind or container is empty, it's a bind mount (ex /mnt/user/appdata/config:/config  )
+                    if ((data[`volume${i}`] == 'on') && (data[`volume_${i}_bind`] != '') && (data[`volume_${i}_container`] != '')) {
+                        compose_file += `\n      - ${data[`volume_${i}_bind`]}:${data[`volume_${i}_container`]}:${data[`volume_${i}_readwrite`]}`
+                    }
+
+                    // if bind is empty create a docker volume (ex container_name_config:/config) convert any '/' in container name to '_'
+                    else if ((data[`volume${i}`] == 'on') && (data[`volume_${i}_bind`] == '') && (data[`volume_${i}_container`] != '')) {
+                        let volume_name = data[`volume_${i}_container`].replace(/\//g, '_');
+                        compose_file += `\n      - ${name}_${volume_name}:${data[`volume_${i}_container`]}:${data[`volume_${i}_readwrite`]}`
+                        docker_volumes.push(`${name}_${volume_name}`);
+                    } 
+                }
+            }
+
+            // Environment variables
+            if (env0 == 'on' || env1 == 'on' || env2 == 'on' || env3 == 'on' || env4 == 'on' || env5 == 'on' || env6 == 'on' || env7 == 'on' || env8 == 'on' || env9 == 'on' || env10 == 'on' || env11 == 'on') {
+                compose_file += `\n    environment:`
+            }
+            for (let i = 0; i < 12; i++) {
+                if (data[`env${i}`] == 'on') {
+                    compose_file += `\n      - ${data[`env_${i}_name`]}=${data[`env_${i}_default`]}`
+
+                }
+            }
+
+            // Add labels
+            if (label0 == 'on' || label1 == 'on' || label2 == 'on' || label3 == 'on' || label4 == 'on' || label5 == 'on' || label6 == 'on' || label7 == 'on' || label8 == 'on' || label9 == 'on' || label10 == 'on' || label11 == 'on') {
+                compose_file += `\n    labels:`
+            }   
+            for (let i = 0; i < 12; i++) {
+                if (data[`label${i}`] == 'on') {
+                    compose_file += `\n      - ${data[`label_${i}_name`]}=${data[`label_${i}_value`]}`
+                }
+            }
+
+            // Add privileged mode 
+
+            if (data.privileged == 'on') {
+                compose_file += `\n    privileged: true`
+            }
+
+
+            // Add hardware acceleration to the docker-compose file if one of the environment variables has the label DRINODE
+            if (env0 == 'on' || env1 == 'on' || env2 == 'on' || env3 == 'on' || env4 == 'on' || env5 == 'on' || env6 == 'on' || env7 == 'on' || env8 == 'on' || env9 == 'on' || env10 == 'on' || env11 == 'on') {
+                for (let i = 0; i < 12; i++) {
+                    if (data[`env${i}`] == 'on') {
+                        if (data[`env_${i}_name`] == 'DRINODE') {
+                            compose_file += `\n    deploy:`
+                            compose_file += `\n      resources:`
+                            compose_file += `\n        reservations:`
+                            compose_file += `\n          devices:`
+                            compose_file += `\n          - driver: nvidia`
+                            compose_file += `\n            count: 1`
+                            compose_file += `\n            capabilities: [gpu]`
+                        }
+                    }
+                }
+            }
+
+    
+            // add any docker volumes to the docker-compose file
+            if ( docker_volumes.length > 0 ) {
+                compose_file += `\n`
+                compose_file += `\nvolumes:`
+
+                // check docker_volumes for duplicates and remove them completely
+                docker_volumes = docker_volumes.filter((item, index) => docker_volumes.indexOf(item) === index)
+
+                for (let i = 0; i < docker_volumes.length; i++) {
+                    if ( docker_volumes[i] != '') {
+                        compose_file += `\n  ${docker_volumes[i]}:`
+                    }
+                }
+            }
+
+            try {   
+                mkdirSync(`./appdata/${name}`, { recursive: true });
+                writeFileSync(`./appdata/${name}/docker-compose.yml`, compose_file, function (err) { console.log(err) });
+
+            } catch { console.log('error creating directory or compose file') }
+
+            try {
+                var compose = new DockerodeCompose(docker, `./appdata/${name}/docker-compose.yml`, `${name}`);
+
+                (async () => {
+                await compose.pull();
+                await compose.up();
+                })();
+
+            } catch { console.log('error running compose file')}
+
+        }
+
+
+}
+
+
+
+module.exports.uninstall = async function (data) {
+    
+
+        if (data.confirm == 'Yes') {
+
+
+            var containerName = docker.getContainer(`${data.service_name}`);
+
+            try {
+                    containerName.stop(function (err, data) {
+                        if (data) {
+                            containerName.remove(function (err, data) {
+                            });
+                        }
+                    });
+                } catch { 
+                    containerName.remove(function (err, data) {
+                    });
+                }
+
+        }
+
+   
+}

+ 7 - 1
functions/package_manager.js

@@ -3,7 +3,8 @@ const yaml = require('js-yaml');
 
 const { exec, execSync } = require("child_process");
 
-const { docker } = require('./system_information');
+const { docker } = require('./system');
+
 var DockerodeCompose = require('dockerode-compose');
 
 
@@ -17,6 +18,11 @@ module.exports.install = async function (data) {
         let { env0, env1, env2, env3, env4, env5, env6, env7, env8, env9, env10, env11 } = data;
         let { label0, label1, label2, label3, label4, label5, label6, label7, label8, label9, label10, label11 } = data;
 
+
+        if ((service_name.includes('caddy')) || (name.includes('caddy'))) {
+            req.app.locals.caddy = 'enabled';
+        }
+
         let docker_volumes = [];
 
         if (image.startsWith('https://')){

+ 256 - 0
functions/system.js

@@ -0,0 +1,256 @@
+const { currentLoad, mem, networkStats, fsSize, dockerContainerStats } = require('systeminformation');
+var Docker = require('dockerode');
+var docker = new Docker({ socketPath: '/var/run/docker.sock' });
+const { dashCard } = require('../components/dashCard');
+
+// export docker
+module.exports.docker = docker;
+
+module.exports.serverStats = async function () {
+    const cpuUsage = await currentLoad();
+    const ramUsage = await mem();
+    const netUsage = await networkStats();
+    const diskUsage = await fsSize();
+
+    const info = {
+        cpu: Math.round(cpuUsage.currentLoad),
+        ram: Math.round((ramUsage.active / ramUsage.total) * 100),
+        tx: netUsage[0].tx_bytes,
+        rx: netUsage[0].rx_bytes,
+        disk: diskUsage[0].use,
+    };
+
+    return info;
+}
+
+
+
+module.exports.containerList = async function () {
+    let card_list = '';
+
+    const data = await docker.listContainers({ all: true });
+    for (const container of data) {
+
+
+        if ((container.Names[0].slice(1) != 'DweebUI') && (container.Names[0].slice(1) != 'DweebCache')) {
+
+            let imageVersion = container.Image.split('/');
+            let service = imageVersion[imageVersion.length - 1].split(':')[0];
+
+            let containerId = docker.getContainer(container.Id);
+            let containerInfo = await containerId.inspect();
+
+            // Get ports //////////////////////////
+            let ports_list = [];
+            try {
+                for (const [key, value] of Object.entries(containerInfo.HostConfig.PortBindings)) {
+                    let ports = {
+                        check : 'checked',
+                        external: value[0].HostPort,
+                        internal: key.split('/')[0],
+                        protocol: key.split('/')[1]
+                    }
+                    ports_list.push(ports);
+                }
+            } catch { console.log('no ports') }
+
+            for (let i = 0; i < 12; i++) {
+                if (ports_list[i] == undefined) {
+                    let ports = {
+                        check : '',
+                        external: '',
+                        internal: '',
+                        protocol: ''
+                    }
+                    ports_list[i] = ports;
+                }
+            } /////////////////////////////////////
+
+
+            // Get volumes ////////////////////////
+            let volumes_list = [];
+            try { for (const [key, value] of Object.entries(containerInfo.HostConfig.Binds)) {
+                    let volumes = {
+                        check : 'checked',
+                        bind: value.split(':')[0],
+                        container: value.split(':')[1],
+                        readwrite: value.split(':')[2]
+                    }
+                    volumes_list.push(volumes);
+                }} catch { console.log('no volumes') }
+            for (let i = 0; i < 12; i++) {
+                if (volumes_list[i] == undefined) {
+                    let volumes = {
+                        check : '',
+                        bind: '',
+                        container: '',
+                        readwrite: ''
+                    }
+                    volumes_list[i] = volumes;
+                }
+            } /////////////////////////////////////
+
+
+            // Get environment variables.
+            let environment_variables = [];
+            try { for (const [key, value] of Object.entries(containerInfo.Config.Env)) {
+                let env = {
+                    check : 'checked',
+                    name: value.split('=')[0],
+                    default: value.split('=')[1]
+                }
+                environment_variables.push(env);
+            }} catch { console.log('no env') }
+            for (let i = 0; i < 12; i++) {
+                if (environment_variables[i] == undefined) {
+                    let env = {
+                        check : '',
+                        name: '',
+                        default: ''
+                    }
+                    environment_variables[i] = env;
+                }
+            }
+
+            // Get labels.
+            let labels = [];
+            for (const [key, value] of Object.entries(containerInfo.Config.Labels)) {
+                let label = {
+                    check : 'checked',
+                    name: key,
+                    value: value
+                }
+                labels.push(label);
+            }
+            for (let i = 0; i < 12; i++) {
+                if (labels[i] == undefined) {
+                    let label = {
+                        check : '',
+                        name: '',
+                        value: ''
+                    }
+                    labels[i] = label;
+                }
+            }
+
+
+            let container_info = {
+                name: container.Names[0].slice(1),
+                service: service,
+                id: container.Id,
+                state: container.State,
+                image: container.Image,
+                external_port: ports_list[0].external || 0,
+                internal_port: ports_list[0].internal || 0, 
+                ports: ports_list,
+                volumes: volumes_list,
+                environment_variables: environment_variables,
+                labels: labels,
+            }
+
+            let dockerCard = dashCard(container_info);
+
+            card_list += dockerCard;
+        }
+        
+    }
+
+    return card_list;
+}
+
+
+
+
+
+
+
+module.exports.containerStats = async function () {
+
+    let container_stats = [];
+
+    const data = await docker.listContainers({ all: true });
+
+    for (const container of data) {
+
+        if ((container.Names[0].slice(1) != 'DweebUI') && (container.Names[0].slice(1) != 'DweebCache')) {
+            const stats = await dockerContainerStats(container.Id);
+            let container_stat = {
+                name: container.Names[0].slice(1),
+                cpu: Math.round(stats[0].cpuPercent),
+                ram: Math.round(stats[0].memPercent)
+            }
+
+            //push stats to an array
+            container_stats.push(container_stat);
+        }
+    }
+    return container_stats;
+}
+
+
+
+
+
+
+module.exports.containerAction = async function (data) {
+
+    let { user, role, action, container, state } = data;
+
+    console.log(`${user} wants to: ${action} ${container}`);
+    
+    if (role == 'admin') {
+        var containerName = docker.getContainer(container);
+
+        if ((action == 'start') && (state == 'stopped')) {
+            containerName.start();
+        } else if ((action == 'start') && (state == 'paused')) {
+            containerName.unpause();
+        } else if ((action == 'stop') && (state != 'stopped')) {
+            containerName.stop();
+        } else if ((action == 'pause') && (state == 'running')) {
+            containerName.pause();
+        } else if ((action == 'pause') && (state == 'paused')) {
+            containerName.unpause();
+        } else if (action == 'restart') {
+            containerName.restart();
+        }
+    } else {
+        console.log('User is not an admin');
+    }
+}
+
+
+
+module.exports.containerExec = async function (data) {
+
+    let { container, command } = data;
+
+    var containerName = docker.getContainer(container);
+
+    var options = {
+        Cmd: ['/bin/sh', '-c', command],
+        AttachStdout: true,
+        AttachStderr: true,
+        Tty: true
+    };
+
+    containerName.exec(options, function (err, exec) {
+        if (err) return;
+
+        exec.start(function (err, stream) {
+            if (err) return;
+
+            containerName.modem.demuxStream(stream, process.stdout, process.stderr);
+
+            exec.inspect(function (err, data) {
+                if (err) return;
+
+              
+            });
+        });
+    });
+    
+}
+
+
+

+ 5 - 4
routes/index.js

@@ -16,10 +16,6 @@ const { Register, processRegister } = require("../controllers/register");
 
 
 router.get("/", Dashboard);
-
-router.post("/install", Install)
-router.post("/uninstall", Uninstall)
-
 router.post("/addsite", AddSite)
 router.post("/removesite", RemoveSite)
 router.get("/refreshsites", RefreshSites)
@@ -27,6 +23,11 @@ router.post("/disablesite", DisableSite)
 router.post("/enablesite", EnableSite)
 
 
+router.post("/install", Install)
+router.post("/uninstall", Uninstall)
+
+
+
 router.get("/users", Users);
 
 router.get("/apps", Apps);