avatars.js 4.9 KB

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