canopycms 0.0.0 → 0.0.2
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.
- package/dist/auth/plugin.d.ts +8 -0
- package/dist/auth/plugin.d.ts.map +1 -1
- package/dist/build-mode.d.ts +15 -5
- package/dist/build-mode.d.ts.map +1 -1
- package/dist/build-mode.js +18 -8
- package/dist/build-mode.js.map +1 -1
- package/dist/cli/init.d.ts +2 -2
- package/dist/cli/init.d.ts.map +1 -1
- package/dist/cli/init.js +37 -36
- package/dist/cli/init.js.map +1 -1
- package/dist/cli/template-files/ai-config.ts.template +21 -0
- package/dist/cli/template-files/ai-route.ts.template +10 -0
- package/dist/cli/template-files/canopy.ts.template +24 -0
- package/dist/cli/templates.d.ts +5 -1
- package/dist/cli/templates.d.ts.map +1 -1
- package/dist/cli/templates.js +9 -2
- package/dist/cli/templates.js.map +1 -1
- package/dist/config/schemas/config.d.ts +4 -0
- package/dist/config/schemas/config.d.ts.map +1 -1
- package/dist/config/schemas/config.js +2 -0
- package/dist/config/schemas/config.js.map +1 -1
- package/dist/config/types.d.ts +5 -0
- package/dist/config/types.d.ts.map +1 -1
- package/dist/content-reader.js +2 -2
- package/dist/content-reader.js.map +1 -1
- package/dist/context.js +5 -5
- package/dist/context.js.map +1 -1
- package/dist/operating-mode/client-unsafe-strategy.d.ts.map +1 -1
- package/dist/operating-mode/client-unsafe-strategy.js +15 -18
- package/dist/operating-mode/client-unsafe-strategy.js.map +1 -1
- package/dist/operating-mode/types.d.ts +8 -0
- package/dist/operating-mode/types.d.ts.map +1 -1
- package/dist/server.d.ts +2 -0
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +2 -0
- package/dist/server.js.map +1 -1
- package/package.json +5 -4
- package/src/cli/init.ts +43 -38
- package/dist/__integration__/fixtures/content-seeds.d.ts +0 -43
- package/dist/__integration__/fixtures/content-seeds.d.ts.map +0 -1
- package/dist/__integration__/fixtures/content-seeds.js +0 -99
- package/dist/__integration__/fixtures/content-seeds.js.map +0 -1
- package/dist/__integration__/fixtures/schemas.d.ts +0 -12
- package/dist/__integration__/fixtures/schemas.d.ts.map +0 -1
- package/dist/__integration__/fixtures/schemas.js +0 -65
- package/dist/__integration__/fixtures/schemas.js.map +0 -1
- package/dist/__integration__/test-utils/api-client.d.ts +0 -123
- package/dist/__integration__/test-utils/api-client.d.ts.map +0 -1
- package/dist/__integration__/test-utils/api-client.js +0 -118
- package/dist/__integration__/test-utils/api-client.js.map +0 -1
- package/dist/__integration__/test-utils/multi-user.d.ts +0 -25
- package/dist/__integration__/test-utils/multi-user.d.ts.map +0 -1
- package/dist/__integration__/test-utils/multi-user.js +0 -105
- package/dist/__integration__/test-utils/multi-user.js.map +0 -1
- package/dist/__integration__/test-utils/test-workspace.d.ts +0 -25
- package/dist/__integration__/test-utils/test-workspace.d.ts.map +0 -1
- package/dist/__integration__/test-utils/test-workspace.js +0 -102
- package/dist/__integration__/test-utils/test-workspace.js.map +0 -1
- package/dist/editor/BranchManager.stories.d.ts +0 -8
- package/dist/editor/BranchManager.stories.d.ts.map +0 -1
- package/dist/editor/BranchManager.stories.js +0 -74
- package/dist/editor/BranchManager.stories.js.map +0 -1
- package/dist/editor/CanopyEditor.stories.d.ts +0 -7
- package/dist/editor/CanopyEditor.stories.d.ts.map +0 -1
- package/dist/editor/CanopyEditor.stories.js +0 -99
- package/dist/editor/CanopyEditor.stories.js.map +0 -1
- package/dist/editor/CommentsPanel.stories.d.ts +0 -10
- package/dist/editor/CommentsPanel.stories.d.ts.map +0 -1
- package/dist/editor/CommentsPanel.stories.js +0 -175
- package/dist/editor/CommentsPanel.stories.js.map +0 -1
- package/dist/editor/Editor.stories.d.ts +0 -7
- package/dist/editor/Editor.stories.d.ts.map +0 -1
- package/dist/editor/Editor.stories.js +0 -95
- package/dist/editor/Editor.stories.js.map +0 -1
- package/dist/editor/EditorPanes.stories.d.ts +0 -7
- package/dist/editor/EditorPanes.stories.d.ts.map +0 -1
- package/dist/editor/EditorPanes.stories.js +0 -116
- package/dist/editor/EditorPanes.stories.js.map +0 -1
- package/dist/editor/EntryNavigator.stories.d.ts +0 -8
- package/dist/editor/EntryNavigator.stories.d.ts.map +0 -1
- package/dist/editor/EntryNavigator.stories.js +0 -42
- package/dist/editor/EntryNavigator.stories.js.map +0 -1
- package/dist/editor/FormRenderer.stories.d.ts +0 -7
- package/dist/editor/FormRenderer.stories.d.ts.map +0 -1
- package/dist/editor/FormRenderer.stories.js +0 -115
- package/dist/editor/FormRenderer.stories.js.map +0 -1
- package/dist/editor/GroupManager.stories.d.ts +0 -19
- package/dist/editor/GroupManager.stories.d.ts.map +0 -1
- package/dist/editor/GroupManager.stories.js +0 -265
- package/dist/editor/GroupManager.stories.js.map +0 -1
- package/dist/editor/PermissionManager.stories.d.ts +0 -20
- package/dist/editor/PermissionManager.stories.d.ts.map +0 -1
- package/dist/editor/PermissionManager.stories.js +0 -506
- package/dist/editor/PermissionManager.stories.js.map +0 -1
- package/dist/editor/comments/FieldWrapper.stories.d.ts +0 -10
- package/dist/editor/comments/FieldWrapper.stories.d.ts.map +0 -1
- package/dist/editor/comments/FieldWrapper.stories.js +0 -173
- package/dist/editor/comments/FieldWrapper.stories.js.map +0 -1
- package/dist/editor/fields/BlockField.stories.d.ts +0 -7
- package/dist/editor/fields/BlockField.stories.d.ts.map +0 -1
- package/dist/editor/fields/BlockField.stories.js +0 -50
- package/dist/editor/fields/BlockField.stories.js.map +0 -1
- package/dist/editor/fields/fields.stories.d.ts +0 -8
- package/dist/editor/fields/fields.stories.d.ts.map +0 -1
- package/dist/editor/fields/fields.stories.js +0 -34
- package/dist/editor/fields/fields.stories.js.map +0 -1
- package/dist/test-utils/api-test-helpers.d.ts +0 -238
- package/dist/test-utils/api-test-helpers.d.ts.map +0 -1
- package/dist/test-utils/api-test-helpers.js +0 -347
- package/dist/test-utils/api-test-helpers.js.map +0 -1
- package/dist/test-utils/console-spy.d.ts +0 -56
- package/dist/test-utils/console-spy.d.ts.map +0 -1
- package/dist/test-utils/console-spy.js +0 -81
- package/dist/test-utils/console-spy.js.map +0 -1
- package/dist/test-utils/git-helpers.d.ts +0 -21
- package/dist/test-utils/git-helpers.d.ts.map +0 -1
- package/dist/test-utils/git-helpers.js +0 -23
- package/dist/test-utils/git-helpers.js.map +0 -1
- package/dist/test-utils/index.d.ts +0 -5
- package/dist/test-utils/index.d.ts.map +0 -1
- package/dist/test-utils/index.js +0 -4
- package/dist/test-utils/index.js.map +0 -1
- package/src/__integration__/errors/invalid-content.test.ts +0 -238
- package/src/__integration__/errors/permission-denied.test.ts +0 -220
- package/src/__integration__/fixtures/content-seeds.ts +0 -105
- package/src/__integration__/fixtures/schemas.ts +0 -67
- package/src/__integration__/initialization/prod-sim-init.test.ts +0 -139
- package/src/__integration__/permissions/path-permissions.test.ts +0 -314
- package/src/__integration__/permissions/role-permissions.test.ts +0 -354
- package/src/__integration__/permissions/settings-branch-isolation.test.ts +0 -317
- package/src/__integration__/settings/groups-api.test.ts +0 -403
- package/src/__integration__/test-utils/api-client.ts +0 -167
- package/src/__integration__/test-utils/multi-user.ts +0 -129
- package/src/__integration__/test-utils/test-workspace.ts +0 -130
- package/src/__integration__/user/user-context.test.ts +0 -174
- package/src/__integration__/validation/input-validation.test.ts +0 -166
- package/src/__integration__/workflows/api-editing-workflow.test.ts +0 -244
- package/src/__integration__/workflows/conflict-resolution.test.ts +0 -259
- package/src/__integration__/workflows/editing-workflow.test.ts +0 -205
- package/src/__integration__/workflows/review-workflow.test.ts +0 -260
- package/src/ai/__tests__/build.integration.test.ts +0 -224
- package/src/ai/__tests__/generate.integration.test.ts +0 -495
- package/src/ai/__tests__/handler.integration.test.ts +0 -212
- package/src/ai/__tests__/json-to-markdown.test.ts +0 -553
- package/src/ai/generate.ts +0 -410
- package/src/ai/handler.ts +0 -123
- package/src/ai/index.ts +0 -26
- package/src/ai/json-to-markdown.ts +0 -424
- package/src/ai/resolve-branch.ts +0 -34
- package/src/ai/types.ts +0 -160
- package/src/api/AGENTS.md +0 -81
- package/src/api/__test__/mock-client.ts +0 -404
- package/src/api/assets.test.ts +0 -140
- package/src/api/assets.ts +0 -154
- package/src/api/branch-merge.test.ts +0 -163
- package/src/api/branch-merge.ts +0 -113
- package/src/api/branch-review.test.ts +0 -297
- package/src/api/branch-review.ts +0 -136
- package/src/api/branch-status.test.ts +0 -85
- package/src/api/branch-status.ts +0 -153
- package/src/api/branch-withdraw.test.ts +0 -146
- package/src/api/branch-withdraw.ts +0 -81
- package/src/api/branch-workflow.integration.test.ts +0 -578
- package/src/api/branch.test.ts +0 -620
- package/src/api/branch.ts +0 -492
- package/src/api/client.test.ts +0 -349
- package/src/api/client.ts +0 -506
- package/src/api/comments.test.ts +0 -285
- package/src/api/comments.ts +0 -210
- package/src/api/content.test.ts +0 -345
- package/src/api/content.ts +0 -454
- package/src/api/entries.test.ts +0 -1339
- package/src/api/entries.ts +0 -650
- package/src/api/github-sync.ts +0 -144
- package/src/api/groups.test.ts +0 -1013
- package/src/api/groups.ts +0 -375
- package/src/api/guards.test.ts +0 -533
- package/src/api/guards.ts +0 -271
- package/src/api/index.ts +0 -87
- package/src/api/permissions.test.ts +0 -766
- package/src/api/permissions.ts +0 -334
- package/src/api/reference-options.ts +0 -118
- package/src/api/resolve-references.ts +0 -107
- package/src/api/route-builder.ts +0 -289
- package/src/api/schema.test.ts +0 -840
- package/src/api/schema.ts +0 -936
- package/src/api/security.test.ts +0 -233
- package/src/api/settings-helpers.ts +0 -84
- package/src/api/types.ts +0 -40
- package/src/api/user.test.ts +0 -127
- package/src/api/user.ts +0 -42
- package/src/api/validators.test.ts +0 -275
- package/src/api/validators.ts +0 -176
- package/src/asset-store.test.ts +0 -37
- package/src/asset-store.ts +0 -110
- package/src/auth/cache.ts +0 -7
- package/src/auth/caching-auth-plugin.test.ts +0 -154
- package/src/auth/caching-auth-plugin.ts +0 -109
- package/src/auth/context-helpers.ts +0 -75
- package/src/auth/file-based-auth-cache.test.ts +0 -257
- package/src/auth/file-based-auth-cache.ts +0 -279
- package/src/auth/index.ts +0 -12
- package/src/auth/plugin.ts +0 -51
- package/src/auth/types.ts +0 -38
- package/src/authorization/__tests__/branch.test.ts +0 -260
- package/src/authorization/__tests__/content.test.ts +0 -142
- package/src/authorization/__tests__/path.test.ts +0 -133
- package/src/authorization/__tests__/permissions-loader.test.ts +0 -200
- package/src/authorization/branch.ts +0 -94
- package/src/authorization/content.ts +0 -93
- package/src/authorization/groups/index.ts +0 -11
- package/src/authorization/groups/loader.ts +0 -127
- package/src/authorization/groups/schema.ts +0 -48
- package/src/authorization/helpers.ts +0 -48
- package/src/authorization/index.ts +0 -84
- package/src/authorization/path.ts +0 -112
- package/src/authorization/permissions/index.ts +0 -11
- package/src/authorization/permissions/loader.ts +0 -116
- package/src/authorization/permissions/schema.ts +0 -66
- package/src/authorization/test-utils.ts +0 -15
- package/src/authorization/types.ts +0 -66
- package/src/authorization/validation.test.ts +0 -100
- package/src/authorization/validation.ts +0 -62
- package/src/branch-metadata.test.ts +0 -168
- package/src/branch-metadata.ts +0 -166
- package/src/branch-registry.test.ts +0 -248
- package/src/branch-registry.ts +0 -152
- package/src/branch-schema-cache.test.ts +0 -275
- package/src/branch-schema-cache.ts +0 -189
- package/src/branch-workspace.test.ts +0 -183
- package/src/branch-workspace.ts +0 -124
- package/src/build/generate-ai-content.ts +0 -78
- package/src/build/index.ts +0 -8
- package/src/build-mode.ts +0 -27
- package/src/cli/generate-ai-content.ts +0 -100
- package/src/cli/init.test.ts +0 -240
- package/src/cli/templates/canopy.ts.template +0 -55
- package/src/cli/templates.ts +0 -47
- package/src/client.ts +0 -12
- package/src/comment-store.test.ts +0 -442
- package/src/comment-store.ts +0 -301
- package/src/config/__tests__/config.test.ts +0 -513
- package/src/config/flatten.ts +0 -174
- package/src/config/helpers.ts +0 -167
- package/src/config/index.ts +0 -86
- package/src/config/schemas/collection.ts +0 -67
- package/src/config/schemas/config.ts +0 -77
- package/src/config/schemas/field.ts +0 -108
- package/src/config/schemas/media.ts +0 -27
- package/src/config/schemas/permissions.ts +0 -21
- package/src/config/types.ts +0 -321
- package/src/config/validation.ts +0 -70
- package/src/config-test.ts +0 -65
- package/src/config.ts +0 -11
- package/src/content-id-index.test.ts +0 -512
- package/src/content-id-index.ts +0 -479
- package/src/content-reader.test.ts +0 -478
- package/src/content-reader.ts +0 -214
- package/src/content-store.test.ts +0 -1126
- package/src/content-store.ts +0 -793
- package/src/context.ts +0 -111
- package/src/editor/BranchManager.stories.tsx +0 -80
- package/src/editor/BranchManager.test.tsx +0 -324
- package/src/editor/BranchManager.tsx +0 -461
- package/src/editor/CanopyEditor.stories.tsx +0 -128
- package/src/editor/CanopyEditor.test.tsx +0 -81
- package/src/editor/CanopyEditor.tsx +0 -73
- package/src/editor/CanopyEditorPage.test.tsx +0 -59
- package/src/editor/CanopyEditorPage.tsx +0 -25
- package/src/editor/CommentsPanel.stories.tsx +0 -184
- package/src/editor/CommentsPanel.tsx +0 -338
- package/src/editor/Editor.integration.test.tsx +0 -227
- package/src/editor/Editor.stories.tsx +0 -119
- package/src/editor/Editor.tsx +0 -1221
- package/src/editor/EditorPanes.stories.tsx +0 -256
- package/src/editor/EditorPanes.test.tsx +0 -77
- package/src/editor/EditorPanes.tsx +0 -180
- package/src/editor/EntryNavigator.stories.tsx +0 -65
- package/src/editor/EntryNavigator.test.tsx +0 -598
- package/src/editor/EntryNavigator.tsx +0 -665
- package/src/editor/FormRenderer.stories.tsx +0 -212
- package/src/editor/FormRenderer.test.tsx +0 -194
- package/src/editor/FormRenderer.tsx +0 -432
- package/src/editor/GroupManager.stories.tsx +0 -301
- package/src/editor/GroupManager.test.tsx +0 -682
- package/src/editor/GroupManager.tsx +0 -9
- package/src/editor/PermissionManager.stories.tsx +0 -539
- package/src/editor/PermissionManager.test.tsx +0 -864
- package/src/editor/PermissionManager.tsx +0 -12
- package/src/editor/canopy-path.test.ts +0 -23
- package/src/editor/canopy-path.ts +0 -52
- package/src/editor/client-reference-resolver.ts +0 -118
- package/src/editor/comments/BranchComments.tsx +0 -93
- package/src/editor/comments/EntryComments.tsx +0 -94
- package/src/editor/comments/FieldWrapper.stories.tsx +0 -210
- package/src/editor/comments/FieldWrapper.tsx +0 -129
- package/src/editor/comments/InlineCommentThread.test.tsx +0 -384
- package/src/editor/comments/InlineCommentThread.tsx +0 -246
- package/src/editor/comments/ThreadCarousel.test.tsx +0 -393
- package/src/editor/comments/ThreadCarousel.tsx +0 -525
- package/src/editor/components/ConfirmDeleteModal.tsx +0 -49
- package/src/editor/components/EditorContext.tsx +0 -49
- package/src/editor/components/EditorFooter.tsx +0 -47
- package/src/editor/components/EditorHeader.tsx +0 -492
- package/src/editor/components/EditorSidebar.tsx +0 -193
- package/src/editor/components/EntryCreateModal.tsx +0 -193
- package/src/editor/components/RenameEntryModal.tsx +0 -152
- package/src/editor/components/UserBadge.test.tsx +0 -274
- package/src/editor/components/UserBadge.tsx +0 -240
- package/src/editor/components/index.ts +0 -6
- package/src/editor/context/ApiClientContext.tsx +0 -56
- package/src/editor/context/EditorStateContext.tsx +0 -221
- package/src/editor/context/index.ts +0 -40
- package/src/editor/editor-config.test.ts +0 -385
- package/src/editor/editor-config.ts +0 -94
- package/src/editor/editor-utils.test.ts +0 -772
- package/src/editor/editor-utils.ts +0 -303
- package/src/editor/env.ts +0 -4
- package/src/editor/fields/BlockField.stories.tsx +0 -79
- package/src/editor/fields/BlockField.tsx +0 -267
- package/src/editor/fields/CodeField.tsx +0 -41
- package/src/editor/fields/MarkdownField.tsx +0 -205
- package/src/editor/fields/ObjectField.tsx +0 -71
- package/src/editor/fields/ReferenceField.tsx +0 -138
- package/src/editor/fields/SelectField.tsx +0 -76
- package/src/editor/fields/TextField.tsx +0 -35
- package/src/editor/fields/ToggleField.tsx +0 -37
- package/src/editor/fields/fields.stories.tsx +0 -40
- package/src/editor/group-manager/ExternalGroupsTab.tsx +0 -114
- package/src/editor/group-manager/GroupCard.tsx +0 -102
- package/src/editor/group-manager/GroupForm.tsx +0 -66
- package/src/editor/group-manager/InternalGroupsTab.tsx +0 -147
- package/src/editor/group-manager/MemberList.tsx +0 -184
- package/src/editor/group-manager/hooks/useExternalGroupSearch.ts +0 -63
- package/src/editor/group-manager/hooks/useGroupState.ts +0 -134
- package/src/editor/group-manager/hooks/useUserSearch.ts +0 -84
- package/src/editor/group-manager/index.tsx +0 -210
- package/src/editor/group-manager/types.ts +0 -28
- package/src/editor/hooks/README.md +0 -26
- package/src/editor/hooks/__test__/test-utils.tsx +0 -183
- package/src/editor/hooks/index.ts +0 -23
- package/src/editor/hooks/useBranchActions.test.tsx +0 -267
- package/src/editor/hooks/useBranchActions.tsx +0 -121
- package/src/editor/hooks/useBranchManager.test.tsx +0 -391
- package/src/editor/hooks/useBranchManager.tsx +0 -326
- package/src/editor/hooks/useCommentSystem.test.ts +0 -615
- package/src/editor/hooks/useCommentSystem.ts +0 -347
- package/src/editor/hooks/useDraftManager.test.ts +0 -375
- package/src/editor/hooks/useDraftManager.ts +0 -259
- package/src/editor/hooks/useEditorLayout.test.ts +0 -147
- package/src/editor/hooks/useEditorLayout.ts +0 -67
- package/src/editor/hooks/useEntryManager.test.ts +0 -588
- package/src/editor/hooks/useEntryManager.ts +0 -387
- package/src/editor/hooks/useGroupManager.test.ts +0 -277
- package/src/editor/hooks/useGroupManager.ts +0 -139
- package/src/editor/hooks/usePermissionManager.test.ts +0 -211
- package/src/editor/hooks/usePermissionManager.ts +0 -113
- package/src/editor/hooks/useReferenceResolution.ts +0 -248
- package/src/editor/hooks/useSchemaManager.test.ts +0 -370
- package/src/editor/hooks/useSchemaManager.ts +0 -310
- package/src/editor/hooks/useUserContext.tsx +0 -57
- package/src/editor/hooks/useUserMetadata.test.ts +0 -191
- package/src/editor/hooks/useUserMetadata.ts +0 -71
- package/src/editor/permission-manager/GroupSelector.tsx +0 -73
- package/src/editor/permission-manager/PermissionEditor.tsx +0 -321
- package/src/editor/permission-manager/PermissionLevelBadge.tsx +0 -53
- package/src/editor/permission-manager/PermissionTree.tsx +0 -237
- package/src/editor/permission-manager/UserSelector.tsx +0 -95
- package/src/editor/permission-manager/constants.tsx +0 -18
- package/src/editor/permission-manager/hooks/useGroupsAndUsers.ts +0 -153
- package/src/editor/permission-manager/hooks/usePermissionTree.ts +0 -200
- package/src/editor/permission-manager/index.tsx +0 -294
- package/src/editor/permission-manager/types.ts +0 -58
- package/src/editor/permission-manager/utils.ts +0 -179
- package/src/editor/preview-bridge.test.tsx +0 -50
- package/src/editor/preview-bridge.tsx +0 -294
- package/src/editor/schema-editor/CollectionEditor.test.tsx +0 -238
- package/src/editor/schema-editor/CollectionEditor.tsx +0 -520
- package/src/editor/schema-editor/EntryTypeEditor.test.tsx +0 -215
- package/src/editor/schema-editor/EntryTypeEditor.tsx +0 -367
- package/src/editor/schema-editor/index.ts +0 -19
- package/src/editor/setup-test-dom.ts +0 -10
- package/src/editor/test-setup.ts +0 -33
- package/src/editor/theme.tsx +0 -119
- package/src/editor/utils/env.ts +0 -39
- package/src/entry-schema-registry.test.ts +0 -281
- package/src/entry-schema-registry.ts +0 -121
- package/src/entry-schema.ts +0 -84
- package/src/git-manager.test.ts +0 -552
- package/src/git-manager.ts +0 -667
- package/src/github-service.test.ts +0 -312
- package/src/github-service.ts +0 -295
- package/src/http/handler.test.ts +0 -275
- package/src/http/handler.ts +0 -280
- package/src/http/index.ts +0 -11
- package/src/http/router.ts +0 -164
- package/src/http/types.ts +0 -44
- package/src/id.test.ts +0 -48
- package/src/id.ts +0 -22
- package/src/index.ts +0 -26
- package/src/operating-mode/__tests__/strategies.test.ts +0 -511
- package/src/operating-mode/client-safe-strategy.ts +0 -184
- package/src/operating-mode/client-unsafe-strategy.ts +0 -303
- package/src/operating-mode/client.ts +0 -13
- package/src/operating-mode/index.ts +0 -34
- package/src/operating-mode/types.ts +0 -186
- package/src/paths/__tests__/branch.test.ts +0 -53
- package/src/paths/__tests__/normalize.test.ts +0 -141
- package/src/paths/__tests__/resolve.test.ts +0 -207
- package/src/paths/__tests__/validation.test.ts +0 -61
- package/src/paths/branch.ts +0 -115
- package/src/paths/index.ts +0 -73
- package/src/paths/normalize-server.ts +0 -40
- package/src/paths/normalize.ts +0 -107
- package/src/paths/resolve.ts +0 -61
- package/src/paths/test-utils.ts +0 -37
- package/src/paths/types.ts +0 -68
- package/src/paths/validation.test.ts +0 -480
- package/src/paths/validation.ts +0 -391
- package/src/reference-resolver.test.ts +0 -107
- package/src/reference-resolver.ts +0 -157
- package/src/schema/index.ts +0 -29
- package/src/schema/meta-loader.ts +0 -366
- package/src/schema/resolver.ts +0 -83
- package/src/schema/schema-store-types.ts +0 -56
- package/src/schema/schema-store.test.ts +0 -816
- package/src/schema/schema-store.ts +0 -795
- package/src/schema/types.ts +0 -33
- package/src/schema-meta-loader.test.ts +0 -447
- package/src/server.ts +0 -15
- package/src/services.test.ts +0 -559
- package/src/services.ts +0 -373
- package/src/settings-branch-utils.ts +0 -53
- package/src/settings-workspace.ts +0 -156
- package/src/task-queue/README.md +0 -144
- package/src/task-queue/index.ts +0 -14
- package/src/task-queue/task-queue.test.ts +0 -524
- package/src/task-queue/task-queue.ts +0 -514
- package/src/task-queue/types.ts +0 -41
- package/src/test-utils/api-test-helpers.ts +0 -445
- package/src/test-utils/console-spy.test.ts +0 -14
- package/src/test-utils/console-spy.ts +0 -125
- package/src/test-utils/git-helpers.ts +0 -31
- package/src/test-utils/index.ts +0 -4
- package/src/types.ts +0 -54
- package/src/user.ts +0 -118
- package/src/utils/debug.test.ts +0 -114
- package/src/utils/debug.ts +0 -127
- package/src/utils/error.test.ts +0 -92
- package/src/utils/error.ts +0 -83
- package/src/utils/format.ts +0 -12
- package/src/validation/__tests__/field-traversal.test.ts +0 -263
- package/src/validation/deletion-checker.ts +0 -234
- package/src/validation/field-traversal.ts +0 -146
- package/src/validation/reference-validator.ts +0 -168
- package/src/worker/cms-worker-rebase.test.ts +0 -473
- package/src/worker/cms-worker.ts +0 -777
- package/src/worker/integration.test.ts +0 -289
- package/src/worker/task-queue-config.ts +0 -25
- package/src/worker/task-queue.test.ts +0 -452
- package/src/worker/task-queue.ts +0 -58
- /package/{src/cli/templates → dist/cli/template-files}/Dockerfile.cms.template +0 -0
- /package/{src/cli/templates → dist/cli/template-files}/canopycms.config.ts.template +0 -0
- /package/{src/cli/templates → dist/cli/template-files}/deploy-cms.yml.template +0 -0
- /package/{src/cli/templates → dist/cli/template-files}/edit-page.tsx.template +0 -0
- /package/{src/cli/templates → dist/cli/template-files}/route.ts.template +0 -0
- /package/{src/cli/templates → dist/cli/template-files}/schemas.ts.template +0 -0
|
@@ -1,513 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it } from 'vitest'
|
|
2
|
-
|
|
3
|
-
import type { CanopyConfigFragment, CollectionConfig } from '../types'
|
|
4
|
-
import type { ContentId } from '../../paths/types'
|
|
5
|
-
import { ROOT_COLLECTION_ID } from '../../paths/types'
|
|
6
|
-
import { composeCanopyConfig, defineCanopyConfig } from '../helpers'
|
|
7
|
-
import { flattenSchema } from '../flatten'
|
|
8
|
-
import { validateCanopyConfig } from '../validation'
|
|
9
|
-
|
|
10
|
-
const gitAuthor = {
|
|
11
|
-
gitBotAuthorName: 'Test Bot',
|
|
12
|
-
gitBotAuthorEmail: 'bot@example.com',
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
describe('config validation', () => {
|
|
16
|
-
it('accepts a valid config with mdx collection and blocks', () => {
|
|
17
|
-
const schema = {
|
|
18
|
-
collections: [
|
|
19
|
-
{
|
|
20
|
-
name: 'posts',
|
|
21
|
-
path: 'posts',
|
|
22
|
-
entries: [
|
|
23
|
-
{
|
|
24
|
-
name: 'entry',
|
|
25
|
-
format: 'mdx' as const,
|
|
26
|
-
schema: [
|
|
27
|
-
{ name: 'title', type: 'string' as const, required: true },
|
|
28
|
-
{ name: 'body', type: 'mdx' as const, required: true },
|
|
29
|
-
{ name: 'tags', type: 'string' as const, list: true },
|
|
30
|
-
{
|
|
31
|
-
name: 'layout',
|
|
32
|
-
type: 'block' as const,
|
|
33
|
-
templates: [
|
|
34
|
-
{
|
|
35
|
-
name: 'hero',
|
|
36
|
-
label: 'Hero',
|
|
37
|
-
fields: [
|
|
38
|
-
{
|
|
39
|
-
name: 'headline',
|
|
40
|
-
type: 'string' as const,
|
|
41
|
-
required: true,
|
|
42
|
-
},
|
|
43
|
-
{ name: 'ctaLabel', type: 'string' as const },
|
|
44
|
-
],
|
|
45
|
-
},
|
|
46
|
-
],
|
|
47
|
-
},
|
|
48
|
-
],
|
|
49
|
-
},
|
|
50
|
-
],
|
|
51
|
-
},
|
|
52
|
-
],
|
|
53
|
-
} as const
|
|
54
|
-
|
|
55
|
-
defineCanopyConfig({
|
|
56
|
-
...gitAuthor,
|
|
57
|
-
media: { adapter: 's3', bucket: 'my-bucket', region: 'us-east-1' },
|
|
58
|
-
})
|
|
59
|
-
|
|
60
|
-
// Verify schema is valid on its own (config no longer contains schema)
|
|
61
|
-
expect(schema.collections).toBeDefined()
|
|
62
|
-
expect(schema.collections[0].name).toBe('posts')
|
|
63
|
-
})
|
|
64
|
-
|
|
65
|
-
it('rejects select fields without options', () => {
|
|
66
|
-
expect(() =>
|
|
67
|
-
validateCanopyConfig({
|
|
68
|
-
...gitAuthor,
|
|
69
|
-
}),
|
|
70
|
-
).not.toThrow() // Config validation no longer includes schema validation
|
|
71
|
-
})
|
|
72
|
-
|
|
73
|
-
it('allows config without schema (schema loaded from .collection.json)', () => {
|
|
74
|
-
// Schema is loaded from .collection.json meta files, not from config
|
|
75
|
-
expect(() =>
|
|
76
|
-
validateCanopyConfig({
|
|
77
|
-
...gitAuthor,
|
|
78
|
-
}),
|
|
79
|
-
).not.toThrow()
|
|
80
|
-
})
|
|
81
|
-
|
|
82
|
-
it('composes config fragments from multiple files', () => {
|
|
83
|
-
const posts: CanopyConfigFragment = {
|
|
84
|
-
...gitAuthor,
|
|
85
|
-
}
|
|
86
|
-
const pages: CanopyConfigFragment = {
|
|
87
|
-
...gitAuthor,
|
|
88
|
-
media: { adapter: 'local' as const },
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
const config = composeCanopyConfig(posts, pages)
|
|
92
|
-
|
|
93
|
-
expect(config.media?.adapter).toBe('local')
|
|
94
|
-
})
|
|
95
|
-
|
|
96
|
-
it('flattens nested paths relative to parents', () => {
|
|
97
|
-
const schema = {
|
|
98
|
-
collections: [
|
|
99
|
-
{
|
|
100
|
-
name: 'content',
|
|
101
|
-
path: 'content',
|
|
102
|
-
entries: [
|
|
103
|
-
{
|
|
104
|
-
name: 'entry',
|
|
105
|
-
format: 'json' as const,
|
|
106
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
107
|
-
},
|
|
108
|
-
],
|
|
109
|
-
collections: [
|
|
110
|
-
{
|
|
111
|
-
name: 'pages',
|
|
112
|
-
path: 'pages',
|
|
113
|
-
entries: [
|
|
114
|
-
{
|
|
115
|
-
name: 'entry',
|
|
116
|
-
format: 'md' as const,
|
|
117
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
118
|
-
},
|
|
119
|
-
],
|
|
120
|
-
},
|
|
121
|
-
],
|
|
122
|
-
},
|
|
123
|
-
],
|
|
124
|
-
} as const
|
|
125
|
-
|
|
126
|
-
const configBundle = defineCanopyConfig({
|
|
127
|
-
...gitAuthor,
|
|
128
|
-
})
|
|
129
|
-
const cfg = configBundle.server
|
|
130
|
-
const flat = flattenSchema(schema, cfg.contentRoot || 'content')
|
|
131
|
-
|
|
132
|
-
const contentCollection = flat.find((item) => item.logicalPath === 'content/content')
|
|
133
|
-
const pagesCollection = flat.find((item) => item.logicalPath === 'content/content/pages')
|
|
134
|
-
|
|
135
|
-
expect(contentCollection).toBeDefined()
|
|
136
|
-
expect(contentCollection?.type).toBe('collection')
|
|
137
|
-
expect(pagesCollection).toBeDefined()
|
|
138
|
-
expect(pagesCollection?.type).toBe('collection')
|
|
139
|
-
expect(pagesCollection?.parentPath).toBe('content/content')
|
|
140
|
-
})
|
|
141
|
-
|
|
142
|
-
it('handles deeply nested collections with correct paths', () => {
|
|
143
|
-
const schema = {
|
|
144
|
-
collections: [
|
|
145
|
-
{
|
|
146
|
-
name: 'docs',
|
|
147
|
-
path: 'docs',
|
|
148
|
-
entries: [
|
|
149
|
-
{
|
|
150
|
-
name: 'entry',
|
|
151
|
-
format: 'md' as const,
|
|
152
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
153
|
-
},
|
|
154
|
-
],
|
|
155
|
-
collections: [
|
|
156
|
-
{
|
|
157
|
-
name: 'api',
|
|
158
|
-
path: 'api',
|
|
159
|
-
entries: [
|
|
160
|
-
{
|
|
161
|
-
name: 'entry',
|
|
162
|
-
format: 'md' as const,
|
|
163
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
164
|
-
},
|
|
165
|
-
],
|
|
166
|
-
collections: [
|
|
167
|
-
{
|
|
168
|
-
name: 'v2',
|
|
169
|
-
path: 'v2',
|
|
170
|
-
entries: [
|
|
171
|
-
{
|
|
172
|
-
name: 'entry',
|
|
173
|
-
format: 'md' as const,
|
|
174
|
-
schema: [{ name: 'content', type: 'markdown' as const }],
|
|
175
|
-
},
|
|
176
|
-
],
|
|
177
|
-
},
|
|
178
|
-
],
|
|
179
|
-
},
|
|
180
|
-
],
|
|
181
|
-
},
|
|
182
|
-
],
|
|
183
|
-
} as const
|
|
184
|
-
|
|
185
|
-
const configBundle = defineCanopyConfig({
|
|
186
|
-
...gitAuthor,
|
|
187
|
-
})
|
|
188
|
-
const cfg = configBundle.server
|
|
189
|
-
const flat = flattenSchema(schema, cfg.contentRoot || 'content')
|
|
190
|
-
|
|
191
|
-
const docsCollection = flat.find((item) => item.logicalPath === 'content/docs')
|
|
192
|
-
const apiCollection = flat.find((item) => item.logicalPath === 'content/docs/api')
|
|
193
|
-
const v2Collection = flat.find((item) => item.logicalPath === 'content/docs/api/v2')
|
|
194
|
-
|
|
195
|
-
expect(docsCollection).toBeDefined()
|
|
196
|
-
expect(docsCollection?.type).toBe('collection')
|
|
197
|
-
|
|
198
|
-
expect(apiCollection).toBeDefined()
|
|
199
|
-
expect(apiCollection?.type).toBe('collection')
|
|
200
|
-
expect(apiCollection?.parentPath).toBe('content/docs')
|
|
201
|
-
|
|
202
|
-
expect(v2Collection).toBeDefined()
|
|
203
|
-
expect(v2Collection?.type).toBe('collection')
|
|
204
|
-
expect(v2Collection?.parentPath).toBe('content/docs/api')
|
|
205
|
-
})
|
|
206
|
-
|
|
207
|
-
it('correctly flattens nested collections without path duplication', () => {
|
|
208
|
-
const schema = {
|
|
209
|
-
collections: [
|
|
210
|
-
{
|
|
211
|
-
name: 'docs',
|
|
212
|
-
path: 'docs',
|
|
213
|
-
entries: [
|
|
214
|
-
{
|
|
215
|
-
name: 'entry',
|
|
216
|
-
format: 'mdx' as const,
|
|
217
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
218
|
-
},
|
|
219
|
-
],
|
|
220
|
-
collections: [
|
|
221
|
-
{
|
|
222
|
-
name: 'api',
|
|
223
|
-
path: 'api',
|
|
224
|
-
entries: [
|
|
225
|
-
{
|
|
226
|
-
name: 'entry',
|
|
227
|
-
format: 'mdx' as const,
|
|
228
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
229
|
-
},
|
|
230
|
-
],
|
|
231
|
-
collections: [
|
|
232
|
-
{
|
|
233
|
-
name: 'v1',
|
|
234
|
-
path: 'v1',
|
|
235
|
-
entries: [
|
|
236
|
-
{
|
|
237
|
-
name: 'entry',
|
|
238
|
-
format: 'mdx' as const,
|
|
239
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
240
|
-
},
|
|
241
|
-
],
|
|
242
|
-
},
|
|
243
|
-
],
|
|
244
|
-
},
|
|
245
|
-
],
|
|
246
|
-
},
|
|
247
|
-
],
|
|
248
|
-
} as const
|
|
249
|
-
|
|
250
|
-
const configBundle = defineCanopyConfig({
|
|
251
|
-
...gitAuthor,
|
|
252
|
-
})
|
|
253
|
-
const cfg = configBundle.server
|
|
254
|
-
const flat = flattenSchema(schema, cfg.contentRoot || 'content')
|
|
255
|
-
|
|
256
|
-
// Find all collections
|
|
257
|
-
const docs = flat.find((item) => item.type === 'collection' && item.name === 'docs')
|
|
258
|
-
const api = flat.find((item) => item.type === 'collection' && item.name === 'api')
|
|
259
|
-
const v1 = flat.find((item) => item.type === 'collection' && item.name === 'v1')
|
|
260
|
-
|
|
261
|
-
// Verify docs collection (child of content root)
|
|
262
|
-
expect(docs).toBeDefined()
|
|
263
|
-
expect(docs?.logicalPath).toBe('content/docs')
|
|
264
|
-
expect(docs?.parentPath).toBe('content') // Now has content root as parent
|
|
265
|
-
|
|
266
|
-
// Verify api collection (nested under docs)
|
|
267
|
-
expect(api).toBeDefined()
|
|
268
|
-
expect(api?.logicalPath).toBe('content/docs/api')
|
|
269
|
-
expect(api?.parentPath).toBe('content/docs')
|
|
270
|
-
|
|
271
|
-
// Verify v1 collection (nested under api)
|
|
272
|
-
expect(v1).toBeDefined()
|
|
273
|
-
expect(v1?.logicalPath).toBe('content/docs/api/v1')
|
|
274
|
-
expect(v1?.parentPath).toBe('content/docs/api')
|
|
275
|
-
})
|
|
276
|
-
|
|
277
|
-
it('handles schema-meta-loader nested structure correctly (from .collection.json pattern)', () => {
|
|
278
|
-
// This simulates the structure created by schema-meta-loader
|
|
279
|
-
// where nested collections have FULL paths (e.g., "docs/api") not relative paths (e.g., "api")
|
|
280
|
-
const schema = {
|
|
281
|
-
collections: [
|
|
282
|
-
{
|
|
283
|
-
name: 'docs',
|
|
284
|
-
path: 'docs', // Top-level path
|
|
285
|
-
entries: [
|
|
286
|
-
{
|
|
287
|
-
name: 'entry',
|
|
288
|
-
format: 'json' as const,
|
|
289
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
290
|
-
},
|
|
291
|
-
],
|
|
292
|
-
collections: [
|
|
293
|
-
{
|
|
294
|
-
name: 'api',
|
|
295
|
-
path: 'docs/api', // FULL path from content root (as set by schema-meta-loader)
|
|
296
|
-
entries: [
|
|
297
|
-
{
|
|
298
|
-
name: 'entry',
|
|
299
|
-
format: 'json' as const,
|
|
300
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
301
|
-
},
|
|
302
|
-
],
|
|
303
|
-
collections: [
|
|
304
|
-
{
|
|
305
|
-
name: 'v1',
|
|
306
|
-
path: 'docs/api/v1', // FULL path from content root
|
|
307
|
-
entries: [
|
|
308
|
-
{
|
|
309
|
-
name: 'entry',
|
|
310
|
-
format: 'json' as const,
|
|
311
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
312
|
-
},
|
|
313
|
-
],
|
|
314
|
-
},
|
|
315
|
-
],
|
|
316
|
-
},
|
|
317
|
-
],
|
|
318
|
-
},
|
|
319
|
-
],
|
|
320
|
-
} as const
|
|
321
|
-
|
|
322
|
-
const configBundle = defineCanopyConfig({
|
|
323
|
-
...gitAuthor,
|
|
324
|
-
})
|
|
325
|
-
const cfg = configBundle.server
|
|
326
|
-
const flat = flattenSchema(schema, cfg.contentRoot || 'content')
|
|
327
|
-
|
|
328
|
-
// Find all collections
|
|
329
|
-
const docs = flat.find((item) => item.type === 'collection' && item.name === 'docs')
|
|
330
|
-
const api = flat.find((item) => item.type === 'collection' && item.name === 'api')
|
|
331
|
-
const v1 = flat.find((item) => item.type === 'collection' && item.name === 'v1')
|
|
332
|
-
|
|
333
|
-
// Verify docs collection (child of content root)
|
|
334
|
-
expect(docs).toBeDefined()
|
|
335
|
-
expect(docs?.logicalPath).toBe('content/docs')
|
|
336
|
-
expect(docs?.parentPath).toBe('content') // Now has content root as parent
|
|
337
|
-
|
|
338
|
-
// Verify api collection (nested under docs)
|
|
339
|
-
expect(api).toBeDefined()
|
|
340
|
-
expect(api?.logicalPath).toBe('content/docs/api') // Should NOT be 'content/docs/docs/api'
|
|
341
|
-
expect(api?.parentPath).toBe('content/docs')
|
|
342
|
-
|
|
343
|
-
// Verify v1 collection (nested under api)
|
|
344
|
-
expect(v1).toBeDefined()
|
|
345
|
-
expect(v1?.logicalPath).toBe('content/docs/api/v1') // Should NOT be 'content/docs/docs/api/api/v1'
|
|
346
|
-
expect(v1?.parentPath).toBe('content/docs/api')
|
|
347
|
-
})
|
|
348
|
-
|
|
349
|
-
it('strips embedded IDs from collection paths for logical identity', () => {
|
|
350
|
-
// This test verifies that embedded IDs in directory names are stripped from logical paths
|
|
351
|
-
// Directory on disk: "docs.bChqT78gcaLd", but logical path should be "docs"
|
|
352
|
-
// This keeps IDs hidden from URLs and the editor while still using them for filesystem uniqueness
|
|
353
|
-
const schema = {
|
|
354
|
-
collections: [
|
|
355
|
-
{
|
|
356
|
-
name: 'docs',
|
|
357
|
-
path: 'docs', // Logical path without ID
|
|
358
|
-
entries: [
|
|
359
|
-
{
|
|
360
|
-
name: 'entry',
|
|
361
|
-
format: 'json' as const,
|
|
362
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
363
|
-
},
|
|
364
|
-
],
|
|
365
|
-
collections: [
|
|
366
|
-
{
|
|
367
|
-
name: 'api',
|
|
368
|
-
path: 'docs/api', // Logical path without ID
|
|
369
|
-
entries: [
|
|
370
|
-
{
|
|
371
|
-
name: 'entry',
|
|
372
|
-
format: 'json' as const,
|
|
373
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
374
|
-
},
|
|
375
|
-
],
|
|
376
|
-
collections: [
|
|
377
|
-
{
|
|
378
|
-
name: 'v1',
|
|
379
|
-
path: 'docs/api/v1', // Logical path without ID
|
|
380
|
-
entries: [
|
|
381
|
-
{
|
|
382
|
-
name: 'entry',
|
|
383
|
-
format: 'json' as const,
|
|
384
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
385
|
-
},
|
|
386
|
-
],
|
|
387
|
-
},
|
|
388
|
-
],
|
|
389
|
-
},
|
|
390
|
-
],
|
|
391
|
-
},
|
|
392
|
-
{
|
|
393
|
-
name: 'posts',
|
|
394
|
-
path: 'posts', // Logical path without ID
|
|
395
|
-
entries: [
|
|
396
|
-
{
|
|
397
|
-
name: 'entry',
|
|
398
|
-
format: 'json' as const,
|
|
399
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
400
|
-
},
|
|
401
|
-
],
|
|
402
|
-
},
|
|
403
|
-
],
|
|
404
|
-
} as const
|
|
405
|
-
|
|
406
|
-
const configBundle = defineCanopyConfig({
|
|
407
|
-
...gitAuthor,
|
|
408
|
-
})
|
|
409
|
-
const cfg = configBundle.server
|
|
410
|
-
const flat = flattenSchema(schema, cfg.contentRoot || 'content')
|
|
411
|
-
|
|
412
|
-
// Find all collections
|
|
413
|
-
const docs = flat.find((item) => item.type === 'collection' && item.name === 'docs')
|
|
414
|
-
const api = flat.find((item) => item.type === 'collection' && item.name === 'api')
|
|
415
|
-
const v1 = flat.find((item) => item.type === 'collection' && item.name === 'v1')
|
|
416
|
-
const posts = flat.find((item) => item.type === 'collection' && item.name === 'posts')
|
|
417
|
-
|
|
418
|
-
// Verify docs collection (child of content root) - NO embedded ID in logical path
|
|
419
|
-
expect(docs).toBeDefined()
|
|
420
|
-
expect(docs?.logicalPath).toBe('content/docs')
|
|
421
|
-
expect(docs?.parentPath).toBe('content') // Now has content root as parent
|
|
422
|
-
|
|
423
|
-
// Verify api collection (nested under docs) - NO embedded ID in logical path
|
|
424
|
-
expect(api).toBeDefined()
|
|
425
|
-
expect(api?.logicalPath).toBe('content/docs/api')
|
|
426
|
-
expect(api?.parentPath).toBe('content/docs')
|
|
427
|
-
|
|
428
|
-
// Verify v1 collection (nested under api) - NO embedded ID in logical path
|
|
429
|
-
expect(v1).toBeDefined()
|
|
430
|
-
expect(v1?.logicalPath).toBe('content/docs/api/v1')
|
|
431
|
-
expect(v1?.parentPath).toBe('content/docs/api')
|
|
432
|
-
|
|
433
|
-
// Verify posts collection (child of content root) - NO embedded ID in logical path
|
|
434
|
-
expect(posts).toBeDefined()
|
|
435
|
-
expect(posts?.logicalPath).toBe('content/posts')
|
|
436
|
-
expect(posts?.parentPath).toBe('content') // Now has content root as parent
|
|
437
|
-
})
|
|
438
|
-
|
|
439
|
-
it('threads contentId from CollectionConfig through to FlatSchemaItem', () => {
|
|
440
|
-
const TEST_ID = 'a1b2c3d4e5f6' as ContentId
|
|
441
|
-
const CHILD_ID = 'Xz9kL2mN4pQr' as ContentId
|
|
442
|
-
|
|
443
|
-
const schema = {
|
|
444
|
-
collections: [
|
|
445
|
-
{
|
|
446
|
-
name: 'posts',
|
|
447
|
-
path: 'posts',
|
|
448
|
-
contentId: TEST_ID,
|
|
449
|
-
entries: [
|
|
450
|
-
{
|
|
451
|
-
name: 'entry',
|
|
452
|
-
format: 'json' as const,
|
|
453
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
454
|
-
},
|
|
455
|
-
],
|
|
456
|
-
collections: [
|
|
457
|
-
{
|
|
458
|
-
name: 'drafts',
|
|
459
|
-
path: 'posts/drafts',
|
|
460
|
-
contentId: CHILD_ID,
|
|
461
|
-
entries: [
|
|
462
|
-
{
|
|
463
|
-
name: 'entry',
|
|
464
|
-
format: 'json' as const,
|
|
465
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
466
|
-
},
|
|
467
|
-
],
|
|
468
|
-
},
|
|
469
|
-
],
|
|
470
|
-
} satisfies CollectionConfig,
|
|
471
|
-
],
|
|
472
|
-
}
|
|
473
|
-
|
|
474
|
-
const flat = flattenSchema(schema, 'content')
|
|
475
|
-
|
|
476
|
-
const root = flat.find((item) => item.type === 'collection' && item.logicalPath === 'content')
|
|
477
|
-
const posts = flat.find((item) => item.type === 'collection' && item.name === 'posts')
|
|
478
|
-
const drafts = flat.find((item) => item.type === 'collection' && item.name === 'drafts')
|
|
479
|
-
|
|
480
|
-
// Root collection gets the sentinel
|
|
481
|
-
expect(root).toBeDefined()
|
|
482
|
-
expect(root?.type === 'collection' && root.contentId).toBe(ROOT_COLLECTION_ID)
|
|
483
|
-
|
|
484
|
-
// Child collections carry their own contentId
|
|
485
|
-
expect(posts?.type === 'collection' && posts.contentId).toBe(TEST_ID)
|
|
486
|
-
expect(drafts?.type === 'collection' && drafts.contentId).toBe(CHILD_ID)
|
|
487
|
-
})
|
|
488
|
-
|
|
489
|
-
it('leaves contentId undefined when CollectionConfig has no contentId', () => {
|
|
490
|
-
const schema = {
|
|
491
|
-
collections: [
|
|
492
|
-
{
|
|
493
|
-
name: 'pages',
|
|
494
|
-
path: 'pages',
|
|
495
|
-
// No contentId — simulates static config (not loaded from filesystem)
|
|
496
|
-
entries: [
|
|
497
|
-
{
|
|
498
|
-
name: 'entry',
|
|
499
|
-
format: 'json' as const,
|
|
500
|
-
schema: [{ name: 'title', type: 'string' as const }],
|
|
501
|
-
},
|
|
502
|
-
],
|
|
503
|
-
},
|
|
504
|
-
],
|
|
505
|
-
}
|
|
506
|
-
|
|
507
|
-
const flat = flattenSchema(schema, 'content')
|
|
508
|
-
const pages = flat.find((item) => item.type === 'collection' && item.name === 'pages')
|
|
509
|
-
|
|
510
|
-
expect(pages).toBeDefined()
|
|
511
|
-
expect(pages?.type === 'collection' && pages.contentId).toBeUndefined()
|
|
512
|
-
})
|
|
513
|
-
})
|
package/src/config/flatten.ts
DELETED
|
@@ -1,174 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Schema flattening utilities for O(1) lookups.
|
|
3
|
-
*/
|
|
4
|
-
|
|
5
|
-
import { join, normalize } from 'pathe'
|
|
6
|
-
|
|
7
|
-
import type {
|
|
8
|
-
CollectionConfig,
|
|
9
|
-
EntryTypeConfig,
|
|
10
|
-
FlatSchemaItem,
|
|
11
|
-
RootCollectionConfig,
|
|
12
|
-
} from './types'
|
|
13
|
-
import { createLogicalPath } from '../paths/normalize'
|
|
14
|
-
import { ROOT_COLLECTION_ID } from '../paths/types'
|
|
15
|
-
|
|
16
|
-
/**
|
|
17
|
-
* Normalize a path value by splitting, filtering empty segments, and rejoining.
|
|
18
|
-
*/
|
|
19
|
-
export const normalizePathValue = (val: string): string =>
|
|
20
|
-
normalize(val).split('/').filter(Boolean).join('/')
|
|
21
|
-
|
|
22
|
-
/**
|
|
23
|
-
* Normalize all paths in the root collection schema.
|
|
24
|
-
* Recursively processes nested collections.
|
|
25
|
-
*/
|
|
26
|
-
export const normalizeSchemaPathsRoot = (root: RootCollectionConfig): RootCollectionConfig => {
|
|
27
|
-
const normalizeCollection = (collection: CollectionConfig, parentPath = ''): CollectionConfig => {
|
|
28
|
-
const logicalPath = parentPath ? join(parentPath, collection.path) : collection.path
|
|
29
|
-
const normalizedFull = normalizePathValue(logicalPath)
|
|
30
|
-
if (!normalizedFull || normalizedFull.includes('..')) {
|
|
31
|
-
throw new Error(`Invalid path for collection "${collection.name}"`)
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
return {
|
|
35
|
-
...collection,
|
|
36
|
-
path: normalizePathValue(collection.path),
|
|
37
|
-
collections: collection.collections?.map((c: CollectionConfig) =>
|
|
38
|
-
normalizeCollection(c, normalizedFull),
|
|
39
|
-
),
|
|
40
|
-
}
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
return {
|
|
44
|
-
...root,
|
|
45
|
-
collections: root.collections?.map((c: CollectionConfig) => normalizeCollection(c)),
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
/**
|
|
50
|
-
* Flatten the root collection schema into a flat array for O(1) lookups.
|
|
51
|
-
* Traverses the nested schema structure and returns all collections and entry types
|
|
52
|
-
* with their full paths resolved.
|
|
53
|
-
*
|
|
54
|
-
* @param root - The root collection configuration
|
|
55
|
-
* @param basePath - Optional base path to prepend (e.g., 'content')
|
|
56
|
-
* @returns Array of flattened schema items with full paths
|
|
57
|
-
*
|
|
58
|
-
* @example
|
|
59
|
-
* const flat = flattenSchema(config.schema, 'content')
|
|
60
|
-
* const map = new Map(flat.map(item => [item.logicalPath, item]))
|
|
61
|
-
* const item = map.get('content/posts') // O(1) lookup
|
|
62
|
-
*/
|
|
63
|
-
export const flattenSchema = (root: RootCollectionConfig, basePath = ''): FlatSchemaItem[] => {
|
|
64
|
-
const flat: FlatSchemaItem[] = []
|
|
65
|
-
const base = normalizePathValue(basePath || '')
|
|
66
|
-
|
|
67
|
-
const walkCollection = (collection: CollectionConfig, parentPath: string) => {
|
|
68
|
-
const normalizedPath = normalizePathValue(collection.path)
|
|
69
|
-
// Build logical path: if we have a parent, join with parent; otherwise use collection path
|
|
70
|
-
let logicalPath: string
|
|
71
|
-
if (parentPath && parentPath !== base) {
|
|
72
|
-
// Nested child collection: use only the collection name (leaf segment), not the full path
|
|
73
|
-
// The full path from collection.path includes parent path segments that are already in parentPath
|
|
74
|
-
logicalPath = join(parentPath, collection.name)
|
|
75
|
-
} else if (parentPath === base) {
|
|
76
|
-
// Root-level collection (direct child of content root): use collection path
|
|
77
|
-
logicalPath = join(base, normalizedPath)
|
|
78
|
-
} else {
|
|
79
|
-
// No parent and no base: use collection path as-is
|
|
80
|
-
logicalPath = normalizedPath
|
|
81
|
-
}
|
|
82
|
-
const normalizedFull = normalizePathValue(logicalPath)
|
|
83
|
-
|
|
84
|
-
// Add the collection itself
|
|
85
|
-
flat.push({
|
|
86
|
-
type: 'collection',
|
|
87
|
-
logicalPath: createLogicalPath(normalizedFull),
|
|
88
|
-
name: collection.name,
|
|
89
|
-
label: collection.label,
|
|
90
|
-
description: collection.description,
|
|
91
|
-
contentId: collection.contentId,
|
|
92
|
-
parentPath: parentPath ? createLogicalPath(parentPath) : undefined,
|
|
93
|
-
entries: collection.entries,
|
|
94
|
-
collections: collection.collections,
|
|
95
|
-
order: collection.order,
|
|
96
|
-
})
|
|
97
|
-
|
|
98
|
-
// Add entry types in this collection
|
|
99
|
-
if (collection.entries) {
|
|
100
|
-
for (const entryType of collection.entries as readonly EntryTypeConfig[]) {
|
|
101
|
-
// Entry type path is collection path + entry type name
|
|
102
|
-
const entryTypePath = join(normalizedFull, entryType.name)
|
|
103
|
-
flat.push({
|
|
104
|
-
type: 'entry-type',
|
|
105
|
-
logicalPath: createLogicalPath(normalizePathValue(entryTypePath)),
|
|
106
|
-
name: entryType.name,
|
|
107
|
-
label: entryType.label,
|
|
108
|
-
description: entryType.description,
|
|
109
|
-
parentPath: createLogicalPath(normalizedFull),
|
|
110
|
-
format: entryType.format,
|
|
111
|
-
schema: entryType.schema,
|
|
112
|
-
schemaRef: entryType.schemaRef,
|
|
113
|
-
default: entryType.default,
|
|
114
|
-
maxItems: entryType.maxItems,
|
|
115
|
-
})
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
// Recursively process nested collections
|
|
120
|
-
if (collection.collections) {
|
|
121
|
-
for (const child of collection.collections) {
|
|
122
|
-
walkCollection(child, normalizedFull)
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
// Add the root collection itself as a normal collection (if we have a base path)
|
|
128
|
-
// This makes content root behave exactly like any other collection, just without a parent
|
|
129
|
-
if (base) {
|
|
130
|
-
flat.push({
|
|
131
|
-
type: 'collection',
|
|
132
|
-
logicalPath: createLogicalPath(base),
|
|
133
|
-
name: base, // Use base path as the name (e.g., 'content')
|
|
134
|
-
label: undefined, // Root collection has no label
|
|
135
|
-
contentId: ROOT_COLLECTION_ID, // Sentinel — root dir has no embedded ID
|
|
136
|
-
parentPath: undefined, // No parent - this is the root
|
|
137
|
-
entries: root.entries,
|
|
138
|
-
collections: root.collections,
|
|
139
|
-
order: root.order,
|
|
140
|
-
})
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
// Add root-level entry types
|
|
144
|
-
// Now their parentPath will reference the root collection we just added above
|
|
145
|
-
if (root.entries) {
|
|
146
|
-
for (const entryType of root.entries as readonly EntryTypeConfig[]) {
|
|
147
|
-
// Root entry type path is base + entry type name
|
|
148
|
-
const entryTypePath = base ? join(base, entryType.name) : entryType.name
|
|
149
|
-
flat.push({
|
|
150
|
-
type: 'entry-type',
|
|
151
|
-
logicalPath: createLogicalPath(normalizePathValue(entryTypePath)),
|
|
152
|
-
name: entryType.name,
|
|
153
|
-
label: entryType.label,
|
|
154
|
-
description: entryType.description,
|
|
155
|
-
parentPath: base ? createLogicalPath(base) : createLogicalPath(''), // Now references the root collection (e.g., 'content')
|
|
156
|
-
format: entryType.format,
|
|
157
|
-
schema: entryType.schema,
|
|
158
|
-
schemaRef: entryType.schemaRef,
|
|
159
|
-
default: entryType.default,
|
|
160
|
-
maxItems: entryType.maxItems,
|
|
161
|
-
})
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
// Process root-level collections
|
|
166
|
-
// Pass base as parentPath so they are children of the content root collection
|
|
167
|
-
if (root.collections) {
|
|
168
|
-
for (const collection of root.collections) {
|
|
169
|
-
walkCollection(collection, base || '')
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
return flat
|
|
174
|
-
}
|