api.js 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. const mime = require("mime-types");
  2. const fs = require('fs');
  3. const Path = require('path');
  4. const Security = require('../src/security.js');
  5. const MediaService = require('../model/mediaService.js');
  6. const MediaFileMetaModel = require('../model/mediaItemMeta.js').MediaFileMetaModel;
  7. const MediaFileTagModel = require('../model/mediaItemTag.js').MediaFileTagModel;
  8. const { AccessModel, ACCESS_TYPE, ACCESS_GRANT } = require('../model/access.js');
  9. function MediaToJson(mediaData) {
  10. if (!mediaData)
  11. return null;
  12. if (mediaData.accessType === ACCESS_GRANT.readNoMeta)
  13. mediaData.meta = {
  14. height: mediaData.meta?.height,
  15. width: mediaData.meta?.width
  16. };
  17. return mediaData;
  18. }
  19. module.exports = { register: app => {
  20. app.router.get("/api/access/list", (req, res) => {
  21. app.routerUtils.onApiRequest(req, res);
  22. app.routerUtils.jsonResponse(res, req.sessionObj?.accessList || {});
  23. });
  24. app.router.post("/api/access/link", async (req, res) => { // /api/access/link, post: { linkIds: [string] (JSON) }
  25. app.routerUtils.onApiRequest(req, res);
  26. if (!req.post?.linkIds?.length)
  27. return app.routerUtils.httpResponse(res, 400, "Missing argument");
  28. try {
  29. for (let i of JSON.parse(req.post.linkIds)) {
  30. if (await app.databaseHelper.findOne(AccessModel, { type: ACCESS_TYPE.link, typeData: i }))
  31. Security.addLinkToSession(req, i);
  32. }
  33. }
  34. catch (err) {
  35. return app.routerUtils.onBadRequest(res);
  36. }
  37. app.routerUtils.jsonResponse(res, req.sessionObj.accessList);
  38. });
  39. app.router.del("/api/access/:id", (req, res) => {
  40. app.routerUtils.onApiRequest(req, res);
  41. const result = Security.removeFromSession(req, req.params.id);
  42. app.routerUtils.jsonResponse(res, result);
  43. });
  44. app.router.del("/api/media/:id/tag/:tag", async (req, res) => {
  45. app.routerUtils.onApiRequest(req, res);
  46. if (!req.params.id ||!req.params.tag)
  47. return app.routerUtils.onBadRequest(res);
  48. let data = await MediaService.fetchOne(app, req.params.id, req.sessionObj?.accessList);
  49. if (!data || data.accessType !== ACCESS_GRANT.write)
  50. return app.routerUtils.onPageNotFound(res);
  51. await app.databaseHelper.remove(MediaFileTagModel, { md5sum: data.fixedSum, tag: decodeURIComponent(req.params.tag), fromMeta: 0 });
  52. app.routerUtils.jsonResponse(res, MediaToJson(await MediaService.fetchOne(app, req.params.id, req.sessionObj?.accessList)));
  53. });
  54. app.router.put("/api/media/:id/tag", async (req, res) => {
  55. app.routerUtils.onApiRequest(req, res);
  56. if (!req.params.id ||!req.body?.tag)
  57. return app.routerUtils.onBadRequest(res);
  58. let data = await MediaService.fetchOne(app, req.params.id, req.sessionObj?.accessList);
  59. if (!data || data.accessType !== ACCESS_GRANT.write)
  60. return app.routerUtils.onPageNotFound(res);
  61. const requestedTag = req.body.tag;
  62. for (let existingTag of [...data.tags, ...data.fixedTags]) {
  63. if (existingTag === requestedTag || existingTag.startsWith(`${requestedTag}/`)) {
  64. return app.routerUtils.jsonResponse(res, data);
  65. }
  66. }
  67. let tag = new MediaFileTagModel(data.fixedSum, requestedTag, false);
  68. await app.databaseHelper.insertOne(tag);
  69. app.routerUtils.jsonResponse(res, MediaToJson(await MediaService.fetchOne(app, req.params.id, req.sessionObj?.accessList)));
  70. });
  71. app.router.patch("/api/media/:id/meta/:key", async (req, res) => {
  72. app.routerUtils.onApiRequest(req, res);
  73. if (!req.params.id ||!req.params.key || !Number.isInteger(req.body?.value?.length))
  74. return app.routerUtils.onBadRequest(res);
  75. let data = await MediaService.fetchOne(app, req.params.id, req.sessionObj?.accessList);
  76. if (!data || data.accessType !== ACCESS_GRANT.write)
  77. return app.routerUtils.onPageNotFound(res);
  78. if (!req.body.value) {
  79. await app.databaseHelper.remove(MediaFileMetaModel, { md5sum: data.fixedSum, key: req.params.key, fromFile: 0 });
  80. } else {
  81. let newMediaItemMedia = new MediaFileMetaModel(data.fixedSum, req.params.key, req.body.value, false);
  82. await app.databaseHelper.upsertOne(newMediaItemMedia);
  83. }
  84. app.routerUtils.jsonResponse(res, MediaToJson(await MediaService.fetchOne(app, req.params.id, req.sessionObj?.accessList)));
  85. });
  86. app.router.get("/api/media/list", async (req, res) => {
  87. app.routerUtils.onApiRequest(req, res);
  88. let first = undefined,
  89. last = undefined;
  90. if (req.body?.chronology !== undefined) {
  91. let range = await MediaService.getMediaRange(app);
  92. first = range[0];
  93. last = range[1];
  94. }
  95. let fromDate = parseInt(req.body?.from);
  96. let count = parseInt(req.body?.count);
  97. app.routerUtils.jsonResponse(res, {
  98. data: (await MediaService.fetchMediasWithAccess(
  99. app,
  100. isNaN(fromDate) ? 0 : fromDate,
  101. isNaN(count) ? 25 : Math.min(350, count),
  102. req.sessionObj?.accessList)).map(MediaToJson),
  103. first: first,
  104. last: last
  105. });
  106. });
  107. app.router.get("/api/media/:md5sum", async (req, res) => {
  108. app.routerUtils.onApiRequest(req, res);
  109. let data = MediaToJson(await MediaService.fetchOne(app, req.params.md5sum, req.sessionObj?.accessList));
  110. if (!data)
  111. return app.routerUtils.onPageNotFound(res);
  112. app.routerUtils.jsonResponse(res, data);
  113. });
  114. app.router.get("/api/media/thumbnail/:md5sum.jpg", async (req, res) => {
  115. app.routerUtils.onApiRequest(req, res);
  116. let data = await MediaService.fetchOne(app, req.params.md5sum, req.sessionObj?.accessList);
  117. if (!data)
  118. return app.routerUtils.onPageNotFound(res);
  119. try {
  120. let thumbnail = null;
  121. req.body = req.body || {};
  122. req.body.w = parseInt(req.body.w || 0);
  123. req.body.h = parseInt(req.body.h || 0);
  124. req.body.q = parseInt(req.body.q || 6);
  125. try {
  126. thumbnail = await (await app.libraryManager.findMedia(data.path))?.createThumbnail(req.body.w, req.body.h, req.body.q);
  127. } catch (err) {
  128. return app.routerUtils.apiError(res);
  129. }
  130. if (!thumbnail)
  131. return app.routerUtils.onPageNotFound(res);
  132. res.setHeader("Content-Type", "image/jpeg");
  133. res.setHeader("Content-Length", fs.statSync(thumbnail.name)?.size || undefined);
  134. res.setHeader("Cache-Control", "private, max-age=2630000"); // 1 month cache
  135. let rd = fs.createReadStream(thumbnail.name);
  136. rd.once('end', () => thumbnail.removeCallback());
  137. rd.pipe(res);
  138. }
  139. catch (err) {
  140. console.error(err);
  141. app.routerUtils.onPageNotFound(res);
  142. }
  143. });
  144. app.router.get("/api/media/original/:md5sum", async (req, res) => {
  145. app.routerUtils.onApiRequest(req, res);
  146. let data = await MediaService.fetchOne(app, req.params.md5sum, req.sessionObj?.accessList);
  147. if (!data)
  148. return app.routerUtils.onPageNotFound(res);
  149. const fileName = Path.basename(data.path);
  150. res.setHeader("Cache-Control", "private, max-age=2630000"); // 1 month cache
  151. if (data.accessType === ACCESS_GRANT.readNoMeta || req.body?.trim !== undefined) {
  152. console.log("remove meta");//-> trim metadata
  153. }
  154. res.setHeader("Content-Disposition", `attachment; filename="${fileName}"`);
  155. res.setHeader("Content-Type", mime.lookup(data.path));
  156. res.setHeader("Content-Length", fs.statSync(data.path)?.size || undefined);
  157. fs.createReadStream(data.path).pipe(res);
  158. });
  159. }};