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,275 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest'
|
|
2
|
-
import { z } from 'zod'
|
|
3
|
-
import {
|
|
4
|
-
branchNameSchema,
|
|
5
|
-
logicalPathSchema,
|
|
6
|
-
contentIdSchema,
|
|
7
|
-
entrySlugSchema,
|
|
8
|
-
collectionSlugSchema,
|
|
9
|
-
permissionPathSchema,
|
|
10
|
-
} from './validators'
|
|
11
|
-
|
|
12
|
-
describe('API validators', () => {
|
|
13
|
-
describe('branchNameSchema', () => {
|
|
14
|
-
it('validates and brands valid branch names', () => {
|
|
15
|
-
const result = branchNameSchema.parse('main')
|
|
16
|
-
expect(result).toBe('main')
|
|
17
|
-
})
|
|
18
|
-
|
|
19
|
-
it('accepts branch names with slashes', () => {
|
|
20
|
-
const result = branchNameSchema.parse('feature/add-dark-mode')
|
|
21
|
-
expect(result).toBe('feature/add-dark-mode')
|
|
22
|
-
})
|
|
23
|
-
|
|
24
|
-
it('rejects empty branch names', () => {
|
|
25
|
-
expect(() => branchNameSchema.parse('')).toThrow()
|
|
26
|
-
})
|
|
27
|
-
|
|
28
|
-
it('rejects branch names with spaces', () => {
|
|
29
|
-
expect(() => branchNameSchema.parse('my branch')).toThrow('space')
|
|
30
|
-
})
|
|
31
|
-
|
|
32
|
-
it('rejects branch names with double dots', () => {
|
|
33
|
-
expect(() => branchNameSchema.parse('feature..bug')).toThrow()
|
|
34
|
-
})
|
|
35
|
-
|
|
36
|
-
it('rejects branch names starting with slash', () => {
|
|
37
|
-
expect(() => branchNameSchema.parse('/feature')).toThrow('slash')
|
|
38
|
-
})
|
|
39
|
-
|
|
40
|
-
it('rejects branch names ending with slash', () => {
|
|
41
|
-
expect(() => branchNameSchema.parse('feature/')).toThrow('slash')
|
|
42
|
-
})
|
|
43
|
-
|
|
44
|
-
it('provides clear error messages', () => {
|
|
45
|
-
try {
|
|
46
|
-
branchNameSchema.parse('my branch')
|
|
47
|
-
} catch (error) {
|
|
48
|
-
expect(error).toBeInstanceOf(z.ZodError)
|
|
49
|
-
if (error instanceof z.ZodError) {
|
|
50
|
-
expect(error.errors[0].message).toContain('space')
|
|
51
|
-
}
|
|
52
|
-
}
|
|
53
|
-
})
|
|
54
|
-
})
|
|
55
|
-
|
|
56
|
-
describe('logicalPathSchema', () => {
|
|
57
|
-
it('validates and brands valid logical paths', () => {
|
|
58
|
-
const result = logicalPathSchema.parse('content/posts')
|
|
59
|
-
expect(result).toBe('content/posts')
|
|
60
|
-
})
|
|
61
|
-
|
|
62
|
-
it('accepts nested paths', () => {
|
|
63
|
-
const result = logicalPathSchema.parse('content/posts/published')
|
|
64
|
-
expect(result).toBe('content/posts/published')
|
|
65
|
-
})
|
|
66
|
-
|
|
67
|
-
it('rejects empty paths', () => {
|
|
68
|
-
expect(() => logicalPathSchema.parse('')).toThrow()
|
|
69
|
-
})
|
|
70
|
-
|
|
71
|
-
it('rejects paths with traversal sequences', () => {
|
|
72
|
-
expect(() => logicalPathSchema.parse('content/../admin')).toThrow('traversal')
|
|
73
|
-
})
|
|
74
|
-
|
|
75
|
-
it('rejects physical paths', () => {
|
|
76
|
-
expect(() => logicalPathSchema.parse('posts.abc123def456')).toThrow('physical path')
|
|
77
|
-
})
|
|
78
|
-
|
|
79
|
-
it('provides clear error messages', () => {
|
|
80
|
-
try {
|
|
81
|
-
logicalPathSchema.parse('content/../admin')
|
|
82
|
-
} catch (error) {
|
|
83
|
-
expect(error).toBeInstanceOf(z.ZodError)
|
|
84
|
-
if (error instanceof z.ZodError) {
|
|
85
|
-
expect(error.errors[0].message).toContain('traversal')
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
})
|
|
89
|
-
})
|
|
90
|
-
|
|
91
|
-
describe('contentIdSchema', () => {
|
|
92
|
-
it('validates and brands valid content IDs', () => {
|
|
93
|
-
const result = contentIdSchema.parse('vh2WdhwAFiSL')
|
|
94
|
-
expect(result).toBe('vh2WdhwAFiSL')
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
it('accepts various Base58 IDs', () => {
|
|
98
|
-
const validIds = ['abc123def456', 'XYZabc123def', 'vh2WdhwAFiSL']
|
|
99
|
-
validIds.forEach((id) => {
|
|
100
|
-
const result = contentIdSchema.parse(id)
|
|
101
|
-
expect(result).toBe(id)
|
|
102
|
-
})
|
|
103
|
-
})
|
|
104
|
-
|
|
105
|
-
it('rejects empty IDs', () => {
|
|
106
|
-
expect(() => contentIdSchema.parse('')).toThrow('required')
|
|
107
|
-
})
|
|
108
|
-
|
|
109
|
-
it('rejects IDs with wrong length', () => {
|
|
110
|
-
expect(() => contentIdSchema.parse('abc123')).toThrow('12 Base58 characters')
|
|
111
|
-
})
|
|
112
|
-
|
|
113
|
-
it('rejects IDs with invalid characters', () => {
|
|
114
|
-
// 0, O, I, l are excluded from Base58
|
|
115
|
-
expect(() => contentIdSchema.parse('0abc23def456')).toThrow('12 Base58 characters')
|
|
116
|
-
})
|
|
117
|
-
|
|
118
|
-
it('provides clear error messages with the invalid ID', () => {
|
|
119
|
-
try {
|
|
120
|
-
contentIdSchema.parse('invalid')
|
|
121
|
-
} catch (error) {
|
|
122
|
-
expect(error).toBeInstanceOf(z.ZodError)
|
|
123
|
-
if (error instanceof z.ZodError) {
|
|
124
|
-
expect(error.errors[0].message).toContain('invalid')
|
|
125
|
-
}
|
|
126
|
-
}
|
|
127
|
-
})
|
|
128
|
-
})
|
|
129
|
-
|
|
130
|
-
describe('entrySlugSchema', () => {
|
|
131
|
-
it('validates and brands valid entry slugs', () => {
|
|
132
|
-
const result = entrySlugSchema.parse('my-first-post')
|
|
133
|
-
expect(result).toBe('my-first-post')
|
|
134
|
-
})
|
|
135
|
-
|
|
136
|
-
it('accepts slugs starting with numbers', () => {
|
|
137
|
-
const result = entrySlugSchema.parse('2023-update')
|
|
138
|
-
expect(result).toBe('2023-update')
|
|
139
|
-
})
|
|
140
|
-
|
|
141
|
-
it('rejects empty slugs', () => {
|
|
142
|
-
expect(() => entrySlugSchema.parse('')).toThrow()
|
|
143
|
-
})
|
|
144
|
-
|
|
145
|
-
it('rejects slugs with path separators', () => {
|
|
146
|
-
expect(() => entrySlugSchema.parse('posts/hello')).toThrow('separator')
|
|
147
|
-
})
|
|
148
|
-
|
|
149
|
-
it('rejects slugs with uppercase', () => {
|
|
150
|
-
expect(() => entrySlugSchema.parse('HelloWorld')).toThrow('lowercase')
|
|
151
|
-
})
|
|
152
|
-
|
|
153
|
-
it('rejects slugs not starting with alphanumeric', () => {
|
|
154
|
-
expect(() => entrySlugSchema.parse('-hello')).toThrow('start with a letter or number')
|
|
155
|
-
})
|
|
156
|
-
|
|
157
|
-
it('rejects slugs longer than 64 characters', () => {
|
|
158
|
-
const longSlug = 'a'.repeat(65)
|
|
159
|
-
expect(() => entrySlugSchema.parse(longSlug)).toThrow('too long')
|
|
160
|
-
})
|
|
161
|
-
|
|
162
|
-
it('provides clear error messages', () => {
|
|
163
|
-
try {
|
|
164
|
-
entrySlugSchema.parse('posts/hello')
|
|
165
|
-
} catch (error) {
|
|
166
|
-
expect(error).toBeInstanceOf(z.ZodError)
|
|
167
|
-
if (error instanceof z.ZodError) {
|
|
168
|
-
expect(error.errors[0].message).toContain('separator')
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
})
|
|
172
|
-
})
|
|
173
|
-
|
|
174
|
-
describe('collectionSlugSchema', () => {
|
|
175
|
-
it('validates and brands valid collection slugs', () => {
|
|
176
|
-
const result = collectionSlugSchema.parse('blog-posts')
|
|
177
|
-
expect(result).toBe('blog-posts')
|
|
178
|
-
})
|
|
179
|
-
|
|
180
|
-
it('applies same validation as entry slugs', () => {
|
|
181
|
-
// Valid
|
|
182
|
-
expect(collectionSlugSchema.parse('posts')).toBe('posts')
|
|
183
|
-
|
|
184
|
-
// Invalid
|
|
185
|
-
expect(() => collectionSlugSchema.parse('')).toThrow()
|
|
186
|
-
expect(() => collectionSlugSchema.parse('posts/items')).toThrow('separator')
|
|
187
|
-
expect(() => collectionSlugSchema.parse('Posts')).toThrow('lowercase')
|
|
188
|
-
})
|
|
189
|
-
})
|
|
190
|
-
|
|
191
|
-
describe('permissionPathSchema', () => {
|
|
192
|
-
it('validates and brands valid permission paths', () => {
|
|
193
|
-
const result = permissionPathSchema.parse('content/posts')
|
|
194
|
-
expect(result).toBe('content/posts')
|
|
195
|
-
})
|
|
196
|
-
|
|
197
|
-
it('accepts nested paths', () => {
|
|
198
|
-
const result = permissionPathSchema.parse('content/docs/guides')
|
|
199
|
-
expect(result).toBe('content/docs/guides')
|
|
200
|
-
})
|
|
201
|
-
|
|
202
|
-
it('accepts simple single-segment paths', () => {
|
|
203
|
-
const result = permissionPathSchema.parse('content')
|
|
204
|
-
expect(result).toBe('content')
|
|
205
|
-
})
|
|
206
|
-
|
|
207
|
-
it('rejects empty paths', () => {
|
|
208
|
-
expect(() => permissionPathSchema.parse('')).toThrow()
|
|
209
|
-
})
|
|
210
|
-
|
|
211
|
-
it('rejects paths with traversal sequences', () => {
|
|
212
|
-
expect(() => permissionPathSchema.parse('../etc/passwd')).toThrow('traversal')
|
|
213
|
-
})
|
|
214
|
-
|
|
215
|
-
it('rejects paths with embedded traversal', () => {
|
|
216
|
-
expect(() => permissionPathSchema.parse('content/../admin')).toThrow('traversal')
|
|
217
|
-
})
|
|
218
|
-
|
|
219
|
-
it('rejects backslash traversal bypass', () => {
|
|
220
|
-
expect(() => permissionPathSchema.parse('content\\..\\etc')).toThrow('traversal')
|
|
221
|
-
})
|
|
222
|
-
|
|
223
|
-
it('rejects paths starting with slash', () => {
|
|
224
|
-
expect(() => permissionPathSchema.parse('/content/posts')).toThrow('slash')
|
|
225
|
-
})
|
|
226
|
-
|
|
227
|
-
it('rejects paths ending with slash', () => {
|
|
228
|
-
expect(() => permissionPathSchema.parse('content/posts/')).toThrow('slash')
|
|
229
|
-
})
|
|
230
|
-
|
|
231
|
-
it('rejects paths with consecutive slashes', () => {
|
|
232
|
-
expect(() => permissionPathSchema.parse('content//posts')).toThrow('consecutive slashes')
|
|
233
|
-
})
|
|
234
|
-
})
|
|
235
|
-
|
|
236
|
-
describe('integration with z.object', () => {
|
|
237
|
-
it('works in combined schemas', () => {
|
|
238
|
-
const schema = z.object({
|
|
239
|
-
branch: branchNameSchema,
|
|
240
|
-
path: logicalPathSchema,
|
|
241
|
-
slug: entrySlugSchema,
|
|
242
|
-
})
|
|
243
|
-
|
|
244
|
-
const result = schema.parse({
|
|
245
|
-
branch: 'main',
|
|
246
|
-
path: 'content/posts',
|
|
247
|
-
slug: 'hello-world',
|
|
248
|
-
})
|
|
249
|
-
|
|
250
|
-
expect(result.branch).toBe('main')
|
|
251
|
-
expect(result.path).toBe('content/posts')
|
|
252
|
-
expect(result.slug).toBe('hello-world')
|
|
253
|
-
})
|
|
254
|
-
|
|
255
|
-
it('reports multiple validation errors', () => {
|
|
256
|
-
const schema = z.object({
|
|
257
|
-
branch: branchNameSchema,
|
|
258
|
-
path: logicalPathSchema,
|
|
259
|
-
})
|
|
260
|
-
|
|
261
|
-
try {
|
|
262
|
-
schema.parse({
|
|
263
|
-
branch: 'my branch',
|
|
264
|
-
path: 'content/../admin',
|
|
265
|
-
})
|
|
266
|
-
} catch (error) {
|
|
267
|
-
expect(error).toBeInstanceOf(z.ZodError)
|
|
268
|
-
if (error instanceof z.ZodError) {
|
|
269
|
-
// Should have errors for both fields
|
|
270
|
-
expect(error.errors.length).toBeGreaterThanOrEqual(1)
|
|
271
|
-
}
|
|
272
|
-
}
|
|
273
|
-
})
|
|
274
|
-
})
|
|
275
|
-
})
|
package/src/api/validators.ts
DELETED
|
@@ -1,176 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Zod schemas for branded type validation at API boundaries.
|
|
3
|
-
*
|
|
4
|
-
* These schemas validate incoming strings from HTTP requests
|
|
5
|
-
* and cast them to branded types for type-safe handling in API handlers.
|
|
6
|
-
*
|
|
7
|
-
* Usage:
|
|
8
|
-
* ```ts
|
|
9
|
-
* import { branchNameSchema, logicalPathSchema } from './validators'
|
|
10
|
-
*
|
|
11
|
-
* const paramsSchema = z.object({
|
|
12
|
-
* branch: branchNameSchema,
|
|
13
|
-
* path: logicalPathSchema,
|
|
14
|
-
* })
|
|
15
|
-
*
|
|
16
|
-
* // TypeScript infers branded types automatically
|
|
17
|
-
* const params = paramsSchema.parse(req.params)
|
|
18
|
-
* // params.branch is BranchName (not string)
|
|
19
|
-
* // params.path is LogicalPath (not string)
|
|
20
|
-
* ```
|
|
21
|
-
*/
|
|
22
|
-
|
|
23
|
-
import { z } from 'zod'
|
|
24
|
-
import {
|
|
25
|
-
parseBranchName,
|
|
26
|
-
parseLogicalPath,
|
|
27
|
-
parseContentId,
|
|
28
|
-
parseSlug,
|
|
29
|
-
type BranchName,
|
|
30
|
-
type LogicalPath,
|
|
31
|
-
type ContentId,
|
|
32
|
-
type EntrySlug,
|
|
33
|
-
type CollectionSlug,
|
|
34
|
-
} from '../paths'
|
|
35
|
-
import { parsePermissionPath, type PermissionPath } from '../authorization'
|
|
36
|
-
|
|
37
|
-
/**
|
|
38
|
-
* Zod schema for BranchName - validates git branch naming rules and brands.
|
|
39
|
-
*
|
|
40
|
-
* Validates:
|
|
41
|
-
* - No empty strings
|
|
42
|
-
* - No double dots (..)
|
|
43
|
-
* - No leading/trailing slashes
|
|
44
|
-
* - No spaces
|
|
45
|
-
* - No leading/trailing dots
|
|
46
|
-
* - No @{ sequences
|
|
47
|
-
*/
|
|
48
|
-
export const branchNameSchema = z
|
|
49
|
-
.string()
|
|
50
|
-
.min(1)
|
|
51
|
-
.transform((val, ctx) => {
|
|
52
|
-
const result = parseBranchName(val)
|
|
53
|
-
if (!result.ok) {
|
|
54
|
-
ctx.addIssue({
|
|
55
|
-
code: z.ZodIssueCode.custom,
|
|
56
|
-
message: result.error,
|
|
57
|
-
})
|
|
58
|
-
return z.NEVER
|
|
59
|
-
}
|
|
60
|
-
return result.name
|
|
61
|
-
}) as unknown as z.ZodType<BranchName>
|
|
62
|
-
|
|
63
|
-
/**
|
|
64
|
-
* Zod schema for LogicalPath - validates and brands logical content paths.
|
|
65
|
-
*
|
|
66
|
-
* Validates:
|
|
67
|
-
* - No empty strings
|
|
68
|
-
* - No path traversal sequences (..)
|
|
69
|
-
* - Not a physical path (no embedded content IDs)
|
|
70
|
-
*/
|
|
71
|
-
export const logicalPathSchema = z
|
|
72
|
-
.string()
|
|
73
|
-
.min(1)
|
|
74
|
-
.transform((val, ctx) => {
|
|
75
|
-
const result = parseLogicalPath(val)
|
|
76
|
-
if (!result.ok) {
|
|
77
|
-
ctx.addIssue({
|
|
78
|
-
code: z.ZodIssueCode.custom,
|
|
79
|
-
message: result.error,
|
|
80
|
-
})
|
|
81
|
-
return z.NEVER
|
|
82
|
-
}
|
|
83
|
-
return result.path
|
|
84
|
-
}) as unknown as z.ZodType<LogicalPath>
|
|
85
|
-
|
|
86
|
-
/**
|
|
87
|
-
* Zod schema for ContentId - validates 12-char Base58 IDs.
|
|
88
|
-
*
|
|
89
|
-
* Validates:
|
|
90
|
-
* - Exactly 12 characters
|
|
91
|
-
* - Base58 alphabet only (no 0, O, I, l)
|
|
92
|
-
*/
|
|
93
|
-
export const contentIdSchema = z.string().transform((val, ctx) => {
|
|
94
|
-
const result = parseContentId(val)
|
|
95
|
-
if (!result.ok) {
|
|
96
|
-
ctx.addIssue({
|
|
97
|
-
code: z.ZodIssueCode.custom,
|
|
98
|
-
message: result.error,
|
|
99
|
-
})
|
|
100
|
-
return z.NEVER
|
|
101
|
-
}
|
|
102
|
-
return result.id
|
|
103
|
-
}) as unknown as z.ZodType<ContentId>
|
|
104
|
-
|
|
105
|
-
/**
|
|
106
|
-
* Zod schema for EntrySlug - validates entry slugs.
|
|
107
|
-
*
|
|
108
|
-
* Validates:
|
|
109
|
-
* - No path separators (/ or \)
|
|
110
|
-
* - Starts with lowercase letter or number
|
|
111
|
-
* - Only lowercase letters, numbers, and hyphens
|
|
112
|
-
* - Max 64 characters
|
|
113
|
-
*/
|
|
114
|
-
export const entrySlugSchema = z
|
|
115
|
-
.string()
|
|
116
|
-
.min(1)
|
|
117
|
-
.transform((val, ctx) => {
|
|
118
|
-
const result = parseSlug(val, 'entry')
|
|
119
|
-
if (!result.ok) {
|
|
120
|
-
ctx.addIssue({
|
|
121
|
-
code: z.ZodIssueCode.custom,
|
|
122
|
-
message: result.error,
|
|
123
|
-
})
|
|
124
|
-
return z.NEVER
|
|
125
|
-
}
|
|
126
|
-
return result.slug as EntrySlug
|
|
127
|
-
}) as unknown as z.ZodType<EntrySlug>
|
|
128
|
-
|
|
129
|
-
/**
|
|
130
|
-
* Zod schema for CollectionSlug - validates collection slugs.
|
|
131
|
-
*
|
|
132
|
-
* Validates:
|
|
133
|
-
* - No path separators (/ or \)
|
|
134
|
-
* - Starts with lowercase letter or number
|
|
135
|
-
* - Only lowercase letters, numbers, and hyphens
|
|
136
|
-
* - Max 64 characters
|
|
137
|
-
*/
|
|
138
|
-
export const collectionSlugSchema = z
|
|
139
|
-
.string()
|
|
140
|
-
.min(1)
|
|
141
|
-
.transform((val, ctx) => {
|
|
142
|
-
const result = parseSlug(val, 'collection')
|
|
143
|
-
if (!result.ok) {
|
|
144
|
-
ctx.addIssue({
|
|
145
|
-
code: z.ZodIssueCode.custom,
|
|
146
|
-
message: result.error,
|
|
147
|
-
})
|
|
148
|
-
return z.NEVER
|
|
149
|
-
}
|
|
150
|
-
return result.slug as CollectionSlug
|
|
151
|
-
}) as unknown as z.ZodType<CollectionSlug>
|
|
152
|
-
|
|
153
|
-
/**
|
|
154
|
-
* Zod schema for PermissionPath - validates permission rule paths.
|
|
155
|
-
*
|
|
156
|
-
* SECURITY: Prevents path traversal attacks in permission rules.
|
|
157
|
-
*
|
|
158
|
-
* Validates:
|
|
159
|
-
* - No path traversal sequences (..)
|
|
160
|
-
* - No leading/trailing slashes
|
|
161
|
-
* - No consecutive slashes
|
|
162
|
-
*/
|
|
163
|
-
export const permissionPathSchema = z
|
|
164
|
-
.string()
|
|
165
|
-
.min(1)
|
|
166
|
-
.transform((val, ctx) => {
|
|
167
|
-
const result = parsePermissionPath(val)
|
|
168
|
-
if (!result.ok) {
|
|
169
|
-
ctx.addIssue({
|
|
170
|
-
code: z.ZodIssueCode.custom,
|
|
171
|
-
message: result.error,
|
|
172
|
-
})
|
|
173
|
-
return z.NEVER
|
|
174
|
-
}
|
|
175
|
-
return result.path
|
|
176
|
-
}) as unknown as z.ZodType<PermissionPath>
|
package/src/asset-store.test.ts
DELETED
|
@@ -1,37 +0,0 @@
|
|
|
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 } from 'vitest'
|
|
6
|
-
|
|
7
|
-
import { LocalAssetStore } from './asset-store'
|
|
8
|
-
|
|
9
|
-
const tmpDir = async () => fs.mkdtemp(path.join(os.tmpdir(), 'canopycms-assets-'))
|
|
10
|
-
|
|
11
|
-
describe('LocalAssetStore', () => {
|
|
12
|
-
it('uploads, lists, and deletes assets', async () => {
|
|
13
|
-
const root = await tmpDir()
|
|
14
|
-
const store = new LocalAssetStore({
|
|
15
|
-
root,
|
|
16
|
-
publicBaseUrl: 'https://cdn.test',
|
|
17
|
-
})
|
|
18
|
-
|
|
19
|
-
const uploaded = await store.upload('images/foo.png', Buffer.from('hello'), 'image/png')
|
|
20
|
-
expect(uploaded.key).toBe('images/foo.png')
|
|
21
|
-
expect(uploaded.url).toBe('https://cdn.test/images/foo.png')
|
|
22
|
-
|
|
23
|
-
const items = await store.list('images')
|
|
24
|
-
expect(items).toHaveLength(1)
|
|
25
|
-
expect(items[0].key).toBe('images/foo.png')
|
|
26
|
-
|
|
27
|
-
await store.delete('images/foo.png')
|
|
28
|
-
const afterDelete = await store.list('images')
|
|
29
|
-
expect(afterDelete).toHaveLength(0)
|
|
30
|
-
})
|
|
31
|
-
|
|
32
|
-
it('prevents path traversal', async () => {
|
|
33
|
-
const root = await tmpDir()
|
|
34
|
-
const store = new LocalAssetStore({ root })
|
|
35
|
-
await expect(store.upload('../evil.txt', Buffer.from('x'))).rejects.toBeInstanceOf(Error)
|
|
36
|
-
})
|
|
37
|
-
})
|
package/src/asset-store.ts
DELETED
|
@@ -1,110 +0,0 @@
|
|
|
1
|
-
import fs from 'node:fs/promises'
|
|
2
|
-
import type { Dirent } from 'node:fs'
|
|
3
|
-
import path from 'node:path'
|
|
4
|
-
|
|
5
|
-
import { isNotFoundError } from './utils/error'
|
|
6
|
-
|
|
7
|
-
export interface AssetItem {
|
|
8
|
-
key: string
|
|
9
|
-
url?: string
|
|
10
|
-
size?: number
|
|
11
|
-
contentType?: string
|
|
12
|
-
updatedAt?: string
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export interface AssetStore {
|
|
16
|
-
list(prefix?: string): Promise<AssetItem[]>
|
|
17
|
-
upload(key: string, data: Buffer | Uint8Array, contentType?: string): Promise<AssetItem>
|
|
18
|
-
delete(key: string): Promise<void>
|
|
19
|
-
getSignedUrl?(key: string, expiresInSeconds?: number): Promise<string>
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
export interface LocalAssetStoreOptions {
|
|
23
|
-
root: string
|
|
24
|
-
publicBaseUrl?: string
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
const normalizeKey = (key: string) => key.replace(/^\/+/, '')
|
|
28
|
-
|
|
29
|
-
/**
|
|
30
|
-
* Local filesystem asset store for dev and tests. In production, swap with S3 adapter.
|
|
31
|
-
*/
|
|
32
|
-
export class LocalAssetStore implements AssetStore {
|
|
33
|
-
private readonly root: string
|
|
34
|
-
private readonly publicBaseUrl?: string
|
|
35
|
-
|
|
36
|
-
constructor(options: LocalAssetStoreOptions) {
|
|
37
|
-
this.root = path.resolve(options.root)
|
|
38
|
-
this.publicBaseUrl = options.publicBaseUrl?.replace(/\/+$/, '')
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
private resolvePath(key: string): string {
|
|
42
|
-
const clean = normalizeKey(key)
|
|
43
|
-
const resolved = path.resolve(this.root, clean)
|
|
44
|
-
if (!resolved.startsWith(this.root)) {
|
|
45
|
-
throw new Error('Path traversal detected')
|
|
46
|
-
}
|
|
47
|
-
return resolved
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
private toUrl(key: string): string | undefined {
|
|
51
|
-
if (!this.publicBaseUrl) return undefined
|
|
52
|
-
return `${this.publicBaseUrl}/${normalizeKey(key)}`
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
async list(prefix = ''): Promise<AssetItem[]> {
|
|
56
|
-
const dir = this.resolvePath(prefix)
|
|
57
|
-
let entries: Dirent[]
|
|
58
|
-
try {
|
|
59
|
-
entries = await fs.readdir(dir, { withFileTypes: true })
|
|
60
|
-
} catch (err: unknown) {
|
|
61
|
-
if (isNotFoundError(err)) return []
|
|
62
|
-
throw err
|
|
63
|
-
}
|
|
64
|
-
const items: AssetItem[] = []
|
|
65
|
-
for (const entry of entries) {
|
|
66
|
-
if (entry.isDirectory()) continue
|
|
67
|
-
const key = path.join(prefix, entry.name).split(path.sep).join('/')
|
|
68
|
-
const stat = await fs.stat(path.join(dir, entry.name))
|
|
69
|
-
items.push({
|
|
70
|
-
key,
|
|
71
|
-
size: stat.size,
|
|
72
|
-
updatedAt: stat.mtime.toISOString(),
|
|
73
|
-
url: this.toUrl(key),
|
|
74
|
-
})
|
|
75
|
-
}
|
|
76
|
-
return items
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
async upload(key: string, data: Buffer | Uint8Array, contentType?: string): Promise<AssetItem> {
|
|
80
|
-
const filePath = this.resolvePath(key)
|
|
81
|
-
await fs.mkdir(path.dirname(filePath), { recursive: true })
|
|
82
|
-
await fs.writeFile(filePath, data)
|
|
83
|
-
const stat = await fs.stat(filePath)
|
|
84
|
-
return {
|
|
85
|
-
key: normalizeKey(key),
|
|
86
|
-
size: stat.size,
|
|
87
|
-
updatedAt: stat.mtime.toISOString(),
|
|
88
|
-
url: this.toUrl(key),
|
|
89
|
-
contentType,
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
async delete(key: string): Promise<void> {
|
|
94
|
-
const filePath = this.resolvePath(key)
|
|
95
|
-
try {
|
|
96
|
-
await fs.unlink(filePath)
|
|
97
|
-
} catch (err: unknown) {
|
|
98
|
-
if (isNotFoundError(err)) return
|
|
99
|
-
throw err
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
async getSignedUrl(key: string): Promise<string> {
|
|
104
|
-
const url = this.toUrl(key)
|
|
105
|
-
if (!url) {
|
|
106
|
-
throw new Error('Public base URL not configured')
|
|
107
|
-
}
|
|
108
|
-
return url
|
|
109
|
-
}
|
|
110
|
-
}
|
package/src/auth/cache.ts
DELETED
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Server-only auth cache exports.
|
|
3
|
-
* These use node:fs/promises and must NOT be imported in client bundles.
|
|
4
|
-
* Use 'canopycms/auth/cache' as the import path.
|
|
5
|
-
*/
|
|
6
|
-
export { FileBasedAuthCache, writeAuthCacheSnapshot } from './file-based-auth-cache'
|
|
7
|
-
export { CachingAuthPlugin } from './caching-auth-plugin'
|