@sync-in/server 1.9.1 → 1.9.6

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 (177) hide show
  1. package/CHANGELOG.md +24 -8
  2. package/package.json +12 -12
  3. package/server/applications/files/interfaces/only-office-config.interface.js.map +1 -1
  4. package/server/applications/files/services/files-manager.service.js +2 -2
  5. package/server/applications/files/services/files-manager.service.js.map +1 -1
  6. package/server/applications/files/services/files-methods.service.spec.js +2 -4
  7. package/server/applications/files/services/files-methods.service.spec.js.map +1 -1
  8. package/server/applications/files/services/files-recents.service.js +4 -0
  9. package/server/applications/files/services/files-recents.service.js.map +1 -1
  10. package/server/applications/files/services/files-scheduler.service.js +23 -4
  11. package/server/applications/files/services/files-scheduler.service.js.map +1 -1
  12. package/server/applications/files/services/files-tasks-manager.service.js +1 -1
  13. package/server/applications/files/services/files-tasks-manager.service.js.map +1 -1
  14. package/server/applications/files/utils/send-file.js +18 -9
  15. package/server/applications/files/utils/send-file.js.map +1 -1
  16. package/server/applications/files/utils/url-file.js +6 -6
  17. package/server/applications/files/utils/url-file.js.map +1 -1
  18. package/server/applications/links/services/links-manager.service.js +1 -1
  19. package/server/applications/links/services/links-manager.service.js.map +1 -1
  20. package/server/applications/shares/interfaces/share-props.interface.js.map +1 -1
  21. package/server/applications/shares/services/shares-manager.service.js +6 -4
  22. package/server/applications/shares/services/shares-manager.service.js.map +1 -1
  23. package/server/applications/spaces/services/spaces-manager.service.js +15 -12
  24. package/server/applications/spaces/services/spaces-manager.service.js.map +1 -1
  25. package/server/applications/spaces/services/spaces-scheduler.service.js +9 -1
  26. package/server/applications/spaces/services/spaces-scheduler.service.js.map +1 -1
  27. package/server/applications/users/services/admin-users-manager.service.js +4 -0
  28. package/server/applications/users/services/admin-users-manager.service.js.map +1 -1
  29. package/server/applications/users/users.controller.js +4 -1
  30. package/server/applications/users/users.controller.js.map +1 -1
  31. package/static/assets/pdfjs/build/pdf.mjs +35 -20
  32. package/static/assets/pdfjs/build/pdf.mjs.map +1 -1
  33. package/static/assets/pdfjs/build/pdf.sandbox.mjs +2 -2
  34. package/static/assets/pdfjs/build/pdf.worker.mjs +1488 -52
  35. package/static/assets/pdfjs/build/pdf.worker.mjs.map +1 -1
  36. package/static/assets/pdfjs/version +1 -1
  37. package/static/assets/pdfjs/web/locale/be/viewer.ftl +2 -12
  38. package/static/assets/pdfjs/web/locale/bg/viewer.ftl +4 -0
  39. package/static/assets/pdfjs/web/locale/bs/viewer.ftl +0 -15
  40. package/static/assets/pdfjs/web/locale/ca/viewer.ftl +0 -4
  41. package/static/assets/pdfjs/web/locale/cs/viewer.ftl +4 -14
  42. package/static/assets/pdfjs/web/locale/cy/viewer.ftl +2 -12
  43. package/static/assets/pdfjs/web/locale/da/viewer.ftl +2 -12
  44. package/static/assets/pdfjs/web/locale/de/viewer.ftl +0 -12
  45. package/static/assets/pdfjs/web/locale/dsb/viewer.ftl +0 -12
  46. package/static/assets/pdfjs/web/locale/el/viewer.ftl +2 -12
  47. package/static/assets/pdfjs/web/locale/en-CA/viewer.ftl +0 -12
  48. package/static/assets/pdfjs/web/locale/en-GB/viewer.ftl +0 -12
  49. package/static/assets/pdfjs/web/locale/eo/viewer.ftl +2 -12
  50. package/static/assets/pdfjs/web/locale/es-AR/viewer.ftl +0 -12
  51. package/static/assets/pdfjs/web/locale/es-CL/viewer.ftl +2 -12
  52. package/static/assets/pdfjs/web/locale/es-ES/viewer.ftl +2 -12
  53. package/static/assets/pdfjs/web/locale/es-MX/viewer.ftl +0 -12
  54. package/static/assets/pdfjs/web/locale/eu/viewer.ftl +0 -12
  55. package/static/assets/pdfjs/web/locale/fi/viewer.ftl +0 -12
  56. package/static/assets/pdfjs/web/locale/fr/viewer.ftl +0 -12
  57. package/static/assets/pdfjs/web/locale/fur/viewer.ftl +0 -12
  58. package/static/assets/pdfjs/web/locale/fy-NL/viewer.ftl +0 -12
  59. package/static/assets/pdfjs/web/locale/gn/viewer.ftl +2 -12
  60. package/static/assets/pdfjs/web/locale/he/viewer.ftl +0 -12
  61. package/static/assets/pdfjs/web/locale/hsb/viewer.ftl +0 -12
  62. package/static/assets/pdfjs/web/locale/hu/viewer.ftl +0 -12
  63. package/static/assets/pdfjs/web/locale/hy-AM/viewer.ftl +0 -15
  64. package/static/assets/pdfjs/web/locale/ia/viewer.ftl +0 -12
  65. package/static/assets/pdfjs/web/locale/id/viewer.ftl +0 -15
  66. package/static/assets/pdfjs/web/locale/is/viewer.ftl +0 -15
  67. package/static/assets/pdfjs/web/locale/it/viewer.ftl +2 -14
  68. package/static/assets/pdfjs/web/locale/ja/viewer.ftl +7 -2
  69. package/static/assets/pdfjs/web/locale/ka/viewer.ftl +0 -12
  70. package/static/assets/pdfjs/web/locale/kab/viewer.ftl +0 -12
  71. package/static/assets/pdfjs/web/locale/kk/viewer.ftl +3 -12
  72. package/static/assets/pdfjs/web/locale/ko/viewer.ftl +0 -12
  73. package/static/assets/pdfjs/web/locale/nb-NO/viewer.ftl +2 -12
  74. package/static/assets/pdfjs/web/locale/nl/viewer.ftl +0 -12
  75. package/static/assets/pdfjs/web/locale/nn-NO/viewer.ftl +0 -12
  76. package/static/assets/pdfjs/web/locale/pa-IN/viewer.ftl +4 -12
  77. package/static/assets/pdfjs/web/locale/pl/viewer.ftl +2 -12
  78. package/static/assets/pdfjs/web/locale/pt-BR/viewer.ftl +2 -12
  79. package/static/assets/pdfjs/web/locale/rm/viewer.ftl +0 -12
  80. package/static/assets/pdfjs/web/locale/ro/viewer.ftl +2 -14
  81. package/static/assets/pdfjs/web/locale/ru/viewer.ftl +0 -12
  82. package/static/assets/pdfjs/web/locale/sc/viewer.ftl +0 -1
  83. package/static/assets/pdfjs/web/locale/sk/viewer.ftl +0 -12
  84. package/static/assets/pdfjs/web/locale/sl/viewer.ftl +2 -12
  85. package/static/assets/pdfjs/web/locale/sq/viewer.ftl +2 -12
  86. package/static/assets/pdfjs/web/locale/sv-SE/viewer.ftl +0 -12
  87. package/static/assets/pdfjs/web/locale/tg/viewer.ftl +2 -12
  88. package/static/assets/pdfjs/web/locale/th/viewer.ftl +0 -12
  89. package/static/assets/pdfjs/web/locale/tr/viewer.ftl +2 -12
  90. package/static/assets/pdfjs/web/locale/vi/viewer.ftl +0 -12
  91. package/static/assets/pdfjs/web/locale/zh-CN/viewer.ftl +2 -12
  92. package/static/assets/pdfjs/web/locale/zh-TW/viewer.ftl +0 -12
  93. package/static/assets/pdfjs/web/viewer.css +53 -41
  94. package/static/assets/pdfjs/web/viewer.html +493 -132
  95. package/static/assets/pdfjs/web/viewer.mjs +225 -150
  96. package/static/assets/pdfjs/web/viewer.mjs.map +1 -1
  97. package/static/{chunk-LYZGJZNP.js → chunk-25PWAXTJ.js} +1 -1
  98. package/static/{chunk-YMAN4LIU.js → chunk-2CAAJBRO.js} +1 -1
  99. package/static/{chunk-GRLHFXGB.js → chunk-2F42MZQ5.js} +1 -1
  100. package/static/{chunk-6B3GGAV3.js → chunk-2U5VKTML.js} +1 -1
  101. package/static/{chunk-HKRGIRKB.js → chunk-3AR5VNJE.js} +1 -1
  102. package/static/chunk-3WS72A6C.js +1 -0
  103. package/static/chunk-4GBA6EJ4.js +1 -0
  104. package/static/{chunk-FNOTGWRW.js → chunk-4ZKAVMB4.js} +1 -1
  105. package/static/{chunk-XQGPSNQB.js → chunk-5ATJIR5S.js} +1 -1
  106. package/static/{chunk-HMOB6XC5.js → chunk-5GIWZKNS.js} +1 -1
  107. package/static/{chunk-ZCOWBVOT.js → chunk-5HYSNQR4.js} +1 -1
  108. package/static/{chunk-BHZEPHRI.js → chunk-5KVI243T.js} +1 -1
  109. package/static/{chunk-Q556XB3S.js → chunk-5NFH4E2B.js} +1 -1
  110. package/static/{chunk-XCLK7NJL.js → chunk-7H5O4BLV.js} +1 -1
  111. package/static/{chunk-RDNTK4YH.js → chunk-7HL5Z6PF.js} +1 -1
  112. package/static/{chunk-V6FA5QY4.js → chunk-7QYALK5T.js} +1 -1
  113. package/static/{chunk-GNZPP2VO.js → chunk-ANH4VNOS.js} +1 -1
  114. package/static/{chunk-EIYRBM4J.js → chunk-AYYJZMBE.js} +1 -1
  115. package/static/chunk-B2A4HNDC.js +1 -0
  116. package/static/{chunk-NOPACN4F.js → chunk-B4TDS6AQ.js} +1 -1
  117. package/static/chunk-BJARRIS6.js +562 -0
  118. package/static/{chunk-PXRT4L57.js → chunk-BVKDW5XO.js} +1 -1
  119. package/static/{chunk-7CKHC72R.js → chunk-BX3QZ7IL.js} +1 -1
  120. package/static/{chunk-MIA5YBOI.js → chunk-C5T7RZSD.js} +1 -1
  121. package/static/{chunk-ACUF7IKP.js → chunk-CHMDM2ZW.js} +1 -1
  122. package/static/{chunk-U4RW6XG5.js → chunk-CUC7R6C2.js} +1 -1
  123. package/static/{chunk-XOF4UW3S.js → chunk-D6QWQHWE.js} +1 -1
  124. package/static/{chunk-6NEBGCAZ.js → chunk-DK2LAJEL.js} +1 -1
  125. package/static/{chunk-C3AAEQKW.js → chunk-DQAQUSVW.js} +1 -1
  126. package/static/{chunk-NN4ONTOT.js → chunk-DU4Q4RWJ.js} +1 -1
  127. package/static/{chunk-M4XL3JN5.js → chunk-E5C4QRNQ.js} +2 -2
  128. package/static/{chunk-BU4ZICZR.js → chunk-EPDWJEPD.js} +1 -1
  129. package/static/{chunk-I2S3XPC5.js → chunk-FCR5AEHR.js} +2 -2
  130. package/static/{chunk-4PZPHJ7L.js → chunk-FEQUP26G.js} +1 -1
  131. package/static/{chunk-ZXXHFBGL.js → chunk-FSGT46LM.js} +1 -1
  132. package/static/{chunk-QMHUIHSR.js → chunk-GENTF6JM.js} +1 -1
  133. package/static/{chunk-DQ3GEMPM.js → chunk-GLPKRULI.js} +1 -1
  134. package/static/{chunk-AZ5TF5Y3.js → chunk-GRV44RYI.js} +1 -1
  135. package/static/{chunk-JY2I3HGL.js → chunk-GYYJ4FWN.js} +1 -1
  136. package/static/{chunk-ERDZ7IVF.js → chunk-HB5DC7RJ.js} +1 -1
  137. package/static/{chunk-DJDRX53V.js → chunk-HLKZCMKV.js} +1 -1
  138. package/static/{chunk-HNMGPG72.js → chunk-IBC7CFBQ.js} +1 -1
  139. package/static/{chunk-5E3TYOL3.js → chunk-IIKL33TV.js} +1 -1
  140. package/static/{chunk-TGLJFALR.js → chunk-JYHTSSKW.js} +1 -1
  141. package/static/{chunk-VQJYCYWI.js → chunk-KAAFVHYE.js} +1 -1
  142. package/static/{chunk-VK7XMFVE.js → chunk-KWKZN53T.js} +1 -1
  143. package/static/{chunk-KDEEERWZ.js → chunk-LBXOAKBD.js} +1 -1
  144. package/static/{chunk-Y5RLD72B.js → chunk-LFAQLJZK.js} +1 -1
  145. package/static/chunk-MGMDT4VN.js +1 -0
  146. package/static/chunk-MWUUM2NK.js +13 -0
  147. package/static/{chunk-JMYAD7E2.js → chunk-NHMYAVJK.js} +1 -1
  148. package/static/{chunk-HHWXIK2M.js → chunk-NQCKX2AD.js} +1 -1
  149. package/static/{chunk-HCSWO7BO.js → chunk-O233BXWK.js} +1 -1
  150. package/static/{chunk-2TB2INBF.js → chunk-ODAQRAPO.js} +1 -1
  151. package/static/{chunk-RK2ONYTL.js → chunk-OVUMPMVM.js} +1 -1
  152. package/static/chunk-Q6B4OVER.js +5 -0
  153. package/static/{chunk-3YDYZLF7.js → chunk-QKMN3S4M.js} +1 -1
  154. package/static/{chunk-23UUFZSR.js → chunk-QUUQOBTF.js} +1 -1
  155. package/static/chunk-QV5LQKTS.js +1 -0
  156. package/static/{chunk-NKGKBQBX.js → chunk-S4UTSOPV.js} +1 -1
  157. package/static/{chunk-5S6KPQRA.js → chunk-SF6Q6VRC.js} +1 -1
  158. package/static/{chunk-BODMMLVB.js → chunk-TOCCCZP2.js} +1 -1
  159. package/static/{chunk-3WLBVJ2S.js → chunk-UO7ATVQG.js} +1 -1
  160. package/static/{chunk-AADK5D2H.js → chunk-X5UDV4ZB.js} +1 -1
  161. package/static/{chunk-S2VBGI6Q.js → chunk-XIQXRSZ2.js} +1 -1
  162. package/static/chunk-YYTDPI5S.js +1 -0
  163. package/static/{chunk-EL6QL4TP.js → chunk-Z2KBIZ5D.js} +1 -1
  164. package/static/{chunk-K657XPXA.js → chunk-ZCOEP4O2.js} +1 -1
  165. package/static/index.html +2 -2
  166. package/static/main-ODUA232E.js +11 -0
  167. package/static/{styles-Q4OZOSSK.css → styles-S5HVK4H5.css} +1 -1
  168. package/static/chunk-5XUIPWOH.js +0 -1
  169. package/static/chunk-EDJAISWO.js +0 -13
  170. package/static/chunk-H6WOTGQ5.js +0 -1
  171. package/static/chunk-HC7F57NA.js +0 -1
  172. package/static/chunk-IOIBQGHN.js +0 -562
  173. package/static/chunk-J6YSFHLZ.js +0 -1
  174. package/static/chunk-JWPXQOS3.js +0 -5
  175. package/static/chunk-QVFPHTOH.js +0 -1
  176. package/static/chunk-YCTCESL4.js +0 -1
  177. package/static/main-56PZQ6TJ.js +0 -11
