@sync-in/server 1.9.6 → 1.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (344) hide show
  1. package/CHANGELOG.md +29 -4
  2. package/environment/environment.dist.yaml +15 -5
  3. package/package.json +13 -14
  4. package/server/app.bootstrap.js +1 -1
  5. package/server/app.bootstrap.js.map +1 -1
  6. package/server/app.constants.js +3 -2
  7. package/server/app.constants.js.map +1 -1
  8. package/server/applications/files/constants/cache.js +2 -5
  9. package/server/applications/files/constants/cache.js.map +1 -1
  10. package/server/applications/files/constants/files.js +4 -0
  11. package/server/applications/files/constants/files.js.map +1 -1
  12. package/server/applications/files/constants/operations.js +4 -0
  13. package/server/applications/files/constants/operations.js.map +1 -1
  14. package/server/applications/files/constants/routes.js +1 -26
  15. package/server/applications/files/constants/routes.js.map +1 -1
  16. package/server/applications/files/files.config.js +15 -39
  17. package/server/applications/files/files.config.js.map +1 -1
  18. package/server/applications/files/files.controller.js +4 -4
  19. package/server/applications/files/files.controller.js.map +1 -1
  20. package/server/applications/files/files.module.js +12 -9
  21. package/server/applications/files/files.module.js.map +1 -1
  22. package/server/applications/files/interfaces/file-lock.interface.js.map +1 -1
  23. package/server/applications/files/interfaces/file-props.interface.js.map +1 -1
  24. package/server/applications/files/modules/collabora-online/collabora-online-environment.decorator.js +32 -0
  25. package/server/applications/files/modules/collabora-online/collabora-online-environment.decorator.js.map +1 -0
  26. package/server/applications/files/modules/collabora-online/collabora-online-manager.service.js +280 -0
  27. package/server/applications/files/modules/collabora-online/collabora-online-manager.service.js.map +1 -0
  28. package/server/applications/files/modules/collabora-online/collabora-online-manager.service.spec.js +552 -0
  29. package/server/applications/files/modules/collabora-online/collabora-online-manager.service.spec.js.map +1 -0
  30. package/server/applications/files/modules/collabora-online/collabora-online.config.js +40 -0
  31. package/server/applications/files/modules/collabora-online/collabora-online.config.js.map +1 -0
  32. package/server/applications/files/modules/collabora-online/collabora-online.constants.js +110 -0
  33. package/server/applications/files/modules/collabora-online/collabora-online.constants.js.map +1 -0
  34. package/server/applications/files/modules/collabora-online/collabora-online.controller.js +128 -0
  35. package/server/applications/files/modules/collabora-online/collabora-online.controller.js.map +1 -0
  36. package/server/applications/files/modules/collabora-online/collabora-online.controller.spec.js +47 -0
  37. package/server/applications/files/modules/collabora-online/collabora-online.controller.spec.js.map +1 -0
  38. package/server/applications/files/{interfaces/only-office-config.interface.js → modules/collabora-online/collabora-online.dtos.js} +1 -1
  39. package/server/applications/files/modules/collabora-online/collabora-online.dtos.js.map +1 -0
  40. package/server/applications/files/{guards/files-only-office.guard.js → modules/collabora-online/collabora-online.guard.js} +7 -21
  41. package/server/applications/files/modules/collabora-online/collabora-online.guard.js.map +1 -0
  42. package/server/applications/files/modules/collabora-online/collabora-online.guard.spec.js +86 -0
  43. package/server/applications/files/modules/collabora-online/collabora-online.guard.spec.js.map +1 -0
  44. package/server/applications/files/modules/collabora-online/collabora-online.interface.js +10 -0
  45. package/server/applications/files/modules/collabora-online/collabora-online.interface.js.map +1 -0
  46. package/server/applications/files/modules/collabora-online/collabora-online.module.js +41 -0
  47. package/server/applications/files/modules/collabora-online/collabora-online.module.js.map +1 -0
  48. package/server/applications/files/modules/collabora-online/collabora-online.routes.js +35 -0
  49. package/server/applications/files/modules/collabora-online/collabora-online.routes.js.map +1 -0
  50. package/server/applications/files/modules/collabora-online/collabora-online.strategy.js +59 -0
  51. package/server/applications/files/modules/collabora-online/collabora-online.strategy.js.map +1 -0
  52. package/server/applications/files/modules/collabora-online/collabora-online.utils.js +28 -0
  53. package/server/applications/files/modules/collabora-online/collabora-online.utils.js.map +1 -0
  54. package/server/applications/files/{decorators → modules/only-office}/only-office-environment.decorator.js +5 -5
  55. package/server/applications/files/modules/only-office/only-office-environment.decorator.js.map +1 -0
  56. package/server/applications/files/{services/files-only-office-manager.service.js → modules/only-office/only-office-manager.service.js} +101 -97
  57. package/server/applications/files/modules/only-office/only-office-manager.service.js.map +1 -0
  58. package/server/applications/files/modules/only-office/only-office-manager.service.spec.js +477 -0
  59. package/server/applications/files/modules/only-office/only-office-manager.service.spec.js.map +1 -0
  60. package/server/applications/files/modules/only-office/only-office.config.js +51 -0
  61. package/server/applications/files/modules/only-office/only-office.config.js.map +1 -0
  62. package/server/applications/files/modules/only-office/only-office.constants.js +417 -0
  63. package/server/applications/files/modules/only-office/only-office.constants.js.map +1 -0
  64. package/server/applications/files/{files-only-office.controller.js → modules/only-office/only-office.controller.js} +35 -52
  65. package/server/applications/files/modules/only-office/only-office.controller.js.map +1 -0
  66. package/server/applications/files/{files-only-office.controller.spec.js → modules/only-office/only-office.controller.spec.js} +24 -21
  67. package/server/applications/files/modules/only-office/only-office.controller.spec.js.map +1 -0
  68. package/server/applications/files/modules/only-office/only-office.dtos.js +10 -0
  69. package/server/applications/files/modules/only-office/only-office.dtos.js.map +1 -0
  70. package/server/applications/files/modules/only-office/only-office.guard.js +40 -0
  71. package/server/applications/files/modules/only-office/only-office.guard.js.map +1 -0
  72. package/server/applications/files/{guards/files-only-office.guard.spec.js → modules/only-office/only-office.guard.spec.js} +15 -21
  73. package/server/applications/files/modules/only-office/only-office.guard.spec.js.map +1 -0
  74. package/server/applications/files/modules/only-office/only-office.interface.js +10 -0
  75. package/server/applications/files/modules/only-office/only-office.interface.js.map +1 -0
  76. package/server/applications/files/modules/only-office/only-office.module.js +41 -0
  77. package/server/applications/files/modules/only-office/only-office.module.js.map +1 -0
  78. package/server/applications/files/modules/only-office/only-office.routes.js +45 -0
  79. package/server/applications/files/modules/only-office/only-office.routes.js.map +1 -0
  80. package/server/applications/files/{guards/files-only-office.strategy.js → modules/only-office/only-office.strategy.js} +11 -11
  81. package/server/applications/files/modules/only-office/only-office.strategy.js.map +1 -0
  82. package/server/applications/files/services/files-lock-manager.service.js +25 -33
  83. package/server/applications/files/services/files-lock-manager.service.js.map +1 -1
  84. package/server/applications/files/services/files-manager.service.js +17 -16
  85. package/server/applications/files/services/files-manager.service.js.map +1 -1
  86. package/server/applications/files/services/files-methods.service.js +2 -2
  87. package/server/applications/files/services/files-methods.service.js.map +1 -1
  88. package/server/applications/files/services/files-methods.service.spec.js +3 -1
  89. package/server/applications/files/services/files-methods.service.spec.js.map +1 -1
  90. package/server/applications/files/services/files-scheduler.service.js +2 -2
  91. package/server/applications/files/services/files-scheduler.service.js.map +1 -1
  92. package/server/applications/files/utils/files.js +10 -2
  93. package/server/applications/files/utils/files.js.map +1 -1
  94. package/server/applications/links/constants/routes.js +5 -0
  95. package/server/applications/links/constants/routes.js.map +1 -1
  96. package/server/applications/links/interfaces/link-space.interface.js.map +1 -1
  97. package/server/applications/links/links.controller.js +25 -5
  98. package/server/applications/links/links.controller.js.map +1 -1
  99. package/server/applications/links/services/links-manager.service.js +43 -21
  100. package/server/applications/links/services/links-manager.service.js.map +1 -1
  101. package/server/applications/links/services/links-manager.service.spec.js +4 -3
  102. package/server/applications/links/services/links-manager.service.spec.js.map +1 -1
  103. package/server/applications/links/services/links-queries.service.js +9 -2
  104. package/server/applications/links/services/links-queries.service.js.map +1 -1
  105. package/server/applications/shares/interfaces/share-link.interface.js.map +1 -1
  106. package/server/applications/shares/services/shares-manager.service.js +3 -0
  107. package/server/applications/shares/services/shares-manager.service.js.map +1 -1
  108. package/server/applications/shares/services/shares-manager.service.spec.js +2 -1
  109. package/server/applications/shares/services/shares-manager.service.spec.js.map +1 -1
  110. package/server/applications/shares/services/shares-queries.service.js +1 -0
  111. package/server/applications/shares/services/shares-queries.service.js.map +1 -1
  112. package/server/applications/spaces/constants/spaces.js +2 -2
  113. package/server/applications/spaces/constants/spaces.js.map +1 -1
  114. package/server/applications/spaces/decorators/space-override-permission.decorator.js +18 -0
  115. package/server/applications/spaces/decorators/space-override-permission.decorator.js.map +1 -0
  116. package/server/applications/spaces/guards/space.guard.js +40 -33
  117. package/server/applications/spaces/guards/space.guard.js.map +1 -1
  118. package/server/applications/spaces/guards/space.guard.spec.js +10 -15
  119. package/server/applications/spaces/guards/space.guard.spec.js.map +1 -1
  120. package/server/applications/webdav/constants/webdav.js +4 -0
  121. package/server/applications/webdav/constants/webdav.js.map +1 -1
  122. package/server/applications/webdav/guards/webdav-protocol.guard.js +9 -8
  123. package/server/applications/webdav/guards/webdav-protocol.guard.js.map +1 -1
  124. package/server/applications/webdav/guards/webdav-protocol.guard.spec.js +1 -1
  125. package/server/applications/webdav/guards/webdav-protocol.guard.spec.js.map +1 -1
  126. package/server/applications/webdav/interfaces/webdav.interface.js.map +1 -1
  127. package/server/applications/webdav/services/webdav-methods.service.js +40 -17
  128. package/server/applications/webdav/services/webdav-methods.service.js.map +1 -1
  129. package/server/applications/webdav/services/webdav-methods.service.spec.js +2157 -1289
  130. package/server/applications/webdav/services/webdav-methods.service.spec.js.map +1 -1
  131. package/server/applications/webdav/utils/webdav.js +8 -4
  132. package/server/applications/webdav/utils/webdav.js.map +1 -1
  133. package/server/applications/webdav/webdav.controller.js +4 -4
  134. package/server/applications/webdav/webdav.controller.js.map +1 -1
  135. package/server/authentication/guards/auth-token-access.guard.js +8 -3
  136. package/server/authentication/guards/auth-token-access.guard.js.map +1 -1
  137. package/server/authentication/services/auth-methods/auth-method-two-fa.service.js +1 -1
  138. package/server/authentication/services/auth-methods/auth-method-two-fa.service.js.map +1 -1
  139. package/server/authentication/services/auth-methods/auth-method-two-fa.service.spec.js +350 -4
  140. package/server/authentication/services/auth-methods/auth-method-two-fa.service.spec.js.map +1 -1
  141. package/server/configuration/config.environment.js +5 -1
  142. package/server/configuration/config.environment.js.map +1 -1
  143. package/server/configuration/config.interfaces.js.map +1 -1
  144. package/static/3rdpartylicenses.txt +507 -507
  145. package/static/assets/pdfjs/build/pdf.mjs +93 -33
  146. package/static/assets/pdfjs/build/pdf.mjs.map +1 -1
  147. package/static/assets/pdfjs/build/pdf.sandbox.mjs +3 -3
  148. package/static/assets/pdfjs/build/pdf.sandbox.mjs.map +1 -1
  149. package/static/assets/pdfjs/build/pdf.worker.mjs +166 -54
  150. package/static/assets/pdfjs/build/pdf.worker.mjs.map +1 -1
  151. package/static/assets/pdfjs/version +1 -1
  152. package/static/assets/pdfjs/web/images/checkmark.svg +5 -0
  153. package/static/assets/pdfjs/web/images/pages_closeButton.svg +3 -0
  154. package/static/assets/pdfjs/web/images/pages_selected.svg +7 -0
  155. package/static/assets/pdfjs/web/images/pages_viewArrow.svg +3 -0
  156. package/static/assets/pdfjs/web/images/pages_viewButton.svg +3 -0
  157. package/static/assets/pdfjs/web/locale/be/viewer.ftl +0 -2
  158. package/static/assets/pdfjs/web/locale/bs/viewer.ftl +0 -5
  159. package/static/assets/pdfjs/web/locale/cs/viewer.ftl +4 -6
  160. package/static/assets/pdfjs/web/locale/cy/viewer.ftl +0 -2
  161. package/static/assets/pdfjs/web/locale/da/viewer.ftl +0 -2
  162. package/static/assets/pdfjs/web/locale/de/viewer.ftl +0 -2
  163. package/static/assets/pdfjs/web/locale/dsb/viewer.ftl +0 -2
  164. package/static/assets/pdfjs/web/locale/el/viewer.ftl +0 -2
  165. package/static/assets/pdfjs/web/locale/en-CA/viewer.ftl +6 -2
  166. package/static/assets/pdfjs/web/locale/en-GB/viewer.ftl +0 -2
  167. package/static/assets/pdfjs/web/locale/en-US/viewer.ftl +82 -17
  168. package/static/assets/pdfjs/web/locale/eo/viewer.ftl +0 -2
  169. package/static/assets/pdfjs/web/locale/es-AR/viewer.ftl +0 -2
  170. package/static/assets/pdfjs/web/locale/es-CL/viewer.ftl +0 -2
  171. package/static/assets/pdfjs/web/locale/es-ES/viewer.ftl +0 -2
  172. package/static/assets/pdfjs/web/locale/es-MX/viewer.ftl +0 -2
  173. package/static/assets/pdfjs/web/locale/eu/viewer.ftl +0 -2
  174. package/static/assets/pdfjs/web/locale/fi/viewer.ftl +0 -2
  175. package/static/assets/pdfjs/web/locale/fr/viewer.ftl +0 -2
  176. package/static/assets/pdfjs/web/locale/fur/viewer.ftl +0 -5
  177. package/static/assets/pdfjs/web/locale/fy-NL/viewer.ftl +3 -5
  178. package/static/assets/pdfjs/web/locale/gn/viewer.ftl +0 -2
  179. package/static/assets/pdfjs/web/locale/he/viewer.ftl +0 -2
  180. package/static/assets/pdfjs/web/locale/hr/viewer.ftl +66 -0
  181. package/static/assets/pdfjs/web/locale/hsb/viewer.ftl +0 -2
  182. package/static/assets/pdfjs/web/locale/hu/viewer.ftl +0 -2
  183. package/static/assets/pdfjs/web/locale/hy-AM/viewer.ftl +3 -8
  184. package/static/assets/pdfjs/web/locale/ia/viewer.ftl +0 -2
  185. package/static/assets/pdfjs/web/locale/id/viewer.ftl +0 -5
  186. package/static/assets/pdfjs/web/locale/is/viewer.ftl +0 -5
  187. package/static/assets/pdfjs/web/locale/it/viewer.ftl +0 -2
  188. package/static/assets/pdfjs/web/locale/ja/viewer.ftl +0 -14
  189. package/static/assets/pdfjs/web/locale/ka/viewer.ftl +4 -6
  190. package/static/assets/pdfjs/web/locale/kab/viewer.ftl +0 -5
  191. package/static/assets/pdfjs/web/locale/kk/viewer.ftl +0 -2
  192. package/static/assets/pdfjs/web/locale/ko/viewer.ftl +0 -2
  193. package/static/assets/pdfjs/web/locale/nb-NO/viewer.ftl +1 -3
  194. package/static/assets/pdfjs/web/locale/nl/viewer.ftl +0 -2
  195. package/static/assets/pdfjs/web/locale/nn-NO/viewer.ftl +4 -2
  196. package/static/assets/pdfjs/web/locale/pa-IN/viewer.ftl +0 -2
  197. package/static/assets/pdfjs/web/locale/pl/viewer.ftl +0 -2
  198. package/static/assets/pdfjs/web/locale/pt-BR/viewer.ftl +0 -2
  199. package/static/assets/pdfjs/web/locale/pt-PT/viewer.ftl +35 -0
  200. package/static/assets/pdfjs/web/locale/rm/viewer.ftl +0 -5
  201. package/static/assets/pdfjs/web/locale/ro/viewer.ftl +4 -6
  202. package/static/assets/pdfjs/web/locale/ru/viewer.ftl +3 -5
  203. package/static/assets/pdfjs/web/locale/sk/viewer.ftl +0 -2
  204. package/static/assets/pdfjs/web/locale/sl/viewer.ftl +0 -2
  205. package/static/assets/pdfjs/web/locale/sq/viewer.ftl +0 -2
  206. package/static/assets/pdfjs/web/locale/sv-SE/viewer.ftl +0 -2
  207. package/static/assets/pdfjs/web/locale/tg/viewer.ftl +0 -2
  208. package/static/assets/pdfjs/web/locale/th/viewer.ftl +2 -2
  209. package/static/assets/pdfjs/web/locale/tr/viewer.ftl +0 -2
  210. package/static/assets/pdfjs/web/locale/vi/viewer.ftl +0 -2
  211. package/static/assets/pdfjs/web/locale/zh-CN/viewer.ftl +0 -2
  212. package/static/assets/pdfjs/web/locale/zh-TW/viewer.ftl +0 -2
  213. package/static/assets/pdfjs/web/viewer.css +1778 -835
  214. package/static/assets/pdfjs/web/viewer.html +167 -86
  215. package/static/assets/pdfjs/web/viewer.mjs +1106 -801
  216. package/static/assets/pdfjs/web/viewer.mjs.map +1 -1
  217. package/static/chunk-27V66YJV.js +2 -0
  218. package/static/{chunk-BVKDW5XO.js → chunk-27Z3SYRL.js} +1 -1
  219. package/static/{chunk-HLKZCMKV.js → chunk-2RWLNKZH.js} +1 -1
  220. package/static/chunk-2YQ4SX3A.js +13 -0
  221. package/static/{chunk-GENTF6JM.js → chunk-3JYMJQYT.js} +1 -1
  222. package/static/chunk-3QTROEHV.js +1 -0
  223. package/static/{chunk-C5T7RZSD.js → chunk-3RPUQ22U.js} +1 -1
  224. package/static/{chunk-EPDWJEPD.js → chunk-3WZ6F3LC.js} +1 -1
  225. package/static/chunk-3ZLBVUCX.js +2 -0
  226. package/static/{chunk-SF6Q6VRC.js → chunk-45AZ6ZML.js} +1 -1
  227. package/static/chunk-46TJLPJY.js +1 -0
  228. package/static/chunk-4NIYCYRS.js +2 -0
  229. package/static/{chunk-GLPKRULI.js → chunk-4TPFERL6.js} +1 -1
  230. package/static/{chunk-KAAFVHYE.js → chunk-5O66CLTD.js} +1 -1
  231. package/static/chunk-6OEOADR6.js +1 -0
  232. package/static/chunk-6WMXMIE4.js +1 -0
  233. package/static/{chunk-QKMN3S4M.js → chunk-7VRYTDX4.js} +1 -1
  234. package/static/{chunk-Z2KBIZ5D.js → chunk-ARS47O5X.js} +1 -1
  235. package/static/chunk-B6HQYQYG.js +1 -0
  236. package/static/chunk-BCN4T5DO.js +2 -0
  237. package/static/{chunk-7HL5Z6PF.js → chunk-CCZWPM7Q.js} +1 -1
  238. package/static/{chunk-DU4Q4RWJ.js → chunk-CMNMPG6Z.js} +1 -1
  239. package/static/{chunk-BX3QZ7IL.js → chunk-CSVPAZHK.js} +1 -1
  240. package/static/{chunk-BJARRIS6.js → chunk-D55YR5X7.js} +4 -4
  241. package/static/{chunk-NHMYAVJK.js → chunk-D5FQ72R4.js} +1 -1
  242. package/static/{chunk-7QYALK5T.js → chunk-DGCVA6BM.js} +1 -1
  243. package/static/{chunk-IBC7CFBQ.js → chunk-DVCN3P7Q.js} +1 -1
  244. package/static/chunk-E32J777S.js +5 -0
  245. package/static/{chunk-FEQUP26G.js → chunk-FIUF2JM4.js} +1 -1
  246. package/static/{chunk-ODAQRAPO.js → chunk-G3PL6YX3.js} +1 -1
  247. package/static/chunk-G7RZN7HN.js +1 -0
  248. package/static/{chunk-IIKL33TV.js → chunk-GQHXYX6Z.js} +1 -1
  249. package/static/{chunk-5HYSNQR4.js → chunk-GWRAGN3M.js} +1 -1
  250. package/static/{chunk-ANH4VNOS.js → chunk-GXWGB7WO.js} +1 -1
  251. package/static/{chunk-25PWAXTJ.js → chunk-HGODIZTV.js} +1 -1
  252. package/static/{chunk-X5UDV4ZB.js → chunk-HZAB6F4Q.js} +1 -1
  253. package/static/chunk-I3FR3A45.js +1 -0
  254. package/static/{chunk-JYHTSSKW.js → chunk-I5SPA4G2.js} +1 -1
  255. package/static/{chunk-DQAQUSVW.js → chunk-IMFO2MI7.js} +1 -1
  256. package/static/{chunk-AYYJZMBE.js → chunk-JNTNMIUH.js} +1 -1
  257. package/static/chunk-JRXG43AA.js +2 -0
  258. package/static/{chunk-3AR5VNJE.js → chunk-KAUCN24H.js} +1 -1
  259. package/static/{chunk-3WS72A6C.js → chunk-KDUAB76O.js} +1 -1
  260. package/static/chunk-KPOQLDWF.js +1 -0
  261. package/static/{chunk-UO7ATVQG.js → chunk-KWFELZTM.js} +1 -1
  262. package/static/{chunk-2CAAJBRO.js → chunk-L3BIP4AA.js} +1 -1
  263. package/static/{chunk-DK2LAJEL.js → chunk-LGIVVJDD.js} +1 -1
  264. package/static/{chunk-5ATJIR5S.js → chunk-LNLBIJZD.js} +1 -1
  265. package/static/chunk-LTJNLOX2.js +1 -0
  266. package/static/{chunk-GRV44RYI.js → chunk-LZUHREOF.js} +1 -1
  267. package/static/{chunk-XIQXRSZ2.js → chunk-NIR4YE2E.js} +1 -1
  268. package/static/{chunk-TOCCCZP2.js → chunk-NJJURHX4.js} +1 -1
  269. package/static/chunk-NNZWSNAW.js +1 -0
  270. package/static/chunk-NWKBB7J4.js +1 -0
  271. package/static/chunk-O3YLAEVE.js +3 -0
  272. package/static/chunk-OUHCDDT6.js +1 -0
  273. package/static/{chunk-B4TDS6AQ.js → chunk-PDG7DOEF.js} +1 -1
  274. package/static/chunk-POUWUMC4.js +1 -0
  275. package/static/{chunk-NQCKX2AD.js → chunk-PPJCVBJH.js} +1 -1
  276. package/static/{chunk-ZCOEP4O2.js → chunk-PQZLR4P3.js} +1 -1
  277. package/static/chunk-PVYVY3GD.js +1 -0
  278. package/static/chunk-Q5X5TPAG.js +1 -0
  279. package/static/{chunk-LFAQLJZK.js → chunk-QHJT5H4M.js} +1 -1
  280. package/static/{chunk-D6QWQHWE.js → chunk-R4VMWCM5.js} +1 -1
  281. package/static/{chunk-O233BXWK.js → chunk-R7PLNX75.js} +1 -1
  282. package/static/chunk-RJULB733.js +1 -0
  283. package/static/{chunk-E5C4QRNQ.js → chunk-RNVPQQKT.js} +5 -5
  284. package/static/chunk-RTNEBRKJ.js +1 -0
  285. package/static/{chunk-GYYJ4FWN.js → chunk-S3TTWPQA.js} +1 -1
  286. package/static/{chunk-4GBA6EJ4.js → chunk-SDJNZULP.js} +1 -1
  287. package/static/chunk-SNOOCDJD.js +1 -0
  288. package/static/chunk-T42BV6TR.js +1 -0
  289. package/static/{chunk-5KVI243T.js → chunk-TNCKNU6I.js} +1 -1
  290. package/static/{chunk-OVUMPMVM.js → chunk-ULSPQ3HP.js} +1 -1
  291. package/static/{chunk-5NFH4E2B.js → chunk-UOK3LKSX.js} +1 -1
  292. package/static/{chunk-CHMDM2ZW.js → chunk-VD5JHSDS.js} +1 -1
  293. package/static/{chunk-2F42MZQ5.js → chunk-XBKCQCBI.js} +1 -1
  294. package/static/{chunk-2U5VKTML.js → chunk-XEWLBWFF.js} +1 -1
  295. package/static/{chunk-FSGT46LM.js → chunk-XTVNHFKX.js} +1 -1
  296. package/static/chunk-ZCSHU3D7.js +1 -0
  297. package/static/{chunk-QUUQOBTF.js → chunk-ZEJLIGAY.js} +1 -1
  298. package/static/{chunk-7H5O4BLV.js → chunk-ZHOE5VEY.js} +1 -1
  299. package/static/chunk-ZOMRIN3G.js +2 -0
  300. package/static/index.html +2 -2
  301. package/static/main-YKDNJ7LK.js +11 -0
  302. package/static/{styles-S5HVK4H5.css → styles-XLLEY5Y3.css} +1 -1
  303. package/server/applications/files/constants/only-office.js +0 -531
  304. package/server/applications/files/constants/only-office.js.map +0 -1
  305. package/server/applications/files/decorators/only-office-environment.decorator.js.map +0 -1
  306. package/server/applications/files/files-only-office.controller.js.map +0 -1
  307. package/server/applications/files/files-only-office.controller.spec.js.map +0 -1
  308. package/server/applications/files/guards/files-only-office.guard.js.map +0 -1
  309. package/server/applications/files/guards/files-only-office.guard.spec.js.map +0 -1
  310. package/server/applications/files/guards/files-only-office.strategy.js.map +0 -1
  311. package/server/applications/files/interfaces/only-office-config.interface.js.map +0 -1
  312. package/server/applications/files/services/files-only-office-manager.service.js.map +0 -1
  313. package/server/applications/files/services/files-only-office-manager.service.spec.js +0 -58
  314. package/server/applications/files/services/files-only-office-manager.service.spec.js.map +0 -1
  315. package/static/chunk-42L6C5MT.js +0 -1
  316. package/static/chunk-4ZKAVMB4.js +0 -1
  317. package/static/chunk-5GIWZKNS.js +0 -1
  318. package/static/chunk-5WCQBTXW.js +0 -1
  319. package/static/chunk-B2A4HNDC.js +0 -1
  320. package/static/chunk-BSB4VROD.js +0 -2
  321. package/static/chunk-CUC7R6C2.js +0 -1
  322. package/static/chunk-DHFQIFOF.js +0 -1
  323. package/static/chunk-DRHPEERW.js +0 -2
  324. package/static/chunk-FCGTI42I.js +0 -1
  325. package/static/chunk-FCR5AEHR.js +0 -3
  326. package/static/chunk-HB5DC7RJ.js +0 -1
  327. package/static/chunk-ITVA26X2.js +0 -2
  328. package/static/chunk-KWKZN53T.js +0 -1
  329. package/static/chunk-LBXOAKBD.js +0 -1
  330. package/static/chunk-LZKI5P5T.js +0 -1
  331. package/static/chunk-MGMDT4VN.js +0 -1
  332. package/static/chunk-MWUUM2NK.js +0 -13
  333. package/static/chunk-MYM43ENO.js +0 -1
  334. package/static/chunk-NAH4V2R6.js +0 -2
  335. package/static/chunk-PCFH5HCI.js +0 -2
  336. package/static/chunk-Q6B4OVER.js +0 -5
  337. package/static/chunk-QV5LQKTS.js +0 -1
  338. package/static/chunk-S4UTSOPV.js +0 -1
  339. package/static/chunk-SRBOO7AO.js +0 -1
  340. package/static/chunk-VZPCXSRG.js +0 -2
  341. package/static/chunk-XKEBQNQJ.js +0 -1
  342. package/static/chunk-YYTDPI5S.js +0 -1
  343. package/static/main-ODUA232E.js +0 -11
  344. /package/static/assets/pdfjs/web/images/{toolbarButton-sidebarToggle.svg → toolbarButton-viewsManagerToggle.svg} +0 -0
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-manager.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpService } from '@nestjs/axios'\nimport { HttpStatus, Injectable, Logger } from '@nestjs/common'\nimport archiver, { Archiver, ArchiverError } from 'archiver'\nimport { AxiosResponse } from 'axios'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport { Readable } from 'node:stream'\nimport { extract as extractTar } from 'tar'\nimport { FastifyAuthenticatedRequest } from '../../../authentication/interfaces/auth-request.interface'\nimport { generateThumbnail } from '../../../common/image'\nimport { ContextManager } from '../../../infrastructure/context/services/context-manager.service'\nimport { HTTP_METHOD } from '../../applications.constants'\nimport { NOTIFICATION_APP, NOTIFICATION_APP_EVENT } from '../../notifications/constants/notifications'\nimport { NotificationContent } from '../../notifications/interfaces/notification-properties.interface'\nimport { NotificationsManager } from '../../notifications/services/notifications-manager.service'\nimport { SPACE_OPERATION } from '../../spaces/constants/spaces'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { realTrashPathFromSpace } from '../../spaces/utils/paths'\nimport { canAccessToSpace, haveSpaceEnvPermissions } from '../../spaces/utils/permissions'\nimport { UserModel } from '../../users/models/user.model'\nimport { DEPTH, LOCK_DEPTH } from '../../webdav/constants/webdav'\nimport { CACHE_LOCK_FILE_TTL } from '../constants/cache'\nimport { tarGzExtension } from '../constants/compress'\nimport { COMPRESSION_EXTENSION, DEFAULT_HIGH_WATER_MARK } from '../constants/files'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { DOCUMENT_TYPE, SAMPLE_PATH_WITHOUT_EXT } from '../constants/samples'\nimport { CompressFileDto } from '../dto/file-operations.dto'\nimport { FileTaskEvent } from '../events/file-task-event'\nimport { FileDBProps } from '../interfaces/file-db-props.interface'\nimport { FileLock } from '../interfaces/file-lock.interface'\nimport { FileLockProps } from '../interfaces/file-props.interface'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport {\n checkFileName,\n copyFileContent,\n copyFiles,\n createEmptyFile,\n dirName,\n dirSize,\n fileName,\n fileSize,\n getMimeType,\n isPathExists,\n isPathIsDir,\n makeDir,\n moveFiles,\n removeFiles,\n touchFile,\n uniqueDatedFilePath,\n uniqueFilePathFromDir,\n writeFromStream,\n writeFromStreamAndChecksum\n} from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { extractZip } from '../utils/unzip-file'\nimport { regExpPrivateIP } from '../utils/url-file'\nimport { FilesLockManager } from './files-lock-manager.service'\nimport { FilesQueries } from './files-queries.service'\n\n@Injectable()\nexport class FilesManager {\n /* Spaces permissions are checked in the space guard, except for the copy/move destination */\n private logger = new Logger(FilesManager.name)\n\n constructor(\n private readonly http: HttpService,\n private readonly filesQueries: FilesQueries,\n private readonly spacesManager: SpacesManager,\n private readonly contextManager: ContextManager,\n private readonly notificationsManager: NotificationsManager,\n public readonly filesLockManager: FilesLockManager\n ) {}\n\n sendFileFromSpace(space: SpaceEnv, downloadName = ''): SendFile {\n return new SendFile(space.realPath, downloadName)\n }\n\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options: {\n checksumAlg: string\n tmpPath?: string\n }\n ): Promise<string>\n async saveStream(user: UserModel, space: SpaceEnv, req: FastifyAuthenticatedRequest, options?: any): Promise<boolean>\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options?: { dav?: { depth: LOCK_DEPTH; lockTokens: string[] }; checksumAlg?: string; tmpPath?: string }\n ): Promise<boolean | string> {\n // if tmpPath is used, we lock the final destination during the transfer\n // space.realPath is replaced by tmpPath (if allowed), if the move operation failed we remove the tmp file\n const fExists = await isPathExists(space.realPath)\n const fTmpExists = options?.tmpPath ? await isPathExists(options.tmpPath) : false\n if (fExists && req.method === HTTP_METHOD.POST) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n }\n if (fExists && (await isPathIsDir(space.realPath))) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'The location is a directory')\n }\n if (options?.tmpPath) {\n // ensure tmpPath parent dir exists\n await makeDir(dirName(options.tmpPath), true)\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n /* File Lock */\n let fileLock: FileLock | undefined\n if (options?.dav) {\n // check locks\n await this.filesLockManager.checkConflicts(space.dbFile, options?.dav?.depth || DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: options.dav?.lockTokens\n })\n } else {\n // create lock if there is no webdav context\n const [ok, lock] = await this.filesLockManager.create(user, space.dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n try {\n // check range\n let startRange = 0\n if ((fExists || fTmpExists) && req.headers['content-range']) {\n // with PUT method, some webdav clients use the `content-range` header,\n // which is normally reserved for a response to a request containing the `range` header.\n // However, for more compatibility let's accept it\n const match = /\\d+/.exec(req.headers['content-range'])\n if (!match.length) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : header is malformed')\n }\n startRange = parseInt(match[0], 10)\n const size = await fileSize(options?.tmpPath || space.realPath)\n if (startRange !== size) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : start offset does not match the current file size')\n }\n }\n // todo: check file in db to update\n // todo : versioning here\n let checksum: string\n if (options?.checksumAlg) {\n checksum = await writeFromStreamAndChecksum(options?.tmpPath || space.realPath, req.raw, startRange, options.checksumAlg)\n } else {\n await writeFromStream(options?.tmpPath || space.realPath, req.raw, startRange)\n }\n if (options?.tmpPath) {\n try {\n // ensure parent path exists\n await makeDir(path.dirname(space.realPath), true)\n // move the uploaded file to destination\n await moveFiles(options.tmpPath, space.realPath, true)\n } catch (e) {\n // cleanup tmp file\n await removeFiles(options.tmpPath)\n this.logger.error(`${this.saveStream.name} - unable to move ${options.tmpPath} -> ${space.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Unable to move tmp file to dst file')\n }\n }\n if (options?.checksumAlg) {\n return checksum\n }\n return fExists\n } finally {\n if (fileLock) {\n try {\n await this.filesLockManager.removeLock(fileLock.key)\n } catch (e) {\n this.logger.warn(`Failed to remove lock ${fileLock.key}: ${e}`)\n }\n }\n }\n }\n\n async saveMultipart(user: UserModel, space: SpaceEnv, req: FastifySpaceRequest) {\n /* Accepted methods:\n POST: Creates new resource\n PUT: Creates or fully replaces a resource at the given URI (even if intermediate paths do not exist)\n PATCH: Updates the content of an existing resource without creating a new one.\n In this text-editing scenario, locking and refreshing occur automatically, but unlocking must be handled explicitly via\n the `unlock` method.\n */\n const overwrite = req.method === HTTP_METHOD.PUT\n const patch = req.method === HTTP_METHOD.PATCH\n const realParentPath = dirName(space.realPath)\n\n if (!overwrite) {\n if (!patch && (await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (!(await isPathExists(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must exists')\n }\n if (!(await isPathIsDir(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must be a directory')\n }\n }\n\n const basePath = realParentPath + path.sep\n\n for await (const part of req.files()) {\n // If the request uses the PATCH method, the file name corresponds to the space\n const partFileName = patch ? fileName(space.realPath) : part.filename\n // `part.filename` may contain a path like foo/bar.txt\n const dstFile = path.resolve(basePath, partFileName)\n // Prevent path traversal\n if (!dstFile.startsWith(basePath)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Location is not allowed')\n }\n\n const dstDir = dirName(dstFile)\n\n if (overwrite) {\n // Prevent errors when an uploaded file would replace a directory with the same name\n // Only applies in `overwrite` cases\n if ((await isPathExists(dstFile)) && (await isPathIsDir(dstFile))) {\n // If a directory already exists at the destination path, delete it to allow overwriting with the uploaded file\n const dstUrl = path.join(path.dirname(space.url), partFileName)\n const dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.delete(user, dstSpace)\n } else if ((await isPathExists(dstDir)) && !(await isPathIsDir(dstDir))) {\n // If the destination's parent exists but is a file, remove it so we can create the directory\n const dstUrl = path.join(path.dirname(space.url), path.dirname(partFileName))\n const dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.delete(user, dstSpace)\n }\n }\n // Create the directory in the space\n if (!(await isPathExists(dstDir))) {\n await makeDir(dstDir, true)\n }\n // Create or refresh lock\n const dbFile = { ...space.dbFile, path: path.join(dirName(space.dbFile.path), partFileName) }\n // Use a short TTL for the PATCH method (which is also used for refreshing)\n const ttl = patch ? CACHE_LOCK_FILE_TTL : undefined\n const [created, fileLock] = await this.filesLockManager.createOrRefresh(user, dbFile, DEPTH.RESOURCE, ttl)\n // Do\n try {\n await writeFromStream(dstFile, part.file)\n } finally {\n if (!patch && created) {\n // Remove the file lock only if it has not been refreshed\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n if (patch) {\n // Only one resource can be updated with the PATCH method.\n break\n }\n }\n }\n\n async touch(user: UserModel, space: SpaceEnv, mtime: number, checkLocks = true): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // todo: update mtime in last files ( & in db file ?)\n await touchFile(space.realPath, mtime)\n }\n\n async mkFile(user: UserModel, space: SpaceEnv, overwrite = false, checkLocks = true, checkDocument = false): Promise<void> {\n checkFileName(space.realPath)\n if (!overwrite && (await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // use sample documents when possible\n const fileExtension = path.extname(space.realPath)\n if (checkDocument && fileExtension !== '.txt' && Object.values(DOCUMENT_TYPE).indexOf(fileExtension) > -1) {\n const srcSample = path.join(__dirname, `${SAMPLE_PATH_WITHOUT_EXT}${fileExtension}`)\n return copyFileContent(srcSample, space.realPath)\n } else {\n return createEmptyFile(space.realPath)\n }\n }\n\n async mkDir(user: UserModel, space: SpaceEnv, recursive = false, dav?: { depth: LOCK_DEPTH; lockTokens: string[] }): Promise<void> {\n checkFileName(space.realPath)\n if (!recursive) {\n if (await isPathExists(space.realPath)) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n await this.filesLockManager.checkConflicts(space.dbFile, dav?.depth || DEPTH.RESOURCE, { userId: user.id, lockTokens: dav?.lockTokens })\n await makeDir(space.realPath, recursive)\n }\n\n async copyMove(\n user: UserModel,\n srcSpace: SpaceEnv,\n dstSpace: SpaceEnv,\n isMove: boolean,\n overwrite = false,\n mkdirDstParentPath = false,\n dav?: { depth: LOCK_DEPTH; lockTokens: string[] }\n ): Promise<void> {\n // checks\n if (!canAccessToSpace(user, dstSpace)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to access to this space repository : ${dstSpace.repository}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to access to this repository')\n }\n if (!haveSpaceEnvPermissions(dstSpace, SPACE_OPERATION.ADD)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to copy/move on this space : *${dstSpace.alias}* (${dstSpace.id}) : ${dstSpace.url}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to copy/move on the destination')\n }\n if (dstSpace.quotaIsExceeded) {\n this.logger.warn(`${this.copyMove.name} - quota is exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Quota is exceeded')\n }\n if (!(await isPathExists(srcSpace.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (!(await isPathExists(dirName(dstSpace.realPath)))) {\n if (mkdirDstParentPath) {\n try {\n await makeDir(dirName(dstSpace.realPath), true)\n } catch (e) {\n this.logger.error(`${this.copyMove.name} - Cannot create parent directory for destination ${dstSpace.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Cannot create parent directory for destination')\n }\n } else {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n if (srcSpace.realPath === dstSpace.realPath) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source onto itself')\n }\n if (`${dstSpace.realPath}/`.startsWith(`${srcSpace.realPath}/`)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source below itself')\n }\n if (dirName(srcSpace.url) === dirName(dstSpace.url) && dirName(srcSpace.realPath) !== dirName(dstSpace.realPath)) {\n /* Handle renaming a space file with the same name as a space root :\n srcSpace.url = '/space/sync-in/code2.ts' (a space file)\n srcSpace.realPath = '/home/sync-in/spaces/sync-in/code2.ts\n dstSpace.url = '/space/sync-in/code.ts' (a space root)\n dstSpace.realPath = '/home/sync-in/users/jo/files/code2.ts !!\n */\n throw new FileError(HttpStatus.BAD_REQUEST, 'An anchored file already has this name')\n }\n if (!overwrite && (await isPathExists(dstSpace.realPath))) {\n /* Handle case-sensitive (in renaming context):\n srcSpace.url = '/space/sync-in/code.ts'\n dstSpace.url = '/space/sync-in/code.TS'\n The destination exists because it's the same file, bypass this\n */\n if (!(isMove && srcSpace.realPath.toLowerCase() === dstSpace.realPath.toLowerCase())) {\n throw new FileError(dav ? HttpStatus.PRECONDITION_FAILED : HttpStatus.BAD_REQUEST, 'The destination already exists')\n }\n }\n\n const isDir = await isPathIsDir(srcSpace.realPath)\n\n if (dstSpace.storageQuota) {\n /* Skip validation when moving to the same space; for copy operations, run all checks. */\n if (!isMove || (isMove && srcSpace.id !== dstSpace.id)) {\n const size = isDir ? (await dirSize(srcSpace.realPath))[0] : await fileSize(srcSpace.realPath)\n if (dstSpace.willExceedQuota(size)) {\n this.logger.warn(`${this.copyMove.name} - storage quota will be exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Storage quota will be exceeded')\n }\n }\n }\n\n // check lock conflicts on source and destination\n let recursive: boolean\n let depth: LOCK_DEPTH\n if (dav?.depth) {\n recursive = dav.depth === DEPTH.INFINITY\n depth = dav.depth\n } else {\n recursive = isDir\n depth = recursive ? DEPTH.INFINITY : DEPTH.RESOURCE\n }\n if (isMove) {\n // check source\n await this.filesLockManager.checkConflicts(srcSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n }\n // check destination\n await this.filesLockManager.checkConflicts(dstSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n\n // overwrite\n if (overwrite && (await isPathExists(dstSpace.realPath))) {\n // todo : versioning here\n await this.delete(user, dstSpace)\n }\n\n // send it to task watcher\n if (srcSpace.task?.cacheKey) {\n if (!isDir) srcSpace.task.props.totalSize = await fileSize(srcSpace.realPath)\n FileTaskEvent.emit('startWatch', srcSpace, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, dstSpace.realPath)\n }\n\n // do\n if (isMove) {\n await moveFiles(srcSpace.realPath, dstSpace.realPath, overwrite)\n return this.filesQueries.moveFiles(srcSpace.dbFile, dstSpace.dbFile, isDir)\n }\n return copyFiles(srcSpace.realPath, dstSpace.realPath, overwrite, recursive)\n }\n\n async delete(user: UserModel, space: SpaceEnv, dav?: { lockTokens: string[] }): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n // check lock conflicts\n const isDir = await isPathIsDir(space.realPath)\n await this.filesLockManager.checkConflicts(space.dbFile, isDir ? DEPTH.INFINITY : DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: dav?.lockTokens\n })\n // file system deletion\n let forceDeleteInDB = false\n if (space.inTrashRepository) {\n await removeFiles(space.realPath)\n } else {\n const baseTrashPath = realTrashPathFromSpace(user, space)\n if (baseTrashPath) {\n const name = fileName(space.realPath)\n const trashDir = path.join(baseTrashPath, dirName(space.dbFile.path))\n const trashFile = path.join(trashDir, name)\n if (!(await isPathExists(trashDir))) {\n await makeDir(trashDir, true)\n }\n if (await isPathExists(trashFile)) {\n // if a resource already exists in the trash, rename it with the date\n const dstTrash = await uniqueDatedFilePath(trashFile)\n // move the resource on fs\n await moveFiles(trashFile, dstTrash.path)\n // move the resource in db\n const trashFileDB: FileDBProps = { ...space.dbFile, inTrash: true }\n const dstTrashFileDB: FileDBProps = { ...trashFileDB, path: path.join(dirName(trashFileDB.path), fileName(dstTrash.path)) }\n await this.filesQueries.moveFiles(trashFileDB, dstTrashFileDB, dstTrash.isDir)\n }\n await moveFiles(space.realPath, trashFile, true)\n } else {\n // unsupported case: delete the file (this shouldn't happen)\n this.logger.error(`Unable to find trash path for space - *${space.alias}* (${space.id}) : delete permanently : ${space.realPath}`)\n forceDeleteInDB = true\n await removeFiles(space.realPath)\n }\n }\n // remove locks, these locks have already been checked in the `checkConflicts` function\n if (isDir) {\n this.filesLockManager.removeChildLocks(user, space.dbFile).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n for (const lock of await this.filesLockManager.getLocksByPath(space.dbFile)) {\n this.filesLockManager.removeLock(lock.key).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n // delete or move to trash the files in db\n return this.filesQueries.deleteFiles(space.dbFile, isDir, forceDeleteInDB)\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, url: string): Promise<void> {\n this.logger.log(`${this.downloadFromUrl.name} : ${url}`)\n // create lock\n const rPath = await uniqueFilePathFromDir(space.realPath)\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(space.realPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task.cacheKey) {\n let headRes: AxiosResponse\n\n try {\n headRes = await this.http.axiosRef({ method: HTTP_METHOD.HEAD, url: url, maxRedirects: 1 })\n } catch (e) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n this.logger.error(`${this.downloadFromUrl.name} - ${url} : ${e}`)\n throw new FileError(HttpStatus.BAD_REQUEST, 'Unable to download file')\n }\n\n if (regExpPrivateIP.test(headRes.request.socket.remoteAddress)) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n // prevent SSRF attack\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n\n // attempt to retrieve the Content-Length header\n try {\n if ('content-length' in headRes.headers) {\n space.task.props.totalSize = parseInt(headRes.headers['content-length'], 10) || null\n }\n } catch (e) {\n this.logger.debug(`${this.downloadFromUrl.name} - content-length : ${e}`)\n }\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DOWNLOAD, rPath)\n }\n // do\n try {\n const getRes = await this.http.axiosRef({ method: HTTP_METHOD.GET, url: url, responseType: 'stream', maxRedirects: 1 })\n if (regExpPrivateIP.test(getRes.request.socket.remoteAddress)) {\n // Prevent SSRF attacks and perform a DNS-rebinding check if a HEAD request has already been made\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n await writeFromStream(rPath, getRes.data)\n } finally {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, dto: CompressFileDto): Promise<void> {\n // This method is currently used only by files-methods.service, which handles input sanitization.\n // If it is used in other services in the future, make sure to refactor accordingly to sanitize inputs properly.\n const srcPath = dirName(space.realPath)\n // todo: a guest link tasksPath should be in specific directory (guest link has no home)\n const archiveExt = dto.name.endsWith(dto.extension) ? '' : `.${dto.extension}`\n const dstPath = await uniqueFilePathFromDir(path.join(dto.compressInDirectory ? srcPath : user.tasksPath, `${dto.name}${archiveExt}`))\n const archive: Archiver = archiver('tar', {\n gzip: dto.extension === tarGzExtension,\n gzipOptions: {\n level: 9\n }\n })\n // create lock\n let fileLock: FileLock\n if (dto.compressInDirectory) {\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, lock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n if (space.task?.cacheKey) {\n space.task.props.compressInDirectory = dto.compressInDirectory\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.COMPRESS, dstPath)\n }\n // do\n try {\n archive.on('error', (error: ArchiverError) => {\n throw error\n })\n const dstStream = fs.createWriteStream(dstPath, { highWaterMark: DEFAULT_HIGH_WATER_MARK })\n archive.pipe(dstStream)\n for (const f of dto.files) {\n if (await isPathIsDir(f.path)) {\n archive.directory(f.path, dto.files.length > 1 ? fileName(f.path) : false)\n } else {\n archive.file(f.path, {\n name: f.rootAlias ? f.name : fileName(f.path)\n })\n }\n }\n await archive.finalize()\n } finally {\n if (fileLock) {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n // checks\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n const extension = path.extname(space.realPath)\n if (!COMPRESSION_EXTENSION.has(extension)) {\n throw new FileError(HttpStatus.BAD_REQUEST, `${extension} is not supported`)\n }\n // make destination folder\n const dstPath = await uniqueFilePathFromDir(path.join(dirName(space.realPath), path.basename(space.realPath, extension)))\n await makeDir(dstPath)\n // create lock\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.INFINITY)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task?.cacheKey) FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DECOMPRESS, dstPath)\n // do\n try {\n if (extension === '.zip') {\n await extractZip(space.realPath, dstPath)\n } else {\n await extractTar({\n file: space.realPath,\n cwd: dstPath,\n gzip: COMPRESSION_EXTENSION.get(extension) === tarGzExtension,\n preserveOwner: false\n })\n }\n } finally {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async generateThumbnail(space: SpaceEnv, size: number): Promise<Readable> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (getMimeType(space.realPath, false).indexOf('image') === -1) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n try {\n return generateThumbnail(space.realPath, size)\n } catch (e) {\n this.logger.warn(e)\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n }\n\n async lock(user: UserModel, space: SpaceEnv): Promise<FileLockProps> {\n const rExists = await isPathExists(space.realPath)\n if (!rExists) {\n this.logger.warn('Lock refresh must specify an existing resource')\n throw new FileError(HttpStatus.BAD_REQUEST, 'Lock refresh must specify an existing resource')\n }\n const [_created, lock] = await this.filesLockManager.createOrRefresh(user, space.dbFile, DEPTH.RESOURCE, CACHE_LOCK_FILE_TTL)\n return this.filesLockManager.convertLockToFileLockProps(lock)\n }\n\n async unlock(user: UserModel, space: SpaceEnv, forceAsOwner = false): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n this.logger.warn(`Unable to unlock: ${space.url} - resource does not exist`)\n throw new FileError(HttpStatus.BAD_REQUEST, 'Unlock must specify an existing resource')\n }\n const fileLocks = await this.filesLockManager.getLocksByPath(space.dbFile)\n if (fileLocks.length === 0) {\n this.logger.warn(`Unable to find lock: ${space.url} - resource does not exist`)\n return\n }\n for (const lock of fileLocks) {\n if ((forceAsOwner && space.dbFile?.ownerId === user.id) || lock.owner.id === user.id) {\n // Refresh if more than half of the TTL has passed\n await this.filesLockManager.removeLock(lock.key)\n } else {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n }\n }\n\n async unlockRequest(user: UserModel, space: SpaceEnv): Promise<void> {\n const fileLocks = await this.filesLockManager.getLocksByPath(space.dbFile)\n if (fileLocks.length === 0) {\n this.logger.warn(`Unable to find lock: ${space.url} - resource does not exist`)\n throw new FileError(HttpStatus.NOT_FOUND, 'Lock not found')\n }\n for (const lock of fileLocks) {\n if (lock.owner.id !== user.id) {\n const notification: NotificationContent = {\n app: NOTIFICATION_APP.UNLOCK_REQUEST,\n event: NOTIFICATION_APP_EVENT.UNLOCK_REQUEST,\n element: fileName(space.url),\n url: dirName(space.url)\n }\n this.notificationsManager\n .create([lock.owner.id], notification, {\n author: user,\n currentUrl: this.contextManager.headerOriginUrl()\n })\n .catch((e: Error) => this.logger.error(`${this.unlockRequest.name} - ${e}`))\n }\n }\n }\n\n async getSize(space: SpaceEnv): Promise<number> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (await isPathIsDir(space.realPath)) {\n return (await dirSize(space.realPath))[0]\n } else {\n return await fileSize(space.realPath)\n }\n }\n}\n"],"names":["FilesManager","sendFileFromSpace","space","downloadName","SendFile","realPath","saveStream","user","req","options","fExists","isPathExists","fTmpExists","tmpPath","method","HTTP_METHOD","POST","FileError","HttpStatus","METHOD_NOT_ALLOWED","isPathIsDir","makeDir","dirName","CONFLICT","fileLock","dav","filesLockManager","checkConflicts","dbFile","depth","DEPTH","RESOURCE","userId","id","lockTokens","ok","lock","create","LockConflict","startRange","headers","match","exec","length","BAD_REQUEST","parseInt","size","fileSize","checksum","checksumAlg","writeFromStreamAndChecksum","raw","writeFromStream","path","dirname","moveFiles","e","removeFiles","logger","error","name","INTERNAL_SERVER_ERROR","removeLock","key","warn","saveMultipart","overwrite","PUT","patch","PATCH","realParentPath","basePath","sep","part","files","partFileName","fileName","filename","dstFile","resolve","startsWith","FORBIDDEN","dstDir","dstUrl","join","url","dstSpace","spacesManager","spaceEnv","split","delete","ttl","CACHE_LOCK_FILE_TTL","undefined","created","createOrRefresh","file","touch","mtime","checkLocks","NOT_FOUND","touchFile","mkFile","checkDocument","checkFileName","fileExtension","extname","Object","values","DOCUMENT_TYPE","indexOf","srcSample","__dirname","SAMPLE_PATH_WITHOUT_EXT","copyFileContent","createEmptyFile","mkDir","recursive","copyMove","srcSpace","isMove","mkdirDstParentPath","canAccessToSpace","repository","haveSpaceEnvPermissions","SPACE_OPERATION","ADD","alias","quotaIsExceeded","INSUFFICIENT_STORAGE","toLowerCase","PRECONDITION_FAILED","isDir","storageQuota","dirSize","willExceedQuota","INFINITY","task","cacheKey","props","totalSize","FileTaskEvent","emit","FILE_OPERATION","MOVE","COPY","filesQueries","copyFiles","forceDeleteInDB","inTrashRepository","baseTrashPath","realTrashPathFromSpace","trashDir","trashFile","dstTrash","uniqueDatedFilePath","trashFileDB","inTrash","dstTrashFileDB","removeChildLocks","catch","getLocksByPath","deleteFiles","downloadFromUrl","log","rPath","uniqueFilePathFromDir","headRes","http","axiosRef","HEAD","maxRedirects","regExpPrivateIP","test","request","socket","remoteAddress","debug","DOWNLOAD","getRes","GET","responseType","data","compress","dto","srcPath","archiveExt","endsWith","extension","dstPath","compressInDirectory","tasksPath","archive","archiver","gzip","tarGzExtension","gzipOptions","level","COMPRESS","on","dstStream","fs","createWriteStream","highWaterMark","DEFAULT_HIGH_WATER_MARK","pipe","f","directory","rootAlias","finalize","decompress","COMPRESSION_EXTENSION","has","basename","DECOMPRESS","extractZip","extractTar","cwd","get","preserveOwner","generateThumbnail","getMimeType","rExists","_created","convertLockToFileLockProps","unlock","forceAsOwner","fileLocks","ownerId","owner","unlockRequest","notification","app","NOTIFICATION_APP","UNLOCK_REQUEST","event","NOTIFICATION_APP_EVENT","element","notificationsManager","author","currentUrl","contextManager","headerOriginUrl","getSize","Logger"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-manager.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpService } from '@nestjs/axios'\nimport { HttpStatus, Injectable, Logger } from '@nestjs/common'\nimport archiver, { Archiver, ArchiverError } from 'archiver'\nimport { AxiosResponse } from 'axios'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport { Readable } from 'node:stream'\nimport { extract as extractTar } from 'tar'\nimport { FastifyAuthenticatedRequest } from '../../../authentication/interfaces/auth-request.interface'\nimport { generateThumbnail } from '../../../common/image'\nimport { SERVER_NAME } from '../../../common/shared'\nimport { ContextManager } from '../../../infrastructure/context/services/context-manager.service'\nimport { HTTP_METHOD } from '../../applications.constants'\nimport { NOTIFICATION_APP, NOTIFICATION_APP_EVENT } from '../../notifications/constants/notifications'\nimport { NotificationContent } from '../../notifications/interfaces/notification-properties.interface'\nimport { NotificationsManager } from '../../notifications/services/notifications-manager.service'\nimport { SPACE_OPERATION } from '../../spaces/constants/spaces'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { realTrashPathFromSpace } from '../../spaces/utils/paths'\nimport { canAccessToSpace, haveSpaceEnvPermissions } from '../../spaces/utils/permissions'\nimport { UserModel } from '../../users/models/user.model'\nimport { DEPTH, LOCK_DEPTH } from '../../webdav/constants/webdav'\nimport { CACHE_LOCK_FILE_TTL } from '../constants/cache'\nimport { tarGzExtension } from '../constants/compress'\nimport { COMPRESSION_EXTENSION, DEFAULT_HIGH_WATER_MARK } from '../constants/files'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { DOCUMENT_TYPE, SAMPLE_PATH_WITHOUT_EXT } from '../constants/samples'\nimport { CompressFileDto } from '../dto/file-operations.dto'\nimport { FileTaskEvent } from '../events/file-task-event'\nimport { FileDBProps } from '../interfaces/file-db-props.interface'\nimport { FileLock } from '../interfaces/file-lock.interface'\nimport { FileLockProps } from '../interfaces/file-props.interface'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport {\n checkFileName,\n copyFileContent,\n copyFiles,\n createEmptyFile,\n dirName,\n dirSize,\n fileName,\n fileSize,\n getMimeType,\n isPathExists,\n isPathIsDir,\n makeDir,\n moveFiles,\n removeFiles,\n touchFile,\n uniqueDatedFilePath,\n uniqueFilePathFromDir,\n writeFromStream,\n writeFromStreamAndChecksum\n} from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { extractZip } from '../utils/unzip-file'\nimport { regExpPrivateIP } from '../utils/url-file'\nimport { FilesLockManager } from './files-lock-manager.service'\nimport { FilesQueries } from './files-queries.service'\n\n@Injectable()\nexport class FilesManager {\n /* Spaces permissions are checked in the space guard, except for the copy/move destination */\n private logger = new Logger(FilesManager.name)\n\n constructor(\n private readonly http: HttpService,\n private readonly filesQueries: FilesQueries,\n private readonly spacesManager: SpacesManager,\n private readonly contextManager: ContextManager,\n private readonly notificationsManager: NotificationsManager,\n public readonly filesLockManager: FilesLockManager\n ) {}\n\n sendFileFromSpace(space: SpaceEnv, downloadName = ''): SendFile {\n return new SendFile(space.realPath, downloadName)\n }\n\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options: {\n checksumAlg: string\n tmpPath?: string\n }\n ): Promise<string>\n async saveStream(user: UserModel, space: SpaceEnv, req: FastifyAuthenticatedRequest, options?: any): Promise<boolean>\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options?: { dav?: { depth: LOCK_DEPTH; lockTokens: string[] }; checksumAlg?: string; tmpPath?: string }\n ): Promise<boolean | string> {\n // If tmpPath is used, we lock the final destination during the transfer\n // space.realPath is replaced by tmpPath (if allowed). If the move operation failed, we remove the tmp file\n const fExists = await isPathExists(space.realPath)\n const fTmpExists = options?.tmpPath ? await isPathExists(options.tmpPath) : false\n if (fExists && req.method === HTTP_METHOD.POST) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n }\n if (fExists && (await isPathIsDir(space.realPath))) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'The location is a directory')\n }\n if (options?.tmpPath) {\n // Ensure tmpPath parent dir exists\n await makeDir(dirName(options.tmpPath), true)\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n /* File Lock */\n let fileLock: FileLock | undefined\n if (options?.dav) {\n // Check locks\n await this.filesLockManager.checkConflicts(space.dbFile, options?.dav?.depth || DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: options.dav?.lockTokens\n })\n } else {\n // Create lock if there is no webdav context\n const [ok, lock] = await this.filesLockManager.create(user, space.dbFile, SERVER_NAME, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n try {\n // Check range\n let startRange = 0\n if ((fExists || fTmpExists) && req.headers['content-range']) {\n // With PUT method, some webdav clients use the `content-range` header,\n // which is normally reserved for a response to a request containing the `range` header.\n // However, for more compatibility let's accept it.\n const match = /\\d+/.exec(req.headers['content-range'])\n if (!match.length) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : header is malformed')\n }\n startRange = parseInt(match[0], 10)\n const size = await fileSize(options?.tmpPath || space.realPath)\n if (startRange !== size) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : start offset does not match the current file size')\n }\n }\n // todo: check file in db to update\n // todo : versioning here\n let checksum: string\n if (options?.checksumAlg) {\n checksum = await writeFromStreamAndChecksum(options?.tmpPath || space.realPath, req.raw, startRange, options.checksumAlg)\n } else {\n await writeFromStream(options?.tmpPath || space.realPath, req.raw, startRange)\n }\n if (options?.tmpPath) {\n try {\n // ensure parent path exists\n await makeDir(path.dirname(space.realPath), true)\n // move the uploaded file to destination\n await moveFiles(options.tmpPath, space.realPath, true)\n } catch (e) {\n // cleanup tmp file\n await removeFiles(options.tmpPath)\n this.logger.error(`${this.saveStream.name} - unable to move ${options.tmpPath} -> ${space.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Unable to move tmp file to dst file')\n }\n }\n if (options?.checksumAlg) {\n return checksum\n }\n return fExists\n } finally {\n if (fileLock) {\n try {\n await this.filesLockManager.removeLock(fileLock.key)\n } catch (e) {\n this.logger.warn(`Failed to remove lock ${fileLock.key}: ${e}`)\n }\n }\n }\n }\n\n async saveMultipart(user: UserModel, space: SpaceEnv, req: FastifySpaceRequest) {\n /* Accepted methods:\n POST: Creates new resource\n PUT: Creates or fully replaces a resource at the given URI (even if intermediate paths do not exist)\n PATCH: Updates the content of an existing resource without creating a new one.\n In this text-editing scenario, locking and refreshing occur automatically, but unlocking must be handled explicitly via\n the `unlock` method.\n */\n const overwrite = req.method === HTTP_METHOD.PUT\n const patch = req.method === HTTP_METHOD.PATCH\n const realParentPath = dirName(space.realPath)\n\n if (!overwrite) {\n if (!patch && (await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (!(await isPathExists(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must exists')\n }\n if (!(await isPathIsDir(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must be a directory')\n }\n }\n\n const basePath = realParentPath + path.sep\n\n for await (const part of req.files()) {\n // If the request uses the PATCH method, the file name corresponds to the space\n const partFileName = patch ? fileName(space.realPath) : part.filename\n // `part.filename` may contain a path like foo/bar.txt\n const dstFile = path.resolve(basePath, partFileName)\n // Prevent path traversal\n if (!dstFile.startsWith(basePath)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Location is not allowed')\n }\n\n const dstDir = dirName(dstFile)\n\n if (overwrite) {\n // Prevent errors when an uploaded file would replace a directory with the same name\n // Only applies in `overwrite` cases\n if ((await isPathExists(dstFile)) && (await isPathIsDir(dstFile))) {\n // If a directory already exists at the destination path, delete it to allow overwriting with the uploaded file\n const dstUrl = path.join(path.dirname(space.url), partFileName)\n const dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.delete(user, dstSpace)\n } else if ((await isPathExists(dstDir)) && !(await isPathIsDir(dstDir))) {\n // If the destination's parent exists but is a file, remove it so we can create the directory\n const dstUrl = path.join(path.dirname(space.url), path.dirname(partFileName))\n const dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.delete(user, dstSpace)\n }\n }\n // Create the directory in the space\n if (!(await isPathExists(dstDir))) {\n await makeDir(dstDir, true)\n }\n // Create or refresh lock\n const dbFile = { ...space.dbFile, path: path.join(dirName(space.dbFile.path), partFileName) }\n // Use a short TTL for the PATCH method (which is also used for refreshing)\n const ttl = patch ? CACHE_LOCK_FILE_TTL : undefined\n const [created, fileLock] = await this.filesLockManager.createOrRefresh(user, dbFile, SERVER_NAME, DEPTH.RESOURCE, ttl)\n // Do\n try {\n await writeFromStream(dstFile, part.file)\n } finally {\n if (!patch && created) {\n // Remove the file lock only if it has not been refreshed\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n if (patch) {\n // Only one resource can be updated with the PATCH method.\n break\n }\n }\n }\n\n async touch(user: UserModel, space: SpaceEnv, mtime: number, checkLocks = true): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // todo: update mtime in last files ( & in db file ?)\n await touchFile(space.realPath, mtime)\n }\n\n async mkFile(user: UserModel, space: SpaceEnv, overwrite = false, checkLocks = true, checkDocument = false): Promise<void> {\n checkFileName(space.realPath)\n if (!overwrite && (await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // use sample documents when possible\n const fileExtension = path.extname(space.realPath)\n if (checkDocument && fileExtension !== '.txt' && Object.values(DOCUMENT_TYPE).indexOf(fileExtension) > -1) {\n const srcSample = path.join(__dirname, `${SAMPLE_PATH_WITHOUT_EXT}${fileExtension}`)\n return copyFileContent(srcSample, space.realPath)\n } else {\n return createEmptyFile(space.realPath)\n }\n }\n\n async mkDir(user: UserModel, space: SpaceEnv, recursive = false, dav?: { depth: LOCK_DEPTH; lockTokens: string[] }): Promise<void> {\n checkFileName(space.realPath)\n if (!recursive) {\n if (await isPathExists(space.realPath)) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n await this.filesLockManager.checkConflicts(space.dbFile, dav?.depth || DEPTH.RESOURCE, { userId: user.id, lockTokens: dav?.lockTokens })\n await makeDir(space.realPath, recursive)\n }\n\n async copyMove(\n user: UserModel,\n srcSpace: SpaceEnv,\n dstSpace: SpaceEnv,\n isMove: boolean,\n overwrite = false,\n mkdirDstParentPath = false,\n dav?: { depth: LOCK_DEPTH; lockTokens: string[] }\n ): Promise<void> {\n // checks\n if (!canAccessToSpace(user, dstSpace)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to access to this space repository : ${dstSpace.repository}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to access to this repository')\n }\n if (!haveSpaceEnvPermissions(dstSpace, SPACE_OPERATION.ADD)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to copy/move on this space : *${dstSpace.alias}* (${dstSpace.id}) : ${dstSpace.url}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to copy/move on the destination')\n }\n if (dstSpace.quotaIsExceeded) {\n this.logger.warn(`${this.copyMove.name} - quota is exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Quota is exceeded')\n }\n if (!(await isPathExists(srcSpace.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (!(await isPathExists(dirName(dstSpace.realPath)))) {\n if (mkdirDstParentPath) {\n try {\n await makeDir(dirName(dstSpace.realPath), true)\n } catch (e) {\n this.logger.error(`${this.copyMove.name} - Cannot create parent directory for destination ${dstSpace.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Cannot create parent directory for destination')\n }\n } else {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n if (srcSpace.realPath === dstSpace.realPath) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source onto itself')\n }\n if (`${dstSpace.realPath}/`.startsWith(`${srcSpace.realPath}/`)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source below itself')\n }\n if (dirName(srcSpace.url) === dirName(dstSpace.url) && dirName(srcSpace.realPath) !== dirName(dstSpace.realPath)) {\n /* Handle renaming a space file with the same name as a space root :\n srcSpace.url = '/space/sync-in/code2.ts' (a space file)\n srcSpace.realPath = '/home/sync-in/spaces/sync-in/code2.ts\n dstSpace.url = '/space/sync-in/code.ts' (a space root)\n dstSpace.realPath = '/home/sync-in/users/jo/files/code2.ts !!\n */\n throw new FileError(HttpStatus.BAD_REQUEST, 'An anchored file already has this name')\n }\n if (!overwrite && (await isPathExists(dstSpace.realPath))) {\n /* Handle case-sensitive (in renaming context):\n srcSpace.url = '/space/sync-in/code.ts'\n dstSpace.url = '/space/sync-in/code.TS'\n The destination exists because it's the same file, bypass this\n */\n if (!(isMove && srcSpace.realPath.toLowerCase() === dstSpace.realPath.toLowerCase())) {\n throw new FileError(dav ? HttpStatus.PRECONDITION_FAILED : HttpStatus.BAD_REQUEST, 'The destination already exists')\n }\n }\n\n const isDir = await isPathIsDir(srcSpace.realPath)\n\n if (dstSpace.storageQuota) {\n /* Skip validation when moving to the same space; for copy operations, run all checks. */\n if (!isMove || (isMove && srcSpace.id !== dstSpace.id)) {\n const size = isDir ? (await dirSize(srcSpace.realPath))[0] : await fileSize(srcSpace.realPath)\n if (dstSpace.willExceedQuota(size)) {\n this.logger.warn(`${this.copyMove.name} - storage quota will be exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Storage quota will be exceeded')\n }\n }\n }\n\n // check lock conflicts on source and destination\n let recursive: boolean\n let depth: LOCK_DEPTH\n if (dav?.depth) {\n recursive = dav.depth === DEPTH.INFINITY\n depth = dav.depth\n } else {\n recursive = isDir\n depth = recursive ? DEPTH.INFINITY : DEPTH.RESOURCE\n }\n if (isMove) {\n // check source\n await this.filesLockManager.checkConflicts(srcSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n }\n // check destination\n await this.filesLockManager.checkConflicts(dstSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n\n // overwrite\n if (overwrite && (await isPathExists(dstSpace.realPath))) {\n // todo : versioning here\n await this.delete(user, dstSpace)\n }\n\n // send it to task watcher\n if (srcSpace.task?.cacheKey) {\n if (!isDir) srcSpace.task.props.totalSize = await fileSize(srcSpace.realPath)\n FileTaskEvent.emit('startWatch', srcSpace, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, dstSpace.realPath)\n }\n\n // do\n if (isMove) {\n await moveFiles(srcSpace.realPath, dstSpace.realPath, overwrite)\n return this.filesQueries.moveFiles(srcSpace.dbFile, dstSpace.dbFile, isDir)\n }\n return copyFiles(srcSpace.realPath, dstSpace.realPath, overwrite, recursive)\n }\n\n async delete(user: UserModel, space: SpaceEnv, dav?: { lockTokens: string[] }): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n // check lock conflicts\n const isDir = await isPathIsDir(space.realPath)\n await this.filesLockManager.checkConflicts(space.dbFile, isDir ? DEPTH.INFINITY : DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: dav?.lockTokens\n })\n // file system deletion\n let forceDeleteInDB = false\n if (space.inTrashRepository) {\n await removeFiles(space.realPath)\n } else {\n const baseTrashPath = realTrashPathFromSpace(user, space)\n if (baseTrashPath) {\n const name = fileName(space.realPath)\n const trashDir = path.join(baseTrashPath, dirName(space.dbFile.path))\n const trashFile = path.join(trashDir, name)\n if (!(await isPathExists(trashDir))) {\n await makeDir(trashDir, true)\n }\n if (await isPathExists(trashFile)) {\n // if a resource already exists in the trash, rename it with the date\n const dstTrash = await uniqueDatedFilePath(trashFile)\n // move the resource on fs\n await moveFiles(trashFile, dstTrash.path)\n // move the resource in db\n const trashFileDB: FileDBProps = { ...space.dbFile, inTrash: true }\n const dstTrashFileDB: FileDBProps = { ...trashFileDB, path: path.join(dirName(trashFileDB.path), fileName(dstTrash.path)) }\n await this.filesQueries.moveFiles(trashFileDB, dstTrashFileDB, dstTrash.isDir)\n }\n await moveFiles(space.realPath, trashFile, true)\n } else {\n // unsupported case: delete the file (this shouldn't happen)\n this.logger.error(`Unable to find trash path for space - *${space.alias}* (${space.id}) : delete permanently : ${space.realPath}`)\n forceDeleteInDB = true\n await removeFiles(space.realPath)\n }\n }\n // remove locks, these locks have already been checked in the `checkConflicts` function\n if (isDir) {\n this.filesLockManager.removeChildLocks(user, space.dbFile).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n for (const lock of await this.filesLockManager.getLocksByPath(space.dbFile)) {\n this.filesLockManager.removeLock(lock.key).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n // delete or move to trash the files in db\n return this.filesQueries.deleteFiles(space.dbFile, isDir, forceDeleteInDB)\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, url: string): Promise<void> {\n this.logger.log(`${this.downloadFromUrl.name} : ${url}`)\n // create lock\n const rPath = await uniqueFilePathFromDir(space.realPath)\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(space.realPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, SERVER_NAME, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task.cacheKey) {\n let headRes: AxiosResponse\n\n try {\n headRes = await this.http.axiosRef({ method: HTTP_METHOD.HEAD, url: url, maxRedirects: 1 })\n } catch (e) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n this.logger.error(`${this.downloadFromUrl.name} - ${url} : ${e}`)\n throw new FileError(HttpStatus.BAD_REQUEST, 'Unable to download file')\n }\n\n if (regExpPrivateIP.test(headRes.request.socket.remoteAddress)) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n // prevent SSRF attack\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n\n // attempt to retrieve the Content-Length header\n try {\n if ('content-length' in headRes.headers) {\n space.task.props.totalSize = parseInt(headRes.headers['content-length'], 10) || null\n }\n } catch (e) {\n this.logger.debug(`${this.downloadFromUrl.name} - content-length : ${e}`)\n }\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DOWNLOAD, rPath)\n }\n // do\n try {\n const getRes = await this.http.axiosRef({ method: HTTP_METHOD.GET, url: url, responseType: 'stream', maxRedirects: 1 })\n if (regExpPrivateIP.test(getRes.request.socket.remoteAddress)) {\n // Prevent SSRF attacks and perform a DNS-rebinding check if a HEAD request has already been made\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n await writeFromStream(rPath, getRes.data)\n } finally {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, dto: CompressFileDto): Promise<void> {\n // This method is currently used only by files-methods.service, which handles input sanitization.\n // If it is used in other services in the future, make sure to refactor accordingly to sanitize inputs properly.\n const srcPath = dirName(space.realPath)\n // todo: a guest link tasksPath should be in specific directory (guest link has no home)\n const archiveExt = dto.name.endsWith(dto.extension) ? '' : `.${dto.extension}`\n const dstPath = await uniqueFilePathFromDir(path.join(dto.compressInDirectory ? srcPath : user.tasksPath, `${dto.name}${archiveExt}`))\n const archive: Archiver = archiver('tar', {\n gzip: dto.extension === tarGzExtension,\n gzipOptions: {\n level: 9\n }\n })\n // create lock\n let fileLock: FileLock\n if (dto.compressInDirectory) {\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, lock] = await this.filesLockManager.create(user, dbFile, SERVER_NAME, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n if (space.task?.cacheKey) {\n space.task.props.compressInDirectory = dto.compressInDirectory\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.COMPRESS, dstPath)\n }\n // do\n try {\n archive.on('error', (error: ArchiverError) => {\n throw error\n })\n const dstStream = fs.createWriteStream(dstPath, { highWaterMark: DEFAULT_HIGH_WATER_MARK })\n archive.pipe(dstStream)\n for (const f of dto.files) {\n if (await isPathIsDir(f.path)) {\n archive.directory(f.path, dto.files.length > 1 ? fileName(f.path) : false)\n } else {\n archive.file(f.path, {\n name: f.rootAlias ? f.name : fileName(f.path)\n })\n }\n }\n await archive.finalize()\n } finally {\n if (fileLock) {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n // checks\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n const extension = path.extname(space.realPath)\n if (!COMPRESSION_EXTENSION.has(extension)) {\n throw new FileError(HttpStatus.BAD_REQUEST, `${extension} is not supported`)\n }\n // make destination folder\n const dstPath = await uniqueFilePathFromDir(path.join(dirName(space.realPath), path.basename(space.realPath, extension)))\n await makeDir(dstPath)\n // create lock\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, SERVER_NAME, DEPTH.INFINITY)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task?.cacheKey) FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DECOMPRESS, dstPath)\n // do\n try {\n if (extension === '.zip') {\n await extractZip(space.realPath, dstPath)\n } else {\n await extractTar({\n file: space.realPath,\n cwd: dstPath,\n gzip: COMPRESSION_EXTENSION.get(extension) === tarGzExtension,\n preserveOwner: false\n })\n }\n } finally {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async generateThumbnail(space: SpaceEnv, size: number): Promise<Readable> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (getMimeType(space.realPath, false).indexOf('image') === -1) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n try {\n return generateThumbnail(space.realPath, size)\n } catch (e) {\n this.logger.warn(e)\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n }\n\n async lock(user: UserModel, space: SpaceEnv): Promise<FileLockProps> {\n const rExists = await isPathExists(space.realPath)\n if (!rExists) {\n this.logger.warn('Lock refresh must specify an existing resource')\n throw new FileError(HttpStatus.BAD_REQUEST, 'Lock refresh must specify an existing resource')\n }\n const [_created, lock] = await this.filesLockManager.createOrRefresh(user, space.dbFile, SERVER_NAME, DEPTH.RESOURCE, CACHE_LOCK_FILE_TTL)\n return this.filesLockManager.convertLockToFileLockProps(lock)\n }\n\n async unlock(user: UserModel, space: SpaceEnv, forceAsFileOwner = false): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n this.logger.warn(`Unable to unlock: ${space.url} - resource does not exist`)\n throw new FileError(HttpStatus.BAD_REQUEST, 'Unlock must specify an existing resource')\n }\n const fileLocks = await this.filesLockManager.getLocksByPath(space.dbFile)\n if (fileLocks.length === 0) {\n this.logger.warn(`Unable to find lock: ${space.url} - resource does not exist`)\n return\n }\n for (const lock of fileLocks) {\n if ((forceAsFileOwner && space.dbFile?.ownerId === user.id) || lock.owner.id === user.id) {\n // Refresh if more than half of the TTL has passed\n await this.filesLockManager.removeLock(lock.key)\n } else {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n }\n }\n\n async unlockRequest(user: UserModel, space: SpaceEnv): Promise<void> {\n const fileLocks = await this.filesLockManager.getLocksByPath(space.dbFile)\n if (fileLocks.length === 0) {\n this.logger.warn(`Unable to find lock: ${space.url} - resource does not exist`)\n throw new FileError(HttpStatus.NOT_FOUND, 'Lock not found')\n }\n for (const lock of fileLocks) {\n if (lock.owner.id !== user.id) {\n const notification: NotificationContent = {\n app: NOTIFICATION_APP.UNLOCK_REQUEST,\n event: NOTIFICATION_APP_EVENT.UNLOCK_REQUEST,\n element: fileName(space.url),\n url: dirName(space.url)\n }\n this.notificationsManager\n .create([lock.owner.id], notification, {\n author: user,\n currentUrl: this.contextManager.headerOriginUrl()\n })\n .catch((e: Error) => this.logger.error(`${this.unlockRequest.name} - ${e}`))\n }\n }\n }\n\n async getSize(space: SpaceEnv): Promise<number> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (await isPathIsDir(space.realPath)) {\n return (await dirSize(space.realPath))[0]\n } else {\n return await fileSize(space.realPath)\n }\n }\n}\n"],"names":["FilesManager","sendFileFromSpace","space","downloadName","SendFile","realPath","saveStream","user","req","options","fExists","isPathExists","fTmpExists","tmpPath","method","HTTP_METHOD","POST","FileError","HttpStatus","METHOD_NOT_ALLOWED","isPathIsDir","makeDir","dirName","CONFLICT","fileLock","dav","filesLockManager","checkConflicts","dbFile","depth","DEPTH","RESOURCE","userId","id","lockTokens","ok","lock","create","SERVER_NAME","LockConflict","startRange","headers","match","exec","length","BAD_REQUEST","parseInt","size","fileSize","checksum","checksumAlg","writeFromStreamAndChecksum","raw","writeFromStream","path","dirname","moveFiles","e","removeFiles","logger","error","name","INTERNAL_SERVER_ERROR","removeLock","key","warn","saveMultipart","overwrite","PUT","patch","PATCH","realParentPath","basePath","sep","part","files","partFileName","fileName","filename","dstFile","resolve","startsWith","FORBIDDEN","dstDir","dstUrl","join","url","dstSpace","spacesManager","spaceEnv","split","delete","ttl","CACHE_LOCK_FILE_TTL","undefined","created","createOrRefresh","file","touch","mtime","checkLocks","NOT_FOUND","touchFile","mkFile","checkDocument","checkFileName","fileExtension","extname","Object","values","DOCUMENT_TYPE","indexOf","srcSample","__dirname","SAMPLE_PATH_WITHOUT_EXT","copyFileContent","createEmptyFile","mkDir","recursive","copyMove","srcSpace","isMove","mkdirDstParentPath","canAccessToSpace","repository","haveSpaceEnvPermissions","SPACE_OPERATION","ADD","alias","quotaIsExceeded","INSUFFICIENT_STORAGE","toLowerCase","PRECONDITION_FAILED","isDir","storageQuota","dirSize","willExceedQuota","INFINITY","task","cacheKey","props","totalSize","FileTaskEvent","emit","FILE_OPERATION","MOVE","COPY","filesQueries","copyFiles","forceDeleteInDB","inTrashRepository","baseTrashPath","realTrashPathFromSpace","trashDir","trashFile","dstTrash","uniqueDatedFilePath","trashFileDB","inTrash","dstTrashFileDB","removeChildLocks","catch","getLocksByPath","deleteFiles","downloadFromUrl","log","rPath","uniqueFilePathFromDir","headRes","http","axiosRef","HEAD","maxRedirects","regExpPrivateIP","test","request","socket","remoteAddress","debug","DOWNLOAD","getRes","GET","responseType","data","compress","dto","srcPath","archiveExt","endsWith","extension","dstPath","compressInDirectory","tasksPath","archive","archiver","gzip","tarGzExtension","gzipOptions","level","COMPRESS","on","dstStream","fs","createWriteStream","highWaterMark","DEFAULT_HIGH_WATER_MARK","pipe","f","directory","rootAlias","finalize","decompress","COMPRESSION_EXTENSION","has","basename","DECOMPRESS","extractZip","extractTar","cwd","get","preserveOwner","generateThumbnail","getMimeType","rExists","_created","convertLockToFileLockProps","unlock","forceAsFileOwner","fileLocks","ownerId","owner","unlockRequest","notification","app","NOTIFICATION_APP","UNLOCK_REQUEST","event","NOTIFICATION_APP_EVENT","element","notificationsManager","author","currentUrl","contextManager","headerOriginUrl","getSize","Logger"],"mappings":"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"}
@@ -141,9 +141,9 @@ let FilesMethods = class FilesMethods {
141
141
  this.handleError(space, _operations.FILE_OPERATION.LOCK, e);
142
142
  }
143
143
  }
