install.js 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. import { writeFileSync, mkdirSync, readFileSync } from "fs";
  2. import yaml from 'js-yaml';
  3. import { execSync } from "child_process";
  4. import { docker } from "../server.js";
  5. import DockerodeCompose from "dockerode-compose";
  6. import { Syslog } from "../database/models.js";
  7. import { addCard } from "../controllers/dashboard.js";
  8. // This entire page hurts to look at.
  9. export const Install = async (req, res) => {
  10. let data = req.body;
  11. let { service_name, name, image, command_check, command, net_mode, restart_policy } = data;
  12. let { port0, port1, port2, port3, port4, port5 } = data;
  13. let { volume0, volume1, volume2, volume3, volume4, volume5 } = data;
  14. let { env0, env1, env2, env3, env4, env5, env6, env7, env8, env9, env10, env11 } = data;
  15. let { label0, label1, label2, label3, label4, label5, label6, label7, label8, label9, label10, label11 } = data;
  16. let ports = [port0, port1, port2, port3, port4, port5]
  17. let docker_volumes = [];
  18. addCard(name, 'installing');
  19. if (image.startsWith('https://')){
  20. mkdirSync(`./appdata/${name}`, { recursive: true });
  21. execSync(`curl -o ./appdata/${name}/${name}_stack.yml -L ${image}`);
  22. console.log(`Downloaded stackfile: ${image}`);
  23. let stackfile = yaml.load(readFileSync(`./appdata/${name}/${name}_stack.yml`, 'utf8'));
  24. let services = Object.keys(stackfile.services);
  25. for ( let i = 0; i < services.length; i++ ) {
  26. try {
  27. console.log(stackfile.services[Object.keys(stackfile.services)[i]].environment);
  28. } catch { console.log('no env') }
  29. }
  30. } else {
  31. let compose_file = `version: '3'`;
  32. compose_file += `\nservices:`
  33. compose_file += `\n ${service_name}:`
  34. compose_file += `\n container_name: ${name}`;
  35. compose_file += `\n image: ${image}`;
  36. // Command
  37. if (command_check == 'on') {
  38. compose_file += `\n command: ${command}`
  39. }
  40. // Network mode
  41. if (net_mode == 'host') {
  42. compose_file += `\n network_mode: 'host'`
  43. }
  44. else if (net_mode != 'host' && net_mode != 'docker') {
  45. compose_file += `\n network_mode: '${net_mode}'`
  46. }
  47. // Restart policy
  48. if (restart_policy != '') {
  49. compose_file += `\n restart: ${restart_policy}`
  50. }
  51. // Ports
  52. for (let i = 0; i < ports.length; i++) {
  53. if ((ports[i] == 'on') && (net_mode != 'host')) {
  54. compose_file += `\n ports:`
  55. break;
  56. }
  57. }
  58. for (let i = 0; i < ports.length; i++) {
  59. if ((ports[i] == 'on') && (net_mode != 'host')) {
  60. compose_file += `\n - ${data[`port_${i}_external`]}:${data[`port_${i}_internal`]}/${data[`port_${i}_protocol`]}`
  61. }
  62. }
  63. // Volumes
  64. let volumes = [volume0, volume1, volume2, volume3, volume4, volume5]
  65. for (let i = 0; i < volumes.length; i++) {
  66. if (volumes[i] == 'on') {
  67. compose_file += `\n volumes:`
  68. break;
  69. }
  70. }
  71. for (let i = 0; i < volumes.length; i++) {
  72. // if volume is on and neither bind or container is empty, it's a bind mount (ex /mnt/user/appdata/config:/config )
  73. if ((data[`volume${i}`] == 'on') && (data[`volume_${i}_bind`] != '') && (data[`volume_${i}_container`] != '')) {
  74. compose_file += `\n - ${data[`volume_${i}_bind`]}:${data[`volume_${i}_container`]}:${data[`volume_${i}_readwrite`]}`
  75. }
  76. // if bind is empty create a docker volume (ex container_name_config:/config) convert any '/' in container name to '_'
  77. else if ((data[`volume${i}`] == 'on') && (data[`volume_${i}_bind`] == '') && (data[`volume_${i}_container`] != '')) {
  78. let volume_name = data[`volume_${i}_container`].replace(/\//g, '_');
  79. compose_file += `\n - ${name}_${volume_name}:${data[`volume_${i}_container`]}:${data[`volume_${i}_readwrite`]}`
  80. docker_volumes.push(`${name}_${volume_name}`);
  81. }
  82. }
  83. // Environment variables
  84. let env_vars = [env0, env1, env2, env3, env4, env5, env6, env7, env8, env9, env10, env11]
  85. for (let i = 0; i < env_vars.length; i++) {
  86. if (env_vars[i] == 'on') {
  87. compose_file += `\n environment:`
  88. break;
  89. }
  90. }
  91. for (let i = 0; i < env_vars.length; i++) {
  92. if (env_vars[i] == 'on') {
  93. compose_file += `\n - ${data[`env_${i}_name`]}=${data[`env_${i}_default`]}`
  94. }
  95. }
  96. // Labels
  97. let labels = [label0, label1, label2, label3, label4, label5, label6, label7, label8, label9, label10, label11]
  98. for (let i = 0; i < labels.length; i++) {
  99. if (labels[i] == 'on') {
  100. compose_file += `\n labels:`
  101. break;
  102. }
  103. }
  104. for (let i = 0; i < 12; i++) {
  105. if (data[`label${i}`] == 'on') {
  106. compose_file += `\n - ${data[`label_${i}_name`]}=${data[`label_${i}_value`]}`
  107. }
  108. }
  109. // Privileged mode
  110. if (data.privileged == 'on') {
  111. compose_file += `\n privileged: true`
  112. }
  113. // Hardware acceleration
  114. for (let i = 0; i < env_vars.length; i++) {
  115. if ((env_vars[i] == 'on') && (data[`env_${i}_name`] == 'DRINODE')) {
  116. compose_file += `\n deploy:`
  117. compose_file += `\n resources:`
  118. compose_file += `\n reservations:`
  119. compose_file += `\n devices:`
  120. compose_file += `\n - driver: nvidia`
  121. compose_file += `\n count: 1`
  122. compose_file += `\n capabilities: [gpu]`
  123. break;
  124. }
  125. }
  126. // add any docker volumes to the docker-compose file
  127. if ( docker_volumes.length > 0 ) {
  128. compose_file += `\n`
  129. compose_file += `\nvolumes:`
  130. // check docker_volumes for duplicates and remove them completely
  131. docker_volumes = docker_volumes.filter((item, index) => docker_volumes.indexOf(item) === index)
  132. for (let i = 0; i < docker_volumes.length; i++) {
  133. if ( docker_volumes[i] != '') {
  134. compose_file += `\n ${docker_volumes[i]}:`
  135. }
  136. }
  137. }
  138. try {
  139. mkdirSync(`./appdata/${name}`, { recursive: true });
  140. writeFileSync(`./appdata/${name}/docker-compose.yml`, compose_file, function (err) { console.log(err) });
  141. var compose = new DockerodeCompose(docker, `./appdata/${name}/docker-compose.yml`, `${name}`);
  142. } catch {
  143. const syslog = await Syslog.create({
  144. user: req.session.user,
  145. email: null,
  146. event: "App Installation",
  147. message: `${name} installation failed - error creating directory or compose file : ${err}`,
  148. ip: req.socket.remoteAddress
  149. });
  150. }
  151. try {
  152. (async () => {
  153. await compose.pull();
  154. await compose.up();
  155. const syslog = await Syslog.create({
  156. user: req.session.user,
  157. email: null,
  158. event: "App Installation",
  159. message: `${name} installed successfully`,
  160. ip: req.socket.remoteAddress
  161. });
  162. })();
  163. } catch (err) {
  164. const syslog = await Syslog.create({
  165. user: req.session.user,
  166. email: null,
  167. event: "App Installation",
  168. message: `${name} installation failed: ${err}`,
  169. ip: req.socket.remoteAddress
  170. });
  171. }
  172. }
  173. res.redirect('/');
  174. }