avatars.js 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. import { Meteor } from 'meteor/meteor';
  2. import { FilesCollection } from 'meteor/ostrio:files';
  3. import { formatFleURL } from 'meteor/ostrio:files/lib';
  4. import { isFileValid } from './fileValidation';
  5. import { createBucket } from './lib/grid/createBucket';
  6. import fs from 'fs';
  7. import path from 'path';
  8. import FileStoreStrategyFactory, { FileStoreStrategyFilesystem, FileStoreStrategyGridFs, STORAGE_NAME_FILESYSTEM } from '/models/lib/fileStoreStrategy';
  9. let avatarsUploadExternalProgram;
  10. let avatarsUploadMimeTypes = [];
  11. let avatarsUploadSize = 72000;
  12. let avatarsBucket;
  13. let storagePath;
  14. if (Meteor.isServer) {
  15. if (process.env.AVATARS_UPLOAD_MIME_TYPES) {
  16. avatarsUploadMimeTypes = process.env.AVATARS_UPLOAD_MIME_TYPES.split(',');
  17. avatarsUploadMimeTypes = avatarsUploadMimeTypes.map(value => value.trim());
  18. }
  19. if (process.env.AVATARS_UPLOAD_MAX_SIZE) {
  20. avatarsUploadSize = parseInt(process.env.AVATARS_UPLOAD_MAX_SIZE);
  21. if (isNaN(avatarsUploadSize)) {
  22. avatarsUploadSize = 0
  23. }
  24. }
  25. if (process.env.AVATARS_UPLOAD_EXTERNAL_PROGRAM) {
  26. avatarsUploadExternalProgram = process.env.AVATARS_UPLOAD_EXTERNAL_PROGRAM;
  27. if (!avatarsUploadExternalProgram.includes("{file}")) {
  28. avatarsUploadExternalProgram = undefined;
  29. }
  30. }
  31. avatarsBucket = createBucket('avatars');
  32. storagePath = path.join(process.env.WRITABLE_PATH, 'avatars');
  33. }
  34. const fileStoreStrategyFactory = new FileStoreStrategyFactory(FileStoreStrategyFilesystem, storagePath, FileStoreStrategyGridFs, avatarsBucket);
  35. Avatars = new FilesCollection({
  36. debug: false, // Change to `true` for debugging
  37. collectionName: 'avatars',
  38. allowClientCode: true,
  39. namingFunction(opts) {
  40. let filenameWithoutExtension = ""
  41. let fileId = "";
  42. if (opts?.name) {
  43. // Client
  44. filenameWithoutExtension = opts.name.replace(/(.+)\..+/, "$1");
  45. fileId = opts.meta.fileId;
  46. delete opts.meta.fileId;
  47. } else if (opts?.file?.name) {
  48. // Server
  49. if (opts.file.extension) {
  50. filenameWithoutExtension = opts.file.name.replace(new RegExp(opts.file.extensionWithDot + "$"), "")
  51. } else {
  52. // file has no extension, so don't replace anything, otherwise the last character is removed (because extensionWithDot = '.')
  53. filenameWithoutExtension = opts.file.name;
  54. }
  55. fileId = opts.fileId;
  56. }
  57. else {
  58. // should never reach here
  59. filenameWithoutExtension = Math.random().toString(36).slice(2);
  60. fileId = Math.random().toString(36).slice(2);
  61. }
  62. const ret = fileId + "-original-" + filenameWithoutExtension;
  63. // remove fileId from meta, it was only stored there to have this information here in the namingFunction function
  64. return ret;
  65. },
  66. sanitize(str, max, replacement) {
  67. // keep the original filename
  68. return str;
  69. },
  70. storagePath() {
  71. const ret = fileStoreStrategyFactory.storagePath;
  72. return ret;
  73. },
  74. onBeforeUpload(file) {
  75. if (file.size <= avatarsUploadSize && file.type.startsWith('image/')) {
  76. return true;
  77. }
  78. return 'avatar-too-big';
  79. },
  80. onAfterUpload(fileObj) {
  81. // current storage is the filesystem, update object and database
  82. Object.keys(fileObj.versions).forEach(versionName => {
  83. fileObj.versions[versionName].storage = STORAGE_NAME_FILESYSTEM;
  84. });
  85. Avatars.update({ _id: fileObj._id }, { $set: { "versions": fileObj.versions } });
  86. const isValid = Promise.await(isFileValid(fileObj, avatarsUploadMimeTypes, avatarsUploadSize, avatarsUploadExternalProgram));
  87. if (isValid) {
  88. Users.findOne(fileObj.userId).setAvatarUrl(`${formatFleURL(fileObj)}?auth=false&brokenIsFine=true`);
  89. } else {
  90. Avatars.remove(fileObj._id);
  91. }
  92. },
  93. interceptDownload(http, fileObj, versionName) {
  94. const ret = fileStoreStrategyFactory.getFileStrategy(fileObj, versionName).interceptDownload(http, this.cacheControl);
  95. return ret;
  96. },
  97. onBeforeRemove(files) {
  98. files.forEach(fileObj => {
  99. if (fileObj.userId) {
  100. Users.findOne(fileObj.userId).setAvatarUrl('');
  101. }
  102. });
  103. return true;
  104. },
  105. onAfterRemove(files) {
  106. files.forEach(fileObj => {
  107. Object.keys(fileObj.versions).forEach(versionName => {
  108. fileStoreStrategyFactory.getFileStrategy(fileObj, versionName).onAfterRemove();
  109. });
  110. });
  111. },
  112. });
  113. function isOwner(userId, doc) {
  114. return userId && userId === doc.userId;
  115. }
  116. if (Meteor.isServer) {
  117. Avatars.allow({
  118. insert: isOwner,
  119. update: isOwner,
  120. remove: isOwner,
  121. fetch: ['userId'],
  122. });
  123. Meteor.startup(() => {
  124. const storagePath = fileStoreStrategyFactory.storagePath;
  125. if (!fs.existsSync(storagePath)) {
  126. console.log("create storagePath because it doesn't exist: " + storagePath);
  127. fs.mkdirSync(storagePath, { recursive: true });
  128. }
  129. });
  130. }
  131. export default Avatars;