@strapi/upload 0.0.0 → 5.0.0-beta.7

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 (217) hide show
  1. package/README.md +1 -1
  2. package/dist/_chunks/ca-B2_I-q1t.mjs +121 -0
  3. package/dist/_chunks/ca-B2_I-q1t.mjs.map +1 -0
  4. package/dist/_chunks/ca-BUpuZx8N.js +121 -0
  5. package/dist/_chunks/ca-BUpuZx8N.js.map +1 -0
  6. package/dist/_chunks/de-A7mEKx6c.mjs +107 -0
  7. package/dist/_chunks/de-A7mEKx6c.mjs.map +1 -0
  8. package/dist/_chunks/de-uGb_Pkq7.js +107 -0
  9. package/dist/_chunks/de-uGb_Pkq7.js.map +1 -0
  10. package/dist/_chunks/dk-BPfkJb9q.mjs +103 -0
  11. package/dist/_chunks/dk-BPfkJb9q.mjs.map +1 -0
  12. package/dist/_chunks/dk-Cd8oFO-O.js +103 -0
  13. package/dist/_chunks/dk-Cd8oFO-O.js.map +1 -0
  14. package/dist/_chunks/en-BcOqhiNe.js +144 -0
  15. package/dist/_chunks/en-BcOqhiNe.js.map +1 -0
  16. package/dist/_chunks/en-oDx2Gnre.mjs +144 -0
  17. package/dist/_chunks/en-oDx2Gnre.mjs.map +1 -0
  18. package/dist/_chunks/es-CuWi2pOn.mjs +120 -0
  19. package/dist/_chunks/es-CuWi2pOn.mjs.map +1 -0
  20. package/dist/_chunks/es-DWFtw_h4.js +120 -0
  21. package/dist/_chunks/es-DWFtw_h4.js.map +1 -0
  22. package/dist/_chunks/fr-BN6ndmWf.mjs +144 -0
  23. package/dist/_chunks/fr-BN6ndmWf.mjs.map +1 -0
  24. package/dist/_chunks/fr-D2bop66d.js +144 -0
  25. package/dist/_chunks/fr-D2bop66d.js.map +1 -0
  26. package/dist/_chunks/graphql-BVCZSNa2.mjs +71 -0
  27. package/dist/_chunks/graphql-BVCZSNa2.mjs.map +1 -0
  28. package/dist/_chunks/graphql-Beol7Y3b.js +71 -0
  29. package/dist/_chunks/graphql-Beol7Y3b.js.map +1 -0
  30. package/dist/_chunks/he-BpxHjaZg.js +76 -0
  31. package/dist/_chunks/he-BpxHjaZg.js.map +1 -0
  32. package/dist/_chunks/he-C9ZOXBB-.mjs +76 -0
  33. package/dist/_chunks/he-C9ZOXBB-.mjs.map +1 -0
  34. package/dist/_chunks/index-2C3ZYD4e.mjs +856 -0
  35. package/dist/_chunks/index-2C3ZYD4e.mjs.map +1 -0
  36. package/dist/_chunks/index-ASLx4V_D.mjs +6548 -0
  37. package/dist/_chunks/index-ASLx4V_D.mjs.map +1 -0
  38. package/dist/_chunks/index-BbFs7_Gx.js +859 -0
  39. package/dist/_chunks/index-BbFs7_Gx.js.map +1 -0
  40. package/dist/_chunks/index-BxUjcUMp.js +266 -0
  41. package/dist/_chunks/index-BxUjcUMp.js.map +1 -0
  42. package/dist/_chunks/index-CRhNpXFr.js +2632 -0
  43. package/dist/_chunks/index-CRhNpXFr.js.map +1 -0
  44. package/dist/_chunks/index-DJC1L27M.js +238 -0
  45. package/dist/_chunks/index-DJC1L27M.js.map +1 -0
  46. package/dist/_chunks/index-PKgX3DYk.js +6574 -0
  47. package/dist/_chunks/index-PKgX3DYk.js.map +1 -0
  48. package/dist/_chunks/index-VKvfSVC7.mjs +2622 -0
  49. package/dist/_chunks/index-VKvfSVC7.mjs.map +1 -0
  50. package/dist/_chunks/index-fmt8afwE.mjs +262 -0
  51. package/dist/_chunks/index-fmt8afwE.mjs.map +1 -0
  52. package/dist/_chunks/index-q-dRjRbA.mjs +233 -0
  53. package/dist/_chunks/index-q-dRjRbA.mjs.map +1 -0
  54. package/dist/_chunks/it-B7rmoZNx.mjs +76 -0
  55. package/dist/_chunks/it-B7rmoZNx.mjs.map +1 -0
  56. package/dist/_chunks/it-BKCWXl8t.js +76 -0
  57. package/dist/_chunks/it-BKCWXl8t.js.map +1 -0
  58. package/dist/_chunks/ja-DlaJTi_3.mjs +76 -0
  59. package/dist/_chunks/ja-DlaJTi_3.mjs.map +1 -0
  60. package/dist/_chunks/ja-ajHzIJz6.js +76 -0
  61. package/dist/_chunks/ja-ajHzIJz6.js.map +1 -0
  62. package/dist/_chunks/ko-Pzj-818C.js +106 -0
  63. package/dist/_chunks/ko-Pzj-818C.js.map +1 -0
  64. package/dist/_chunks/ko-vJl9kPpn.mjs +106 -0
  65. package/dist/_chunks/ko-vJl9kPpn.mjs.map +1 -0
  66. package/dist/_chunks/ms-CqwG8v8l.mjs +68 -0
  67. package/dist/_chunks/ms-CqwG8v8l.mjs.map +1 -0
  68. package/dist/_chunks/ms-h3gjldBy.js +68 -0
  69. package/dist/_chunks/ms-h3gjldBy.js.map +1 -0
  70. package/dist/_chunks/pl-Cj8jChOO.mjs +105 -0
  71. package/dist/_chunks/pl-Cj8jChOO.mjs.map +1 -0
  72. package/dist/_chunks/pl-esgZ7ltN.js +105 -0
  73. package/dist/_chunks/pl-esgZ7ltN.js.map +1 -0
  74. package/dist/_chunks/pt-BR-B4LJHJIp.mjs +76 -0
  75. package/dist/_chunks/pt-BR-B4LJHJIp.mjs.map +1 -0
  76. package/dist/_chunks/pt-BR-Cazr7Z5I.js +76 -0
  77. package/dist/_chunks/pt-BR-Cazr7Z5I.js.map +1 -0
  78. package/dist/_chunks/pt-CNOOM_7x.mjs +76 -0
  79. package/dist/_chunks/pt-CNOOM_7x.mjs.map +1 -0
  80. package/dist/_chunks/pt-cbUnkHM5.js +76 -0
  81. package/dist/_chunks/pt-cbUnkHM5.js.map +1 -0
  82. package/dist/_chunks/ru-DqglvSUC.mjs +76 -0
  83. package/dist/_chunks/ru-DqglvSUC.mjs.map +1 -0
  84. package/dist/_chunks/ru-H6MzFUxp.js +76 -0
  85. package/dist/_chunks/ru-H6MzFUxp.js.map +1 -0
  86. package/dist/_chunks/sk-CZxC4dFY.js +122 -0
  87. package/dist/_chunks/sk-CZxC4dFY.js.map +1 -0
  88. package/dist/_chunks/sk-Dgpb3lnz.mjs +122 -0
  89. package/dist/_chunks/sk-Dgpb3lnz.mjs.map +1 -0
  90. package/dist/_chunks/th-C6unJZ8j.js +76 -0
  91. package/dist/_chunks/th-C6unJZ8j.js.map +1 -0
  92. package/dist/_chunks/th-DRfzuiFf.mjs +76 -0
  93. package/dist/_chunks/th-DRfzuiFf.mjs.map +1 -0
  94. package/dist/_chunks/tr--GzWXE_A.mjs +122 -0
  95. package/dist/_chunks/tr--GzWXE_A.mjs.map +1 -0
  96. package/dist/_chunks/tr-CY6AwX50.js +122 -0
  97. package/dist/_chunks/tr-CY6AwX50.js.map +1 -0
  98. package/dist/_chunks/uk-BniyNsD4.js +73 -0
  99. package/dist/_chunks/uk-BniyNsD4.js.map +1 -0
  100. package/dist/_chunks/uk-DVMT2Piq.mjs +73 -0
  101. package/dist/_chunks/uk-DVMT2Piq.mjs.map +1 -0
  102. package/dist/_chunks/zh-CsZw0IpM.js +129 -0
  103. package/dist/_chunks/zh-CsZw0IpM.js.map +1 -0
  104. package/dist/_chunks/zh-HOnih0is.mjs +129 -0
  105. package/dist/_chunks/zh-HOnih0is.mjs.map +1 -0
  106. package/dist/_chunks/zh-Hans-Cpmhg8uH.mjs +152 -0
  107. package/dist/_chunks/zh-Hans-Cpmhg8uH.mjs.map +1 -0
  108. package/dist/_chunks/zh-Hans-k_xAc6nm.js +152 -0
  109. package/dist/_chunks/zh-Hans-k_xAc6nm.js.map +1 -0
  110. package/dist/admin/index.js +5 -0
  111. package/dist/admin/index.js.map +1 -0
  112. package/dist/admin/index.mjs +6 -0
  113. package/dist/admin/index.mjs.map +1 -0
  114. package/dist/server/index.js +4 -0
  115. package/dist/server/index.js.map +1 -0
  116. package/dist/server/index.mjs +5 -0
  117. package/dist/server/index.mjs.map +1 -0
  118. package/dist/server/src/bootstrap.d.ts +5 -0
  119. package/dist/server/src/bootstrap.d.ts.map +1 -0
  120. package/dist/server/src/config.d.ts +10 -0
  121. package/dist/server/src/config.d.ts.map +1 -0
  122. package/dist/server/src/constants.d.ts +20 -0
  123. package/dist/server/src/constants.d.ts.map +1 -0
  124. package/dist/server/src/content-types/file.d.ts +110 -0
  125. package/dist/server/src/content-types/file.d.ts.map +1 -0
  126. package/dist/server/src/content-types/folder.d.ts +61 -0
  127. package/dist/server/src/content-types/folder.d.ts.map +1 -0
  128. package/dist/server/src/content-types/index.d.ts +170 -0
  129. package/dist/server/src/content-types/index.d.ts.map +1 -0
  130. package/dist/server/src/controllers/admin-file.d.ts +16 -0
  131. package/dist/server/src/controllers/admin-file.d.ts.map +1 -0
  132. package/dist/server/src/controllers/admin-folder-file.d.ts +7 -0
  133. package/dist/server/src/controllers/admin-folder-file.d.ts.map +1 -0
  134. package/dist/server/src/controllers/admin-folder.d.ts +10 -0
  135. package/dist/server/src/controllers/admin-folder.d.ts.map +1 -0
  136. package/dist/server/src/controllers/admin-settings.d.ts +7 -0
  137. package/dist/server/src/controllers/admin-settings.d.ts.map +1 -0
  138. package/dist/server/src/controllers/admin-upload.d.ts +9 -0
  139. package/dist/server/src/controllers/admin-upload.d.ts.map +1 -0
  140. package/dist/server/src/controllers/content-api.d.ts +15 -0
  141. package/dist/server/src/controllers/content-api.d.ts.map +1 -0
  142. package/dist/server/src/controllers/index.d.ts +53 -0
  143. package/dist/server/src/controllers/index.d.ts.map +1 -0
  144. package/dist/server/src/controllers/utils/find-entity-and-check-permissions.d.ts +6 -0
  145. package/dist/server/src/controllers/utils/find-entity-and-check-permissions.d.ts.map +1 -0
  146. package/dist/server/src/controllers/utils/folders.d.ts +4 -0
  147. package/dist/server/src/controllers/utils/folders.d.ts.map +1 -0
  148. package/dist/server/src/controllers/validation/admin/configureView.d.ts +15 -0
  149. package/dist/server/src/controllers/validation/admin/configureView.d.ts.map +1 -0
  150. package/dist/server/src/controllers/validation/admin/folder-file.d.ts +3 -0
  151. package/dist/server/src/controllers/validation/admin/folder-file.d.ts.map +1 -0
  152. package/dist/server/src/controllers/validation/admin/folder.d.ts +3 -0
  153. package/dist/server/src/controllers/validation/admin/folder.d.ts.map +1 -0
  154. package/dist/server/src/controllers/validation/admin/settings.d.ts +18 -0
  155. package/dist/server/src/controllers/validation/admin/settings.d.ts.map +1 -0
  156. package/dist/server/src/controllers/validation/admin/upload.d.ts +109 -0
  157. package/dist/server/src/controllers/validation/admin/upload.d.ts.map +1 -0
  158. package/dist/server/src/controllers/validation/admin/utils.d.ts +3 -0
  159. package/dist/server/src/controllers/validation/admin/utils.d.ts.map +1 -0
  160. package/dist/server/src/controllers/validation/content-api/upload.d.ts +91 -0
  161. package/dist/server/src/controllers/validation/content-api/upload.d.ts.map +1 -0
  162. package/dist/server/src/controllers/view-configuration.d.ts +7 -0
  163. package/dist/server/src/controllers/view-configuration.d.ts.map +1 -0
  164. package/dist/server/src/graphql.d.ts +5 -0
  165. package/dist/server/src/graphql.d.ts.map +1 -0
  166. package/dist/server/src/index.d.ts +475 -0
  167. package/dist/server/src/index.d.ts.map +1 -0
  168. package/dist/server/src/middlewares/upload.d.ts +9 -0
  169. package/dist/server/src/middlewares/upload.d.ts.map +1 -0
  170. package/dist/server/src/register.d.ts +8 -0
  171. package/dist/server/src/register.d.ts.map +1 -0
  172. package/dist/server/src/routes/admin.d.ts +17 -0
  173. package/dist/server/src/routes/admin.d.ts.map +1 -0
  174. package/dist/server/src/routes/content-api.d.ts +9 -0
  175. package/dist/server/src/routes/content-api.d.ts.map +1 -0
  176. package/dist/server/src/routes/index.d.ts +43 -0
  177. package/dist/server/src/routes/index.d.ts.map +1 -0
  178. package/dist/server/src/routes/view-configuration.d.ts +17 -0
  179. package/dist/server/src/routes/view-configuration.d.ts.map +1 -0
  180. package/dist/server/src/services/api-upload-folder.d.ts +5 -0
  181. package/dist/server/src/services/api-upload-folder.d.ts.map +1 -0
  182. package/dist/server/src/services/extensions/content-manager/entity-manager.d.ts +15 -0
  183. package/dist/server/src/services/extensions/content-manager/entity-manager.d.ts.map +1 -0
  184. package/dist/server/src/services/extensions/content-manager/index.d.ts +4 -0
  185. package/dist/server/src/services/extensions/content-manager/index.d.ts.map +1 -0
  186. package/dist/server/src/services/extensions/core/entity-service.d.ts +4 -0
  187. package/dist/server/src/services/extensions/core/entity-service.d.ts.map +1 -0
  188. package/dist/server/src/services/extensions/core/index.d.ts +4 -0
  189. package/dist/server/src/services/extensions/core/index.d.ts.map +1 -0
  190. package/dist/server/src/services/extensions/index.d.ts +8 -0
  191. package/dist/server/src/services/extensions/index.d.ts.map +1 -0
  192. package/dist/server/src/services/extensions/utils.d.ts +14 -0
  193. package/dist/server/src/services/extensions/utils.d.ts.map +1 -0
  194. package/dist/server/src/services/file.d.ts +8 -0
  195. package/dist/server/src/services/file.d.ts.map +1 -0
  196. package/dist/server/src/services/folder.d.ts +32 -0
  197. package/dist/server/src/services/folder.d.ts.map +1 -0
  198. package/dist/server/src/services/image-manipulation.d.ts +63 -0
  199. package/dist/server/src/services/image-manipulation.d.ts.map +1 -0
  200. package/dist/server/src/services/index.d.ts +196 -0
  201. package/dist/server/src/services/index.d.ts.map +1 -0
  202. package/dist/server/src/services/metrics.d.ts +8 -0
  203. package/dist/server/src/services/metrics.d.ts.map +1 -0
  204. package/dist/server/src/services/provider.d.ts +10 -0
  205. package/dist/server/src/services/provider.d.ts.map +1 -0
  206. package/dist/server/src/services/upload.d.ts +67 -0
  207. package/dist/server/src/services/upload.d.ts.map +1 -0
  208. package/dist/server/src/services/weekly-metrics.d.ts +17 -0
  209. package/dist/server/src/services/weekly-metrics.d.ts.map +1 -0
  210. package/dist/server/src/types.d.ts +74 -0
  211. package/dist/server/src/types.d.ts.map +1 -0
  212. package/dist/server/src/utils/cron.d.ts +3 -0
  213. package/dist/server/src/utils/cron.d.ts.map +1 -0
  214. package/dist/server/src/utils/index.d.ts +23 -0
  215. package/dist/server/src/utils/index.d.ts.map +1 -0
  216. package/package.json +99 -7
  217. package/strapi-server.js +3 -0
