decap-cms-core 2.56.0-beta.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 (290) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +9 -0
  3. package/dist/decap-cms-core.js +47 -0
  4. package/dist/decap-cms-core.js.LICENSE.txt +109 -0
  5. package/dist/decap-cms-core.js.map +1 -0
  6. package/dist/esm/actions/auth.js +118 -0
  7. package/dist/esm/actions/collections.js +23 -0
  8. package/dist/esm/actions/config.js +482 -0
  9. package/dist/esm/actions/deploys.js +89 -0
  10. package/dist/esm/actions/editorialWorkflow.js +511 -0
  11. package/dist/esm/actions/entries.js +940 -0
  12. package/dist/esm/actions/media.js +168 -0
  13. package/dist/esm/actions/mediaLibrary.js +606 -0
  14. package/dist/esm/actions/notifications.js +32 -0
  15. package/dist/esm/actions/search.js +156 -0
  16. package/dist/esm/actions/status.js +93 -0
  17. package/dist/esm/actions/waitUntil.js +39 -0
  18. package/dist/esm/backend.js +1073 -0
  19. package/dist/esm/bootstrap.js +107 -0
  20. package/dist/esm/components/App/App.js +301 -0
  21. package/dist/esm/components/App/Header.js +162 -0
  22. package/dist/esm/components/App/NotFoundPage.js +27 -0
  23. package/dist/esm/components/Collection/Collection.js +210 -0
  24. package/dist/esm/components/Collection/CollectionControls.js +54 -0
  25. package/dist/esm/components/Collection/CollectionSearch.js +235 -0
  26. package/dist/esm/components/Collection/CollectionTop.js +77 -0
  27. package/dist/esm/components/Collection/ControlButton.js +23 -0
  28. package/dist/esm/components/Collection/Entries/Entries.js +74 -0
  29. package/dist/esm/components/Collection/Entries/EntriesCollection.js +193 -0
  30. package/dist/esm/components/Collection/Entries/EntriesSearch.js +117 -0
  31. package/dist/esm/components/Collection/Entries/EntryCard.js +134 -0
  32. package/dist/esm/components/Collection/Entries/EntryListing.js +120 -0
  33. package/dist/esm/components/Collection/FilterControl.js +41 -0
  34. package/dist/esm/components/Collection/GroupControl.js +41 -0
  35. package/dist/esm/components/Collection/NestedCollection.js +311 -0
  36. package/dist/esm/components/Collection/Sidebar.js +102 -0
  37. package/dist/esm/components/Collection/SortControl.js +67 -0
  38. package/dist/esm/components/Collection/ViewStyleControl.js +47 -0
  39. package/dist/esm/components/Editor/Editor.js +479 -0
  40. package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +393 -0
  41. package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +259 -0
  42. package/dist/esm/components/Editor/EditorControlPane/Widget.js +355 -0
  43. package/dist/esm/components/Editor/EditorInterface.js +401 -0
  44. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +56 -0
  45. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +34 -0
  46. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +252 -0
  47. package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +39 -0
  48. package/dist/esm/components/Editor/EditorToolbar.js +551 -0
  49. package/dist/esm/components/Editor/withWorkflow.js +64 -0
  50. package/dist/esm/components/EditorWidgets/Unknown/UnknownControl.js +26 -0
  51. package/dist/esm/components/EditorWidgets/Unknown/UnknownPreview.js +28 -0
  52. package/dist/esm/components/EditorWidgets/index.js +7 -0
  53. package/dist/esm/components/MediaLibrary/EmptyMessage.js +30 -0
  54. package/dist/esm/components/MediaLibrary/MediaLibrary.js +444 -0
  55. package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +110 -0
  56. package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +108 -0
  57. package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +211 -0
  58. package/dist/esm/components/MediaLibrary/MediaLibraryHeader.js +40 -0
  59. package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +163 -0
  60. package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +60 -0
  61. package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +134 -0
  62. package/dist/esm/components/UI/DragDrop.js +84 -0
  63. package/dist/esm/components/UI/ErrorBoundary.js +184 -0
  64. package/dist/esm/components/UI/FileUploadButton.js +34 -0
  65. package/dist/esm/components/UI/Modal.js +113 -0
  66. package/dist/esm/components/UI/Notifications.js +74 -0
  67. package/dist/esm/components/UI/SettingsDropdown.js +118 -0
  68. package/dist/esm/components/UI/index.js +60 -0
  69. package/dist/esm/components/Workflow/Workflow.js +144 -0
  70. package/dist/esm/components/Workflow/WorkflowCard.js +137 -0
  71. package/dist/esm/components/Workflow/WorkflowList.js +219 -0
  72. package/dist/esm/constants/collectionTypes.js +10 -0
  73. package/dist/esm/constants/collectionViews.js +10 -0
  74. package/dist/esm/constants/commitProps.js +10 -0
  75. package/dist/esm/constants/configSchema.js +594 -0
  76. package/dist/esm/constants/fieldInference.js +67 -0
  77. package/dist/esm/constants/publishModes.js +28 -0
  78. package/dist/esm/constants/validationErrorTypes.js +13 -0
  79. package/dist/esm/formats/formats.js +82 -0
  80. package/dist/esm/formats/frontmatter.js +163 -0
  81. package/dist/esm/formats/helpers.js +18 -0
  82. package/dist/esm/formats/json.js +15 -0
  83. package/dist/esm/formats/toml.js +40 -0
  84. package/dist/esm/formats/yaml.js +63 -0
  85. package/dist/esm/index.js +20 -0
  86. package/dist/esm/integrations/index.js +37 -0
  87. package/dist/esm/integrations/providers/algolia/implementation.js +184 -0
  88. package/dist/esm/integrations/providers/assetStore/implementation.js +175 -0
  89. package/dist/esm/lib/consoleError.js +9 -0
  90. package/dist/esm/lib/formatters.js +204 -0
  91. package/dist/esm/lib/i18n.js +381 -0
  92. package/dist/esm/lib/phrases.js +13 -0
  93. package/dist/esm/lib/registry.js +322 -0
  94. package/dist/esm/lib/serializeEntryValues.js +74 -0
  95. package/dist/esm/lib/textHelper.js +15 -0
  96. package/dist/esm/lib/urlHelper.js +128 -0
  97. package/dist/esm/mediaLibrary.js +42 -0
  98. package/dist/esm/reducers/auth.js +35 -0
  99. package/dist/esm/reducers/collections.js +424 -0
  100. package/dist/esm/reducers/combinedReducer.js +19 -0
  101. package/dist/esm/reducers/config.js +42 -0
  102. package/dist/esm/reducers/cursors.js +38 -0
  103. package/dist/esm/reducers/deploys.js +53 -0
  104. package/dist/esm/reducers/editorialWorkflow.js +98 -0
  105. package/dist/esm/reducers/entries.js +591 -0
  106. package/dist/esm/reducers/entryDraft.js +184 -0
  107. package/dist/esm/reducers/globalUI.js +32 -0
  108. package/dist/esm/reducers/index.js +84 -0
  109. package/dist/esm/reducers/integrations.js +68 -0
  110. package/dist/esm/reducers/mediaLibrary.js +265 -0
  111. package/dist/esm/reducers/medias.js +76 -0
  112. package/dist/esm/reducers/notifications.js +36 -0
  113. package/dist/esm/reducers/search.js +89 -0
  114. package/dist/esm/reducers/status.js +37 -0
  115. package/dist/esm/redux/index.js +14 -0
  116. package/dist/esm/redux/middleware/waitUntilAction.js +56 -0
  117. package/dist/esm/routing/history.js +21 -0
  118. package/dist/esm/types/diacritics.d.js +1 -0
  119. package/dist/esm/types/global.d.js +5 -0
  120. package/dist/esm/types/immutable.js +5 -0
  121. package/dist/esm/types/redux.js +16 -0
  122. package/dist/esm/types/tomlify-j0.4.d.js +1 -0
  123. package/dist/esm/valueObjects/AssetProxy.js +60 -0
  124. package/dist/esm/valueObjects/EditorComponent.js +48 -0
  125. package/dist/esm/valueObjects/Entry.js +27 -0
  126. package/index.d.ts +587 -0
  127. package/package.json +102 -0
  128. package/src/__tests__/backend.spec.js +934 -0
  129. package/src/actions/__tests__/config.spec.js +1009 -0
  130. package/src/actions/__tests__/editorialWorkflow.spec.js +216 -0
  131. package/src/actions/__tests__/entries.spec.js +575 -0
  132. package/src/actions/__tests__/media.spec.ts +171 -0
  133. package/src/actions/__tests__/mediaLibrary.spec.js +327 -0
  134. package/src/actions/__tests__/search.spec.js +209 -0
  135. package/src/actions/auth.ts +127 -0
  136. package/src/actions/collections.ts +18 -0
  137. package/src/actions/config.ts +538 -0
  138. package/src/actions/deploys.ts +104 -0
  139. package/src/actions/editorialWorkflow.ts +567 -0
  140. package/src/actions/entries.ts +1041 -0
  141. package/src/actions/media.ts +139 -0
  142. package/src/actions/mediaLibrary.ts +574 -0
  143. package/src/actions/notifications.ts +36 -0
  144. package/src/actions/search.ts +194 -0
  145. package/src/actions/status.ts +99 -0
  146. package/src/actions/waitUntil.ts +49 -0
  147. package/src/backend.ts +1342 -0
  148. package/src/bootstrap.js +102 -0
  149. package/src/components/App/App.js +280 -0
  150. package/src/components/App/Header.js +236 -0
  151. package/src/components/App/NotFoundPage.js +23 -0
  152. package/src/components/Collection/Collection.js +205 -0
  153. package/src/components/Collection/CollectionControls.js +58 -0
  154. package/src/components/Collection/CollectionSearch.js +238 -0
  155. package/src/components/Collection/CollectionTop.js +81 -0
  156. package/src/components/Collection/ControlButton.js +27 -0
  157. package/src/components/Collection/Entries/Entries.js +73 -0
  158. package/src/components/Collection/Entries/EntriesCollection.js +165 -0
  159. package/src/components/Collection/Entries/EntriesSearch.js +91 -0
  160. package/src/components/Collection/Entries/EntryCard.js +167 -0
  161. package/src/components/Collection/Entries/EntryListing.js +86 -0
  162. package/src/components/Collection/Entries/__tests__/EntriesCollection.spec.js +155 -0
  163. package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +49 -0
  164. package/src/components/Collection/FilterControl.js +39 -0
  165. package/src/components/Collection/GroupControl.js +39 -0
  166. package/src/components/Collection/NestedCollection.js +309 -0
  167. package/src/components/Collection/Sidebar.js +130 -0
  168. package/src/components/Collection/SortControl.js +68 -0
  169. package/src/components/Collection/ViewStyleControl.js +50 -0
  170. package/src/components/Collection/__tests__/Collection.spec.js +75 -0
  171. package/src/components/Collection/__tests__/NestedCollection.spec.js +442 -0
  172. package/src/components/Collection/__tests__/Sidebar.spec.js +87 -0
  173. package/src/components/Collection/__tests__/__snapshots__/Collection.spec.js.snap +144 -0
  174. package/src/components/Collection/__tests__/__snapshots__/NestedCollection.spec.js.snap +550 -0
  175. package/src/components/Collection/__tests__/__snapshots__/Sidebar.spec.js.snap +308 -0
  176. package/src/components/Editor/Editor.js +494 -0
  177. package/src/components/Editor/EditorControlPane/EditorControl.js +428 -0
  178. package/src/components/Editor/EditorControlPane/EditorControlPane.js +256 -0
  179. package/src/components/Editor/EditorControlPane/Widget.js +351 -0
  180. package/src/components/Editor/EditorInterface.js +434 -0
  181. package/src/components/Editor/EditorPreviewPane/EditorPreview.js +40 -0
  182. package/src/components/Editor/EditorPreviewPane/EditorPreviewContent.js +34 -0
  183. package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +278 -0
  184. package/src/components/Editor/EditorPreviewPane/PreviewHOC.js +33 -0
  185. package/src/components/Editor/EditorToolbar.js +678 -0
  186. package/src/components/Editor/__tests__/Editor.spec.js +219 -0
  187. package/src/components/Editor/__tests__/EditorToolbar.spec.js +120 -0
  188. package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +45 -0
  189. package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +4020 -0
  190. package/src/components/Editor/withWorkflow.js +61 -0
  191. package/src/components/EditorWidgets/Unknown/UnknownControl.js +17 -0
  192. package/src/components/EditorWidgets/Unknown/UnknownPreview.js +19 -0
  193. package/src/components/EditorWidgets/index.js +5 -0
  194. package/src/components/MediaLibrary/EmptyMessage.js +28 -0
  195. package/src/components/MediaLibrary/MediaLibrary.js +407 -0
  196. package/src/components/MediaLibrary/MediaLibraryButtons.js +136 -0
  197. package/src/components/MediaLibrary/MediaLibraryCard.js +128 -0
  198. package/src/components/MediaLibrary/MediaLibraryCardGrid.js +199 -0
  199. package/src/components/MediaLibrary/MediaLibraryHeader.js +48 -0
  200. package/src/components/MediaLibrary/MediaLibraryModal.js +200 -0
  201. package/src/components/MediaLibrary/MediaLibrarySearch.js +61 -0
  202. package/src/components/MediaLibrary/MediaLibraryTop.js +143 -0
  203. package/src/components/MediaLibrary/__tests__/MediaLibraryButtons.spec.js +45 -0
  204. package/src/components/MediaLibrary/__tests__/MediaLibraryCard.spec.js +49 -0
  205. package/src/components/MediaLibrary/__tests__/__snapshots__/MediaLibraryCard.spec.js.snap +216 -0
  206. package/src/components/UI/DragDrop.js +66 -0
  207. package/src/components/UI/ErrorBoundary.js +209 -0
  208. package/src/components/UI/FileUploadButton.js +24 -0
  209. package/src/components/UI/Modal.js +109 -0
  210. package/src/components/UI/Notifications.tsx +83 -0
  211. package/src/components/UI/SettingsDropdown.js +103 -0
  212. package/src/components/UI/__tests__/ErrorBoundary.spec.js +57 -0
  213. package/src/components/UI/index.js +6 -0
  214. package/src/components/Workflow/Workflow.js +166 -0
  215. package/src/components/Workflow/WorkflowCard.js +177 -0
  216. package/src/components/Workflow/WorkflowList.js +269 -0
  217. package/src/constants/__tests__/configSchema.spec.js +511 -0
  218. package/src/constants/collectionTypes.ts +2 -0
  219. package/src/constants/collectionViews.js +2 -0
  220. package/src/constants/commitProps.ts +2 -0
  221. package/src/constants/configSchema.js +399 -0
  222. package/src/constants/fieldInference.tsx +78 -0
  223. package/src/constants/publishModes.ts +22 -0
  224. package/src/constants/validationErrorTypes.js +6 -0
  225. package/src/formats/__tests__/frontmatter.spec.js +429 -0
  226. package/src/formats/__tests__/toml.spec.js +9 -0
  227. package/src/formats/__tests__/yaml.spec.js +162 -0
  228. package/src/formats/formats.ts +86 -0
  229. package/src/formats/frontmatter.ts +150 -0
  230. package/src/formats/helpers.ts +14 -0
  231. package/src/formats/json.ts +9 -0
  232. package/src/formats/toml.ts +33 -0
  233. package/src/formats/yaml.ts +58 -0
  234. package/src/index.js +8 -0
  235. package/src/integrations/index.js +35 -0
  236. package/src/integrations/providers/algolia/implementation.js +176 -0
  237. package/src/integrations/providers/assetStore/implementation.js +147 -0
  238. package/src/lib/__tests__/formatters.spec.js +723 -0
  239. package/src/lib/__tests__/i18n.spec.js +792 -0
  240. package/src/lib/__tests__/phrases.spec.js +119 -0
  241. package/src/lib/__tests__/registry.spec.js +246 -0
  242. package/src/lib/__tests__/serializeEntryValues.spec.js +22 -0
  243. package/src/lib/__tests__/urlHelper.spec.js +138 -0
  244. package/src/lib/consoleError.js +7 -0
  245. package/src/lib/formatters.ts +267 -0
  246. package/src/lib/i18n.ts +427 -0
  247. package/src/lib/phrases.js +8 -0
  248. package/src/lib/registry.js +282 -0
  249. package/src/lib/serializeEntryValues.js +75 -0
  250. package/src/lib/textHelper.js +11 -0
  251. package/src/lib/urlHelper.ts +125 -0
  252. package/src/mediaLibrary.ts +51 -0
  253. package/src/reducers/__tests__/auth.spec.ts +38 -0
  254. package/src/reducers/__tests__/collections.spec.js +571 -0
  255. package/src/reducers/__tests__/config.spec.js +38 -0
  256. package/src/reducers/__tests__/entries.spec.js +694 -0
  257. package/src/reducers/__tests__/entryDraft.spec.js +198 -0
  258. package/src/reducers/__tests__/globalUI.js +43 -0
  259. package/src/reducers/__tests__/integrations.spec.ts +76 -0
  260. package/src/reducers/__tests__/mediaLibrary.spec.js +154 -0
  261. package/src/reducers/__tests__/medias.spec.ts +49 -0
  262. package/src/reducers/auth.ts +46 -0
  263. package/src/reducers/collections.ts +483 -0
  264. package/src/reducers/combinedReducer.ts +11 -0
  265. package/src/reducers/config.ts +38 -0
  266. package/src/reducers/cursors.js +36 -0
  267. package/src/reducers/deploys.ts +52 -0
  268. package/src/reducers/editorialWorkflow.ts +163 -0
  269. package/src/reducers/entries.ts +806 -0
  270. package/src/reducers/entryDraft.js +218 -0
  271. package/src/reducers/globalUI.ts +45 -0
  272. package/src/reducers/index.ts +82 -0
  273. package/src/reducers/integrations.ts +59 -0
  274. package/src/reducers/mediaLibrary.ts +296 -0
  275. package/src/reducers/medias.ts +66 -0
  276. package/src/reducers/notifications.ts +53 -0
  277. package/src/reducers/search.ts +89 -0
  278. package/src/reducers/status.ts +40 -0
  279. package/src/redux/index.ts +18 -0
  280. package/src/redux/middleware/waitUntilAction.ts +64 -0
  281. package/src/routing/__tests__/history.spec.ts +49 -0
  282. package/src/routing/history.ts +17 -0
  283. package/src/types/diacritics.d.ts +1 -0
  284. package/src/types/global.d.ts +8 -0
  285. package/src/types/immutable.ts +39 -0
  286. package/src/types/redux.ts +810 -0
  287. package/src/types/tomlify-j0.4.d.ts +13 -0
  288. package/src/valueObjects/AssetProxy.ts +48 -0
  289. package/src/valueObjects/EditorComponent.js +38 -0
  290. package/src/valueObjects/Entry.ts +63 -0
