apps.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590
  1. import { readFileSync, readdirSync, renameSync, mkdirSync, unlinkSync, existsSync } from 'fs';
  2. import { parse } from 'yaml';
  3. import multer from 'multer';
  4. import AdmZip from 'adm-zip';
  5. const upload = multer({storage: multer.diskStorage({
  6. destination: function (req, file, cb) { cb(null, 'templates/tmp/') },
  7. filename: function (req, file, cb) { cb(null, file.originalname) },
  8. })});
  9. let alert = '';
  10. let templates_global = '';
  11. let json_templates = '';
  12. let remove_button = '';
  13. export const Apps = async (req, res) => {
  14. let page = Number(req.params.page) || 1;
  15. let template_param = req.params.template || 'default';
  16. if ((template_param != 'default') && (template_param != 'compose')) {
  17. remove_button = `<a href="/remove_template/${template_param}" class="btn" hx-confirm="Are you sure you want to remove this template?">Remove</a>`;
  18. } else {
  19. remove_button = '';
  20. }
  21. json_templates = '';
  22. let json_files = readdirSync('templates/json/');
  23. for (let i = 0; i < json_files.length; i++) {
  24. if (json_files[i] != 'default.json') {
  25. let filename = json_files[i].split('.')[0];
  26. let link = `<li><a class="dropdown-item" href="/apps/1/${filename}">${filename}</a></li>`
  27. json_templates += link;
  28. }
  29. }
  30. let apps_list = '';
  31. let app_count = '';
  32. let list_start = (page - 1) * 28;
  33. let list_end = (page * 28);
  34. let last_page = '';
  35. let pages = `<li class="page-item"><a class="page-link" href="/apps/1/${template_param}">1</a></li>
  36. <li class="page-item"><a class="page-link" href="/apps/2/${template_param}">2</a></li>
  37. <li class="page-item"><a class="page-link" href="/apps/3/${template_param}">3</a></li>
  38. <li class="page-item"><a class="page-link" href="/apps/4/${template_param}">4</a></li>
  39. <li class="page-item"><a class="page-link" href="/apps/5/${template_param}">5</a></li>`
  40. let prev = '/apps/' + (page - 1) + '/' + template_param;
  41. let next = '/apps/' + (page + 1) + '/' + template_param;
  42. if (page == 1) { prev = '/apps/' + (page) + '/' + template_param; }
  43. if (page == last_page) { next = '/apps/' + (page) + '/' + template_param;}
  44. if (template_param == 'compose') {
  45. let compose_files = readdirSync('templates/compose/');
  46. app_count = compose_files.length;
  47. last_page = Math.ceil(compose_files.length/28);
  48. compose_files.forEach(file => {
  49. if (file == '.gitignore') { return; }
  50. let compose = readFileSync(`templates/compose/${file}/compose.yaml`, 'utf8');
  51. let compose_data = parse(compose);
  52. let service_name = Object.keys(compose_data.services)
  53. let container = compose_data.services[service_name].container_name;
  54. let image = compose_data.services[service_name].image;
  55. let appCard = readFileSync('./views/partials/appCard.html', 'utf8');
  56. appCard = appCard.replace(/AppName/g, service_name);
  57. appCard = appCard.replace(/AppShortName/g, service_name);
  58. appCard = appCard.replace(/AppDesc/g, 'Compose File');
  59. appCard = appCard.replace(/AppLogo/g, `https://raw.githubusercontent.com/lllllllillllllillll/DweebUI-Icons/main/${service_name}.png`);
  60. appCard = appCard.replace(/AppCategories/g, '<span class="badge bg-orange-lt">Compose</span> ');
  61. appCard = appCard.replace(/AppType/g, 'compose');
  62. apps_list += appCard;
  63. });
  64. } else {
  65. let template_file = readFileSync(`./templates/json/${template_param}.json`);
  66. let templates = JSON.parse(template_file).templates;
  67. templates = templates.sort((a, b) => { if (a.name < b.name) { return -1; } });
  68. app_count = templates.length;
  69. templates_global = templates;
  70. apps_list = '';
  71. for (let i = list_start; i < list_end && i < templates.length; i++) {
  72. let appCard = readFileSync('./views/partials/appCard.html', 'utf8');
  73. let name = templates[i].name || templates[i].title.toLowerCase();
  74. let title = templates[i].title || templates[i].name;
  75. let desc = templates[i].description.slice(0, 60) + "...";
  76. let description = templates[i].description.replaceAll(". ", ".\n") || "no description available";
  77. let note = templates[i].note ? templates[i].note.replaceAll(". ", ".\n") : "no notes available";
  78. let image = templates[i].image;
  79. let logo = templates[i].logo;
  80. let categories = '';
  81. // set data.catagories to 'other' if data.catagories is empty or undefined
  82. if (templates[i].categories == null || templates[i].categories == undefined || templates[i].categories == '') {
  83. templates[i].categories = ['Other'];
  84. }
  85. // loop through the categories and add the badge to the card
  86. for (let j = 0; j < templates[i].categories.length; j++) {
  87. categories += CatagoryColor(templates[i].categories[j]);
  88. }
  89. appCard = appCard.replace(/AppName/g, name);
  90. appCard = appCard.replace(/AppTitle/g, title);
  91. appCard = appCard.replace(/AppShortName/g, name);
  92. appCard = appCard.replace(/AppDesc/g, desc);
  93. appCard = appCard.replace(/AppLogo/g, logo);
  94. appCard = appCard.replace(/AppCategories/g, categories);
  95. appCard = appCard.replace(/AppType/g, 'json');
  96. apps_list += appCard;
  97. }
  98. }
  99. res.render("apps", {
  100. name: req.session.user,
  101. role: req.session.role,
  102. avatar: req.session.user.charAt(0).toUpperCase(),
  103. list_start: list_start + 1,
  104. list_end: list_end,
  105. app_count: app_count,
  106. prev: prev,
  107. next: next,
  108. apps_list: apps_list,
  109. alert: alert,
  110. template_list: '',
  111. json_templates: json_templates,
  112. pages: pages,
  113. remove_button: remove_button
  114. });
  115. alert = '';
  116. }
  117. export const removeTemplate = async (req, res) => {
  118. let template = req.params.template;
  119. unlinkSync(`templates/json/${template}.json`);
  120. res.redirect('/apps');
  121. }
  122. export const appSearch = async (req, res) => {
  123. let search = req.body.search;
  124. let page = Number(req.params.page) || 1;
  125. let template_param = req.params.template || 'default';
  126. let template_file = readFileSync(`./templates/json/${template_param}.json`);
  127. let templates = JSON.parse(template_file).templates;
  128. templates = templates.sort((a, b) => {
  129. if (a.name < b.name) { return -1; }
  130. });
  131. let pages = `<li class="page-item"><a class="page-link" href="/apps/1/${template_param}">1</a></li>
  132. <li class="page-item"><a class="page-link" href="/apps/2/${template_param}">2</a></li>
  133. <li class="page-item"><a class="page-link" href="/apps/3/${template_param}">3</a></li>
  134. <li class="page-item"><a class="page-link" href="/apps/4/${template_param}">4</a></li>
  135. <li class="page-item"><a class="page-link" href="/apps/5/${template_param}">5</a></li>`
  136. let list_start = (page-1)*28;
  137. let list_end = (page*28);
  138. let last_page = Math.ceil(templates.length/28);
  139. let prev = '/apps/' + (page-1);
  140. let next = '/apps/' + (page+1);
  141. if (page == 1) { prev = '/apps/' + (page); }
  142. if (page == last_page) { next = '/apps/' + (page); }
  143. let apps_list = '';
  144. let results = [];
  145. let [cat_1, cat_2, cat_3] = ['','',''];
  146. function searchTemplates(terms) {
  147. terms = terms.toLowerCase();
  148. for (let i = 0; i < templates.length; i++) {
  149. if (templates[i].categories) {
  150. if (templates[i].categories[0]) {
  151. cat_1 = (templates[i].categories[0]).toLowerCase();
  152. }
  153. if (templates[i].categories[1]) {
  154. cat_2 = (templates[i].categories[1]).toLowerCase();
  155. }
  156. if (templates[i].categories[2]) {
  157. cat_3 = (templates[i].categories[2]).toLowerCase();
  158. }
  159. }
  160. if ((templates[i].description.includes(terms)) || (templates[i].name.includes(terms)) || (templates[i].title.includes(terms)) || (cat_1.includes(terms)) || (cat_2.includes(terms)) || (cat_3.includes(terms))){
  161. results.push(templates[i]);
  162. }
  163. }
  164. }
  165. searchTemplates(search);
  166. for (let i = 0; i < results.length; i++) {
  167. let appCard = readFileSync('./views/partials/appCard.html', 'utf8');
  168. let name = results[i].name || results[i].title.toLowerCase();
  169. let desc = results[i].description.slice(0, 60) + "...";
  170. let description = results[i].description.replaceAll(". ", ".\n") || "no description available";
  171. let note = results[i].note ? results[i].note.replaceAll(". ", ".\n") : "no notes available";
  172. let image = results[i].image;
  173. let logo = results[i].logo;let categories = '';
  174. // set data.catagories to 'other' if data.catagories is empty or undefined
  175. if (results[i].categories == null || results[i].categories == undefined || results[i].categories == '') {
  176. results[i].categories = ['Other'];
  177. }
  178. // loop through the categories and add the badge to the card
  179. for (let j = 0; j < results[i].categories.length; j++) {
  180. categories += CatagoryColor(results[i].categories[j]);
  181. }
  182. appCard = appCard.replace(/AppName/g, name);
  183. appCard = appCard.replace(/AppShortName/g, name);
  184. appCard = appCard.replace(/AppDesc/g, desc);
  185. appCard = appCard.replace(/AppLogo/g, logo);
  186. appCard = appCard.replace(/AppCategories/g, categories);
  187. appCard = appCard.replace(/AppType/g, 'json');
  188. apps_list += appCard;
  189. }
  190. res.render("apps", {
  191. name: req.session.user,
  192. role: req.session.role,
  193. avatar: req.session.avatar,
  194. list_start: list_start + 1,
  195. list_end: list_end,
  196. app_count: results.length,
  197. prev: prev,
  198. next: next,
  199. apps_list: apps_list,
  200. alert: alert,
  201. template_list: '',
  202. json_templates: json_templates,
  203. pages: pages,
  204. remove_button: remove_button
  205. });
  206. }
  207. function CatagoryColor(category) {
  208. switch (category) {
  209. case 'Other':
  210. return '<span class="badge bg-blue-lt">Other</span> ';
  211. case 'Productivity':
  212. return '<span class="badge bg-blue-lt">Productivity</span> ';
  213. case 'Tools':
  214. return '<span class="badge bg-blue-lt">Tools</span> ';
  215. case 'Dashboard':
  216. return '<span class="badge bg-blue-lt">Dashboard</span> ';
  217. case 'Communication':
  218. return '<span class="badge bg-azure-lt">Communication</span> ';
  219. case 'Media':
  220. return '<span class="badge bg-azure-lt">Media</span> ';
  221. case 'CMS':
  222. return '<span class="badge bg-azure-lt">CMS</span> ';
  223. case 'Monitoring':
  224. return '<span class="badge bg-indigo-lt">Monitoring</span> ';
  225. case 'LDAP':
  226. return '<span class="badge bg-purple-lt">LDAP</span> ';
  227. case 'Arr':
  228. return '<span class="badge bg-purple-lt">Arr</span> ';
  229. case 'Database':
  230. return '<span class="badge bg-red-lt">Database</span> ';
  231. case 'Paid':
  232. return '<span class="badge bg-red-lt" title="This is a paid product or contains paid features.">Paid</span> ';
  233. case 'Gaming':
  234. return '<span class="badge bg-pink-lt">Gaming</span> ';
  235. case 'Finance':
  236. return '<span class="badge bg-orange-lt">Finance</span> ';
  237. case 'Networking':
  238. return '<span class="badge bg-yellow-lt">Networking</span> ';
  239. case 'Authentication':
  240. return '<span class="badge bg-lime-lt">Authentication</span> ';
  241. case 'Development':
  242. return '<span class="badge bg-green-lt">Development</span> ';
  243. case 'Media Server':
  244. return '<span class="badge bg-teal-lt">Media Server</span> ';
  245. case 'Downloaders':
  246. return '<span class="badge bg-cyan-lt">Downloaders</span> ';
  247. default:
  248. return ''; // default to other if the category is not recognized
  249. }
  250. }
  251. export const InstallModal = async (req, res) => {
  252. let input = req.header('hx-trigger-name');
  253. let type = req.header('hx-trigger');
  254. if (type == 'compose') {
  255. let compose = readFileSync(`templates/compose/${input}/compose.yaml`, 'utf8');
  256. let modal = readFileSync('./views/modals/compose.html', 'utf8');
  257. modal = modal.replace(/AppName/g, input);
  258. modal = modal.replace(/COMPOSE_CONTENT/g, compose);
  259. res.send(modal);
  260. return;
  261. } else {
  262. let result = templates_global.find(t => t.name == input);
  263. let name = result.name || result.title.toLowerCase();
  264. let short_name = name.slice(0, 25) + "...";
  265. let desc = result.description.replaceAll(". ", ".\n") || "no description available";
  266. let short_desc = desc.slice(0, 60) + "...";
  267. let modal_name = name.replaceAll(" ", "-");
  268. let form_id = name.replaceAll("-", "_");
  269. let note = result.note ? result.note.replaceAll(". ", ".\n") : "no notes available";
  270. let command = result.command ? result.command : "";
  271. let command_check = command ? "checked" : "";
  272. let privileged = result.privileged || "";
  273. let privileged_check = privileged ? "checked" : "";
  274. let repository = result.repository || "";
  275. let image = result.image || "";
  276. let net_host, net_bridge, net_docker = '';
  277. let net_name = 'AppBridge';
  278. let restart_policy = result.restart_policy || 'unless-stopped';
  279. switch (result.network) {
  280. case 'host':
  281. net_host = 'checked';
  282. break;
  283. case 'bridge':
  284. net_bridge = 'checked';
  285. net_name = result.network;
  286. break;
  287. default:
  288. net_docker = 'checked';
  289. }
  290. if (repository != "") {
  291. image = (`${repository.url}/raw/master/${repository.stackfile}`);
  292. }
  293. let [ports_data, volumes_data, env_data, label_data] = [[], [], [], []];
  294. for (let i = 0; i < 12; i++) {
  295. // Get port details
  296. try {
  297. let ports = result.ports[i];
  298. let port_check = ports ? "checked" : "";
  299. let port_external = ports.split(":")[0] ? ports.split(":")[0] : ports.split("/")[0];
  300. let port_internal = ports.split(":")[1] ? ports.split(":")[1].split("/")[0] : ports.split("/")[0];
  301. let port_protocol = ports.split("/")[1] ? ports.split("/")[1] : "";
  302. // remove /tcp or /udp from port_external if it exists
  303. if (port_external.includes("/")) {
  304. port_external = port_external.split("/")[0];
  305. }
  306. ports_data.push({
  307. check: port_check,
  308. external: port_external,
  309. internal: port_internal,
  310. protocol: port_protocol
  311. });
  312. } catch {
  313. ports_data.push({
  314. check: "",
  315. external: "",
  316. internal: "",
  317. protocol: ""
  318. });
  319. }
  320. // Get volume details
  321. try {
  322. let volumes = result.volumes[i];
  323. let volume_check = volumes ? "checked" : "";
  324. let volume_bind = volumes.bind ? volumes.bind : "";
  325. let volume_container = volumes.container ? volumes.container.split(":")[0] : "";
  326. let volume_readwrite = "rw";
  327. if (volumes.readonly == true) {
  328. volume_readwrite = "ro";
  329. }
  330. volumes_data.push({
  331. check: volume_check,
  332. bind: volume_bind,
  333. container: volume_container,
  334. readwrite: volume_readwrite
  335. });
  336. } catch {
  337. volumes_data.push({
  338. check: "",
  339. bind: "",
  340. container: "",
  341. readwrite: ""
  342. });
  343. }
  344. // Get environment details
  345. try {
  346. let env = result.env[i];
  347. let env_check = "";
  348. let env_default = env.default ? env.default : "";
  349. if (env.set) { env_default = env.set;}
  350. let env_description = env.description ? env.description : "";
  351. let env_label = env.label ? env.label : "";
  352. let env_name = env.name ? env.name : "";
  353. env_data.push({
  354. check: env_check,
  355. default: env_default,
  356. description: env_description,
  357. label: env_label,
  358. name: env_name
  359. });
  360. } catch {
  361. env_data.push({
  362. check: "",
  363. default: "",
  364. description: "",
  365. label: "",
  366. name: ""
  367. });
  368. }
  369. // Get label details
  370. try {
  371. let label = result.labels[i];
  372. let label_check = "";
  373. let label_name = label.name ? label.name : "";
  374. let label_value = label.value ? label.value : "";
  375. label_data.push({
  376. check: label_check,
  377. name: label_name,
  378. value: label_value
  379. });
  380. } catch {
  381. label_data.push({
  382. check: "",
  383. name: "",
  384. value: ""
  385. });
  386. }
  387. }
  388. let modal = readFileSync('./views/modals/json.html', 'utf8');
  389. modal = modal.replace(/AppName/g, name);
  390. modal = modal.replace(/AppNote/g, note);
  391. modal = modal.replace(/AppImage/g, image);
  392. modal = modal.replace(/RestartPolicy/g, restart_policy);
  393. modal = modal.replace(/NetHost/g, net_host);
  394. modal = modal.replace(/NetBridge/g, net_bridge);
  395. modal = modal.replace(/NetDocker/g, net_docker);
  396. modal = modal.replace(/NetName/g, net_name);
  397. modal = modal.replace(/ModalName/g, modal_name);
  398. modal = modal.replace(/FormId/g, form_id);
  399. modal = modal.replace(/CommandCheck/g, command_check);
  400. modal = modal.replace(/CommandValue/g, command);
  401. modal = modal.replace(/PrivilegedCheck/g, privileged_check);
  402. for (let i = 0; i < 12; i++) {
  403. modal = modal.replaceAll(`Port${i}Check`, ports_data[i].check);
  404. modal = modal.replaceAll(`Port${i}External`, ports_data[i].external);
  405. modal = modal.replaceAll(`Port${i}Internal`, ports_data[i].internal);
  406. modal = modal.replaceAll(`Port${i}Protocol`, ports_data[i].protocol);
  407. modal = modal.replaceAll(`Volume${i}Check`, volumes_data[i].check);
  408. modal = modal.replaceAll(`Volume${i}Bind`, volumes_data[i].bind);
  409. modal = modal.replaceAll(`Volume${i}Container`, volumes_data[i].container);
  410. modal = modal.replaceAll(`Volume${i}RW`, volumes_data[i].readwrite);
  411. modal = modal.replaceAll(`Env${i}Check`, env_data[i].check);
  412. modal = modal.replaceAll(`Env${i}Default`, env_data[i].default);
  413. modal = modal.replaceAll(`Env${i}Description`, env_data[i].description);
  414. modal = modal.replaceAll(`Env${i}Label`, env_data[i].label);
  415. modal = modal.replaceAll(`Env${i}Name`, env_data[i].name);
  416. modal = modal.replaceAll(`Label${i}Check`, label_data[i].check);
  417. modal = modal.replaceAll(`Label${i}Name`, label_data[i].name);
  418. modal = modal.replaceAll(`Label${i}Value`, label_data[i].value);
  419. }
  420. res.send(modal);
  421. }
  422. }
  423. export const LearnMore = async (req, res) => {
  424. let name = req.header('hx-trigger-name');
  425. let id = req.header('hx-trigger');
  426. if (id == 'compose') {
  427. let modal = readFileSync('./views/modals/learnmore.html', 'utf8');
  428. modal = modal.replace(/AppName/g, name);
  429. modal = modal.replace(/AppDesc/g, 'Compose File');
  430. res.send(modal);
  431. return;
  432. }
  433. let result = templates_global.find(t => t.name == name);
  434. let modal = readFileSync('./views/modals/learnmore.html', 'utf8');
  435. modal = modal.replace(/AppName/g, result.title);
  436. modal = modal.replace(/AppDesc/g, result.description);
  437. res.send(modal);
  438. }
  439. export const ImportModal = async (req, res) => {
  440. let modal = readFileSync('./views/modals/import.html', 'utf8');
  441. res.send(modal);
  442. }
  443. export const Upload = (req, res) => {
  444. upload.array('files', 10)(req, res, () => {
  445. alert = `<div class="alert alert-success alert-dismissible mb-0 py-2" role="alert">
  446. <div class="d-flex">
  447. <div><svg xmlns="http://www.w3.org/2000/svg" class="icon alert-icon" 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="M5 12l5 5l10 -10"></path></svg> </div>
  448. <div>Template(s) Uploaded!</div>
  449. </div>
  450. <a class="btn-close" data-bs-dismiss="alert" aria-label="close" style="padding-top: 0.5rem;"></a>
  451. </div>`;
  452. let exists_alert = `<div class="alert alert-danger alert-dismissible mb-0 py-2" role="alert">
  453. <div class="d-flex">
  454. <div><svg xmlns="http://www.w3.org/2000/svg" class="icon alert-icon" 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="M5 12l5 5l10 -10"></path></svg> </div>
  455. <div>Template already exists</div>
  456. </div>
  457. <a class="btn-close" data-bs-dismiss="alert" aria-label="close" style="padding-top: 0.5rem;"></a>
  458. </div>`;
  459. let files = readdirSync('templates/tmp/');
  460. for (let i = 0; i < files.length; i++) {
  461. if (files[i].endsWith('.zip')) {
  462. let zip = new AdmZip(`templates/tmp/${files[i]}`);
  463. zip.extractAllTo('templates/compose', true);
  464. unlinkSync(`templates/tmp/${files[i]}`);
  465. } else if (files[i].endsWith('.json')) {
  466. if (existsSync(`templates/json/${files[i]}`)) {
  467. unlinkSync(`templates/tmp/${files[i]}`);
  468. alert = exists_alert;
  469. res.redirect('/apps');
  470. return;
  471. }
  472. renameSync(`templates/tmp/${files[i]}`, `templates/json/${files[i]}`);
  473. } else if (files[i].endsWith('.yml')) {
  474. let compose = readFileSync(`templates/tmp/${files[i]}`, 'utf8');
  475. let compose_data = parse(compose);
  476. let service_name = Object.keys(compose_data.services);
  477. if (existsSync(`templates/compose/${service_name}`)) {
  478. unlinkSync(`templates/tmp/${files[i]}`);
  479. alert = exists_alert;
  480. res.redirect('/apps');
  481. return;
  482. }
  483. mkdirSync(`templates/compose/${service_name}`);
  484. renameSync(`templates/tmp/${files[i]}`, `templates/compose/${service_name}/compose.yaml`);
  485. } else if (files[i].endsWith('.yaml')) {
  486. let compose = readFileSync(`templates/tmp/${files[i]}`, 'utf8');
  487. let compose_data = parse(compose);
  488. let service_name = Object.keys(compose_data.services);
  489. if (existsSync(`templates/compose/${service_name}`)) {
  490. unlinkSync(`templates/tmp/${files[i]}`);
  491. alert = exists_alert;
  492. res.redirect('/apps');
  493. return;
  494. }
  495. mkdirSync(`templates/compose/${service_name}`);
  496. renameSync(`templates/tmp/${files[i]}`, `templates/compose/${service_name}/compose.yaml`);
  497. } else {
  498. // unsupported file type
  499. unlinkSync(`templates/tmp/${files[i]}`);
  500. }
  501. }
  502. res.redirect('/apps');
  503. });
  504. };