@evfrenkel/decap-cms-core 3.13.0-image-conversions.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (306) hide show
  1. package/README.md +9 -0
  2. package/dist/20.decap-cms-core.js +2 -0
  3. package/dist/20.decap-cms-core.js.map +1 -0
  4. package/dist/3802306e7b58a11862fb.wasm +0 -0
  5. package/dist/@evfrenkel/20.decap-cms-core.js +2 -0
  6. package/dist/@evfrenkel/20.decap-cms-core.js.map +1 -0
  7. package/dist/@evfrenkel/decap-cms-core.js +44 -0
  8. package/dist/@evfrenkel/decap-cms-core.js.LICENSE.txt +126 -0
  9. package/dist/@evfrenkel/decap-cms-core.js.map +1 -0
  10. package/dist/decap-cms-core.js +47 -0
  11. package/dist/decap-cms-core.js.LICENSE.txt +126 -0
  12. package/dist/decap-cms-core.js.map +1 -0
  13. package/dist/esm/actions/auth.js +97 -0
  14. package/dist/esm/actions/collections.js +15 -0
  15. package/dist/esm/actions/config.js +503 -0
  16. package/dist/esm/actions/deploys.js +80 -0
  17. package/dist/esm/actions/editorialWorkflow.js +480 -0
  18. package/dist/esm/actions/entries.js +876 -0
  19. package/dist/esm/actions/media.js +147 -0
  20. package/dist/esm/actions/mediaLibrary.js +599 -0
  21. package/dist/esm/actions/notifications.js +21 -0
  22. package/dist/esm/actions/search.js +149 -0
  23. package/dist/esm/actions/status.js +74 -0
  24. package/dist/esm/actions/waitUntil.js +32 -0
  25. package/dist/esm/backend.js +1090 -0
  26. package/dist/esm/bootstrap.js +101 -0
  27. package/dist/esm/components/App/App.js +296 -0
  28. package/dist/esm/components/App/Header.js +172 -0
  29. package/dist/esm/components/App/NotFoundPage.js +19 -0
  30. package/dist/esm/components/Collection/Collection.js +198 -0
  31. package/dist/esm/components/Collection/CollectionControls.js +47 -0
  32. package/dist/esm/components/Collection/CollectionSearch.js +222 -0
  33. package/dist/esm/components/Collection/CollectionTop.js +68 -0
  34. package/dist/esm/components/Collection/ControlButton.js +17 -0
  35. package/dist/esm/components/Collection/Entries/Entries.js +73 -0
  36. package/dist/esm/components/Collection/Entries/EntriesCollection.js +241 -0
  37. package/dist/esm/components/Collection/Entries/EntriesSearch.js +113 -0
  38. package/dist/esm/components/Collection/Entries/EntryCard.js +192 -0
  39. package/dist/esm/components/Collection/Entries/EntryListing.js +143 -0
  40. package/dist/esm/components/Collection/FilterControl.js +33 -0
  41. package/dist/esm/components/Collection/GroupControl.js +33 -0
  42. package/dist/esm/components/Collection/NestedCollection.js +308 -0
  43. package/dist/esm/components/Collection/Sidebar.js +91 -0
  44. package/dist/esm/components/Collection/SortControl.js +59 -0
  45. package/dist/esm/components/Collection/ViewStyleControl.js +41 -0
  46. package/dist/esm/components/Editor/Editor.js +466 -0
  47. package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +398 -0
  48. package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +254 -0
  49. package/dist/esm/components/Editor/EditorControlPane/Widget.js +374 -0
  50. package/dist/esm/components/Editor/EditorInterface.js +386 -0
  51. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +47 -0
  52. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +67 -0
  53. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +306 -0
  54. package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +27 -0
  55. package/dist/esm/components/Editor/EditorToolbar.js +554 -0
  56. package/dist/esm/components/Editor/withWorkflow.js +56 -0
  57. package/dist/esm/components/EditorWidgets/Unknown/UnknownControl.js +18 -0
  58. package/dist/esm/components/EditorWidgets/Unknown/UnknownPreview.js +20 -0
  59. package/dist/esm/components/EditorWidgets/index.js +4 -0
  60. package/dist/esm/components/MediaLibrary/EmptyMessage.js +22 -0
  61. package/dist/esm/components/MediaLibrary/MediaLibrary.js +446 -0
  62. package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +92 -0
  63. package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +99 -0
  64. package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +198 -0
  65. package/dist/esm/components/MediaLibrary/MediaLibraryHeader.js +34 -0
  66. package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +156 -0
  67. package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +51 -0
  68. package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +124 -0
  69. package/dist/esm/components/UI/DragDrop.js +67 -0
  70. package/dist/esm/components/UI/ErrorBoundary.js +170 -0
  71. package/dist/esm/components/UI/FileUploadButton.js +27 -0
  72. package/dist/esm/components/UI/Modal.js +104 -0
  73. package/dist/esm/components/UI/Notifications.js +62 -0
  74. package/dist/esm/components/UI/SettingsDropdown.js +107 -0
  75. package/dist/esm/components/UI/index.js +6 -0
  76. package/dist/esm/components/Workflow/Workflow.js +133 -0
  77. package/dist/esm/components/Workflow/WorkflowCard.js +128 -0
  78. package/dist/esm/components/Workflow/WorkflowList.js +204 -0
  79. package/dist/esm/constants/collectionTypes.js +2 -0
  80. package/dist/esm/constants/collectionViews.js +2 -0
  81. package/dist/esm/constants/commitProps.js +2 -0
  82. package/dist/esm/constants/configSchema.js +695 -0
  83. package/dist/esm/constants/fieldInference.js +57 -0
  84. package/dist/esm/constants/publishModes.js +18 -0
  85. package/dist/esm/constants/validationErrorTypes.js +6 -0
  86. package/dist/esm/formats/formats.js +83 -0
  87. package/dist/esm/formats/frontmatter.js +146 -0
  88. package/dist/esm/formats/helpers.js +12 -0
  89. package/dist/esm/formats/json.js +8 -0
  90. package/dist/esm/formats/toml.js +32 -0
  91. package/dist/esm/formats/yaml.js +60 -0
  92. package/dist/esm/index.js +7 -0
  93. package/dist/esm/integrations/index.js +28 -0
  94. package/dist/esm/integrations/providers/algolia/implementation.js +174 -0
  95. package/dist/esm/integrations/providers/assetStore/implementation.js +165 -0
  96. package/dist/esm/lib/consoleError.js +3 -0
  97. package/dist/esm/lib/formatters.js +201 -0
  98. package/dist/esm/lib/i18n.js +372 -0
  99. package/dist/esm/lib/imageTransformations.js +143 -0
  100. package/dist/esm/lib/phrases.js +6 -0
  101. package/dist/esm/lib/polyfill.js +8 -0
  102. package/dist/esm/lib/registry.js +332 -0
  103. package/dist/esm/lib/serializeEntryValues.js +67 -0
  104. package/dist/esm/lib/stega.js +142 -0
  105. package/dist/esm/lib/textHelper.js +9 -0
  106. package/dist/esm/lib/urlHelper.js +129 -0
  107. package/dist/esm/mediaLibrary.js +37 -0
  108. package/dist/esm/reducers/auth.js +27 -0
  109. package/dist/esm/reducers/collections.js +428 -0
  110. package/dist/esm/reducers/combinedReducer.js +8 -0
  111. package/dist/esm/reducers/config.js +29 -0
  112. package/dist/esm/reducers/cursors.js +31 -0
  113. package/dist/esm/reducers/deploys.js +50 -0
  114. package/dist/esm/reducers/editorialWorkflow.js +83 -0
  115. package/dist/esm/reducers/entries.js +568 -0
  116. package/dist/esm/reducers/entryDraft.js +211 -0
  117. package/dist/esm/reducers/globalUI.js +25 -0
  118. package/dist/esm/reducers/index.js +66 -0
  119. package/dist/esm/reducers/integrations.js +53 -0
  120. package/dist/esm/reducers/mediaLibrary.js +252 -0
  121. package/dist/esm/reducers/medias.js +68 -0
  122. package/dist/esm/reducers/notifications.js +23 -0
  123. package/dist/esm/reducers/search.js +92 -0
  124. package/dist/esm/reducers/status.js +30 -0
  125. package/dist/esm/redux/index.js +7 -0
  126. package/dist/esm/redux/middleware/waitUntilAction.js +48 -0
  127. package/dist/esm/routing/history.js +12 -0
  128. package/dist/esm/types/diacritics.d.js +0 -0
  129. package/dist/esm/types/global.d.js +1 -0
  130. package/dist/esm/types/immutable.js +7 -0
  131. package/dist/esm/types/redux.js +14 -0
  132. package/dist/esm/types/tomlify-j0.4.d.js +0 -0
  133. package/dist/esm/valueObjects/AssetProxy.js +44 -0
  134. package/dist/esm/valueObjects/EditorComponent.js +34 -0
  135. package/dist/esm/valueObjects/Entry.js +20 -0
  136. package/index.d.ts +662 -0
  137. package/package.json +100 -0
  138. package/src/__tests__/backend.spec.js +1223 -0
  139. package/src/actions/__tests__/config.spec.js +1023 -0
  140. package/src/actions/__tests__/editorialWorkflow.spec.js +216 -0
  141. package/src/actions/__tests__/entries.spec.js +610 -0
  142. package/src/actions/__tests__/media.spec.ts +171 -0
  143. package/src/actions/__tests__/mediaLibrary.spec.js +462 -0
  144. package/src/actions/__tests__/search.spec.js +209 -0
  145. package/src/actions/auth.ts +127 -0
  146. package/src/actions/collections.ts +18 -0
  147. package/src/actions/config.ts +574 -0
  148. package/src/actions/deploys.ts +105 -0
  149. package/src/actions/editorialWorkflow.ts +567 -0
  150. package/src/actions/entries.ts +1070 -0
  151. package/src/actions/media.ts +139 -0
  152. package/src/actions/mediaLibrary.ts +639 -0
  153. package/src/actions/notifications.ts +36 -0
  154. package/src/actions/search.ts +221 -0
  155. package/src/actions/status.ts +99 -0
  156. package/src/actions/waitUntil.ts +49 -0
  157. package/src/backend.ts +1411 -0
  158. package/src/bootstrap.js +104 -0
  159. package/src/components/App/App.js +295 -0
  160. package/src/components/App/Header.js +291 -0
  161. package/src/components/App/NotFoundPage.js +23 -0
  162. package/src/components/Collection/Collection.js +210 -0
  163. package/src/components/Collection/CollectionControls.js +58 -0
  164. package/src/components/Collection/CollectionSearch.js +243 -0
  165. package/src/components/Collection/CollectionTop.js +81 -0
  166. package/src/components/Collection/ControlButton.js +27 -0
  167. package/src/components/Collection/Entries/Entries.js +82 -0
  168. package/src/components/Collection/Entries/EntriesCollection.js +277 -0
  169. package/src/components/Collection/Entries/EntriesSearch.js +102 -0
  170. package/src/components/Collection/Entries/EntryCard.js +256 -0
  171. package/src/components/Collection/Entries/EntryListing.js +151 -0
  172. package/src/components/Collection/Entries/__tests__/EntriesCollection.spec.js +163 -0
  173. package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +46 -0
  174. package/src/components/Collection/FilterControl.js +39 -0
  175. package/src/components/Collection/GroupControl.js +39 -0
  176. package/src/components/Collection/NestedCollection.js +330 -0
  177. package/src/components/Collection/Sidebar.js +136 -0
  178. package/src/components/Collection/SortControl.js +68 -0
  179. package/src/components/Collection/ViewStyleControl.js +52 -0
  180. package/src/components/Collection/__tests__/Collection.spec.js +75 -0
  181. package/src/components/Collection/__tests__/NestedCollection.spec.js +445 -0
  182. package/src/components/Collection/__tests__/Sidebar.spec.js +87 -0
  183. package/src/components/Collection/__tests__/__snapshots__/Collection.spec.js.snap +144 -0
  184. package/src/components/Collection/__tests__/__snapshots__/NestedCollection.spec.js.snap +550 -0
  185. package/src/components/Collection/__tests__/__snapshots__/Sidebar.spec.js.snap +312 -0
  186. package/src/components/Editor/Editor.js +497 -0
  187. package/src/components/Editor/EditorControlPane/EditorControl.js +453 -0
  188. package/src/components/Editor/EditorControlPane/EditorControlPane.js +269 -0
  189. package/src/components/Editor/EditorControlPane/Widget.js +384 -0
  190. package/src/components/Editor/EditorInterface.js +444 -0
  191. package/src/components/Editor/EditorPreviewPane/EditorPreview.js +40 -0
  192. package/src/components/Editor/EditorPreviewPane/EditorPreviewContent.js +75 -0
  193. package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +337 -0
  194. package/src/components/Editor/EditorPreviewPane/PreviewHOC.js +33 -0
  195. package/src/components/Editor/EditorToolbar.js +728 -0
  196. package/src/components/Editor/__tests__/Editor.spec.js +221 -0
  197. package/src/components/Editor/__tests__/EditorToolbar.spec.js +166 -0
  198. package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +45 -0
  199. package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +4265 -0
  200. package/src/components/Editor/withWorkflow.js +61 -0
  201. package/src/components/EditorWidgets/Unknown/UnknownControl.js +17 -0
  202. package/src/components/EditorWidgets/Unknown/UnknownPreview.js +19 -0
  203. package/src/components/EditorWidgets/index.js +5 -0
  204. package/src/components/MediaLibrary/EmptyMessage.js +28 -0
  205. package/src/components/MediaLibrary/MediaLibrary.js +411 -0
  206. package/src/components/MediaLibrary/MediaLibraryButtons.js +135 -0
  207. package/src/components/MediaLibrary/MediaLibraryCard.js +128 -0
  208. package/src/components/MediaLibrary/MediaLibraryCardGrid.js +199 -0
  209. package/src/components/MediaLibrary/MediaLibraryHeader.js +48 -0
  210. package/src/components/MediaLibrary/MediaLibraryModal.js +200 -0
  211. package/src/components/MediaLibrary/MediaLibrarySearch.js +61 -0
  212. package/src/components/MediaLibrary/MediaLibraryTop.js +144 -0
  213. package/src/components/MediaLibrary/__tests__/MediaLibraryButtons.spec.js +45 -0
  214. package/src/components/MediaLibrary/__tests__/MediaLibraryCard.spec.js +49 -0
  215. package/src/components/MediaLibrary/__tests__/__snapshots__/MediaLibraryCard.spec.js.snap +264 -0
  216. package/src/components/UI/DragDrop.js +66 -0
  217. package/src/components/UI/ErrorBoundary.js +213 -0
  218. package/src/components/UI/FileUploadButton.js +24 -0
  219. package/src/components/UI/Modal.js +112 -0
  220. package/src/components/UI/Notifications.tsx +83 -0
  221. package/src/components/UI/SettingsDropdown.js +130 -0
  222. package/src/components/UI/__tests__/ErrorBoundary.spec.js +57 -0
  223. package/src/components/UI/index.js +6 -0
  224. package/src/components/Workflow/Workflow.js +169 -0
  225. package/src/components/Workflow/WorkflowCard.js +177 -0
  226. package/src/components/Workflow/WorkflowList.js +272 -0
  227. package/src/constants/__tests__/configSchema.spec.js +644 -0
  228. package/src/constants/collectionTypes.ts +2 -0
  229. package/src/constants/collectionViews.js +2 -0
  230. package/src/constants/commitProps.ts +2 -0
  231. package/src/constants/configSchema.js +489 -0
  232. package/src/constants/fieldInference.tsx +78 -0
  233. package/src/constants/publishModes.ts +22 -0
  234. package/src/constants/validationErrorTypes.js +6 -0
  235. package/src/formats/__tests__/formats.spec.js +87 -0
  236. package/src/formats/__tests__/frontmatter.spec.js +450 -0
  237. package/src/formats/__tests__/toml.spec.js +9 -0
  238. package/src/formats/__tests__/yaml.spec.js +189 -0
  239. package/src/formats/formats.ts +97 -0
  240. package/src/formats/frontmatter.ts +150 -0
  241. package/src/formats/helpers.ts +14 -0
  242. package/src/formats/json.ts +9 -0
  243. package/src/formats/toml.ts +33 -0
  244. package/src/formats/yaml.ts +73 -0
  245. package/src/index.js +8 -0
  246. package/src/integrations/index.js +35 -0
  247. package/src/integrations/providers/algolia/implementation.js +176 -0
  248. package/src/integrations/providers/assetStore/implementation.js +148 -0
  249. package/src/lib/__tests__/formatters.spec.js +844 -0
  250. package/src/lib/__tests__/i18n.spec.js +792 -0
  251. package/src/lib/__tests__/imageTransformations.spec.ts +97 -0
  252. package/src/lib/__tests__/phrases.spec.js +119 -0
  253. package/src/lib/__tests__/registry.spec.js +261 -0
  254. package/src/lib/__tests__/serializeEntryValues.spec.js +22 -0
  255. package/src/lib/__tests__/urlHelper.spec.js +145 -0
  256. package/src/lib/consoleError.js +7 -0
  257. package/src/lib/formatters.ts +297 -0
  258. package/src/lib/i18n.ts +462 -0
  259. package/src/lib/imageTransformations.ts +212 -0
  260. package/src/lib/phrases.js +8 -0
  261. package/src/lib/polyfill.js +9 -0
  262. package/src/lib/registry.js +315 -0
  263. package/src/lib/serializeEntryValues.js +75 -0
  264. package/src/lib/stega.ts +145 -0
  265. package/src/lib/textHelper.js +11 -0
  266. package/src/lib/urlHelper.ts +152 -0
  267. package/src/mediaLibrary.ts +51 -0
  268. package/src/reducers/__tests__/auth.spec.ts +38 -0
  269. package/src/reducers/__tests__/collections.spec.js +610 -0
  270. package/src/reducers/__tests__/config.spec.js +38 -0
  271. package/src/reducers/__tests__/deploys.spec.ts +111 -0
  272. package/src/reducers/__tests__/entries.spec.js +694 -0
  273. package/src/reducers/__tests__/entryDraft.spec.js +315 -0
  274. package/src/reducers/__tests__/globalUI.js +43 -0
  275. package/src/reducers/__tests__/integrations.spec.ts +76 -0
  276. package/src/reducers/__tests__/mediaLibrary.spec.js +154 -0
  277. package/src/reducers/__tests__/medias.spec.ts +49 -0
  278. package/src/reducers/auth.ts +46 -0
  279. package/src/reducers/collections.ts +535 -0
  280. package/src/reducers/combinedReducer.ts +11 -0
  281. package/src/reducers/config.ts +38 -0
  282. package/src/reducers/cursors.js +36 -0
  283. package/src/reducers/deploys.ts +54 -0
  284. package/src/reducers/editorialWorkflow.ts +163 -0
  285. package/src/reducers/entries.ts +819 -0
  286. package/src/reducers/entryDraft.js +259 -0
  287. package/src/reducers/globalUI.ts +45 -0
  288. package/src/reducers/index.ts +82 -0
  289. package/src/reducers/integrations.ts +59 -0
  290. package/src/reducers/mediaLibrary.ts +296 -0
  291. package/src/reducers/medias.ts +66 -0
  292. package/src/reducers/notifications.ts +52 -0
  293. package/src/reducers/search.ts +111 -0
  294. package/src/reducers/status.ts +40 -0
  295. package/src/redux/index.ts +18 -0
  296. package/src/redux/middleware/waitUntilAction.ts +64 -0
  297. package/src/routing/__tests__/history.spec.ts +49 -0
  298. package/src/routing/history.ts +17 -0
  299. package/src/types/diacritics.d.ts +1 -0
  300. package/src/types/global.d.ts +8 -0
  301. package/src/types/immutable.ts +49 -0
  302. package/src/types/redux.ts +875 -0
  303. package/src/types/tomlify-j0.4.d.ts +13 -0
  304. package/src/valueObjects/AssetProxy.ts +48 -0
  305. package/src/valueObjects/EditorComponent.js +38 -0
  306. package/src/valueObjects/Entry.ts +63 -0