@@ -0,0 +1,934 @@
1
+ import { Map, List, fromJS } from 'immutable';
2
+
3
+ import {
4
+ resolveBackend,
5
+ Backend,
6
+ extractSearchFields,
7
+ expandSearchEntries,
8
+ mergeExpandedEntries,
9
+ } from '../backend';
10
+ import { getBackend } from '../lib/registry';
11
+ import { FOLDER, FILES } from '../constants/collectionTypes';
12
+
13
+ jest.mock('../lib/registry');
14
+ jest.mock('decap-cms-lib-util');
15
+ jest.mock('../lib/urlHelper');
16
+
17
+ describe('Backend', () => {
18
+ describe('filterEntries', () => {
19
+ let backend;
20
+
21
+ beforeEach(() => {
22
+ getBackend.mockReturnValue({
23
+ init: jest.fn(),
24
+ });
25
+ backend = resolveBackend({
26
+ backend: {
27
+ name: 'git-gateway',
28
+ },
29
+ });
30
+ });
31
+
32
+ it('filters string values', () => {
33
+ const result = backend.filterEntries(
34
+ {
35
+ entries: [
36
+ {
37
+ data: {
38
+ testField: 'testValue',
39
+ },
40
+ },
41
+ {
42
+ data: {
43
+ testField: 'testValue2',
44
+ },
45
+ },
46
+ ],
47
+ },
48
+ Map({ field: 'testField', value: 'testValue' }),
49
+ );
50
+
51
+ expect(result.length).toBe(1);
52
+ });
53
+
54
+ it('filters number values', () => {
55
+ const result = backend.filterEntries(
56
+ {
57
+ entries: [
58
+ {
59
+ data: {
60
+ testField: 42,
61
+ },
62
+ },
63
+ {
64
+ data: {
65
+ testField: 5,
66
+ },
67
+ },
68
+ ],
69
+ },
70
+ Map({ field: 'testField', value: 42 }),
71
+ );
72
+
73
+ expect(result.length).toBe(1);
74
+ });
75
+
76
+ it('filters boolean values', () => {
77
+ const result = backend.filterEntries(
78
+ {
79
+ entries: [
80
+ {
81
+ data: {
82
+ testField: false,
83
+ },
84
+ },
85
+ {
86
+ data: {
87
+ testField: true,
88
+ },
89
+ },
90
+ ],
91
+ },
92
+ Map({ field: 'testField', value: false }),
93
+ );
94
+
95
+ expect(result.length).toBe(1);
96
+ });
97
+
98
+ it('filters list values', () => {
99
+ const result = backend.filterEntries(
100
+ {
101
+ entries: [
102
+ {
103
+ data: {
104
+ testField: ['valueOne', 'valueTwo', 'testValue'],
105
+ },
106
+ },
107
+ {
108
+ data: {
109
+ testField: ['valueThree'],
110
+ },
111
+ },
112
+ ],
113
+ },
114
+ Map({ field: 'testField', value: 'testValue' }),
115
+ );
116
+
117
+ expect(result.length).toBe(1);
118
+ });
119
+ });
120
+
121
+ describe('getLocalDraftBackup', () => {
122
+ const { localForage, asyncLock } = require('decap-cms-lib-util');
123
+
124
+ asyncLock.mockImplementation(() => ({ acquire: jest.fn(), release: jest.fn() }));
125
+
126
+ beforeEach(() => {
127
+ jest.clearAllMocks();
128
+ });
129
+
130
+ it('should return empty object on no item', async () => {
131
+ const implementation = {
132
+ init: jest.fn(() => implementation),
133
+ };
134
+
135
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
136
+
137
+ const collection = Map({
138
+ name: 'posts',
139
+ });
140
+ const slug = 'slug';
141
+
142
+ localForage.getItem.mockReturnValue();
143
+
144
+ const result = await backend.getLocalDraftBackup(collection, slug);
145
+
146
+ expect(result).toEqual({});
147
+ expect(localForage.getItem).toHaveBeenCalledTimes(1);
148
+ expect(localForage.getItem).toHaveBeenCalledWith('backup.posts.slug');
149
+ });
150
+
151
+ it('should return empty object on item with empty content', async () => {
152
+ const implementation = {
153
+ init: jest.fn(() => implementation),
154
+ };
155
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
156
+
157
+ const collection = Map({
158
+ name: 'posts',
159
+ });
160
+ const slug = 'slug';
161
+
162
+ localForage.getItem.mockReturnValue({ raw: '' });
163
+
164
+ const result = await backend.getLocalDraftBackup(collection, slug);
165
+
166
+ expect(result).toEqual({});
167
+ expect(localForage.getItem).toHaveBeenCalledTimes(1);
168
+ expect(localForage.getItem).toHaveBeenCalledWith('backup.posts.slug');
169
+ });
170
+
171
+ it('should return backup entry, empty media files and assets when only raw property was saved', async () => {
172
+ const implementation = {
173
+ init: jest.fn(() => implementation),
174
+ };
175
+
176
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
177
+
178
+ const collection = Map({
179
+ name: 'posts',
180
+ });
181
+ const slug = 'slug';
182
+
183
+ localForage.getItem.mockReturnValue({
184
+ raw: '---\ntitle: "Hello World"\n---\n',
185
+ });
186
+
187
+ const result = await backend.getLocalDraftBackup(collection, slug);
188
+
189
+ expect(result).toEqual({
190
+ entry: {
191
+ author: '',
192
+ mediaFiles: [],
193
+ collection: 'posts',
194
+ slug: 'slug',
195
+ path: '',
196
+ partial: false,
197
+ raw: '---\ntitle: "Hello World"\n---\n',
198
+ data: { title: 'Hello World' },
199
+ meta: {},
200
+ i18n: {},
201
+ label: null,
202
+ isModification: null,
203
+ status: '',
204
+ updatedOn: '',
205
+ },
206
+ });
207
+ expect(localForage.getItem).toHaveBeenCalledTimes(1);
208
+ expect(localForage.getItem).toHaveBeenCalledWith('backup.posts.slug');
209
+ });
210
+
211
+ it('should return backup entry, media files and assets when all were backed up', async () => {
212
+ const implementation = {
213
+ init: jest.fn(() => implementation),
214
+ };
215
+
216
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
217
+
218
+ const collection = Map({
219
+ name: 'posts',
220
+ });
221
+ const slug = 'slug';
222
+
223
+ localForage.getItem.mockReturnValue({
224
+ raw: '---\ntitle: "Hello World"\n---\n',
225
+ mediaFiles: [{ id: '1' }],
226
+ });
227
+
228
+ const result = await backend.getLocalDraftBackup(collection, slug);
229
+
230
+ expect(result).toEqual({
231
+ entry: {
232
+ author: '',
233
+ mediaFiles: [{ id: '1' }],
234
+ collection: 'posts',
235
+ slug: 'slug',
236
+ path: '',
237
+ partial: false,
238
+ raw: '---\ntitle: "Hello World"\n---\n',
239
+ data: { title: 'Hello World' },
240
+ meta: {},
241
+ i18n: {},
242
+ label: null,
243
+ isModification: null,
244
+ status: '',
245
+ updatedOn: '',
246
+ },
247
+ });
248
+ expect(localForage.getItem).toHaveBeenCalledTimes(1);
249
+ expect(localForage.getItem).toHaveBeenCalledWith('backup.posts.slug');
250
+ });
251
+ });
252
+
253
+ describe('persistLocalDraftBackup', () => {
254
+ const { localForage } = require('decap-cms-lib-util');
255
+
256
+ beforeEach(() => {
257
+ jest.clearAllMocks();
258
+ });
259
+
260
+ it('should not persist empty entry', async () => {
261
+ const implementation = {
262
+ init: jest.fn(() => implementation),
263
+ };
264
+
265
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
266
+
267
+ backend.entryToRaw = jest.fn().mockReturnValue('');
268
+
269
+ const collection = Map({
270
+ name: 'posts',
271
+ });
272
+
273
+ const slug = 'slug';
274
+
275
+ const entry = Map({
276
+ slug,
277
+ });
278
+
279
+ await backend.persistLocalDraftBackup(entry, collection);
280
+
281
+ expect(backend.entryToRaw).toHaveBeenCalledTimes(1);
282
+ expect(backend.entryToRaw).toHaveBeenCalledWith(collection, entry);
283
+ expect(localForage.setItem).toHaveBeenCalledTimes(0);
284
+ });
285
+
286
+ it('should persist non empty entry', async () => {
287
+ const implementation = {
288
+ init: jest.fn(() => implementation),
289
+ };
290
+
291
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
292
+
293
+ backend.entryToRaw = jest.fn().mockReturnValue('content');
294
+
295
+ const collection = Map({
296
+ name: 'posts',
297
+ });
298
+
299
+ const slug = 'slug';
300
+
301
+ const entry = Map({
302
+ slug,
303
+ path: 'content/posts/entry.md',
304
+ mediaFiles: List([{ id: '1' }]),
305
+ });
306
+
307
+ await backend.persistLocalDraftBackup(entry, collection);
308
+
309
+ expect(backend.entryToRaw).toHaveBeenCalledTimes(1);
310
+ expect(backend.entryToRaw).toHaveBeenCalledWith(collection, entry);
311
+ expect(localForage.setItem).toHaveBeenCalledTimes(2);
312
+ expect(localForage.setItem).toHaveBeenCalledWith('backup.posts.slug', {
313
+ mediaFiles: [{ id: '1' }],
314
+ path: 'content/posts/entry.md',
315
+ raw: 'content',
316
+ });
317
+ expect(localForage.setItem).toHaveBeenCalledWith('backup', 'content');
318
+ });
319
+ });
320
+
321
+ describe('persistMedia', () => {
322
+ it('should persist media', async () => {
323
+ const persistMediaResult = {};
324
+ const implementation = {
325
+ init: jest.fn(() => implementation),
326
+ persistMedia: jest.fn().mockResolvedValue(persistMediaResult),
327
+ };
328
+ const config = { backend: { name: 'github' } };
329
+
330
+ const backend = new Backend(implementation, { config, backendName: config.backend.name });
331
+ const user = { login: 'login', name: 'name' };
332
+ backend.currentUser = jest.fn().mockResolvedValue(user);
333
+
334
+ const file = { path: 'static/media/image.png' };
335
+
336
+ const result = await backend.persistMedia(config, file);
337
+ expect(result).toBe(persistMediaResult);
338
+ expect(implementation.persistMedia).toHaveBeenCalledTimes(1);
339
+ expect(implementation.persistMedia).toHaveBeenCalledWith(
340
+ { path: 'static/media/image.png' },
341
+ { commitMessage: 'Upload “static/media/image.png”' },
342
+ );
343
+ });
344
+ });
345
+
346
+ describe('unpublishedEntry', () => {
347
+ it('should return unpublished entry', async () => {
348
+ const unpublishedEntryResult = {
349
+ diffs: [{ path: 'src/posts/index.md', newFile: false }, { path: 'netlify.png' }],
350
+ };
351
+ const implementation = {
352
+ init: jest.fn(() => implementation),
353
+ unpublishedEntry: jest.fn().mockResolvedValue(unpublishedEntryResult),
354
+ unpublishedEntryDataFile: jest
355
+ .fn()
356
+ .mockResolvedValueOnce('---\ntitle: "Hello World"\n---\n'),
357
+ unpublishedEntryMediaFile: jest.fn().mockResolvedValueOnce({ id: '1' }),
358
+ };
359
+ const config = {
360
+ media_folder: 'static/images',
361
+ };
362
+
363
+ const backend = new Backend(implementation, { config, backendName: 'github' });
364
+
365
+ const collection = fromJS({
366
+ name: 'posts',
367
+ folder: 'src/posts',
368
+ fields: [],
369
+ });
370
+
371
+ const state = {
372
+ config,
373
+ integrations: Map({}),
374
+ mediaLibrary: Map({}),
375
+ };
376
+
377
+ const slug = 'slug';
378
+
379
+ const result = await backend.unpublishedEntry(state, collection, slug);
380
+ expect(result).toEqual({
381
+ author: '',
382
+ collection: 'posts',
383
+ slug: '',
384
+ path: 'src/posts/index.md',
385
+ partial: false,
386
+ raw: '---\ntitle: "Hello World"\n---\n',
387
+ data: { title: 'Hello World' },
388
+ meta: { path: 'src/posts/index.md' },
389
+ i18n: {},
390
+ label: null,
391
+ isModification: true,
392
+ mediaFiles: [{ id: '1', draft: true }],
393
+ status: '',
394
+ updatedOn: '',
395
+ });
396
+ });
397
+ });
398
+
399
+ describe('generateUniqueSlug', () => {
400
+ beforeEach(() => {
401
+ jest.resetAllMocks();
402
+ });
403
+
404
+ it("should return unique slug when entry doesn't exist", async () => {
405
+ const { sanitizeSlug } = require('../lib/urlHelper');
406
+ sanitizeSlug.mockReturnValue('some-post-title');
407
+
408
+ const implementation = {
409
+ init: jest.fn(() => implementation),
410
+ getEntry: jest.fn(() => Promise.resolve()),
411
+ };
412
+
413
+ const collection = fromJS({
414
+ name: 'posts',
415
+ fields: [
416
+ {
417
+ name: 'title',
418
+ },
419
+ ],
420
+ type: FOLDER,
421
+ folder: 'posts',
422
+ slug: '{{slug}}',
423
+ path: 'sub_dir/{{slug}}',
424
+ });
425
+
426
+ const entry = Map({
427
+ title: 'some post title',
428
+ });
429
+
430
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
431
+
432
+ await expect(backend.generateUniqueSlug(collection, entry, Map({}), [])).resolves.toBe(
433
+ 'sub_dir/some-post-title',
434
+ );
435
+ });
436
+
437
+ it('should return unique slug when entry exists', async () => {
438
+ const { sanitizeSlug, sanitizeChar } = require('../lib/urlHelper');
439
+ sanitizeSlug.mockReturnValue('some-post-title');
440
+ sanitizeChar.mockReturnValue('-');
441
+
442
+ const implementation = {
443
+ init: jest.fn(() => implementation),
444
+ getEntry: jest.fn(),
445
+ };
446
+
447
+ implementation.getEntry.mockResolvedValueOnce({ data: 'data' });
448
+ implementation.getEntry.mockResolvedValueOnce();
449
+
450
+ const collection = fromJS({
451
+ name: 'posts',
452
+ fields: [
453
+ {
454
+ name: 'title',
455
+ },
456
+ ],
457
+ type: FOLDER,
458
+ folder: 'posts',
459
+ slug: '{{slug}}',
460
+ path: 'sub_dir/{{slug}}',
461
+ });
462
+
463
+ const entry = Map({
464
+ title: 'some post title',
465
+ });
466
+
467
+ const backend = new Backend(implementation, { config: {}, backendName: 'github' });
468
+
469
+ await expect(backend.generateUniqueSlug(collection, entry, Map({}), [])).resolves.toBe(
470
+ 'sub_dir/some-post-title-1',
471
+ );
472
+ });
473
+ });
474
+
475
+ describe('extractSearchFields', () => {
476
+ it('should extract slug', () => {
477
+ expect(extractSearchFields(['slug'])({ slug: 'entry-slug', data: {} })).toEqual(
478
+ ' entry-slug',
479
+ );
480
+ });
481
+
482
+ it('should extract path', () => {
483
+ expect(extractSearchFields(['path'])({ path: 'entry-path', data: {} })).toEqual(
484
+ ' entry-path',
485
+ );
486
+ });
487
+
488
+ it('should extract fields', () => {
489
+ expect(
490
+ extractSearchFields(['title', 'order'])({ data: { title: 'Entry Title', order: 5 } }),
491
+ ).toEqual(' Entry Title 5');
492
+ });
493
+
494
+ it('should extract nested fields', () => {
495
+ expect(
496
+ extractSearchFields(['nested.title'])({ data: { nested: { title: 'nested title' } } }),
497
+ ).toEqual(' nested title');
498
+ });
499
+ });
500
+
501
+ describe('search/query', () => {
502
+ const collections = [
503
+ fromJS({
504
+ name: 'posts',
505
+ folder: 'posts',
506
+ fields: [
507
+ { name: 'title', widget: 'string' },
508
+ { name: 'short_title', widget: 'string' },
509
+ { name: 'author', widget: 'string' },
510
+ { name: 'description', widget: 'string' },
511
+ { name: 'nested', widget: 'object', fields: { name: 'title', widget: 'string' } },
512
+ ],
513
+ }),
514
+ fromJS({
515
+ name: 'pages',
516
+ folder: 'pages',
517
+ fields: [
518
+ { name: 'title', widget: 'string' },
519
+ { name: 'short_title', widget: 'string' },
520
+ { name: 'author', widget: 'string' },
521
+ { name: 'description', widget: 'string' },
522
+ { name: 'nested', widget: 'object', fields: { name: 'title', widget: 'string' } },
523
+ ],
524
+ }),
525
+ ];
526
+
527
+ const posts = [
528
+ {
529
+ path: 'posts/find-me.md',
530
+ slug: 'find-me',
531
+ data: {
532
+ title: 'find me by title',
533
+ short_title: 'find me by short title',
534
+ author: 'find me by author',
535
+ description: 'find me by description',
536
+ nested: { title: 'find me by nested title' },
537
+ },
538
+ },
539
+ { path: 'posts/not-me.md', slug: 'not-me', data: { title: 'not me' } },
540
+ ];
541
+
542
+ const pages = [
543
+ {
544
+ path: 'pages/find-me.md',
545
+ slug: 'find-me',
546
+ data: {
547
+ title: 'find me by title',
548
+ short_title: 'find me by short title',
549
+ author: 'find me by author',
550
+ description: 'find me by description',
551
+ nested: { title: 'find me by nested title' },
552
+ },
553
+ },
554
+ { path: 'pages/not-me.md', slug: 'not-me', data: { title: 'not me' } },
555
+ ];
556
+
557
+ const files = [
558
+ {
559
+ path: 'files/file1.md',
560
+ slug: 'file1',
561
+ data: {
562
+ author: 'find me by author',
563
+ },
564
+ },
565
+ {
566
+ path: 'files/file2.md',
567
+ slug: 'file2',
568
+ data: {
569
+ other: 'find me by other',
570
+ },
571
+ },
572
+ ];
573
+
574
+ const implementation = {
575
+ init: jest.fn(() => implementation),
576
+ };
577
+
578
+ let backend;
579
+ beforeEach(() => {
580
+ backend = new Backend(implementation, { config: {}, backendName: 'github' });
581
+ backend.listAllEntries = jest.fn(collection => {
582
+ if (collection.get('name') === 'posts') {
583
+ return Promise.resolve(posts);
584
+ }
585
+ if (collection.get('name') === 'pages') {
586
+ return Promise.resolve(pages);
587
+ }
588
+ if (collection.get('name') === 'files') {
589
+ return Promise.resolve(files);
590
+ }
591
+ return Promise.resolve([]);
592
+ });
593
+ });
594
+
595
+ it('should search collections by title', async () => {
596
+ const results = await backend.search(collections, 'find me by title');
597
+
598
+ expect(results).toEqual({
599
+ entries: [posts[0], pages[0]],
600
+ });
601
+ });
602
+
603
+ it('should search collections by short title', async () => {
604
+ const results = await backend.search(collections, 'find me by short title');
605
+
606
+ expect(results).toEqual({
607
+ entries: [posts[0], pages[0]],
608
+ });
609
+ });
610
+
611
+ it('should search collections by author', async () => {
612
+ const results = await backend.search(collections, 'find me by author');
613
+
614
+ expect(results).toEqual({
615
+ entries: [posts[0], pages[0]],
616
+ });
617
+ });
618
+
619
+ it('should search collections by summary description', async () => {
620
+ const results = await backend.search(
621
+ collections.map(c => c.set('summary', '{{description}}')),
622
+ 'find me by description',
623
+ );
624
+
625
+ expect(results).toEqual({
626
+ entries: [posts[0], pages[0]],
627
+ });
628
+ });
629
+
630
+ it('should search in file collection using top level fields', async () => {
631
+ const collections = [
632
+ fromJS({
633
+ name: 'files',
634
+ files: [
635
+ {
636
+ name: 'file1',
637
+ fields: [{ name: 'author', widget: 'string' }],
638
+ },
639
+ {
640
+ name: 'file2',
641
+ fields: [{ name: 'other', widget: 'string' }],
642
+ },
643
+ ],
644
+ type: FILES,
645
+ }),
646
+ ];
647
+
648
+ expect(await backend.search(collections, 'find me by author')).toEqual({
649
+ entries: [files[0]],
650
+ });
651
+ expect(await backend.search(collections, 'find me by other')).toEqual({
652
+ entries: [files[1]],
653
+ });
654
+ });
655
+
656
+ it('should query collections by title', async () => {
657
+ const results = await backend.query(collections[0], ['title'], 'find me by title');
658
+
659
+ expect(results).toEqual({
660
+ hits: [posts[0]],
661
+ query: 'find me by title',
662
+ });
663
+ });
664
+
665
+ it('should query collections by slug', async () => {
666
+ const results = await backend.query(collections[0], ['slug'], 'find-me');
667
+
668
+ expect(results).toEqual({
669
+ hits: [posts[0]],
670
+ query: 'find-me',
671
+ });
672
+ });
673
+
674
+ it('should query collections by path', async () => {
675
+ const results = await backend.query(collections[0], ['path'], 'posts/find-me.md');
676
+
677
+ expect(results).toEqual({
678
+ hits: [posts[0]],
679
+ query: 'posts/find-me.md',
680
+ });
681
+ });
682
+
683
+ it('should query collections by nested field', async () => {
684
+ const results = await backend.query(
685
+ collections[0],
686
+ ['nested.title'],
687
+ 'find me by nested title',
688
+ );
689
+
690
+ expect(results).toEqual({
691
+ hits: [posts[0]],
692
+ query: 'find me by nested title',
693
+ });
694
+ });
695
+ });
696
+
697
+ describe('expandSearchEntries', () => {
698
+ it('should expand entry with list to multiple entries', () => {
699
+ const entry = {
700
+ data: {
701
+ field: {
702
+ nested: {
703
+ list: [
704
+ { id: 1, name: '1' },
705
+ { id: 2, name: '2' },
706
+ ],
707
+ },
708
+ },
709
+ list: [1, 2],
710
+ },
711
+ };
712
+
713
+ expect(expandSearchEntries([entry], ['list.*', 'field.nested.list.*.name'])).toEqual([
714
+ {
715
+ data: {
716
+ field: {
717
+ nested: {
718
+ list: [
719
+ { id: 1, name: '1' },
720
+ { id: 2, name: '2' },
721
+ ],
722
+ },
723
+ },
724
+ list: [1, 2],
725
+ },
726
+ field: 'list.0',
727
+ },
728
+ {
729
+ data: {
730
+ field: {
731
+ nested: {
732
+ list: [
733
+ { id: 1, name: '1' },
734
+ { id: 2, name: '2' },
735
+ ],
736
+ },
737
+ },
738
+ list: [1, 2],
739
+ },
740
+ field: 'list.1',
741
+ },
742
+ {
743
+ data: {
744
+ field: {
745
+ nested: {
746
+ list: [
747
+ { id: 1, name: '1' },
748
+ { id: 2, name: '2' },
749
+ ],
750
+ },
751
+ },
752
+ list: [1, 2],
753
+ },
754
+ field: 'field.nested.list.0.name',
755
+ },
756
+ {
757
+ data: {
758
+ field: {
759
+ nested: {
760
+ list: [
761
+ { id: 1, name: '1' },
762
+ { id: 2, name: '2' },
763
+ ],
764
+ },
765
+ },
766
+ list: [1, 2],
767
+ },
768
+ field: 'field.nested.list.1.name',
769
+ },
770
+ ]);
771
+ });
772
+ });
773
+
774
+ describe('mergeExpandedEntries', () => {
775
+ it('should merge entries and filter data', () => {
776
+ const expanded = [
777
+ {
778
+ data: {
779
+ field: {
780
+ nested: {
781
+ list: [
782
+ { id: 1, name: '1' },
783
+ { id: 2, name: '2' },
784
+ { id: 3, name: '3' },
785
+ { id: 4, name: '4' },
786
+ ],
787
+ },
788
+ },
789
+ list: [1, 2],
790
+ },
791
+ field: 'field.nested.list.0.name',
792
+ },
793
+ {
794
+ data: {
795
+ field: {
796
+ nested: {
797
+ list: [
798
+ { id: 1, name: '1' },
799
+ { id: 2, name: '2' },
800
+ { id: 3, name: '3' },
801
+ { id: 4, name: '4' },
802
+ ],
803
+ },
804
+ },
805
+ list: [1, 2],
806
+ },
807
+ field: 'field.nested.list.3.name',
808
+ },
809
+ ];
810
+
811
+ expect(mergeExpandedEntries(expanded)).toEqual([
812
+ {
813
+ data: {
814
+ field: {
815
+ nested: {
816
+ list: [
817
+ { id: 1, name: '1' },
818
+ { id: 4, name: '4' },
819
+ ],
820
+ },
821
+ },
822
+ list: [1, 2],
823
+ },
824
+ },
825
+ ]);
826
+ });
827
+
828
+ it('should merge entries and filter data based on different fields', () => {
829
+ const expanded = [
830
+ {
831
+ data: {
832
+ field: {
833
+ nested: {
834
+ list: [
835
+ { id: 1, name: '1' },
836
+ { id: 2, name: '2' },
837
+ { id: 3, name: '3' },
838
+ { id: 4, name: '4' },
839
+ ],
840
+ },
841
+ },
842
+ list: [1, 2],
843
+ },
844
+ field: 'field.nested.list.0.name',
845
+ },
846
+ {
847
+ data: {
848
+ field: {
849
+ nested: {
850
+ list: [
851
+ { id: 1, name: '1' },
852
+ { id: 2, name: '2' },
853
+ { id: 3, name: '3' },
854
+ { id: 4, name: '4' },
855
+ ],
856
+ },
857
+ },
858
+ list: [1, 2],
859
+ },
860
+ field: 'field.nested.list.3.name',
861
+ },
862
+ {
863
+ data: {
864
+ field: {
865
+ nested: {
866
+ list: [
867
+ { id: 1, name: '1' },
868
+ { id: 2, name: '2' },
869
+ { id: 3, name: '3' },
870
+ { id: 4, name: '4' },
871
+ ],
872
+ },
873
+ },
874
+ list: [1, 2],
875
+ },
876
+ field: 'list.1',
877
+ },
878
+ ];
879
+
880
+ expect(mergeExpandedEntries(expanded)).toEqual([
881
+ {
882
+ data: {
883
+ field: {
884
+ nested: {
885
+ list: [
886
+ { id: 1, name: '1' },
887
+ { id: 4, name: '4' },
888
+ ],
889
+ },
890
+ },
891
+ list: [2],
892
+ },
893
+ },
894
+ ]);
895
+ });
896
+
897
+ it('should merge entries and keep sort by entry index', () => {
898
+ const expanded = [
899
+ {
900
+ data: {
901
+ list: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12],
902
+ },
903
+ field: 'list.5',
904
+ },
905
+ {
906
+ data: {
907
+ list: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12],
908
+ },
909
+ field: 'list.0',
910
+ },
911
+ {
912
+ data: {
913
+ list: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12],
914
+ },
915
+ field: 'list.11',
916
+ },
917
+ {
918
+ data: {
919
+ list: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12],
920
+ },
921
+ field: 'list.1',
922
+ },
923
+ ];
924
+
925
+ expect(mergeExpandedEntries(expanded)).toEqual([
926
+ {
927
+ data: {
928
+ list: [5, 0, 11, 1],
929
+ },
930
+ },
931
+ ]);
932
+ });
933
+ });
934
+ });