canopycms 0.0.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 (1276) hide show
  1. package/README.md +404 -0
  2. package/dist/__integration__/fixtures/content-seeds.d.ts +43 -0
  3. package/dist/__integration__/fixtures/content-seeds.d.ts.map +1 -0
  4. package/dist/__integration__/fixtures/content-seeds.js +99 -0
  5. package/dist/__integration__/fixtures/content-seeds.js.map +1 -0
  6. package/dist/__integration__/fixtures/schemas.d.ts +12 -0
  7. package/dist/__integration__/fixtures/schemas.d.ts.map +1 -0
  8. package/dist/__integration__/fixtures/schemas.js +65 -0
  9. package/dist/__integration__/fixtures/schemas.js.map +1 -0
  10. package/dist/__integration__/test-utils/api-client.d.ts +123 -0
  11. package/dist/__integration__/test-utils/api-client.d.ts.map +1 -0
  12. package/dist/__integration__/test-utils/api-client.js +118 -0
  13. package/dist/__integration__/test-utils/api-client.js.map +1 -0
  14. package/dist/__integration__/test-utils/multi-user.d.ts +25 -0
  15. package/dist/__integration__/test-utils/multi-user.d.ts.map +1 -0
  16. package/dist/__integration__/test-utils/multi-user.js +105 -0
  17. package/dist/__integration__/test-utils/multi-user.js.map +1 -0
  18. package/dist/__integration__/test-utils/test-workspace.d.ts +25 -0
  19. package/dist/__integration__/test-utils/test-workspace.d.ts.map +1 -0
  20. package/dist/__integration__/test-utils/test-workspace.js +102 -0
  21. package/dist/__integration__/test-utils/test-workspace.js.map +1 -0
  22. package/dist/ai/generate.d.ts +31 -0
  23. package/dist/ai/generate.d.ts.map +1 -0
  24. package/dist/ai/generate.js +277 -0
  25. package/dist/ai/generate.js.map +1 -0
  26. package/dist/ai/handler.d.ts +38 -0
  27. package/dist/ai/handler.d.ts.map +1 -0
  28. package/dist/ai/handler.js +99 -0
  29. package/dist/ai/handler.js.map +1 -0
  30. package/dist/ai/index.d.ts +13 -0
  31. package/dist/ai/index.d.ts.map +1 -0
  32. package/dist/ai/index.js +10 -0
  33. package/dist/ai/index.js.map +1 -0
  34. package/dist/ai/json-to-markdown.d.ts +15 -0
  35. package/dist/ai/json-to-markdown.d.ts.map +1 -0
  36. package/dist/ai/json-to-markdown.js +318 -0
  37. package/dist/ai/json-to-markdown.js.map +1 -0
  38. package/dist/ai/resolve-branch.d.ts +13 -0
  39. package/dist/ai/resolve-branch.d.ts.map +1 -0
  40. package/dist/ai/resolve-branch.js +27 -0
  41. package/dist/ai/resolve-branch.js.map +1 -0
  42. package/dist/ai/types.d.ts +133 -0
  43. package/dist/ai/types.d.ts.map +1 -0
  44. package/dist/ai/types.js +14 -0
  45. package/dist/ai/types.js.map +1 -0
  46. package/dist/api/__test__/mock-client.d.ts +194 -0
  47. package/dist/api/__test__/mock-client.d.ts.map +1 -0
  48. package/dist/api/__test__/mock-client.js +322 -0
  49. package/dist/api/__test__/mock-client.js.map +1 -0
  50. package/dist/api/assets.d.ts +52 -0
  51. package/dist/api/assets.d.ts.map +1 -0
  52. package/dist/api/assets.js +106 -0
  53. package/dist/api/assets.js.map +1 -0
  54. package/dist/api/branch-merge.d.ts +28 -0
  55. package/dist/api/branch-merge.d.ts.map +1 -0
  56. package/dist/api/branch-merge.js +87 -0
  57. package/dist/api/branch-merge.js.map +1 -0
  58. package/dist/api/branch-review.d.ts +42 -0
  59. package/dist/api/branch-review.d.ts.map +1 -0
  60. package/dist/api/branch-review.js +102 -0
  61. package/dist/api/branch-review.js.map +1 -0
  62. package/dist/api/branch-status.d.ts +81 -0
  63. package/dist/api/branch-status.d.ts.map +1 -0
  64. package/dist/api/branch-status.js +125 -0
  65. package/dist/api/branch-status.js.map +1 -0
  66. package/dist/api/branch-withdraw.d.ts +18 -0
  67. package/dist/api/branch-withdraw.d.ts.map +1 -0
  68. package/dist/api/branch-withdraw.js +64 -0
  69. package/dist/api/branch-withdraw.js.map +1 -0
  70. package/dist/api/branch.d.ts +188 -0
  71. package/dist/api/branch.d.ts.map +1 -0
  72. package/dist/api/branch.js +381 -0
  73. package/dist/api/branch.js.map +1 -0
  74. package/dist/api/client.d.ts +289 -0
  75. package/dist/api/client.d.ts.map +1 -0
  76. package/dist/api/client.js +412 -0
  77. package/dist/api/client.js.map +1 -0
  78. package/dist/api/comments.d.ts +84 -0
  79. package/dist/api/comments.d.ts.map +1 -0
  80. package/dist/api/comments.js +142 -0
  81. package/dist/api/comments.js.map +1 -0
  82. package/dist/api/content.d.ts +154 -0
  83. package/dist/api/content.d.ts.map +1 -0
  84. package/dist/api/content.js +308 -0
  85. package/dist/api/content.js.map +1 -0
  86. package/dist/api/entries.d.ts +155 -0
  87. package/dist/api/entries.d.ts.map +1 -0
  88. package/dist/api/entries.js +466 -0
  89. package/dist/api/entries.js.map +1 -0
  90. package/dist/api/github-sync.d.ts +22 -0
  91. package/dist/api/github-sync.d.ts.map +1 -0
  92. package/dist/api/github-sync.js +121 -0
  93. package/dist/api/github-sync.js.map +1 -0
  94. package/dist/api/groups.d.ts +91 -0
  95. package/dist/api/groups.d.ts.map +1 -0
  96. package/dist/api/groups.js +270 -0
  97. package/dist/api/groups.js.map +1 -0
  98. package/dist/api/guards.d.ts +58 -0
  99. package/dist/api/guards.d.ts.map +1 -0
  100. package/dist/api/guards.js +166 -0
  101. package/dist/api/guards.js.map +1 -0
  102. package/dist/api/index.d.ts +18 -0
  103. package/dist/api/index.d.ts.map +1 -0
  104. package/dist/api/index.js +8 -0
  105. package/dist/api/index.js.map +1 -0
  106. package/dist/api/permissions.d.ts +278 -0
  107. package/dist/api/permissions.d.ts.map +1 -0
  108. package/dist/api/permissions.js +257 -0
  109. package/dist/api/permissions.js.map +1 -0
  110. package/dist/api/reference-options.d.ts +27 -0
  111. package/dist/api/reference-options.d.ts.map +1 -0
  112. package/dist/api/reference-options.js +88 -0
  113. package/dist/api/reference-options.js.map +1 -0
  114. package/dist/api/resolve-references.d.ts +32 -0
  115. package/dist/api/resolve-references.d.ts.map +1 -0
  116. package/dist/api/resolve-references.js +77 -0
  117. package/dist/api/resolve-references.js.map +1 -0
  118. package/dist/api/route-builder.d.ts +120 -0
  119. package/dist/api/route-builder.d.ts.map +1 -0
  120. package/dist/api/route-builder.js +108 -0
  121. package/dist/api/route-builder.js.map +1 -0
  122. package/dist/api/schema.d.ts +713 -0
  123. package/dist/api/schema.d.ts.map +1 -0
  124. package/dist/api/schema.js +656 -0
  125. package/dist/api/schema.js.map +1 -0
  126. package/dist/api/settings-helpers.d.ts +33 -0
  127. package/dist/api/settings-helpers.d.ts.map +1 -0
  128. package/dist/api/settings-helpers.js +61 -0
  129. package/dist/api/settings-helpers.js.map +1 -0
  130. package/dist/api/types.d.ts +37 -0
  131. package/dist/api/types.d.ts.map +1 -0
  132. package/dist/api/types.js +2 -0
  133. package/dist/api/types.js.map +1 -0
  134. package/dist/api/user.d.ts +9 -0
  135. package/dist/api/user.d.ts.map +1 -0
  136. package/dist/api/user.js +34 -0
  137. package/dist/api/user.js.map +1 -0
  138. package/dist/api/validators.d.ts +85 -0
  139. package/dist/api/validators.d.ts.map +1 -0
  140. package/dist/api/validators.js +160 -0
  141. package/dist/api/validators.js.map +1 -0
  142. package/dist/asset-store.d.ts +32 -0
  143. package/dist/asset-store.d.ts.map +1 -0
  144. package/dist/asset-store.js +84 -0
  145. package/dist/asset-store.js.map +1 -0
  146. package/dist/auth/cache.d.ts +8 -0
  147. package/dist/auth/cache.d.ts.map +1 -0
  148. package/dist/auth/cache.js +8 -0
  149. package/dist/auth/cache.js.map +1 -0
  150. package/dist/auth/caching-auth-plugin.d.ts +42 -0
  151. package/dist/auth/caching-auth-plugin.d.ts.map +1 -0
  152. package/dist/auth/caching-auth-plugin.js +78 -0
  153. package/dist/auth/caching-auth-plugin.js.map +1 -0
  154. package/dist/auth/context-helpers.d.ts +31 -0
  155. package/dist/auth/context-helpers.d.ts.map +1 -0
  156. package/dist/auth/context-helpers.js +53 -0
  157. package/dist/auth/context-helpers.js.map +1 -0
  158. package/dist/auth/file-based-auth-cache.d.ts +46 -0
  159. package/dist/auth/file-based-auth-cache.d.ts.map +1 -0
  160. package/dist/auth/file-based-auth-cache.js +226 -0
  161. package/dist/auth/file-based-auth-cache.js.map +1 -0
  162. package/dist/auth/index.d.ts +6 -0
  163. package/dist/auth/index.d.ts.map +1 -0
  164. package/dist/auth/index.js +2 -0
  165. package/dist/auth/index.js.map +1 -0
  166. package/dist/auth/plugin.d.ts +48 -0
  167. package/dist/auth/plugin.d.ts.map +1 -0
  168. package/dist/auth/plugin.js +2 -0
  169. package/dist/auth/plugin.js.map +1 -0
  170. package/dist/auth/types.d.ts +36 -0
  171. package/dist/auth/types.d.ts.map +1 -0
  172. package/dist/auth/types.js +2 -0
  173. package/dist/auth/types.js.map +1 -0
  174. package/dist/authorization/branch.d.ts +29 -0
  175. package/dist/authorization/branch.d.ts.map +1 -0
  176. package/dist/authorization/branch.js +67 -0
  177. package/dist/authorization/branch.js.map +1 -0
  178. package/dist/authorization/content.d.ts +38 -0
  179. package/dist/authorization/content.d.ts.map +1 -0
  180. package/dist/authorization/content.js +65 -0
  181. package/dist/authorization/content.js.map +1 -0
  182. package/dist/authorization/groups/index.d.ts +6 -0
  183. package/dist/authorization/groups/index.d.ts.map +1 -0
  184. package/dist/authorization/groups/index.js +6 -0
  185. package/dist/authorization/groups/index.js.map +1 -0
  186. package/dist/authorization/groups/loader.d.ts +24 -0
  187. package/dist/authorization/groups/loader.d.ts.map +1 -0
  188. package/dist/authorization/groups/loader.js +98 -0
  189. package/dist/authorization/groups/loader.js.map +1 -0
  190. package/dist/authorization/groups/schema.d.ts +67 -0
  191. package/dist/authorization/groups/schema.d.ts.map +1 -0
  192. package/dist/authorization/groups/schema.js +31 -0
  193. package/dist/authorization/groups/schema.js.map +1 -0
  194. package/dist/authorization/helpers.d.ts +35 -0
  195. package/dist/authorization/helpers.d.ts.map +1 -0
  196. package/dist/authorization/helpers.js +42 -0
  197. package/dist/authorization/helpers.js.map +1 -0
  198. package/dist/authorization/index.d.ts +36 -0
  199. package/dist/authorization/index.d.ts.map +1 -0
  200. package/dist/authorization/index.js +42 -0
  201. package/dist/authorization/index.js.map +1 -0
  202. package/dist/authorization/path.d.ts +29 -0
  203. package/dist/authorization/path.d.ts.map +1 -0
  204. package/dist/authorization/path.js +71 -0
  205. package/dist/authorization/path.js.map +1 -0
  206. package/dist/authorization/permissions/index.d.ts +6 -0
  207. package/dist/authorization/permissions/index.d.ts.map +1 -0
  208. package/dist/authorization/permissions/index.js +6 -0
  209. package/dist/authorization/permissions/index.js.map +1 -0
  210. package/dist/authorization/permissions/loader.d.ts +33 -0
  211. package/dist/authorization/permissions/loader.d.ts.map +1 -0
  212. package/dist/authorization/permissions/loader.js +85 -0
  213. package/dist/authorization/permissions/loader.js.map +1 -0
  214. package/dist/authorization/permissions/schema.d.ts +133 -0
  215. package/dist/authorization/permissions/schema.d.ts.map +1 -0
  216. package/dist/authorization/permissions/schema.js +56 -0
  217. package/dist/authorization/permissions/schema.js.map +1 -0
  218. package/dist/authorization/test-utils.d.ts +14 -0
  219. package/dist/authorization/test-utils.d.ts.map +1 -0
  220. package/dist/authorization/test-utils.js +13 -0
  221. package/dist/authorization/test-utils.js.map +1 -0
  222. package/dist/authorization/types.d.ts +55 -0
  223. package/dist/authorization/types.d.ts.map +1 -0
  224. package/dist/authorization/types.js +8 -0
  225. package/dist/authorization/types.js.map +1 -0
  226. package/dist/authorization/validation.d.ts +32 -0
  227. package/dist/authorization/validation.d.ts.map +1 -0
  228. package/dist/authorization/validation.js +53 -0
  229. package/dist/authorization/validation.js.map +1 -0
  230. package/dist/branch-metadata.d.ts +52 -0
  231. package/dist/branch-metadata.d.ts.map +1 -0
  232. package/dist/branch-metadata.js +125 -0
  233. package/dist/branch-metadata.js.map +1 -0
  234. package/dist/branch-registry.d.ts +44 -0
  235. package/dist/branch-registry.d.ts.map +1 -0
  236. package/dist/branch-registry.js +129 -0
  237. package/dist/branch-registry.js.map +1 -0
  238. package/dist/branch-schema-cache.d.ts +60 -0
  239. package/dist/branch-schema-cache.d.ts.map +1 -0
  240. package/dist/branch-schema-cache.js +142 -0
  241. package/dist/branch-schema-cache.js.map +1 -0
  242. package/dist/branch-workspace.d.ts +25 -0
  243. package/dist/branch-workspace.d.ts.map +1 -0
  244. package/dist/branch-workspace.js +87 -0
  245. package/dist/branch-workspace.js.map +1 -0
  246. package/dist/build/generate-ai-content.d.ts +28 -0
  247. package/dist/build/generate-ai-content.d.ts.map +1 -0
  248. package/dist/build/generate-ai-content.js +56 -0
  249. package/dist/build/generate-ai-content.js.map +1 -0
  250. package/dist/build/index.d.ts +8 -0
  251. package/dist/build/index.d.ts.map +1 -0
  252. package/dist/build/index.js +7 -0
  253. package/dist/build/index.js.map +1 -0
  254. package/dist/build-mode.d.ts +12 -0
  255. package/dist/build-mode.d.ts.map +1 -0
  256. package/dist/build-mode.js +25 -0
  257. package/dist/build-mode.js.map +1 -0
  258. package/dist/cli/generate-ai-content.d.ts +13 -0
  259. package/dist/cli/generate-ai-content.d.ts.map +1 -0
  260. package/dist/cli/generate-ai-content.js +78 -0
  261. package/dist/cli/generate-ai-content.js.map +1 -0
  262. package/dist/cli/init.d.ts +34 -0
  263. package/dist/cli/init.d.ts.map +1 -0
  264. package/dist/cli/init.js +372 -0
  265. package/dist/cli/init.js.map +1 -0
  266. package/dist/cli/templates.d.ts +20 -0
  267. package/dist/cli/templates.d.ts.map +1 -0
  268. package/dist/cli/templates.js +38 -0
  269. package/dist/cli/templates.js.map +1 -0
  270. package/dist/client.d.ts +11 -0
  271. package/dist/client.d.ts.map +1 -0
  272. package/dist/client.js +11 -0
  273. package/dist/client.js.map +1 -0
  274. package/dist/comment-store.d.ts +94 -0
  275. package/dist/comment-store.d.ts.map +1 -0
  276. package/dist/comment-store.js +234 -0
  277. package/dist/comment-store.js.map +1 -0
  278. package/dist/config/flatten.d.ts +29 -0
  279. package/dist/config/flatten.d.ts.map +1 -0
  280. package/dist/config/flatten.js +153 -0
  281. package/dist/config/flatten.js.map +1 -0
  282. package/dist/config/helpers.d.ts +48 -0
  283. package/dist/config/helpers.d.ts.map +1 -0
  284. package/dist/config/helpers.js +138 -0
  285. package/dist/config/helpers.js.map +1 -0
  286. package/dist/config/index.d.ts +21 -0
  287. package/dist/config/index.d.ts.map +1 -0
  288. package/dist/config/index.js +23 -0
  289. package/dist/config/index.js.map +1 -0
  290. package/dist/config/schemas/collection.d.ts +94 -0
  291. package/dist/config/schemas/collection.d.ts.map +1 -0
  292. package/dist/config/schemas/collection.js +57 -0
  293. package/dist/config/schemas/collection.js.map +1 -0
  294. package/dist/config/schemas/config.d.ts +304 -0
  295. package/dist/config/schemas/config.d.ts.map +1 -0
  296. package/dist/config/schemas/config.js +70 -0
  297. package/dist/config/schemas/config.js.map +1 -0
  298. package/dist/config/schemas/field.d.ts +276 -0
  299. package/dist/config/schemas/field.d.ts.map +1 -0
  300. package/dist/config/schemas/field.js +90 -0
  301. package/dist/config/schemas/field.js.map +1 -0
  302. package/dist/config/schemas/media.d.ts +48 -0
  303. package/dist/config/schemas/media.d.ts.map +1 -0
  304. package/dist/config/schemas/media.js +26 -0
  305. package/dist/config/schemas/media.js.map +1 -0
  306. package/dist/config/schemas/permissions.d.ts +76 -0
  307. package/dist/config/schemas/permissions.d.ts.map +1 -0
  308. package/dist/config/schemas/permissions.js +17 -0
  309. package/dist/config/schemas/permissions.js.map +1 -0
  310. package/dist/config/types.d.ts +272 -0
  311. package/dist/config/types.d.ts.map +1 -0
  312. package/dist/config/types.js +24 -0
  313. package/dist/config/types.js.map +1 -0
  314. package/dist/config/validation.d.ts +19 -0
  315. package/dist/config/validation.d.ts.map +1 -0
  316. package/dist/config/validation.js +66 -0
  317. package/dist/config/validation.js.map +1 -0
  318. package/dist/config-test.d.ts +18 -0
  319. package/dist/config-test.d.ts.map +1 -0
  320. package/dist/config-test.js +42 -0
  321. package/dist/config-test.js.map +1 -0
  322. package/dist/config.d.ts +10 -0
  323. package/dist/config.d.ts.map +1 -0
  324. package/dist/config.js +11 -0
  325. package/dist/config.js.map +1 -0
  326. package/dist/content-id-index.d.ts +152 -0
  327. package/dist/content-id-index.d.ts.map +1 -0
  328. package/dist/content-id-index.js +416 -0
  329. package/dist/content-id-index.js.map +1 -0
  330. package/dist/content-reader.d.ts +36 -0
  331. package/dist/content-reader.d.ts.map +1 -0
  332. package/dist/content-reader.js +149 -0
  333. package/dist/content-reader.js.map +1 -0
  334. package/dist/content-store.d.ts +129 -0
  335. package/dist/content-store.d.ts.map +1 -0
  336. package/dist/content-store.js +611 -0
  337. package/dist/content-store.js.map +1 -0
  338. package/dist/context.d.ts +44 -0
  339. package/dist/context.d.ts.map +1 -0
  340. package/dist/context.js +69 -0
  341. package/dist/context.js.map +1 -0
  342. package/dist/editor/BranchManager.d.ts +62 -0
  343. package/dist/editor/BranchManager.d.ts.map +1 -0
  344. package/dist/editor/BranchManager.js +87 -0
  345. package/dist/editor/BranchManager.js.map +1 -0
  346. package/dist/editor/BranchManager.stories.d.ts +8 -0
  347. package/dist/editor/BranchManager.stories.d.ts.map +1 -0
  348. package/dist/editor/BranchManager.stories.js +74 -0
  349. package/dist/editor/BranchManager.stories.js.map +1 -0
  350. package/dist/editor/CanopyEditor.d.ts +10 -0
  351. package/dist/editor/CanopyEditor.d.ts.map +1 -0
  352. package/dist/editor/CanopyEditor.js +20 -0
  353. package/dist/editor/CanopyEditor.js.map +1 -0
  354. package/dist/editor/CanopyEditor.stories.d.ts +7 -0
  355. package/dist/editor/CanopyEditor.stories.d.ts.map +1 -0
  356. package/dist/editor/CanopyEditor.stories.js +99 -0
  357. package/dist/editor/CanopyEditor.stories.js.map +1 -0
  358. package/dist/editor/CanopyEditorPage.d.ts +12 -0
  359. package/dist/editor/CanopyEditorPage.d.ts.map +1 -0
  360. package/dist/editor/CanopyEditorPage.js +13 -0
  361. package/dist/editor/CanopyEditorPage.js.map +1 -0
  362. package/dist/editor/CommentsPanel.d.ts +21 -0
  363. package/dist/editor/CommentsPanel.d.ts.map +1 -0
  364. package/dist/editor/CommentsPanel.js +77 -0
  365. package/dist/editor/CommentsPanel.js.map +1 -0
  366. package/dist/editor/CommentsPanel.stories.d.ts +10 -0
  367. package/dist/editor/CommentsPanel.stories.d.ts.map +1 -0
  368. package/dist/editor/CommentsPanel.stories.js +175 -0
  369. package/dist/editor/CommentsPanel.stories.js.map +1 -0
  370. package/dist/editor/Editor.d.ts +72 -0
  371. package/dist/editor/Editor.d.ts.map +1 -0
  372. package/dist/editor/Editor.js +568 -0
  373. package/dist/editor/Editor.js.map +1 -0
  374. package/dist/editor/Editor.stories.d.ts +7 -0
  375. package/dist/editor/Editor.stories.d.ts.map +1 -0
  376. package/dist/editor/Editor.stories.js +95 -0
  377. package/dist/editor/Editor.stories.js.map +1 -0
  378. package/dist/editor/EditorPanes.d.ts +11 -0
  379. package/dist/editor/EditorPanes.d.ts.map +1 -0
  380. package/dist/editor/EditorPanes.js +99 -0
  381. package/dist/editor/EditorPanes.js.map +1 -0
  382. package/dist/editor/EditorPanes.stories.d.ts +7 -0
  383. package/dist/editor/EditorPanes.stories.d.ts.map +1 -0
  384. package/dist/editor/EditorPanes.stories.js +116 -0
  385. package/dist/editor/EditorPanes.stories.js.map +1 -0
  386. package/dist/editor/EntryNavigator.d.ts +48 -0
  387. package/dist/editor/EntryNavigator.d.ts.map +1 -0
  388. package/dist/editor/EntryNavigator.js +331 -0
  389. package/dist/editor/EntryNavigator.js.map +1 -0
  390. package/dist/editor/EntryNavigator.stories.d.ts +8 -0
  391. package/dist/editor/EntryNavigator.stories.d.ts.map +1 -0
  392. package/dist/editor/EntryNavigator.stories.js +42 -0
  393. package/dist/editor/EntryNavigator.stories.js.map +1 -0
  394. package/dist/editor/FormRenderer.d.ts +34 -0
  395. package/dist/editor/FormRenderer.d.ts.map +1 -0
  396. package/dist/editor/FormRenderer.js +127 -0
  397. package/dist/editor/FormRenderer.js.map +1 -0
  398. package/dist/editor/FormRenderer.stories.d.ts +7 -0
  399. package/dist/editor/FormRenderer.stories.d.ts.map +1 -0
  400. package/dist/editor/FormRenderer.stories.js +115 -0
  401. package/dist/editor/FormRenderer.stories.js.map +1 -0
  402. package/dist/editor/GroupManager.d.ts +7 -0
  403. package/dist/editor/GroupManager.d.ts.map +1 -0
  404. package/dist/editor/GroupManager.js +7 -0
  405. package/dist/editor/GroupManager.js.map +1 -0
  406. package/dist/editor/GroupManager.stories.d.ts +19 -0
  407. package/dist/editor/GroupManager.stories.d.ts.map +1 -0
  408. package/dist/editor/GroupManager.stories.js +265 -0
  409. package/dist/editor/GroupManager.stories.js.map +1 -0
  410. package/dist/editor/PermissionManager.d.ts +9 -0
  411. package/dist/editor/PermissionManager.d.ts.map +1 -0
  412. package/dist/editor/PermissionManager.js +9 -0
  413. package/dist/editor/PermissionManager.js.map +1 -0
  414. package/dist/editor/PermissionManager.stories.d.ts +20 -0
  415. package/dist/editor/PermissionManager.stories.d.ts.map +1 -0
  416. package/dist/editor/PermissionManager.stories.js +506 -0
  417. package/dist/editor/PermissionManager.stories.js.map +1 -0
  418. package/dist/editor/canopy-path.d.ts +17 -0
  419. package/dist/editor/canopy-path.d.ts.map +1 -0
  420. package/dist/editor/canopy-path.js +49 -0
  421. package/dist/editor/canopy-path.js.map +1 -0
  422. package/dist/editor/client-reference-resolver.d.ts +18 -0
  423. package/dist/editor/client-reference-resolver.d.ts.map +1 -0
  424. package/dist/editor/client-reference-resolver.js +87 -0
  425. package/dist/editor/client-reference-resolver.js.map +1 -0
  426. package/dist/editor/comments/BranchComments.d.ts +27 -0
  427. package/dist/editor/comments/BranchComments.d.ts.map +1 -0
  428. package/dist/editor/comments/BranchComments.js +26 -0
  429. package/dist/editor/comments/BranchComments.js.map +1 -0
  430. package/dist/editor/comments/EntryComments.d.ts +26 -0
  431. package/dist/editor/comments/EntryComments.d.ts.map +1 -0
  432. package/dist/editor/comments/EntryComments.js +26 -0
  433. package/dist/editor/comments/EntryComments.js.map +1 -0
  434. package/dist/editor/comments/FieldWrapper.d.ts +32 -0
  435. package/dist/editor/comments/FieldWrapper.d.ts.map +1 -0
  436. package/dist/editor/comments/FieldWrapper.js +41 -0
  437. package/dist/editor/comments/FieldWrapper.js.map +1 -0
  438. package/dist/editor/comments/FieldWrapper.stories.d.ts +10 -0
  439. package/dist/editor/comments/FieldWrapper.stories.d.ts.map +1 -0
  440. package/dist/editor/comments/FieldWrapper.stories.js +173 -0
  441. package/dist/editor/comments/FieldWrapper.stories.js.map +1 -0
  442. package/dist/editor/comments/InlineCommentThread.d.ts +23 -0
  443. package/dist/editor/comments/InlineCommentThread.d.ts.map +1 -0
  444. package/dist/editor/comments/InlineCommentThread.js +82 -0
  445. package/dist/editor/comments/InlineCommentThread.js.map +1 -0
  446. package/dist/editor/comments/ThreadCarousel.d.ts +110 -0
  447. package/dist/editor/comments/ThreadCarousel.d.ts.map +1 -0
  448. package/dist/editor/comments/ThreadCarousel.js +223 -0
  449. package/dist/editor/comments/ThreadCarousel.js.map +1 -0
  450. package/dist/editor/components/ConfirmDeleteModal.d.ts +15 -0
  451. package/dist/editor/components/ConfirmDeleteModal.d.ts.map +1 -0
  452. package/dist/editor/components/ConfirmDeleteModal.js +11 -0
  453. package/dist/editor/components/ConfirmDeleteModal.js.map +1 -0
  454. package/dist/editor/components/EditorContext.d.ts +35 -0
  455. package/dist/editor/components/EditorContext.d.ts.map +1 -0
  456. package/dist/editor/components/EditorContext.js +19 -0
  457. package/dist/editor/components/EditorContext.js.map +1 -0
  458. package/dist/editor/components/EditorFooter.d.ts +20 -0
  459. package/dist/editor/components/EditorFooter.d.ts.map +1 -0
  460. package/dist/editor/components/EditorFooter.js +15 -0
  461. package/dist/editor/components/EditorFooter.js.map +1 -0
  462. package/dist/editor/components/EditorHeader.d.ts +166 -0
  463. package/dist/editor/components/EditorHeader.d.ts.map +1 -0
  464. package/dist/editor/components/EditorHeader.js +99 -0
  465. package/dist/editor/components/EditorHeader.js.map +1 -0
  466. package/dist/editor/components/EditorSidebar.d.ts +76 -0
  467. package/dist/editor/components/EditorSidebar.d.ts.map +1 -0
  468. package/dist/editor/components/EditorSidebar.js +40 -0
  469. package/dist/editor/components/EditorSidebar.js.map +1 -0
  470. package/dist/editor/components/EntryCreateModal.d.ts +27 -0
  471. package/dist/editor/components/EntryCreateModal.d.ts.map +1 -0
  472. package/dist/editor/components/EntryCreateModal.js +78 -0
  473. package/dist/editor/components/EntryCreateModal.js.map +1 -0
  474. package/dist/editor/components/RenameEntryModal.d.ts +18 -0
  475. package/dist/editor/components/RenameEntryModal.d.ts.map +1 -0
  476. package/dist/editor/components/RenameEntryModal.js +62 -0
  477. package/dist/editor/components/RenameEntryModal.js.map +1 -0
  478. package/dist/editor/components/UserBadge.d.ts +28 -0
  479. package/dist/editor/components/UserBadge.d.ts.map +1 -0
  480. package/dist/editor/components/UserBadge.js +93 -0
  481. package/dist/editor/components/UserBadge.js.map +1 -0
  482. package/dist/editor/components/index.d.ts +4 -0
  483. package/dist/editor/components/index.d.ts.map +1 -0
  484. package/dist/editor/components/index.js +6 -0
  485. package/dist/editor/components/index.js.map +1 -0
  486. package/dist/editor/context/ApiClientContext.d.ts +33 -0
  487. package/dist/editor/context/ApiClientContext.d.ts.map +1 -0
  488. package/dist/editor/context/ApiClientContext.js +43 -0
  489. package/dist/editor/context/ApiClientContext.js.map +1 -0
  490. package/dist/editor/context/EditorStateContext.d.ts +93 -0
  491. package/dist/editor/context/EditorStateContext.d.ts.map +1 -0
  492. package/dist/editor/context/EditorStateContext.js +143 -0
  493. package/dist/editor/context/EditorStateContext.js.map +1 -0
  494. package/dist/editor/context/index.d.ts +20 -0
  495. package/dist/editor/context/index.d.ts.map +1 -0
  496. package/dist/editor/context/index.js +20 -0
  497. package/dist/editor/context/index.js.map +1 -0
  498. package/dist/editor/editor-config.d.ts +14 -0
  499. package/dist/editor/editor-config.d.ts.map +1 -0
  500. package/dist/editor/editor-config.js +79 -0
  501. package/dist/editor/editor-config.js.map +1 -0
  502. package/dist/editor/editor-utils.d.ts +92 -0
  503. package/dist/editor/editor-utils.d.ts.map +1 -0
  504. package/dist/editor/editor-utils.js +238 -0
  505. package/dist/editor/editor-utils.js.map +1 -0
  506. package/dist/editor/env.d.ts +5 -0
  507. package/dist/editor/env.d.ts.map +1 -0
  508. package/dist/editor/env.js +5 -0
  509. package/dist/editor/env.js.map +1 -0
  510. package/dist/editor/fields/BlockField.d.ts +19 -0
  511. package/dist/editor/fields/BlockField.d.ts.map +1 -0
  512. package/dist/editor/fields/BlockField.js +86 -0
  513. package/dist/editor/fields/BlockField.js.map +1 -0
  514. package/dist/editor/fields/BlockField.stories.d.ts +7 -0
  515. package/dist/editor/fields/BlockField.stories.d.ts.map +1 -0
  516. package/dist/editor/fields/BlockField.stories.js +50 -0
  517. package/dist/editor/fields/BlockField.stories.js.map +1 -0
  518. package/dist/editor/fields/CodeField.d.ts +12 -0
  519. package/dist/editor/fields/CodeField.d.ts.map +1 -0
  520. package/dist/editor/fields/CodeField.js +11 -0
  521. package/dist/editor/fields/CodeField.js.map +1 -0
  522. package/dist/editor/fields/MarkdownField.d.ts +11 -0
  523. package/dist/editor/fields/MarkdownField.d.ts.map +1 -0
  524. package/dist/editor/fields/MarkdownField.js +94 -0
  525. package/dist/editor/fields/MarkdownField.js.map +1 -0
  526. package/dist/editor/fields/ObjectField.d.ts +15 -0
  527. package/dist/editor/fields/ObjectField.d.ts.map +1 -0
  528. package/dist/editor/fields/ObjectField.js +12 -0
  529. package/dist/editor/fields/ObjectField.js.map +1 -0
  530. package/dist/editor/fields/ReferenceField.d.ts +20 -0
  531. package/dist/editor/fields/ReferenceField.d.ts.map +1 -0
  532. package/dist/editor/fields/ReferenceField.js +66 -0
  533. package/dist/editor/fields/ReferenceField.js.map +1 -0
  534. package/dist/editor/fields/SelectField.d.ts +18 -0
  535. package/dist/editor/fields/SelectField.d.ts.map +1 -0
  536. package/dist/editor/fields/SelectField.js +17 -0
  537. package/dist/editor/fields/SelectField.js.map +1 -0
  538. package/dist/editor/fields/TextField.d.ts +11 -0
  539. package/dist/editor/fields/TextField.d.ts.map +1 -0
  540. package/dist/editor/fields/TextField.js +10 -0
  541. package/dist/editor/fields/TextField.js.map +1 -0
  542. package/dist/editor/fields/ToggleField.d.ts +12 -0
  543. package/dist/editor/fields/ToggleField.d.ts.map +1 -0
  544. package/dist/editor/fields/ToggleField.js +10 -0
  545. package/dist/editor/fields/ToggleField.js.map +1 -0
  546. package/dist/editor/fields/fields.stories.d.ts +8 -0
  547. package/dist/editor/fields/fields.stories.d.ts.map +1 -0
  548. package/dist/editor/fields/fields.stories.js +34 -0
  549. package/dist/editor/fields/fields.stories.js.map +1 -0
  550. package/dist/editor/group-manager/ExternalGroupsTab.d.ts +16 -0
  551. package/dist/editor/group-manager/ExternalGroupsTab.d.ts.map +1 -0
  552. package/dist/editor/group-manager/ExternalGroupsTab.js +16 -0
  553. package/dist/editor/group-manager/ExternalGroupsTab.js.map +1 -0
  554. package/dist/editor/group-manager/GroupCard.d.ts +24 -0
  555. package/dist/editor/group-manager/GroupCard.d.ts.map +1 -0
  556. package/dist/editor/group-manager/GroupCard.js +9 -0
  557. package/dist/editor/group-manager/GroupCard.js.map +1 -0
  558. package/dist/editor/group-manager/GroupForm.d.ts +15 -0
  559. package/dist/editor/group-manager/GroupForm.d.ts.map +1 -0
  560. package/dist/editor/group-manager/GroupForm.js +7 -0
  561. package/dist/editor/group-manager/GroupForm.js.map +1 -0
  562. package/dist/editor/group-manager/InternalGroupsTab.d.ts +26 -0
  563. package/dist/editor/group-manager/InternalGroupsTab.d.ts.map +1 -0
  564. package/dist/editor/group-manager/InternalGroupsTab.js +18 -0
  565. package/dist/editor/group-manager/InternalGroupsTab.js.map +1 -0
  566. package/dist/editor/group-manager/MemberList.d.ts +23 -0
  567. package/dist/editor/group-manager/MemberList.d.ts.map +1 -0
  568. package/dist/editor/group-manager/MemberList.js +13 -0
  569. package/dist/editor/group-manager/MemberList.js.map +1 -0
  570. package/dist/editor/group-manager/hooks/useExternalGroupSearch.d.ts +16 -0
  571. package/dist/editor/group-manager/hooks/useExternalGroupSearch.d.ts.map +1 -0
  572. package/dist/editor/group-manager/hooks/useExternalGroupSearch.js +43 -0
  573. package/dist/editor/group-manager/hooks/useExternalGroupSearch.js.map +1 -0
  574. package/dist/editor/group-manager/hooks/useGroupState.d.ts +24 -0
  575. package/dist/editor/group-manager/hooks/useGroupState.d.ts.map +1 -0
  576. package/dist/editor/group-manager/hooks/useGroupState.js +97 -0
  577. package/dist/editor/group-manager/hooks/useGroupState.js.map +1 -0
  578. package/dist/editor/group-manager/hooks/useUserSearch.d.ts +20 -0
  579. package/dist/editor/group-manager/hooks/useUserSearch.d.ts.map +1 -0
  580. package/dist/editor/group-manager/hooks/useUserSearch.js +60 -0
  581. package/dist/editor/group-manager/hooks/useUserSearch.js.map +1 -0
  582. package/dist/editor/group-manager/index.d.ts +14 -0
  583. package/dist/editor/group-manager/index.d.ts.map +1 -0
  584. package/dist/editor/group-manager/index.js +71 -0
  585. package/dist/editor/group-manager/index.js.map +1 -0
  586. package/dist/editor/group-manager/types.d.ts +24 -0
  587. package/dist/editor/group-manager/types.d.ts.map +1 -0
  588. package/dist/editor/group-manager/types.js +5 -0
  589. package/dist/editor/group-manager/types.js.map +1 -0
  590. package/dist/editor/hooks/__test__/test-utils.d.ts +129 -0
  591. package/dist/editor/hooks/__test__/test-utils.d.ts.map +1 -0
  592. package/dist/editor/hooks/__test__/test-utils.js +157 -0
  593. package/dist/editor/hooks/__test__/test-utils.js.map +1 -0
  594. package/dist/editor/hooks/index.d.ts +12 -0
  595. package/dist/editor/hooks/index.d.ts.map +1 -0
  596. package/dist/editor/hooks/index.js +15 -0
  597. package/dist/editor/hooks/index.js.map +1 -0
  598. package/dist/editor/hooks/useBranchActions.d.ts +34 -0
  599. package/dist/editor/hooks/useBranchActions.d.ts.map +1 -0
  600. package/dist/editor/hooks/useBranchActions.js +95 -0
  601. package/dist/editor/hooks/useBranchActions.js.map +1 -0
  602. package/dist/editor/hooks/useBranchManager.d.ts +88 -0
  603. package/dist/editor/hooks/useBranchManager.d.ts.map +1 -0
  604. package/dist/editor/hooks/useBranchManager.js +245 -0
  605. package/dist/editor/hooks/useBranchManager.js.map +1 -0
  606. package/dist/editor/hooks/useCommentSystem.d.ts +93 -0
  607. package/dist/editor/hooks/useCommentSystem.d.ts.map +1 -0
  608. package/dist/editor/hooks/useCommentSystem.js +240 -0
  609. package/dist/editor/hooks/useCommentSystem.js.map +1 -0
  610. package/dist/editor/hooks/useDraftManager.d.ts +64 -0
  611. package/dist/editor/hooks/useDraftManager.d.ts.map +1 -0
  612. package/dist/editor/hooks/useDraftManager.js +228 -0
  613. package/dist/editor/hooks/useDraftManager.js.map +1 -0
  614. package/dist/editor/hooks/useEditorLayout.d.ts +34 -0
  615. package/dist/editor/hooks/useEditorLayout.d.ts.map +1 -0
  616. package/dist/editor/hooks/useEditorLayout.js +52 -0
  617. package/dist/editor/hooks/useEditorLayout.js.map +1 -0
  618. package/dist/editor/hooks/useEntryManager.d.ts +66 -0
  619. package/dist/editor/hooks/useEntryManager.d.ts.map +1 -0
  620. package/dist/editor/hooks/useEntryManager.js +305 -0
  621. package/dist/editor/hooks/useEntryManager.js.map +1 -0
  622. package/dist/editor/hooks/useGroupManager.d.ts +43 -0
  623. package/dist/editor/hooks/useGroupManager.d.ts.map +1 -0
  624. package/dist/editor/hooks/useGroupManager.js +122 -0
  625. package/dist/editor/hooks/useGroupManager.js.map +1 -0
  626. package/dist/editor/hooks/usePermissionManager.d.ts +40 -0
  627. package/dist/editor/hooks/usePermissionManager.d.ts.map +1 -0
  628. package/dist/editor/hooks/usePermissionManager.js +94 -0
  629. package/dist/editor/hooks/usePermissionManager.js.map +1 -0
  630. package/dist/editor/hooks/useReferenceResolution.d.ts +47 -0
  631. package/dist/editor/hooks/useReferenceResolution.d.ts.map +1 -0
  632. package/dist/editor/hooks/useReferenceResolution.js +207 -0
  633. package/dist/editor/hooks/useReferenceResolution.js.map +1 -0
  634. package/dist/editor/hooks/useSchemaManager.d.ts +46 -0
  635. package/dist/editor/hooks/useSchemaManager.d.ts.map +1 -0
  636. package/dist/editor/hooks/useSchemaManager.js +235 -0
  637. package/dist/editor/hooks/useSchemaManager.js.map +1 -0
  638. package/dist/editor/hooks/useUserContext.d.ts +22 -0
  639. package/dist/editor/hooks/useUserContext.d.ts.map +1 -0
  640. package/dist/editor/hooks/useUserContext.js +49 -0
  641. package/dist/editor/hooks/useUserContext.js.map +1 -0
  642. package/dist/editor/hooks/useUserMetadata.d.ts +16 -0
  643. package/dist/editor/hooks/useUserMetadata.d.ts.map +1 -0
  644. package/dist/editor/hooks/useUserMetadata.js +55 -0
  645. package/dist/editor/hooks/useUserMetadata.js.map +1 -0
  646. package/dist/editor/permission-manager/GroupSelector.d.ts +13 -0
  647. package/dist/editor/permission-manager/GroupSelector.d.ts.map +1 -0
  648. package/dist/editor/permission-manager/GroupSelector.js +22 -0
  649. package/dist/editor/permission-manager/GroupSelector.js.map +1 -0
  650. package/dist/editor/permission-manager/PermissionEditor.d.ts +34 -0
  651. package/dist/editor/permission-manager/PermissionEditor.d.ts.map +1 -0
  652. package/dist/editor/permission-manager/PermissionEditor.js +30 -0
  653. package/dist/editor/permission-manager/PermissionEditor.js.map +1 -0
  654. package/dist/editor/permission-manager/PermissionLevelBadge.d.ts +16 -0
  655. package/dist/editor/permission-manager/PermissionLevelBadge.d.ts.map +1 -0
  656. package/dist/editor/permission-manager/PermissionLevelBadge.js +22 -0
  657. package/dist/editor/permission-manager/PermissionLevelBadge.js.map +1 -0
  658. package/dist/editor/permission-manager/PermissionTree.d.ts +37 -0
  659. package/dist/editor/permission-manager/PermissionTree.d.ts.map +1 -0
  660. package/dist/editor/permission-manager/PermissionTree.js +45 -0
  661. package/dist/editor/permission-manager/PermissionTree.js.map +1 -0
  662. package/dist/editor/permission-manager/UserSelector.d.ts +16 -0
  663. package/dist/editor/permission-manager/UserSelector.d.ts.map +1 -0
  664. package/dist/editor/permission-manager/UserSelector.js +14 -0
  665. package/dist/editor/permission-manager/UserSelector.js.map +1 -0
  666. package/dist/editor/permission-manager/constants.d.ts +12 -0
  667. package/dist/editor/permission-manager/constants.d.ts.map +1 -0
  668. package/dist/editor/permission-manager/constants.js +9 -0
  669. package/dist/editor/permission-manager/constants.js.map +1 -0
  670. package/dist/editor/permission-manager/hooks/useGroupsAndUsers.d.ts +30 -0
  671. package/dist/editor/permission-manager/hooks/useGroupsAndUsers.d.ts.map +1 -0
  672. package/dist/editor/permission-manager/hooks/useGroupsAndUsers.js +102 -0
  673. package/dist/editor/permission-manager/hooks/useGroupsAndUsers.js.map +1 -0
  674. package/dist/editor/permission-manager/hooks/usePermissionTree.d.ts +41 -0
  675. package/dist/editor/permission-manager/hooks/usePermissionTree.d.ts.map +1 -0
  676. package/dist/editor/permission-manager/hooks/usePermissionTree.js +125 -0
  677. package/dist/editor/permission-manager/hooks/usePermissionTree.js.map +1 -0
  678. package/dist/editor/permission-manager/index.d.ts +14 -0
  679. package/dist/editor/permission-manager/index.d.ts.map +1 -0
  680. package/dist/editor/permission-manager/index.js +125 -0
  681. package/dist/editor/permission-manager/index.js.map +1 -0
  682. package/dist/editor/permission-manager/types.d.ts +53 -0
  683. package/dist/editor/permission-manager/types.d.ts.map +1 -0
  684. package/dist/editor/permission-manager/types.js +5 -0
  685. package/dist/editor/permission-manager/types.js.map +1 -0
  686. package/dist/editor/permission-manager/utils.d.ts +60 -0
  687. package/dist/editor/permission-manager/utils.d.ts.map +1 -0
  688. package/dist/editor/permission-manager/utils.js +152 -0
  689. package/dist/editor/permission-manager/utils.js.map +1 -0
  690. package/dist/editor/preview-bridge.d.ts +64 -0
  691. package/dist/editor/preview-bridge.d.ts.map +1 -0
  692. package/dist/editor/preview-bridge.js +225 -0
  693. package/dist/editor/preview-bridge.js.map +1 -0
  694. package/dist/editor/schema-editor/CollectionEditor.d.ts +53 -0
  695. package/dist/editor/schema-editor/CollectionEditor.d.ts.map +1 -0
  696. package/dist/editor/schema-editor/CollectionEditor.js +220 -0
  697. package/dist/editor/schema-editor/CollectionEditor.js.map +1 -0
  698. package/dist/editor/schema-editor/EntryTypeEditor.d.ts +39 -0
  699. package/dist/editor/schema-editor/EntryTypeEditor.d.ts.map +1 -0
  700. package/dist/editor/schema-editor/EntryTypeEditor.js +161 -0
  701. package/dist/editor/schema-editor/EntryTypeEditor.js.map +1 -0
  702. package/dist/editor/schema-editor/index.d.ts +10 -0
  703. package/dist/editor/schema-editor/index.d.ts.map +1 -0
  704. package/dist/editor/schema-editor/index.js +10 -0
  705. package/dist/editor/schema-editor/index.js.map +1 -0
  706. package/dist/editor/setup-test-dom.d.ts +1 -0
  707. package/dist/editor/setup-test-dom.d.ts.map +1 -0
  708. package/dist/editor/setup-test-dom.js +12 -0
  709. package/dist/editor/setup-test-dom.js.map +1 -0
  710. package/dist/editor/test-setup.d.ts +6 -0
  711. package/dist/editor/test-setup.d.ts.map +1 -0
  712. package/dist/editor/test-setup.js +34 -0
  713. package/dist/editor/test-setup.js.map +1 -0
  714. package/dist/editor/theme.d.ts +165 -0
  715. package/dist/editor/theme.d.ts.map +1 -0
  716. package/dist/editor/theme.js +66 -0
  717. package/dist/editor/theme.js.map +1 -0
  718. package/dist/editor/utils/env.d.ts +23 -0
  719. package/dist/editor/utils/env.d.ts.map +1 -0
  720. package/dist/editor/utils/env.js +38 -0
  721. package/dist/editor/utils/env.js.map +1 -0
  722. package/dist/entry-schema-registry.d.ts +44 -0
  723. package/dist/entry-schema-registry.d.ts.map +1 -0
  724. package/dist/entry-schema-registry.js +101 -0
  725. package/dist/entry-schema-registry.js.map +1 -0
  726. package/dist/entry-schema.d.ts +74 -0
  727. package/dist/entry-schema.d.ts.map +1 -0
  728. package/dist/entry-schema.js +5 -0
  729. package/dist/entry-schema.js.map +1 -0
  730. package/dist/git-manager.d.ts +144 -0
  731. package/dist/git-manager.d.ts.map +1 -0
  732. package/dist/git-manager.js +563 -0
  733. package/dist/git-manager.js.map +1 -0
  734. package/dist/github-service.d.ts +93 -0
  735. package/dist/github-service.d.ts.map +1 -0
  736. package/dist/github-service.js +228 -0
  737. package/dist/github-service.js.map +1 -0
  738. package/dist/http/handler.d.ts +57 -0
  739. package/dist/http/handler.d.ts.map +1 -0
  740. package/dist/http/handler.js +197 -0
  741. package/dist/http/handler.js.map +1 -0
  742. package/dist/http/index.d.ts +7 -0
  743. package/dist/http/index.d.ts.map +1 -0
  744. package/dist/http/index.js +4 -0
  745. package/dist/http/index.js.map +1 -0
  746. package/dist/http/router.d.ts +49 -0
  747. package/dist/http/router.d.ts.map +1 -0
  748. package/dist/http/router.js +100 -0
  749. package/dist/http/router.js.map +1 -0
  750. package/dist/http/types.d.ts +34 -0
  751. package/dist/http/types.d.ts.map +1 -0
  752. package/dist/http/types.js +7 -0
  753. package/dist/http/types.js.map +1 -0
  754. package/dist/id.d.ts +17 -0
  755. package/dist/id.d.ts.map +1 -0
  756. package/dist/id.js +20 -0
  757. package/dist/id.js.map +1 -0
  758. package/dist/index.d.ts +11 -0
  759. package/dist/index.d.ts.map +1 -0
  760. package/dist/index.js +9 -0
  761. package/dist/index.js.map +1 -0
  762. package/dist/operating-mode/client-safe-strategy.d.ts +57 -0
  763. package/dist/operating-mode/client-safe-strategy.d.ts.map +1 -0
  764. package/dist/operating-mode/client-safe-strategy.js +154 -0
  765. package/dist/operating-mode/client-safe-strategy.js.map +1 -0
  766. package/dist/operating-mode/client-unsafe-strategy.d.ts +26 -0
  767. package/dist/operating-mode/client-unsafe-strategy.d.ts.map +1 -0
  768. package/dist/operating-mode/client-unsafe-strategy.js +234 -0
  769. package/dist/operating-mode/client-unsafe-strategy.js.map +1 -0
  770. package/dist/operating-mode/client.d.ts +13 -0
  771. package/dist/operating-mode/client.d.ts.map +1 -0
  772. package/dist/operating-mode/client.js +11 -0
  773. package/dist/operating-mode/client.js.map +1 -0
  774. package/dist/operating-mode/index.d.ts +28 -0
  775. package/dist/operating-mode/index.d.ts.map +1 -0
  776. package/dist/operating-mode/index.js +28 -0
  777. package/dist/operating-mode/index.js.map +1 -0
  778. package/dist/operating-mode/types.d.ts +130 -0
  779. package/dist/operating-mode/types.d.ts.map +1 -0
  780. package/dist/operating-mode/types.js +9 -0
  781. package/dist/operating-mode/types.js.map +1 -0
  782. package/dist/paths/branch.d.ts +45 -0
  783. package/dist/paths/branch.d.ts.map +1 -0
  784. package/dist/paths/branch.js +84 -0
  785. package/dist/paths/branch.js.map +1 -0
  786. package/dist/paths/index.d.ts +22 -0
  787. package/dist/paths/index.d.ts.map +1 -0
  788. package/dist/paths/index.js +30 -0
  789. package/dist/paths/index.js.map +1 -0
  790. package/dist/paths/normalize-server.d.ts +20 -0
  791. package/dist/paths/normalize-server.d.ts.map +1 -0
  792. package/dist/paths/normalize-server.js +35 -0
  793. package/dist/paths/normalize-server.js.map +1 -0
  794. package/dist/paths/normalize.d.ts +59 -0
  795. package/dist/paths/normalize.d.ts.map +1 -0
  796. package/dist/paths/normalize.js +96 -0
  797. package/dist/paths/normalize.js.map +1 -0
  798. package/dist/paths/resolve.d.ts +30 -0
  799. package/dist/paths/resolve.d.ts.map +1 -0
  800. package/dist/paths/resolve.js +51 -0
  801. package/dist/paths/resolve.js.map +1 -0
  802. package/dist/paths/test-utils.d.ts +24 -0
  803. package/dist/paths/test-utils.d.ts.map +1 -0
  804. package/dist/paths/test-utils.js +23 -0
  805. package/dist/paths/test-utils.js.map +1 -0
  806. package/dist/paths/types.d.ts +72 -0
  807. package/dist/paths/types.d.ts.map +1 -0
  808. package/dist/paths/types.js +13 -0
  809. package/dist/paths/types.js.map +1 -0
  810. package/dist/paths/validation.d.ts +181 -0
  811. package/dist/paths/validation.d.ts.map +1 -0
  812. package/dist/paths/validation.js +322 -0
  813. package/dist/paths/validation.js.map +1 -0
  814. package/dist/reference-resolver.d.ts +58 -0
  815. package/dist/reference-resolver.d.ts.map +1 -0
  816. package/dist/reference-resolver.js +118 -0
  817. package/dist/reference-resolver.js.map +1 -0
  818. package/dist/schema/index.d.ts +18 -0
  819. package/dist/schema/index.d.ts.map +1 -0
  820. package/dist/schema/index.js +19 -0
  821. package/dist/schema/index.js.map +1 -0
  822. package/dist/schema/meta-loader.d.ts +73 -0
  823. package/dist/schema/meta-loader.d.ts.map +1 -0
  824. package/dist/schema/meta-loader.js +273 -0
  825. package/dist/schema/meta-loader.js.map +1 -0
  826. package/dist/schema/resolver.d.ts +42 -0
  827. package/dist/schema/resolver.d.ts.map +1 -0
  828. package/dist/schema/resolver.js +70 -0
  829. package/dist/schema/resolver.js.map +1 -0
  830. package/dist/schema/schema-store-types.d.ts +48 -0
  831. package/dist/schema/schema-store-types.d.ts.map +1 -0
  832. package/dist/schema/schema-store-types.js +8 -0
  833. package/dist/schema/schema-store-types.js.map +1 -0
  834. package/dist/schema/schema-store.d.ts +249 -0
  835. package/dist/schema/schema-store.d.ts.map +1 -0
  836. package/dist/schema/schema-store.js +624 -0
  837. package/dist/schema/schema-store.js.map +1 -0
  838. package/dist/schema/types.d.ts +30 -0
  839. package/dist/schema/types.d.ts.map +1 -0
  840. package/dist/schema/types.js +5 -0
  841. package/dist/schema/types.js.map +1 -0
  842. package/dist/server.d.ts +11 -0
  843. package/dist/server.d.ts.map +1 -0
  844. package/dist/server.js +10 -0
  845. package/dist/server.js.map +1 -0
  846. package/dist/services.d.ts +94 -0
  847. package/dist/services.d.ts.map +1 -0
  848. package/dist/services.js +251 -0
  849. package/dist/services.js.map +1 -0
  850. package/dist/settings-branch-utils.d.ts +21 -0
  851. package/dist/settings-branch-utils.d.ts.map +1 -0
  852. package/dist/settings-branch-utils.js +33 -0
  853. package/dist/settings-branch-utils.js.map +1 -0
  854. package/dist/settings-workspace.d.ts +30 -0
  855. package/dist/settings-workspace.d.ts.map +1 -0
  856. package/dist/settings-workspace.js +137 -0
  857. package/dist/settings-workspace.js.map +1 -0
  858. package/dist/task-queue/index.d.ts +3 -0
  859. package/dist/task-queue/index.d.ts.map +1 -0
  860. package/dist/task-queue/index.js +2 -0
  861. package/dist/task-queue/index.js.map +1 -0
  862. package/dist/task-queue/task-queue.d.ts +81 -0
  863. package/dist/task-queue/task-queue.d.ts.map +1 -0
  864. package/dist/task-queue/task-queue.js +434 -0
  865. package/dist/task-queue/task-queue.js.map +1 -0
  866. package/dist/task-queue/types.d.ts +39 -0
  867. package/dist/task-queue/types.d.ts.map +1 -0
  868. package/dist/task-queue/types.js +2 -0
  869. package/dist/task-queue/types.js.map +1 -0
  870. package/dist/test-utils/api-test-helpers.d.ts +238 -0
  871. package/dist/test-utils/api-test-helpers.d.ts.map +1 -0
  872. package/dist/test-utils/api-test-helpers.js +347 -0
  873. package/dist/test-utils/api-test-helpers.js.map +1 -0
  874. package/dist/test-utils/console-spy.d.ts +56 -0
  875. package/dist/test-utils/console-spy.d.ts.map +1 -0
  876. package/dist/test-utils/console-spy.js +81 -0
  877. package/dist/test-utils/console-spy.js.map +1 -0
  878. package/dist/test-utils/git-helpers.d.ts +21 -0
  879. package/dist/test-utils/git-helpers.d.ts.map +1 -0
  880. package/dist/test-utils/git-helpers.js +23 -0
  881. package/dist/test-utils/git-helpers.js.map +1 -0
  882. package/dist/test-utils/index.d.ts +5 -0
  883. package/dist/test-utils/index.d.ts.map +1 -0
  884. package/dist/test-utils/index.js +4 -0
  885. package/dist/test-utils/index.js.map +1 -0
  886. package/dist/types.d.ts +46 -0
  887. package/dist/types.d.ts.map +1 -0
  888. package/dist/types.js +2 -0
  889. package/dist/types.js.map +1 -0
  890. package/dist/user.d.ts +62 -0
  891. package/dist/user.d.ts.map +1 -0
  892. package/dist/user.js +69 -0
  893. package/dist/user.js.map +1 -0
  894. package/dist/utils/debug.d.ts +44 -0
  895. package/dist/utils/debug.d.ts.map +1 -0
  896. package/dist/utils/debug.js +99 -0
  897. package/dist/utils/debug.js.map +1 -0
  898. package/dist/utils/error.d.ts +63 -0
  899. package/dist/utils/error.d.ts.map +1 -0
  900. package/dist/utils/error.js +79 -0
  901. package/dist/utils/error.js.map +1 -0
  902. package/dist/utils/format.d.ts +8 -0
  903. package/dist/utils/format.d.ts.map +1 -0
  904. package/dist/utils/format.js +13 -0
  905. package/dist/utils/format.js.map +1 -0
  906. package/dist/validation/deletion-checker.d.ts +66 -0
  907. package/dist/validation/deletion-checker.d.ts.map +1 -0
  908. package/dist/validation/deletion-checker.js +163 -0
  909. package/dist/validation/deletion-checker.js.map +1 -0
  910. package/dist/validation/field-traversal.d.ts +62 -0
  911. package/dist/validation/field-traversal.d.ts.map +1 -0
  912. package/dist/validation/field-traversal.js +95 -0
  913. package/dist/validation/field-traversal.js.map +1 -0
  914. package/dist/validation/reference-validator.d.ts +45 -0
  915. package/dist/validation/reference-validator.d.ts.map +1 -0
  916. package/dist/validation/reference-validator.js +140 -0
  917. package/dist/validation/reference-validator.js.map +1 -0
  918. package/dist/worker/cms-worker.d.ts +122 -0
  919. package/dist/worker/cms-worker.d.ts.map +1 -0
  920. package/dist/worker/cms-worker.js +660 -0
  921. package/dist/worker/cms-worker.js.map +1 -0
  922. package/dist/worker/task-queue-config.d.ts +11 -0
  923. package/dist/worker/task-queue-config.d.ts.map +1 -0
  924. package/dist/worker/task-queue-config.js +24 -0
  925. package/dist/worker/task-queue-config.js.map +1 -0
  926. package/dist/worker/task-queue.d.ts +18 -0
  927. package/dist/worker/task-queue.d.ts.map +1 -0
  928. package/dist/worker/task-queue.js +20 -0
  929. package/dist/worker/task-queue.js.map +1 -0
  930. package/package.json +144 -0
  931. package/src/__integration__/errors/invalid-content.test.ts +238 -0
  932. package/src/__integration__/errors/permission-denied.test.ts +220 -0
  933. package/src/__integration__/fixtures/content-seeds.ts +105 -0
  934. package/src/__integration__/fixtures/schemas.ts +67 -0
  935. package/src/__integration__/initialization/prod-sim-init.test.ts +139 -0
  936. package/src/__integration__/permissions/path-permissions.test.ts +314 -0
  937. package/src/__integration__/permissions/role-permissions.test.ts +354 -0
  938. package/src/__integration__/permissions/settings-branch-isolation.test.ts +317 -0
  939. package/src/__integration__/settings/groups-api.test.ts +403 -0
  940. package/src/__integration__/test-utils/api-client.ts +167 -0
  941. package/src/__integration__/test-utils/multi-user.ts +129 -0
  942. package/src/__integration__/test-utils/test-workspace.ts +130 -0
  943. package/src/__integration__/user/user-context.test.ts +174 -0
  944. package/src/__integration__/validation/input-validation.test.ts +166 -0
  945. package/src/__integration__/workflows/api-editing-workflow.test.ts +244 -0
  946. package/src/__integration__/workflows/conflict-resolution.test.ts +259 -0
  947. package/src/__integration__/workflows/editing-workflow.test.ts +205 -0
  948. package/src/__integration__/workflows/review-workflow.test.ts +260 -0
  949. package/src/ai/__tests__/build.integration.test.ts +224 -0
  950. package/src/ai/__tests__/generate.integration.test.ts +495 -0
  951. package/src/ai/__tests__/handler.integration.test.ts +212 -0
  952. package/src/ai/__tests__/json-to-markdown.test.ts +553 -0
  953. package/src/ai/generate.ts +410 -0
  954. package/src/ai/handler.ts +123 -0
  955. package/src/ai/index.ts +26 -0
  956. package/src/ai/json-to-markdown.ts +424 -0
  957. package/src/ai/resolve-branch.ts +34 -0
  958. package/src/ai/types.ts +160 -0
  959. package/src/api/AGENTS.md +81 -0
  960. package/src/api/__test__/mock-client.ts +404 -0
  961. package/src/api/assets.test.ts +140 -0
  962. package/src/api/assets.ts +154 -0
  963. package/src/api/branch-merge.test.ts +163 -0
  964. package/src/api/branch-merge.ts +113 -0
  965. package/src/api/branch-review.test.ts +297 -0
  966. package/src/api/branch-review.ts +136 -0
  967. package/src/api/branch-status.test.ts +85 -0
  968. package/src/api/branch-status.ts +153 -0
  969. package/src/api/branch-withdraw.test.ts +146 -0
  970. package/src/api/branch-withdraw.ts +81 -0
  971. package/src/api/branch-workflow.integration.test.ts +578 -0
  972. package/src/api/branch.test.ts +620 -0
  973. package/src/api/branch.ts +492 -0
  974. package/src/api/client.test.ts +349 -0
  975. package/src/api/client.ts +506 -0
  976. package/src/api/comments.test.ts +285 -0
  977. package/src/api/comments.ts +210 -0
  978. package/src/api/content.test.ts +345 -0
  979. package/src/api/content.ts +454 -0
  980. package/src/api/entries.test.ts +1339 -0
  981. package/src/api/entries.ts +650 -0
  982. package/src/api/github-sync.ts +144 -0
  983. package/src/api/groups.test.ts +1013 -0
  984. package/src/api/groups.ts +375 -0
  985. package/src/api/guards.test.ts +533 -0
  986. package/src/api/guards.ts +271 -0
  987. package/src/api/index.ts +87 -0
  988. package/src/api/permissions.test.ts +766 -0
  989. package/src/api/permissions.ts +334 -0
  990. package/src/api/reference-options.ts +118 -0
  991. package/src/api/resolve-references.ts +107 -0
  992. package/src/api/route-builder.ts +289 -0
  993. package/src/api/schema.test.ts +840 -0
  994. package/src/api/schema.ts +936 -0
  995. package/src/api/security.test.ts +233 -0
  996. package/src/api/settings-helpers.ts +84 -0
  997. package/src/api/types.ts +40 -0
  998. package/src/api/user.test.ts +127 -0
  999. package/src/api/user.ts +42 -0
  1000. package/src/api/validators.test.ts +275 -0
  1001. package/src/api/validators.ts +176 -0
  1002. package/src/asset-store.test.ts +37 -0
  1003. package/src/asset-store.ts +110 -0
  1004. package/src/auth/cache.ts +7 -0
  1005. package/src/auth/caching-auth-plugin.test.ts +154 -0
  1006. package/src/auth/caching-auth-plugin.ts +109 -0
  1007. package/src/auth/context-helpers.ts +75 -0
  1008. package/src/auth/file-based-auth-cache.test.ts +257 -0
  1009. package/src/auth/file-based-auth-cache.ts +279 -0
  1010. package/src/auth/index.ts +12 -0
  1011. package/src/auth/plugin.ts +51 -0
  1012. package/src/auth/types.ts +38 -0
  1013. package/src/authorization/__tests__/branch.test.ts +260 -0
  1014. package/src/authorization/__tests__/content.test.ts +142 -0
  1015. package/src/authorization/__tests__/path.test.ts +133 -0
  1016. package/src/authorization/__tests__/permissions-loader.test.ts +200 -0
  1017. package/src/authorization/branch.ts +94 -0
  1018. package/src/authorization/content.ts +93 -0
  1019. package/src/authorization/groups/index.ts +11 -0
  1020. package/src/authorization/groups/loader.ts +127 -0
  1021. package/src/authorization/groups/schema.ts +48 -0
  1022. package/src/authorization/helpers.ts +48 -0
  1023. package/src/authorization/index.ts +84 -0
  1024. package/src/authorization/path.ts +112 -0
  1025. package/src/authorization/permissions/index.ts +11 -0
  1026. package/src/authorization/permissions/loader.ts +116 -0
  1027. package/src/authorization/permissions/schema.ts +66 -0
  1028. package/src/authorization/test-utils.ts +15 -0
  1029. package/src/authorization/types.ts +66 -0
  1030. package/src/authorization/validation.test.ts +100 -0
  1031. package/src/authorization/validation.ts +62 -0
  1032. package/src/branch-metadata.test.ts +168 -0
  1033. package/src/branch-metadata.ts +166 -0
  1034. package/src/branch-registry.test.ts +248 -0
  1035. package/src/branch-registry.ts +152 -0
  1036. package/src/branch-schema-cache.test.ts +275 -0
  1037. package/src/branch-schema-cache.ts +189 -0
  1038. package/src/branch-workspace.test.ts +183 -0
  1039. package/src/branch-workspace.ts +124 -0
  1040. package/src/build/generate-ai-content.ts +78 -0
  1041. package/src/build/index.ts +8 -0
  1042. package/src/build-mode.ts +27 -0
  1043. package/src/cli/generate-ai-content.ts +100 -0
  1044. package/src/cli/init.test.ts +240 -0
  1045. package/src/cli/init.ts +457 -0
  1046. package/src/cli/templates/Dockerfile.cms.template +19 -0
  1047. package/src/cli/templates/canopy.ts.template +55 -0
  1048. package/src/cli/templates/canopycms.config.ts.template +11 -0
  1049. package/src/cli/templates/deploy-cms.yml.template +27 -0
  1050. package/src/cli/templates/edit-page.tsx.template +32 -0
  1051. package/src/cli/templates/route.ts.template +12 -0
  1052. package/src/cli/templates/schemas.ts.template +16 -0
  1053. package/src/cli/templates.ts +47 -0
  1054. package/src/client.ts +12 -0
  1055. package/src/comment-store.test.ts +442 -0
  1056. package/src/comment-store.ts +301 -0
  1057. package/src/config/__tests__/config.test.ts +513 -0
  1058. package/src/config/flatten.ts +174 -0
  1059. package/src/config/helpers.ts +167 -0
  1060. package/src/config/index.ts +86 -0
  1061. package/src/config/schemas/collection.ts +67 -0
  1062. package/src/config/schemas/config.ts +77 -0
  1063. package/src/config/schemas/field.ts +108 -0
  1064. package/src/config/schemas/media.ts +27 -0
  1065. package/src/config/schemas/permissions.ts +21 -0
  1066. package/src/config/types.ts +321 -0
  1067. package/src/config/validation.ts +70 -0
  1068. package/src/config-test.ts +65 -0
  1069. package/src/config.ts +11 -0
  1070. package/src/content-id-index.test.ts +512 -0
  1071. package/src/content-id-index.ts +479 -0
  1072. package/src/content-reader.test.ts +478 -0
  1073. package/src/content-reader.ts +214 -0
  1074. package/src/content-store.test.ts +1126 -0
  1075. package/src/content-store.ts +793 -0
  1076. package/src/context.ts +111 -0
  1077. package/src/editor/BranchManager.stories.tsx +80 -0
  1078. package/src/editor/BranchManager.test.tsx +324 -0
  1079. package/src/editor/BranchManager.tsx +461 -0
  1080. package/src/editor/CanopyEditor.stories.tsx +128 -0
  1081. package/src/editor/CanopyEditor.test.tsx +81 -0
  1082. package/src/editor/CanopyEditor.tsx +73 -0
  1083. package/src/editor/CanopyEditorPage.test.tsx +59 -0
  1084. package/src/editor/CanopyEditorPage.tsx +25 -0
  1085. package/src/editor/CommentsPanel.stories.tsx +184 -0
  1086. package/src/editor/CommentsPanel.tsx +338 -0
  1087. package/src/editor/Editor.integration.test.tsx +227 -0
  1088. package/src/editor/Editor.stories.tsx +119 -0
  1089. package/src/editor/Editor.tsx +1221 -0
  1090. package/src/editor/EditorPanes.stories.tsx +256 -0
  1091. package/src/editor/EditorPanes.test.tsx +77 -0
  1092. package/src/editor/EditorPanes.tsx +180 -0
  1093. package/src/editor/EntryNavigator.stories.tsx +65 -0
  1094. package/src/editor/EntryNavigator.test.tsx +598 -0
  1095. package/src/editor/EntryNavigator.tsx +665 -0
  1096. package/src/editor/FormRenderer.stories.tsx +212 -0
  1097. package/src/editor/FormRenderer.test.tsx +194 -0
  1098. package/src/editor/FormRenderer.tsx +432 -0
  1099. package/src/editor/GroupManager.stories.tsx +301 -0
  1100. package/src/editor/GroupManager.test.tsx +682 -0
  1101. package/src/editor/GroupManager.tsx +9 -0
  1102. package/src/editor/PermissionManager.stories.tsx +539 -0
  1103. package/src/editor/PermissionManager.test.tsx +864 -0
  1104. package/src/editor/PermissionManager.tsx +12 -0
  1105. package/src/editor/canopy-path.test.ts +23 -0
  1106. package/src/editor/canopy-path.ts +52 -0
  1107. package/src/editor/client-reference-resolver.ts +118 -0
  1108. package/src/editor/comments/BranchComments.tsx +93 -0
  1109. package/src/editor/comments/EntryComments.tsx +94 -0
  1110. package/src/editor/comments/FieldWrapper.stories.tsx +210 -0
  1111. package/src/editor/comments/FieldWrapper.tsx +129 -0
  1112. package/src/editor/comments/InlineCommentThread.test.tsx +384 -0
  1113. package/src/editor/comments/InlineCommentThread.tsx +246 -0
  1114. package/src/editor/comments/ThreadCarousel.test.tsx +393 -0
  1115. package/src/editor/comments/ThreadCarousel.tsx +525 -0
  1116. package/src/editor/components/ConfirmDeleteModal.tsx +49 -0
  1117. package/src/editor/components/EditorContext.tsx +49 -0
  1118. package/src/editor/components/EditorFooter.tsx +47 -0
  1119. package/src/editor/components/EditorHeader.tsx +492 -0
  1120. package/src/editor/components/EditorSidebar.tsx +193 -0
  1121. package/src/editor/components/EntryCreateModal.tsx +193 -0
  1122. package/src/editor/components/RenameEntryModal.tsx +152 -0
  1123. package/src/editor/components/UserBadge.test.tsx +274 -0
  1124. package/src/editor/components/UserBadge.tsx +240 -0
  1125. package/src/editor/components/index.ts +6 -0
  1126. package/src/editor/context/ApiClientContext.tsx +56 -0
  1127. package/src/editor/context/EditorStateContext.tsx +221 -0
  1128. package/src/editor/context/index.ts +40 -0
  1129. package/src/editor/editor-config.test.ts +385 -0
  1130. package/src/editor/editor-config.ts +94 -0
  1131. package/src/editor/editor-utils.test.ts +772 -0
  1132. package/src/editor/editor-utils.ts +303 -0
  1133. package/src/editor/env.ts +4 -0
  1134. package/src/editor/fields/BlockField.stories.tsx +79 -0
  1135. package/src/editor/fields/BlockField.tsx +267 -0
  1136. package/src/editor/fields/CodeField.tsx +41 -0
  1137. package/src/editor/fields/MarkdownField.tsx +205 -0
  1138. package/src/editor/fields/ObjectField.tsx +71 -0
  1139. package/src/editor/fields/ReferenceField.tsx +138 -0
  1140. package/src/editor/fields/SelectField.tsx +76 -0
  1141. package/src/editor/fields/TextField.tsx +35 -0
  1142. package/src/editor/fields/ToggleField.tsx +37 -0
  1143. package/src/editor/fields/fields.stories.tsx +40 -0
  1144. package/src/editor/group-manager/ExternalGroupsTab.tsx +114 -0
  1145. package/src/editor/group-manager/GroupCard.tsx +102 -0
  1146. package/src/editor/group-manager/GroupForm.tsx +66 -0
  1147. package/src/editor/group-manager/InternalGroupsTab.tsx +147 -0
  1148. package/src/editor/group-manager/MemberList.tsx +184 -0
  1149. package/src/editor/group-manager/hooks/useExternalGroupSearch.ts +63 -0
  1150. package/src/editor/group-manager/hooks/useGroupState.ts +134 -0
  1151. package/src/editor/group-manager/hooks/useUserSearch.ts +84 -0
  1152. package/src/editor/group-manager/index.tsx +210 -0
  1153. package/src/editor/group-manager/types.ts +28 -0
  1154. package/src/editor/hooks/README.md +26 -0
  1155. package/src/editor/hooks/__test__/test-utils.tsx +183 -0
  1156. package/src/editor/hooks/index.ts +23 -0
  1157. package/src/editor/hooks/useBranchActions.test.tsx +267 -0
  1158. package/src/editor/hooks/useBranchActions.tsx +121 -0
  1159. package/src/editor/hooks/useBranchManager.test.tsx +391 -0
  1160. package/src/editor/hooks/useBranchManager.tsx +326 -0
  1161. package/src/editor/hooks/useCommentSystem.test.ts +615 -0
  1162. package/src/editor/hooks/useCommentSystem.ts +347 -0
  1163. package/src/editor/hooks/useDraftManager.test.ts +375 -0
  1164. package/src/editor/hooks/useDraftManager.ts +259 -0
  1165. package/src/editor/hooks/useEditorLayout.test.ts +147 -0
  1166. package/src/editor/hooks/useEditorLayout.ts +67 -0
  1167. package/src/editor/hooks/useEntryManager.test.ts +588 -0
  1168. package/src/editor/hooks/useEntryManager.ts +387 -0
  1169. package/src/editor/hooks/useGroupManager.test.ts +277 -0
  1170. package/src/editor/hooks/useGroupManager.ts +139 -0
  1171. package/src/editor/hooks/usePermissionManager.test.ts +211 -0
  1172. package/src/editor/hooks/usePermissionManager.ts +113 -0
  1173. package/src/editor/hooks/useReferenceResolution.ts +248 -0
  1174. package/src/editor/hooks/useSchemaManager.test.ts +370 -0
  1175. package/src/editor/hooks/useSchemaManager.ts +310 -0
  1176. package/src/editor/hooks/useUserContext.tsx +57 -0
  1177. package/src/editor/hooks/useUserMetadata.test.ts +191 -0
  1178. package/src/editor/hooks/useUserMetadata.ts +71 -0
  1179. package/src/editor/permission-manager/GroupSelector.tsx +73 -0
  1180. package/src/editor/permission-manager/PermissionEditor.tsx +321 -0
  1181. package/src/editor/permission-manager/PermissionLevelBadge.tsx +53 -0
  1182. package/src/editor/permission-manager/PermissionTree.tsx +237 -0
  1183. package/src/editor/permission-manager/UserSelector.tsx +95 -0
  1184. package/src/editor/permission-manager/constants.tsx +18 -0
  1185. package/src/editor/permission-manager/hooks/useGroupsAndUsers.ts +153 -0
  1186. package/src/editor/permission-manager/hooks/usePermissionTree.ts +200 -0
  1187. package/src/editor/permission-manager/index.tsx +294 -0
  1188. package/src/editor/permission-manager/types.ts +58 -0
  1189. package/src/editor/permission-manager/utils.ts +179 -0
  1190. package/src/editor/preview-bridge.test.tsx +50 -0
  1191. package/src/editor/preview-bridge.tsx +294 -0
  1192. package/src/editor/schema-editor/CollectionEditor.test.tsx +238 -0
  1193. package/src/editor/schema-editor/CollectionEditor.tsx +520 -0
  1194. package/src/editor/schema-editor/EntryTypeEditor.test.tsx +215 -0
  1195. package/src/editor/schema-editor/EntryTypeEditor.tsx +367 -0
  1196. package/src/editor/schema-editor/index.ts +19 -0
  1197. package/src/editor/setup-test-dom.ts +10 -0
  1198. package/src/editor/test-setup.ts +33 -0
  1199. package/src/editor/theme.tsx +119 -0
  1200. package/src/editor/utils/env.ts +39 -0
  1201. package/src/entry-schema-registry.test.ts +281 -0
  1202. package/src/entry-schema-registry.ts +121 -0
  1203. package/src/entry-schema.ts +84 -0
  1204. package/src/git-manager.test.ts +552 -0
  1205. package/src/git-manager.ts +667 -0
  1206. package/src/github-service.test.ts +312 -0
  1207. package/src/github-service.ts +295 -0
  1208. package/src/http/handler.test.ts +275 -0
  1209. package/src/http/handler.ts +280 -0
  1210. package/src/http/index.ts +11 -0
  1211. package/src/http/router.ts +164 -0
  1212. package/src/http/types.ts +44 -0
  1213. package/src/id.test.ts +48 -0
  1214. package/src/id.ts +22 -0
  1215. package/src/index.ts +26 -0
  1216. package/src/operating-mode/__tests__/strategies.test.ts +511 -0
  1217. package/src/operating-mode/client-safe-strategy.ts +184 -0
  1218. package/src/operating-mode/client-unsafe-strategy.ts +303 -0
  1219. package/src/operating-mode/client.ts +13 -0
  1220. package/src/operating-mode/index.ts +34 -0
  1221. package/src/operating-mode/types.ts +186 -0
  1222. package/src/paths/__tests__/branch.test.ts +53 -0
  1223. package/src/paths/__tests__/normalize.test.ts +141 -0
  1224. package/src/paths/__tests__/resolve.test.ts +207 -0
  1225. package/src/paths/__tests__/validation.test.ts +61 -0
  1226. package/src/paths/branch.ts +115 -0
  1227. package/src/paths/index.ts +73 -0
  1228. package/src/paths/normalize-server.ts +40 -0
  1229. package/src/paths/normalize.ts +107 -0
  1230. package/src/paths/resolve.ts +61 -0
  1231. package/src/paths/test-utils.ts +37 -0
  1232. package/src/paths/types.ts +68 -0
  1233. package/src/paths/validation.test.ts +480 -0
  1234. package/src/paths/validation.ts +391 -0
  1235. package/src/reference-resolver.test.ts +107 -0
  1236. package/src/reference-resolver.ts +157 -0
  1237. package/src/schema/index.ts +29 -0
  1238. package/src/schema/meta-loader.ts +366 -0
  1239. package/src/schema/resolver.ts +83 -0
  1240. package/src/schema/schema-store-types.ts +56 -0
  1241. package/src/schema/schema-store.test.ts +816 -0
  1242. package/src/schema/schema-store.ts +795 -0
  1243. package/src/schema/types.ts +33 -0
  1244. package/src/schema-meta-loader.test.ts +447 -0
  1245. package/src/server.ts +15 -0
  1246. package/src/services.test.ts +559 -0
  1247. package/src/services.ts +373 -0
  1248. package/src/settings-branch-utils.ts +53 -0
  1249. package/src/settings-workspace.ts +156 -0
  1250. package/src/task-queue/README.md +144 -0
  1251. package/src/task-queue/index.ts +14 -0
  1252. package/src/task-queue/task-queue.test.ts +524 -0
  1253. package/src/task-queue/task-queue.ts +514 -0
  1254. package/src/task-queue/types.ts +41 -0
  1255. package/src/test-utils/api-test-helpers.ts +445 -0
  1256. package/src/test-utils/console-spy.test.ts +14 -0
  1257. package/src/test-utils/console-spy.ts +125 -0
  1258. package/src/test-utils/git-helpers.ts +31 -0
  1259. package/src/test-utils/index.ts +4 -0
  1260. package/src/types.ts +54 -0
  1261. package/src/user.ts +118 -0
  1262. package/src/utils/debug.test.ts +114 -0
  1263. package/src/utils/debug.ts +127 -0
  1264. package/src/utils/error.test.ts +92 -0
  1265. package/src/utils/error.ts +83 -0
  1266. package/src/utils/format.ts +12 -0
  1267. package/src/validation/__tests__/field-traversal.test.ts +263 -0
  1268. package/src/validation/deletion-checker.ts +234 -0
  1269. package/src/validation/field-traversal.ts +146 -0
  1270. package/src/validation/reference-validator.ts +168 -0
  1271. package/src/worker/cms-worker-rebase.test.ts +473 -0
  1272. package/src/worker/cms-worker.ts +777 -0
  1273. package/src/worker/integration.test.ts +289 -0
  1274. package/src/worker/task-queue-config.ts +25 -0
  1275. package/src/worker/task-queue.test.ts +452 -0
  1276. package/src/worker/task-queue.ts +58 -0