144
- async unlock(user, space, forceAsOwner = false) {
144
+ async unlock(user, space, forceAsFileOwner = false) {
145
145
  try {
146
- return await this.filesManager.unlock(user, space, forceAsOwner);
146
+ return await this.filesManager.unlock(user, space, forceAsFileOwner);
147
147
  } catch (e) {
148
148
  this.handleError(space, _operations.FILE_OPERATION.UNLOCK, e);
149
149
  }
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-methods.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpException, HttpStatus, Injectable, Logger, StreamableFile } from '@nestjs/common'\nimport { FastifyReply } from 'fastify'\nimport path from 'node:path'\nimport { Readable } from 'node:stream'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { CompressFileDto, CopyMoveFileDto, DownloadFileDto, MakeFileDto } from '../dto/file-operations.dto'\nimport { FileLockProps } from '../interfaces/file-props.interface'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport { checkFileName, dirName, fileName, isPathExists, sanitizeName } from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { FilesManager } from './files-manager.service'\n\n@Injectable()\nexport class FilesMethods {\n private readonly logger = new Logger(FilesMethods.name)\n\n constructor(\n private readonly spacesManager: SpacesManager,\n private readonly filesManager: FilesManager\n ) {}\n\n async headOrGet(req: FastifySpaceRequest, res: FastifyReply): Promise<StreamableFile> {\n const sendFile: SendFile = this.filesManager.sendFileFromSpace(req.space)\n try {\n await sendFile.checks()\n return await sendFile.stream(req, res)\n } catch (e) {\n this.handleError(req.space, req.method, e)\n }\n }\n\n async upload(req: FastifySpaceRequest): Promise<void> {\n try {\n await this.filesManager.saveMultipart(req.user, req.space, req)\n } catch (e) {\n this.handleError(req.space, FILE_OPERATION.UPLOAD, e)\n }\n }\n\n async make(user: UserModel, space: SpaceEnv, makeFileDto: MakeFileDto): Promise<void> {\n try {\n if (makeFileDto.type === 'directory') {\n return await this.filesManager.mkDir(user, space)\n } else {\n return await this.filesManager.mkFile(user, space, false, true, true)\n }\n } catch (e) {\n this.handleError(space, `${FILE_OPERATION.MAKE} ${makeFileDto.type}`, e)\n }\n }\n\n copy(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, false)\n }\n\n move(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, true)\n }\n\n async delete(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.delete(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DELETE, e)\n }\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, downloadDto: DownloadFileDto): Promise<void> {\n checkFileName(space.realPath)\n try {\n return await this.filesManager.downloadFromUrl(user, space, downloadDto.url)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DOWNLOAD, e)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, compressFileDto: CompressFileDto): Promise<void> {\n compressFileDto.name = checkFileName(space.realPath)\n try {\n for (const f of compressFileDto.files) {\n // sanitize file name\n f.name = sanitizeName(f.name)\n // handles the case where the file is an anchored file\n let baseSpace: SpaceEnv\n if (f.path) {\n baseSpace = await this.spacesManager.spaceEnv(user, f.path.split('/'))\n f.path = baseSpace.realPath\n } else {\n if (f.rootAlias) {\n baseSpace = await this.spacesManager.spaceEnv(user, path.join(dirName(space.url), f.rootAlias).split('/'))\n f.path = baseSpace.realPath\n } else {\n baseSpace = space\n f.path = path.resolve(dirName(space.realPath), f.name)\n }\n }\n // prevent path traversal\n if (!f.path.startsWith(baseSpace.realBasePath)) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.FORBIDDEN, `${f.name} not allowed`))\n }\n if (!(await isPathExists(f.path))) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.NOT_FOUND, `${f.name} does not exist`))\n }\n }\n return await this.filesManager.compress(user, space, compressFileDto)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.COMPRESS, e)\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.decompress(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DECOMPRESS, e)\n }\n }\n\n async genThumbnail(space: SpaceEnv, size: number): Promise<Readable> {\n try {\n return await this.filesManager.generateThumbnail(space, size)\n } catch (e) {\n this.handleError(space, this.genThumbnail.name, e)\n }\n }\n\n async lock(user: UserModel, space: SpaceEnv): Promise<FileLockProps> {\n try {\n return await this.filesManager.lock(user, space)\n } catch (e) {\n if (e instanceof LockConflict) {\n const fileLockProps = this.filesManager.filesLockManager.convertLockToFileLockProps(e.lock)\n throw new HttpException(fileLockProps, HttpStatus.LOCKED)\n }\n this.handleError(space, FILE_OPERATION.LOCK, e)\n }\n }\n\n async unlock(user: UserModel, space: SpaceEnv, forceAsOwner = false): Promise<void> {\n try {\n return await this.filesManager.unlock(user, space, forceAsOwner)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.UNLOCK, e)\n }\n }\n\n async unlockRequest(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.unlockRequest(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.UNLOCK_REQUEST, e)\n }\n }\n\n async getSize(space: SpaceEnv): Promise<{ size: number }> {\n try {\n return { size: await this.filesManager.getSize(space) }\n } catch (e) {\n this.handleError(space, FILE_OPERATION.GET_SIZE, e)\n }\n }\n\n private async copyMove(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto,\n isMove: boolean\n ): Promise<{\n path: string\n name: string\n }> {\n const dstUrl = path.join(copyMoveFileDto.dstDirectory, copyMoveFileDto.dstName ? copyMoveFileDto.dstName : fileName(space.realPath))\n let dstSpace: SpaceEnv\n try {\n dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.filesManager.copyMove(user, space, dstSpace, isMove, copyMoveFileDto.overwrite)\n } catch (e) {\n this.handleError(space, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, e, dstSpace)\n }\n return { path: dirName(dstUrl), name: fileName(dstUrl) }\n }\n\n private handleError(space: SpaceEnv, action: string, e: any, dstSpace?: SpaceEnv) {\n this.logger.error(`unable to ${action} ${space.url}${dstSpace?.url ? ` -> ${dstSpace.url}` : ''} : ${e}`)\n // Remove the last part to avoid exposing the path\n const errorMsg = e.message.split(',')[0]\n if (e instanceof LockConflict) {\n throw new HttpException('The file is locked', HttpStatus.LOCKED)\n } else if (e instanceof FileError) {\n throw new HttpException(errorMsg, e.httpCode)\n }\n throw new HttpException(errorMsg, HttpStatus.INTERNAL_SERVER_ERROR)\n }\n}\n"],"names":["FilesMethods","headOrGet","req","res","sendFile","filesManager","sendFileFromSpace","space","checks","stream","e","handleError","method","upload","saveMultipart","user","FILE_OPERATION","UPLOAD","make","makeFileDto","type","mkDir","mkFile","MAKE","copy","copyMoveFileDto","copyMove","move","delete","DELETE","downloadFromUrl","downloadDto","checkFileName","realPath","url","DOWNLOAD","compress","compressFileDto","name","f","files","sanitizeName","baseSpace","path","spacesManager","spaceEnv","split","rootAlias","join","dirName","resolve","startsWith","realBasePath","COMPRESS","FileError","HttpStatus","FORBIDDEN","isPathExists","NOT_FOUND","decompress","DECOMPRESS","genThumbnail","size","generateThumbnail","lock","LockConflict","fileLockProps","filesLockManager","convertLockToFileLockProps","HttpException","LOCKED","LOCK","unlock","forceAsOwner","UNLOCK","unlockRequest","UNLOCK_REQUEST","getSize","GET_SIZE","isMove","dstUrl","dstDirectory","dstName","fileName","dstSpace","overwrite","MOVE","COPY","action","logger","error","errorMsg","message","httpCode","INTERNAL_SERVER_ERROR","Logger"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-methods.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpException, HttpStatus, Injectable, Logger, StreamableFile } from '@nestjs/common'\nimport { FastifyReply } from 'fastify'\nimport path from 'node:path'\nimport { Readable } from 'node:stream'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { CompressFileDto, CopyMoveFileDto, DownloadFileDto, MakeFileDto } from '../dto/file-operations.dto'\nimport { FileLockProps } from '../interfaces/file-props.interface'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport { checkFileName, dirName, fileName, isPathExists, sanitizeName } from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { FilesManager } from './files-manager.service'\n\n@Injectable()\nexport class FilesMethods {\n private readonly logger = new Logger(FilesMethods.name)\n\n constructor(\n private readonly spacesManager: SpacesManager,\n private readonly filesManager: FilesManager\n ) {}\n\n async headOrGet(req: FastifySpaceRequest, res: FastifyReply): Promise<StreamableFile> {\n const sendFile: SendFile = this.filesManager.sendFileFromSpace(req.space)\n try {\n await sendFile.checks()\n return await sendFile.stream(req, res)\n } catch (e) {\n this.handleError(req.space, req.method, e)\n }\n }\n\n async upload(req: FastifySpaceRequest): Promise<void> {\n try {\n await this.filesManager.saveMultipart(req.user, req.space, req)\n } catch (e) {\n this.handleError(req.space, FILE_OPERATION.UPLOAD, e)\n }\n }\n\n async make(user: UserModel, space: SpaceEnv, makeFileDto: MakeFileDto): Promise<void> {\n try {\n if (makeFileDto.type === 'directory') {\n return await this.filesManager.mkDir(user, space)\n } else {\n return await this.filesManager.mkFile(user, space, false, true, true)\n }\n } catch (e) {\n this.handleError(space, `${FILE_OPERATION.MAKE} ${makeFileDto.type}`, e)\n }\n }\n\n copy(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, false)\n }\n\n move(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, true)\n }\n\n async delete(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.delete(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DELETE, e)\n }\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, downloadDto: DownloadFileDto): Promise<void> {\n checkFileName(space.realPath)\n try {\n return await this.filesManager.downloadFromUrl(user, space, downloadDto.url)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DOWNLOAD, e)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, compressFileDto: CompressFileDto): Promise<void> {\n compressFileDto.name = checkFileName(space.realPath)\n try {\n for (const f of compressFileDto.files) {\n // sanitize file name\n f.name = sanitizeName(f.name)\n // handles the case where the file is an anchored file\n let baseSpace: SpaceEnv\n if (f.path) {\n baseSpace = await this.spacesManager.spaceEnv(user, f.path.split('/'))\n f.path = baseSpace.realPath\n } else {\n if (f.rootAlias) {\n baseSpace = await this.spacesManager.spaceEnv(user, path.join(dirName(space.url), f.rootAlias).split('/'))\n f.path = baseSpace.realPath\n } else {\n baseSpace = space\n f.path = path.resolve(dirName(space.realPath), f.name)\n }\n }\n // prevent path traversal\n if (!f.path.startsWith(baseSpace.realBasePath)) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.FORBIDDEN, `${f.name} not allowed`))\n }\n if (!(await isPathExists(f.path))) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.NOT_FOUND, `${f.name} does not exist`))\n }\n }\n return await this.filesManager.compress(user, space, compressFileDto)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.COMPRESS, e)\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.decompress(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DECOMPRESS, e)\n }\n }\n\n async genThumbnail(space: SpaceEnv, size: number): Promise<Readable> {\n try {\n return await this.filesManager.generateThumbnail(space, size)\n } catch (e) {\n this.handleError(space, this.genThumbnail.name, e)\n }\n }\n\n async lock(user: UserModel, space: SpaceEnv): Promise<FileLockProps> {\n try {\n return await this.filesManager.lock(user, space)\n } catch (e) {\n if (e instanceof LockConflict) {\n const fileLockProps = this.filesManager.filesLockManager.convertLockToFileLockProps(e.lock)\n throw new HttpException(fileLockProps, HttpStatus.LOCKED)\n }\n this.handleError(space, FILE_OPERATION.LOCK, e)\n }\n }\n\n async unlock(user: UserModel, space: SpaceEnv, forceAsFileOwner = false): Promise<void> {\n try {\n return await this.filesManager.unlock(user, space, forceAsFileOwner)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.UNLOCK, e)\n }\n }\n\n async unlockRequest(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.unlockRequest(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.UNLOCK_REQUEST, e)\n }\n }\n\n async getSize(space: SpaceEnv): Promise<{ size: number }> {\n try {\n return { size: await this.filesManager.getSize(space) }\n } catch (e) {\n this.handleError(space, FILE_OPERATION.GET_SIZE, e)\n }\n }\n\n private async copyMove(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto,\n isMove: boolean\n ): Promise<{\n path: string\n name: string\n }> {\n const dstUrl = path.join(copyMoveFileDto.dstDirectory, copyMoveFileDto.dstName ? copyMoveFileDto.dstName : fileName(space.realPath))\n let dstSpace: SpaceEnv\n try {\n dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.filesManager.copyMove(user, space, dstSpace, isMove, copyMoveFileDto.overwrite)\n } catch (e) {\n this.handleError(space, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, e, dstSpace)\n }\n return { path: dirName(dstUrl), name: fileName(dstUrl) }\n }\n\n private handleError(space: SpaceEnv, action: string, e: any, dstSpace?: SpaceEnv) {\n this.logger.error(`unable to ${action} ${space.url}${dstSpace?.url ? ` -> ${dstSpace.url}` : ''} : ${e}`)\n // Remove the last part to avoid exposing the path\n const errorMsg = e.message.split(',')[0]\n if (e instanceof LockConflict) {\n throw new HttpException('The file is locked', HttpStatus.LOCKED)\n } else if (e instanceof FileError) {\n throw new HttpException(errorMsg, e.httpCode)\n }\n throw new HttpException(errorMsg, HttpStatus.INTERNAL_SERVER_ERROR)\n }\n}\n"],"names":["FilesMethods","headOrGet","req","res","sendFile","filesManager","sendFileFromSpace","space","checks","stream","e","handleError","method","upload","saveMultipart","user","FILE_OPERATION","UPLOAD","make","makeFileDto","type","mkDir","mkFile","MAKE","copy","copyMoveFileDto","copyMove","move","delete","DELETE","downloadFromUrl","downloadDto","checkFileName","realPath","url","DOWNLOAD","compress","compressFileDto","name","f","files","sanitizeName","baseSpace","path","spacesManager","spaceEnv","split","rootAlias","join","dirName","resolve","startsWith","realBasePath","COMPRESS","FileError","HttpStatus","FORBIDDEN","isPathExists","NOT_FOUND","decompress","DECOMPRESS","genThumbnail","size","generateThumbnail","lock","LockConflict","fileLockProps","filesLockManager","convertLockToFileLockProps","HttpException","LOCKED","LOCK","unlock","forceAsFileOwner","UNLOCK","unlockRequest","UNLOCK_REQUEST","getSize","GET_SIZE","isMove","dstUrl","dstDirectory","dstName","fileName","dstSpace","overwrite","MOVE","COPY","action","logger","error","errorMsg","message","httpCode","INTERNAL_SERVER_ERROR","Logger"],"mappings":"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"}
@@ -85,7 +85,9 @@ describe(_filesmethodsservice.FilesMethods.name, ()=>{
85
85
  },
86
86
  {
87
87
  provide: _filesmanagerservice.FilesManager,
88
- useValue: {}
88
+ useValue: {
89
+ compress: jest.fn()
90
+ }
89
91
  }
90
92
  ]
91
93
  }).compile();
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-methods.service.spec.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Test, TestingModule } from '@nestjs/testing'\nimport path from 'node:path'\nimport { transformAndValidate } from '../../../common/functions'\nimport { Cache } from '../../../infrastructure/cache/services/cache.service'\nimport { ContextManager } from '../../../infrastructure/context/services/context-manager.service'\nimport { DB_TOKEN_PROVIDER } from '../../../infrastructure/database/constants'\nimport { NotificationsManager } from '../../notifications/services/notifications-manager.service'\nimport { SharesManager } from '../../shares/services/shares-manager.service'\nimport { SPACE_REPOSITORY } from '../../spaces/constants/spaces'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpaceModel } from '../../spaces/models/space.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { SpacesQueries } from '../../spaces/services/spaces-queries.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { UsersQueries } from '../../users/services/users-queries.service'\nimport { generateUserTest } from '../../users/utils/test'\nimport { tarExtension } from '../constants/compress'\nimport { CompressFileDto, CopyMoveFileDto } from '../dto/file-operations.dto'\nimport { FilesManager } from './files-manager.service'\nimport { FilesMethods } from './files-methods.service'\n\ndescribe(FilesMethods.name, () => {\n let filesMethods: FilesMethods\n let spacesManager: SpacesManager\n let userTest: UserModel\n const spaceEnv = {\n id: 1,\n alias: 'project',\n name: 'project',\n enabled: true,\n permissions: 'a:d:m:so',\n role: 0,\n realBasePath: SpaceModel.getFilesPath('project'),\n realPath: path.join(SpaceModel.getFilesPath('project'), 'foo'),\n url: `${SPACE_REPOSITORY.FILES}/project/foo`\n } as SpaceEnv\n\n beforeAll(async () => {\n const module: TestingModule = await Test.createTestingModule({\n imports: [],\n providers: [\n FilesMethods,\n SpacesManager,\n { provide: DB_TOKEN_PROVIDER, useValue: {} },\n {\n provide: Cache,\n useValue: { get: () => null }\n },\n { provide: ContextManager, useValue: {} },\n {\n provide: NotificationsManager,\n useValue: {}\n },\n { provide: UsersQueries, useValue: {} },\n { provide: SharesManager, useValue: {} },\n {\n provide: SpacesQueries,\n useValue: {\n permissions: () => spaceEnv\n }\n },\n { provide: FilesManager, useValue: {} }\n ]\n }).compile()\n\n module.useLogger(['fatal'])\n filesMethods = module.get<FilesMethods>(FilesMethods)\n spacesManager = module.get<SpacesManager>(SpacesManager)\n userTest = new UserModel(generateUserTest())\n // mock\n spacesManager.updateSpacesQuota = jest.fn().mockReturnValue(undefined)\n })\n\n it('should be defined', () => {\n expect(filesMethods).toBeDefined()\n expect(spacesManager).toBeDefined()\n expect(userTest).toBeDefined()\n })\n\n it('should avoid path traversal on CopyMove action', async () => {\n const copyMoveFileDto: CopyMoveFileDto = { dstDirectory: '../../../foo', dstName: '../bar/../' }\n expect(() => transformAndValidate(CopyMoveFileDto, copyMoveFileDto)).toThrow()\n await expect((filesMethods as any).copyMove(userTest, spaceEnv, copyMoveFileDto, false)).rejects.toThrow(/is not valid/i)\n })\n\n it('should avoid path traversal on Compress action', async () => {\n const compressFileDto: CompressFileDto = {\n name: '../../archive',\n compressInDirectory: false,\n files: [{ name: '../../foo', rootAlias: undefined }],\n extension: tarExtension\n }\n expect(() => transformAndValidate(CompressFileDto, compressFileDto)).toThrow()\n await expect(filesMethods.compress(userTest, spaceEnv, compressFileDto)).rejects.toThrow(/does not exist/i)\n compressFileDto.files[0].path = '../../../bar/../'\n await expect(filesMethods.compress(userTest, spaceEnv, compressFileDto)).rejects.toThrow(/is not valid/i)\n })\n})\n"],"names":["describe","FilesMethods","name","filesMethods","spacesManager","userTest","spaceEnv","id","alias","enabled","permissions","role","realBasePath","SpaceModel","getFilesPath","realPath","path","join","url","SPACE_REPOSITORY","FILES","beforeAll","module","Test","createTestingModule","imports","providers","SpacesManager","provide","DB_TOKEN_PROVIDER","useValue","Cache","get","ContextManager","NotificationsManager","UsersQueries","SharesManager","SpacesQueries","FilesManager","compile","useLogger","UserModel","generateUserTest","updateSpacesQuota","jest","fn","mockReturnValue","undefined","it","expect","toBeDefined","copyMoveFileDto","dstDirectory","dstName","transformAndValidate","CopyMoveFileDto","toThrow","copyMove","rejects","compressFileDto","compressInDirectory","files","rootAlias","extension","tarExtension","CompressFileDto","compress"],"mappings":"AAAA;;;;CAIC;;;;yBAEmC;iEACnB;2BACoB;8BACf;uCACS;2BACG;6CACG;sCACP;wBACG;4BAEN;sCACG;sCACA;2BACJ;qCACG;sBACI;0BACJ;mCACoB;qCACpB;qCACA;;;;;;AAE7BA,SAASC,iCAAY,CAACC,IAAI,EAAE;IAC1B,IAAIC;IACJ,IAAIC;IACJ,IAAIC;IACJ,MAAMC,WAAW;QACfC,IAAI;QACJC,OAAO;QACPN,MAAM;QACNO,SAAS;QACTC,aAAa;QACbC,MAAM;QACNC,cAAcC,sBAAU,CAACC,YAAY,CAAC;QACtCC,UAAUC,iBAAI,CAACC,IAAI,CAACJ,sBAAU,CAACC,YAAY,CAAC,YAAY;QACxDI,KAAK,GAAGC,wBAAgB,CAACC,KAAK,CAAC,YAAY,CAAC;IAC9C;IAEAC,UAAU;QACR,MAAMC,SAAwB,MAAMC,aAAI,CAACC,mBAAmB,CAAC;YAC3DC,SAAS,EAAE;YACXC,WAAW;gBACTzB,iCAAY;gBACZ0B,mCAAa;gBACb;oBAAEC,SAASC,4BAAiB;oBAAEC,UAAU,CAAC;gBAAE;gBAC3C;oBACEF,SAASG,mBAAK;oBACdD,UAAU;wBAAEE,KAAK,IAAM;oBAAK;gBAC9B;gBACA;oBAAEJ,SAASK,qCAAc;oBAAEH,UAAU,CAAC;gBAAE;gBACxC;oBACEF,SAASM,iDAAoB;oBAC7BJ,UAAU,CAAC;gBACb;gBACA;oBAAEF,SAASO,iCAAY;oBAAEL,UAAU,CAAC;gBAAE;gBACtC;oBAAEF,SAASQ,mCAAa;oBAAEN,UAAU,CAAC;gBAAE;gBACvC;oBACEF,SAASS,mCAAa;oBACtBP,UAAU;wBACRpB,aAAa,IAAMJ;oBACrB;gBACF;gBACA;oBAAEsB,SAASU,iCAAY;oBAAER,UAAU,CAAC;gBAAE;aACvC;QACH,GAAGS,OAAO;QAEVjB,OAAOkB,SAAS,CAAC;YAAC;SAAQ;QAC1BrC,eAAemB,OAAOU,GAAG,CAAe/B,iCAAY;QACpDG,gBAAgBkB,OAAOU,GAAG,CAAgBL,mCAAa;QACvDtB,WAAW,IAAIoC,oBAAS,CAACC,IAAAA,sBAAgB;QACzC,OAAO;QACPtC,cAAcuC,iBAAiB,GAAGC,KAAKC,EAAE,GAAGC,eAAe,CAACC;IAC9D;IAEAC,GAAG,qBAAqB;QACtBC,OAAO9C,cAAc+C,WAAW;QAChCD,OAAO7C,eAAe8C,WAAW;QACjCD,OAAO5C,UAAU6C,WAAW;IAC9B;IAEAF,GAAG,kDAAkD;QACnD,MAAMG,kBAAmC;YAAEC,cAAc;YAAgBC,SAAS;QAAa;QAC/FJ,OAAO,IAAMK,IAAAA,+BAAoB,EAACC,kCAAe,EAAEJ,kBAAkBK,OAAO;QAC5E,MAAMP,OAAO,AAAC9C,aAAqBsD,QAAQ,CAACpD,UAAUC,UAAU6C,iBAAiB,QAAQO,OAAO,CAACF,OAAO,CAAC;IAC3G;IAEAR,GAAG,kDAAkD;QACnD,MAAMW,kBAAmC;YACvCzD,MAAM;YACN0D,qBAAqB;YACrBC,OAAO;gBAAC;oBAAE3D,MAAM;oBAAa4D,WAAWf;gBAAU;aAAE;YACpDgB,WAAWC,sBAAY;QACzB;QACAf,OAAO,IAAMK,IAAAA,+BAAoB,EAACW,kCAAe,EAAEN,kBAAkBH,OAAO;QAC5E,MAAMP,OAAO9C,aAAa+D,QAAQ,CAAC7D,UAAUC,UAAUqD,kBAAkBD,OAAO,CAACF,OAAO,CAAC;QACzFG,gBAAgBE,KAAK,CAAC,EAAE,CAAC7C,IAAI,GAAG;QAChC,MAAMiC,OAAO9C,aAAa+D,QAAQ,CAAC7D,UAAUC,UAAUqD,kBAAkBD,OAAO,CAACF,OAAO,CAAC;IAC3F;AACF"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-methods.service.spec.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Test, TestingModule } from '@nestjs/testing'\nimport path from 'node:path'\nimport { transformAndValidate } from '../../../common/functions'\nimport { Cache } from '../../../infrastructure/cache/services/cache.service'\nimport { ContextManager } from '../../../infrastructure/context/services/context-manager.service'\nimport { DB_TOKEN_PROVIDER } from '../../../infrastructure/database/constants'\nimport { NotificationsManager } from '../../notifications/services/notifications-manager.service'\nimport { SharesManager } from '../../shares/services/shares-manager.service'\nimport { SPACE_REPOSITORY } from '../../spaces/constants/spaces'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpaceModel } from '../../spaces/models/space.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { SpacesQueries } from '../../spaces/services/spaces-queries.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { UsersQueries } from '../../users/services/users-queries.service'\nimport { generateUserTest } from '../../users/utils/test'\nimport { tarExtension } from '../constants/compress'\nimport { CompressFileDto, CopyMoveFileDto } from '../dto/file-operations.dto'\nimport { FilesManager } from './files-manager.service'\nimport { FilesMethods } from './files-methods.service'\n\ndescribe(FilesMethods.name, () => {\n let filesMethods: FilesMethods\n let spacesManager: SpacesManager\n let userTest: UserModel\n const spaceEnv = {\n id: 1,\n alias: 'project',\n name: 'project',\n enabled: true,\n permissions: 'a:d:m:so',\n role: 0,\n realBasePath: SpaceModel.getFilesPath('project'),\n realPath: path.join(SpaceModel.getFilesPath('project'), 'foo'),\n url: `${SPACE_REPOSITORY.FILES}/project/foo`\n } as SpaceEnv\n\n beforeAll(async () => {\n const module: TestingModule = await Test.createTestingModule({\n imports: [],\n providers: [\n FilesMethods,\n SpacesManager,\n { provide: DB_TOKEN_PROVIDER, useValue: {} },\n {\n provide: Cache,\n useValue: { get: () => null }\n },\n { provide: ContextManager, useValue: {} },\n {\n provide: NotificationsManager,\n useValue: {}\n },\n { provide: UsersQueries, useValue: {} },\n { provide: SharesManager, useValue: {} },\n {\n provide: SpacesQueries,\n useValue: {\n permissions: () => spaceEnv\n }\n },\n {\n provide: FilesManager,\n useValue: {\n compress: jest.fn()\n }\n }\n ]\n }).compile()\n\n module.useLogger(['fatal'])\n filesMethods = module.get<FilesMethods>(FilesMethods)\n spacesManager = module.get<SpacesManager>(SpacesManager)\n userTest = new UserModel(generateUserTest())\n // mock\n spacesManager.updateSpacesQuota = jest.fn().mockReturnValue(undefined)\n })\n\n it('should be defined', () => {\n expect(filesMethods).toBeDefined()\n expect(spacesManager).toBeDefined()\n expect(userTest).toBeDefined()\n })\n\n it('should avoid path traversal on CopyMove action', async () => {\n const copyMoveFileDto: CopyMoveFileDto = { dstDirectory: '../../../foo', dstName: '../bar/../' }\n expect(() => transformAndValidate(CopyMoveFileDto, copyMoveFileDto)).toThrow()\n await expect((filesMethods as any).copyMove(userTest, spaceEnv, copyMoveFileDto, false)).rejects.toThrow(/is not valid/i)\n })\n\n it('should avoid path traversal on Compress action', async () => {\n const compressFileDto: CompressFileDto = {\n name: '../../archive',\n compressInDirectory: false,\n files: [{ name: '../../foo', rootAlias: undefined }],\n extension: tarExtension\n }\n expect(() => transformAndValidate(CompressFileDto, compressFileDto)).toThrow()\n await expect(filesMethods.compress(userTest, spaceEnv, compressFileDto)).rejects.toThrow(/does not exist/i)\n compressFileDto.files[0].path = '../../../bar/../'\n await expect(filesMethods.compress(userTest, spaceEnv, compressFileDto)).rejects.toThrow(/is not valid/i)\n })\n})\n"],"names":["describe","FilesMethods","name","filesMethods","spacesManager","userTest","spaceEnv","id","alias","enabled","permissions","role","realBasePath","SpaceModel","getFilesPath","realPath","path","join","url","SPACE_REPOSITORY","FILES","beforeAll","module","Test","createTestingModule","imports","providers","SpacesManager","provide","DB_TOKEN_PROVIDER","useValue","Cache","get","ContextManager","NotificationsManager","UsersQueries","SharesManager","SpacesQueries","FilesManager","compress","jest","fn","compile","useLogger","UserModel","generateUserTest","updateSpacesQuota","mockReturnValue","undefined","it","expect","toBeDefined","copyMoveFileDto","dstDirectory","dstName","transformAndValidate","CopyMoveFileDto","toThrow","copyMove","rejects","compressFileDto","compressInDirectory","files","rootAlias","extension","tarExtension","CompressFileDto"],"mappings":"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"}
@@ -133,7 +133,7 @@ let FilesScheduler = class FilesScheduler {
133
133
  WHERE ${fk} IS NOT NULL
134
134
  AND id NOT IN (SELECT id
135
135
  FROM (SELECT id,
136
- ROW_NUMBER() OVER (PARTITION BY ${fk} ORDER BY ${_filesrecentsschema.filesRecents.mtime}) AS rn
136
+ ROW_NUMBER() OVER (PARTITION BY ${fk} ORDER BY ${_filesrecentsschema.filesRecents.mtime} DESC) AS rn
137
137
  FROM ${_filesrecentsschema.filesRecents}
138
138
  WHERE ${fk} IS NOT NULL) AS ranked
139
139
  WHERE ranked.rn <= ${keepNumber})
@@ -197,7 +197,7 @@ let FilesScheduler = class FilesScheduler {
197
197
  }
198
198
  };
199
199
  _ts_decorate([
200
- (0, _schedule.Timeout)(30_000),
200
+ (0, _schedule.Timeout)(10_000),
201
201
  _ts_metadata("design:type", Function),
202
202
  _ts_metadata("design:paramtypes", []),
203
203
  _ts_metadata("design:returntype", Promise)
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-scheduler.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Inject, Injectable, Logger } from '@nestjs/common'\nimport { Cron, CronExpression, Timeout } from '@nestjs/schedule'\nimport { isNotNull, sql } from 'drizzle-orm'\nimport { unionAll } from 'drizzle-orm/mysql-core'\nimport fs from 'node:fs/promises'\nimport path from 'node:path'\nimport { configuration } from '../../../configuration/config.environment'\nimport { Cache } from '../../../infrastructure/cache/services/cache.service'\nimport { DB_TOKEN_PROVIDER } from '../../../infrastructure/database/constants'\nimport { DBSchema } from '../../../infrastructure/database/interfaces/database.interface'\nimport { getTablesWithFileIdColumn } from '../../../infrastructure/database/utils'\nimport { USER_ROLE } from '../../users/constants/user'\nimport { UserModel } from '../../users/models/user.model'\nimport { users } from '../../users/schemas/users.schema'\nimport { CACHE_TASK_PREFIX } from '../constants/cache'\nimport { FileTask, FileTaskStatus } from '../models/file-task'\nimport { filesRecents } from '../schemas/files-recents.schema'\nimport { files } from '../schemas/files.schema'\nimport { dirHasChildren, isPathExists, removeFiles } from '../utils/files'\nimport { FilesContentManager } from './files-content-manager.service'\nimport { FilesTasksManager } from './files-tasks-manager.service'\n\n@Injectable()\nexport class FilesScheduler {\n private readonly logger = new Logger(FilesScheduler.name)\n\n constructor(\n @Inject(DB_TOKEN_PROVIDER) private readonly db: DBSchema,\n private readonly cache: Cache,\n private readonly filesContentManager: FilesContentManager\n ) {}\n\n @Timeout(30_000)\n async onStartup(): Promise<void> {\n try {\n await this.cleanupInterruptedTasks()\n await this.clearRecentFiles()\n } catch (e) {\n this.logger.error(e)\n }\n }\n\n @Timeout(180_000)\n async afterStartup(): Promise<void> {\n try {\n await this.indexContentFiles()\n } catch (e) {\n this.logger.error(e)\n }\n }\n\n async cleanupInterruptedTasks(): Promise<void> {\n this.logger.log(`${this.cleanupInterruptedTasks.name} - START`)\n try {\n let nb = 0\n const keys = await this.cache.keys(`${CACHE_TASK_PREFIX}-*`)\n for (const key of keys) {\n const task = await this.cache.get(key)\n if (task && task.status === FileTaskStatus.PENDING) {\n task.status = FileTaskStatus.ERROR\n task.result = 'Interrupted'\n nb++\n this.cache.set(key, task).catch((e: Error) => this.logger.error(`${this.cleanupInterruptedTasks.name} - ${e}`))\n }\n }\n this.logger.log(`${this.cleanupInterruptedTasks.name} - ${nb} tasks cleaned : END`)\n } catch (e) {\n this.logger.error(`${this.cleanupInterruptedTasks.name} - ${e}`)\n }\n }\n\n @Cron(CronExpression.EVERY_DAY_AT_MIDNIGHT)\n async cleanupUserTaskFiles(): Promise<void> {\n this.logger.log(`${this.cleanupUserTaskFiles.name} - START`)\n try {\n for (const user of await this.db\n .select({\n id: users.id,\n login: users.login,\n role: users.role\n })\n .from(users)) {\n const userTasksPath = UserModel.getTasksPath(user.login, user.role === USER_ROLE.GUEST, user.role === USER_ROLE.LINK)\n if (!(await isPathExists(userTasksPath))) {\n continue\n }\n if (await dirHasChildren(userTasksPath, false)) {\n const cacheKey = FilesTasksManager.getCacheKey(user.id)\n const keys = await this.cache.keys(cacheKey)\n const excludeFiles = (await this.cache.mget(keys))\n .filter((task: FileTask) => task && task.status === FileTaskStatus.PENDING && task.props.compressInDirectory === false)\n .map((task: FileTask) => task.name)\n for (const f of (await fs.readdir(userTasksPath)).filter((f: string) => excludeFiles.indexOf(f) === -1)) {\n try {\n removeFiles(path.join(userTasksPath, f)).catch((e: Error) => this.logger.error(`${this.cleanupUserTaskFiles.name} - ${e}`))\n } catch (e) {\n this.logger.error(`${this.cleanupUserTaskFiles.name} - unable to remove ${path.join(userTasksPath, f)} : ${e}`)\n }\n }\n }\n }\n } catch (e) {\n this.logger.error(`${this.cleanupUserTaskFiles.name} - ${e}`)\n }\n this.logger.log(`${this.cleanupUserTaskFiles.name} - END`)\n }\n\n @Cron(CronExpression.EVERY_8_HOURS)\n async clearRecentFiles(): Promise<void> {\n this.logger.log(`${this.clearRecentFiles.name} - START`)\n const keepNumber = 100\n let nbCleared = 0\n try {\n for (const fk of [filesRecents.ownerId, filesRecents.spaceId, filesRecents.shareId]) {\n const [r] = await this.db.execute(sql`\n DELETE\n FROM ${filesRecents}\n WHERE ${fk} IS NOT NULL\n AND id NOT IN (SELECT id\n FROM (SELECT id,\n ROW_NUMBER() OVER (PARTITION BY ${fk} ORDER BY ${filesRecents.mtime}) AS rn\n FROM ${filesRecents}\n WHERE ${fk} IS NOT NULL) AS ranked\n WHERE ranked.rn <= ${keepNumber})\n `)\n nbCleared += r.affectedRows\n }\n } catch (e) {\n this.logger.error(`${this.clearRecentFiles.name} - ${e}`)\n }\n this.logger.log(`${this.clearRecentFiles.name} - ${nbCleared} records cleared - END`)\n }\n\n @Cron(CronExpression.EVERY_4_HOURS)\n async indexContentFiles(): Promise<void> {\n // Conditional loading of file content indexing\n if (!configuration.applications.files.contentIndexing) return\n this.logger.log(`${this.indexContentFiles.name} - START`)\n await this.filesContentManager.parseAndIndexAllFiles()\n this.logger.log(`${this.indexContentFiles.name} - END`)\n }\n\n @Cron(CronExpression.EVERY_DAY_AT_4AM)\n async deleteOrphanFiles() {\n this.logger.log(`${this.deleteOrphanFiles.name} - START`)\n const selects: any[] = []\n for (const table of getTablesWithFileIdColumn()) {\n selects.push(this.db.selectDistinct({ id: table.fileId }).from(table).where(isNotNull(table.fileId)))\n }\n if (selects.length === 0) {\n this.logger.warn(`${this.deleteOrphanFiles.name} - no tables with fileId column`)\n return\n }\n const unionSub = (selects.length === 1 ? selects[0] : unionAll(...(selects as [any, any, ...any[]]))).as('u')\n // Debug\n // const [preview] = (await this.db.execute(sql`\n // SELECT f.id\n // FROM ${files} AS f\n // LEFT JOIN ${unionSub} ON ${unionSub.id} = f.id\n // WHERE ${unionSub.id} IS NULL\n // `)) as any[]\n // console.log(preview.length, preview)\n const deleteQuery = sql`\n DELETE f\n FROM ${files} AS f\n LEFT JOIN ${unionSub} ON ${unionSub.id} = f.id\n WHERE ${unionSub.id} IS NULL\n `\n try {\n await this.db.transaction(async (tx) => {\n const [r] = await tx.execute(deleteQuery)\n this.logger.log(`${this.deleteOrphanFiles.name} - files: ${r.affectedRows}`)\n })\n } catch (e) {\n this.logger.log(`${this.deleteOrphanFiles.name} - ${e}`)\n }\n this.logger.log(`${this.deleteOrphanFiles.name} - END`)\n }\n}\n"],"names":["FilesScheduler","onStartup","cleanupInterruptedTasks","clearRecentFiles","e","logger","error","afterStartup","indexContentFiles","log","name","nb","keys","cache","CACHE_TASK_PREFIX","key","task","get","status","FileTaskStatus","PENDING","ERROR","result","set","catch","cleanupUserTaskFiles","user","db","select","id","users","login","role","from","userTasksPath","UserModel","getTasksPath","USER_ROLE","GUEST","LINK","isPathExists","dirHasChildren","cacheKey","FilesTasksManager","getCacheKey","excludeFiles","mget","filter","props","compressInDirectory","map","f","fs","readdir","indexOf","removeFiles","path","join","keepNumber","nbCleared","fk","filesRecents","ownerId","spaceId","shareId","r","execute","sql","mtime","affectedRows","configuration","applications","files","contentIndexing","filesContentManager","parseAndIndexAllFiles","deleteOrphanFiles","selects","table","getTablesWithFileIdColumn","push","selectDistinct","fileId","where","isNotNull","length","warn","unionSub","unionAll","as","deleteQuery","transaction","tx","Logger","EVERY_DAY_AT_MIDNIGHT","EVERY_8_HOURS","EVERY_4_HOURS","EVERY_DAY_AT_4AM"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-scheduler.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Inject, Injectable, Logger } from '@nestjs/common'\nimport { Cron, CronExpression, Timeout } from '@nestjs/schedule'\nimport { isNotNull, sql } from 'drizzle-orm'\nimport { unionAll } from 'drizzle-orm/mysql-core'\nimport fs from 'node:fs/promises'\nimport path from 'node:path'\nimport { configuration } from '../../../configuration/config.environment'\nimport { Cache } from '../../../infrastructure/cache/services/cache.service'\nimport { DB_TOKEN_PROVIDER } from '../../../infrastructure/database/constants'\nimport { DBSchema } from '../../../infrastructure/database/interfaces/database.interface'\nimport { getTablesWithFileIdColumn } from '../../../infrastructure/database/utils'\nimport { USER_ROLE } from '../../users/constants/user'\nimport { UserModel } from '../../users/models/user.model'\nimport { users } from '../../users/schemas/users.schema'\nimport { CACHE_TASK_PREFIX } from '../constants/cache'\nimport { FileTask, FileTaskStatus } from '../models/file-task'\nimport { filesRecents } from '../schemas/files-recents.schema'\nimport { files } from '../schemas/files.schema'\nimport { dirHasChildren, isPathExists, removeFiles } from '../utils/files'\nimport { FilesContentManager } from './files-content-manager.service'\nimport { FilesTasksManager } from './files-tasks-manager.service'\n\n@Injectable()\nexport class FilesScheduler {\n private readonly logger = new Logger(FilesScheduler.name)\n\n constructor(\n @Inject(DB_TOKEN_PROVIDER) private readonly db: DBSchema,\n private readonly cache: Cache,\n private readonly filesContentManager: FilesContentManager\n ) {}\n\n @Timeout(10_000)\n async onStartup(): Promise<void> {\n try {\n await this.cleanupInterruptedTasks()\n await this.clearRecentFiles()\n } catch (e) {\n this.logger.error(e)\n }\n }\n\n @Timeout(180_000)\n async afterStartup(): Promise<void> {\n try {\n await this.indexContentFiles()\n } catch (e) {\n this.logger.error(e)\n }\n }\n\n async cleanupInterruptedTasks(): Promise<void> {\n this.logger.log(`${this.cleanupInterruptedTasks.name} - START`)\n try {\n let nb = 0\n const keys = await this.cache.keys(`${CACHE_TASK_PREFIX}-*`)\n for (const key of keys) {\n const task = await this.cache.get(key)\n if (task && task.status === FileTaskStatus.PENDING) {\n task.status = FileTaskStatus.ERROR\n task.result = 'Interrupted'\n nb++\n this.cache.set(key, task).catch((e: Error) => this.logger.error(`${this.cleanupInterruptedTasks.name} - ${e}`))\n }\n }\n this.logger.log(`${this.cleanupInterruptedTasks.name} - ${nb} tasks cleaned : END`)\n } catch (e) {\n this.logger.error(`${this.cleanupInterruptedTasks.name} - ${e}`)\n }\n }\n\n @Cron(CronExpression.EVERY_DAY_AT_MIDNIGHT)\n async cleanupUserTaskFiles(): Promise<void> {\n this.logger.log(`${this.cleanupUserTaskFiles.name} - START`)\n try {\n for (const user of await this.db\n .select({\n id: users.id,\n login: users.login,\n role: users.role\n })\n .from(users)) {\n const userTasksPath = UserModel.getTasksPath(user.login, user.role === USER_ROLE.GUEST, user.role === USER_ROLE.LINK)\n if (!(await isPathExists(userTasksPath))) {\n continue\n }\n if (await dirHasChildren(userTasksPath, false)) {\n const cacheKey = FilesTasksManager.getCacheKey(user.id)\n const keys = await this.cache.keys(cacheKey)\n const excludeFiles = (await this.cache.mget(keys))\n .filter((task: FileTask) => task && task.status === FileTaskStatus.PENDING && task.props.compressInDirectory === false)\n .map((task: FileTask) => task.name)\n for (const f of (await fs.readdir(userTasksPath)).filter((f: string) => excludeFiles.indexOf(f) === -1)) {\n try {\n removeFiles(path.join(userTasksPath, f)).catch((e: Error) => this.logger.error(`${this.cleanupUserTaskFiles.name} - ${e}`))\n } catch (e) {\n this.logger.error(`${this.cleanupUserTaskFiles.name} - unable to remove ${path.join(userTasksPath, f)} : ${e}`)\n }\n }\n }\n }\n } catch (e) {\n this.logger.error(`${this.cleanupUserTaskFiles.name} - ${e}`)\n }\n this.logger.log(`${this.cleanupUserTaskFiles.name} - END`)\n }\n\n @Cron(CronExpression.EVERY_8_HOURS)\n async clearRecentFiles(): Promise<void> {\n this.logger.log(`${this.clearRecentFiles.name} - START`)\n const keepNumber = 100\n let nbCleared = 0\n try {\n for (const fk of [filesRecents.ownerId, filesRecents.spaceId, filesRecents.shareId]) {\n const [r] = await this.db.execute(sql`\n DELETE\n FROM ${filesRecents}\n WHERE ${fk} IS NOT NULL\n AND id NOT IN (SELECT id\n FROM (SELECT id,\n ROW_NUMBER() OVER (PARTITION BY ${fk} ORDER BY ${filesRecents.mtime} DESC) AS rn\n FROM ${filesRecents}\n WHERE ${fk} IS NOT NULL) AS ranked\n WHERE ranked.rn <= ${keepNumber})\n `)\n nbCleared += r.affectedRows\n }\n } catch (e) {\n this.logger.error(`${this.clearRecentFiles.name} - ${e}`)\n }\n this.logger.log(`${this.clearRecentFiles.name} - ${nbCleared} records cleared - END`)\n }\n\n @Cron(CronExpression.EVERY_4_HOURS)\n async indexContentFiles(): Promise<void> {\n // Conditional loading of file content indexing\n if (!configuration.applications.files.contentIndexing) return\n this.logger.log(`${this.indexContentFiles.name} - START`)\n await this.filesContentManager.parseAndIndexAllFiles()\n this.logger.log(`${this.indexContentFiles.name} - END`)\n }\n\n @Cron(CronExpression.EVERY_DAY_AT_4AM)\n async deleteOrphanFiles() {\n this.logger.log(`${this.deleteOrphanFiles.name} - START`)\n const selects: any[] = []\n for (const table of getTablesWithFileIdColumn()) {\n selects.push(this.db.selectDistinct({ id: table.fileId }).from(table).where(isNotNull(table.fileId)))\n }\n if (selects.length === 0) {\n this.logger.warn(`${this.deleteOrphanFiles.name} - no tables with fileId column`)\n return\n }\n const unionSub = (selects.length === 1 ? selects[0] : unionAll(...(selects as [any, any, ...any[]]))).as('u')\n // Debug\n // const [preview] = (await this.db.execute(sql`\n // SELECT f.id\n // FROM ${files} AS f\n // LEFT JOIN ${unionSub} ON ${unionSub.id} = f.id\n // WHERE ${unionSub.id} IS NULL\n // `)) as any[]\n // console.log(preview.length, preview)\n const deleteQuery = sql`\n DELETE f\n FROM ${files} AS f\n LEFT JOIN ${unionSub} ON ${unionSub.id} = f.id\n WHERE ${unionSub.id} IS NULL\n `\n try {\n await this.db.transaction(async (tx) => {\n const [r] = await tx.execute(deleteQuery)\n this.logger.log(`${this.deleteOrphanFiles.name} - files: ${r.affectedRows}`)\n })\n } catch (e) {\n this.logger.log(`${this.deleteOrphanFiles.name} - ${e}`)\n }\n this.logger.log(`${this.deleteOrphanFiles.name} - END`)\n }\n}\n"],"names":["FilesScheduler","onStartup","cleanupInterruptedTasks","clearRecentFiles","e","logger","error","afterStartup","indexContentFiles","log","name","nb","keys","cache","CACHE_TASK_PREFIX","key","task","get","status","FileTaskStatus","PENDING","ERROR","result","set","catch","cleanupUserTaskFiles","user","db","select","id","users","login","role","from","userTasksPath","UserModel","getTasksPath","USER_ROLE","GUEST","LINK","isPathExists","dirHasChildren","cacheKey","FilesTasksManager","getCacheKey","excludeFiles","mget","filter","props","compressInDirectory","map","f","fs","readdir","indexOf","removeFiles","path","join","keepNumber","nbCleared","fk","filesRecents","ownerId","spaceId","shareId","r","execute","sql","mtime","affectedRows","configuration","applications","files","contentIndexing","filesContentManager","parseAndIndexAllFiles","deleteOrphanFiles","selects","table","getTablesWithFileIdColumn","push","selectDistinct","fileId","where","isNotNull","length","warn","unionSub","unionAll","as","deleteQuery","transaction","tx","Logger","EVERY_DAY_AT_MIDNIGHT","EVERY_8_HOURS","EVERY_4_HOURS","EVERY_DAY_AT_4AM"],"mappings":"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"}
@@ -55,6 +55,9 @@ _export(exports, {
55
55
  get genEtag () {
56
56
  return genEtag;
57
57
  },
58
+ get genUniqHashFromFileDBProps () {
59
+ return genUniqHashFromFileDBProps;
60
+ },
58
61
  get getMimeType () {
59
62
  return getMimeType;
60
63
  },
@@ -192,7 +195,7 @@ function getMimeType(fPath, isDir) {
192
195
  }
193
196
  return 'file';
194
197
  }
195
- function genEtag(file, rPath) {
198
+ function genEtag(file, rPath, weakPrefix = true) {
196
199
  if (!file) {
197
200
  if (!rPath) throw new Error('File or path are missing');
198
201
  const stats = (0, _nodefs.statSync)(rPath);
@@ -201,7 +204,12 @@ function genEtag(file, rPath) {
201
204
  mtime: stats.mtime.getTime()
202
205
  };
203
206
  }
204
- return `W/"${file.size.toString(16)}-${file.mtime.toString(16)}"`;
207
+ const etag = `${file.size.toString(16)}-${file.mtime.toString(16)}`;
208
+ return weakPrefix ? `W/"${etag}"` : etag;
209
+ }
210
+ function genUniqHashFromFileDBProps(dbFile) {
211
+ const dbFileString = `${Object.keys(dbFile).sort().map((k)=>`${k}=${String(dbFile[k])}`).join('|')}`;
212
+ return _nodecrypto.default.createHash(_files.DEFAULT_CHECKSUM_ALGORITHM).update(dbFileString, 'utf-8').digest('hex');
205
213
  }
206
214
  function removeFiles(rPath) {
207
215
  // if the file does not exist, no error is thrown