package/CHANGELOG.md CHANGED
@@ -1,16 +1,32 @@
1
1
 
2
- ## [1.9.1](https://github.com/Sync-in/server/compare/v1.8.1...v1.9.1) (2025-11-25)
2
+ ## [1.9.6](https://github.com/Sync-in/server/compare/v1.9.5...v1.9.6) (2025-12-16)
3
+
4
+ ### Bug Fixes
5
+
6
+ * **backend:files:** skip adding recents for trashed files ([c445196](https://github.com/Sync-in/server/commit/c445196914b2d351fba9218698a24496b1d6036c))
7
+ * **backend:schedulers:** resolve scheduled methods being skipped because of @Timeout decorator overlap ([50f4140](https://github.com/Sync-in/server/commit/50f4140a7b0b478e6b499ea8884b43f13595bb71))
8
+ * **frontend:files:** enable autoplay for video in media viewer component ([20fe25f](https://github.com/Sync-in/server/commit/20fe25fba00987994076d09489febd5593e08cef))
9
+ * **frontend:files:** remove hidden class from buttons for consistent visibility across breakpoints ([a60538a](https://github.com/Sync-in/server/commit/a60538ad01c675dacdac7ed4d80ca2bdf5f369ba))
10
+ * **frontend:files:** update file metadata timestamps on save and align OnlyOffice state change handlers ([db768e1](https://github.com/Sync-in/server/commit/db768e14452f4712df9f443350c214e0700b7270))
11
+ * **frontend:search:** improve search input layout and update filter button visibility for responsiveness ([09ebce6](https://github.com/Sync-in/server/commit/09ebce612fa2d72699a4d60bf9896f8e3c0fc4e4))
12
+ * **frontend:spaces:** show disabled space message to space managers ([f8bcdf7](https://github.com/Sync-in/server/commit/f8bcdf7fdd4b25abc2ba4b74715adbb0ae04a3e3))
13
+
14
+ ## [1.9.3](https://github.com/Sync-in/server/compare/v1.9.1...v1.9.3) (2025-12-07)
3
15
 
4
- ### Highlights 1.9
16
+ ### Security Fixes
17
+ * **backend:security:** prevent stored XSS by serving files with `Content-Disposition: attachment` to avoid arbitrary JavaScript execution in the browser ([a6276d0](https://github.com/Sync-in/server/commit/a6276d067725637310e4e83a3eee337aae81f439))
5
18
 
6
- * 🎨 **Visual identity updated** — refreshed branding.
7
- * 🖼️ **Enhanced image viewer** browse all images in a directory, start slideshows, use fullscreen mode, and view detailed properties.
8
- * ✏️ **Native text editor** search, syntax highlighting, undo/redo, and lock/unlock capabilities built directly into the UI.
9
- * 🔐 **Improved file lock management** users can request unlocks from lock owners, and file owners can now remove locks themselves.
10
- * ⚠️ **Overwrite confirmation dialogs** added confirmation prompts for uploads, renaming, copying, and moving files to prevent accidental overwrites.
19
+ ### Bug Fixes
20
+ * **ci:** update Dockerfile to use alpine3.22 to avoid errors with busybox-1.37.0-r29 ([ede1bec](https://github.com/Sync-in/server/commit/ede1bec4b3c33f17c3b94c32d68c4b642ee710c0))
21
+ * **backend:users:** clear whitelist caches when group visibility changes ([071c3ae](https://github.com/Sync-in/server/commit/071c3aed68d3bdacead571d39a1f4006b2380915))
22
+ * **frontend:files:** fix DataTransfer usage after async operations and delay overwrite until analysis completes to restore overwrite on dropped files ([d9935e5](https://github.com/Sync-in/server/commit/d9935e5a3887448635c30fd49f22657461177610))
23
+ * **frontend:styles:** add min-width on app-auth background class ([dffd5e5](https://github.com/Sync-in/server/commit/dffd5e5c7a1a65994970bedf33a95dd00827aa94))
11
24
 
12
- ➡️ [More details](https://sync-in.com/news/sync-in-server-1-9)
25
+ ### Community Highlights ❤️
13
26
 
27
+ We would like to thank @x0root for reporting this vulnerability and helping improve the security of the project.
28
+
29
+ ## [1.9.1](https://github.com/Sync-in/server/compare/v1.8.1...v1.9.1) (2025-11-25)
14
30
 
15
31
  ### Features
16
32
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sync-in/server",
3
- "version": "1.9.1",
3
+ "version": "1.9.6",
4
4
  "description": "The secure, open-source platform for file storage, sharing, collaboration, and sync",
5
5
  "author": {
6
6
  "name": "Johan Legrand",
@@ -83,11 +83,11 @@
83
83
  "@nestjs/common": "11.1.9",
84
84
  "@nestjs/config": "4.0.2",
85
85
  "@nestjs/core": "11.1.9",
86
- "@nestjs/jwt": "11.0.1",
86
+ "@nestjs/jwt": "11.0.2",
87
87
  "@nestjs/passport": "11.0.5",
88
88
  "@nestjs/platform-fastify": "11.1.9",
89
89
  "@nestjs/platform-socket.io": "11.1.9",
90
- "@nestjs/schedule": "6.0.1",
90
+ "@nestjs/schedule": "6.1.0",
91
91
  "@nestjs/websockets": "11.1.9",
92
92
  "@socket.io/cluster-adapter": "0.3.0",
93
93
  "@socket.io/redis-adapter": "8.3.0",
@@ -95,24 +95,24 @@
95
95
  "archiver": "7.0.1",
96
96
  "bcryptjs": "3.0.3",
97
97
  "class-transformer": "0.5.1",
98
- "class-validator": "0.14.2",
98
+ "class-validator": "0.14.3",
99
99
  "deepmerge": "4.3.1",
100
- "drizzle-kit": "0.31.7",
100
+ "drizzle-kit": "0.31.8",
101
101
  "drizzle-orm": "0.44.7",
102
- "fast-xml-parser": "5.3.2",
102
+ "fast-xml-parser": "5.3.3",
103
103
  "fs-extra": "11.3.2",
104
104
  "html-to-text": "9.0.5",
105
105
  "js-yaml": "4.1.1",
106
- "ldapts": "8.0.9",
107
- "mime-types": "3.0.1",
108
- "mysql2": "3.15.3",
109
- "nestjs-pino": "4.4.1",
110
- "nodemailer": "7.0.10",
106
+ "ldapts": "8.0.25",
107
+ "mime-types": "3.0.2",
108
+ "mysql2": "3.16.0",
109
+ "nestjs-pino": "4.5.0",
110
+ "nodemailer": "7.0.11",
111
111
  "passport-http": "0.3.0",
112
112
  "passport-jwt": "4.0.1",
113
113
  "passport-local": "1.0.0",
114
114
  "passport": "0.7.0",
115
- "pino-pretty": "13.1.2",
115
+ "pino-pretty": "13.1.3",
116
116
  "qrcode-generator": "2.0.4",
117
117
  "redis": "5.10.0",
118
118
  "sax": "1.4.3",
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/interfaces/only-office-config.interface.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 { FILE_MODE } from '../constants/operations'\n\nexport interface OnlyOfficeReqConfig {\n documentServerUrl: string\n config: OnlyOfficeConfig\n}\n\nexport interface OnlyOfficeConvertForm {\n key: string\n url: string\n outputtype: string\n filetype: string\n async: boolean\n token?: string\n}\n\nexport interface OnlyOfficeConfig {\n documentType?: string\n token?: string\n type?: 'mobile' | 'desktop'\n height?: string\n width?: string\n document?: {\n fileType: string\n key: string\n referenceData?: {\n fileKey: string\n instanceId: string\n }\n title: string\n url: string\n info?: {\n owner?: string\n uploaded?: string\n favorite?: boolean\n folder?: string\n sharingSettings?: any[]\n }\n permissions?: {\n /**\n * @deprecated Deprecated since version 5.5, please add the onRequestRestore field instead.\n */\n changeHistory?: boolean\n chat?: boolean\n comment?: boolean\n commentGroups?: any\n copy?: boolean\n deleteCommentAuthorOnly?: boolean\n download?: boolean\n edit?: boolean\n editCommentAuthorOnly?: boolean\n fillForms?: boolean\n modifyContentControl?: boolean\n modifyFilter?: boolean\n print?: boolean\n protect?: boolean\n review?: boolean\n reviewGroups?: string[]\n userInfoGroups?: string[]\n }\n }\n editorConfig?: {\n actionLink?: any\n callbackUrl?: string\n coEditing?: {\n mode: string\n change: boolean\n }\n createUrl?: string\n lang?: string\n mode?: FILE_MODE\n recent?: any[]\n region?: string\n templates?: any[]\n user?: {\n group?: string\n id?: string\n image?: string\n name?: string\n }\n customization?: {\n anonymous?: {\n request?: boolean\n label?: string\n }\n autosave?: boolean\n forcesave?: boolean\n close?: {\n visible: boolean\n text: string\n }\n comments?: boolean\n compactHeader?: boolean\n compactToolbar?: boolean\n compatibleFeatures?: boolean\n customer?: {\n address?: string\n info?: string\n logo?: string\n logoDark?: string\n mail?: string\n name?: string\n phone?: string\n www?: string\n }\n features?: any\n feedback?: any\n goback?: any\n help?: boolean\n hideNotes?: boolean\n hideRightMenu?: boolean\n hideRulers?: boolean\n integrationMode?: string\n logo?: {\n image?: string\n imageDark?: string\n imageLight?: string\n imageEmbedded?: string\n url?: string\n visible?: boolean\n }\n macros?: boolean\n macrosMode?: string\n mentionShare?: boolean\n mobileForceView?: boolean\n plugins?: boolean\n review?: {\n hideReviewDisplay?: boolean\n hoverMode?: boolean\n reviewDisplay?: string\n showReviewChanges?: boolean\n trackChanges?: boolean\n }\n submitForm?: boolean\n toolbarHideFileName?: boolean\n toolbarNoTabs?: boolean\n uiTheme?: string\n unit?: string\n zoom?: number\n about?: boolean\n }\n embedded?: {\n embedUrl?: string\n fullscreenUrl?: string\n saveUrl?: string\n shareUrl?: string\n toolbarDocked?: string\n }\n plugins?: {\n autostart?: string[]\n options?: {\n all?: any\n pluginGuid: any\n }\n pluginsData?: string[]\n }\n }\n events?: {\n onAppReady?: (event: object) => void\n onCollaborativeChanges?: (event: object) => void\n onDocumentReady?: (event: object) => void\n onDocumentStateChange?: (event: object) => void\n onDownloadAs?: (event: object) => void\n onError?: (event: object) => void\n onInfo?: (event: object) => void\n onMetaChange?: (event: object) => void\n onMakeActionLink?: (event: object) => void\n onRequestRefreshFile?: (event: object) => void\n onPluginsReady?: (event: object) => void\n onReady?: (event: object) => void\n onRequestClose?: (event: object) => void\n onRequestCreateNew?: (event: object) => void\n onRequestEditRights?: (event: object) => void\n onRequestHistory?: (event: object) => void\n onRequestHistoryClose?: (event: object) => void\n onRequestHistoryData?: (event: object) => void\n onRequestInsertImage?: (event: object) => void\n onRequestOpen?: (event: object) => void\n onRequestReferenceData?: (event: object) => void\n onRequestReferenceSource?: (event: object) => void\n onRequestRename?: (event: object) => void\n onRequestRestore?: (event: object) => void\n onRequestSaveAs?: (event: object) => void\n onRequestSelectDocument?: (event: object) => void\n onRequestSelectSpreadsheet?: (event: object) => void\n onRequestSendNotify?: (event: object) => void\n onRequestSharingSettings?: (event: object) => void\n onRequestStartFilling: (event: object) => void\n onRequestUsers?: (event: object) => void\n onSubmit?: (event: object) => void\n onWarning?: (event: object) => void\n }\n}\n\nexport interface OnlyOfficeCallBack {\n /* documentation : https://api.onlyoffice.com/docs/docs-api/usage-api/callback-handler/ */\n key: string // document key\n /*\n status:\n 1 - document is being edited\n 2 - document is ready for saving\n 3 - document saving error has occurred\n 4 - document is closed with no changes\n 6 - document is being edited, but the current document state is saved\n 7 - error has occurred while force saving the document\n */\n status: 1 | 2 | 3 | 4 | 6 | 7\n url?: string // link to download the modified version (for status: 2, 3, 6 or 7)\n notmodified?: boolean // only with status 2\n /*\n actions:\n 0 - the user disconnects from the document co-editing\n 1 - the new user connects to the document co-editing\n 2 - the user clicks the forcesave button.\n */\n actions?: { type: 0 | 1 | 2; userid: string }[]\n forcesavetype?: 0 | 1 | 2 | 3 // The type is present when the status value is equal to 6 or 7 only\n /*\n forcesavetype:\n 0 - to the command service\n 1 - each time the saving is done (e.g. the Save button is clicked), which is only available when the forcesave option is set to true\n 2 - by timer with the settings from the server config\n 3 - each time the form is submitted (e.g. the Complete & Submit button is clicked)\n */\n users?: string[] // when multiple users are editing the document\n}\n"],"names":[],"mappings":"AAAA;;;;CAIC"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/interfaces/only-office-config.interface.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 { FILE_MODE } from '../constants/operations'\n\nexport interface OnlyOfficeReqConfig {\n documentServerUrl: string\n config: OnlyOfficeConfig\n}\n\nexport interface OnlyOfficeConvertForm {\n key: string\n url: string\n outputtype: string\n filetype: string\n async: boolean\n token?: string\n}\n\nexport interface OnlyOfficeConfig {\n documentType?: string\n token?: string\n type?: 'mobile' | 'desktop'\n height?: string\n width?: string\n document?: {\n fileType: string\n key: string\n referenceData?: {\n fileKey: string\n instanceId: string\n }\n title: string\n url: string\n info?: {\n owner?: string\n uploaded?: string\n favorite?: boolean\n folder?: string\n sharingSettings?: any[]\n }\n permissions?: {\n /**\n * @deprecated Deprecated since version 5.5, please add the onRequestRestore field instead.\n */\n changeHistory?: boolean\n chat?: boolean\n comment?: boolean\n commentGroups?: any\n copy?: boolean\n deleteCommentAuthorOnly?: boolean\n download?: boolean\n edit?: boolean\n editCommentAuthorOnly?: boolean\n fillForms?: boolean\n modifyContentControl?: boolean\n modifyFilter?: boolean\n print?: boolean\n protect?: boolean\n review?: boolean\n reviewGroups?: string[]\n userInfoGroups?: string[]\n }\n }\n editorConfig?: {\n actionLink?: any\n callbackUrl?: string\n coEditing?: {\n mode: string\n change: boolean\n }\n createUrl?: string\n lang?: string\n mode?: FILE_MODE\n recent?: any[]\n region?: string\n templates?: any[]\n user?: {\n group?: string\n id?: string\n image?: string\n name?: string\n }\n customization?: {\n anonymous?: {\n request?: boolean\n label?: string\n }\n autosave?: boolean\n forcesave?: boolean\n close?: {\n visible: boolean\n text: string\n }\n comments?: boolean\n compactHeader?: boolean\n compactToolbar?: boolean\n compatibleFeatures?: boolean\n customer?: {\n address?: string\n info?: string\n logo?: string\n logoDark?: string\n mail?: string\n name?: string\n phone?: string\n www?: string\n }\n features?: any\n feedback?: any\n goback?: any\n help?: boolean\n hideNotes?: boolean\n hideRightMenu?: boolean\n hideRulers?: boolean\n integrationMode?: string\n logo?: {\n image?: string\n imageDark?: string\n imageLight?: string\n imageEmbedded?: string\n url?: string\n visible?: boolean\n }\n macros?: boolean\n macrosMode?: string\n mentionShare?: boolean\n mobileForceView?: boolean\n plugins?: boolean\n review?: {\n hideReviewDisplay?: boolean\n hoverMode?: boolean\n reviewDisplay?: string\n showReviewChanges?: boolean\n trackChanges?: boolean\n }\n submitForm?: boolean\n toolbarHideFileName?: boolean\n toolbarNoTabs?: boolean\n uiTheme?: string\n unit?: string\n zoom?: number\n about?: boolean\n }\n embedded?: {\n embedUrl?: string\n fullscreenUrl?: string\n saveUrl?: string\n shareUrl?: string\n toolbarDocked?: string\n }\n plugins?: {\n autostart?: string[]\n options?: {\n all?: any\n pluginGuid: any\n }\n pluginsData?: string[]\n }\n }\n events?: {\n onAppReady?: (event: object) => void\n onCollaborativeChanges?: (event: object) => void\n onDocumentReady?: (event: object) => void\n onDocumentStateChange?: (event: { data: boolean }) => void\n onDownloadAs?: (event: object) => void\n onError?: (event: object) => void\n onInfo?: (event: object) => void\n onMetaChange?: (event: object) => void\n onMakeActionLink?: (event: object) => void\n onRequestRefreshFile?: (event: object) => void\n onPluginsReady?: (event: object) => void\n onReady?: (event: object) => void\n onRequestClose?: (event: object) => void\n onRequestCreateNew?: (event: object) => void\n onRequestEditRights?: (event: object) => void\n onRequestHistory?: (event: object) => void\n onRequestHistoryClose?: (event: object) => void\n onRequestHistoryData?: (event: object) => void\n onRequestInsertImage?: (event: object) => void\n onRequestOpen?: (event: object) => void\n onRequestReferenceData?: (event: object) => void\n onRequestReferenceSource?: (event: object) => void\n onRequestRename?: (event: object) => void\n onRequestRestore?: (event: object) => void\n onRequestSaveAs?: (event: object) => void\n onRequestSelectDocument?: (event: object) => void\n onRequestSelectSpreadsheet?: (event: object) => void\n onRequestSendNotify?: (event: object) => void\n onRequestSharingSettings?: (event: object) => void\n onRequestStartFilling?: (event: object) => void\n onRequestUsers?: (event: object) => void\n onSubmit?: (event: object) => void\n onWarning?: (event: object) => void\n }\n}\n\nexport interface OnlyOfficeCallBack {\n /* documentation : https://api.onlyoffice.com/docs/docs-api/usage-api/callback-handler/ */\n key: string // document key\n /*\n status:\n 1 - document is being edited\n 2 - document is ready for saving\n 3 - document saving error has occurred\n 4 - document is closed with no changes\n 6 - document is being edited, but the current document state is saved\n 7 - error has occurred while force saving the document\n */\n status: 1 | 2 | 3 | 4 | 6 | 7\n url?: string // link to download the modified version (for status: 2, 3, 6 or 7)\n notmodified?: boolean // only with status 2\n /*\n actions:\n 0 - the user disconnects from the document co-editing\n 1 - the new user connects to the document co-editing\n 2 - the user clicks the forcesave button.\n */\n actions?: { type: 0 | 1 | 2; userid: string }[]\n forcesavetype?: 0 | 1 | 2 | 3 // The type is present when the status value is equal to 6 or 7 only\n /*\n forcesavetype:\n 0 - to the command service\n 1 - each time the saving is done (e.g. the Save button is clicked), which is only available when the forcesave option is set to true\n 2 - by timer with the settings from the server config\n 3 - each time the form is submitted (e.g. the Complete & Submit button is clicked)\n */\n users?: string[] // when multiple users are editing the document\n}\n"],"names":[],"mappings":"AAAA;;;;CAIC"}
@@ -57,8 +57,8 @@ function _ts_metadata(k, v) {
57
57
  if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
58
58
  }
59
59
  let FilesManager = class FilesManager {
60
- sendFileFromSpace(space, asAttachment = false, downloadName = '') {
61
- return new _sendfile.SendFile(space.realPath, asAttachment, downloadName);
60
+ sendFileFromSpace(space, downloadName = '') {
61
+ return new _sendfile.SendFile(space.realPath, downloadName);
62
62
  }
63
63
  async saveStream(user, space, req, options) {
64
64
  // if tmpPath is used, we lock the final destination during the transfer
@@ -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, asAttachment = false, downloadName = ''): SendFile {\n return new SendFile(space.realPath, asAttachment, 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","asAttachment","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 { 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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"}
@@ -108,7 +108,7 @@ describe(_filesmethodsservice.FilesMethods.name, ()=>{
108
108
  dstDirectory: '../../../foo',
109
109
  dstName: '../bar/../'
110
110
  };
111
- expect(()=>(0, _functions.transformAndValidate)(_fileoperationsdto.CompressFileDto, copyMoveFileDto)).toThrow();
111
+ expect(()=>(0, _functions.transformAndValidate)(_fileoperationsdto.CopyMoveFileDto, copyMoveFileDto)).toThrow();
112
112
  await expect(filesMethods.copyMove(userTest, spaceEnv, copyMoveFileDto, false)).rejects.toThrow(/is not valid/i);
113
113
  });
114
114
  it('should avoid path traversal on Compress action', async ()=>{
@@ -124,9 +124,7 @@ describe(_filesmethodsservice.FilesMethods.name, ()=>{
124
124
  extension: _compress.tarExtension
125
125
  };
126
126
  expect(()=>(0, _functions.transformAndValidate)(_fileoperationsdto.CompressFileDto, compressFileDto)).toThrow();
127
- await expect(filesMethods.compress(userTest, spaceEnv, {
128
- ...compressFileDto
129
- })).rejects.toThrow(/does not exist/i);
127
+ await expect(filesMethods.compress(userTest, spaceEnv, compressFileDto)).rejects.toThrow(/does not exist/i);
130
128
  compressFileDto.files[0].path = '../../../bar/../';
131
129
  await expect(filesMethods.compress(userTest, spaceEnv, compressFileDto)).rejects.toThrow(/is not valid/i);
132
130
  });
@@ -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: Partial<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 }\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 = { dstDirectory: '../../../foo', dstName: '../bar/../' } satisfies CopyMoveFileDto\n expect(() => transformAndValidate(CompressFileDto, copyMoveFileDto satisfies CopyMoveFileDto)).toThrow()\n await expect((filesMethods as any).copyMove(userTest, spaceEnv as SpaceEnv, copyMoveFileDto satisfies CopyMoveFileDto, false)).rejects.toThrow(\n /is not valid/i\n )\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 as typeof tarExtension\n }\n expect(() => transformAndValidate(CompressFileDto, compressFileDto satisfies CompressFileDto)).toThrow()\n await expect(filesMethods.compress(userTest, spaceEnv as SpaceEnv, { ...(compressFileDto as CompressFileDto) })).rejects.toThrow(\n /does not exist/i\n )\n compressFileDto.files[0].path = '../../../bar/../'\n await expect(filesMethods.compress(userTest, spaceEnv as SpaceEnv, compressFileDto as 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","CompressFileDto","toThrow","copyMove","rejects","compressFileDto","compressInDirectory","files","rootAlias","extension","tarExtension","compress"],"mappings":"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"}
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"}
@@ -36,6 +36,10 @@ let FilesRecents = class FilesRecents {
36
36
  return this.filesQueries.getRecentsFromUser(user.id, spaceIds, shareIds, limit);
37
37
  }
38
38
  async updateRecents(user, space, files) {
39
+ if (space.inTrashRepository) {
40
+ // Ignore trashed files
41
+ return;
42
+ }
39
43
  const timestamp = (0, _shared.currentTimeStamp)(null, true) - (0, _functions.convertHumanTimeToMs)(this.keepTime);
40
44
  const location = this.getLocation(user, space, files);
41
45
  // only store files, ignore dirs
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-recents.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 { Injectable } from '@nestjs/common'\nimport { convertDiffUpdate, convertHumanTimeToMs, diffCollection } from '../../../common/functions'\nimport { currentTimeStamp } from '../../../common/shared'\nimport { SharesQueries } from '../../shares/services/shares-queries.service'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesQueries } from '../../spaces/services/spaces-queries.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { FileProps } from '../interfaces/file-props.interface'\nimport { FileRecentLocation } from '../interfaces/file-recent-location.interface'\nimport { FileRecent } from '../schemas/file-recent.interface'\nimport { FilesQueries } from './files-queries.service'\n\n@Injectable()\nexport class FilesRecents {\n private readonly keepTime = '14d'\n\n constructor(\n private readonly filesQueries: FilesQueries,\n private readonly spacesQueries: SpacesQueries,\n private readonly sharesQueries: SharesQueries\n ) {}\n\n async getRecents(user: UserModel, limit: number): Promise<FileRecent[]> {\n const [spaceIds, shareIds] = await Promise.all([this.spacesQueries.spaceIds(user.id), this.sharesQueries.shareIds(user.id, +user.isAdmin)])\n return this.filesQueries.getRecentsFromUser(user.id, spaceIds, shareIds, limit)\n }\n\n async updateRecents(user: UserModel, space: SpaceEnv, files: FileProps[]): Promise<void> {\n const timestamp = currentTimeStamp(null, true) - convertHumanTimeToMs(this.keepTime)\n const location = this.getLocation(user, space, files)\n // only store files, ignore dirs\n const fsRecents = files.filter((f) => !f.isDir && f.size > 0 && f.mtime > timestamp)\n const dbRecents = await this.filesQueries.getRecentsFromLocation(location)\n if (!fsRecents.length && !dbRecents.length) {\n return\n }\n const [add, update, remove] = diffCollection(dbRecents as any, fsRecents as any, ['mtime', 'name'])\n const toAdd: Partial<FileRecent>[] = add.map(\n (f: FileProps): Partial<FileRecent> =>\n ({\n id: f.id,\n name: f.name,\n mtime: f.mtime,\n mime: f.mime,\n ...location,\n ...(space.inSharesList && { shareId: f.root.id })\n }) as FileRecent\n )\n const toUpdate: Record<string | 'object', Partial<FileProps> | FileProps>[] = convertDiffUpdate(update)\n const toRemove: number[] = remove.map((f: FileRecent) => f.id)\n await this.filesQueries.updateRecents(location, toAdd, toUpdate, toRemove)\n }\n\n private getLocation(user: UserModel, space: SpaceEnv, files: FileProps[]): FileRecentLocation {\n const location: FileRecentLocation = { path: space.url }\n if (space.inPersonalSpace) {\n location.ownerId = user.id\n } else if (space.inSharesList) {\n location.shareId = files.map((f) => f.root.id)\n } else if (space.inSharesRepository) {\n location.shareId = space.id\n } else {\n location.spaceId = space.id\n }\n return location\n }\n}\n"],"names":["FilesRecents","getRecents","user","limit","spaceIds","shareIds","Promise","all","spacesQueries","id","sharesQueries","isAdmin","filesQueries","getRecentsFromUser","updateRecents","space","files","timestamp","currentTimeStamp","convertHumanTimeToMs","keepTime","location","getLocation","fsRecents","filter","f","isDir","size","mtime","dbRecents","getRecentsFromLocation","length","add","update","remove","diffCollection","toAdd","map","name","mime","inSharesList","shareId","root","toUpdate","convertDiffUpdate","toRemove","path","url","inPersonalSpace","ownerId","inSharesRepository","spaceId"],"mappings":"AAAA;;;;CAIC;;;;+BAeYA;;;eAAAA;;;wBAbc;2BAC6C;wBACvC;sCACH;sCAEA;qCAKD;;;;;;;;;;AAGtB,IAAA,AAAMA,eAAN,MAAMA;IASX,MAAMC,WAAWC,IAAe,EAAEC,KAAa,EAAyB;QACtE,MAAM,CAACC,UAAUC,SAAS,GAAG,MAAMC,QAAQC,GAAG,CAAC;YAAC,IAAI,CAACC,aAAa,CAACJ,QAAQ,CAACF,KAAKO,EAAE;YAAG,IAAI,CAACC,aAAa,CAACL,QAAQ,CAACH,KAAKO,EAAE,EAAE,CAACP,KAAKS,OAAO;SAAE;QAC1I,OAAO,IAAI,CAACC,YAAY,CAACC,kBAAkB,CAACX,KAAKO,EAAE,EAAEL,UAAUC,UAAUF;IAC3E;IAEA,MAAMW,cAAcZ,IAAe,EAAEa,KAAe,EAAEC,KAAkB,EAAiB;QACvF,MAAMC,YAAYC,IAAAA,wBAAgB,EAAC,MAAM,QAAQC,IAAAA,+BAAoB,EAAC,IAAI,CAACC,QAAQ;QACnF,MAAMC,WAAW,IAAI,CAACC,WAAW,CAACpB,MAAMa,OAAOC;QAC/C,gCAAgC;QAChC,MAAMO,YAAYP,MAAMQ,MAAM,CAAC,CAACC,IAAM,CAACA,EAAEC,KAAK,IAAID,EAAEE,IAAI,GAAG,KAAKF,EAAEG,KAAK,GAAGX;QAC1E,MAAMY,YAAY,MAAM,IAAI,CAACjB,YAAY,CAACkB,sBAAsB,CAACT;QACjE,IAAI,CAACE,UAAUQ,MAAM,IAAI,CAACF,UAAUE,MAAM,EAAE;YAC1C;QACF;QACA,MAAM,CAACC,KAAKC,QAAQC,OAAO,GAAGC,IAAAA,yBAAc,EAACN,WAAkBN,WAAkB;YAAC;YAAS;SAAO;QAClG,MAAMa,QAA+BJ,IAAIK,GAAG,CAC1C,CAACZ,IACE,CAAA;gBACChB,IAAIgB,EAAEhB,EAAE;gBACR6B,MAAMb,EAAEa,IAAI;gBACZV,OAAOH,EAAEG,KAAK;gBACdW,MAAMd,EAAEc,IAAI;gBACZ,GAAGlB,QAAQ;gBACX,GAAIN,MAAMyB,YAAY,IAAI;oBAAEC,SAAShB,EAAEiB,IAAI,CAACjC,EAAE;gBAAC,CAAC;YAClD,CAAA;QAEJ,MAAMkC,WAAwEC,IAAAA,4BAAiB,EAACX;QAChG,MAAMY,WAAqBX,OAAOG,GAAG,CAAC,CAACZ,IAAkBA,EAAEhB,EAAE;QAC7D,MAAM,IAAI,CAACG,YAAY,CAACE,aAAa,CAACO,UAAUe,OAAOO,UAAUE;IACnE;IAEQvB,YAAYpB,IAAe,EAAEa,KAAe,EAAEC,KAAkB,EAAsB;QAC5F,MAAMK,WAA+B;YAAEyB,MAAM/B,MAAMgC,GAAG;QAAC;QACvD,IAAIhC,MAAMiC,eAAe,EAAE;YACzB3B,SAAS4B,OAAO,GAAG/C,KAAKO,EAAE;QAC5B,OAAO,IAAIM,MAAMyB,YAAY,EAAE;YAC7BnB,SAASoB,OAAO,GAAGzB,MAAMqB,GAAG,CAAC,CAACZ,IAAMA,EAAEiB,IAAI,CAACjC,EAAE;QAC/C,OAAO,IAAIM,MAAMmC,kBAAkB,EAAE;YACnC7B,SAASoB,OAAO,GAAG1B,MAAMN,EAAE;QAC7B,OAAO;YACLY,SAAS8B,OAAO,GAAGpC,MAAMN,EAAE;QAC7B;QACA,OAAOY;IACT;IAjDA,YACE,AAAiBT,YAA0B,EAC3C,AAAiBJ,aAA4B,EAC7C,AAAiBE,aAA4B,CAC7C;aAHiBE,eAAAA;aACAJ,gBAAAA;aACAE,gBAAAA;aALFU,WAAW;IAMzB;AA8CL"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-recents.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 { Injectable } from '@nestjs/common'\nimport { convertDiffUpdate, convertHumanTimeToMs, diffCollection } from '../../../common/functions'\nimport { currentTimeStamp } from '../../../common/shared'\nimport { SharesQueries } from '../../shares/services/shares-queries.service'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesQueries } from '../../spaces/services/spaces-queries.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { FileProps } from '../interfaces/file-props.interface'\nimport { FileRecentLocation } from '../interfaces/file-recent-location.interface'\nimport { FileRecent } from '../schemas/file-recent.interface'\nimport { FilesQueries } from './files-queries.service'\n\n@Injectable()\nexport class FilesRecents {\n private readonly keepTime = '14d'\n\n constructor(\n private readonly filesQueries: FilesQueries,\n private readonly spacesQueries: SpacesQueries,\n private readonly sharesQueries: SharesQueries\n ) {}\n\n async getRecents(user: UserModel, limit: number): Promise<FileRecent[]> {\n const [spaceIds, shareIds] = await Promise.all([this.spacesQueries.spaceIds(user.id), this.sharesQueries.shareIds(user.id, +user.isAdmin)])\n return this.filesQueries.getRecentsFromUser(user.id, spaceIds, shareIds, limit)\n }\n\n async updateRecents(user: UserModel, space: SpaceEnv, files: FileProps[]): Promise<void> {\n if (space.inTrashRepository) {\n // Ignore trashed files\n return\n }\n const timestamp = currentTimeStamp(null, true) - convertHumanTimeToMs(this.keepTime)\n const location = this.getLocation(user, space, files)\n // only store files, ignore dirs\n const fsRecents = files.filter((f) => !f.isDir && f.size > 0 && f.mtime > timestamp)\n const dbRecents = await this.filesQueries.getRecentsFromLocation(location)\n if (!fsRecents.length && !dbRecents.length) {\n return\n }\n const [add, update, remove] = diffCollection(dbRecents as any, fsRecents as any, ['mtime', 'name'])\n const toAdd: Partial<FileRecent>[] = add.map(\n (f: FileProps): Partial<FileRecent> =>\n ({\n id: f.id,\n name: f.name,\n mtime: f.mtime,\n mime: f.mime,\n ...location,\n ...(space.inSharesList && { shareId: f.root.id })\n }) as FileRecent\n )\n const toUpdate: Record<string | 'object', Partial<FileProps> | FileProps>[] = convertDiffUpdate(update)\n const toRemove: number[] = remove.map((f: FileRecent) => f.id)\n await this.filesQueries.updateRecents(location, toAdd, toUpdate, toRemove)\n }\n\n private getLocation(user: UserModel, space: SpaceEnv, files: FileProps[]): FileRecentLocation {\n const location: FileRecentLocation = { path: space.url }\n if (space.inPersonalSpace) {\n location.ownerId = user.id\n } else if (space.inSharesList) {\n location.shareId = files.map((f) => f.root.id)\n } else if (space.inSharesRepository) {\n location.shareId = space.id\n } else {\n location.spaceId = space.id\n }\n return location\n }\n}\n"],"names":["FilesRecents","getRecents","user","limit","spaceIds","shareIds","Promise","all","spacesQueries","id","sharesQueries","isAdmin","filesQueries","getRecentsFromUser","updateRecents","space","files","inTrashRepository","timestamp","currentTimeStamp","convertHumanTimeToMs","keepTime","location","getLocation","fsRecents","filter","f","isDir","size","mtime","dbRecents","getRecentsFromLocation","length","add","update","remove","diffCollection","toAdd","map","name","mime","inSharesList","shareId","root","toUpdate","convertDiffUpdate","toRemove","path","url","inPersonalSpace","ownerId","inSharesRepository","spaceId"],"mappings":"AAAA;;;;CAIC;;;;+BAeYA;;;eAAAA;;;wBAbc;2BAC6C;wBACvC;sCACH;sCAEA;qCAKD;;;;;;;;;;AAGtB,IAAA,AAAMA,eAAN,MAAMA;IASX,MAAMC,WAAWC,IAAe,EAAEC,KAAa,EAAyB;QACtE,MAAM,CAACC,UAAUC,SAAS,GAAG,MAAMC,QAAQC,GAAG,CAAC;YAAC,IAAI,CAACC,aAAa,CAACJ,QAAQ,CAACF,KAAKO,EAAE;YAAG,IAAI,CAACC,aAAa,CAACL,QAAQ,CAACH,KAAKO,EAAE,EAAE,CAACP,KAAKS,OAAO;SAAE;QAC1I,OAAO,IAAI,CAACC,YAAY,CAACC,kBAAkB,CAACX,KAAKO,EAAE,EAAEL,UAAUC,UAAUF;IAC3E;IAEA,MAAMW,cAAcZ,IAAe,EAAEa,KAAe,EAAEC,KAAkB,EAAiB;QACvF,IAAID,MAAME,iBAAiB,EAAE;YAC3B,uBAAuB;YACvB;QACF;QACA,MAAMC,YAAYC,IAAAA,wBAAgB,EAAC,MAAM,QAAQC,IAAAA,+BAAoB,EAAC,IAAI,CAACC,QAAQ;QACnF,MAAMC,WAAW,IAAI,CAACC,WAAW,CAACrB,MAAMa,OAAOC;QAC/C,gCAAgC;QAChC,MAAMQ,YAAYR,MAAMS,MAAM,CAAC,CAACC,IAAM,CAACA,EAAEC,KAAK,IAAID,EAAEE,IAAI,GAAG,KAAKF,EAAEG,KAAK,GAAGX;QAC1E,MAAMY,YAAY,MAAM,IAAI,CAAClB,YAAY,CAACmB,sBAAsB,CAACT;QACjE,IAAI,CAACE,UAAUQ,MAAM,IAAI,CAACF,UAAUE,MAAM,EAAE;YAC1C;QACF;QACA,MAAM,CAACC,KAAKC,QAAQC,OAAO,GAAGC,IAAAA,yBAAc,EAACN,WAAkBN,WAAkB;YAAC;YAAS;SAAO;QAClG,MAAMa,QAA+BJ,IAAIK,GAAG,CAC1C,CAACZ,IACE,CAAA;gBACCjB,IAAIiB,EAAEjB,EAAE;gBACR8B,MAAMb,EAAEa,IAAI;gBACZV,OAAOH,EAAEG,KAAK;gBACdW,MAAMd,EAAEc,IAAI;gBACZ,GAAGlB,QAAQ;gBACX,GAAIP,MAAM0B,YAAY,IAAI;oBAAEC,SAAShB,EAAEiB,IAAI,CAAClC,EAAE;gBAAC,CAAC;YAClD,CAAA;QAEJ,MAAMmC,WAAwEC,IAAAA,4BAAiB,EAACX;QAChG,MAAMY,WAAqBX,OAAOG,GAAG,CAAC,CAACZ,IAAkBA,EAAEjB,EAAE;QAC7D,MAAM,IAAI,CAACG,YAAY,CAACE,aAAa,CAACQ,UAAUe,OAAOO,UAAUE;IACnE;IAEQvB,YAAYrB,IAAe,EAAEa,KAAe,EAAEC,KAAkB,EAAsB;QAC5F,MAAMM,WAA+B;YAAEyB,MAAMhC,MAAMiC,GAAG;QAAC;QACvD,IAAIjC,MAAMkC,eAAe,EAAE;YACzB3B,SAAS4B,OAAO,GAAGhD,KAAKO,EAAE;QAC5B,OAAO,IAAIM,MAAM0B,YAAY,EAAE;YAC7BnB,SAASoB,OAAO,GAAG1B,MAAMsB,GAAG,CAAC,CAACZ,IAAMA,EAAEiB,IAAI,CAAClC,EAAE;QAC/C,OAAO,IAAIM,MAAMoC,kBAAkB,EAAE;YACnC7B,SAASoB,OAAO,GAAG3B,MAAMN,EAAE;QAC7B,OAAO;YACLa,SAAS8B,OAAO,GAAGrC,MAAMN,EAAE;QAC7B;QACA,OAAOa;IACT;IArDA,YACE,AAAiBV,YAA0B,EAC3C,AAAiBJ,aAA4B,EAC7C,AAAiBE,aAA4B,CAC7C;aAHiBE,eAAAA;aACAJ,gBAAAA;aACAE,gBAAAA;aALFW,WAAW;IAMzB;AAkDL"}
@@ -53,6 +53,21 @@ function _ts_param(paramIndex, decorator) {
53
53
  };
54
54
  }
55
55
  let FilesScheduler = class FilesScheduler {
56
+ async onStartup() {
57
+ try {
58
+ await this.cleanupInterruptedTasks();
59
+ await this.clearRecentFiles();
60
+ } catch (e) {
61
+ this.logger.error(e);
62
+ }
63
+ }
64
+ async afterStartup() {
65
+ try {
66
+ await this.indexContentFiles();
67
+ } catch (e) {
68
+ this.logger.error(e);
69
+ }
70
+ }
56
71
  async cleanupInterruptedTasks() {
57
72
  this.logger.log(`${this.cleanupInterruptedTasks.name} - START`);
58
73
  try {
@@ -182,11 +197,17 @@ let FilesScheduler = class FilesScheduler {
182
197
  }
183
198
  };
184
199
  _ts_decorate([
185
- (0, _schedule.Timeout)(30000),
200
+ (0, _schedule.Timeout)(30_000),
201
+ _ts_metadata("design:type", Function),
202
+ _ts_metadata("design:paramtypes", []),
203
+ _ts_metadata("design:returntype", Promise)
204
+ ], FilesScheduler.prototype, "onStartup", null);
205
+ _ts_decorate([
206
+ (0, _schedule.Timeout)(180_000),
186
207
  _ts_metadata("design:type", Function),
187
208
  _ts_metadata("design:paramtypes", []),
188
209
  _ts_metadata("design:returntype", Promise)
189
- ], FilesScheduler.prototype, "cleanupInterruptedTasks", null);
210
+ ], FilesScheduler.prototype, "afterStartup", null);
190
211
  _ts_decorate([
191
212
  (0, _schedule.Cron)(_schedule.CronExpression.EVERY_DAY_AT_MIDNIGHT),
192
213
  _ts_metadata("design:type", Function),
@@ -194,14 +215,12 @@ _ts_decorate([
194
215
  _ts_metadata("design:returntype", Promise)
195
216
  ], FilesScheduler.prototype, "cleanupUserTaskFiles", null);
196
217
  _ts_decorate([
197
- (0, _schedule.Timeout)(30000),
198
218
  (0, _schedule.Cron)(_schedule.CronExpression.EVERY_8_HOURS),
199
219
  _ts_metadata("design:type", Function),
200
220
  _ts_metadata("design:paramtypes", []),
201
221
  _ts_metadata("design:returntype", Promise)
202
222
  ], FilesScheduler.prototype, "clearRecentFiles", null);
203
223
  _ts_decorate([
204
- (0, _schedule.Timeout)(120000),
205
224
  (0, _schedule.Cron)(_schedule.CronExpression.EVERY_4_HOURS),
206
225
  _ts_metadata("design:type", Function),
207
226
  _ts_metadata("design:paramtypes", []),