@@ -0,0 +1,171 @@
1
+ import { Map } from 'immutable';
2
+ import configureMockStore from 'redux-mock-store';
3
+ import thunk from 'redux-thunk';
4
+ import { mocked } from 'jest-mock';
5
+
6
+ import { getAsset, ADD_ASSET, LOAD_ASSET_REQUEST } from '../media';
7
+ import { selectMediaFilePath } from '../../reducers/entries';
8
+ import AssetProxy from '../../valueObjects/AssetProxy';
9
+
10
+ import type { State } from '../../types/redux';
11
+ import type { AnyAction } from 'redux';
12
+ import type { ThunkDispatch } from 'redux-thunk';
13
+
14
+ const middlewares = [thunk];
15
+ const mockStore = configureMockStore<Partial<State>, ThunkDispatch<State, {}, AnyAction>>(
16
+ middlewares,
17
+ );
18
+ const mockedSelectMediaFilePath = mocked(selectMediaFilePath);
19
+
20
+ jest.mock('../../reducers/entries');
21
+ jest.mock('../mediaLibrary');
22
+
23
+ describe('media', () => {
24
+ const emptyAsset = new AssetProxy({
25
+ path: 'empty.svg',
26
+ file: new File([`<svg xmlns="http://www.w3.org/2000/svg"></svg>`], 'empty.svg', {
27
+ type: 'image/svg+xml',
28
+ }),
29
+ });
30
+
31
+ describe('getAsset', () => {
32
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
33
+ // @ts-ignore
34
+ global.URL = { createObjectURL: jest.fn() };
35
+
36
+ beforeEach(() => {
37
+ jest.resetAllMocks();
38
+ });
39
+
40
+ it('should return empty asset for null path', () => {
41
+ const store = mockStore({});
42
+
43
+ const payload = { collection: null, entryPath: null, entry: null, path: null };
44
+
45
+ // TODO change to proper payload when immutable is removed
46
+ // from 'collections' and 'entries' state slices
47
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
48
+ // @ts-ignore
49
+ const result = store.dispatch(getAsset(payload));
50
+ const actions = store.getActions();
51
+ expect(actions).toHaveLength(0);
52
+ expect(result).toEqual(emptyAsset);
53
+ });
54
+
55
+ it('should return asset from medias state', () => {
56
+ const path = 'static/media/image.png';
57
+ const asset = new AssetProxy({ file: new File([], 'empty'), path });
58
+ const store = mockStore({
59
+ // TODO change to proper store data when immutable is removed
60
+ // from 'config' state slice
61
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
62
+ // @ts-ignore
63
+ config: Map(),
64
+ medias: {
65
+ [path]: { asset, isLoading: false, error: null },
66
+ },
67
+ });
68
+
69
+ mockedSelectMediaFilePath.mockReturnValue(path);
70
+ const payload = { collection: Map(), entry: Map({ path: 'entryPath' }), path };
71
+
72
+ // TODO change to proper payload when immutable is removed
73
+ // from 'collections' and 'entries' state slices
74
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
75
+ // @ts-ignore
76
+ const result = store.dispatch(getAsset(payload));
77
+ const actions = store.getActions();
78
+ expect(actions).toHaveLength(0);
79
+
80
+ expect(result).toBe(asset);
81
+ expect(mockedSelectMediaFilePath).toHaveBeenCalledTimes(1);
82
+ expect(mockedSelectMediaFilePath).toHaveBeenCalledWith(
83
+ store.getState().config,
84
+ payload.collection,
85
+ payload.entry,
86
+ path,
87
+ undefined,
88
+ );
89
+ });
90
+
91
+ it('should create asset for absolute path when not in medias state', () => {
92
+ const path = 'https://asset.netlify.com/image.png';
93
+
94
+ const asset = new AssetProxy({ url: path, path });
95
+ const store = mockStore({
96
+ medias: {},
97
+ });
98
+
99
+ mockedSelectMediaFilePath.mockReturnValue(path);
100
+ const payload = { collection: null, entryPath: null, path };
101
+
102
+ // TODO change to proper payload when immutable is removed
103
+ // from 'collections' state slice
104
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
105
+ // @ts-ignore
106
+ const result = store.dispatch(getAsset(payload));
107
+ const actions = store.getActions();
108
+ expect(actions).toHaveLength(1);
109
+ expect(actions[0]).toEqual({
110
+ type: ADD_ASSET,
111
+ payload: asset,
112
+ });
113
+ expect(result).toEqual(asset);
114
+ });
115
+
116
+ it('should return empty asset and initiate load when not in medias state', () => {
117
+ const path = 'static/media/image.png';
118
+ const store = mockStore({
119
+ medias: {},
120
+ });
121
+
122
+ mockedSelectMediaFilePath.mockReturnValue(path);
123
+ const payload = { path };
124
+
125
+ // TODO change to proper payload when immutable is removed
126
+ // from 'collections' and 'entries' state slices
127
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
128
+ // @ts-ignore
129
+ const result = store.dispatch(getAsset(payload));
130
+ const actions = store.getActions();
131
+ expect(actions).toHaveLength(1);
132
+ expect(actions[0]).toEqual({
133
+ type: LOAD_ASSET_REQUEST,
134
+ payload: { path },
135
+ });
136
+ expect(result).toEqual(emptyAsset);
137
+ });
138
+
139
+ it('should return asset with original path on load error', () => {
140
+ const path = 'static/media/image.png';
141
+ const resolvePath = 'resolvePath';
142
+ const store = mockStore({
143
+ medias: {
144
+ [resolvePath]: {
145
+ asset: undefined,
146
+ error: new Error('test'),
147
+ isLoading: false,
148
+ },
149
+ },
150
+ });
151
+
152
+ mockedSelectMediaFilePath.mockReturnValue(resolvePath);
153
+ const payload = { path };
154
+
155
+ // TODO change to proper payload when immutable is removed
156
+ // from 'collections' and 'entries' state slices
157
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
158
+ // @ts-ignore
159
+ const result = store.dispatch(getAsset(payload));
160
+ const actions = store.getActions();
161
+
162
+ const asset = new AssetProxy({ url: path, path: resolvePath });
163
+ expect(actions).toHaveLength(1);
164
+ expect(actions[0]).toEqual({
165
+ type: ADD_ASSET,
166
+ payload: asset,
167
+ });
168
+ expect(result).toEqual(asset);
169
+ });
170
+ });
171
+ });
@@ -0,0 +1,462 @@
1
+ import configureMockStore from 'redux-mock-store';
2
+ import thunk from 'redux-thunk';
3
+ import { List, Map } from 'immutable';
4
+
5
+ import { insertMedia, persistMedia, deleteMedia } from '../mediaLibrary';
6
+
7
+ jest.mock('../../backend');
8
+ jest.mock('../waitUntil');
9
+ jest.mock('../../lib/imageTransformations', () => {
10
+ const actual = jest.requireActual('../../lib/imageTransformations');
11
+ return {
12
+ ...actual,
13
+ transformImage: jest.fn(),
14
+ };
15
+ });
16
+ jest.mock('decap-cms-lib-util', () => {
17
+ const lib = jest.requireActual('decap-cms-lib-util');
18
+ return {
19
+ ...lib,
20
+ getBlobSHA: jest.fn(),
21
+ };
22
+ });
23
+
24
+ const middlewares = [thunk];
25
+ const mockStore = configureMockStore(middlewares);
26
+
27
+ describe('mediaLibrary', () => {
28
+ describe('insertMedia', () => {
29
+ it('should return mediaPath as string when string is given', () => {
30
+ const store = mockStore({
31
+ config: {
32
+ public_folder: '/media',
33
+ },
34
+ collections: Map({
35
+ posts: Map({ name: 'posts' }),
36
+ }),
37
+ entryDraft: Map({
38
+ entry: Map({ isPersisting: false, collection: 'posts' }),
39
+ }),
40
+ });
41
+
42
+ store.dispatch(insertMedia('foo.png'));
43
+ expect(store.getActions()[0]).toEqual({
44
+ type: 'MEDIA_INSERT',
45
+ payload: { mediaPath: '/media/foo.png' },
46
+ });
47
+ });
48
+
49
+ it('should return mediaPath as array of strings when array of strings is given', () => {
50
+ const store = mockStore({
51
+ config: {
52
+ public_folder: '/media',
53
+ },
54
+ collections: Map({
55
+ posts: Map({ name: 'posts' }),
56
+ }),
57
+ entryDraft: Map({
58
+ entry: Map({ isPersisting: false, collection: 'posts' }),
59
+ }),
60
+ });
61
+
62
+ store.dispatch(insertMedia(['foo.png']));
63
+ expect(store.getActions()[0]).toEqual({
64
+ type: 'MEDIA_INSERT',
65
+ payload: { mediaPath: ['/media/foo.png'] },
66
+ });
67
+ });
68
+ });
69
+
70
+ const { currentBackend } = require('../../backend');
71
+
72
+ const backend = {
73
+ persistMedia: jest.fn(() => ({ id: 'id' })),
74
+ deleteMedia: jest.fn(),
75
+ };
76
+
77
+ currentBackend.mockReturnValue(backend);
78
+
79
+ describe('persistMedia', () => {
80
+ global.URL = { createObjectURL: jest.fn().mockReturnValue('displayURL') };
81
+
82
+ beforeEach(() => {
83
+ jest.clearAllMocks();
84
+ window.confirm = jest.fn(() => true);
85
+ });
86
+
87
+ it('should not persist media when editing draft', () => {
88
+ const { getBlobSHA } = require('decap-cms-lib-util');
89
+
90
+ getBlobSHA.mockReturnValue('000000000000000');
91
+
92
+ const store = mockStore({
93
+ config: {
94
+ media_folder: 'static/media',
95
+ slug: {
96
+ encoding: 'unicode',
97
+ clean_accents: false,
98
+ sanitize_replacement: '-',
99
+ },
100
+ },
101
+ collections: Map({
102
+ posts: Map({ name: 'posts' }),
103
+ }),
104
+ integrations: Map(),
105
+ mediaLibrary: Map({
106
+ files: List(),
107
+ }),
108
+ entryDraft: Map({
109
+ entry: Map({ isPersisting: false, collection: 'posts' }),
110
+ }),
111
+ });
112
+
113
+ const file = new File([''], 'name.png');
114
+
115
+ return store.dispatch(persistMedia(file)).then(() => {
116
+ const actions = store.getActions();
117
+
118
+ expect(actions).toHaveLength(2);
119
+ expect(actions[0].type).toEqual('ADD_ASSET');
120
+ expect(actions[0].payload).toEqual(
121
+ expect.objectContaining({
122
+ path: 'static/media/name.png',
123
+ }),
124
+ );
125
+ expect(actions[1].type).toEqual('ADD_DRAFT_ENTRY_MEDIA_FILE');
126
+ expect(actions[1].payload).toEqual(
127
+ expect.objectContaining({
128
+ draft: true,
129
+ id: '000000000000000',
130
+ path: 'static/media/name.png',
131
+ size: file.size,
132
+ name: file.name,
133
+ }),
134
+ );
135
+
136
+ expect(getBlobSHA).toHaveBeenCalledTimes(1);
137
+ expect(getBlobSHA).toHaveBeenCalledWith(file);
138
+ expect(backend.persistMedia).toHaveBeenCalledTimes(0);
139
+ });
140
+ });
141
+
142
+ it('should persist media when not editing draft', () => {
143
+ const store = mockStore({
144
+ config: {
145
+ media_folder: 'static/media',
146
+ slug: {
147
+ encoding: 'unicode',
148
+ clean_accents: false,
149
+ sanitize_replacement: '-',
150
+ },
151
+ },
152
+ collections: Map({
153
+ posts: Map({ name: 'posts' }),
154
+ }),
155
+ integrations: Map(),
156
+ mediaLibrary: Map({
157
+ files: List([{ name: 'kittens.jpg' }]),
158
+ }),
159
+ entryDraft: Map({
160
+ entry: Map(),
161
+ }),
162
+ });
163
+
164
+ const file = new File([''], 'name.png');
165
+
166
+ return store.dispatch(persistMedia(file)).then(() => {
167
+ const actions = store.getActions();
168
+
169
+ expect(actions).toHaveLength(3);
170
+
171
+ expect(actions).toHaveLength(3);
172
+ expect(actions[0]).toEqual({ type: 'MEDIA_PERSIST_REQUEST' });
173
+ expect(actions[1].type).toEqual('ADD_ASSET');
174
+ expect(actions[1].payload).toEqual(
175
+ expect.objectContaining({
176
+ path: 'static/media/name.png',
177
+ }),
178
+ );
179
+ expect(actions[2]).toEqual({
180
+ type: 'MEDIA_PERSIST_SUCCESS',
181
+ payload: {
182
+ file: { id: 'id' },
183
+ },
184
+ });
185
+
186
+ expect(backend.persistMedia).toHaveBeenCalledTimes(1);
187
+ expect(backend.persistMedia).toHaveBeenCalledWith(
188
+ store.getState().config,
189
+ expect.objectContaining({
190
+ path: 'static/media/name.png',
191
+ }),
192
+ );
193
+ });
194
+ });
195
+
196
+ it('should sanitize media name if needed when persisting', () => {
197
+ const store = mockStore({
198
+ config: {
199
+ media_folder: 'static/media',
200
+ slug: {
201
+ encoding: 'ascii',
202
+ clean_accents: true,
203
+ sanitize_replacement: '_',
204
+ },
205
+ },
206
+ collections: Map({
207
+ posts: Map({ name: 'posts' }),
208
+ }),
209
+ integrations: Map(),
210
+ mediaLibrary: Map({
211
+ files: List(),
212
+ }),
213
+ entryDraft: Map({
214
+ entry: Map(),
215
+ }),
216
+ });
217
+
218
+ const file = new File([''], 'abc DEF éâçÖ $;, .png');
219
+
220
+ return store.dispatch(persistMedia(file)).then(() => {
221
+ const actions = store.getActions();
222
+
223
+ expect(actions).toHaveLength(3);
224
+
225
+ expect(actions[0]).toEqual({ type: 'MEDIA_PERSIST_REQUEST' });
226
+
227
+ expect(actions[1].type).toEqual('ADD_ASSET');
228
+ expect(actions[1].payload).toEqual(
229
+ expect.objectContaining({
230
+ path: 'static/media/abc_def_eaco_.png',
231
+ }),
232
+ );
233
+
234
+ expect(actions[2]).toEqual({
235
+ type: 'MEDIA_PERSIST_SUCCESS',
236
+ payload: {
237
+ file: { id: 'id' },
238
+ },
239
+ });
240
+
241
+ expect(backend.persistMedia).toHaveBeenCalledTimes(1);
242
+ expect(backend.persistMedia).toHaveBeenCalledWith(
243
+ store.getState().config,
244
+ expect.objectContaining({
245
+ path: 'static/media/abc_def_eaco_.png',
246
+ }),
247
+ );
248
+ });
249
+ });
250
+
251
+ it('should persist image transformations and select default transformation', () => {
252
+ const { transformImage } = require('../../lib/imageTransformations');
253
+ backend.persistMedia.mockImplementation((_config, assetProxy) => ({
254
+ id: assetProxy.path,
255
+ path: assetProxy.path,
256
+ }));
257
+
258
+ const store = mockStore({
259
+ config: {
260
+ media_folder: 'static/media',
261
+ image_transformations: {
262
+ keep_original: true,
263
+ variants: [
264
+ { name: 'small', width: 100 },
265
+ { name: 'medium', width: 400, default: true },
266
+ ],
267
+ },
268
+ slug: {
269
+ encoding: 'unicode',
270
+ clean_accents: false,
271
+ sanitize_replacement: '-',
272
+ },
273
+ },
274
+ collections: Map({
275
+ posts: Map({ name: 'posts' }),
276
+ }),
277
+ integrations: Map(),
278
+ mediaLibrary: Map({
279
+ files: List(),
280
+ }),
281
+ entryDraft: Map({
282
+ entry: Map(),
283
+ }),
284
+ });
285
+
286
+ const file = new File(['original'], 'kittens.jpg', { type: 'image/jpeg' });
287
+ const small = new File(['small'], 'kittens.jpg', { type: 'image/jpeg' });
288
+ const medium = new File(['medium'], 'kittens.jpg', { type: 'image/jpeg' });
289
+
290
+ transformImage.mockResolvedValue([
291
+ { file, path: 'static/media/kittens.jpg', original: true },
292
+ { file: small, path: 'static/media/_transformations/small/kittens.jpg' },
293
+ { file: medium, path: 'static/media/_transformations/medium/kittens.jpg', default: true },
294
+ ]);
295
+
296
+ return store.dispatch(persistMedia(file)).then(() => {
297
+ const actions = store.getActions();
298
+ const addAssetActions = actions.filter(action => action.type === 'ADD_ASSET');
299
+ const persistedActions = actions.filter(action => action.type === 'MEDIA_PERSIST_SUCCESS');
300
+
301
+ expect(transformImage).toHaveBeenCalledWith(file, 'static/media/kittens.jpg', {
302
+ keepOriginal: true,
303
+ variants: [
304
+ { name: 'small', width: 100 },
305
+ { name: 'medium', width: 400, default: true },
306
+ ],
307
+ });
308
+ expect(addAssetActions.map(action => action.payload.path)).toEqual([
309
+ 'static/media/kittens.jpg',
310
+ 'static/media/_transformations/small/kittens.jpg',
311
+ 'static/media/_transformations/medium/kittens.jpg',
312
+ ]);
313
+ expect(persistedActions.map(action => action.payload.file.path)).toEqual([
314
+ 'static/media/kittens.jpg',
315
+ 'static/media/_transformations/small/kittens.jpg',
316
+ 'static/media/_transformations/medium/kittens.jpg',
317
+ ]);
318
+ expect(persistedActions[persistedActions.length - 1].payload.file.path).toBe(
319
+ 'static/media/_transformations/medium/kittens.jpg',
320
+ );
321
+ });
322
+ });
323
+
324
+ it('should skip the original image when keep_original is false', () => {
325
+ const { transformImage } = require('../../lib/imageTransformations');
326
+ backend.persistMedia.mockImplementation((_config, assetProxy) => ({
327
+ id: assetProxy.path,
328
+ path: assetProxy.path,
329
+ }));
330
+
331
+ const store = mockStore({
332
+ config: {
333
+ media_folder: 'static/media',
334
+ image_transformations: {
335
+ keep_original: false,
336
+ variants: [{ name: 'compressed', quality: 70, keep_original_size: true }],
337
+ },
338
+ slug: {
339
+ encoding: 'unicode',
340
+ clean_accents: false,
341
+ sanitize_replacement: '-',
342
+ },
343
+ },
344
+ collections: Map({
345
+ posts: Map({ name: 'posts' }),
346
+ }),
347
+ integrations: Map(),
348
+ mediaLibrary: Map({
349
+ files: List(),
350
+ }),
351
+ entryDraft: Map({
352
+ entry: Map(),
353
+ }),
354
+ });
355
+
356
+ const file = new File(['original'], 'kittens.jpg', { type: 'image/jpeg' });
357
+ const compressed = new File(['compressed'], 'kittens.jpg', { type: 'image/jpeg' });
358
+
359
+ transformImage.mockResolvedValue([
360
+ { file: compressed, path: 'static/media/_transformations/compressed/kittens.jpg' },
361
+ ]);
362
+
363
+ return store.dispatch(persistMedia(file)).then(() => {
364
+ const addAssetActions = store.getActions().filter(action => action.type === 'ADD_ASSET');
365
+
366
+ expect(transformImage).toHaveBeenCalledWith(file, 'static/media/kittens.jpg', {
367
+ keepOriginal: false,
368
+ variants: [{ name: 'compressed', quality: 70, keep_original_size: true }],
369
+ });
370
+ expect(addAssetActions.map(action => action.payload.path)).toEqual([
371
+ 'static/media/_transformations/compressed/kittens.jpg',
372
+ ]);
373
+ expect(window.confirm).not.toHaveBeenCalled();
374
+ expect(backend.persistMedia).toHaveBeenCalledTimes(1);
375
+ });
376
+ });
377
+ });
378
+
379
+ describe('deleteMedia', () => {
380
+ beforeEach(() => {
381
+ jest.clearAllMocks();
382
+ });
383
+
384
+ it('should delete non draft file', () => {
385
+ const store = mockStore({
386
+ config: {
387
+ publish_mode: 'editorial_workflow',
388
+ },
389
+ collections: Map(),
390
+ integrations: Map(),
391
+ mediaLibrary: Map({
392
+ files: List(),
393
+ }),
394
+ entryDraft: Map({
395
+ entry: Map({ isPersisting: false }),
396
+ }),
397
+ });
398
+
399
+ const file = { name: 'name.png', id: 'id', path: 'static/media/name.png', draft: false };
400
+
401
+ return store.dispatch(deleteMedia(file)).then(() => {
402
+ const actions = store.getActions();
403
+
404
+ expect(actions).toHaveLength(4);
405
+ expect(actions[0]).toEqual({ type: 'MEDIA_DELETE_REQUEST' });
406
+ expect(actions[1]).toEqual({
407
+ type: 'REMOVE_ASSET',
408
+ payload: 'static/media/name.png',
409
+ });
410
+ expect(actions[2]).toEqual({
411
+ type: 'MEDIA_DELETE_SUCCESS',
412
+ payload: { file },
413
+ });
414
+ expect(actions[3]).toEqual({
415
+ type: 'REMOVE_DRAFT_ENTRY_MEDIA_FILE',
416
+ payload: { id: 'id' },
417
+ });
418
+
419
+ expect(backend.deleteMedia).toHaveBeenCalledTimes(1);
420
+ expect(backend.deleteMedia).toHaveBeenCalledWith(
421
+ store.getState().config,
422
+ 'static/media/name.png',
423
+ );
424
+ });
425
+ });
426
+
427
+ it('should not delete a draft file', () => {
428
+ const store = mockStore({
429
+ config: {
430
+ publish_mode: 'editorial_workflow',
431
+ },
432
+ collections: Map(),
433
+ integrations: Map(),
434
+ mediaLibrary: Map({
435
+ files: List(),
436
+ }),
437
+ entryDraft: Map({
438
+ entry: Map({ isPersisting: false }),
439
+ }),
440
+ });
441
+
442
+ const file = { name: 'name.png', id: 'id', path: 'static/media/name.png', draft: true };
443
+
444
+ return store.dispatch(deleteMedia(file)).then(() => {
445
+ const actions = store.getActions();
446
+
447
+ expect(actions).toHaveLength(2);
448
+ expect(actions[0]).toEqual({
449
+ type: 'REMOVE_ASSET',
450
+ payload: 'static/media/name.png',
451
+ });
452
+
453
+ expect(actions[1]).toEqual({
454
+ type: 'REMOVE_DRAFT_ENTRY_MEDIA_FILE',
455
+ payload: { id: 'id' },
456
+ });
457
+
458
+ expect(backend.deleteMedia).toHaveBeenCalledTimes(0);
459
+ });
460
+ });
461
+ });
462
+ });