@pranaysahith/decap-cms-core 3.9.1

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 (299) hide show
  1. package/README.md +9 -0
  2. package/dist/@pranaysahith/decap-cms-core.js +52 -0
  3. package/dist/@pranaysahith/decap-cms-core.js.LICENSE.txt +141 -0
  4. package/dist/@pranaysahith/decap-cms-core.js.map +1 -0
  5. package/dist/decap-cms-core.js +47 -0
  6. package/dist/decap-cms-core.js.LICENSE.txt +116 -0
  7. package/dist/decap-cms-core.js.map +1 -0
  8. package/dist/esm/actions/auth.js +97 -0
  9. package/dist/esm/actions/collections.js +15 -0
  10. package/dist/esm/actions/config.js +493 -0
  11. package/dist/esm/actions/deploys.js +79 -0
  12. package/dist/esm/actions/editorialWorkflow.js +480 -0
  13. package/dist/esm/actions/entries.js +865 -0
  14. package/dist/esm/actions/media.js +147 -0
  15. package/dist/esm/actions/mediaLibrary.js +552 -0
  16. package/dist/esm/actions/notifications.js +21 -0
  17. package/dist/esm/actions/search.js +149 -0
  18. package/dist/esm/actions/status.js +74 -0
  19. package/dist/esm/actions/waitUntil.js +32 -0
  20. package/dist/esm/backend.js +1082 -0
  21. package/dist/esm/bootstrap.js +101 -0
  22. package/dist/esm/components/App/App.js +289 -0
  23. package/dist/esm/components/App/Header.js +172 -0
  24. package/dist/esm/components/App/NotFoundPage.js +19 -0
  25. package/dist/esm/components/Collection/Collection.js +198 -0
  26. package/dist/esm/components/Collection/CollectionControls.js +46 -0
  27. package/dist/esm/components/Collection/CollectionSearch.js +222 -0
  28. package/dist/esm/components/Collection/CollectionTop.js +68 -0
  29. package/dist/esm/components/Collection/ControlButton.js +17 -0
  30. package/dist/esm/components/Collection/Entries/Entries.js +73 -0
  31. package/dist/esm/components/Collection/Entries/EntriesCollection.js +241 -0
  32. package/dist/esm/components/Collection/Entries/EntriesSearch.js +113 -0
  33. package/dist/esm/components/Collection/Entries/EntryCard.js +177 -0
  34. package/dist/esm/components/Collection/Entries/EntryListing.js +143 -0
  35. package/dist/esm/components/Collection/FilterControl.js +33 -0
  36. package/dist/esm/components/Collection/FolderRenameControl.js +403 -0
  37. package/dist/esm/components/Collection/GroupControl.js +33 -0
  38. package/dist/esm/components/Collection/NestedCollection.js +308 -0
  39. package/dist/esm/components/Collection/Sidebar.js +91 -0
  40. package/dist/esm/components/Collection/SortControl.js +59 -0
  41. package/dist/esm/components/Collection/ViewStyleControl.js +38 -0
  42. package/dist/esm/components/Editor/Editor.js +466 -0
  43. package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +395 -0
  44. package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +254 -0
  45. package/dist/esm/components/Editor/EditorControlPane/Widget.js +374 -0
  46. package/dist/esm/components/Editor/EditorInterface.js +386 -0
  47. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +47 -0
  48. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +66 -0
  49. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +288 -0
  50. package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +27 -0
  51. package/dist/esm/components/Editor/EditorToolbar.js +536 -0
  52. package/dist/esm/components/Editor/EntryPathEditor.js +272 -0
  53. package/dist/esm/components/Editor/withWorkflow.js +56 -0
  54. package/dist/esm/components/EditorWidgets/Unknown/UnknownControl.js +18 -0
  55. package/dist/esm/components/EditorWidgets/Unknown/UnknownPreview.js +20 -0
  56. package/dist/esm/components/EditorWidgets/index.js +4 -0
  57. package/dist/esm/components/MediaLibrary/EmptyMessage.js +22 -0
  58. package/dist/esm/components/MediaLibrary/MediaLibrary.js +446 -0
  59. package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +93 -0
  60. package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +99 -0
  61. package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +198 -0
  62. package/dist/esm/components/MediaLibrary/MediaLibraryHeader.js +32 -0
  63. package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +156 -0
  64. package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +51 -0
  65. package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +123 -0
  66. package/dist/esm/components/UI/DragDrop.js +67 -0
  67. package/dist/esm/components/UI/ErrorBoundary.js +173 -0
  68. package/dist/esm/components/UI/FileUploadButton.js +27 -0
  69. package/dist/esm/components/UI/Modal.js +104 -0
  70. package/dist/esm/components/UI/Notifications.js +62 -0
  71. package/dist/esm/components/UI/SettingsDropdown.js +107 -0
  72. package/dist/esm/components/UI/index.js +6 -0
  73. package/dist/esm/components/Workflow/Workflow.js +133 -0
  74. package/dist/esm/components/Workflow/WorkflowCard.js +128 -0
  75. package/dist/esm/components/Workflow/WorkflowList.js +204 -0
  76. package/dist/esm/constants/collectionTypes.js +2 -0
  77. package/dist/esm/constants/collectionViews.js +2 -0
  78. package/dist/esm/constants/commitProps.js +2 -0
  79. package/dist/esm/constants/configSchema.js +644 -0
  80. package/dist/esm/constants/fieldInference.js +57 -0
  81. package/dist/esm/constants/publishModes.js +18 -0
  82. package/dist/esm/constants/validationErrorTypes.js +6 -0
  83. package/dist/esm/formats/formats.js +83 -0
  84. package/dist/esm/formats/frontmatter.js +146 -0
  85. package/dist/esm/formats/helpers.js +12 -0
  86. package/dist/esm/formats/json.js +8 -0
  87. package/dist/esm/formats/toml.js +32 -0
  88. package/dist/esm/formats/yaml.js +51 -0
  89. package/dist/esm/index.js +7 -0
  90. package/dist/esm/integrations/index.js +28 -0
  91. package/dist/esm/integrations/providers/algolia/implementation.js +174 -0
  92. package/dist/esm/integrations/providers/assetStore/implementation.js +165 -0
  93. package/dist/esm/lib/consoleError.js +3 -0
  94. package/dist/esm/lib/formatters.js +191 -0
  95. package/dist/esm/lib/i18n.js +367 -0
  96. package/dist/esm/lib/phrases.js +6 -0
  97. package/dist/esm/lib/polyfill.js +8 -0
  98. package/dist/esm/lib/registry.js +329 -0
  99. package/dist/esm/lib/serializeEntryValues.js +67 -0
  100. package/dist/esm/lib/stega.js +142 -0
  101. package/dist/esm/lib/textHelper.js +9 -0
  102. package/dist/esm/lib/urlHelper.js +111 -0
  103. package/dist/esm/mediaLibrary.js +37 -0
  104. package/dist/esm/reducers/auth.js +27 -0
  105. package/dist/esm/reducers/collections.js +428 -0
  106. package/dist/esm/reducers/combinedReducer.js +8 -0
  107. package/dist/esm/reducers/config.js +29 -0
  108. package/dist/esm/reducers/cursors.js +31 -0
  109. package/dist/esm/reducers/deploys.js +45 -0
  110. package/dist/esm/reducers/editorialWorkflow.js +83 -0
  111. package/dist/esm/reducers/entries.js +568 -0
  112. package/dist/esm/reducers/entryDraft.js +212 -0
  113. package/dist/esm/reducers/globalUI.js +25 -0
  114. package/dist/esm/reducers/index.js +66 -0
  115. package/dist/esm/reducers/integrations.js +53 -0
  116. package/dist/esm/reducers/mediaLibrary.js +252 -0
  117. package/dist/esm/reducers/medias.js +68 -0
  118. package/dist/esm/reducers/notifications.js +23 -0
  119. package/dist/esm/reducers/search.js +92 -0
  120. package/dist/esm/reducers/status.js +30 -0
  121. package/dist/esm/redux/index.js +7 -0
  122. package/dist/esm/redux/middleware/waitUntilAction.js +48 -0
  123. package/dist/esm/routing/history.js +12 -0
  124. package/dist/esm/types/diacritics.d.js +0 -0
  125. package/dist/esm/types/global.d.js +1 -0
  126. package/dist/esm/types/immutable.js +7 -0
  127. package/dist/esm/types/redux.js +14 -0
  128. package/dist/esm/types/tomlify-j0.4.d.js +0 -0
  129. package/dist/esm/valueObjects/AssetProxy.js +44 -0
  130. package/dist/esm/valueObjects/EditorComponent.js +34 -0
  131. package/dist/esm/valueObjects/Entry.js +20 -0
  132. package/index.d.ts +618 -0
  133. package/package.json +106 -0
  134. package/src/__tests__/backend.spec.js +1161 -0
  135. package/src/actions/__tests__/config.spec.js +1009 -0
  136. package/src/actions/__tests__/editorialWorkflow.spec.js +216 -0
  137. package/src/actions/__tests__/entries.spec.js +596 -0
  138. package/src/actions/__tests__/media.spec.ts +171 -0
  139. package/src/actions/__tests__/mediaLibrary.spec.js +327 -0
  140. package/src/actions/__tests__/search.spec.js +209 -0
  141. package/src/actions/auth.ts +127 -0
  142. package/src/actions/collections.ts +18 -0
  143. package/src/actions/config.ts +565 -0
  144. package/src/actions/deploys.ts +104 -0
  145. package/src/actions/editorialWorkflow.ts +567 -0
  146. package/src/actions/entries.ts +1055 -0
  147. package/src/actions/media.ts +139 -0
  148. package/src/actions/mediaLibrary.ts +574 -0
  149. package/src/actions/notifications.ts +36 -0
  150. package/src/actions/search.ts +221 -0
  151. package/src/actions/status.ts +99 -0
  152. package/src/actions/waitUntil.ts +49 -0
  153. package/src/backend.ts +1400 -0
  154. package/src/bootstrap.js +104 -0
  155. package/src/components/App/App.js +286 -0
  156. package/src/components/App/Header.js +266 -0
  157. package/src/components/App/NotFoundPage.js +23 -0
  158. package/src/components/Collection/Collection.js +210 -0
  159. package/src/components/Collection/CollectionControls.js +58 -0
  160. package/src/components/Collection/CollectionSearch.js +243 -0
  161. package/src/components/Collection/CollectionTop.js +81 -0
  162. package/src/components/Collection/ControlButton.js +27 -0
  163. package/src/components/Collection/Entries/Entries.js +82 -0
  164. package/src/components/Collection/Entries/EntriesCollection.js +277 -0
  165. package/src/components/Collection/Entries/EntriesSearch.js +102 -0
  166. package/src/components/Collection/Entries/EntryCard.js +246 -0
  167. package/src/components/Collection/Entries/EntryListing.js +151 -0
  168. package/src/components/Collection/Entries/__tests__/EntriesCollection.spec.js +163 -0
  169. package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +46 -0
  170. package/src/components/Collection/FilterControl.js +39 -0
  171. package/src/components/Collection/GroupControl.js +39 -0
  172. package/src/components/Collection/NestedCollection.js +330 -0
  173. package/src/components/Collection/Sidebar.js +136 -0
  174. package/src/components/Collection/SortControl.js +68 -0
  175. package/src/components/Collection/ViewStyleControl.js +50 -0
  176. package/src/components/Collection/__tests__/Collection.spec.js +75 -0
  177. package/src/components/Collection/__tests__/NestedCollection.spec.js +445 -0
  178. package/src/components/Collection/__tests__/Sidebar.spec.js +87 -0
  179. package/src/components/Collection/__tests__/__snapshots__/Collection.spec.js.snap +144 -0
  180. package/src/components/Collection/__tests__/__snapshots__/NestedCollection.spec.js.snap +550 -0
  181. package/src/components/Collection/__tests__/__snapshots__/Sidebar.spec.js.snap +312 -0
  182. package/src/components/Editor/Editor.js +497 -0
  183. package/src/components/Editor/EditorControlPane/EditorControl.js +452 -0
  184. package/src/components/Editor/EditorControlPane/EditorControlPane.js +269 -0
  185. package/src/components/Editor/EditorControlPane/Widget.js +384 -0
  186. package/src/components/Editor/EditorInterface.js +444 -0
  187. package/src/components/Editor/EditorPreviewPane/EditorPreview.js +40 -0
  188. package/src/components/Editor/EditorPreviewPane/EditorPreviewContent.js +74 -0
  189. package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +333 -0
  190. package/src/components/Editor/EditorPreviewPane/PreviewHOC.js +33 -0
  191. package/src/components/Editor/EditorToolbar.js +691 -0
  192. package/src/components/Editor/__tests__/Editor.spec.js +221 -0
  193. package/src/components/Editor/__tests__/EditorToolbar.spec.js +120 -0
  194. package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +45 -0
  195. package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +4233 -0
  196. package/src/components/Editor/withWorkflow.js +61 -0
  197. package/src/components/EditorWidgets/Unknown/UnknownControl.js +17 -0
  198. package/src/components/EditorWidgets/Unknown/UnknownPreview.js +19 -0
  199. package/src/components/EditorWidgets/index.js +5 -0
  200. package/src/components/MediaLibrary/EmptyMessage.js +28 -0
  201. package/src/components/MediaLibrary/MediaLibrary.js +411 -0
  202. package/src/components/MediaLibrary/MediaLibraryButtons.js +136 -0
  203. package/src/components/MediaLibrary/MediaLibraryCard.js +128 -0
  204. package/src/components/MediaLibrary/MediaLibraryCardGrid.js +199 -0
  205. package/src/components/MediaLibrary/MediaLibraryHeader.js +48 -0
  206. package/src/components/MediaLibrary/MediaLibraryModal.js +200 -0
  207. package/src/components/MediaLibrary/MediaLibrarySearch.js +61 -0
  208. package/src/components/MediaLibrary/MediaLibraryTop.js +143 -0
  209. package/src/components/MediaLibrary/__tests__/MediaLibraryButtons.spec.js +45 -0
  210. package/src/components/MediaLibrary/__tests__/MediaLibraryCard.spec.js +49 -0
  211. package/src/components/MediaLibrary/__tests__/__snapshots__/MediaLibraryCard.spec.js.snap +264 -0
  212. package/src/components/UI/DragDrop.js +66 -0
  213. package/src/components/UI/ErrorBoundary.js +214 -0
  214. package/src/components/UI/FileUploadButton.js +24 -0
  215. package/src/components/UI/Modal.js +112 -0
  216. package/src/components/UI/Notifications.tsx +83 -0
  217. package/src/components/UI/SettingsDropdown.js +103 -0
  218. package/src/components/UI/__tests__/ErrorBoundary.spec.js +57 -0
  219. package/src/components/UI/index.js +6 -0
  220. package/src/components/Workflow/Workflow.js +169 -0
  221. package/src/components/Workflow/WorkflowCard.js +177 -0
  222. package/src/components/Workflow/WorkflowList.js +272 -0
  223. package/src/constants/__tests__/configSchema.spec.js +611 -0
  224. package/src/constants/collectionTypes.ts +2 -0
  225. package/src/constants/collectionViews.js +2 -0
  226. package/src/constants/commitProps.ts +2 -0
  227. package/src/constants/configSchema.js +441 -0
  228. package/src/constants/fieldInference.tsx +78 -0
  229. package/src/constants/publishModes.ts +22 -0
  230. package/src/constants/validationErrorTypes.js +6 -0
  231. package/src/formats/__tests__/formats.spec.js +87 -0
  232. package/src/formats/__tests__/frontmatter.spec.js +429 -0
  233. package/src/formats/__tests__/toml.spec.js +9 -0
  234. package/src/formats/__tests__/yaml.spec.js +162 -0
  235. package/src/formats/formats.ts +97 -0
  236. package/src/formats/frontmatter.ts +150 -0
  237. package/src/formats/helpers.ts +14 -0
  238. package/src/formats/json.ts +9 -0
  239. package/src/formats/toml.ts +33 -0
  240. package/src/formats/yaml.ts +58 -0
  241. package/src/index.js +8 -0
  242. package/src/integrations/index.js +35 -0
  243. package/src/integrations/providers/algolia/implementation.js +176 -0
  244. package/src/integrations/providers/assetStore/implementation.js +148 -0
  245. package/src/lib/__tests__/formatters.spec.js +751 -0
  246. package/src/lib/__tests__/i18n.spec.js +792 -0
  247. package/src/lib/__tests__/phrases.spec.js +119 -0
  248. package/src/lib/__tests__/registry.spec.js +261 -0
  249. package/src/lib/__tests__/serializeEntryValues.spec.js +22 -0
  250. package/src/lib/__tests__/urlHelper.spec.js +138 -0
  251. package/src/lib/consoleError.js +7 -0
  252. package/src/lib/formatters.ts +286 -0
  253. package/src/lib/i18n.ts +454 -0
  254. package/src/lib/phrases.js +8 -0
  255. package/src/lib/polyfill.js +9 -0
  256. package/src/lib/registry.js +312 -0
  257. package/src/lib/serializeEntryValues.js +75 -0
  258. package/src/lib/stega.ts +145 -0
  259. package/src/lib/textHelper.js +11 -0
  260. package/src/lib/urlHelper.ts +128 -0
  261. package/src/mediaLibrary.ts +51 -0
  262. package/src/reducers/__tests__/auth.spec.ts +38 -0
  263. package/src/reducers/__tests__/collections.spec.js +610 -0
  264. package/src/reducers/__tests__/config.spec.js +38 -0
  265. package/src/reducers/__tests__/entries.spec.js +694 -0
  266. package/src/reducers/__tests__/entryDraft.spec.js +297 -0
  267. package/src/reducers/__tests__/globalUI.js +43 -0
  268. package/src/reducers/__tests__/integrations.spec.ts +76 -0
  269. package/src/reducers/__tests__/mediaLibrary.spec.js +154 -0
  270. package/src/reducers/__tests__/medias.spec.ts +49 -0
  271. package/src/reducers/auth.ts +46 -0
  272. package/src/reducers/collections.ts +535 -0
  273. package/src/reducers/combinedReducer.ts +11 -0
  274. package/src/reducers/config.ts +38 -0
  275. package/src/reducers/cursors.js +36 -0
  276. package/src/reducers/deploys.ts +52 -0
  277. package/src/reducers/editorialWorkflow.ts +163 -0
  278. package/src/reducers/entries.ts +819 -0
  279. package/src/reducers/entryDraft.js +260 -0
  280. package/src/reducers/globalUI.ts +45 -0
  281. package/src/reducers/index.ts +82 -0
  282. package/src/reducers/integrations.ts +59 -0
  283. package/src/reducers/mediaLibrary.ts +296 -0
  284. package/src/reducers/medias.ts +66 -0
  285. package/src/reducers/notifications.ts +52 -0
  286. package/src/reducers/search.ts +111 -0
  287. package/src/reducers/status.ts +40 -0
  288. package/src/redux/index.ts +18 -0
  289. package/src/redux/middleware/waitUntilAction.ts +64 -0
  290. package/src/routing/__tests__/history.spec.ts +49 -0
  291. package/src/routing/history.ts +17 -0
  292. package/src/types/diacritics.d.ts +1 -0
  293. package/src/types/global.d.ts +8 -0
  294. package/src/types/immutable.ts +49 -0
  295. package/src/types/redux.ts +827 -0
  296. package/src/types/tomlify-j0.4.d.ts +13 -0
  297. package/src/valueObjects/AssetProxy.ts +48 -0
  298. package/src/valueObjects/EditorComponent.js +38 -0
  299. 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,327 @@
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('decap-cms-lib-util', () => {
10
+ const lib = jest.requireActual('decap-cms-lib-util');
11
+ return {
12
+ ...lib,
13
+ getBlobSHA: jest.fn(),
14
+ };
15
+ });
16
+
17
+ const middlewares = [thunk];
18
+ const mockStore = configureMockStore(middlewares);
19
+
20
+ describe('mediaLibrary', () => {
21
+ describe('insertMedia', () => {
22
+ it('should return mediaPath as string when string is given', () => {
23
+ const store = mockStore({
24
+ config: {
25
+ public_folder: '/media',
26
+ },
27
+ collections: Map({
28
+ posts: Map({ name: 'posts' }),
29
+ }),
30
+ entryDraft: Map({
31
+ entry: Map({ isPersisting: false, collection: 'posts' }),
32
+ }),
33
+ });
34
+
35
+ store.dispatch(insertMedia('foo.png'));
36
+ expect(store.getActions()[0]).toEqual({
37
+ type: 'MEDIA_INSERT',
38
+ payload: { mediaPath: '/media/foo.png' },
39
+ });
40
+ });
41
+
42
+ it('should return mediaPath as array of strings when array of strings is given', () => {
43
+ const store = mockStore({
44
+ config: {
45
+ public_folder: '/media',
46
+ },
47
+ collections: Map({
48
+ posts: Map({ name: 'posts' }),
49
+ }),
50
+ entryDraft: Map({
51
+ entry: Map({ isPersisting: false, collection: 'posts' }),
52
+ }),
53
+ });
54
+
55
+ store.dispatch(insertMedia(['foo.png']));
56
+ expect(store.getActions()[0]).toEqual({
57
+ type: 'MEDIA_INSERT',
58
+ payload: { mediaPath: ['/media/foo.png'] },
59
+ });
60
+ });
61
+ });
62
+
63
+ const { currentBackend } = require('../../backend');
64
+
65
+ const backend = {
66
+ persistMedia: jest.fn(() => ({ id: 'id' })),
67
+ deleteMedia: jest.fn(),
68
+ };
69
+
70
+ currentBackend.mockReturnValue(backend);
71
+
72
+ describe('persistMedia', () => {
73
+ global.URL = { createObjectURL: jest.fn().mockReturnValue('displayURL') };
74
+
75
+ beforeEach(() => {
76
+ jest.clearAllMocks();
77
+ });
78
+
79
+ it('should not persist media when editing draft', () => {
80
+ const { getBlobSHA } = require('decap-cms-lib-util');
81
+
82
+ getBlobSHA.mockReturnValue('000000000000000');
83
+
84
+ const store = mockStore({
85
+ config: {
86
+ media_folder: 'static/media',
87
+ slug: {
88
+ encoding: 'unicode',
89
+ clean_accents: false,
90
+ sanitize_replacement: '-',
91
+ },
92
+ },
93
+ collections: Map({
94
+ posts: Map({ name: 'posts' }),
95
+ }),
96
+ integrations: Map(),
97
+ mediaLibrary: Map({
98
+ files: List(),
99
+ }),
100
+ entryDraft: Map({
101
+ entry: Map({ isPersisting: false, collection: 'posts' }),
102
+ }),
103
+ });
104
+
105
+ const file = new File([''], 'name.png');
106
+
107
+ return store.dispatch(persistMedia(file)).then(() => {
108
+ const actions = store.getActions();
109
+
110
+ expect(actions).toHaveLength(2);
111
+ expect(actions[0].type).toEqual('ADD_ASSET');
112
+ expect(actions[0].payload).toEqual(
113
+ expect.objectContaining({
114
+ path: 'static/media/name.png',
115
+ }),
116
+ );
117
+ expect(actions[1].type).toEqual('ADD_DRAFT_ENTRY_MEDIA_FILE');
118
+ expect(actions[1].payload).toEqual(
119
+ expect.objectContaining({
120
+ draft: true,
121
+ id: '000000000000000',
122
+ path: 'static/media/name.png',
123
+ size: file.size,
124
+ name: file.name,
125
+ }),
126
+ );
127
+
128
+ expect(getBlobSHA).toHaveBeenCalledTimes(1);
129
+ expect(getBlobSHA).toHaveBeenCalledWith(file);
130
+ expect(backend.persistMedia).toHaveBeenCalledTimes(0);
131
+ });
132
+ });
133
+
134
+ it('should persist media when not editing draft', () => {
135
+ const store = mockStore({
136
+ config: {
137
+ media_folder: 'static/media',
138
+ slug: {
139
+ encoding: 'unicode',
140
+ clean_accents: false,
141
+ sanitize_replacement: '-',
142
+ },
143
+ },
144
+ collections: Map({
145
+ posts: Map({ name: 'posts' }),
146
+ }),
147
+ integrations: Map(),
148
+ mediaLibrary: Map({
149
+ files: List(),
150
+ }),
151
+ entryDraft: Map({
152
+ entry: Map(),
153
+ }),
154
+ });
155
+
156
+ const file = new File([''], 'name.png');
157
+
158
+ return store.dispatch(persistMedia(file)).then(() => {
159
+ const actions = store.getActions();
160
+
161
+ expect(actions).toHaveLength(3);
162
+
163
+ expect(actions).toHaveLength(3);
164
+ expect(actions[0]).toEqual({ type: 'MEDIA_PERSIST_REQUEST' });
165
+ expect(actions[1].type).toEqual('ADD_ASSET');
166
+ expect(actions[1].payload).toEqual(
167
+ expect.objectContaining({
168
+ path: 'static/media/name.png',
169
+ }),
170
+ );
171
+ expect(actions[2]).toEqual({
172
+ type: 'MEDIA_PERSIST_SUCCESS',
173
+ payload: {
174
+ file: { id: 'id' },
175
+ },
176
+ });
177
+
178
+ expect(backend.persistMedia).toHaveBeenCalledTimes(1);
179
+ expect(backend.persistMedia).toHaveBeenCalledWith(
180
+ store.getState().config,
181
+ expect.objectContaining({
182
+ path: 'static/media/name.png',
183
+ }),
184
+ );
185
+ });
186
+ });
187
+
188
+ it('should sanitize media name if needed when persisting', () => {
189
+ const store = mockStore({
190
+ config: {
191
+ media_folder: 'static/media',
192
+ slug: {
193
+ encoding: 'ascii',
194
+ clean_accents: true,
195
+ sanitize_replacement: '_',
196
+ },
197
+ },
198
+ collections: Map({
199
+ posts: Map({ name: 'posts' }),
200
+ }),
201
+ integrations: Map(),
202
+ mediaLibrary: Map({
203
+ files: List(),
204
+ }),
205
+ entryDraft: Map({
206
+ entry: Map(),
207
+ }),
208
+ });
209
+
210
+ const file = new File([''], 'abc DEF éâçÖ $;, .png');
211
+
212
+ return store.dispatch(persistMedia(file)).then(() => {
213
+ const actions = store.getActions();
214
+
215
+ expect(actions).toHaveLength(3);
216
+
217
+ expect(actions[0]).toEqual({ type: 'MEDIA_PERSIST_REQUEST' });
218
+
219
+ expect(actions[1].type).toEqual('ADD_ASSET');
220
+ expect(actions[1].payload).toEqual(
221
+ expect.objectContaining({
222
+ path: 'static/media/abc_def_eaco_.png',
223
+ }),
224
+ );
225
+
226
+ expect(actions[2]).toEqual({
227
+ type: 'MEDIA_PERSIST_SUCCESS',
228
+ payload: {
229
+ file: { id: 'id' },
230
+ },
231
+ });
232
+
233
+ expect(backend.persistMedia).toHaveBeenCalledTimes(1);
234
+ expect(backend.persistMedia).toHaveBeenCalledWith(
235
+ store.getState().config,
236
+ expect.objectContaining({
237
+ path: 'static/media/abc_def_eaco_.png',
238
+ }),
239
+ );
240
+ });
241
+ });
242
+ });
243
+
244
+ describe('deleteMedia', () => {
245
+ beforeEach(() => {
246
+ jest.clearAllMocks();
247
+ });
248
+
249
+ it('should delete non draft file', () => {
250
+ const store = mockStore({
251
+ config: {
252
+ publish_mode: 'editorial_workflow',
253
+ },
254
+ collections: Map(),
255
+ integrations: Map(),
256
+ mediaLibrary: Map({
257
+ files: List(),
258
+ }),
259
+ entryDraft: Map({
260
+ entry: Map({ isPersisting: false }),
261
+ }),
262
+ });
263
+
264
+ const file = { name: 'name.png', id: 'id', path: 'static/media/name.png', draft: false };
265
+
266
+ return store.dispatch(deleteMedia(file)).then(() => {
267
+ const actions = store.getActions();
268
+
269
+ expect(actions).toHaveLength(4);
270
+ expect(actions[0]).toEqual({ type: 'MEDIA_DELETE_REQUEST' });
271
+ expect(actions[1]).toEqual({
272
+ type: 'REMOVE_ASSET',
273
+ payload: 'static/media/name.png',
274
+ });
275
+ expect(actions[2]).toEqual({
276
+ type: 'MEDIA_DELETE_SUCCESS',
277
+ payload: { file },
278
+ });
279
+ expect(actions[3]).toEqual({
280
+ type: 'REMOVE_DRAFT_ENTRY_MEDIA_FILE',
281
+ payload: { id: 'id' },
282
+ });
283
+
284
+ expect(backend.deleteMedia).toHaveBeenCalledTimes(1);
285
+ expect(backend.deleteMedia).toHaveBeenCalledWith(
286
+ store.getState().config,
287
+ 'static/media/name.png',
288
+ );
289
+ });
290
+ });
291
+
292
+ it('should not delete a draft file', () => {
293
+ const store = mockStore({
294
+ config: {
295
+ publish_mode: 'editorial_workflow',
296
+ },
297
+ collections: Map(),
298
+ integrations: Map(),
299
+ mediaLibrary: Map({
300
+ files: List(),
301
+ }),
302
+ entryDraft: Map({
303
+ entry: Map({ isPersisting: false }),
304
+ }),
305
+ });
306
+
307
+ const file = { name: 'name.png', id: 'id', path: 'static/media/name.png', draft: true };
308
+
309
+ return store.dispatch(deleteMedia(file)).then(() => {
310
+ const actions = store.getActions();
311
+
312
+ expect(actions).toHaveLength(2);
313
+ expect(actions[0]).toEqual({
314
+ type: 'REMOVE_ASSET',
315
+ payload: 'static/media/name.png',
316
+ });
317
+
318
+ expect(actions[1]).toEqual({
319
+ type: 'REMOVE_DRAFT_ENTRY_MEDIA_FILE',
320
+ payload: { id: 'id' },
321
+ });
322
+
323
+ expect(backend.deleteMedia).toHaveBeenCalledTimes(0);
324
+ });
325
+ });
326
+ });
327
+ });
@@ -0,0 +1,209 @@
1
+ import { fromJS } from 'immutable';
2
+ import configureMockStore from 'redux-mock-store';
3
+ import thunk from 'redux-thunk';
4
+
5
+ import { searchEntries } from '../search';
6
+
7
+ const middlewares = [thunk];
8
+ const mockStore = configureMockStore(middlewares);
9
+
10
+ jest.mock('../../reducers');
11
+ jest.mock('../../backend');
12
+ jest.mock('../../integrations');
13
+
14
+ describe('search', () => {
15
+ describe('searchEntries', () => {
16
+ const { currentBackend } = require('../../backend');
17
+ const { selectIntegration } = require('../../reducers');
18
+ const { getIntegrationProvider } = require('../../integrations');
19
+
20
+ beforeEach(() => {
21
+ jest.resetAllMocks();
22
+ });
23
+ it('should search entries in all collections using integration', async () => {
24
+ const store = mockStore({
25
+ collections: fromJS({ posts: { name: 'posts' }, pages: { name: 'pages' } }),
26
+ search: {},
27
+ });
28
+
29
+ selectIntegration.mockReturnValue('search_integration');
30
+ currentBackend.mockReturnValue({});
31
+ const response = { entries: [{ name: '1' }, { name: '' }], pagination: 1 };
32
+ const integration = { search: jest.fn().mockResolvedValue(response) };
33
+ getIntegrationProvider.mockReturnValue(integration);
34
+
35
+ await store.dispatch(searchEntries('find me'));
36
+ const actions = store.getActions();
37
+ expect(actions).toHaveLength(2);
38
+
39
+ expect(actions[0]).toEqual({
40
+ type: 'SEARCH_ENTRIES_REQUEST',
41
+ payload: {
42
+ searchTerm: 'find me',
43
+ searchCollections: ['posts', 'pages'],
44
+ page: 0,
45
+ },
46
+ });
47
+ expect(actions[1]).toEqual({
48
+ type: 'SEARCH_ENTRIES_SUCCESS',
49
+ payload: {
50
+ entries: response.entries,
51
+ page: response.pagination,
52
+ },
53
+ });
54
+
55
+ expect(integration.search).toHaveBeenCalledTimes(1);
56
+ expect(integration.search).toHaveBeenCalledWith(['posts', 'pages'], 'find me', 0);
57
+ });
58
+
59
+ it('should search entries in a subset of collections using integration', async () => {
60
+ const store = mockStore({
61
+ collections: fromJS({ posts: { name: 'posts' }, pages: { name: 'pages' } }),
62
+ search: {},
63
+ });
64
+
65
+ selectIntegration.mockReturnValue('search_integration');
66
+ currentBackend.mockReturnValue({});
67
+ const response = { entries: [{ name: '1' }, { name: '' }], pagination: 1 };
68
+ const integration = { search: jest.fn().mockResolvedValue(response) };
69
+ getIntegrationProvider.mockReturnValue(integration);
70
+
71
+ await store.dispatch(searchEntries('find me', ['pages']));
72
+ const actions = store.getActions();
73
+ expect(actions).toHaveLength(2);
74
+
75
+ expect(actions[0]).toEqual({
76
+ type: 'SEARCH_ENTRIES_REQUEST',
77
+ payload: {
78
+ searchTerm: 'find me',
79
+ searchCollections: ['pages'],
80
+ page: 0,
81
+ },
82
+ });
83
+ expect(actions[1]).toEqual({
84
+ type: 'SEARCH_ENTRIES_SUCCESS',
85
+ payload: {
86
+ entries: response.entries,
87
+ page: response.pagination,
88
+ },
89
+ });
90
+
91
+ expect(integration.search).toHaveBeenCalledTimes(1);
92
+ expect(integration.search).toHaveBeenCalledWith(['pages'], 'find me', 0);
93
+ });
94
+
95
+ it('should search entries in all collections using backend', async () => {
96
+ const store = mockStore({
97
+ collections: fromJS({ posts: { name: 'posts' }, pages: { name: 'pages' } }),
98
+ search: {},
99
+ });
100
+
101
+ const response = { entries: [{ name: '1' }, { name: '' }], pagination: 1 };
102
+ const backend = { search: jest.fn().mockResolvedValue(response) };
103
+ currentBackend.mockReturnValue(backend);
104
+
105
+ await store.dispatch(searchEntries('find me'));
106
+
107
+ const actions = store.getActions();
108
+ expect(actions).toHaveLength(2);
109
+
110
+ expect(actions[0]).toEqual({
111
+ type: 'SEARCH_ENTRIES_REQUEST',
112
+ payload: {
113
+ searchTerm: 'find me',
114
+ searchCollections: ['posts', 'pages'],
115
+ page: 0,
116
+ },
117
+ });
118
+ expect(actions[1]).toEqual({
119
+ type: 'SEARCH_ENTRIES_SUCCESS',
120
+ payload: {
121
+ entries: response.entries,
122
+ page: response.pagination,
123
+ },
124
+ });
125
+
126
+ expect(backend.search).toHaveBeenCalledTimes(1);
127
+ expect(backend.search).toHaveBeenCalledWith(
128
+ [fromJS({ name: 'posts' }), fromJS({ name: 'pages' })],
129
+ 'find me',
130
+ );
131
+ });
132
+
133
+ it('should search entries in a subset of collections using backend', async () => {
134
+ const store = mockStore({
135
+ collections: fromJS({ posts: { name: 'posts' }, pages: { name: 'pages' } }),
136
+ search: {},
137
+ });
138
+
139
+ const response = { entries: [{ name: '1' }, { name: '' }], pagination: 1 };
140
+ const backend = { search: jest.fn().mockResolvedValue(response) };
141
+ currentBackend.mockReturnValue(backend);
142
+
143
+ await store.dispatch(searchEntries('find me', ['pages']));
144
+
145
+ const actions = store.getActions();
146
+ expect(actions).toHaveLength(2);
147
+
148
+ expect(actions[0]).toEqual({
149
+ type: 'SEARCH_ENTRIES_REQUEST',
150
+ payload: {
151
+ searchTerm: 'find me',
152
+ searchCollections: ['pages'],
153
+ page: 0,
154
+ },
155
+ });
156
+ expect(actions[1]).toEqual({
157
+ type: 'SEARCH_ENTRIES_SUCCESS',
158
+ payload: {
159
+ entries: response.entries,
160
+ page: response.pagination,
161
+ },
162
+ });
163
+
164
+ expect(backend.search).toHaveBeenCalledTimes(1);
165
+ expect(backend.search).toHaveBeenCalledWith([fromJS({ name: 'pages' })], 'find me');
166
+ });
167
+
168
+ it('should ignore identical search in all collections', async () => {
169
+ const store = mockStore({
170
+ collections: fromJS({ posts: { name: 'posts' }, pages: { name: 'pages' } }),
171
+ search: { isFetching: true, term: 'find me', collections: ['posts', 'pages'] },
172
+ });
173
+
174
+ await store.dispatch(searchEntries('find me'));
175
+
176
+ const actions = store.getActions();
177
+ expect(actions).toHaveLength(0);
178
+ });
179
+
180
+ it('should ignore identical search in a subset of collections', async () => {
181
+ const store = mockStore({
182
+ collections: fromJS({ posts: { name: 'posts' }, pages: { name: 'pages' } }),
183
+ search: { isFetching: true, term: 'find me', collections: ['pages'] },
184
+ });
185
+
186
+ await store.dispatch(searchEntries('find me', ['pages']));
187
+
188
+ const actions = store.getActions();
189
+ expect(actions).toHaveLength(0);
190
+ });
191
+
192
+ it('should not ignore same search term in different search collections', async () => {
193
+ const store = mockStore({
194
+ collections: fromJS({ posts: { name: 'posts' }, pages: { name: 'pages' } }),
195
+ search: { isFetching: true, term: 'find me', collections: ['pages'] },
196
+ });
197
+ const backend = { search: jest.fn().mockResolvedValue({}) };
198
+ currentBackend.mockReturnValue(backend);
199
+
200
+ await store.dispatch(searchEntries('find me', ['posts', 'pages']));
201
+
202
+ expect(backend.search).toHaveBeenCalledTimes(1);
203
+ expect(backend.search).toHaveBeenCalledWith(
204
+ [fromJS({ name: 'posts' }), fromJS({ name: 'pages' })],
205
+ 'find me',
206
+ );
207
+ });
208
+ });
209
+ });