apps.js 17 KB

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