@@ -0,0 +1,2632 @@
1
+ "use strict";
2
+ const _ = require("lodash");
3
+ const utils = require("@strapi/utils");
4
+ const range = require("koa-range");
5
+ const koaStatic = require("koa-static");
6
+ const fp = require("lodash/fp");
7
+ const os = require("os");
8
+ const path = require("path");
9
+ const crypto = require("crypto");
10
+ const fs = require("fs");
11
+ const fse = require("fs-extra");
12
+ const mimeTypes = require("mime-types");
13
+ const sharp = require("sharp");
14
+ const dateFns = require("date-fns");
15
+ const _interopDefault = (e) => e && e.__esModule ? e : { default: e };
16
+ const ___default = /* @__PURE__ */ _interopDefault(_);
17
+ const utils__default = /* @__PURE__ */ _interopDefault(utils);
18
+ const range__default = /* @__PURE__ */ _interopDefault(range);
19
+ const koaStatic__default = /* @__PURE__ */ _interopDefault(koaStatic);
20
+ const os__default = /* @__PURE__ */ _interopDefault(os);
21
+ const path__default = /* @__PURE__ */ _interopDefault(path);
22
+ const crypto__default = /* @__PURE__ */ _interopDefault(crypto);
23
+ const fs__default = /* @__PURE__ */ _interopDefault(fs);
24
+ const fse__default = /* @__PURE__ */ _interopDefault(fse);
25
+ const sharp__default = /* @__PURE__ */ _interopDefault(sharp);
26
+ const registerUploadMiddleware = ({ strapi: strapi2 }) => {
27
+ strapi2.server.app.on("error", (err) => {
28
+ if (err.code === "EPIPE") {
29
+ return;
30
+ }
31
+ strapi2.server.app.onerror(err);
32
+ });
33
+ const localServerConfig = strapi2.config.get("plugin::upload.providerOptions.localServer", {});
34
+ strapi2.server.routes([
35
+ {
36
+ method: "GET",
37
+ path: "/uploads/(.*)",
38
+ handler: [range__default.default, koaStatic__default.default(strapi2.dirs.static.public, { defer: true, ...localServerConfig })],
39
+ config: { auth: false }
40
+ }
41
+ ]);
42
+ };
43
+ const paths = {
44
+ "/upload": {
45
+ post: {
46
+ description: "Upload files",
47
+ responses: {
48
+ "200": {
49
+ description: "response",
50
+ content: {
51
+ "application/json": {
52
+ schema: {
53
+ type: "array",
54
+ items: {
55
+ $ref: "#/components/schemas/UploadFile"
56
+ }
57
+ }
58
+ }
59
+ }
60
+ }
61
+ },
62
+ summary: "",
63
+ tags: [
64
+ "Upload - File"
65
+ ],
66
+ requestBody: {
67
+ description: "Upload files",
68
+ required: true,
69
+ content: {
70
+ "multipart/form-data": {
71
+ schema: {
72
+ required: [
73
+ "files"
74
+ ],
75
+ type: "object",
76
+ properties: {
77
+ path: {
78
+ type: "string",
79
+ description: "The folder where the file(s) will be uploaded to (only supported on strapi-provider-upload-aws-s3)."
80
+ },
81
+ refId: {
82
+ type: "string",
83
+ description: "The ID of the entry which the file(s) will be linked to"
84
+ },
85
+ ref: {
86
+ type: "string",
87
+ description: "The unique ID (uid) of the model which the file(s) will be linked to (api::restaurant.restaurant)."
88
+ },
89
+ field: {
90
+ type: "string",
91
+ description: "The field of the entry which the file(s) will be precisely linked to."
92
+ },
93
+ files: {
94
+ type: "array",
95
+ items: {
96
+ type: "string",
97
+ format: "binary"
98
+ }
99
+ }
100
+ }
101
+ }
102
+ }
103
+ }
104
+ }
105
+ }
106
+ },
107
+ "/upload?id={id}": {
108
+ post: {
109
+ parameters: [
110
+ {
111
+ name: "id",
112
+ "in": "query",
113
+ description: "File id",
114
+ required: true,
115
+ schema: {
116
+ type: "string"
117
+ }
118
+ }
119
+ ],
120
+ description: "Upload file information",
121
+ responses: {
122
+ "200": {
123
+ description: "response",
124
+ content: {
125
+ "application/json": {
126
+ schema: {
127
+ type: "array",
128
+ items: {
129
+ $ref: "#/components/schemas/UploadFile"
130
+ }
131
+ }
132
+ }
133
+ }
134
+ }
135
+ },
136
+ summary: "",
137
+ tags: [
138
+ "Upload - File"
139
+ ],
140
+ requestBody: {
141
+ description: "Upload files",
142
+ required: true,
143
+ content: {
144
+ "multipart/form-data": {
145
+ schema: {
146
+ type: "object",
147
+ properties: {
148
+ fileInfo: {
149
+ type: "object",
150
+ properties: {
151
+ name: {
152
+ type: "string"
153
+ },
154
+ alternativeText: {
155
+ type: "string"
156
+ },
157
+ caption: {
158
+ type: "string"
159
+ }
160
+ }
161
+ },
162
+ files: {
163
+ type: "string",
164
+ format: "binary"
165
+ }
166
+ }
167
+ }
168
+ }
169
+ }
170
+ }
171
+ }
172
+ },
173
+ "/upload/files": {
174
+ get: {
175
+ tags: [
176
+ "Upload - File"
177
+ ],
178
+ responses: {
179
+ "200": {
180
+ description: "Get a list of files",
181
+ content: {
182
+ "application/json": {
183
+ schema: {
184
+ type: "array",
185
+ items: {
186
+ $ref: "#/components/schemas/UploadFile"
187
+ }
188
+ }
189
+ }
190
+ }
191
+ }
192
+ }
193
+ }
194
+ },
195
+ "/upload/files/{id}": {
196
+ get: {
197
+ parameters: [
198
+ {
199
+ name: "id",
200
+ "in": "path",
201
+ description: "",
202
+ deprecated: false,
203
+ required: true,
204
+ schema: {
205
+ type: "string"
206
+ }
207
+ }
208
+ ],
209
+ tags: [
210
+ "Upload - File"
211
+ ],
212
+ responses: {
213
+ "200": {
214
+ description: "Get a specific file",
215
+ content: {
216
+ "application/json": {
217
+ schema: {
218
+ $ref: "#/components/schemas/UploadFile"
219
+ }
220
+ }
221
+ }
222
+ }
223
+ }
224
+ },
225
+ "delete": {
226
+ parameters: [
227
+ {
228
+ name: "id",
229
+ "in": "path",
230
+ description: "",
231
+ deprecated: false,
232
+ required: true,
233
+ schema: {
234
+ type: "string"
235
+ }
236
+ }
237
+ ],
238
+ tags: [
239
+ "Upload - File"
240
+ ],
241
+ responses: {
242
+ "200": {
243
+ description: "Delete a file",
244
+ content: {
245
+ "application/json": {
246
+ schema: {
247
+ $ref: "#/components/schemas/UploadFile"
248
+ }
249
+ }
250
+ }
251
+ }
252
+ }
253
+ }
254
+ }
255
+ };
256
+ const components = {
257
+ schemas: {
258
+ UploadFile: {
259
+ properties: {
260
+ id: {
261
+ type: "number"
262
+ },
263
+ name: {
264
+ type: "string"
265
+ },
266
+ alternativeText: {
267
+ type: "string"
268
+ },
269
+ caption: {
270
+ type: "string"
271
+ },
272
+ width: {
273
+ type: "number",
274
+ format: "integer"
275
+ },
276
+ height: {
277
+ type: "number",
278
+ format: "integer"
279
+ },
280
+ formats: {
281
+ type: "number"
282
+ },
283
+ hash: {
284
+ type: "string"
285
+ },
286
+ ext: {
287
+ type: "string"
288
+ },
289
+ mime: {
290
+ type: "string"
291
+ },
292
+ size: {
293
+ type: "number",
294
+ format: "double"
295
+ },
296
+ url: {
297
+ type: "string"
298
+ },
299
+ previewUrl: {
300
+ type: "string"
301
+ },
302
+ provider: {
303
+ type: "string"
304
+ },
305
+ provider_metadata: {
306
+ type: "object"
307
+ },
308
+ createdAt: {
309
+ type: "string",
310
+ format: "date-time"
311
+ },
312
+ updatedAt: {
313
+ type: "string",
314
+ format: "date-time"
315
+ }
316
+ }
317
+ }
318
+ }
319
+ };
320
+ const spec = {
321
+ paths,
322
+ components
323
+ };
324
+ const { PayloadTooLargeError } = utils.errors;
325
+ const { bytesToHumanReadable, kbytesToBytes } = utils.file;
326
+ async function register({ strapi: strapi2 }) {
327
+ strapi2.plugin("upload").provider = createProvider(strapi2.config.get("plugin::upload"));
328
+ await registerUploadMiddleware({ strapi: strapi2 });
329
+ if (strapi2.plugin("graphql")) {
330
+ const { installGraphqlExtension } = await Promise.resolve().then(() => require("./graphql-Beol7Y3b.js"));
331
+ installGraphqlExtension({ strapi: strapi2 });
332
+ }
333
+ if (strapi2.plugin("documentation")) {
334
+ strapi2.plugin("documentation").service("override").registerOverride(spec, {
335
+ pluginOrigin: "upload",
336
+ excludeFromGeneration: ["upload"]
337
+ });
338
+ }
339
+ }
340
+ const createProvider = (config2) => {
341
+ const { providerOptions, actionOptions = {} } = config2;
342
+ const providerName = ___default.default.toLower(config2.provider);
343
+ let provider2;
344
+ let modulePath;
345
+ try {
346
+ modulePath = require.resolve(`@strapi/provider-upload-${providerName}`);
347
+ } catch (error) {
348
+ if (typeof error === "object" && error !== null && "code" in error && error.code === "MODULE_NOT_FOUND") {
349
+ modulePath = providerName;
350
+ } else {
351
+ throw error;
352
+ }
353
+ }
354
+ try {
355
+ provider2 = require(modulePath);
356
+ } catch (err) {
357
+ const newError = new Error(`Could not load upload provider "${providerName}".`);
358
+ if (err instanceof Error) {
359
+ newError.stack = err.stack;
360
+ }
361
+ throw newError;
362
+ }
363
+ const providerInstance = provider2.init(providerOptions);
364
+ if (!providerInstance.delete) {
365
+ throw new Error(`The upload provider "${providerName}" doesn't implement the delete method.`);
366
+ }
367
+ if (!providerInstance.upload && !providerInstance.uploadStream) {
368
+ throw new Error(
369
+ `The upload provider "${providerName}" doesn't implement the uploadStream nor the upload method.`
370
+ );
371
+ }
372
+ if (!providerInstance.uploadStream) {
373
+ process.emitWarning(
374
+ `The upload provider "${providerName}" doesn't implement the uploadStream function. Strapi will fallback on the upload method. Some performance issues may occur.`
375
+ );
376
+ }
377
+ const wrappedProvider = ___default.default.mapValues(providerInstance, (method, methodName) => {
378
+ return async (file2, options = actionOptions[methodName]) => providerInstance[methodName](file2, options);
379
+ });
380
+ return Object.assign(Object.create(baseProvider), wrappedProvider);
381
+ };
382
+ const baseProvider = {
383
+ extend(obj) {
384
+ Object.assign(this, obj);
385
+ },
386
+ checkFileSize(file2, { sizeLimit }) {
387
+ if (sizeLimit && kbytesToBytes(file2.size) > sizeLimit) {
388
+ throw new PayloadTooLargeError(
389
+ `${file2.originalFilename} exceeds size limit of ${bytesToHumanReadable(sizeLimit)}.`
390
+ );
391
+ }
392
+ },
393
+ getSignedUrl(file2) {
394
+ return file2;
395
+ },
396
+ isPrivate() {
397
+ return false;
398
+ }
399
+ };
400
+ const getService = (name) => {
401
+ return strapi.plugin("upload").service(name);
402
+ };
403
+ const ACTIONS = {
404
+ read: "plugin::upload.read",
405
+ readSettings: "plugin::upload.settings.read",
406
+ create: "plugin::upload.assets.create",
407
+ update: "plugin::upload.assets.update",
408
+ download: "plugin::upload.assets.download",
409
+ copyLink: "plugin::upload.assets.copy-link",
410
+ configureView: "plugin::upload.configure-view"
411
+ };
412
+ const ALLOWED_SORT_STRINGS = [
413
+ "createdAt:DESC",
414
+ "createdAt:ASC",
415
+ "name:ASC",
416
+ "name:DESC",
417
+ "updatedAt:DESC",
418
+ "updatedAt:ASC"
419
+ ];
420
+ const ALLOWED_WEBHOOK_EVENTS = {
421
+ MEDIA_CREATE: "media.create",
422
+ MEDIA_UPDATE: "media.update",
423
+ MEDIA_DELETE: "media.delete"
424
+ };
425
+ const FOLDER_MODEL_UID = "plugin::upload.folder";
426
+ const FILE_MODEL_UID = "plugin::upload.file";
427
+ const API_UPLOAD_FOLDER_BASE_NAME = "API Uploads";
428
+ async function bootstrap({ strapi: strapi2 }) {
429
+ const defaultConfig = {
430
+ settings: {
431
+ sizeOptimization: true,
432
+ responsiveDimensions: true,
433
+ autoOrientation: false
434
+ },
435
+ view_configuration: {
436
+ pageSize: 10,
437
+ sort: ALLOWED_SORT_STRINGS[0]
438
+ }
439
+ };
440
+ for (const [key, defaultValue] of Object.entries(defaultConfig)) {
441
+ const configurator = strapi2.store({ type: "plugin", name: "upload", key });
442
+ const config2 = await configurator.get({});
443
+ if (config2 && Object.keys(defaultValue).every((key2) => Object.prototype.hasOwnProperty.call(config2, key2))) {
444
+ continue;
445
+ }
446
+ await configurator.set({
447
+ value: Object.assign(defaultValue, config2 || {})
448
+ });
449
+ }
450
+ await registerPermissionActions();
451
+ await registerWebhookEvents();
452
+ await getService("weeklyMetrics").registerCron();
453
+ getService("metrics").sendUploadPluginMetrics();
454
+ if (strapi2.config.get("plugin::upload.signAdminURLsOnly", false)) {
455
+ getService("extensions").contentManager.entityManager.addSignedFileUrlsToAdmin();
456
+ } else {
457
+ getService("extensions").core.entityService.addSignedFileUrlsToEntityService();
458
+ }
459
+ }
460
+ const registerWebhookEvents = async () => Object.entries(ALLOWED_WEBHOOK_EVENTS).forEach(([key, value]) => {
461
+ strapi.get("webhookStore").addAllowedEvent(key, value);
462
+ });
463
+ const registerPermissionActions = async () => {
464
+ const actions = [
465
+ {
466
+ section: "plugins",
467
+ displayName: "Access the Media Library",
468
+ uid: "read",
469
+ pluginName: "upload"
470
+ },
471
+ {
472
+ section: "plugins",
473
+ displayName: "Create (upload)",
474
+ uid: "assets.create",
475
+ subCategory: "assets",
476
+ pluginName: "upload"
477
+ },
478
+ {
479
+ section: "plugins",
480
+ displayName: "Update (crop, details, replace) + delete",
481
+ uid: "assets.update",
482
+ subCategory: "assets",
483
+ pluginName: "upload"
484
+ },
485
+ {
486
+ section: "plugins",
487
+ displayName: "Download",
488
+ uid: "assets.download",
489
+ subCategory: "assets",
490
+ pluginName: "upload"
491
+ },
492
+ {
493
+ section: "plugins",
494
+ displayName: "Copy link",
495
+ uid: "assets.copy-link",
496
+ subCategory: "assets",
497
+ pluginName: "upload"
498
+ },
499
+ {
500
+ section: "plugins",
501
+ displayName: "Configure view",
502
+ uid: "configure-view",
503
+ pluginName: "upload"
504
+ },
505
+ {
506
+ section: "settings",
507
+ displayName: "Access the Media Library settings page",
508
+ uid: "settings.read",
509
+ category: "media library",
510
+ pluginName: "upload"
511
+ }
512
+ ];
513
+ await strapi.service("admin::permission").actionProvider.registerMany(actions);
514
+ };
515
+ const file$1 = {
516
+ schema: {
517
+ collectionName: "files",
518
+ info: {
519
+ singularName: "file",
520
+ pluralName: "files",
521
+ displayName: "File",
522
+ description: ""
523
+ },
524
+ options: {},
525
+ pluginOptions: {
526
+ "content-manager": {
527
+ visible: false
528
+ },
529
+ "content-type-builder": {
530
+ visible: false
531
+ }
532
+ },
533
+ attributes: {
534
+ name: {
535
+ type: "string",
536
+ configurable: false,
537
+ required: true
538
+ },
539
+ alternativeText: {
540
+ type: "string",
541
+ configurable: false
542
+ },
543
+ caption: {
544
+ type: "string",
545
+ configurable: false
546
+ },
547
+ width: {
548
+ type: "integer",
549
+ configurable: false
550
+ },
551
+ height: {
552
+ type: "integer",
553
+ configurable: false
554
+ },
555
+ formats: {
556
+ type: "json",
557
+ configurable: false
558
+ },
559
+ hash: {
560
+ type: "string",
561
+ configurable: false,
562
+ required: true
563
+ },
564
+ ext: {
565
+ type: "string",
566
+ configurable: false
567
+ },
568
+ mime: {
569
+ type: "string",
570
+ configurable: false,
571
+ required: true
572
+ },
573
+ size: {
574
+ type: "decimal",
575
+ configurable: false,
576
+ required: true
577
+ },
578
+ url: {
579
+ type: "string",
580
+ configurable: false,
581
+ required: true
582
+ },
583
+ previewUrl: {
584
+ type: "string",
585
+ configurable: false
586
+ },
587
+ provider: {
588
+ type: "string",
589
+ configurable: false,
590
+ required: true
591
+ },
592
+ provider_metadata: {
593
+ type: "json",
594
+ configurable: false
595
+ },
596
+ related: {
597
+ type: "relation",
598
+ relation: "morphToMany",
599
+ configurable: false
600
+ },
601
+ folder: {
602
+ type: "relation",
603
+ relation: "manyToOne",
604
+ target: FOLDER_MODEL_UID,
605
+ inversedBy: "files",
606
+ private: true
607
+ },
608
+ folderPath: {
609
+ type: "string",
610
+ minLength: 1,
611
+ required: true,
612
+ private: true,
613
+ searchable: false
614
+ }
615
+ },
616
+ // experimental feature:
617
+ indexes: [
618
+ {
619
+ name: "upload_files_folder_path_index",
620
+ columns: ["folder_path"],
621
+ type: null
622
+ },
623
+ {
624
+ name: `upload_files_created_at_index`,
625
+ columns: ["created_at"],
626
+ type: null
627
+ },
628
+ {
629
+ name: `upload_files_updated_at_index`,
630
+ columns: ["updated_at"],
631
+ type: null
632
+ },
633
+ {
634
+ name: `upload_files_name_index`,
635
+ columns: ["name"],
636
+ type: null
637
+ },
638
+ {
639
+ name: `upload_files_size_index`,
640
+ columns: ["size"],
641
+ type: null
642
+ },
643
+ {
644
+ name: `upload_files_ext_index`,
645
+ columns: ["ext"],
646
+ type: null
647
+ }
648
+ ]
649
+ }
650
+ };
651
+ const folder$1 = {
652
+ schema: {
653
+ collectionName: "upload_folders",
654
+ info: {
655
+ singularName: "folder",
656
+ pluralName: "folders",
657
+ displayName: "Folder"
658
+ },
659
+ options: {},
660
+ pluginOptions: {
661
+ "content-manager": {
662
+ visible: false
663
+ },
664
+ "content-type-builder": {
665
+ visible: false
666
+ }
667
+ },
668
+ attributes: {
669
+ name: {
670
+ type: "string",
671
+ minLength: 1,
672
+ required: true
673
+ },
674
+ pathId: {
675
+ type: "integer",
676
+ unique: true,
677
+ required: true
678
+ },
679
+ parent: {
680
+ type: "relation",
681
+ relation: "manyToOne",
682
+ target: FOLDER_MODEL_UID,
683
+ inversedBy: "children"
684
+ },
685
+ children: {
686
+ type: "relation",
687
+ relation: "oneToMany",
688
+ target: FOLDER_MODEL_UID,
689
+ mappedBy: "parent"
690
+ },
691
+ files: {
692
+ type: "relation",
693
+ relation: "oneToMany",
694
+ target: FILE_MODEL_UID,
695
+ mappedBy: "folder"
696
+ },
697
+ path: {
698
+ type: "string",
699
+ minLength: 1,
700
+ required: true
701
+ }
702
+ },
703
+ // experimental feature:
704
+ indexes: [
705
+ {
706
+ name: "upload_folders_path_id_index",
707
+ columns: ["path_id"],
708
+ type: "unique"
709
+ },
710
+ {
711
+ name: "upload_folders_path_index",
712
+ columns: ["path"],
713
+ type: "unique"
714
+ }
715
+ ]
716
+ }
717
+ };
718
+ const contentTypes = {
719
+ file: file$1,
720
+ folder: folder$1
721
+ };
722
+ const provider = ({ strapi: strapi2 }) => ({
723
+ async checkFileSize(file2) {
724
+ const { sizeLimit } = strapi2.config.get("plugin::upload");
725
+ await strapi2.plugin("upload").provider.checkFileSize(file2, { sizeLimit });
726
+ },
727
+ async upload(file2) {
728
+ if (fp.isFunction(strapi2.plugin("upload").provider.uploadStream)) {
729
+ file2.stream = file2.getStream();
730
+ await strapi2.plugin("upload").provider.uploadStream(file2);
731
+ delete file2.stream;
732
+ } else {
733
+ file2.buffer = await utils.file.streamToBuffer(file2.getStream());
734
+ await strapi2.plugin("upload").provider.upload(file2);
735
+ delete file2.buffer;
736
+ }
737
+ }
738
+ });
739
+ const { UPDATED_BY_ATTRIBUTE, CREATED_BY_ATTRIBUTE } = utils.contentTypes.constants;
740
+ const { MEDIA_CREATE, MEDIA_UPDATE, MEDIA_DELETE } = ALLOWED_WEBHOOK_EVENTS;
741
+ const { ApplicationError, NotFoundError } = utils.errors;
742
+ const { bytesToKbytes: bytesToKbytes$1 } = utils.file;
743
+ const upload = ({ strapi: strapi2 }) => {
744
+ const randomSuffix = () => crypto__default.default.randomBytes(5).toString("hex");
745
+ const generateFileName = (name) => {
746
+ const baseName = utils.strings.nameToSlug(name, { separator: "_", lowercase: false });
747
+ return `${baseName}_${randomSuffix()}`;
748
+ };
749
+ const sendMediaMetrics = (data) => {
750
+ if (___default.default.has(data, "caption") && !___default.default.isEmpty(data.caption)) {
751
+ strapi2.telemetry.send("didSaveMediaWithCaption");
752
+ }
753
+ if (___default.default.has(data, "alternativeText") && !___default.default.isEmpty(data.alternativeText)) {
754
+ strapi2.telemetry.send("didSaveMediaWithAlternativeText");
755
+ }
756
+ };
757
+ const createAndAssignTmpWorkingDirectoryToFiles = async (files) => {
758
+ const tmpWorkingDirectory = await fse__default.default.mkdtemp(path__default.default.join(os__default.default.tmpdir(), "strapi-upload-"));
759
+ if (Array.isArray(files)) {
760
+ files.forEach((file2) => {
761
+ file2.tmpWorkingDirectory = tmpWorkingDirectory;
762
+ });
763
+ } else {
764
+ files.tmpWorkingDirectory = tmpWorkingDirectory;
765
+ }
766
+ return tmpWorkingDirectory;
767
+ };
768
+ function filenameReservedRegex() {
769
+ return /[<>:"/\\|?*\u0000-\u001F]/g;
770
+ }
771
+ function windowsReservedNameRegex() {
772
+ return /^(con|prn|aux|nul|com\d|lpt\d)$/i;
773
+ }
774
+ function isValidFilename(string) {
775
+ if (!string || string.length > 255) {
776
+ return false;
777
+ }
778
+ if (filenameReservedRegex().test(string) || windowsReservedNameRegex().test(string)) {
779
+ return false;
780
+ }
781
+ if (string === "." || string === "..") {
782
+ return false;
783
+ }
784
+ return true;
785
+ }
786
+ async function emitEvent(event, data) {
787
+ const modelDef = strapi2.getModel(FILE_MODEL_UID);
788
+ const sanitizedData = await utils.sanitize.sanitizers.defaultSanitizeOutput(
789
+ {
790
+ schema: modelDef,
791
+ getModel(uid) {
792
+ return strapi2.getModel(uid);
793
+ }
794
+ },
795
+ data
796
+ );
797
+ strapi2.eventHub.emit(event, { media: sanitizedData });
798
+ }
799
+ async function formatFileInfo({ filename, type, size }, fileInfo = {}, metas = {}) {
800
+ const fileService = getService("file");
801
+ if (!isValidFilename(filename)) {
802
+ throw new ApplicationError("File name contains invalid characters");
803
+ }
804
+ let ext = path__default.default.extname(filename);
805
+ if (!ext) {
806
+ ext = `.${mimeTypes.extension(type)}`;
807
+ }
808
+ const usedName = (fileInfo.name || filename).normalize();
809
+ const basename = path__default.default.basename(usedName, ext);
810
+ if (!isValidFilename(filename)) {
811
+ throw new ApplicationError("File name contains invalid characters");
812
+ }
813
+ const entity = {
814
+ name: usedName,
815
+ alternativeText: fileInfo.alternativeText,
816
+ caption: fileInfo.caption,
817
+ folder: fileInfo.folder,
818
+ folderPath: await fileService.getFolderPath(fileInfo.folder),
819
+ hash: generateFileName(basename),
820
+ ext,
821
+ mime: type,
822
+ size: bytesToKbytes$1(size),
823
+ sizeInBytes: size
824
+ };
825
+ const { refId, ref, field } = metas;
826
+ if (refId && ref && field) {
827
+ entity.related = [
828
+ {
829
+ id: refId,
830
+ __type: ref,
831
+ __pivot: { field }
832
+ }
833
+ ];
834
+ }
835
+ if (metas.path) {
836
+ entity.path = metas.path;
837
+ }
838
+ if (metas.tmpWorkingDirectory) {
839
+ entity.tmpWorkingDirectory = metas.tmpWorkingDirectory;
840
+ }
841
+ return entity;
842
+ }
843
+ async function enhanceAndValidateFile(file2, fileInfo, metas) {
844
+ const currentFile = await formatFileInfo(
845
+ {
846
+ filename: file2.originalFilename ?? "unamed",
847
+ type: file2.mimetype ?? "application/octet-stream",
848
+ size: file2.size
849
+ },
850
+ fileInfo,
851
+ {
852
+ ...metas,
853
+ tmpWorkingDirectory: file2.tmpWorkingDirectory
854
+ }
855
+ );
856
+ currentFile.getStream = () => fs__default.default.createReadStream(file2.filepath);
857
+ const { optimize: optimize2, isImage: isImage2, isFaultyImage: isFaultyImage2, isOptimizableImage: isOptimizableImage2 } = strapi2.plugin("upload").service("image-manipulation");
858
+ if (await isImage2(currentFile)) {
859
+ if (await isFaultyImage2(currentFile)) {
860
+ throw new ApplicationError("File is not a valid image");
861
+ }
862
+ if (await isOptimizableImage2(currentFile)) {
863
+ return optimize2(currentFile);
864
+ }
865
+ }
866
+ return currentFile;
867
+ }
868
+ async function upload2({
869
+ data,
870
+ files
871
+ }, opts) {
872
+ const { user } = opts ?? {};
873
+ const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(files);
874
+ let uploadedFiles = [];
875
+ try {
876
+ const { fileInfo, ...metas } = data;
877
+ const fileArray = Array.isArray(files) ? files : [files];
878
+ const fileInfoArray = Array.isArray(fileInfo) ? fileInfo : [fileInfo];
879
+ const doUpload = async (file2, fileInfo2) => {
880
+ const fileData = await enhanceAndValidateFile(file2, fileInfo2, metas);
881
+ return uploadFileAndPersist(fileData, { user });
882
+ };
883
+ uploadedFiles = await Promise.all(
884
+ fileArray.map((file2, idx) => doUpload(file2, fileInfoArray[idx] || {}))
885
+ );
886
+ } finally {
887
+ await fse__default.default.remove(tmpWorkingDirectory);
888
+ }
889
+ return uploadedFiles;
890
+ }
891
+ async function uploadImage(fileData) {
892
+ const { getDimensions: getDimensions2, generateThumbnail: generateThumbnail2, generateResponsiveFormats: generateResponsiveFormats2, isResizableImage: isResizableImage2 } = getService("image-manipulation");
893
+ const { width, height } = await getDimensions2(fileData);
894
+ ___default.default.assign(fileData, {
895
+ width,
896
+ height
897
+ });
898
+ const uploadThumbnail = async (thumbnailFile) => {
899
+ await getService("provider").upload(thumbnailFile);
900
+ ___default.default.set(fileData, "formats.thumbnail", thumbnailFile);
901
+ };
902
+ const uploadResponsiveFormat = async (format) => {
903
+ const { key, file: file2 } = format;
904
+ await getService("provider").upload(file2);
905
+ ___default.default.set(fileData, ["formats", key], file2);
906
+ };
907
+ const uploadPromises = [];
908
+ uploadPromises.push(getService("provider").upload(fileData));
909
+ if (await isResizableImage2(fileData)) {
910
+ const thumbnailFile = await generateThumbnail2(fileData);
911
+ if (thumbnailFile) {
912
+ uploadPromises.push(uploadThumbnail(thumbnailFile));
913
+ }
914
+ const formats = await generateResponsiveFormats2(fileData);
915
+ if (Array.isArray(formats) && formats.length > 0) {
916
+ for (const format of formats) {
917
+ if (!format)
918
+ continue;
919
+ uploadPromises.push(uploadResponsiveFormat(format));
920
+ }
921
+ }
922
+ }
923
+ await Promise.all(uploadPromises);
924
+ }
925
+ async function uploadFileAndPersist(fileData, opts) {
926
+ const { user } = opts ?? {};
927
+ const config2 = strapi2.config.get("plugin::upload");
928
+ const { isImage: isImage2 } = getService("image-manipulation");
929
+ await getService("provider").checkFileSize(fileData);
930
+ if (await isImage2(fileData)) {
931
+ await uploadImage(fileData);
932
+ } else {
933
+ await getService("provider").upload(fileData);
934
+ }
935
+ ___default.default.set(fileData, "provider", config2.provider);
936
+ return add(fileData, { user });
937
+ }
938
+ async function updateFileInfo(id, { name, alternativeText, caption, folder: folder2 }, opts) {
939
+ const { user } = opts ?? {};
940
+ const dbFile = await findOne(id);
941
+ if (!dbFile) {
942
+ throw new NotFoundError();
943
+ }
944
+ const fileService = getService("file");
945
+ const newName = ___default.default.isNil(name) ? dbFile.name : name;
946
+ const newInfos = {
947
+ name: newName,
948
+ alternativeText: ___default.default.isNil(alternativeText) ? dbFile.alternativeText : alternativeText,
949
+ caption: ___default.default.isNil(caption) ? dbFile.caption : caption,
950
+ folder: ___default.default.isUndefined(folder2) ? dbFile.folder : folder2,
951
+ folderPath: ___default.default.isUndefined(folder2) ? dbFile.path : await fileService.getFolderPath(folder2)
952
+ };
953
+ return update2(id, newInfos, { user });
954
+ }
955
+ async function replace(id, { data, file: file2 }, opts) {
956
+ const { user } = opts ?? {};
957
+ const config2 = strapi2.config.get("plugin::upload");
958
+ const { isImage: isImage2 } = getService("image-manipulation");
959
+ const dbFile = await findOne(id);
960
+ if (!dbFile) {
961
+ throw new NotFoundError();
962
+ }
963
+ const tmpWorkingDirectory = await createAndAssignTmpWorkingDirectoryToFiles(file2);
964
+ let fileData;
965
+ try {
966
+ const { fileInfo } = data;
967
+ fileData = await enhanceAndValidateFile(file2, fileInfo);
968
+ ___default.default.assign(fileData, {
969
+ hash: dbFile.hash,
970
+ ext: dbFile.ext
971
+ });
972
+ if (dbFile.provider === config2.provider) {
973
+ await strapi2.plugin("upload").provider.delete(dbFile);
974
+ if (dbFile.formats) {
975
+ await Promise.all(
976
+ Object.keys(dbFile.formats).map((key) => {
977
+ return strapi2.plugin("upload").provider.delete(dbFile.formats[key]);
978
+ })
979
+ );
980
+ }
981
+ }
982
+ ___default.default.set(fileData, "formats", {});
983
+ if (await isImage2(fileData)) {
984
+ await uploadImage(fileData);
985
+ } else {
986
+ await getService("provider").upload(fileData);
987
+ }
988
+ ___default.default.set(fileData, "provider", config2.provider);
989
+ } finally {
990
+ await fse__default.default.remove(tmpWorkingDirectory);
991
+ }
992
+ return update2(id, fileData, { user });
993
+ }
994
+ async function update2(id, values, opts) {
995
+ const { user } = opts ?? {};
996
+ const fileValues = { ...values };
997
+ if (user) {
998
+ Object.assign(fileValues, {
999
+ [UPDATED_BY_ATTRIBUTE]: user.id
1000
+ });
1001
+ }
1002
+ sendMediaMetrics(fileValues);
1003
+ const res = await strapi2.db.query(FILE_MODEL_UID).update({ where: { id }, data: fileValues });
1004
+ await emitEvent(MEDIA_UPDATE, res);
1005
+ return res;
1006
+ }
1007
+ async function add(values, opts) {
1008
+ const { user } = opts ?? {};
1009
+ const fileValues = { ...values };
1010
+ if (user) {
1011
+ Object.assign(fileValues, {
1012
+ [UPDATED_BY_ATTRIBUTE]: user.id,
1013
+ [CREATED_BY_ATTRIBUTE]: user.id
1014
+ });
1015
+ }
1016
+ sendMediaMetrics(fileValues);
1017
+ const res = await strapi2.db.query(FILE_MODEL_UID).create({ data: fileValues });
1018
+ await emitEvent(MEDIA_CREATE, res);
1019
+ return res;
1020
+ }
1021
+ function findOne(id, populate = {}) {
1022
+ const query = strapi2.get("query-params").transform(FILE_MODEL_UID, {
1023
+ populate
1024
+ });
1025
+ return strapi2.db.query(FILE_MODEL_UID).findOne({
1026
+ where: { id },
1027
+ ...query
1028
+ });
1029
+ }
1030
+ function findMany(query = {}) {
1031
+ return strapi2.db.query(FILE_MODEL_UID).findMany(strapi2.get("query-params").transform(FILE_MODEL_UID, query));
1032
+ }
1033
+ function findPage(query = {}) {
1034
+ return strapi2.db.query(FILE_MODEL_UID).findPage(strapi2.get("query-params").transform(FILE_MODEL_UID, query));
1035
+ }
1036
+ async function remove(file2) {
1037
+ const config2 = strapi2.config.get("plugin::upload");
1038
+ if (file2.provider === config2.provider) {
1039
+ await strapi2.plugin("upload").provider.delete(file2);
1040
+ if (file2.formats) {
1041
+ const keys = Object.keys(file2.formats);
1042
+ await Promise.all(
1043
+ keys.map((key) => {
1044
+ return strapi2.plugin("upload").provider.delete(file2.formats[key]);
1045
+ })
1046
+ );
1047
+ }
1048
+ }
1049
+ const media = await strapi2.db.query(FILE_MODEL_UID).findOne({
1050
+ where: { id: file2.id }
1051
+ });
1052
+ await emitEvent(MEDIA_DELETE, media);
1053
+ return strapi2.db.query(FILE_MODEL_UID).delete({ where: { id: file2.id } });
1054
+ }
1055
+ async function getSettings() {
1056
+ const res = await strapi2.store({ type: "plugin", name: "upload", key: "settings" }).get({});
1057
+ return res;
1058
+ }
1059
+ function setSettings(value) {
1060
+ if (value.responsiveDimensions === true) {
1061
+ strapi2.telemetry.send("didEnableResponsiveDimensions");
1062
+ } else {
1063
+ strapi2.telemetry.send("didDisableResponsiveDimensions");
1064
+ }
1065
+ return strapi2.store({ type: "plugin", name: "upload", key: "settings" }).set({ value });
1066
+ }
1067
+ async function getConfiguration() {
1068
+ const res = await strapi2.store({
1069
+ type: "plugin",
1070
+ name: "upload",
1071
+ key: "view_configuration"
1072
+ }).get({});
1073
+ return res;
1074
+ }
1075
+ function setConfiguration(value) {
1076
+ return strapi2.store({ type: "plugin", name: "upload", key: "view_configuration" }).set({
1077
+ value
1078
+ });
1079
+ }
1080
+ return {
1081
+ formatFileInfo,
1082
+ upload: upload2,
1083
+ updateFileInfo,
1084
+ replace,
1085
+ findOne,
1086
+ findMany,
1087
+ findPage,
1088
+ remove,
1089
+ getSettings,
1090
+ setSettings,
1091
+ getConfiguration,
1092
+ setConfiguration,
1093
+ /**
1094
+ * exposed for testing only
1095
+ * @internal
1096
+ */
1097
+ _uploadImage: uploadImage
1098
+ };
1099
+ };
1100
+ const { bytesToKbytes, writableDiscardStream } = utils.file;
1101
+ const FORMATS_TO_RESIZE = ["jpeg", "png", "webp", "tiff", "gif"];
1102
+ const FORMATS_TO_PROCESS = ["jpeg", "png", "webp", "tiff", "svg", "gif", "avif"];
1103
+ const FORMATS_TO_OPTIMIZE = ["jpeg", "png", "webp", "tiff", "avif"];
1104
+ const isOptimizableFormat = (format) => format !== void 0 && FORMATS_TO_OPTIMIZE.includes(format);
1105
+ const writeStreamToFile = (stream, path2) => new Promise((resolve, reject) => {
1106
+ const writeStream = fs__default.default.createWriteStream(path2);
1107
+ stream.on("error", reject);
1108
+ stream.pipe(writeStream);
1109
+ writeStream.on("close", resolve);
1110
+ writeStream.on("error", reject);
1111
+ });
1112
+ const getMetadata = (file2) => new Promise((resolve, reject) => {
1113
+ const pipeline = sharp__default.default();
1114
+ pipeline.metadata().then(resolve).catch(reject);
1115
+ file2.getStream().pipe(pipeline);
1116
+ });
1117
+ const getDimensions = async (file2) => {
1118
+ const { width = null, height = null } = await getMetadata(file2);
1119
+ return { width, height };
1120
+ };
1121
+ const THUMBNAIL_RESIZE_OPTIONS = {
1122
+ width: 245,
1123
+ height: 156,
1124
+ fit: "inside"
1125
+ };
1126
+ const resizeFileTo = async (file2, options, {
1127
+ name,
1128
+ hash
1129
+ }) => {
1130
+ const filePath = file2.tmpWorkingDirectory ? path.join(file2.tmpWorkingDirectory, hash) : hash;
1131
+ await writeStreamToFile(file2.getStream().pipe(sharp__default.default().resize(options)), filePath);
1132
+ const newFile = {
1133
+ name,
1134
+ hash,
1135
+ ext: file2.ext,
1136
+ mime: file2.mime,
1137
+ path: file2.path || null,
1138
+ getStream: () => fs__default.default.createReadStream(filePath)
1139
+ };
1140
+ const { width, height, size } = await getMetadata(newFile);
1141
+ Object.assign(newFile, {
1142
+ width,
1143
+ height,
1144
+ size: size ? bytesToKbytes(size) : 0,
1145
+ sizeInBytes: size
1146
+ });
1147
+ return newFile;
1148
+ };
1149
+ const generateThumbnail = async (file2) => {
1150
+ if (file2.width && file2.height && (file2.width > THUMBNAIL_RESIZE_OPTIONS.width || file2.height > THUMBNAIL_RESIZE_OPTIONS.height)) {
1151
+ const newFile = await resizeFileTo(file2, THUMBNAIL_RESIZE_OPTIONS, {
1152
+ name: `thumbnail_${file2.name}`,
1153
+ hash: `thumbnail_${file2.hash}`
1154
+ });
1155
+ return newFile;
1156
+ }
1157
+ return null;
1158
+ };
1159
+ const optimize = async (file2) => {
1160
+ const { sizeOptimization = false, autoOrientation = false } = await getService("upload").getSettings() ?? {};
1161
+ const newFile = { ...file2 };
1162
+ const { width, height, size, format } = await getMetadata(newFile);
1163
+ if ((sizeOptimization || autoOrientation) && isOptimizableFormat(format)) {
1164
+ const transformer = sharp__default.default();
1165
+ transformer[format]({ quality: sizeOptimization ? 80 : 100 });
1166
+ if (autoOrientation) {
1167
+ transformer.rotate();
1168
+ }
1169
+ const filePath = file2.tmpWorkingDirectory ? path.join(file2.tmpWorkingDirectory, `optimized-${file2.hash}`) : `optimized-${file2.hash}`;
1170
+ await writeStreamToFile(file2.getStream().pipe(transformer), filePath);
1171
+ newFile.getStream = () => fs__default.default.createReadStream(filePath);
1172
+ }
1173
+ const { width: newWidth, height: newHeight, size: newSize } = await getMetadata(newFile);
1174
+ if (newSize && size && newSize > size) {
1175
+ return { ...file2, width, height, size: bytesToKbytes(size), sizeInBytes: size };
1176
+ }
1177
+ return Object.assign(newFile, {
1178
+ width: newWidth,
1179
+ height: newHeight,
1180
+ size: newSize ? bytesToKbytes(newSize) : 0,
1181
+ sizeInBytes: newSize
1182
+ });
1183
+ };
1184
+ const DEFAULT_BREAKPOINTS = {
1185
+ large: 1e3,
1186
+ medium: 750,
1187
+ small: 500
1188
+ };
1189
+ const getBreakpoints = () => strapi.config.get("plugin::upload.breakpoints", DEFAULT_BREAKPOINTS);
1190
+ const generateResponsiveFormats = async (file2) => {
1191
+ const { responsiveDimensions = false } = await getService("upload").getSettings() ?? {};
1192
+ if (!responsiveDimensions)
1193
+ return [];
1194
+ const originalDimensions = await getDimensions(file2);
1195
+ const breakpoints = getBreakpoints();
1196
+ return Promise.all(
1197
+ Object.keys(breakpoints).map((key) => {
1198
+ const breakpoint = breakpoints[key];
1199
+ if (breakpointSmallerThan(breakpoint, originalDimensions)) {
1200
+ return generateBreakpoint(key, { file: file2, breakpoint });
1201
+ }
1202
+ return void 0;
1203
+ })
1204
+ );
1205
+ };
1206
+ const generateBreakpoint = async (key, { file: file2, breakpoint }) => {
1207
+ const newFile = await resizeFileTo(
1208
+ file2,
1209
+ {
1210
+ width: breakpoint,
1211
+ height: breakpoint,
1212
+ fit: "inside"
1213
+ },
1214
+ {
1215
+ name: `${key}_${file2.name}`,
1216
+ hash: `${key}_${file2.hash}`
1217
+ }
1218
+ );
1219
+ return {
1220
+ key,
1221
+ file: newFile
1222
+ };
1223
+ };
1224
+ const breakpointSmallerThan = (breakpoint, { width, height }) => {
1225
+ return breakpoint < (width ?? 0) || breakpoint < (height ?? 0);
1226
+ };
1227
+ const isFaultyImage = (file2) => new Promise((resolve) => {
1228
+ file2.getStream().pipe(sharp__default.default().rotate()).on("error", () => resolve(true)).pipe(writableDiscardStream()).on("error", () => resolve(true)).on("close", () => resolve(false));
1229
+ });
1230
+ const isOptimizableImage = async (file2) => {
1231
+ let format;
1232
+ try {
1233
+ const metadata = await getMetadata(file2);
1234
+ format = metadata.format;
1235
+ } catch (e) {
1236
+ return false;
1237
+ }
1238
+ return format && FORMATS_TO_OPTIMIZE.includes(format);
1239
+ };
1240
+ const isResizableImage = async (file2) => {
1241
+ let format;
1242
+ try {
1243
+ const metadata = await getMetadata(file2);
1244
+ format = metadata.format;
1245
+ } catch (e) {
1246
+ return false;
1247
+ }
1248
+ return format && FORMATS_TO_RESIZE.includes(format);
1249
+ };
1250
+ const isImage = async (file2) => {
1251
+ let format;
1252
+ try {
1253
+ const metadata = await getMetadata(file2);
1254
+ format = metadata.format;
1255
+ } catch (e) {
1256
+ return false;
1257
+ }
1258
+ return format && FORMATS_TO_PROCESS.includes(format);
1259
+ };
1260
+ const imageManipulation = {
1261
+ isFaultyImage,
1262
+ isOptimizableImage,
1263
+ isResizableImage,
1264
+ isImage,
1265
+ getDimensions,
1266
+ generateResponsiveFormats,
1267
+ generateThumbnail,
1268
+ optimize
1269
+ };
1270
+ const setPathIdAndPath = async (folder2) => {
1271
+ const { max } = await strapi.db.queryBuilder(FOLDER_MODEL_UID).max("pathId").first().execute();
1272
+ const pathId = max + 1;
1273
+ let parentPath = "/";
1274
+ if (folder2.parent) {
1275
+ const parentFolder = await strapi.db.query(FOLDER_MODEL_UID).findOne({ where: { id: folder2.parent } });
1276
+ parentPath = parentFolder.path;
1277
+ }
1278
+ return Object.assign(folder2, {
1279
+ pathId,
1280
+ path: utils.strings.joinBy("/", parentPath, `${pathId}`)
1281
+ });
1282
+ };
1283
+ const create = async (folderData, opts) => {
1284
+ const folderService = getService("folder");
1285
+ const { user } = opts || {};
1286
+ let enrichedFolder = await folderService.setPathIdAndPath(folderData);
1287
+ if (user) {
1288
+ enrichedFolder = await utils.setCreatorFields({ user })(enrichedFolder);
1289
+ }
1290
+ const folder2 = await strapi.db.query(FOLDER_MODEL_UID).create({ data: enrichedFolder });
1291
+ strapi.eventHub.emit("media-folder.create", { folder: folder2 });
1292
+ return folder2;
1293
+ };
1294
+ const deleteByIds$1 = async (ids = []) => {
1295
+ const folders = await strapi.db.query(FOLDER_MODEL_UID).findMany({ where: { id: { $in: ids } } });
1296
+ if (folders.length === 0) {
1297
+ return {
1298
+ folders: [],
1299
+ totalFolderNumber: 0,
1300
+ totalFileNumber: 0
1301
+ };
1302
+ }
1303
+ const pathsToDelete = fp.map("path", folders);
1304
+ const filesToDelete = await strapi.db.query(FILE_MODEL_UID).findMany({
1305
+ where: {
1306
+ $or: pathsToDelete.flatMap((path2) => [
1307
+ { folderPath: { $eq: path2 } },
1308
+ { folderPath: { $startsWith: `${path2}/` } }
1309
+ ])
1310
+ }
1311
+ });
1312
+ await Promise.all(filesToDelete.map((file2) => getService("upload").remove(file2)));
1313
+ const { count: totalFolderNumber } = await strapi.db.query(FOLDER_MODEL_UID).deleteMany({
1314
+ where: {
1315
+ $or: pathsToDelete.flatMap((path2) => [
1316
+ { path: { $eq: path2 } },
1317
+ { path: { $startsWith: `${path2}/` } }
1318
+ ])
1319
+ }
1320
+ });
1321
+ strapi.eventHub.emit("media-folder.delete", { folders });
1322
+ return {
1323
+ folders,
1324
+ totalFolderNumber,
1325
+ totalFileNumber: filesToDelete.length
1326
+ };
1327
+ };
1328
+ const update = async (id, {
1329
+ name,
1330
+ parent
1331
+ }, { user }) => {
1332
+ if (fp.isUndefined(parent)) {
1333
+ const existingFolder = await strapi.db.query(FOLDER_MODEL_UID).findOne({ where: { id } });
1334
+ if (!existingFolder) {
1335
+ return void 0;
1336
+ }
1337
+ const newFolder = utils.setCreatorFields({ user, isEdition: true })({ name, parent });
1338
+ if (fp.isUndefined(parent)) {
1339
+ const folder2 = await strapi.db.query(FOLDER_MODEL_UID).update({ where: { id }, data: newFolder });
1340
+ return folder2;
1341
+ }
1342
+ } else {
1343
+ const trx = await strapi.db.transaction();
1344
+ try {
1345
+ const existingFolder = await strapi.db.queryBuilder(FOLDER_MODEL_UID).select(["pathId", "path"]).where({ id }).transacting(trx.get()).forUpdate().first().execute();
1346
+ const { joinTable } = strapi.db.metadata.get(FOLDER_MODEL_UID).attributes.parent;
1347
+ await strapi.db.queryBuilder(joinTable.name).transacting(trx.get()).delete().where({ [joinTable.joinColumn.name]: id }).execute();
1348
+ if (parent !== null) {
1349
+ await strapi.db.queryBuilder(joinTable.name).transacting(trx.get()).insert({ [joinTable.inverseJoinColumn.name]: parent, [joinTable.joinColumn.name]: id }).where({ [joinTable.joinColumn.name]: id }).execute();
1350
+ }
1351
+ let destinationFolderPath = "/";
1352
+ if (parent !== null) {
1353
+ const destinationFolder = await strapi.db.queryBuilder(FOLDER_MODEL_UID).select("path").where({ id: parent }).transacting(trx.get()).first().execute();
1354
+ destinationFolderPath = destinationFolder.path;
1355
+ }
1356
+ const folderTable = strapi.getModel(FOLDER_MODEL_UID).collectionName;
1357
+ const fileTable = strapi.getModel(FILE_MODEL_UID).collectionName;
1358
+ const folderPathColumnName = (
1359
+ // @ts-expect-error - no dynamic types
1360
+ strapi.db.metadata.get(FILE_MODEL_UID).attributes.folderPath.columnName
1361
+ );
1362
+ const pathColumnName = strapi.db.metadata.get(FOLDER_MODEL_UID).attributes.path.columnName;
1363
+ await strapi.db.getConnection(folderTable).transacting(trx.get()).where(pathColumnName, existingFolder.path).orWhere(pathColumnName, "like", `${existingFolder.path}/%`).update(
1364
+ pathColumnName,
1365
+ strapi.db.connection.raw("REPLACE(??, ?, ?)", [
1366
+ pathColumnName,
1367
+ existingFolder.path,
1368
+ utils.strings.joinBy("/", destinationFolderPath, `${existingFolder.pathId}`)
1369
+ ])
1370
+ );
1371
+ await strapi.db.getConnection(fileTable).transacting(trx.get()).where(folderPathColumnName, existingFolder.path).orWhere(folderPathColumnName, "like", `${existingFolder.path}/%`).update(
1372
+ folderPathColumnName,
1373
+ strapi.db.connection.raw("REPLACE(??, ?, ?)", [
1374
+ folderPathColumnName,
1375
+ existingFolder.path,
1376
+ utils.strings.joinBy("/", destinationFolderPath, `${existingFolder.pathId}`)
1377
+ ])
1378
+ );
1379
+ await trx.commit();
1380
+ } catch (e) {
1381
+ await trx.rollback();
1382
+ throw e;
1383
+ }
1384
+ const newFolder = utils.setCreatorFields({ user, isEdition: true })({ name });
1385
+ const folder2 = await strapi.db.query(FOLDER_MODEL_UID).update({ where: { id }, data: newFolder });
1386
+ strapi.eventHub.emit("media-folder.update", { folder: folder2 });
1387
+ return folder2;
1388
+ }
1389
+ };
1390
+ const exists = async (params = {}) => {
1391
+ const count = await strapi.db.query(FOLDER_MODEL_UID).count({ where: params });
1392
+ return count > 0;
1393
+ };
1394
+ const getStructure = async () => {
1395
+ const { joinTable } = strapi.db.metadata.get(FOLDER_MODEL_UID).attributes.parent;
1396
+ const qb = strapi.db.queryBuilder(FOLDER_MODEL_UID);
1397
+ const alias = qb.getAlias();
1398
+ const folders = await qb.select(["id", "name", `${alias}.${joinTable.inverseJoinColumn.name} as parent`]).join({
1399
+ alias,
1400
+ referencedTable: joinTable.name,
1401
+ referencedColumn: joinTable.joinColumn.name,
1402
+ rootColumn: joinTable.joinColumn.referencedColumn,
1403
+ rootTable: qb.alias
1404
+ }).execute({ mapResults: false });
1405
+ const folderMap = {
1406
+ null: { children: [] }
1407
+ };
1408
+ folders.forEach((f) => {
1409
+ folderMap[f.id] = { ...f, children: [] };
1410
+ });
1411
+ folders.forEach((f) => {
1412
+ const parentId = f.parent || "null";
1413
+ if (!folderMap[parentId]) {
1414
+ folderMap[parentId] = { children: [] };
1415
+ }
1416
+ folderMap[parentId].children.push(folderMap[f.id]);
1417
+ folderMap[parentId].children = fp.sortBy("name", folderMap[parentId].children);
1418
+ delete folderMap[f.id].parent;
1419
+ });
1420
+ return folderMap.null.children;
1421
+ };
1422
+ const folder = {
1423
+ create,
1424
+ exists,
1425
+ deleteByIds: deleteByIds$1,
1426
+ update,
1427
+ setPathIdAndPath,
1428
+ getStructure
1429
+ };
1430
+ const getFolderPath = async (folderId) => {
1431
+ if (!folderId)
1432
+ return "/";
1433
+ const parentFolder = await strapi.db.query(FOLDER_MODEL_UID).findOne({ where: { id: folderId } });
1434
+ return parentFolder.path;
1435
+ };
1436
+ const deleteByIds = async (ids = []) => {
1437
+ const filesToDelete = await strapi.db.query(FILE_MODEL_UID).findMany({ where: { id: { $in: ids } } });
1438
+ await Promise.all(filesToDelete.map((file2) => getService("upload").remove(file2)));
1439
+ return filesToDelete;
1440
+ };
1441
+ const signFileUrls = async (file2) => {
1442
+ const { provider: provider2 } = strapi.plugins.upload;
1443
+ const { provider: providerConfig } = strapi.config.get("plugin::upload");
1444
+ const isPrivate = await provider2.isPrivate();
1445
+ file2.isUrlSigned = false;
1446
+ if (file2.provider !== providerConfig || !isPrivate) {
1447
+ return file2;
1448
+ }
1449
+ const signUrl = async (file22) => {
1450
+ const signedUrl = await provider2.getSignedUrl(file22);
1451
+ file22.url = signedUrl.url;
1452
+ file22.isUrlSigned = true;
1453
+ };
1454
+ const signedFile = fp.cloneDeep(file2);
1455
+ await signUrl(signedFile);
1456
+ if (file2.formats) {
1457
+ await utils.async.map(Object.values(signedFile.formats ?? {}), signUrl);
1458
+ }
1459
+ return signedFile;
1460
+ };
1461
+ const file = { getFolderPath, deleteByIds, signFileUrls };
1462
+ const getWeeklyCronScheduleAt = (date) => `${date.getSeconds()} ${date.getMinutes()} ${date.getHours()} * * ${date.getDay()}`;
1463
+ const ONE_WEEK = 7 * 24 * 60 * 60 * 1e3;
1464
+ const getMetricsStoreValue = async () => {
1465
+ const value = await strapi.store.get({ type: "plugin", name: "upload", key: "metrics" });
1466
+ return fp.defaultTo({}, value);
1467
+ };
1468
+ const setMetricsStoreValue = (value) => strapi.store.set({ type: "plugin", name: "upload", key: "metrics", value });
1469
+ const weeklyMetrics = ({ strapi: strapi2 }) => ({
1470
+ async computeMetrics() {
1471
+ const pathColName = strapi2.db.metadata.get(FOLDER_MODEL_UID).attributes.path.columnName;
1472
+ const folderTable = strapi2.getModel(FOLDER_MODEL_UID).collectionName;
1473
+ let keepOnlySlashesSQLString = "??";
1474
+ const queryParams = [pathColName];
1475
+ for (let i = 0; i < 10; i += 1) {
1476
+ keepOnlySlashesSQLString = `REPLACE(${keepOnlySlashesSQLString}, ?, ?)`;
1477
+ queryParams.push(String(i), "");
1478
+ }
1479
+ const res = await strapi2.db.getConnection(folderTable).select(
1480
+ strapi2.db.connection.raw(
1481
+ `LENGTH(${keepOnlySlashesSQLString}) AS depth, COUNT(*) AS occurence`,
1482
+ queryParams
1483
+ )
1484
+ ).groupBy("depth");
1485
+ const folderLevelsArray = res.map((map) => ({
1486
+ depth: Number(map.depth),
1487
+ occurence: Number(map.occurence)
1488
+ }));
1489
+ let product = 0;
1490
+ let folderNumber = 0;
1491
+ let maxDepth = 0;
1492
+ for (const folderLevel of folderLevelsArray) {
1493
+ product += folderLevel.depth * folderLevel.occurence;
1494
+ folderNumber += folderLevel.occurence;
1495
+ if (folderLevel.depth > maxDepth) {
1496
+ maxDepth = folderLevel.depth;
1497
+ }
1498
+ }
1499
+ const averageDepth = folderNumber !== 0 ? product / folderNumber : 0;
1500
+ let sumOfDeviation = 0;
1501
+ for (const folderLevel of folderLevelsArray) {
1502
+ sumOfDeviation += Math.abs(folderLevel.depth - averageDepth) * folderLevel.occurence;
1503
+ }
1504
+ const averageDeviationDepth = folderNumber !== 0 ? sumOfDeviation / folderNumber : 0;
1505
+ const assetNumber = await strapi2.db.query(FILE_MODEL_UID).count();
1506
+ return {
1507
+ assetNumber,
1508
+ folderNumber,
1509
+ averageDepth,
1510
+ maxDepth,
1511
+ averageDeviationDepth
1512
+ };
1513
+ },
1514
+ async sendMetrics() {
1515
+ const metrics2 = await this.computeMetrics();
1516
+ strapi2.telemetry.send("didSendUploadPropertiesOnceAWeek", {
1517
+ groupProperties: { metrics: metrics2 }
1518
+ });
1519
+ const metricsInfoStored = await getMetricsStoreValue();
1520
+ await setMetricsStoreValue({ ...metricsInfoStored, lastWeeklyUpdate: (/* @__PURE__ */ new Date()).getTime() });
1521
+ },
1522
+ async ensureWeeklyStoredCronSchedule() {
1523
+ const metricsInfoStored = await getMetricsStoreValue();
1524
+ const { weeklySchedule: currentSchedule, lastWeeklyUpdate } = metricsInfoStored;
1525
+ const now = /* @__PURE__ */ new Date();
1526
+ let weeklySchedule = currentSchedule;
1527
+ if (!weeklySchedule || !lastWeeklyUpdate || lastWeeklyUpdate + ONE_WEEK < now.getTime()) {
1528
+ weeklySchedule = getWeeklyCronScheduleAt(dateFns.add(now, { minutes: 5 }));
1529
+ await setMetricsStoreValue({ ...metricsInfoStored, weeklySchedule });
1530
+ return weeklySchedule;
1531
+ }
1532
+ return weeklySchedule;
1533
+ },
1534
+ async registerCron() {
1535
+ const weeklySchedule = await this.ensureWeeklyStoredCronSchedule();
1536
+ strapi2.cron.add({ [weeklySchedule]: this.sendMetrics.bind(this) });
1537
+ }
1538
+ });
1539
+ const getProviderName = () => strapi.config.get("plugin::upload.provider", "local");
1540
+ const isProviderPrivate = async () => strapi.plugin("upload").provider.isPrivate();
1541
+ const metrics = ({ strapi: strapi2 }) => ({
1542
+ async sendUploadPluginMetrics() {
1543
+ const uploadProvider = getProviderName();
1544
+ const privateProvider = await isProviderPrivate();
1545
+ strapi2.telemetry.send("didInitializePluginUpload", {
1546
+ groupProperties: {
1547
+ uploadProvider,
1548
+ privateProvider
1549
+ }
1550
+ });
1551
+ }
1552
+ });
1553
+ const getStore = () => strapi.store({ type: "plugin", name: "upload", key: "api-folder" });
1554
+ const createApiUploadFolder = async () => {
1555
+ let name = API_UPLOAD_FOLDER_BASE_NAME;
1556
+ const folderService = getService("folder");
1557
+ let exists2 = true;
1558
+ let index2 = 1;
1559
+ while (exists2) {
1560
+ exists2 = await folderService.exists({ name, parent: null });
1561
+ if (exists2) {
1562
+ name = `${API_UPLOAD_FOLDER_BASE_NAME} (${index2})`;
1563
+ index2 += 1;
1564
+ }
1565
+ }
1566
+ const folder2 = await folderService.create({ name });
1567
+ await getStore().set({ value: { id: folder2.id } });
1568
+ return folder2;
1569
+ };
1570
+ const getAPIUploadFolder = async () => {
1571
+ const storeValue = await getStore().get({});
1572
+ const folderId = fp.get("id", storeValue);
1573
+ const folder2 = folderId ? await strapi.db.query(FOLDER_MODEL_UID).findOne({ where: { id: folderId } }) : null;
1574
+ return fp.isNil(folder2) ? createApiUploadFolder() : folder2;
1575
+ };
1576
+ const apiUploadFolder = {
1577
+ getAPIUploadFolder
1578
+ };
1579
+ const addSignedFileUrlsToAdmin = async () => {
1580
+ };
1581
+ const entityManager = { addSignedFileUrlsToAdmin };
1582
+ const contentManager = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
1583
+ __proto__: null,
1584
+ entityManager
1585
+ }, Symbol.toStringTag, { value: "Module" }));
1586
+ const addSignedFileUrlsToEntityService = async () => {
1587
+ };
1588
+ const entityService = { addSignedFileUrlsToEntityService };
1589
+ const core = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
1590
+ __proto__: null,
1591
+ entityService
1592
+ }, Symbol.toStringTag, { value: "Module" }));
1593
+ const extensions = {
1594
+ contentManager,
1595
+ core
1596
+ };
1597
+ const services = {
1598
+ provider,
1599
+ upload,
1600
+ folder,
1601
+ file,
1602
+ weeklyMetrics,
1603
+ metrics,
1604
+ "image-manipulation": imageManipulation,
1605
+ "api-upload-folder": apiUploadFolder,
1606
+ extensions
1607
+ };
1608
+ const routes$3 = {
1609
+ type: "admin",
1610
+ routes: [
1611
+ {
1612
+ method: "GET",
1613
+ path: "/settings",
1614
+ handler: "admin-settings.getSettings",
1615
+ config: {
1616
+ policies: [
1617
+ "admin::isAuthenticatedAdmin",
1618
+ {
1619
+ name: "admin::hasPermissions",
1620
+ config: {
1621
+ actions: ["plugin::upload.settings.read"]
1622
+ }
1623
+ }
1624
+ ]
1625
+ }
1626
+ },
1627
+ {
1628
+ method: "PUT",
1629
+ path: "/settings",
1630
+ handler: "admin-settings.updateSettings",
1631
+ config: {
1632
+ policies: [
1633
+ "admin::isAuthenticatedAdmin",
1634
+ {
1635
+ name: "admin::hasPermissions",
1636
+ config: {
1637
+ actions: ["plugin::upload.settings.read"]
1638
+ }
1639
+ }
1640
+ ]
1641
+ }
1642
+ },
1643
+ {
1644
+ method: "POST",
1645
+ path: "/",
1646
+ handler: "admin-upload.upload",
1647
+ config: {
1648
+ policies: ["admin::isAuthenticatedAdmin"]
1649
+ }
1650
+ },
1651
+ {
1652
+ method: "GET",
1653
+ path: "/files",
1654
+ handler: "admin-file.find",
1655
+ config: {
1656
+ policies: [
1657
+ "admin::isAuthenticatedAdmin",
1658
+ {
1659
+ name: "admin::hasPermissions",
1660
+ config: {
1661
+ actions: ["plugin::upload.read"]
1662
+ }
1663
+ }
1664
+ ]
1665
+ }
1666
+ },
1667
+ {
1668
+ method: "GET",
1669
+ path: "/files/:id",
1670
+ handler: "admin-file.findOne",
1671
+ config: {
1672
+ policies: [
1673
+ "admin::isAuthenticatedAdmin",
1674
+ {
1675
+ name: "admin::hasPermissions",
1676
+ config: {
1677
+ actions: ["plugin::upload.read"]
1678
+ }
1679
+ }
1680
+ ]
1681
+ }
1682
+ },
1683
+ {
1684
+ method: "DELETE",
1685
+ path: "/files/:id",
1686
+ handler: "admin-file.destroy",
1687
+ config: {
1688
+ policies: [
1689
+ "admin::isAuthenticatedAdmin",
1690
+ {
1691
+ name: "admin::hasPermissions",
1692
+ config: {
1693
+ actions: ["plugin::upload.assets.update"]
1694
+ }
1695
+ }
1696
+ ]
1697
+ }
1698
+ },
1699
+ {
1700
+ method: "GET",
1701
+ path: "/folders/:id",
1702
+ handler: "admin-folder.findOne",
1703
+ config: {
1704
+ policies: [
1705
+ "admin::isAuthenticatedAdmin",
1706
+ {
1707
+ name: "admin::hasPermissions",
1708
+ config: {
1709
+ actions: ["plugin::upload.read"]
1710
+ }
1711
+ }
1712
+ ]
1713
+ }
1714
+ },
1715
+ {
1716
+ method: "GET",
1717
+ path: "/folders",
1718
+ handler: "admin-folder.find",
1719
+ config: {
1720
+ policies: [
1721
+ "admin::isAuthenticatedAdmin",
1722
+ {
1723
+ name: "admin::hasPermissions",
1724
+ config: {
1725
+ actions: ["plugin::upload.read"]
1726
+ }
1727
+ }
1728
+ ]
1729
+ }
1730
+ },
1731
+ {
1732
+ method: "POST",
1733
+ path: "/folders",
1734
+ handler: "admin-folder.create",
1735
+ config: {
1736
+ policies: [
1737
+ "admin::isAuthenticatedAdmin",
1738
+ {
1739
+ name: "admin::hasPermissions",
1740
+ config: {
1741
+ actions: ["plugin::upload.assets.create"]
1742
+ }
1743
+ }
1744
+ ]
1745
+ }
1746
+ },
1747
+ {
1748
+ method: "PUT",
1749
+ path: "/folders/:id",
1750
+ handler: "admin-folder.update",
1751
+ config: {
1752
+ policies: [
1753
+ "admin::isAuthenticatedAdmin",
1754
+ {
1755
+ name: "admin::hasPermissions",
1756
+ config: {
1757
+ actions: ["plugin::upload.assets.update"]
1758
+ }
1759
+ }
1760
+ ]
1761
+ }
1762
+ },
1763
+ {
1764
+ method: "GET",
1765
+ path: "/folder-structure",
1766
+ handler: "admin-folder.getStructure",
1767
+ config: {
1768
+ policies: [
1769
+ "admin::isAuthenticatedAdmin",
1770
+ {
1771
+ name: "admin::hasPermissions",
1772
+ config: {
1773
+ actions: ["plugin::upload.read"]
1774
+ }
1775
+ }
1776
+ ]
1777
+ }
1778
+ },
1779
+ {
1780
+ method: "POST",
1781
+ path: "/actions/bulk-delete",
1782
+ handler: "admin-folder-file.deleteMany",
1783
+ config: {
1784
+ policies: [
1785
+ "admin::isAuthenticatedAdmin",
1786
+ {
1787
+ name: "admin::hasPermissions",
1788
+ config: {
1789
+ actions: ["plugin::upload.assets.update"]
1790
+ }
1791
+ }
1792
+ ]
1793
+ }
1794
+ },
1795
+ {
1796
+ method: "POST",
1797
+ path: "/actions/bulk-move",
1798
+ handler: "admin-folder-file.moveMany",
1799
+ config: {
1800
+ policies: [
1801
+ "admin::isAuthenticatedAdmin",
1802
+ {
1803
+ name: "admin::hasPermissions",
1804
+ config: {
1805
+ actions: ["plugin::upload.assets.update"]
1806
+ }
1807
+ }
1808
+ ]
1809
+ }
1810
+ }
1811
+ ]
1812
+ };
1813
+ const routes$2 = {
1814
+ type: "content-api",
1815
+ routes: [
1816
+ {
1817
+ method: "POST",
1818
+ path: "/",
1819
+ handler: "content-api.upload"
1820
+ },
1821
+ {
1822
+ method: "GET",
1823
+ path: "/files",
1824
+ handler: "content-api.find"
1825
+ },
1826
+ {
1827
+ method: "GET",
1828
+ path: "/files/:id",
1829
+ handler: "content-api.findOne"
1830
+ },
1831
+ {
1832
+ method: "DELETE",
1833
+ path: "/files/:id",
1834
+ handler: "content-api.destroy"
1835
+ }
1836
+ ]
1837
+ };
1838
+ const routes$1 = {
1839
+ type: "admin",
1840
+ routes: [
1841
+ {
1842
+ method: "GET",
1843
+ path: "/configuration",
1844
+ handler: "view-configuration.findViewConfiguration",
1845
+ config: {
1846
+ policies: ["admin::isAuthenticatedAdmin"]
1847
+ }
1848
+ },
1849
+ {
1850
+ method: "PUT",
1851
+ path: "/configuration",
1852
+ handler: "view-configuration.updateViewConfiguration",
1853
+ config: {
1854
+ policies: [
1855
+ "admin::isAuthenticatedAdmin",
1856
+ {
1857
+ name: "admin::hasPermissions",
1858
+ config: {
1859
+ actions: [ACTIONS.configureView]
1860
+ }
1861
+ }
1862
+ ]
1863
+ }
1864
+ }
1865
+ ]
1866
+ };
1867
+ const routes = {
1868
+ admin: routes$3,
1869
+ "content-api": routes$2,
1870
+ viewConfiguration: routes$1
1871
+ };
1872
+ const config = {
1873
+ default: {
1874
+ enabled: true,
1875
+ provider: "local",
1876
+ sizeLimit: 1e9,
1877
+ // 1GB
1878
+ actionOptions: {}
1879
+ },
1880
+ validator() {
1881
+ }
1882
+ };
1883
+ const findEntityAndCheckPermissions = async (ability, action, model, id) => {
1884
+ const file2 = await getService("upload").findOne(id, [
1885
+ utils.contentTypes.constants.CREATED_BY_ATTRIBUTE,
1886
+ "folder"
1887
+ ]);
1888
+ if (___default.default.isNil(file2)) {
1889
+ throw new utils.errors.NotFoundError();
1890
+ }
1891
+ const pm = strapi.service("admin::permission").createPermissionsManager({ ability, action, model });
1892
+ const creatorId = ___default.default.get(file2, [utils.contentTypes.constants.CREATED_BY_ATTRIBUTE, "id"]);
1893
+ const author = creatorId ? await strapi.service("admin::user").findOne(creatorId, ["roles"]) : null;
1894
+ const fileWithRoles = ___default.default.set(___default.default.cloneDeep(file2), "createdBy", author);
1895
+ if (pm.ability.cannot(pm.action, pm.toSubject(fileWithRoles))) {
1896
+ throw new utils.errors.ForbiddenError();
1897
+ }
1898
+ return { pm, file: file2 };
1899
+ };
1900
+ const adminFile = {
1901
+ async find(ctx) {
1902
+ const {
1903
+ state: { userAbility }
1904
+ } = ctx;
1905
+ const defaultQuery = { populate: { folder: true } };
1906
+ const pm = strapi.service("admin::permission").createPermissionsManager({
1907
+ ability: userAbility,
1908
+ action: ACTIONS.read,
1909
+ model: FILE_MODEL_UID
1910
+ });
1911
+ if (!pm.isAllowed) {
1912
+ return ctx.forbidden();
1913
+ }
1914
+ const pmQuery = pm.addPermissionsQueryTo(fp.merge(defaultQuery, ctx.query));
1915
+ await pm.validateQuery(pmQuery);
1916
+ const query = await pm.sanitizeQuery(pmQuery);
1917
+ const { results: files, pagination } = await getService("upload").findPage(query);
1918
+ const signedFiles = await utils.async.map(files, getService("file").signFileUrls);
1919
+ const sanitizedFiles = await pm.sanitizeOutput(signedFiles);
1920
+ return { results: sanitizedFiles, pagination };
1921
+ },
1922
+ async findOne(ctx) {
1923
+ const {
1924
+ state: { userAbility },
1925
+ params: { id }
1926
+ } = ctx;
1927
+ const { pm, file: file2 } = await findEntityAndCheckPermissions(
1928
+ userAbility,
1929
+ ACTIONS.read,
1930
+ FILE_MODEL_UID,
1931
+ id
1932
+ );
1933
+ const signedFile = await getService("file").signFileUrls(file2);
1934
+ ctx.body = await pm.sanitizeOutput(signedFile);
1935
+ },
1936
+ async destroy(ctx) {
1937
+ const { id } = ctx.params;
1938
+ const { userAbility } = ctx.state;
1939
+ const { pm, file: file2 } = await findEntityAndCheckPermissions(
1940
+ userAbility,
1941
+ ACTIONS.update,
1942
+ FILE_MODEL_UID,
1943
+ id
1944
+ );
1945
+ const [body] = await Promise.all([
1946
+ pm.sanitizeOutput(file2, { action: ACTIONS.read }),
1947
+ getService("upload").remove(file2)
1948
+ ]);
1949
+ ctx.body = body;
1950
+ }
1951
+ };
1952
+ const folderExists = async (folderId) => {
1953
+ if (fp.isNil(folderId)) {
1954
+ return true;
1955
+ }
1956
+ const exists2 = await getService("folder").exists({ id: folderId });
1957
+ return exists2;
1958
+ };
1959
+ const isFolderOrChild = (folderOrChild, folder2) => folderOrChild.path === folder2.path || folderOrChild.path.startsWith(`${folder2.path}/`);
1960
+ const NO_SLASH_REGEX = /^[^/]+$/;
1961
+ const NO_SPACES_AROUND = /^(?! ).+(?<! )$/;
1962
+ const isNameUniqueInFolder = (id) => {
1963
+ return async function test(name) {
1964
+ const { exists: exists2 } = getService("folder");
1965
+ const filters = { name, parent: this.parent.parent || null };
1966
+ if (id) {
1967
+ filters.id = { $ne: id };
1968
+ if (fp.isUndefined(name)) {
1969
+ const existingFolder = await strapi.db.query(FOLDER_MODEL_UID).findOne({ where: { id } });
1970
+ filters.name = fp.get("name", existingFolder);
1971
+ }
1972
+ }
1973
+ const doesExist = await exists2(filters);
1974
+ return !doesExist;
1975
+ };
1976
+ };
1977
+ const validateCreateFolderSchema = utils.yup.object().shape({
1978
+ name: utils.yup.string().min(1).matches(NO_SLASH_REGEX, "name cannot contain slashes").matches(NO_SPACES_AROUND, "name cannot start or end with a whitespace").required().test("is-folder-unique", "A folder with this name already exists", isNameUniqueInFolder()),
1979
+ parent: utils.yup.strapiID().nullable().test("folder-exists", "parent folder does not exist", folderExists)
1980
+ }).noUnknown().required();
1981
+ const validateUpdateFolderSchema = (id) => utils.yup.object().shape({
1982
+ name: utils.yup.string().min(1).matches(NO_SLASH_REGEX, "name cannot contain slashes").matches(NO_SPACES_AROUND, "name cannot start or end with a whitespace").test(
1983
+ "is-folder-unique",
1984
+ "A folder with this name already exists",
1985
+ isNameUniqueInFolder(id)
1986
+ ),
1987
+ parent: utils.yup.strapiID().nullable().test("folder-exists", "parent folder does not exist", folderExists).test(
1988
+ "dont-move-inside-self",
1989
+ "folder cannot be moved inside itself",
1990
+ async function test(parent) {
1991
+ if (fp.isNil(parent))
1992
+ return true;
1993
+ const destinationFolder = await strapi.db.query(FOLDER_MODEL_UID).findOne({
1994
+ select: ["path"],
1995
+ where: { id: parent }
1996
+ });
1997
+ const currentFolder = await strapi.db.query(FOLDER_MODEL_UID).findOne({
1998
+ select: ["path"],
1999
+ where: { id }
2000
+ });
2001
+ if (!destinationFolder || !currentFolder)
2002
+ return true;
2003
+ return !isFolderOrChild(destinationFolder, currentFolder);
2004
+ }
2005
+ )
2006
+ }).noUnknown().required();
2007
+ const validateCreateFolder = utils.validateYupSchema(validateCreateFolderSchema);
2008
+ const validateUpdateFolder = (id) => utils.validateYupSchema(validateUpdateFolderSchema(id));
2009
+ const adminFolder = {
2010
+ async findOne(ctx) {
2011
+ const { id } = ctx.params;
2012
+ const permissionsManager = strapi.service("admin::permission").createPermissionsManager({
2013
+ ability: ctx.state.userAbility,
2014
+ model: FOLDER_MODEL_UID
2015
+ });
2016
+ await permissionsManager.validateQuery(ctx.query);
2017
+ const query = await permissionsManager.sanitizeQuery(ctx.query);
2018
+ const { results } = await strapi.db.query(FOLDER_MODEL_UID).findPage(
2019
+ strapi.get("query-params").transform(
2020
+ FOLDER_MODEL_UID,
2021
+ fp.defaultsDeep(
2022
+ {
2023
+ filters: { id },
2024
+ populate: {
2025
+ children: {
2026
+ count: true
2027
+ },
2028
+ files: {
2029
+ count: true
2030
+ }
2031
+ }
2032
+ },
2033
+ query
2034
+ )
2035
+ )
2036
+ );
2037
+ if (results.length === 0) {
2038
+ return ctx.notFound("folder not found");
2039
+ }
2040
+ ctx.body = {
2041
+ data: await permissionsManager.sanitizeOutput(results[0])
2042
+ };
2043
+ },
2044
+ async find(ctx) {
2045
+ const permissionsManager = strapi.service("admin::permission").createPermissionsManager({
2046
+ ability: ctx.state.userAbility,
2047
+ model: FOLDER_MODEL_UID
2048
+ });
2049
+ await permissionsManager.validateQuery(ctx.query);
2050
+ const query = await permissionsManager.sanitizeQuery(ctx.query);
2051
+ const results = await strapi.db.query(FOLDER_MODEL_UID).findMany(
2052
+ strapi.get("query-params").transform(
2053
+ FOLDER_MODEL_UID,
2054
+ fp.defaultsDeep(
2055
+ {
2056
+ populate: {
2057
+ children: {
2058
+ count: true
2059
+ },
2060
+ files: {
2061
+ count: true
2062
+ }
2063
+ }
2064
+ },
2065
+ query
2066
+ )
2067
+ )
2068
+ );
2069
+ ctx.body = {
2070
+ data: await permissionsManager.sanitizeOutput(results)
2071
+ };
2072
+ },
2073
+ async create(ctx) {
2074
+ const { user } = ctx.state;
2075
+ const { body } = ctx.request;
2076
+ await validateCreateFolder(body);
2077
+ const folderService = getService("folder");
2078
+ const folder2 = await folderService.create(body, { user });
2079
+ const permissionsManager = strapi.service("admin::permission").createPermissionsManager({
2080
+ ability: ctx.state.userAbility,
2081
+ model: FOLDER_MODEL_UID
2082
+ });
2083
+ ctx.created({
2084
+ data: await permissionsManager.sanitizeOutput(folder2)
2085
+ });
2086
+ },
2087
+ async update(ctx) {
2088
+ const { id } = ctx.params;
2089
+ const { user } = ctx.state;
2090
+ const { body } = ctx.request;
2091
+ const permissionsManager = strapi.service("admin::permission").createPermissionsManager({
2092
+ ability: ctx.state.userAbility,
2093
+ model: FOLDER_MODEL_UID
2094
+ });
2095
+ await validateUpdateFolder(id)(body);
2096
+ const folderService = getService("folder");
2097
+ const updatedFolder = await folderService.update(id, body, { user });
2098
+ if (!updatedFolder) {
2099
+ return ctx.notFound("folder not found");
2100
+ }
2101
+ ctx.body = {
2102
+ data: await permissionsManager.sanitizeOutput(updatedFolder)
2103
+ };
2104
+ },
2105
+ async getStructure(ctx) {
2106
+ const { getStructure: getStructure2 } = getService("folder");
2107
+ const structure = await getStructure2();
2108
+ ctx.body = {
2109
+ data: structure
2110
+ };
2111
+ }
2112
+ };
2113
+ const validateDeleteManyFoldersFilesSchema = utils.yup.object().shape({
2114
+ fileIds: utils.yup.array().of(utils.yup.strapiID().required()),
2115
+ folderIds: utils.yup.array().of(utils.yup.strapiID().required())
2116
+ }).noUnknown().required();
2117
+ const validateStructureMoveManyFoldersFilesSchema = utils.yup.object().shape({
2118
+ destinationFolderId: utils.yup.strapiID().nullable().defined().test("folder-exists", "destination folder does not exist", folderExists),
2119
+ fileIds: utils.yup.array().of(utils.yup.strapiID().required()),
2120
+ folderIds: utils.yup.array().of(utils.yup.strapiID().required())
2121
+ }).noUnknown().required();
2122
+ const validateDuplicatesMoveManyFoldersFilesSchema = utils.yup.object().test("are-folders-unique", "some folders already exist", async function(value) {
2123
+ const { folderIds, destinationFolderId } = value;
2124
+ if (fp.isEmpty(folderIds))
2125
+ return true;
2126
+ const folders = await strapi.db.query(FOLDER_MODEL_UID).findMany({
2127
+ select: ["name"],
2128
+ where: { id: { $in: folderIds } }
2129
+ });
2130
+ const existingFolders = await strapi.db.query(FOLDER_MODEL_UID).findMany({
2131
+ select: ["name"],
2132
+ where: { parent: { id: destinationFolderId } }
2133
+ });
2134
+ const duplicatedNames = fp.intersection(fp.map("name", folders), fp.map("name", existingFolders));
2135
+ if (duplicatedNames.length > 0) {
2136
+ return this.createError({
2137
+ message: `some folders already exists: ${duplicatedNames.join(", ")}`
2138
+ });
2139
+ }
2140
+ return true;
2141
+ });
2142
+ const validateMoveFoldersNotInsideThemselvesSchema = utils.yup.object().test(
2143
+ "dont-move-inside-self",
2144
+ "folders cannot be moved inside themselves or one of its children",
2145
+ async function(value) {
2146
+ const { folderIds, destinationFolderId } = value;
2147
+ if (destinationFolderId === null || fp.isEmpty(folderIds))
2148
+ return true;
2149
+ const destinationFolder = await strapi.db.query(FOLDER_MODEL_UID).findOne({
2150
+ select: ["path"],
2151
+ where: { id: destinationFolderId }
2152
+ });
2153
+ const folders = await strapi.db.query(FOLDER_MODEL_UID).findMany({
2154
+ select: ["name", "path"],
2155
+ where: { id: { $in: folderIds } }
2156
+ });
2157
+ const unmovableFoldersNames = folders.filter((folder2) => isFolderOrChild(destinationFolder, folder2)).map((f) => f.name);
2158
+ if (unmovableFoldersNames.length > 0) {
2159
+ return this.createError({
2160
+ message: `folders cannot be moved inside themselves or one of its children: ${unmovableFoldersNames.join(
2161
+ ", "
2162
+ )}`
2163
+ });
2164
+ }
2165
+ return true;
2166
+ }
2167
+ );
2168
+ const validateDeleteManyFoldersFiles = utils.validateYupSchema(
2169
+ validateDeleteManyFoldersFilesSchema
2170
+ );
2171
+ async function validateMoveManyFoldersFiles(body) {
2172
+ await utils.validateYupSchema(validateStructureMoveManyFoldersFilesSchema)(body);
2173
+ await utils.validateYupSchema(validateDuplicatesMoveManyFoldersFilesSchema)(body);
2174
+ await utils.validateYupSchema(validateMoveFoldersNotInsideThemselvesSchema)(body);
2175
+ }
2176
+ const adminFolderFile = {
2177
+ async deleteMany(ctx) {
2178
+ const { body } = ctx.request;
2179
+ const {
2180
+ state: { userAbility }
2181
+ } = ctx;
2182
+ const pmFolder = strapi.service("admin::permission").createPermissionsManager({
2183
+ ability: ctx.state.userAbility,
2184
+ model: FOLDER_MODEL_UID
2185
+ });
2186
+ const pmFile = strapi.service("admin::permission").createPermissionsManager({
2187
+ ability: userAbility,
2188
+ action: ACTIONS.read,
2189
+ model: FILE_MODEL_UID
2190
+ });
2191
+ await validateDeleteManyFoldersFiles(body);
2192
+ const fileService = getService("file");
2193
+ const folderService = getService("folder");
2194
+ const deletedFiles = await fileService.deleteByIds(body.fileIds);
2195
+ const {
2196
+ folders: deletedFolders,
2197
+ totalFolderNumber,
2198
+ totalFileNumber
2199
+ } = await folderService.deleteByIds(body.folderIds);
2200
+ if (deletedFiles.length + deletedFolders.length > 1) {
2201
+ strapi.telemetry.send("didBulkDeleteMediaLibraryElements", {
2202
+ eventProperties: {
2203
+ rootFolderNumber: deletedFolders.length,
2204
+ rootAssetNumber: deletedFiles.length,
2205
+ totalFolderNumber,
2206
+ totalAssetNumber: totalFileNumber + deletedFiles.length
2207
+ }
2208
+ });
2209
+ }
2210
+ ctx.body = {
2211
+ data: {
2212
+ files: await pmFile.sanitizeOutput(deletedFiles),
2213
+ folders: await pmFolder.sanitizeOutput(deletedFolders)
2214
+ }
2215
+ };
2216
+ },
2217
+ async moveMany(ctx) {
2218
+ const { body } = ctx.request;
2219
+ const {
2220
+ state: { userAbility }
2221
+ } = ctx;
2222
+ const pmFolder = strapi.service("admin::permission").createPermissionsManager({
2223
+ ability: ctx.state.userAbility,
2224
+ model: FOLDER_MODEL_UID
2225
+ });
2226
+ const pmFile = strapi.service("admin::permission").createPermissionsManager({
2227
+ ability: userAbility,
2228
+ action: ACTIONS.read,
2229
+ model: FILE_MODEL_UID
2230
+ });
2231
+ await validateMoveManyFoldersFiles(body);
2232
+ const { folderIds = [], fileIds = [], destinationFolderId } = body;
2233
+ let totalFolderNumber = 0;
2234
+ let totalFileNumber = 0;
2235
+ const trx = await strapi.db.transaction();
2236
+ try {
2237
+ const existingFolders = await strapi.db.queryBuilder(FOLDER_MODEL_UID).select(["id", "pathId", "path"]).where({ id: { $in: folderIds } }).transacting(trx.get()).forUpdate().execute();
2238
+ const existingFiles = await strapi.db.queryBuilder(FILE_MODEL_UID).select(["id"]).where({ id: { $in: fileIds } }).transacting(trx.get()).forUpdate().execute();
2239
+ let destinationFolderPath = "/";
2240
+ if (destinationFolderId !== null) {
2241
+ const destinationFolder = await strapi.db.queryBuilder(FOLDER_MODEL_UID).select("path").where({ id: destinationFolderId }).transacting(trx.get()).first().execute();
2242
+ destinationFolderPath = destinationFolder.path;
2243
+ }
2244
+ const fileTable = strapi.getModel(FILE_MODEL_UID).collectionName;
2245
+ const folderTable = strapi.getModel(FOLDER_MODEL_UID).collectionName;
2246
+ const folderPathColName = (
2247
+ // @ts-expect-error - no dynamic typings for the models
2248
+ strapi.db.metadata.get(FILE_MODEL_UID).attributes.folderPath.columnName
2249
+ );
2250
+ const pathColName = strapi.db.metadata.get(FOLDER_MODEL_UID).attributes.path.columnName;
2251
+ if (existingFolders.length > 0) {
2252
+ const { joinTable } = strapi.db.metadata.get(FOLDER_MODEL_UID).attributes.parent;
2253
+ await strapi.db.queryBuilder(joinTable.name).transacting(trx.get()).delete().where({ [joinTable.joinColumn.name]: { $in: folderIds } }).execute();
2254
+ if (destinationFolderId !== null) {
2255
+ await strapi.db.queryBuilder(joinTable.name).transacting(trx.get()).insert(
2256
+ existingFolders.map((folder2) => ({
2257
+ [joinTable.inverseJoinColumn.name]: destinationFolderId,
2258
+ [joinTable.joinColumn.name]: folder2.id
2259
+ }))
2260
+ ).execute();
2261
+ }
2262
+ for (const existingFolder of existingFolders) {
2263
+ let replaceQuery;
2264
+ switch (strapi.db.dialect.client) {
2265
+ case "sqlite":
2266
+ replaceQuery = "? || SUBSTRING(??, ?)";
2267
+ break;
2268
+ case "postgres":
2269
+ replaceQuery = "CONCAT(?::TEXT, SUBSTRING(??, ?::INTEGER))";
2270
+ break;
2271
+ default:
2272
+ replaceQuery = "CONCAT(?, SUBSTRING(??, ?))";
2273
+ }
2274
+ totalFolderNumber = await strapi.db.getConnection(folderTable).transacting(trx.get()).where(pathColName, existingFolder.path).orWhere(pathColName, "like", `${existingFolder.path}/%`).update(
2275
+ pathColName,
2276
+ strapi.db.connection.raw(replaceQuery, [
2277
+ utils.strings.joinBy("/", destinationFolderPath, `${existingFolder.pathId}`),
2278
+ pathColName,
2279
+ existingFolder.path.length + 1
2280
+ ])
2281
+ );
2282
+ totalFileNumber = await strapi.db.getConnection(fileTable).transacting(trx.get()).where(folderPathColName, existingFolder.path).orWhere(folderPathColName, "like", `${existingFolder.path}/%`).update(
2283
+ folderPathColName,
2284
+ strapi.db.connection.raw(replaceQuery, [
2285
+ utils.strings.joinBy("/", destinationFolderPath, `${existingFolder.pathId}`),
2286
+ folderPathColName,
2287
+ existingFolder.path.length + 1
2288
+ ])
2289
+ );
2290
+ }
2291
+ }
2292
+ if (existingFiles.length > 0) {
2293
+ const fileJoinTable = strapi.db.metadata.get(FILE_MODEL_UID).attributes.folder.joinTable;
2294
+ await strapi.db.queryBuilder(fileJoinTable.name).transacting(trx.get()).delete().where({ [fileJoinTable.joinColumn.name]: { $in: fileIds } }).execute();
2295
+ if (destinationFolderId !== null) {
2296
+ await strapi.db.queryBuilder(fileJoinTable.name).transacting(trx.get()).insert(
2297
+ existingFiles.map((file2) => ({
2298
+ [fileJoinTable.inverseJoinColumn.name]: destinationFolderId,
2299
+ [fileJoinTable.joinColumn.name]: file2.id
2300
+ }))
2301
+ ).execute();
2302
+ }
2303
+ await strapi.db.getConnection(fileTable).transacting(trx.get()).whereIn("id", fileIds).update(folderPathColName, destinationFolderPath);
2304
+ }
2305
+ await trx.commit();
2306
+ } catch (e) {
2307
+ await trx.rollback();
2308
+ throw e;
2309
+ }
2310
+ const updatedFolders = await strapi.db.query(FOLDER_MODEL_UID).findMany({
2311
+ where: { id: { $in: folderIds } }
2312
+ });
2313
+ const updatedFiles = await strapi.db.query(FILE_MODEL_UID).findMany({
2314
+ where: { id: { $in: fileIds } }
2315
+ });
2316
+ strapi.telemetry.send("didBulkMoveMediaLibraryElements", {
2317
+ eventProperties: {
2318
+ rootFolderNumber: updatedFolders.length,
2319
+ rootAssetNumber: updatedFiles.length,
2320
+ totalFolderNumber,
2321
+ totalAssetNumber: totalFileNumber + updatedFiles.length
2322
+ }
2323
+ });
2324
+ ctx.body = {
2325
+ data: {
2326
+ files: await pmFile.sanitizeOutput(updatedFiles),
2327
+ folders: await pmFolder.sanitizeOutput(updatedFolders)
2328
+ }
2329
+ };
2330
+ }
2331
+ };
2332
+ const settingsSchema = utils.yup.object({
2333
+ sizeOptimization: utils.yup.boolean().required(),
2334
+ responsiveDimensions: utils.yup.boolean().required(),
2335
+ autoOrientation: utils.yup.boolean()
2336
+ });
2337
+ const validateSettings = utils.validateYupSchema(settingsSchema);
2338
+ const adminSettings = {
2339
+ async updateSettings(ctx) {
2340
+ const {
2341
+ request: { body },
2342
+ state: { userAbility }
2343
+ } = ctx;
2344
+ if (userAbility.cannot(ACTIONS.readSettings, FILE_MODEL_UID)) {
2345
+ return ctx.forbidden();
2346
+ }
2347
+ const data = await validateSettings(body);
2348
+ await getService("upload").setSettings(data);
2349
+ ctx.body = { data };
2350
+ },
2351
+ async getSettings(ctx) {
2352
+ const {
2353
+ state: { userAbility }
2354
+ } = ctx;
2355
+ if (userAbility.cannot(ACTIONS.readSettings, FILE_MODEL_UID)) {
2356
+ return ctx.forbidden();
2357
+ }
2358
+ const data = await getService("upload").getSettings();
2359
+ ctx.body = { data };
2360
+ }
2361
+ };
2362
+ const fileInfoSchema$1 = utils.yup.object({
2363
+ name: utils.yup.string().nullable(),
2364
+ alternativeText: utils.yup.string().nullable(),
2365
+ caption: utils.yup.string().nullable(),
2366
+ folder: utils.yup.strapiID().nullable().test("folder-exists", "the folder does not exist", async (folderId) => {
2367
+ if (fp.isNil(folderId)) {
2368
+ return true;
2369
+ }
2370
+ const exists2 = await getService("folder").exists({ id: folderId });
2371
+ return exists2;
2372
+ })
2373
+ });
2374
+ const uploadSchema$1 = utils.yup.object({
2375
+ fileInfo: fileInfoSchema$1
2376
+ });
2377
+ const multiUploadSchema$1 = utils.yup.object({
2378
+ fileInfo: utils.yup.array().of(fileInfoSchema$1)
2379
+ });
2380
+ const validateUploadBody$1 = (data = {}, isMulti = false) => {
2381
+ const schema = isMulti ? multiUploadSchema$1 : uploadSchema$1;
2382
+ return utils.validateYupSchema(schema, { strict: false })(data);
2383
+ };
2384
+ const adminUpload = {
2385
+ async updateFileInfo(ctx) {
2386
+ const {
2387
+ state: { userAbility, user },
2388
+ query: { id },
2389
+ request: { body }
2390
+ } = ctx;
2391
+ if (typeof id !== "string") {
2392
+ throw new utils.errors.ValidationError("File id is required");
2393
+ }
2394
+ const uploadService = getService("upload");
2395
+ const { pm } = await findEntityAndCheckPermissions(
2396
+ userAbility,
2397
+ ACTIONS.update,
2398
+ FILE_MODEL_UID,
2399
+ id
2400
+ );
2401
+ const data = await validateUploadBody$1(body);
2402
+ const file2 = await uploadService.updateFileInfo(id, data.fileInfo, { user });
2403
+ ctx.body = await pm.sanitizeOutput(file2, { action: ACTIONS.read });
2404
+ },
2405
+ async replaceFile(ctx) {
2406
+ const {
2407
+ state: { userAbility, user },
2408
+ query: { id },
2409
+ request: { body, files: { files } = {} }
2410
+ } = ctx;
2411
+ if (typeof id !== "string") {
2412
+ throw new utils.errors.ValidationError("File id is required");
2413
+ }
2414
+ const uploadService = getService("upload");
2415
+ const { pm } = await findEntityAndCheckPermissions(
2416
+ userAbility,
2417
+ ACTIONS.update,
2418
+ FILE_MODEL_UID,
2419
+ id
2420
+ );
2421
+ if (Array.isArray(files)) {
2422
+ throw new utils.errors.ApplicationError("Cannot replace a file with multiple ones");
2423
+ }
2424
+ const data = await validateUploadBody$1(body);
2425
+ const replacedFile = await uploadService.replace(id, { data, file: files }, { user });
2426
+ const signedFile = await getService("file").signFileUrls(replacedFile);
2427
+ ctx.body = await pm.sanitizeOutput(signedFile, { action: ACTIONS.read });
2428
+ },
2429
+ async uploadFiles(ctx) {
2430
+ const {
2431
+ state: { userAbility, user },
2432
+ request: { body, files: { files } = {} }
2433
+ } = ctx;
2434
+ const uploadService = getService("upload");
2435
+ const pm = strapi.service("admin::permission").createPermissionsManager({
2436
+ ability: userAbility,
2437
+ action: ACTIONS.create,
2438
+ model: FILE_MODEL_UID
2439
+ });
2440
+ if (!pm.isAllowed) {
2441
+ return ctx.forbidden();
2442
+ }
2443
+ const data = await validateUploadBody$1(body);
2444
+ const uploadedFiles = await uploadService.upload({ data, files }, { user });
2445
+ const signedFiles = await utils.async.map(uploadedFiles, getService("file").signFileUrls);
2446
+ ctx.body = await pm.sanitizeOutput(signedFiles, { action: ACTIONS.read });
2447
+ ctx.status = 201;
2448
+ },
2449
+ // TODO: split into multiple endpoints
2450
+ async upload(ctx) {
2451
+ const {
2452
+ query: { id },
2453
+ request: { files: { files } = {} }
2454
+ } = ctx;
2455
+ if (___default.default.isEmpty(files) || !Array.isArray(files) && files.size === 0) {
2456
+ if (id) {
2457
+ return this.updateFileInfo(ctx);
2458
+ }
2459
+ throw new utils.errors.ApplicationError("Files are empty");
2460
+ }
2461
+ await (id ? this.replaceFile : this.uploadFiles)(ctx);
2462
+ }
2463
+ };
2464
+ const fileInfoSchema = utils.yup.object({
2465
+ name: utils.yup.string().nullable(),
2466
+ alternativeText: utils.yup.string().nullable(),
2467
+ caption: utils.yup.string().nullable()
2468
+ }).noUnknown();
2469
+ const uploadSchema = utils.yup.object({
2470
+ fileInfo: fileInfoSchema
2471
+ });
2472
+ const multiUploadSchema = utils.yup.object({
2473
+ fileInfo: utils.yup.array().of(fileInfoSchema)
2474
+ });
2475
+ const validateUploadBody = (data = {}, isMulti = false) => {
2476
+ const schema = isMulti ? multiUploadSchema : uploadSchema;
2477
+ return utils.validateYupSchema(schema, { strict: false })(data);
2478
+ };
2479
+ const { ValidationError } = utils__default.default.errors;
2480
+ const contentApi = ({ strapi: strapi2 }) => {
2481
+ const sanitizeOutput = async (data, ctx) => {
2482
+ const schema = strapi2.getModel(FILE_MODEL_UID);
2483
+ const { auth } = ctx.state;
2484
+ return strapi2.contentAPI.sanitize.output(data, schema, { auth });
2485
+ };
2486
+ const validateQuery = async (data, ctx) => {
2487
+ const schema = strapi2.getModel(FILE_MODEL_UID);
2488
+ const { auth } = ctx.state;
2489
+ return strapi2.contentAPI.validate.query(data, schema, { auth });
2490
+ };
2491
+ const sanitizeQuery = async (data, ctx) => {
2492
+ const schema = strapi2.getModel(FILE_MODEL_UID);
2493
+ const { auth } = ctx.state;
2494
+ return strapi2.contentAPI.sanitize.query(data, schema, { auth });
2495
+ };
2496
+ return {
2497
+ async find(ctx) {
2498
+ await validateQuery(ctx.query, ctx);
2499
+ const sanitizedQuery = await sanitizeQuery(ctx.query, ctx);
2500
+ const files = await getService("upload").findMany(sanitizedQuery);
2501
+ ctx.body = await sanitizeOutput(files, ctx);
2502
+ },
2503
+ async findOne(ctx) {
2504
+ const {
2505
+ params: { id }
2506
+ } = ctx;
2507
+ await validateQuery(ctx.query, ctx);
2508
+ const sanitizedQuery = await sanitizeQuery(ctx.query, ctx);
2509
+ const file2 = await getService("upload").findOne(id, sanitizedQuery.populate);
2510
+ if (!file2) {
2511
+ return ctx.notFound("file.notFound");
2512
+ }
2513
+ ctx.body = await sanitizeOutput(file2, ctx);
2514
+ },
2515
+ async destroy(ctx) {
2516
+ const {
2517
+ params: { id }
2518
+ } = ctx;
2519
+ const file2 = await getService("upload").findOne(id);
2520
+ if (!file2) {
2521
+ return ctx.notFound("file.notFound");
2522
+ }
2523
+ await getService("upload").remove(file2);
2524
+ ctx.body = await sanitizeOutput(file2, ctx);
2525
+ },
2526
+ async updateFileInfo(ctx) {
2527
+ const {
2528
+ query: { id },
2529
+ request: { body }
2530
+ } = ctx;
2531
+ const data = await validateUploadBody(body);
2532
+ if (!id || typeof id !== "string" && typeof id !== "number") {
2533
+ throw new ValidationError("File id is required and must be a single value");
2534
+ }
2535
+ const result = await getService("upload").updateFileInfo(id, data.fileInfo);
2536
+ ctx.body = await sanitizeOutput(result, ctx);
2537
+ },
2538
+ async replaceFile(ctx) {
2539
+ const {
2540
+ query: { id },
2541
+ request: { body, files: { files } = {} }
2542
+ } = ctx;
2543
+ if (Array.isArray(files)) {
2544
+ throw new ValidationError("Cannot replace a file with multiple ones");
2545
+ }
2546
+ if (!id || typeof id !== "string" && typeof id !== "number") {
2547
+ throw new ValidationError("File id is required and must be a single value");
2548
+ }
2549
+ const data = await validateUploadBody(body);
2550
+ const replacedFiles = await getService("upload").replace(id, { data, file: files });
2551
+ ctx.body = await sanitizeOutput(replacedFiles, ctx);
2552
+ },
2553
+ async uploadFiles(ctx) {
2554
+ const {
2555
+ request: { body, files: { files } = {} }
2556
+ } = ctx;
2557
+ const data = await validateUploadBody(body, Array.isArray(files));
2558
+ const apiUploadFolderService = getService("api-upload-folder");
2559
+ const apiUploadFolder2 = await apiUploadFolderService.getAPIUploadFolder();
2560
+ if (Array.isArray(files)) {
2561
+ data.fileInfo = data.fileInfo || [];
2562
+ data.fileInfo = files.map((_f, i) => ({ ...data.fileInfo[i], folder: apiUploadFolder2.id }));
2563
+ } else {
2564
+ data.fileInfo = { ...data.fileInfo, folder: apiUploadFolder2.id };
2565
+ }
2566
+ const uploadedFiles = await getService("upload").upload({
2567
+ data,
2568
+ files
2569
+ });
2570
+ ctx.body = await sanitizeOutput(uploadedFiles, ctx);
2571
+ ctx.status = 201;
2572
+ },
2573
+ // TODO: split into multiple endpoints
2574
+ async upload(ctx) {
2575
+ const {
2576
+ query: { id },
2577
+ request: { files: { files } = {} }
2578
+ } = ctx;
2579
+ if (___default.default.isEmpty(files) || !Array.isArray(files) && files.size === 0) {
2580
+ if (id) {
2581
+ return this.updateFileInfo(ctx);
2582
+ }
2583
+ throw new ValidationError("Files are empty");
2584
+ }
2585
+ await (id ? this.replaceFile : this.uploadFiles)(ctx);
2586
+ }
2587
+ };
2588
+ };
2589
+ const configSchema = utils.yup.object({
2590
+ pageSize: utils.yup.number().required(),
2591
+ sort: utils.yup.mixed().oneOf(ALLOWED_SORT_STRINGS)
2592
+ });
2593
+ const validateViewConfiguration = utils.validateYupSchema(configSchema);
2594
+ const viewConfiguration = {
2595
+ async updateViewConfiguration(ctx) {
2596
+ const {
2597
+ request: { body },
2598
+ state: { userAbility }
2599
+ } = ctx;
2600
+ if (userAbility.cannot(ACTIONS.configureView)) {
2601
+ return ctx.forbidden();
2602
+ }
2603
+ const data = await validateViewConfiguration(body);
2604
+ await getService("upload").setConfiguration(data);
2605
+ ctx.body = { data };
2606
+ },
2607
+ async findViewConfiguration(ctx) {
2608
+ const data = await getService("upload").getConfiguration();
2609
+ ctx.body = { data };
2610
+ }
2611
+ };
2612
+ const controllers = {
2613
+ "admin-file": adminFile,
2614
+ "admin-folder": adminFolder,
2615
+ "admin-folder-file": adminFolderFile,
2616
+ "admin-settings": adminSettings,
2617
+ "admin-upload": adminUpload,
2618
+ "content-api": contentApi,
2619
+ "view-configuration": viewConfiguration
2620
+ };
2621
+ const index = () => ({
2622
+ register,
2623
+ bootstrap,
2624
+ config,
2625
+ routes,
2626
+ controllers,
2627
+ contentTypes,
2628
+ services
2629
+ });
2630
+ exports.FILE_MODEL_UID = FILE_MODEL_UID;
2631
+ exports.index = index;
2632
+ //# sourceMappingURL=index-CRhNpXFr.js.map