@@ -0,0 +1,1339 @@
1
+ import fs from 'node:fs/promises'
2
+ import os from 'node:os'
3
+ import path from 'node:path'
4
+
5
+ import { describe, expect, it, vi } from 'vitest'
6
+
7
+ import { defineCanopyTestConfig } from '../config-test'
8
+ import { flattenSchema } from '../config'
9
+ import { createCheckBranchAccess } from '../authorization'
10
+ import { createCheckContentAccess } from '../authorization'
11
+ import { unsafeAsPermissionPath } from '../authorization/test-utils'
12
+ import type { PathPermission } from '../config'
13
+ import { listEntries } from './entries'
14
+ import { createMockApiContext, createMockBranchContext } from '../test-utils'
15
+ import { loadCollectionMetaFiles, resolveCollectionReferences } from '../schema'
16
+ import { unsafeAsBranchName, unsafeAsLogicalPath } from '../paths/test-utils'
17
+
18
+ const tmpDir = async () => fs.mkdtemp(path.join(os.tmpdir(), 'canopycms-entries-'))
19
+
20
+ describe('listEntries', () => {
21
+ it('lists entries with access filtering and pagination', async () => {
22
+ const root = await tmpDir()
23
+ await fs.mkdir(path.join(root, 'content/posts'), { recursive: true })
24
+ await fs.writeFile(
25
+ path.join(root, 'content/posts/entry.first.abc123def456.json'),
26
+ JSON.stringify({ title: 'First Post' }),
27
+ 'utf8',
28
+ )
29
+ await fs.writeFile(
30
+ path.join(root, 'content/posts/entry.hidden.xyz789abcDEF.json'),
31
+ JSON.stringify({ title: 'Hidden Post' }),
32
+ 'utf8',
33
+ )
34
+ await fs.writeFile(
35
+ path.join(root, 'content/settings.abc123XYZ789.json'),
36
+ JSON.stringify({ siteName: 'CanopyCMS' }),
37
+ 'utf8',
38
+ )
39
+
40
+ const schema = {
41
+ collections: [
42
+ {
43
+ name: 'posts',
44
+ path: 'posts',
45
+ entries: [
46
+ {
47
+ name: 'entry',
48
+ format: 'json' as const,
49
+ schema: [{ name: 'title', type: 'string' as const }],
50
+ },
51
+ ],
52
+ },
53
+ ],
54
+ } as const
55
+
56
+ const config = defineCanopyTestConfig({
57
+ defaultBranchAccess: 'allow',
58
+ schema,
59
+ })
60
+
61
+ // Mock loadPathPermissions to return rules that hide 'entry.hidden.xyz789abcDEF.json' from user 'u1'
62
+ // Use 'read' access restriction to actually hide the file from listing
63
+ const pathRules: PathPermission[] = [
64
+ {
65
+ path: unsafeAsPermissionPath('content/posts/entry.hidden.xyz789abcDEF.json'),
66
+ read: { allowedUsers: ['other'] },
67
+ },
68
+ ]
69
+ const mockLoadPermissions = vi.fn().mockResolvedValue(pathRules)
70
+
71
+ const checkBranchAccess = createCheckBranchAccess('allow')
72
+ const checkContentAccess = createCheckContentAccess({
73
+ checkBranchAccess,
74
+ loadPathPermissions: mockLoadPermissions,
75
+ defaultPathAccess: 'allow',
76
+ mode: 'dev',
77
+ })
78
+
79
+ const ctx = createMockApiContext({
80
+ services: {
81
+ config,
82
+ checkBranchAccess,
83
+ checkContentAccess,
84
+ },
85
+ branchContext: {
86
+ ...createMockBranchContext({
87
+ branchName: 'main',
88
+ baseRoot: root,
89
+ branchRoot: root,
90
+ createdBy: 'u1',
91
+ }),
92
+ flatSchema: flattenSchema(schema, config.contentRoot),
93
+ },
94
+ })
95
+
96
+ // Request limit=2 to get entries
97
+ const res = await listEntries.handler(
98
+ ctx,
99
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
100
+ { branch: unsafeAsBranchName('main'), limit: 2 },
101
+ )
102
+
103
+ expect(res.ok).toBe(true)
104
+ // Should include posts but not hidden.json (restricted by permission)
105
+ expect(res.data?.entries.some((e) => e.slug === 'first')).toBe(true)
106
+ expect(res.data?.entries.some((e) => e.slug === 'hidden')).toBe(false)
107
+ })
108
+
109
+ it('returns 404 when branch is missing', async () => {
110
+ const ctx = createMockApiContext({ branchContext: null })
111
+ const res = await listEntries.handler(
112
+ ctx,
113
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
114
+ { branch: unsafeAsBranchName('missing') },
115
+ )
116
+ expect(res.status).toBe(404)
117
+ expect(res.ok).toBe(false)
118
+ })
119
+
120
+ it('lists entries recursively from deeply nested collections', async () => {
121
+ const root = await tmpDir()
122
+
123
+ // Create 3-level nested structure with embedded IDs: docs/api/v2
124
+ // This mirrors the real example1 structure
125
+ const docsId = 'bChqT78gcaLd'
126
+ const apiId = 'meiuwxTSo7UN'
127
+ const v2Id = 'muwmyafM6mEJ'
128
+
129
+ await fs.mkdir(path.join(root, `content/docs.${docsId}`), {
130
+ recursive: true,
131
+ })
132
+ await fs.mkdir(path.join(root, `content/docs.${docsId}/api.${apiId}`), {
133
+ recursive: true,
134
+ })
135
+ await fs.mkdir(path.join(root, `content/docs.${docsId}/api.${apiId}/v2.${v2Id}`), {
136
+ recursive: true,
137
+ })
138
+
139
+ // Create entries at each level with embedded IDs in filenames
140
+ const overviewId = 'gnVmHnnMjWrD'
141
+ const introId = 'k396pBDVP8tC'
142
+ const authId = 'kmtzTh2k9Axq'
143
+ const usersId = 'ppqJw61uKkV5'
144
+
145
+ await fs.writeFile(
146
+ path.join(root, `content/docs.${docsId}/entry.overview.${overviewId}.json`),
147
+ JSON.stringify({ title: 'Overview' }),
148
+ 'utf8',
149
+ )
150
+ await fs.writeFile(
151
+ path.join(root, `content/docs.${docsId}/api.${apiId}/entry.intro.${introId}.json`),
152
+ JSON.stringify({ title: 'API Introduction' }),
153
+ 'utf8',
154
+ )
155
+ await fs.writeFile(
156
+ path.join(root, `content/docs.${docsId}/api.${apiId}/v2.${v2Id}/entry.auth.${authId}.json`),
157
+ JSON.stringify({ title: 'Authentication' }),
158
+ 'utf8',
159
+ )
160
+ await fs.writeFile(
161
+ path.join(root, `content/docs.${docsId}/api.${apiId}/v2.${v2Id}/entry.users.${usersId}.json`),
162
+ JSON.stringify({ title: 'Users API' }),
163
+ 'utf8',
164
+ )
165
+
166
+ const schema = {
167
+ collections: [
168
+ {
169
+ name: 'docs',
170
+ path: 'docs',
171
+ entries: [
172
+ {
173
+ name: 'entry',
174
+ format: 'json' as const,
175
+ schema: [{ name: 'title', type: 'string' as const }],
176
+ },
177
+ ],
178
+ collections: [
179
+ {
180
+ name: 'api',
181
+ path: 'api',
182
+ entries: [
183
+ {
184
+ name: 'entry',
185
+ format: 'json' as const,
186
+ schema: [{ name: 'title', type: 'string' as const }],
187
+ },
188
+ ],
189
+ collections: [
190
+ {
191
+ name: 'v2',
192
+ path: 'v2',
193
+ entries: [
194
+ {
195
+ name: 'entry',
196
+ format: 'json' as const,
197
+ schema: [{ name: 'title', type: 'string' as const }],
198
+ },
199
+ ],
200
+ },
201
+ ],
202
+ },
203
+ ],
204
+ },
205
+ ],
206
+ } as const
207
+
208
+ const config = defineCanopyTestConfig({
209
+ defaultBranchAccess: 'allow',
210
+ schema,
211
+ })
212
+
213
+ const checkBranchAccess = createCheckBranchAccess('allow')
214
+ const checkContentAccess = createCheckContentAccess({
215
+ checkBranchAccess,
216
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
217
+ defaultPathAccess: 'allow',
218
+ mode: 'dev',
219
+ })
220
+
221
+ const ctx = createMockApiContext({
222
+ services: {
223
+ config,
224
+ checkBranchAccess,
225
+ checkContentAccess,
226
+ },
227
+ branchContext: {
228
+ ...createMockBranchContext({
229
+ branchName: 'main',
230
+ baseRoot: root,
231
+ branchRoot: root,
232
+ createdBy: 'u1',
233
+ }),
234
+ flatSchema: flattenSchema(schema, config.contentRoot),
235
+ },
236
+ })
237
+
238
+ // Test 1: Without collection filter - lists entries from all collections (flat list)
239
+ const allEntriesRes = await listEntries.handler(
240
+ ctx,
241
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
242
+ { branch: unsafeAsBranchName('main') },
243
+ )
244
+
245
+ expect(allEntriesRes.ok).toBe(true)
246
+ expect(allEntriesRes.data?.entries.length).toBe(4) // All entries from all collections
247
+ expect(allEntriesRes.data?.entries.some((e) => e.slug === 'overview')).toBe(true)
248
+ expect(allEntriesRes.data?.entries.some((e) => e.slug === 'intro')).toBe(true)
249
+ expect(allEntriesRes.data?.entries.some((e) => e.slug === 'auth')).toBe(true)
250
+ expect(allEntriesRes.data?.entries.some((e) => e.slug === 'users')).toBe(true)
251
+
252
+ // Test 2: With collection filter, non-recursive - only gets entries from 'content/docs' collection (no children)
253
+ const docsNonRecursiveRes = await listEntries.handler(
254
+ ctx,
255
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
256
+ {
257
+ branch: unsafeAsBranchName('main'),
258
+ collection: unsafeAsLogicalPath('content/docs'),
259
+ },
260
+ )
261
+
262
+ expect(docsNonRecursiveRes.ok).toBe(true)
263
+ expect(docsNonRecursiveRes.data?.entries.length).toBe(1) // Only 'overview' from docs
264
+ expect(docsNonRecursiveRes.data?.entries.some((e) => e.slug === 'overview')).toBe(true)
265
+ expect(docsNonRecursiveRes.data?.entries.some((e) => e.slug === 'intro')).toBe(false) // From child collection
266
+ expect(docsNonRecursiveRes.data?.entries.some((e) => e.slug === 'auth')).toBe(false) // From grandchild collection
267
+
268
+ // Test 3: With collection filter and recursive flag - gets entries from 'content/docs' and all children
269
+ const docsRecursiveRes = await listEntries.handler(
270
+ ctx,
271
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
272
+ {
273
+ branch: unsafeAsBranchName('main'),
274
+ collection: unsafeAsLogicalPath('content/docs'),
275
+ recursive: true,
276
+ },
277
+ )
278
+
279
+ expect(docsRecursiveRes.ok).toBe(true)
280
+ expect(docsRecursiveRes.data?.entries.length).toBe(4) // All entries from docs tree
281
+ expect(docsRecursiveRes.data?.entries.some((e) => e.slug === 'overview')).toBe(true)
282
+ expect(docsRecursiveRes.data?.entries.some((e) => e.slug === 'intro')).toBe(true)
283
+ expect(docsRecursiveRes.data?.entries.some((e) => e.slug === 'auth')).toBe(true)
284
+ expect(docsRecursiveRes.data?.entries.some((e) => e.slug === 'users')).toBe(true)
285
+
286
+ // Test 4: Nested collection with recursive - gets entries from 'content/docs/api' and its children
287
+ const apiRecursiveRes = await listEntries.handler(
288
+ ctx,
289
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
290
+ {
291
+ branch: unsafeAsBranchName('main'),
292
+ collection: unsafeAsLogicalPath('content/docs/api'),
293
+ recursive: true,
294
+ },
295
+ )
296
+
297
+ expect(apiRecursiveRes.ok).toBe(true)
298
+ expect(apiRecursiveRes.data?.entries.length).toBe(3) // intro, auth, users (not overview)
299
+ expect(apiRecursiveRes.data?.entries.some((e) => e.slug === 'overview')).toBe(false) // From parent
300
+ expect(apiRecursiveRes.data?.entries.some((e) => e.slug === 'intro')).toBe(true)
301
+ expect(apiRecursiveRes.data?.entries.some((e) => e.slug === 'auth')).toBe(true)
302
+ expect(apiRecursiveRes.data?.entries.some((e) => e.slug === 'users')).toBe(true)
303
+
304
+ // Verify collectionPath values match the nested structure
305
+ const authEntry = docsRecursiveRes.data?.entries.find((e) => e.slug === 'auth')
306
+ expect(authEntry?.collectionPath).toBe('content/docs/api/v2')
307
+ })
308
+
309
+ it('returns entries with schemas using new schema format', async () => {
310
+ const root = await tmpDir()
311
+ await fs.mkdir(path.join(root, 'content/pages'), { recursive: true })
312
+ await fs.writeFile(
313
+ path.join(root, 'content/pages/page.home.abc123XYZ789.json'),
314
+ JSON.stringify({ title: 'Home Page', tagline: 'Welcome' }),
315
+ 'utf8',
316
+ )
317
+
318
+ const schema = {
319
+ collections: [
320
+ {
321
+ name: 'pages',
322
+ label: 'Pages',
323
+ path: 'pages',
324
+ entries: [
325
+ {
326
+ name: 'page',
327
+ format: 'json' as const,
328
+ schema: [
329
+ { name: 'title', type: 'string' as const },
330
+ { name: 'tagline', type: 'string' as const },
331
+ ],
332
+ },
333
+ ],
334
+ },
335
+ ],
336
+ } as const
337
+
338
+ const config = defineCanopyTestConfig({
339
+ defaultBranchAccess: 'allow',
340
+ contentRoot: 'content',
341
+ schema,
342
+ })
343
+
344
+ const checkBranchAccess = createCheckBranchAccess('allow')
345
+ const checkContentAccess = createCheckContentAccess({
346
+ checkBranchAccess,
347
+ loadPathPermissions: async () => [],
348
+ defaultPathAccess: 'allow',
349
+ mode: 'dev',
350
+ })
351
+
352
+ const ctx = createMockApiContext({
353
+ services: {
354
+ config,
355
+ checkBranchAccess,
356
+ checkContentAccess,
357
+ },
358
+ branchContext: {
359
+ ...createMockBranchContext({
360
+ branchName: 'main',
361
+ baseRoot: root,
362
+ branchRoot: root,
363
+ createdBy: 'u1',
364
+ }),
365
+ flatSchema: flattenSchema(schema, config.contentRoot),
366
+ },
367
+ })
368
+
369
+ const res = await listEntries.handler(
370
+ ctx,
371
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
372
+ { branch: unsafeAsBranchName('main') },
373
+ )
374
+
375
+ expect(res.ok).toBe(true)
376
+
377
+ // Verify entry is returned
378
+ const homeEntry = res.data?.entries.find((e) => e.slug === 'home')
379
+ expect(homeEntry).toBeDefined()
380
+ expect(homeEntry?.collectionPath).toBe('content/pages')
381
+
382
+ // Collections are now fetched from schema API, not entries API
383
+ })
384
+
385
+ it('includes canEdit flag based on edit permissions', async () => {
386
+ const root = await tmpDir()
387
+ await fs.mkdir(path.join(root, 'content/posts'), { recursive: true })
388
+ await fs.writeFile(
389
+ path.join(root, 'content/posts/entry.public.abcDEFghj123.json'),
390
+ JSON.stringify({ title: 'Public Post' }),
391
+ 'utf8',
392
+ )
393
+ await fs.writeFile(
394
+ path.join(root, 'content/posts/entry.readonly.defGHJkmn456.json'),
395
+ JSON.stringify({ title: 'Read-Only Post' }),
396
+ 'utf8',
397
+ )
398
+
399
+ const schema = {
400
+ collections: [
401
+ {
402
+ name: 'posts',
403
+ path: 'posts',
404
+ entries: [
405
+ {
406
+ name: 'entry',
407
+ format: 'json' as const,
408
+ schema: [{ name: 'title', type: 'string' as const }],
409
+ },
410
+ ],
411
+ },
412
+ ],
413
+ } as const
414
+
415
+ const config = defineCanopyTestConfig({
416
+ defaultBranchAccess: 'allow',
417
+ schema,
418
+ })
419
+
420
+ // Mock loadPathPermissions: 'entry.readonly.defGHJkmn456.json' is read-only for user 'u1'
421
+ const pathRules: PathPermission[] = [
422
+ {
423
+ path: unsafeAsPermissionPath('content/posts/entry.readonly.defGHJkmn456.json'),
424
+ read: { allowedUsers: ['u1'] },
425
+ edit: { allowedUsers: ['admin'] }, // u1 cannot edit
426
+ },
427
+ ]
428
+ const mockLoadPermissions = vi.fn().mockResolvedValue(pathRules)
429
+
430
+ const checkBranchAccess = createCheckBranchAccess('allow')
431
+ const checkContentAccess = createCheckContentAccess({
432
+ checkBranchAccess,
433
+ loadPathPermissions: mockLoadPermissions,
434
+ defaultPathAccess: 'allow',
435
+ mode: 'dev',
436
+ })
437
+
438
+ const ctx = createMockApiContext({
439
+ services: {
440
+ config,
441
+ checkBranchAccess,
442
+ checkContentAccess,
443
+ },
444
+ branchContext: {
445
+ ...createMockBranchContext({
446
+ branchName: 'main',
447
+ baseRoot: root,
448
+ branchRoot: root,
449
+ createdBy: 'u1',
450
+ }),
451
+ flatSchema: flattenSchema(schema, config.contentRoot),
452
+ },
453
+ })
454
+
455
+ const res = await listEntries.handler(
456
+ ctx,
457
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
458
+ { branch: unsafeAsBranchName('main') },
459
+ )
460
+
461
+ expect(res.ok).toBe(true)
462
+ expect(res.data?.entries).toHaveLength(2)
463
+
464
+ // Check canEdit flag on entries
465
+ const publicEntry = res.data?.entries.find((e) => e.slug === 'public')
466
+ const readonlyEntry = res.data?.entries.find((e) => e.slug === 'readonly')
467
+
468
+ expect(publicEntry?.canEdit).toBe(true) // u1 can edit public post (default allow)
469
+ expect(readonlyEntry?.canEdit).toBe(false) // u1 cannot edit readonly post (restricted to admin)
470
+ })
471
+
472
+ it('lists entries with embedded IDs in filenames', async () => {
473
+ const root = await tmpDir()
474
+
475
+ // Create content directory
476
+ await fs.mkdir(path.join(root, 'content'), { recursive: true })
477
+
478
+ // Create collection folder with ID (like authors.q52DCVPuH4ga)
479
+ const authorsId = 'q52DCVPuH4ga'
480
+ await fs.mkdir(path.join(root, `content/authors.${authorsId}`), {
481
+ recursive: true,
482
+ })
483
+
484
+ // Create .collection.json file (matching example1's approach)
485
+ await fs.writeFile(
486
+ path.join(root, `content/authors.${authorsId}/.collection.json`),
487
+ JSON.stringify({
488
+ name: 'authors',
489
+ label: 'Authors',
490
+ entries: [
491
+ {
492
+ name: 'author',
493
+ format: 'json',
494
+ schema: 'authorSchema',
495
+ },
496
+ ],
497
+ order: [],
498
+ }),
499
+ 'utf8',
500
+ )
501
+
502
+ // Create entry files with embedded IDs: {type}.{slug}.{id}.{ext}
503
+ const aliceId = '5NVkkrB1MJUv'
504
+ const bobId = 'jm6FYVAtJie8'
505
+ await fs.writeFile(
506
+ path.join(root, `content/authors.${authorsId}/author.alice.${aliceId}.json`),
507
+ JSON.stringify({ name: 'Alice', bio: 'Developer' }),
508
+ 'utf8',
509
+ )
510
+ await fs.writeFile(
511
+ path.join(root, `content/authors.${authorsId}/author.bob.${bobId}.json`),
512
+ JSON.stringify({ name: 'Bob', bio: 'Designer' }),
513
+ 'utf8',
514
+ )
515
+
516
+ // Load schema from .collection.json files (like services do)
517
+ const entrySchemaRegistry = {
518
+ authorSchema: [
519
+ { name: 'name', type: 'string' },
520
+ { name: 'bio', type: 'string' },
521
+ ],
522
+ }
523
+
524
+ const metaFiles = await loadCollectionMetaFiles(path.join(root, 'content'))
525
+ const schema = resolveCollectionReferences(metaFiles, entrySchemaRegistry)
526
+
527
+ // Create config with the loaded schema
528
+ const config = defineCanopyTestConfig({
529
+ defaultBranchAccess: 'allow',
530
+ contentRoot: 'content',
531
+ schema,
532
+ })
533
+
534
+ const checkBranchAccess = createCheckBranchAccess('allow')
535
+ const checkContentAccess = createCheckContentAccess({
536
+ checkBranchAccess,
537
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
538
+ defaultPathAccess: 'allow',
539
+ mode: 'dev',
540
+ })
541
+
542
+ const ctx = createMockApiContext({
543
+ services: {
544
+ config,
545
+ entrySchemaRegistry: entrySchemaRegistry,
546
+ checkBranchAccess,
547
+ checkContentAccess,
548
+ },
549
+ branchContext: {
550
+ ...createMockBranchContext({
551
+ branchName: 'main',
552
+ baseRoot: root,
553
+ branchRoot: root,
554
+ createdBy: 'u1',
555
+ }),
556
+ flatSchema: flattenSchema(schema, config.contentRoot),
557
+ },
558
+ })
559
+
560
+ const res = await listEntries.handler(
561
+ ctx,
562
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
563
+ { branch: unsafeAsBranchName('main') },
564
+ )
565
+
566
+ expect(res.ok).toBe(true)
567
+ expect(res.data?.entries).toHaveLength(2) // alice + bob
568
+
569
+ // Verify slugs are extracted correctly (without IDs) for collection entries
570
+ const aliceEntry = res.data?.entries.find((e) => e.slug === 'alice')
571
+ const bobEntry = res.data?.entries.find((e) => e.slug === 'bob')
572
+
573
+ expect(aliceEntry).toBeDefined()
574
+ expect(aliceEntry?.slug).toBe('alice')
575
+ expect(aliceEntry?.title).toBe('Alice')
576
+ expect(aliceEntry?.collectionPath).toBe('content/authors') // Logical path, no ID
577
+
578
+ expect(bobEntry).toBeDefined()
579
+ expect(bobEntry?.slug).toBe('bob')
580
+ expect(bobEntry?.title).toBe('Bob')
581
+ expect(bobEntry?.collectionPath).toBe('content/authors') // Logical path, no ID
582
+ })
583
+
584
+ it('lists root-level entry types with maxItems: 1', async () => {
585
+ const root = await tmpDir()
586
+
587
+ // Create content directory
588
+ await fs.mkdir(path.join(root, 'content'), { recursive: true })
589
+
590
+ // Create root .collection.json with entries (not singletons)
591
+ await fs.writeFile(
592
+ path.join(root, 'content/.collection.json'),
593
+ JSON.stringify({
594
+ entries: [
595
+ {
596
+ name: 'home',
597
+ label: 'Home',
598
+ format: 'json',
599
+ schema: 'homeSchema',
600
+ maxItems: 1,
601
+ },
602
+ {
603
+ name: 'settings',
604
+ label: 'Settings',
605
+ format: 'json',
606
+ schema: 'settingsSchema',
607
+ maxItems: 1,
608
+ },
609
+ ],
610
+ order: [],
611
+ }),
612
+ 'utf8',
613
+ )
614
+
615
+ // Create root-level entry files (pattern: {type}.{slug}.{id}.{ext})
616
+ const homeId = 'agfzDt2RLpSn'
617
+ const settingsId = 'Xp7qR2sL9mKn'
618
+ await fs.writeFile(
619
+ path.join(root, `content/home.home.${homeId}.json`),
620
+ JSON.stringify({ title: 'Welcome Home', hero: 'Hello World' }),
621
+ 'utf8',
622
+ )
623
+ await fs.writeFile(
624
+ path.join(root, `content/settings.settings.${settingsId}.json`),
625
+ JSON.stringify({ siteName: 'My Site', theme: 'dark' }),
626
+ 'utf8',
627
+ )
628
+
629
+ // Also create a collection to verify both work together
630
+ const postsId = '916jXZabYCxu'
631
+ await fs.mkdir(path.join(root, `content/posts.${postsId}`), {
632
+ recursive: true,
633
+ })
634
+ await fs.writeFile(
635
+ path.join(root, `content/posts.${postsId}/.collection.json`),
636
+ JSON.stringify({
637
+ name: 'posts',
638
+ label: 'Posts',
639
+ entries: [{ name: 'post', format: 'json', schema: 'postSchema' }],
640
+ order: [],
641
+ }),
642
+ 'utf8',
643
+ )
644
+ await fs.writeFile(
645
+ path.join(root, `content/posts.${postsId}/post.first.abc123def456.json`),
646
+ JSON.stringify({ title: 'First Post' }),
647
+ 'utf8',
648
+ )
649
+
650
+ // Load schema from .collection.json files
651
+ const entrySchemaRegistry = {
652
+ homeSchema: [
653
+ { name: 'title', type: 'string' },
654
+ { name: 'hero', type: 'string' },
655
+ ],
656
+ settingsSchema: [
657
+ { name: 'siteName', type: 'string' },
658
+ { name: 'theme', type: 'string' },
659
+ ],
660
+ postSchema: [{ name: 'title', type: 'string' }],
661
+ }
662
+
663
+ const metaFiles = await loadCollectionMetaFiles(path.join(root, 'content'))
664
+ const schema = resolveCollectionReferences(metaFiles, entrySchemaRegistry)
665
+
666
+ const config = defineCanopyTestConfig({
667
+ defaultBranchAccess: 'allow',
668
+ contentRoot: 'content',
669
+ schema,
670
+ })
671
+
672
+ const checkBranchAccess = createCheckBranchAccess('allow')
673
+ const checkContentAccess = createCheckContentAccess({
674
+ checkBranchAccess,
675
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
676
+ defaultPathAccess: 'allow',
677
+ mode: 'dev',
678
+ })
679
+
680
+ const ctx = createMockApiContext({
681
+ services: {
682
+ config,
683
+ entrySchemaRegistry: entrySchemaRegistry,
684
+ checkBranchAccess,
685
+ checkContentAccess,
686
+ },
687
+ branchContext: {
688
+ ...createMockBranchContext({
689
+ branchName: 'main',
690
+ baseRoot: root,
691
+ branchRoot: root,
692
+ createdBy: 'u1',
693
+ }),
694
+ flatSchema: flattenSchema(schema, config.contentRoot),
695
+ },
696
+ })
697
+
698
+ const res = await listEntries.handler(
699
+ ctx,
700
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
701
+ { branch: unsafeAsBranchName('main') },
702
+ )
703
+
704
+ expect(res.ok).toBe(true)
705
+ // Should have 3 entries: home, settings (root-level) + first post (collection)
706
+ expect(res.data?.entries).toHaveLength(3)
707
+
708
+ // Check root-level entry types
709
+ const homeEntry = res.data?.entries.find(
710
+ (e) => e.slug === 'home' && e.collectionPath === 'content',
711
+ )
712
+ expect(homeEntry).toBeDefined()
713
+ expect(homeEntry?.slug).toBe('home') // Name acts as slug
714
+ expect(homeEntry?.title).toBe('Welcome Home')
715
+ expect(homeEntry?.collectionPath).toBe('content') // Parent path, not full path
716
+ expect(homeEntry?.entryType).toBe('home')
717
+
718
+ const settingsEntry = res.data?.entries.find(
719
+ (e) => e.slug === 'settings' && e.collectionPath === 'content',
720
+ )
721
+ expect(settingsEntry).toBeDefined()
722
+ expect(settingsEntry?.slug).toBe('settings') // Name acts as slug
723
+ expect(settingsEntry?.title).toBe('Settings') // Falls back to label since siteName isn't title
724
+ expect(settingsEntry?.collectionPath).toBe('content') // Parent path, not full path
725
+ expect(settingsEntry?.entryType).toBe('settings')
726
+
727
+ // Check collection entry still works
728
+ const postEntry = res.data?.entries.find((e) => e.slug === 'first')
729
+ expect(postEntry).toBeDefined()
730
+ expect(postEntry?.collectionPath).toBe('content/posts')
731
+ })
732
+ })
733
+
734
+ describe('sortEntriesByOrder', () => {
735
+ // Import the function for testing
736
+ // Since it's not exported, we'll test it indirectly through the list handler
737
+ // or we can add a describe block that tests ordering behavior
738
+
739
+ it('returns entries sorted by order array when order is provided', async () => {
740
+ const root = await tmpDir()
741
+
742
+ // Create collection folder with embedded ID
743
+ const postsId = 'q52DCVPuH4ga'
744
+ await fs.mkdir(path.join(root, `content/posts.${postsId}`), {
745
+ recursive: true,
746
+ })
747
+
748
+ // Create .collection.json with an order array
749
+ await fs.writeFile(
750
+ path.join(root, `content/posts.${postsId}/.collection.json`),
751
+ JSON.stringify({
752
+ name: 'posts',
753
+ label: 'Posts',
754
+ entries: [{ name: 'post', format: 'json', schema: 'postSchema' }],
755
+ order: ['ccc333def456', 'aaa111abc123', 'bbb222xyz789'], // Custom order
756
+ }),
757
+ 'utf8',
758
+ )
759
+
760
+ // Create entries (alphabetically: aaa < bbb < ccc, but order says ccc, aaa, bbb)
761
+ await fs.writeFile(
762
+ path.join(root, `content/posts.${postsId}/post.alpha.aaa111abc123.json`),
763
+ JSON.stringify({ title: 'Alpha Post' }),
764
+ 'utf8',
765
+ )
766
+ await fs.writeFile(
767
+ path.join(root, `content/posts.${postsId}/post.beta.bbb222xyz789.json`),
768
+ JSON.stringify({ title: 'Beta Post' }),
769
+ 'utf8',
770
+ )
771
+ await fs.writeFile(
772
+ path.join(root, `content/posts.${postsId}/post.gamma.ccc333def456.json`),
773
+ JSON.stringify({ title: 'Gamma Post' }),
774
+ 'utf8',
775
+ )
776
+
777
+ const entrySchemaRegistry = {
778
+ postSchema: [{ name: 'title', type: 'string' }],
779
+ }
780
+
781
+ const metaFiles = await loadCollectionMetaFiles(path.join(root, 'content'))
782
+ const schema = resolveCollectionReferences(metaFiles, entrySchemaRegistry)
783
+
784
+ const config = defineCanopyTestConfig({
785
+ defaultBranchAccess: 'allow',
786
+ contentRoot: 'content',
787
+ schema,
788
+ })
789
+
790
+ const checkBranchAccess = createCheckBranchAccess('allow')
791
+ const checkContentAccess = createCheckContentAccess({
792
+ checkBranchAccess,
793
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
794
+ defaultPathAccess: 'allow',
795
+ mode: 'dev',
796
+ })
797
+
798
+ const ctx = createMockApiContext({
799
+ services: {
800
+ config,
801
+ entrySchemaRegistry: entrySchemaRegistry,
802
+ checkBranchAccess,
803
+ checkContentAccess,
804
+ },
805
+ branchContext: {
806
+ ...createMockBranchContext({
807
+ branchName: 'main',
808
+ baseRoot: root,
809
+ branchRoot: root,
810
+ createdBy: 'u1',
811
+ }),
812
+ flatSchema: flattenSchema(schema, config.contentRoot),
813
+ },
814
+ })
815
+
816
+ const res = await listEntries.handler(
817
+ ctx,
818
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
819
+ {
820
+ branch: unsafeAsBranchName('main'),
821
+ collection: unsafeAsLogicalPath('content/posts'),
822
+ },
823
+ )
824
+
825
+ expect(res.ok).toBe(true)
826
+ expect(res.data?.entries).toHaveLength(3)
827
+
828
+ // Verify entries are sorted according to order array: gamma, alpha, beta
829
+ const slugs = res.data?.entries.map((e) => e.slug)
830
+ expect(slugs).toEqual(['gamma', 'alpha', 'beta'])
831
+
832
+ // Verify entries have contentId
833
+ const contentIds = res.data?.entries.map((e) => e.contentId)
834
+ expect(contentIds).toEqual(['ccc333def456', 'aaa111abc123', 'bbb222xyz789'])
835
+ })
836
+
837
+ it('puts unordered entries at the end alphabetically', async () => {
838
+ const root = await tmpDir()
839
+
840
+ const postsId = 'q52DCVPuH4ga'
841
+ await fs.mkdir(path.join(root, `content/posts.${postsId}`), {
842
+ recursive: true,
843
+ })
844
+
845
+ // Order only has one entry
846
+ await fs.writeFile(
847
+ path.join(root, `content/posts.${postsId}/.collection.json`),
848
+ JSON.stringify({
849
+ name: 'posts',
850
+ entries: [{ name: 'post', format: 'json', schema: 'postSchema' }],
851
+ order: ['bbb222xyz789'], // Only beta is in the order
852
+ }),
853
+ 'utf8',
854
+ )
855
+
856
+ // Create entries
857
+ await fs.writeFile(
858
+ path.join(root, `content/posts.${postsId}/post.alpha.aaa111abc123.json`),
859
+ JSON.stringify({ title: 'Alpha' }),
860
+ 'utf8',
861
+ )
862
+ await fs.writeFile(
863
+ path.join(root, `content/posts.${postsId}/post.beta.bbb222xyz789.json`),
864
+ JSON.stringify({ title: 'Beta' }),
865
+ 'utf8',
866
+ )
867
+ await fs.writeFile(
868
+ path.join(root, `content/posts.${postsId}/post.gamma.ccc333def456.json`),
869
+ JSON.stringify({ title: 'Gamma' }),
870
+ 'utf8',
871
+ )
872
+
873
+ const entrySchemaRegistry = {
874
+ postSchema: [{ name: 'title', type: 'string' }],
875
+ }
876
+ const metaFiles = await loadCollectionMetaFiles(path.join(root, 'content'))
877
+ const schema = resolveCollectionReferences(metaFiles, entrySchemaRegistry)
878
+
879
+ const config = defineCanopyTestConfig({
880
+ defaultBranchAccess: 'allow',
881
+ contentRoot: 'content',
882
+ schema,
883
+ })
884
+
885
+ const checkBranchAccess = createCheckBranchAccess('allow')
886
+ const checkContentAccess = createCheckContentAccess({
887
+ checkBranchAccess,
888
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
889
+ defaultPathAccess: 'allow',
890
+ mode: 'dev',
891
+ })
892
+
893
+ const ctx = createMockApiContext({
894
+ services: {
895
+ config,
896
+ entrySchemaRegistry: entrySchemaRegistry,
897
+ checkBranchAccess,
898
+ checkContentAccess,
899
+ },
900
+ branchContext: {
901
+ ...createMockBranchContext({
902
+ branchName: 'main',
903
+ baseRoot: root,
904
+ branchRoot: root,
905
+ createdBy: 'u1',
906
+ }),
907
+ flatSchema: flattenSchema(schema, config.contentRoot),
908
+ },
909
+ })
910
+
911
+ const res = await listEntries.handler(
912
+ ctx,
913
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
914
+ {
915
+ branch: unsafeAsBranchName('main'),
916
+ collection: unsafeAsLogicalPath('content/posts'),
917
+ },
918
+ )
919
+
920
+ expect(res.ok).toBe(true)
921
+ // Beta first (in order), then alpha and gamma alphabetically
922
+ const slugs = res.data?.entries.map((e) => e.slug)
923
+ expect(slugs).toEqual(['beta', 'alpha', 'gamma'])
924
+ })
925
+ })
926
+
927
+ describe('dynamic collection discovery', () => {
928
+ it('discovers collections from .collection.json files not in flatSchema', async () => {
929
+ const root = await tmpDir()
930
+
931
+ // Create initial collection folder
932
+ const docsId = 'bChqT78gcaLd'
933
+ await fs.mkdir(path.join(root, `content/docs.${docsId}`), {
934
+ recursive: true,
935
+ })
936
+
937
+ await fs.writeFile(
938
+ path.join(root, `content/docs.${docsId}/.collection.json`),
939
+ JSON.stringify({
940
+ name: 'docs',
941
+ label: 'Documentation',
942
+ entries: [{ name: 'doc', format: 'json', schema: 'docSchema' }],
943
+ order: [],
944
+ }),
945
+ 'utf8',
946
+ )
947
+
948
+ // Create an entry in docs
949
+ await fs.writeFile(
950
+ path.join(root, `content/docs.${docsId}/doc.overview.gnVmHnnMjWrD.json`),
951
+ JSON.stringify({ title: 'Overview' }),
952
+ 'utf8',
953
+ )
954
+
955
+ // Now simulate a dynamically created subcollection that is NOT in the flatSchema
956
+ // This is the scenario where a user creates a collection via the schema API
957
+ const innerId = '2XWmsdeEU2Li'
958
+ await fs.mkdir(path.join(root, `content/docs.${docsId}/inner.${innerId}`), {
959
+ recursive: true,
960
+ })
961
+
962
+ await fs.writeFile(
963
+ path.join(root, `content/docs.${docsId}/inner.${innerId}/.collection.json`),
964
+ JSON.stringify({
965
+ name: 'inner',
966
+ label: 'Inner Docs',
967
+ entries: [{ name: 'doc', format: 'json', schema: 'docSchema' }],
968
+ order: [],
969
+ }),
970
+ 'utf8',
971
+ )
972
+
973
+ const entrySchemaRegistry = {
974
+ docSchema: [
975
+ { name: 'title', type: 'string' },
976
+ { name: 'body', type: 'markdown' },
977
+ ],
978
+ }
979
+
980
+ // Only load the ORIGINAL schema (docs only, not inner)
981
+ // This simulates the flatSchema being cached at startup before "inner" was created
982
+ const originalSchema = {
983
+ collections: [
984
+ {
985
+ name: 'docs',
986
+ label: 'Documentation',
987
+ path: 'docs',
988
+ entries: [
989
+ {
990
+ name: 'doc',
991
+ format: 'json' as const,
992
+ schema: entrySchemaRegistry.docSchema,
993
+ },
994
+ ],
995
+ },
996
+ ],
997
+ }
998
+
999
+ const config = defineCanopyTestConfig({
1000
+ defaultBranchAccess: 'allow',
1001
+ contentRoot: 'content',
1002
+ schema: originalSchema,
1003
+ })
1004
+
1005
+ const checkBranchAccess = createCheckBranchAccess('allow')
1006
+ const checkContentAccess = createCheckContentAccess({
1007
+ checkBranchAccess,
1008
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
1009
+ defaultPathAccess: 'allow',
1010
+ mode: 'dev',
1011
+ })
1012
+
1013
+ const ctx = createMockApiContext({
1014
+ services: {
1015
+ config,
1016
+ entrySchemaRegistry: entrySchemaRegistry,
1017
+ checkBranchAccess,
1018
+ checkContentAccess,
1019
+ },
1020
+ branchContext: {
1021
+ ...createMockBranchContext({
1022
+ branchName: 'main',
1023
+ baseRoot: root,
1024
+ branchRoot: root,
1025
+ createdBy: 'u1',
1026
+ }),
1027
+ // flatSchema only knows about 'docs', NOT 'docs/inner'
1028
+ flatSchema: flattenSchema(originalSchema, config.contentRoot),
1029
+ },
1030
+ })
1031
+
1032
+ const res = await listEntries.handler(
1033
+ ctx,
1034
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
1035
+ { branch: unsafeAsBranchName('main') },
1036
+ )
1037
+
1038
+ expect(res.ok).toBe(true)
1039
+
1040
+ // Dynamic collection discovery has moved to schema API
1041
+ // This test now only verifies entries API returns entries, not collections
1042
+ expect(res.data?.entries).toBeDefined()
1043
+ })
1044
+ })
1045
+
1046
+ describe('deleteEntry', () => {
1047
+ it('deletes an entry and returns success', async () => {
1048
+ const root = await tmpDir()
1049
+
1050
+ const postsId = 'q52DCVPuH4ga'
1051
+ await fs.mkdir(path.join(root, `content/posts.${postsId}`), {
1052
+ recursive: true,
1053
+ })
1054
+
1055
+ await fs.writeFile(
1056
+ path.join(root, `content/posts.${postsId}/.collection.json`),
1057
+ JSON.stringify({
1058
+ name: 'posts',
1059
+ entries: [{ name: 'post', format: 'json', schema: 'postSchema' }],
1060
+ order: [],
1061
+ }),
1062
+ 'utf8',
1063
+ )
1064
+
1065
+ const entryId = 'abc123def456'
1066
+ await fs.writeFile(
1067
+ path.join(root, `content/posts.${postsId}/post.to-delete.${entryId}.json`),
1068
+ JSON.stringify({ title: 'Delete Me' }),
1069
+ 'utf8',
1070
+ )
1071
+
1072
+ const entrySchemaRegistry = {
1073
+ postSchema: [{ name: 'title', type: 'string' }],
1074
+ }
1075
+ const metaFiles = await loadCollectionMetaFiles(path.join(root, 'content'))
1076
+ const schema = resolveCollectionReferences(metaFiles, entrySchemaRegistry)
1077
+
1078
+ const config = defineCanopyTestConfig({
1079
+ defaultBranchAccess: 'allow',
1080
+ contentRoot: 'content',
1081
+ schema,
1082
+ })
1083
+
1084
+ const checkBranchAccess = createCheckBranchAccess('allow')
1085
+ const checkContentAccess = createCheckContentAccess({
1086
+ checkBranchAccess,
1087
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
1088
+ defaultPathAccess: 'allow',
1089
+ mode: 'dev',
1090
+ })
1091
+
1092
+ const ctx = createMockApiContext({
1093
+ services: {
1094
+ config,
1095
+ entrySchemaRegistry: entrySchemaRegistry,
1096
+ checkBranchAccess,
1097
+ checkContentAccess,
1098
+ },
1099
+ branchContext: {
1100
+ ...createMockBranchContext({
1101
+ branchName: 'main',
1102
+ baseRoot: root,
1103
+ branchRoot: root,
1104
+ createdBy: 'u1',
1105
+ }),
1106
+ flatSchema: flattenSchema(schema, config.contentRoot),
1107
+ },
1108
+ })
1109
+
1110
+ // Import deleteEntry handler
1111
+ const { deleteEntry } = await import('./entries')
1112
+
1113
+ const res = await deleteEntry.handler(
1114
+ ctx,
1115
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
1116
+ {
1117
+ branch: unsafeAsBranchName('main'),
1118
+ entryPath: unsafeAsLogicalPath('content/posts/to-delete'),
1119
+ },
1120
+ )
1121
+
1122
+ expect(res.ok).toBe(true)
1123
+ expect(res.data?.deleted).toBe(true)
1124
+
1125
+ // Verify file was deleted
1126
+ const files = await fs.readdir(path.join(root, `content/posts.${postsId}`))
1127
+ expect(files.filter((f) => f.endsWith('.json') && f !== '.collection.json')).toHaveLength(0)
1128
+ })
1129
+
1130
+ it('returns 403 when user lacks edit permission', async () => {
1131
+ const root = await tmpDir()
1132
+
1133
+ const postsId = 'q52DCVPuH4ga'
1134
+ await fs.mkdir(path.join(root, `content/posts.${postsId}`), {
1135
+ recursive: true,
1136
+ })
1137
+
1138
+ await fs.writeFile(
1139
+ path.join(root, `content/posts.${postsId}/.collection.json`),
1140
+ JSON.stringify({
1141
+ name: 'posts',
1142
+ entries: [{ name: 'post', format: 'json', schema: 'postSchema' }],
1143
+ order: [],
1144
+ }),
1145
+ 'utf8',
1146
+ )
1147
+
1148
+ await fs.writeFile(
1149
+ path.join(root, `content/posts.${postsId}/post.protected.abc123def456.json`),
1150
+ JSON.stringify({ title: 'Protected' }),
1151
+ 'utf8',
1152
+ )
1153
+
1154
+ const entrySchemaRegistry = {
1155
+ postSchema: [{ name: 'title', type: 'string' }],
1156
+ }
1157
+ const metaFiles = await loadCollectionMetaFiles(path.join(root, 'content'))
1158
+ const schema = resolveCollectionReferences(metaFiles, entrySchemaRegistry)
1159
+
1160
+ const config = defineCanopyTestConfig({
1161
+ defaultBranchAccess: 'allow',
1162
+ contentRoot: 'content',
1163
+ schema,
1164
+ })
1165
+
1166
+ // Mock edit access denied
1167
+ const pathRules: PathPermission[] = [
1168
+ {
1169
+ path: unsafeAsPermissionPath(`content/posts.${postsId}/post.protected.abc123def456.json`),
1170
+ edit: { allowedUsers: ['admin'] },
1171
+ },
1172
+ ]
1173
+
1174
+ const checkBranchAccess = createCheckBranchAccess('allow')
1175
+ const checkContentAccess = createCheckContentAccess({
1176
+ checkBranchAccess,
1177
+ loadPathPermissions: vi.fn().mockResolvedValue(pathRules),
1178
+ defaultPathAccess: 'allow',
1179
+ mode: 'dev',
1180
+ })
1181
+
1182
+ const ctx = createMockApiContext({
1183
+ services: {
1184
+ config,
1185
+ entrySchemaRegistry: entrySchemaRegistry,
1186
+ checkBranchAccess,
1187
+ checkContentAccess,
1188
+ },
1189
+ branchContext: {
1190
+ ...createMockBranchContext({
1191
+ branchName: 'main',
1192
+ baseRoot: root,
1193
+ branchRoot: root,
1194
+ createdBy: 'u1',
1195
+ }),
1196
+ flatSchema: flattenSchema(schema, config.contentRoot),
1197
+ },
1198
+ })
1199
+
1200
+ const { deleteEntry } = await import('./entries')
1201
+
1202
+ const res = await deleteEntry.handler(
1203
+ ctx,
1204
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
1205
+ {
1206
+ branch: unsafeAsBranchName('main'),
1207
+ entryPath: unsafeAsLogicalPath('content/posts/protected'),
1208
+ },
1209
+ )
1210
+
1211
+ expect(res.ok).toBe(false)
1212
+ expect(res.status).toBe(403)
1213
+ expect(res.error).toContain('Edit permission required')
1214
+ })
1215
+
1216
+ it('returns 404 for non-existent entry', async () => {
1217
+ const root = await tmpDir()
1218
+
1219
+ const postsId = 'q52DCVPuH4ga'
1220
+ await fs.mkdir(path.join(root, `content/posts.${postsId}`), {
1221
+ recursive: true,
1222
+ })
1223
+
1224
+ await fs.writeFile(
1225
+ path.join(root, `content/posts.${postsId}/.collection.json`),
1226
+ JSON.stringify({
1227
+ name: 'posts',
1228
+ entries: [{ name: 'post', format: 'json', schema: 'postSchema' }],
1229
+ order: [],
1230
+ }),
1231
+ 'utf8',
1232
+ )
1233
+
1234
+ const entrySchemaRegistry = {
1235
+ postSchema: [{ name: 'title', type: 'string' }],
1236
+ }
1237
+ const metaFiles = await loadCollectionMetaFiles(path.join(root, 'content'))
1238
+ const schema = resolveCollectionReferences(metaFiles, entrySchemaRegistry)
1239
+
1240
+ const config = defineCanopyTestConfig({
1241
+ defaultBranchAccess: 'allow',
1242
+ contentRoot: 'content',
1243
+ schema,
1244
+ })
1245
+
1246
+ const checkBranchAccess = createCheckBranchAccess('allow')
1247
+ const checkContentAccess = createCheckContentAccess({
1248
+ checkBranchAccess,
1249
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
1250
+ defaultPathAccess: 'allow',
1251
+ mode: 'dev',
1252
+ })
1253
+
1254
+ const ctx = createMockApiContext({
1255
+ services: {
1256
+ config,
1257
+ entrySchemaRegistry: entrySchemaRegistry,
1258
+ checkBranchAccess,
1259
+ checkContentAccess,
1260
+ },
1261
+ branchContext: {
1262
+ ...createMockBranchContext({
1263
+ branchName: 'main',
1264
+ baseRoot: root,
1265
+ branchRoot: root,
1266
+ createdBy: 'u1',
1267
+ }),
1268
+ flatSchema: flattenSchema(schema, config.contentRoot),
1269
+ },
1270
+ })
1271
+
1272
+ const { deleteEntry } = await import('./entries')
1273
+
1274
+ const res = await deleteEntry.handler(
1275
+ ctx,
1276
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
1277
+ {
1278
+ branch: unsafeAsBranchName('main'),
1279
+ entryPath: unsafeAsLogicalPath('content/posts/nonexistent'),
1280
+ },
1281
+ )
1282
+
1283
+ expect(res.ok).toBe(false)
1284
+ expect(res.status).toBe(404)
1285
+ })
1286
+
1287
+ it('returns 400 for invalid entry path format', async () => {
1288
+ const root = await tmpDir()
1289
+ await fs.mkdir(path.join(root, 'content'), { recursive: true })
1290
+
1291
+ const config = defineCanopyTestConfig({
1292
+ defaultBranchAccess: 'allow',
1293
+ contentRoot: 'content',
1294
+ schema: { collections: [] },
1295
+ })
1296
+
1297
+ const checkBranchAccess = createCheckBranchAccess('allow')
1298
+ const checkContentAccess = createCheckContentAccess({
1299
+ checkBranchAccess,
1300
+ loadPathPermissions: vi.fn().mockResolvedValue([]),
1301
+ defaultPathAccess: 'allow',
1302
+ mode: 'dev',
1303
+ })
1304
+
1305
+ const ctx = createMockApiContext({
1306
+ services: {
1307
+ config,
1308
+ entrySchemaRegistry: {},
1309
+ checkBranchAccess,
1310
+ checkContentAccess,
1311
+ },
1312
+ branchContext: {
1313
+ ...createMockBranchContext({
1314
+ branchName: 'main',
1315
+ baseRoot: root,
1316
+ branchRoot: root,
1317
+ createdBy: 'u1',
1318
+ }),
1319
+ flatSchema: [],
1320
+ },
1321
+ })
1322
+
1323
+ const { deleteEntry } = await import('./entries')
1324
+
1325
+ // Path without slash is invalid
1326
+ const res = await deleteEntry.handler(
1327
+ ctx,
1328
+ { user: { type: 'authenticated', userId: 'u1', groups: [] } },
1329
+ {
1330
+ branch: unsafeAsBranchName('main'),
1331
+ entryPath: unsafeAsLogicalPath('invalid-no-slash'),
1332
+ },
1333
+ )
1334
+
1335
+ expect(res.ok).toBe(false)
1336
+ expect(res.status).toBe(400)
1337
+ expect(res.error).toContain('Invalid entry path format')
1338
+ })
1339
+ })