@pilotiq/pilotiq 0.24.1 → 0.24.3
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/CHANGELOG.md +57 -0
- package/boost/guidelines.md +571 -0
- package/boost/skills/pilotiq-actions/SKILL.md +49 -0
- package/boost/skills/pilotiq-actions/rules/dispatch-modes.md +177 -0
- package/boost/skills/pilotiq-actions/rules/factories.md +130 -0
- package/boost/skills/pilotiq-actions/rules/visibility-and-authorization.md +125 -0
- package/boost/skills/pilotiq-fields/SKILL.md +47 -0
- package/boost/skills/pilotiq-fields/rules/field-catalog.md +288 -0
- package/boost/skills/pilotiq-fields/rules/reactive-fields.md +199 -0
- package/boost/skills/pilotiq-fields/rules/validation.md +198 -0
- package/boost/skills/pilotiq-relations/SKILL.md +47 -0
- package/boost/skills/pilotiq-relations/rules/relation-managers.md +256 -0
- package/boost/skills/pilotiq-relations/rules/repeater-relationship.md +177 -0
- package/boost/skills/pilotiq-resource/SKILL.md +61 -0
- package/boost/skills/pilotiq-resource/rules/authorization.md +242 -0
- package/boost/skills/pilotiq-resource/rules/defining-resources.md +228 -0
- package/boost/skills/pilotiq-resource/rules/page-overrides.md +296 -0
- package/dist/Pilotiq.d.ts +31 -0
- package/dist/Pilotiq.d.ts.map +1 -1
- package/dist/Pilotiq.js +3 -1
- package/dist/Pilotiq.js.map +1 -1
- package/dist/PilotiqRegistry.d.ts +13 -0
- package/dist/PilotiqRegistry.d.ts.map +1 -1
- package/dist/PilotiqRegistry.js +15 -0
- package/dist/PilotiqRegistry.js.map +1 -1
- package/dist/pageData/misc.d.ts.map +1 -1
- package/dist/pageData/misc.js +6 -0
- package/dist/pageData/misc.js.map +1 -1
- package/dist/pageData/navigation.d.ts +1 -0
- package/dist/pageData/navigation.d.ts.map +1 -1
- package/dist/pageData/navigation.js +3 -0
- package/dist/pageData/navigation.js.map +1 -1
- package/dist/pageData/relationPages.d.ts.map +1 -1
- package/dist/pageData/relationPages.js +3 -0
- package/dist/pageData/relationPages.js.map +1 -1
- package/dist/pageData/resourcePages.d.ts.map +1 -1
- package/dist/pageData/resourcePages.js +8 -0
- package/dist/pageData/resourcePages.js.map +1 -1
- package/dist/react/AppShell.d.ts +8 -0
- package/dist/react/AppShell.d.ts.map +1 -1
- package/dist/react/AppShell.js.map +1 -1
- package/dist/react/layouts/SidebarLayout.d.ts.map +1 -1
- package/dist/react/layouts/SidebarLayout.js +10 -2
- package/dist/react/layouts/SidebarLayout.js.map +1 -1
- package/dist/react/widgets/StatsOverviewRenderer.d.ts.map +1 -1
- package/dist/react/widgets/StatsOverviewRenderer.js +32 -18
- package/dist/react/widgets/StatsOverviewRenderer.js.map +1 -1
- package/dist/routes/relations.d.ts.map +1 -1
- package/dist/routes/relations.js +25 -18
- package/dist/routes/relations.js.map +1 -1
- package/dist/routes/resources.js.map +1 -1
- package/package.json +10 -5
- package/.turbo/turbo-build.log +0 -8
- package/CLAUDE.md +0 -265
- package/src/Cluster.test.ts +0 -283
- package/src/Cluster.ts +0 -83
- package/src/Column.test.ts +0 -199
- package/src/Column.ts +0 -710
- package/src/Global.test.ts +0 -367
- package/src/Global.ts +0 -169
- package/src/Page.test.ts +0 -114
- package/src/Page.ts +0 -208
- package/src/Pilotiq.perf.test.ts +0 -252
- package/src/Pilotiq.test.ts +0 -129
- package/src/Pilotiq.ts +0 -1158
- package/src/PilotiqRegistry.ts +0 -36
- package/src/PilotiqServiceProvider.ts +0 -121
- package/src/RelationManager.test.ts +0 -400
- package/src/RelationManager.ts +0 -527
- package/src/RenderHook.test.ts +0 -252
- package/src/RenderHook.ts +0 -242
- package/src/Resource.test.ts +0 -284
- package/src/Resource.ts +0 -526
- package/src/RightPanel.test.ts +0 -202
- package/src/RightPanel.ts +0 -132
- package/src/Tab.test.ts +0 -91
- package/src/Tab.ts +0 -156
- package/src/UserMenuItem.ts +0 -145
- package/src/actions/Action.test.ts +0 -2526
- package/src/actions/Action.ts +0 -1515
- package/src/actions/ActionGroup.test.ts +0 -112
- package/src/actions/ActionGroup.ts +0 -173
- package/src/actions/attachFactory.ts +0 -172
- package/src/actions/bulkFactories.ts +0 -168
- package/src/actions/crudFactories.ts +0 -220
- package/src/actions/exportFactory.ts +0 -225
- package/src/actions/factoryHelpers.ts +0 -177
- package/src/actions/importFactory.ts +0 -243
- package/src/actions/index.ts +0 -17
- package/src/actions/m2mFactories.ts +0 -193
- package/src/actions/relationFactories.ts +0 -372
- package/src/applyPageHooks.test.ts +0 -463
- package/src/applyPageHooks.ts +0 -330
- package/src/authorization.test.ts +0 -483
- package/src/breadcrumbs.test.ts +0 -238
- package/src/cells/coerce.test.ts +0 -85
- package/src/cells/coerce.ts +0 -84
- package/src/clusterPaths.ts +0 -35
- package/src/columns/BadgeColumn.test.ts +0 -54
- package/src/columns/BadgeColumn.ts +0 -32
- package/src/columns/BooleanColumn.test.ts +0 -41
- package/src/columns/BooleanColumn.ts +0 -18
- package/src/columns/ColorColumn.test.ts +0 -37
- package/src/columns/ColorColumn.ts +0 -38
- package/src/columns/IconColumn.test.ts +0 -54
- package/src/columns/IconColumn.ts +0 -37
- package/src/columns/ImageColumn.test.ts +0 -41
- package/src/columns/ImageColumn.ts +0 -28
- package/src/columns/SelectColumn.ts +0 -98
- package/src/columns/TextColumn.test.ts +0 -190
- package/src/columns/TextColumn.ts +0 -20
- package/src/columns/TextInputColumn.ts +0 -68
- package/src/columns/ToggleColumn.ts +0 -46
- package/src/columns/editableColumns.test.ts +0 -238
- package/src/columns/index.ts +0 -9
- package/src/defaultGlobalPages.ts +0 -95
- package/src/defaultPages.test.ts +0 -634
- package/src/defaultPages.ts +0 -617
- package/src/defaultViewPage.test.ts +0 -147
- package/src/elements/Form.test.ts +0 -223
- package/src/elements/Form.ts +0 -416
- package/src/elements/ListTabs.ts +0 -28
- package/src/elements/Table.test.ts +0 -422
- package/src/elements/Table.ts +0 -850
- package/src/elements/TableGroup.test.ts +0 -260
- package/src/elements/TableGroup.ts +0 -334
- package/src/elements/dispatchAction.test.ts +0 -463
- package/src/elements/dispatchAction.ts +0 -355
- package/src/elements/dispatchForm.test.ts +0 -477
- package/src/elements/dispatchForm.ts +0 -1993
- package/src/elements/dispatchTable.test.ts +0 -1514
- package/src/elements/dispatchTable.ts +0 -745
- package/src/elements/index.ts +0 -21
- package/src/entries/BadgeEntry.ts +0 -39
- package/src/entries/CodeEntry.test.ts +0 -40
- package/src/entries/CodeEntry.ts +0 -52
- package/src/entries/ColorEntry.ts +0 -63
- package/src/entries/ComponentEntry.test.ts +0 -173
- package/src/entries/ComponentEntry.ts +0 -95
- package/src/entries/Entry.ts +0 -304
- package/src/entries/IconEntry.ts +0 -49
- package/src/entries/ImageEntry.ts +0 -61
- package/src/entries/KeyValueEntry.ts +0 -47
- package/src/entries/RepeatableEntry.test.ts +0 -239
- package/src/entries/RepeatableEntry.ts +0 -173
- package/src/entries/TextEntry.test.ts +0 -394
- package/src/entries/TextEntry.ts +0 -60
- package/src/entries/index.ts +0 -12
- package/src/entries/leaves.test.ts +0 -306
- package/src/entries/registry.ts +0 -54
- package/src/fields/BuilderField.test.ts +0 -1188
- package/src/fields/BuilderField.ts +0 -605
- package/src/fields/BuilderRelationship.test.ts +0 -811
- package/src/fields/CheckboxField.test.ts +0 -44
- package/src/fields/CheckboxField.ts +0 -27
- package/src/fields/CheckboxListField.test.ts +0 -99
- package/src/fields/CheckboxListField.ts +0 -66
- package/src/fields/ColorPickerField.test.ts +0 -33
- package/src/fields/ColorPickerField.ts +0 -25
- package/src/fields/DateField.ts +0 -54
- package/src/fields/DateTimeField.test.ts +0 -55
- package/src/fields/EmailField.ts +0 -16
- package/src/fields/Field.test.ts +0 -654
- package/src/fields/Field.ts +0 -817
- package/src/fields/FileUploadField.test.ts +0 -143
- package/src/fields/FileUploadField.ts +0 -159
- package/src/fields/HiddenField.test.ts +0 -27
- package/src/fields/HiddenField.ts +0 -28
- package/src/fields/KeyValueField.test.ts +0 -105
- package/src/fields/KeyValueField.ts +0 -55
- package/src/fields/MarkdownField.test.ts +0 -167
- package/src/fields/MarkdownField.ts +0 -162
- package/src/fields/NumberField.ts +0 -33
- package/src/fields/RadioField.test.ts +0 -94
- package/src/fields/RadioField.ts +0 -67
- package/src/fields/RepeaterField.test.ts +0 -1806
- package/src/fields/RepeaterField.ts +0 -939
- package/src/fields/RepeaterRelationship.test.ts +0 -1923
- package/src/fields/RepeaterSimple.test.ts +0 -248
- package/src/fields/RowButton.test.ts +0 -219
- package/src/fields/RowButton.ts +0 -135
- package/src/fields/SelectField.test.ts +0 -192
- package/src/fields/SelectField.ts +0 -235
- package/src/fields/SliderField.test.ts +0 -50
- package/src/fields/SliderField.ts +0 -53
- package/src/fields/SlugField.ts +0 -24
- package/src/fields/TagsInputField.test.ts +0 -154
- package/src/fields/TagsInputField.ts +0 -133
- package/src/fields/TextField.test.ts +0 -213
- package/src/fields/TextField.ts +0 -177
- package/src/fields/TextareaField.test.ts +0 -58
- package/src/fields/TextareaField.ts +0 -59
- package/src/fields/ToggleButtonsField.test.ts +0 -106
- package/src/fields/ToggleButtonsField.ts +0 -59
- package/src/fields/ToggleField.ts +0 -16
- package/src/fields/disableOptionsWhenSelectedInSiblingRepeaterItems.test.ts +0 -319
- package/src/fields/optionsResolver.ts +0 -95
- package/src/fields/resolveField.ts +0 -28
- package/src/filters/BooleanFilter.ts +0 -35
- package/src/filters/DateRangeFilter.test.ts +0 -194
- package/src/filters/DateRangeFilter.ts +0 -148
- package/src/filters/Filter.test.ts +0 -268
- package/src/filters/Filter.ts +0 -184
- package/src/filters/FormFilter.test.ts +0 -238
- package/src/filters/FormFilter.ts +0 -215
- package/src/filters/MultiSelectFilter.test.ts +0 -119
- package/src/filters/MultiSelectFilter.ts +0 -78
- package/src/filters/QueryBuilderFilter.test.ts +0 -662
- package/src/filters/QueryBuilderFilter.ts +0 -398
- package/src/filters/SelectFilter.ts +0 -46
- package/src/filters/TernaryFilter.test.ts +0 -160
- package/src/filters/TernaryFilter.ts +0 -72
- package/src/filters/TrashedFilter.test.ts +0 -149
- package/src/filters/TrashedFilter.ts +0 -55
- package/src/filters/queryBuilder/BooleanConstraint.ts +0 -31
- package/src/filters/queryBuilder/Constraint.ts +0 -115
- package/src/filters/queryBuilder/DateConstraint.ts +0 -69
- package/src/filters/queryBuilder/NumberConstraint.ts +0 -66
- package/src/filters/queryBuilder/SelectConstraint.ts +0 -72
- package/src/filters/queryBuilder/TextConstraint.ts +0 -64
- package/src/filters/queryBuilder/index.ts +0 -12
- package/src/icons/index.ts +0 -2
- package/src/icons/lucide.ts +0 -204
- package/src/icons/registry.test.ts +0 -56
- package/src/icons/registry.ts +0 -41
- package/src/icons/types.ts +0 -47
- package/src/index.ts +0 -525
- package/src/io/csv.test.ts +0 -142
- package/src/io/csv.ts +0 -170
- package/src/nestedRelationManagerData.test.ts +0 -547
- package/src/notifications/Notification.test.ts +0 -210
- package/src/notifications/Notification.ts +0 -354
- package/src/notifications/broadcast.test.ts +0 -110
- package/src/notifications/broadcast.ts +0 -95
- package/src/notifications/database.test.ts +0 -383
- package/src/notifications/database.ts +0 -398
- package/src/notifications/databaseNotifications.test.ts +0 -187
- package/src/notifications/dispatchNotificationAction.test.ts +0 -341
- package/src/notifications/dispatchNotificationAction.ts +0 -142
- package/src/notifications/flash.test.ts +0 -89
- package/src/notifications/flash.ts +0 -71
- package/src/notifications/index.ts +0 -45
- package/src/notifications/registerBroadcastAuth.test.ts +0 -134
- package/src/notifications/registerBroadcastAuth.ts +0 -100
- package/src/notifications/resolveSavedNotification.test.ts +0 -82
- package/src/notifications/resolveSavedNotification.ts +0 -59
- package/src/notifications/types.ts +0 -93
- package/src/orm/m2mAccessor.ts +0 -66
- package/src/orm/modelDefaults.test.ts +0 -633
- package/src/orm/modelDefaults.ts +0 -666
- package/src/pageData/breadcrumbs.ts +0 -288
- package/src/pageData/forms.ts +0 -578
- package/src/pageData/helpers.ts +0 -857
- package/src/pageData/misc.ts +0 -347
- package/src/pageData/navigation.ts +0 -842
- package/src/pageData/relationPages.ts +0 -1248
- package/src/pageData/relationTabs.ts +0 -286
- package/src/pageData/resourcePages.ts +0 -609
- package/src/pageData.test.ts +0 -1545
- package/src/pageData.ts +0 -341
- package/src/plugins/index.ts +0 -8
- package/src/plugins/themeEditor.test.ts +0 -36
- package/src/plugins/themeEditor.ts +0 -45
- package/src/react/AppShell.tsx +0 -251
- package/src/react/CollabExtensionFactoryRegistry.ts +0 -55
- package/src/react/CollabRoomContext.ts +0 -98
- package/src/react/CollabTextRendererRegistry.ts +0 -102
- package/src/react/CommandPalette.tsx +0 -375
- package/src/react/CurrentUserContext.tsx +0 -50
- package/src/react/CustomPageWrapperGate.tsx +0 -69
- package/src/react/CustomPageWrapperRegistry.ts +0 -45
- package/src/react/FieldFocusReporterRegistry.ts +0 -37
- package/src/react/FieldLabelSlotRegistry.ts +0 -30
- package/src/react/FieldPresenceRegistry.ts +0 -46
- package/src/react/FormCollabBindingRegistry.ts +0 -242
- package/src/react/FormStateContext.tsx +0 -591
- package/src/react/HeadHooks.tsx +0 -126
- package/src/react/MarkdownEditorRegistry.test.ts +0 -38
- package/src/react/MarkdownEditorRegistry.ts +0 -107
- package/src/react/NotificationActionStrip.tsx +0 -263
- package/src/react/NotificationBell.tsx +0 -426
- package/src/react/PendingSuggestionApplierRegistry.test.ts +0 -97
- package/src/react/PendingSuggestionApplierRegistry.ts +0 -98
- package/src/react/PendingSuggestionOverlayRegistry.ts +0 -54
- package/src/react/PendingSuggestionsContext.tsx +0 -172
- package/src/react/RecordWrapperGate.tsx +0 -58
- package/src/react/RecordWrapperRegistry.ts +0 -39
- package/src/react/RenderHookSlot.tsx +0 -32
- package/src/react/RightSidebar.tsx +0 -257
- package/src/react/RightSidebarContext.tsx +0 -234
- package/src/react/RightSidebarTrigger.tsx +0 -53
- package/src/react/RowCoordsContext.tsx +0 -23
- package/src/react/SchemaRenderer.tsx +0 -549
- package/src/react/SearchTrigger.tsx +0 -46
- package/src/react/ThemeProvider.tsx +0 -93
- package/src/react/ThemeSettingsPage.tsx +0 -579
- package/src/react/ThemeToggle.tsx +0 -20
- package/src/react/Toaster.tsx +0 -158
- package/src/react/UserMenu.tsx +0 -196
- package/src/react/WidgetDataContext.tsx +0 -157
- package/src/react/cells/EditableCell.tsx +0 -389
- package/src/react/component-slots.test.ts +0 -103
- package/src/react/component-slots.ts +0 -116
- package/src/react/fieldJsHandler.test.ts +0 -166
- package/src/react/fieldJsHandler.ts +0 -79
- package/src/react/fields/BuilderInput.tsx +0 -1078
- package/src/react/fields/CheckboxInput.tsx +0 -39
- package/src/react/fields/CheckboxListInput.tsx +0 -102
- package/src/react/fields/ColorInput.tsx +0 -71
- package/src/react/fields/DateFieldInput.tsx +0 -70
- package/src/react/fields/DateTimeInput.tsx +0 -62
- package/src/react/fields/FieldShell.tsx +0 -348
- package/src/react/fields/FileUploadInput.tsx +0 -639
- package/src/react/fields/HiddenInput.tsx +0 -17
- package/src/react/fields/KeyValueInput.tsx +0 -230
- package/src/react/fields/MarkdownInput.tsx +0 -560
- package/src/react/fields/RadioInput.tsx +0 -81
- package/src/react/fields/RepeaterInput.test.ts +0 -116
- package/src/react/fields/RepeaterInput.tsx +0 -1420
- package/src/react/fields/SelectFieldInput.tsx +0 -280
- package/src/react/fields/SliderInput.tsx +0 -81
- package/src/react/fields/TagsInput.tsx +0 -283
- package/src/react/fields/TextLikeInput.tsx +0 -256
- package/src/react/fields/ToggleButtonsInput.tsx +0 -60
- package/src/react/fields/ToggleFieldInput.tsx +0 -56
- package/src/react/fields/relationshipRenameDispatch.test.ts +0 -106
- package/src/react/fields/relationshipRenameDispatch.ts +0 -97
- package/src/react/fields/repeaterReconcile.test.ts +0 -114
- package/src/react/fields/repeaterReconcile.ts +0 -104
- package/src/react/fields/rowChromeButton.tsx +0 -336
- package/src/react/fields/rowState.ts +0 -106
- package/src/react/fields/syncRowGates.test.ts +0 -202
- package/src/react/fields/syncRowGates.ts +0 -66
- package/src/react/fields/textInputControls.tsx +0 -238
- package/src/react/fields/useRowReorderDnd.ts +0 -78
- package/src/react/formStateHelpers.test.ts +0 -508
- package/src/react/formStateHelpers.ts +0 -381
- package/src/react/hooks/use-mobile.ts +0 -19
- package/src/react/icon-context.tsx +0 -60
- package/src/react/index.ts +0 -194
- package/src/react/layouts/SidebarLayout.tsx +0 -250
- package/src/react/layouts/TopbarLayout.tsx +0 -258
- package/src/react/navigate.tsx +0 -37
- package/src/react/onProviderSynced.test.ts +0 -90
- package/src/react/parseRecordEditUrl.test.ts +0 -122
- package/src/react/parseRecordEditUrl.ts +0 -94
- package/src/react/persistedState.ts +0 -40
- package/src/react/registry.ts +0 -48
- package/src/react/right-panel-registry.tsx +0 -47
- package/src/react/schemaRenderer/AlertRenderer.tsx +0 -112
- package/src/react/schemaRenderer/EntryRenderer.tsx +0 -501
- package/src/react/schemaRenderer/SectionRenderer.tsx +0 -120
- package/src/react/schemaRenderer/SimpleElements.tsx +0 -306
- package/src/react/schemaRenderer/TabsRenderer.tsx +0 -62
- package/src/react/schemaRenderer/WizardRenderer.tsx +0 -338
- package/src/react/schemaRenderer/action/ActionGroupTrigger.tsx +0 -177
- package/src/react/schemaRenderer/action/ActionModalDialog.tsx +0 -273
- package/src/react/schemaRenderer/action/ConfirmActionDialog.tsx +0 -61
- package/src/react/schemaRenderer/action/HandlerActionButton.tsx +0 -43
- package/src/react/schemaRenderer/action/MethodActionButton.tsx +0 -64
- package/src/react/schemaRenderer/action/buttons.tsx +0 -99
- package/src/react/schemaRenderer/action/helpers.ts +0 -140
- package/src/react/schemaRenderer/action/renderAction.tsx +0 -245
- package/src/react/schemaRenderer/columnFormat.ts +0 -65
- package/src/react/schemaRenderer/constants.ts +0 -50
- package/src/react/schemaRenderer/form/FormRenderer.tsx +0 -274
- package/src/react/schemaRenderer/form/renderField.tsx +0 -511
- package/src/react/schemaRenderer/helpers.tsx +0 -81
- package/src/react/schemaRenderer/table/CardsLayoutBody.tsx +0 -308
- package/src/react/schemaRenderer/table/TableRenderer.tsx +0 -123
- package/src/react/schemaRenderer/table/TableRendererBody.tsx +0 -974
- package/src/react/schemaRenderer/table/filters.tsx +0 -1233
- package/src/react/schemaRenderer/table/formatCell.tsx +0 -264
- package/src/react/schemaRenderer/table/links.tsx +0 -112
- package/src/react/schemaRenderer/table/renderRowActions.tsx +0 -52
- package/src/react/schemaRenderer/table/url.tsx +0 -143
- package/src/react/theme-preview/apply.ts +0 -99
- package/src/react/theme-preview/build-html.ts +0 -436
- package/src/react/ui/button.tsx +0 -51
- package/src/react/ui/calendar.tsx +0 -67
- package/src/react/ui/checkbox.tsx +0 -29
- package/src/react/ui/dialog.tsx +0 -108
- package/src/react/ui/dropdown-menu.tsx +0 -97
- package/src/react/ui/input.tsx +0 -20
- package/src/react/ui/label.tsx +0 -21
- package/src/react/ui/popover.tsx +0 -50
- package/src/react/ui/select.tsx +0 -169
- package/src/react/ui/separator.tsx +0 -25
- package/src/react/ui/sheet.tsx +0 -136
- package/src/react/ui/sidebar.tsx +0 -723
- package/src/react/ui/skeleton.tsx +0 -13
- package/src/react/ui/slider.tsx +0 -34
- package/src/react/ui/switch.tsx +0 -28
- package/src/react/ui/table.tsx +0 -105
- package/src/react/ui/tabs.tsx +0 -63
- package/src/react/ui/textarea.tsx +0 -18
- package/src/react/ui/tooltip.tsx +0 -64
- package/src/react/useResizableWidth.ts +0 -139
- package/src/react/utils.ts +0 -6
- package/src/react/widgetRegistry.test.ts +0 -43
- package/src/react/widgetRegistry.ts +0 -50
- package/src/react/widgets/StatsOverviewRenderer.tsx +0 -232
- package/src/react/widgets/TableWidgetRenderer.tsx +0 -231
- package/src/react/widgets/ViewRenderer.tsx +0 -71
- package/src/relationManagerData.test.ts +0 -1595
- package/src/richtext/index.ts +0 -8
- package/src/richtext/registry.ts +0 -89
- package/src/routes/globals.ts +0 -148
- package/src/routes/guard.test.ts +0 -325
- package/src/routes/helpers.ts +0 -704
- package/src/routes/pages.ts +0 -175
- package/src/routes/panel.ts +0 -204
- package/src/routes/relations.ts +0 -1243
- package/src/routes/resources.ts +0 -781
- package/src/routes/theme.ts +0 -91
- package/src/routes-nested-relations.test.ts +0 -676
- package/src/routes-relations.test.ts +0 -972
- package/src/routes.test.ts +0 -2027
- package/src/routes.ts +0 -303
- package/src/schema/Alert.test.ts +0 -109
- package/src/schema/Alert.ts +0 -131
- package/src/schema/Block.ts +0 -169
- package/src/schema/Breadcrumbs.ts +0 -40
- package/src/schema/Card.ts +0 -35
- package/src/schema/Divider.ts +0 -20
- package/src/schema/Element.ts +0 -219
- package/src/schema/EmptyState.test.ts +0 -37
- package/src/schema/EmptyState.ts +0 -63
- package/src/schema/Fieldset.ts +0 -43
- package/src/schema/Grid.ts +0 -43
- package/src/schema/Group.ts +0 -30
- package/src/schema/Heading.ts +0 -39
- package/src/schema/Html.ts +0 -67
- package/src/schema/Icon.ts +0 -54
- package/src/schema/Image.ts +0 -57
- package/src/schema/LinkTag.ts +0 -41
- package/src/schema/Markdown.ts +0 -85
- package/src/schema/MetaTag.ts +0 -41
- package/src/schema/RelationTabs.ts +0 -71
- package/src/schema/ScriptTag.ts +0 -55
- package/src/schema/Section.ts +0 -160
- package/src/schema/ServerDataElement.test.ts +0 -140
- package/src/schema/ServerDataElement.ts +0 -156
- package/src/schema/SlotComponent.test.ts +0 -77
- package/src/schema/SlotComponent.ts +0 -71
- package/src/schema/Split.ts +0 -50
- package/src/schema/Stat.test.ts +0 -118
- package/src/schema/Stat.ts +0 -154
- package/src/schema/StatsOverview.test.ts +0 -141
- package/src/schema/StatsOverview.ts +0 -119
- package/src/schema/StyleTag.ts +0 -35
- package/src/schema/TableWidget.test.ts +0 -297
- package/src/schema/TableWidget.ts +0 -289
- package/src/schema/Tabs.ts +0 -79
- package/src/schema/Text.ts +0 -58
- package/src/schema/UnorderedList.ts +0 -49
- package/src/schema/View.test.ts +0 -111
- package/src/schema/View.ts +0 -127
- package/src/schema/Wizard.ts +0 -220
- package/src/schema/containers.test.ts +0 -564
- package/src/schema/headTags.test.ts +0 -134
- package/src/schema/index.ts +0 -40
- package/src/schema/primes.test.ts +0 -269
- package/src/schema/resolveSchema.test.ts +0 -379
- package/src/schema/resolveSchema.ts +0 -917
- package/src/schema/sanitize.ts +0 -58
- package/src/search.test.ts +0 -446
- package/src/search.ts +0 -178
- package/src/sessionFilters.test.ts +0 -375
- package/src/sessionFilters.ts +0 -143
- package/src/slot-components/index.ts +0 -10
- package/src/slot-components/registry.ts +0 -56
- package/src/styles/file-upload.css +0 -13
- package/src/summarizers/Summarizer.test.ts +0 -84
- package/src/summarizers/Summarizer.ts +0 -123
- package/src/summarizers/index.ts +0 -11
- package/src/theme/base-colors.ts +0 -68
- package/src/theme/chart-colors.ts +0 -50
- package/src/theme/colors.ts +0 -447
- package/src/theme/generate-css.test.ts +0 -139
- package/src/theme/generate-css.ts +0 -44
- package/src/theme/generate-scale.test.ts +0 -106
- package/src/theme/generate-scale.ts +0 -97
- package/src/theme/icon-map.ts +0 -42
- package/src/theme/index.ts +0 -34
- package/src/theme/migrate.test.ts +0 -178
- package/src/theme/migrate.ts +0 -81
- package/src/theme/presets.ts +0 -135
- package/src/theme/radius.ts +0 -18
- package/src/theme/resolve.test.ts +0 -238
- package/src/theme/resolve.ts +0 -96
- package/src/theme/spacing.ts +0 -18
- package/src/theme/storage.test.ts +0 -126
- package/src/theme/storage.ts +0 -106
- package/src/theme/theme-colors.ts +0 -88
- package/src/theme/types.ts +0 -125
- package/src/uploads/UploadAdapter.ts +0 -35
- package/src/uploads/index.ts +0 -2
- package/src/uploads/localUpload.test.ts +0 -70
- package/src/uploads/localUpload.ts +0 -84
- package/src/validation/Validator.ts +0 -49
- package/src/validation/index.ts +0 -28
- package/src/validation/rules.ts +0 -78
- package/src/validation/runValidators.ts +0 -435
- package/src/validation/uniqueValidator.test.ts +0 -196
- package/src/validation/uniqueValidator.ts +0 -133
- package/src/validation/validators.test.ts +0 -268
- package/src/vite.test.ts +0 -184
- package/src/vite.ts +0 -787
- package/src/widgets/index.ts +0 -10
- package/src/widgets/registry.ts +0 -45
- package/src/widgets.test.ts +0 -592
- package/tsconfig.build.json +0 -11
- package/tsconfig.json +0 -4
- package/tsconfig.test.json +0 -10
- package/views/react/Dashboard.tsx +0 -27
- package/views/react/Resources/Form.tsx +0 -102
- package/views/react/Resources/Index.tsx +0 -49
|
@@ -1,196 +0,0 @@
|
|
|
1
|
-
import { describe, it } from 'node:test'
|
|
2
|
-
import assert from 'node:assert/strict'
|
|
3
|
-
|
|
4
|
-
import { unique } from './uniqueValidator.js'
|
|
5
|
-
import type { ModelLike, ModelQuery } from '../orm/modelDefaults.js'
|
|
6
|
-
|
|
7
|
-
// ── Fake ModelLike — records every where() call so tests can assert ──
|
|
8
|
-
|
|
9
|
-
interface FakeQueryOp { op: string; args: unknown[] }
|
|
10
|
-
|
|
11
|
-
class FakeQuery implements ModelQuery {
|
|
12
|
-
ops: FakeQueryOp[] = []
|
|
13
|
-
constructor(private readonly _data: unknown[]) {}
|
|
14
|
-
|
|
15
|
-
where(...args: unknown[]): ModelQuery { this.ops.push({ op: 'where', args }); return this }
|
|
16
|
-
orWhere(...args: unknown[]): ModelQuery { this.ops.push({ op: 'orWhere', args }); return this }
|
|
17
|
-
orderBy(column: string, direction: 'ASC' | 'DESC' = 'ASC'): ModelQuery {
|
|
18
|
-
this.ops.push({ op: 'orderBy', args: [column, direction] }); return this
|
|
19
|
-
}
|
|
20
|
-
async paginate(page: number, perPage?: number) {
|
|
21
|
-
this.ops.push({ op: 'paginate', args: [page, perPage] })
|
|
22
|
-
return { data: this._data, total: this._data.length }
|
|
23
|
-
}
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
function makeModel(opts: { rows?: unknown[]; primaryKey?: string } = {}) {
|
|
27
|
-
let lastQuery: FakeQuery | null = null
|
|
28
|
-
const model = {
|
|
29
|
-
primaryKey: opts.primaryKey,
|
|
30
|
-
async find() { return null },
|
|
31
|
-
async create(data: Record<string, unknown>) { return { id: 'new', ...data } },
|
|
32
|
-
async update(id: string | number, data: Record<string, unknown>) { return { id, ...data } },
|
|
33
|
-
async delete() {},
|
|
34
|
-
query(): ModelQuery {
|
|
35
|
-
const q = new FakeQuery(opts.rows ?? [])
|
|
36
|
-
lastQuery = q
|
|
37
|
-
return q
|
|
38
|
-
},
|
|
39
|
-
get lastQuery() { return lastQuery },
|
|
40
|
-
} as unknown as ModelLike & { lastQuery: FakeQuery | null }
|
|
41
|
-
return model
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
describe('unique() validator', () => {
|
|
45
|
-
it('passes when value is empty (defers to required())', async () => {
|
|
46
|
-
const model = makeModel({ rows: [{ id: 1, email: 'a@b.com' }] })
|
|
47
|
-
const v = unique({ model, column: 'email' })
|
|
48
|
-
assert.equal(await v(''), null)
|
|
49
|
-
assert.equal(await v(null), null)
|
|
50
|
-
assert.equal(await v(undefined), null)
|
|
51
|
-
})
|
|
52
|
-
|
|
53
|
-
it('passes when no row matches', async () => {
|
|
54
|
-
const model = makeModel({ rows: [] })
|
|
55
|
-
const v = unique({ model, column: 'email' })
|
|
56
|
-
assert.equal(await v('new@b.com', { values: { email: 'new@b.com' } }), null)
|
|
57
|
-
})
|
|
58
|
-
|
|
59
|
-
it('rejects when a row matches and there is no current record (create)', async () => {
|
|
60
|
-
const model = makeModel({ rows: [{ id: 1, email: 'a@b.com' }] })
|
|
61
|
-
const v = unique({ model, column: 'email' })
|
|
62
|
-
assert.equal(await v('a@b.com', { values: { email: 'a@b.com' } }), 'Already taken')
|
|
63
|
-
})
|
|
64
|
-
|
|
65
|
-
it('passes on edit when the only matching row is the record under edit', async () => {
|
|
66
|
-
const model = makeModel({ rows: [{ id: 7, email: 'a@b.com' }] })
|
|
67
|
-
const v = unique({ model, column: 'email' })
|
|
68
|
-
const ctx = { values: { email: 'a@b.com' }, record: { id: 7, email: 'a@b.com' } }
|
|
69
|
-
assert.equal(await v('a@b.com', ctx), null)
|
|
70
|
-
})
|
|
71
|
-
|
|
72
|
-
it('rejects on edit when matching row is a DIFFERENT record', async () => {
|
|
73
|
-
const model = makeModel({ rows: [{ id: 99, email: 'a@b.com' }] })
|
|
74
|
-
const v = unique({ model, column: 'email' })
|
|
75
|
-
const ctx = { values: { email: 'a@b.com' }, record: { id: 7, email: 'old@b.com' } }
|
|
76
|
-
assert.equal(await v('a@b.com', ctx), 'Already taken')
|
|
77
|
-
})
|
|
78
|
-
|
|
79
|
-
it('honors primaryKey override on the model', async () => {
|
|
80
|
-
const model = makeModel({ rows: [{ uuid: 'abc', email: 'a@b.com' }], primaryKey: 'uuid' })
|
|
81
|
-
const v = unique({ model, column: 'email' })
|
|
82
|
-
const ctx = { values: { email: 'a@b.com' }, record: { uuid: 'abc' } }
|
|
83
|
-
assert.equal(await v('a@b.com', ctx), null)
|
|
84
|
-
})
|
|
85
|
-
|
|
86
|
-
it('ignoreRecord:false rejects even when the match is the current record', async () => {
|
|
87
|
-
const model = makeModel({ rows: [{ id: 7, email: 'a@b.com' }] })
|
|
88
|
-
const v = unique({ model, column: 'email', ignoreRecord: false })
|
|
89
|
-
const ctx = { values: { email: 'a@b.com' }, record: { id: 7 } }
|
|
90
|
-
assert.equal(await v('a@b.com', ctx), 'Already taken')
|
|
91
|
-
})
|
|
92
|
-
|
|
93
|
-
it('emits a custom message when configured', async () => {
|
|
94
|
-
const model = makeModel({ rows: [{ id: 1, email: 'a@b.com' }] })
|
|
95
|
-
const v = unique({ model, column: 'email', message: 'That email is taken' })
|
|
96
|
-
assert.equal(await v('a@b.com', { values: { email: 'a@b.com' } }), 'That email is taken')
|
|
97
|
-
})
|
|
98
|
-
|
|
99
|
-
it('caseInsensitive uses LIKE with escaped wildcards', async () => {
|
|
100
|
-
const model = makeModel({ rows: [] })
|
|
101
|
-
const v = unique({ model, column: 'name', caseInsensitive: true })
|
|
102
|
-
await v('100% solid_steel\\bar', { values: { name: '100% solid_steel\\bar' } })
|
|
103
|
-
const where = model.lastQuery!.ops.find(o => o.op === 'where')
|
|
104
|
-
assert.deepEqual(where!.args, ['name', 'LIKE', '100\\% solid\\_steel\\\\bar'])
|
|
105
|
-
})
|
|
106
|
-
|
|
107
|
-
it('caseInsensitive uses plain where for non-strings', async () => {
|
|
108
|
-
const model = makeModel({ rows: [] })
|
|
109
|
-
const v = unique({ model, column: 'count', caseInsensitive: true })
|
|
110
|
-
await v(42, { values: { count: 42 } })
|
|
111
|
-
const where = model.lastQuery!.ops.find(o => o.op === 'where')
|
|
112
|
-
assert.deepEqual(where!.args, ['count', 42])
|
|
113
|
-
})
|
|
114
|
-
|
|
115
|
-
it('applies extra where() clauses for scoped uniqueness', async () => {
|
|
116
|
-
const model = makeModel({ rows: [] })
|
|
117
|
-
const v = unique({
|
|
118
|
-
model, column: 'name',
|
|
119
|
-
where: ({ values }) => ({ tenantId: values?.tenantId }),
|
|
120
|
-
})
|
|
121
|
-
await v('Alpha', { values: { name: 'Alpha', tenantId: 't-1' } })
|
|
122
|
-
const wheres = model.lastQuery!.ops.filter(o => o.op === 'where')
|
|
123
|
-
assert.equal(wheres.length, 2)
|
|
124
|
-
assert.deepEqual(wheres[0]!.args, ['name', 'Alpha'])
|
|
125
|
-
assert.deepEqual(wheres[1]!.args, ['tenantId', 't-1'])
|
|
126
|
-
})
|
|
127
|
-
|
|
128
|
-
it('skips where() entries with undefined value', async () => {
|
|
129
|
-
const model = makeModel({ rows: [] })
|
|
130
|
-
const v = unique({
|
|
131
|
-
model, column: 'name',
|
|
132
|
-
where: () => ({ tenantId: undefined, status: 'active' }),
|
|
133
|
-
})
|
|
134
|
-
await v('Alpha', { values: { name: 'Alpha' } })
|
|
135
|
-
const wheres = model.lastQuery!.ops.filter(o => o.op === 'where')
|
|
136
|
-
assert.equal(wheres.length, 2) // primary + status, NOT tenantId
|
|
137
|
-
assert.deepEqual(wheres[1]!.args, ['status', 'active'])
|
|
138
|
-
})
|
|
139
|
-
|
|
140
|
-
it('falls back to ctx.values key when column option is omitted', async () => {
|
|
141
|
-
// The validator doesn't know its owning Field's name, so without
|
|
142
|
-
// `column` it scans ctx.values for a key whose value matches `value`.
|
|
143
|
-
const model = makeModel({ rows: [{ id: 1, slug: 'taken' }] })
|
|
144
|
-
const v = unique({ model })
|
|
145
|
-
assert.equal(await v('taken', { values: { slug: 'taken' } }), 'Already taken')
|
|
146
|
-
const where = model.lastQuery!.ops.find(o => o.op === 'where')
|
|
147
|
-
assert.deepEqual(where!.args, ['slug', 'taken'])
|
|
148
|
-
})
|
|
149
|
-
|
|
150
|
-
it('paginates with (1, 2) so we can distinguish own-record from real conflict', async () => {
|
|
151
|
-
const model = makeModel({ rows: [] })
|
|
152
|
-
const v = unique({ model, column: 'email' })
|
|
153
|
-
await v('x@y.com', { values: { email: 'x@y.com' } })
|
|
154
|
-
const page = model.lastQuery!.ops.find(o => o.op === 'paginate')
|
|
155
|
-
assert.deepEqual(page!.args, [1, 2])
|
|
156
|
-
})
|
|
157
|
-
|
|
158
|
-
it('rejects on edit when 2+ rows match, even if one is the current record', async () => {
|
|
159
|
-
// Defensive: shouldn't happen with a real unique index, but if it
|
|
160
|
-
// does, "every match is me" is false → conflict.
|
|
161
|
-
const model = makeModel({
|
|
162
|
-
rows: [{ id: 7, email: 'a@b.com' }, { id: 99, email: 'a@b.com' }],
|
|
163
|
-
})
|
|
164
|
-
const v = unique({ model, column: 'email' })
|
|
165
|
-
const ctx = { values: { email: 'a@b.com' }, record: { id: 7 } }
|
|
166
|
-
assert.equal(await v('a@b.com', ctx), 'Already taken')
|
|
167
|
-
})
|
|
168
|
-
|
|
169
|
-
it('thrown errors propagate (no silent fail-closed)', async () => {
|
|
170
|
-
const model = {
|
|
171
|
-
async find() { return null },
|
|
172
|
-
async create() { return null },
|
|
173
|
-
async update() { return null },
|
|
174
|
-
async delete() {},
|
|
175
|
-
query(): ModelQuery {
|
|
176
|
-
return {
|
|
177
|
-
where() { return this },
|
|
178
|
-
orWhere() { return this },
|
|
179
|
-
orderBy() { return this },
|
|
180
|
-
async paginate() { throw new Error('db down') },
|
|
181
|
-
} as unknown as ModelQuery
|
|
182
|
-
},
|
|
183
|
-
} as ModelLike
|
|
184
|
-
const v = unique({ model, column: 'email' })
|
|
185
|
-
await assert.rejects(
|
|
186
|
-
async () => { await v('a@b.com', { values: { email: 'a@b.com' } }) },
|
|
187
|
-
/db down/,
|
|
188
|
-
)
|
|
189
|
-
})
|
|
190
|
-
|
|
191
|
-
it('carries no serialized descriptor (skips client mirroring)', () => {
|
|
192
|
-
const model = makeModel({ rows: [] })
|
|
193
|
-
const v = unique({ model, column: 'email' })
|
|
194
|
-
assert.equal(v.serialized, undefined)
|
|
195
|
-
})
|
|
196
|
-
})
|
|
@@ -1,133 +0,0 @@
|
|
|
1
|
-
import { makeValidator, type Validator, type ValidatorContext } from './Validator.js'
|
|
2
|
-
import { getPrimaryKey, type ModelLike } from '../orm/modelDefaults.js'
|
|
3
|
-
|
|
4
|
-
export interface UniqueOptions {
|
|
5
|
-
/**
|
|
6
|
-
* The model to probe. Any `ModelLike` works — `query().where(col, value)
|
|
7
|
-
* .paginate(1, 2)` is the only contract used, so no new ORM method is
|
|
8
|
-
* required. A class extending `@rudderjs/orm`'s `Model` satisfies this
|
|
9
|
-
* automatically.
|
|
10
|
-
*/
|
|
11
|
-
model: ModelLike
|
|
12
|
-
|
|
13
|
-
/**
|
|
14
|
-
* Column to match the field's value against. Defaults to the field's own
|
|
15
|
-
* name — sufficient for the common `email` / `slug` / `username` case.
|
|
16
|
-
*/
|
|
17
|
-
column?: string
|
|
18
|
-
|
|
19
|
-
/**
|
|
20
|
-
* Whether to exclude the record under edit from the probe. Defaults to
|
|
21
|
-
* `true` — on edit, the validator skips when the only matching row IS
|
|
22
|
-
* the current record (so saving "no change" doesn't trigger a conflict).
|
|
23
|
-
* Pass `false` to forbid all matches even when editing the same row.
|
|
24
|
-
*
|
|
25
|
-
* Has no effect on create — there's no `ctx.record` to ignore.
|
|
26
|
-
*/
|
|
27
|
-
ignoreRecord?: boolean
|
|
28
|
-
|
|
29
|
-
/**
|
|
30
|
-
* Extra equality clauses for scoped uniqueness. Receives the validator
|
|
31
|
-
* context (`{ values, record }`) so the scope can read sibling fields:
|
|
32
|
-
*
|
|
33
|
-
* unique({ model: Project, where: ({ values }) => ({ tenantId: values.tenantId }) })
|
|
34
|
-
*
|
|
35
|
-
* Each entry is applied as `query().where(key, value)`. Skipped entirely
|
|
36
|
-
* when the function returns `undefined` or an empty object.
|
|
37
|
-
*/
|
|
38
|
-
where?: (ctx: ValidatorContext) => Record<string, unknown> | undefined
|
|
39
|
-
|
|
40
|
-
/**
|
|
41
|
-
* Treat string comparisons as case-insensitive. Implemented via SQL
|
|
42
|
-
* `LIKE` on the value with `%` / `_` / `\` escaped — so it stays an
|
|
43
|
-
* exact match, just collation-folded. Works on SQLite (default `NOCASE`
|
|
44
|
-
* for ASCII) and MySQL (default collation), and is collation-dependent
|
|
45
|
-
* on Postgres. For Postgres apps that need locale-aware folding, use a
|
|
46
|
-
* `citext` column or a custom `where(fn)` against a generated lower-cased
|
|
47
|
-
* column instead of this flag.
|
|
48
|
-
*/
|
|
49
|
-
caseInsensitive?: boolean
|
|
50
|
-
|
|
51
|
-
/** Override the default error message. */
|
|
52
|
-
message?: string
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
/**
|
|
56
|
-
* Async validator: rejects when another row in the table already has the
|
|
57
|
-
* same value in `column` (subject to `where` / `ignoreRecord`). Empty
|
|
58
|
-
* values pass — pair with `required()` if the field is mandatory.
|
|
59
|
-
*
|
|
60
|
-
* The probe is `M.query().where(column, value).paginate(1, 2)` — pulls at
|
|
61
|
-
* most 2 rows so we can distinguish "no match" / "only own record" /
|
|
62
|
-
* "real conflict" without a `count(*)` contract.
|
|
63
|
-
*
|
|
64
|
-
* Carries no `serialized` descriptor, so the client doesn't try to mirror
|
|
65
|
-
* the rule for live UX. The roundtrip happens once on submit.
|
|
66
|
-
*/
|
|
67
|
-
export function unique(opts: UniqueOptions): Validator {
|
|
68
|
-
const message = opts.message ?? 'Already taken'
|
|
69
|
-
const ignoreRecord = opts.ignoreRecord ?? true
|
|
70
|
-
|
|
71
|
-
return makeValidator(async (value, ctx) => {
|
|
72
|
-
if (value === undefined || value === null || value === '') return null
|
|
73
|
-
|
|
74
|
-
// Resolve column lazily: validators don't know their owning Field's name,
|
|
75
|
-
// so we accept that the caller either passes `column` explicitly or we
|
|
76
|
-
// fall back to whichever key in `ctx.values` matches `value`. The
|
|
77
|
-
// explicit `column` path is the recommended one — the fallback is a
|
|
78
|
-
// best-effort convenience for the trivial single-field case.
|
|
79
|
-
const column = opts.column ?? findColumnForValue(ctx, value)
|
|
80
|
-
if (!column) return null
|
|
81
|
-
|
|
82
|
-
let q = opts.model.query()
|
|
83
|
-
|
|
84
|
-
if (opts.caseInsensitive && typeof value === 'string') {
|
|
85
|
-
q = q.where(column, 'LIKE', escapeLikeLiteral(value))
|
|
86
|
-
} else {
|
|
87
|
-
q = q.where(column, value)
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
const extra = opts.where?.(ctx ?? {})
|
|
91
|
-
if (extra) {
|
|
92
|
-
for (const [key, val] of Object.entries(extra)) {
|
|
93
|
-
if (val === undefined) continue
|
|
94
|
-
q = q.where(key, val)
|
|
95
|
-
}
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
const { data } = await q.paginate(1, 2)
|
|
99
|
-
if (data.length === 0) return null
|
|
100
|
-
|
|
101
|
-
if (ignoreRecord && ctx?.record) {
|
|
102
|
-
const pk = getPrimaryKey(opts.model)
|
|
103
|
-
const ownPk = (ctx.record as Record<string, unknown>)[pk]
|
|
104
|
-
if (ownPk !== undefined && ownPk !== null) {
|
|
105
|
-
const onlyOwn = data.every(row => {
|
|
106
|
-
const rowPk = (row as Record<string, unknown>)[pk]
|
|
107
|
-
return rowPk === ownPk
|
|
108
|
-
})
|
|
109
|
-
if (onlyOwn) return null
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
return message
|
|
114
|
-
})
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
function findColumnForValue(ctx: ValidatorContext | undefined, value: unknown): string | undefined {
|
|
118
|
-
const values = ctx?.values
|
|
119
|
-
if (!values) return undefined
|
|
120
|
-
for (const [k, v] of Object.entries(values)) {
|
|
121
|
-
if (v === value) return k
|
|
122
|
-
}
|
|
123
|
-
return undefined
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
/**
|
|
127
|
-
* Escape SQL `LIKE`'s wildcard chars so the comparison stays an exact
|
|
128
|
-
* match. `%` matches any sequence, `_` any single char, `\` is the
|
|
129
|
-
* escape char itself (with the default LIKE escape).
|
|
130
|
-
*/
|
|
131
|
-
function escapeLikeLiteral(s: string): string {
|
|
132
|
-
return s.replace(/[%_\\]/g, '\\$&')
|
|
133
|
-
}
|
|
@@ -1,268 +0,0 @@
|
|
|
1
|
-
import { describe, it } from 'node:test'
|
|
2
|
-
import assert from 'node:assert/strict'
|
|
3
|
-
|
|
4
|
-
import { TextField } from '../fields/TextField.js'
|
|
5
|
-
import { EmailField } from '../fields/EmailField.js'
|
|
6
|
-
import { NumberField } from '../fields/NumberField.js'
|
|
7
|
-
import { Section } from '../schema/Section.js'
|
|
8
|
-
import { Card } from '../schema/Card.js'
|
|
9
|
-
import { Tabs, Tab } from '../schema/Tabs.js'
|
|
10
|
-
|
|
11
|
-
import { makeValidator } from './Validator.js'
|
|
12
|
-
import { required, email, minLength, maxLength, min, max, pattern } from './rules.js'
|
|
13
|
-
import { validateSchema, isValid } from './runValidators.js'
|
|
14
|
-
|
|
15
|
-
describe('built-in validators', () => {
|
|
16
|
-
describe('required', () => {
|
|
17
|
-
it('rejects empty string, null, and undefined', () => {
|
|
18
|
-
const v = required()
|
|
19
|
-
assert.equal(v(''), 'This field is required')
|
|
20
|
-
assert.equal(v(null), 'This field is required')
|
|
21
|
-
assert.equal(v(undefined), 'This field is required')
|
|
22
|
-
})
|
|
23
|
-
|
|
24
|
-
it('accepts non-empty values (including 0 and false)', () => {
|
|
25
|
-
const v = required()
|
|
26
|
-
assert.equal(v('hi'), null)
|
|
27
|
-
assert.equal(v(0), null)
|
|
28
|
-
assert.equal(v(false), null)
|
|
29
|
-
})
|
|
30
|
-
|
|
31
|
-
it('honors a custom message', () => {
|
|
32
|
-
assert.equal(required('Required!')(''), 'Required!')
|
|
33
|
-
})
|
|
34
|
-
|
|
35
|
-
it('serializes to { rule: "required", message }', () => {
|
|
36
|
-
assert.deepEqual(required().serialized, { rule: 'required', message: 'This field is required' })
|
|
37
|
-
})
|
|
38
|
-
})
|
|
39
|
-
|
|
40
|
-
describe('email', () => {
|
|
41
|
-
it('passes empty values (combine with required for "must have an email")', () => {
|
|
42
|
-
assert.equal(email()(''), null)
|
|
43
|
-
assert.equal(email()(undefined), null)
|
|
44
|
-
})
|
|
45
|
-
|
|
46
|
-
it('rejects malformed emails', () => {
|
|
47
|
-
assert.equal(email()('not-an-email'), 'Must be a valid email')
|
|
48
|
-
assert.equal(email()('a@b'), 'Must be a valid email')
|
|
49
|
-
assert.equal(email()('a@.com'), 'Must be a valid email')
|
|
50
|
-
})
|
|
51
|
-
|
|
52
|
-
it('accepts well-formed emails', () => {
|
|
53
|
-
assert.equal(email()('user@example.com'), null)
|
|
54
|
-
})
|
|
55
|
-
})
|
|
56
|
-
|
|
57
|
-
describe('minLength / maxLength', () => {
|
|
58
|
-
it('minLength rejects shorter strings, accepts equal/longer', () => {
|
|
59
|
-
const v = minLength(3)
|
|
60
|
-
assert.equal(v('ab'), 'Must be at least 3 characters')
|
|
61
|
-
assert.equal(v('abc'), null)
|
|
62
|
-
assert.equal(v('abcd'),null)
|
|
63
|
-
})
|
|
64
|
-
|
|
65
|
-
it('maxLength rejects longer strings', () => {
|
|
66
|
-
const v = maxLength(3)
|
|
67
|
-
assert.equal(v('abcd'), 'Must be at most 3 characters')
|
|
68
|
-
assert.equal(v('abc'), null)
|
|
69
|
-
})
|
|
70
|
-
|
|
71
|
-
it('skips non-strings and empty values', () => {
|
|
72
|
-
assert.equal(minLength(3)(undefined), null)
|
|
73
|
-
assert.equal(minLength(3)(42), null)
|
|
74
|
-
})
|
|
75
|
-
})
|
|
76
|
-
|
|
77
|
-
describe('min / max', () => {
|
|
78
|
-
it('min/max bound numbers', () => {
|
|
79
|
-
assert.equal(min(0)(-1), 'Must be at least 0')
|
|
80
|
-
assert.equal(min(0)(0), null)
|
|
81
|
-
assert.equal(max(10)(11), 'Must be at most 10')
|
|
82
|
-
assert.equal(max(10)(10), null)
|
|
83
|
-
})
|
|
84
|
-
|
|
85
|
-
it('skips non-numbers and empty', () => {
|
|
86
|
-
assert.equal(min(0)('hi'), null)
|
|
87
|
-
assert.equal(min(0)(undefined), null)
|
|
88
|
-
})
|
|
89
|
-
})
|
|
90
|
-
|
|
91
|
-
describe('pattern', () => {
|
|
92
|
-
it('rejects non-matching strings', () => {
|
|
93
|
-
assert.equal(pattern(/^[a-z]+$/)('Abc'), 'Invalid format')
|
|
94
|
-
assert.equal(pattern(/^[a-z]+$/)('abc'), null)
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
it('serializes source + flags', () => {
|
|
98
|
-
const re = /^foo$/i
|
|
99
|
-
assert.deepEqual(pattern(re).serialized, {
|
|
100
|
-
rule: 'pattern', source: '^foo$', flags: 'i', message: 'Invalid format',
|
|
101
|
-
})
|
|
102
|
-
})
|
|
103
|
-
})
|
|
104
|
-
|
|
105
|
-
describe('makeValidator', () => {
|
|
106
|
-
it('produces a callable validator without serialized when none provided', () => {
|
|
107
|
-
const v = makeValidator(val => (val === 'bad' ? 'no' : null))
|
|
108
|
-
assert.equal(v('bad'), 'no')
|
|
109
|
-
assert.equal(v('ok'), null)
|
|
110
|
-
assert.equal(v.serialized, undefined)
|
|
111
|
-
})
|
|
112
|
-
})
|
|
113
|
-
})
|
|
114
|
-
|
|
115
|
-
describe('Field.validate / runValidators', () => {
|
|
116
|
-
it('accumulates validators across calls', async () => {
|
|
117
|
-
const f = TextField.make('x').validate(minLength(3)).validate(maxLength(5))
|
|
118
|
-
assert.equal((await f.runValidators('a')).length, 1) // minLength
|
|
119
|
-
assert.equal((await f.runValidators('abcdef')).length, 1) // maxLength
|
|
120
|
-
assert.equal((await f.runValidators('abcd')).length, 0)
|
|
121
|
-
})
|
|
122
|
-
|
|
123
|
-
it('reports every error, not just the first', async () => {
|
|
124
|
-
const f = EmailField.make('x').validate([minLength(20), email()])
|
|
125
|
-
const errors = await f.runValidators('a@b')
|
|
126
|
-
assert.equal(errors.length, 2)
|
|
127
|
-
})
|
|
128
|
-
|
|
129
|
-
it('honors validator order', async () => {
|
|
130
|
-
const f = TextField.make('x').validate([minLength(5), maxLength(2)])
|
|
131
|
-
const errors = await f.runValidators('abc')
|
|
132
|
-
// both fail; order is [minLength, maxLength]
|
|
133
|
-
assert.match(errors[0]!, /at least 5/)
|
|
134
|
-
assert.match(errors[1]!, /at most 2/)
|
|
135
|
-
})
|
|
136
|
-
|
|
137
|
-
it('passes ctx through to validators', async () => {
|
|
138
|
-
const seenValues: unknown[] = []
|
|
139
|
-
const v = makeValidator((_value, ctx) => {
|
|
140
|
-
seenValues.push(ctx?.values)
|
|
141
|
-
return null
|
|
142
|
-
})
|
|
143
|
-
const f = TextField.make('x').validate(v)
|
|
144
|
-
await f.runValidators('a', { values: { other: 1 } })
|
|
145
|
-
assert.deepEqual(seenValues, [{ other: 1 }])
|
|
146
|
-
})
|
|
147
|
-
|
|
148
|
-
it('required() flag implicitly adds a required check', async () => {
|
|
149
|
-
const f = TextField.make('x').required()
|
|
150
|
-
assert.deepEqual(await f.runValidators(''), ['This field is required'])
|
|
151
|
-
assert.deepEqual(await f.runValidators('ok'), [])
|
|
152
|
-
})
|
|
153
|
-
|
|
154
|
-
it('does not double-fire required when both flag and validator are set', async () => {
|
|
155
|
-
const f = TextField.make('x').required().validate(required('Custom required'))
|
|
156
|
-
const errors = await f.runValidators('')
|
|
157
|
-
assert.equal(errors.length, 1)
|
|
158
|
-
assert.equal(errors[0], 'Custom required')
|
|
159
|
-
})
|
|
160
|
-
|
|
161
|
-
it('awaits async validators in declaration order', async () => {
|
|
162
|
-
const seen: string[] = []
|
|
163
|
-
const slow = makeValidator(async _v => {
|
|
164
|
-
await new Promise(r => setTimeout(r, 5))
|
|
165
|
-
seen.push('slow')
|
|
166
|
-
return 'slow-error'
|
|
167
|
-
})
|
|
168
|
-
const fast = makeValidator(_v => { seen.push('fast'); return 'fast-error' })
|
|
169
|
-
const f = TextField.make('x').validate([slow, fast])
|
|
170
|
-
const errors = await f.runValidators('hi')
|
|
171
|
-
assert.deepEqual(seen, ['slow', 'fast'])
|
|
172
|
-
assert.deepEqual(errors, ['slow-error', 'fast-error'])
|
|
173
|
-
})
|
|
174
|
-
})
|
|
175
|
-
|
|
176
|
-
describe('Field.toMeta serialized rules', () => {
|
|
177
|
-
it('omits rules array when no validators and not required', () => {
|
|
178
|
-
const meta = TextField.make('x').toMeta()
|
|
179
|
-
assert.equal('rules' in meta, false)
|
|
180
|
-
})
|
|
181
|
-
|
|
182
|
-
it('includes a required rule when .required() is set', () => {
|
|
183
|
-
const meta = TextField.make('x').required().toMeta()
|
|
184
|
-
assert.deepEqual(meta.rules, [{ rule: 'required', message: 'This field is required' }])
|
|
185
|
-
})
|
|
186
|
-
|
|
187
|
-
it('serializes each validator that carries a descriptor', () => {
|
|
188
|
-
const meta = EmailField.make('x').validate([email(), minLength(5)]).toMeta()
|
|
189
|
-
assert.deepEqual(meta.rules, [
|
|
190
|
-
{ rule: 'email', message: 'Must be a valid email' },
|
|
191
|
-
{ rule: 'minLength', value: 5, message: 'Must be at least 5 characters' },
|
|
192
|
-
])
|
|
193
|
-
})
|
|
194
|
-
|
|
195
|
-
it('skips validators without a serialized descriptor', () => {
|
|
196
|
-
const customOnly = makeValidator(() => null) // no serialized
|
|
197
|
-
const meta = TextField.make('x').validate(customOnly).toMeta()
|
|
198
|
-
assert.equal('rules' in meta, false)
|
|
199
|
-
})
|
|
200
|
-
|
|
201
|
-
it('does not duplicate required when both flag and validator are set', () => {
|
|
202
|
-
const meta = TextField.make('x').required().validate(required('Custom required')).toMeta()
|
|
203
|
-
assert.equal(meta.rules!.length, 1)
|
|
204
|
-
assert.equal((meta.rules![0] as { message: string }).message, 'Custom required')
|
|
205
|
-
})
|
|
206
|
-
})
|
|
207
|
-
|
|
208
|
-
describe('validateSchema (tree-level runner)', () => {
|
|
209
|
-
it('walks containers and gathers errors keyed by field name', async () => {
|
|
210
|
-
const schema = [
|
|
211
|
-
Section.make('Profile').schema([
|
|
212
|
-
TextField.make('name').required(),
|
|
213
|
-
EmailField.make('email').validate(email()),
|
|
214
|
-
]),
|
|
215
|
-
Card.make().schema([
|
|
216
|
-
NumberField.make('age').validate(min(0)),
|
|
217
|
-
]),
|
|
218
|
-
]
|
|
219
|
-
const errors = await validateSchema(schema, { name: '', email: 'bad', age: -1 })
|
|
220
|
-
assert.deepEqual(errors['name'], ['This field is required'])
|
|
221
|
-
assert.deepEqual(errors['email'], ['Must be a valid email'])
|
|
222
|
-
assert.deepEqual(errors['age'], ['Must be at least 0'])
|
|
223
|
-
assert.equal(isValid(errors), false)
|
|
224
|
-
})
|
|
225
|
-
|
|
226
|
-
it('walks deeply-nested Tabs/Tab containers', async () => {
|
|
227
|
-
const schema = [
|
|
228
|
-
Tabs.make().tabs([
|
|
229
|
-
Tab.make('Settings').schema([
|
|
230
|
-
TextField.make('alias').required(),
|
|
231
|
-
]),
|
|
232
|
-
]),
|
|
233
|
-
]
|
|
234
|
-
const errors = await validateSchema(schema, { alias: '' })
|
|
235
|
-
assert.deepEqual(errors['alias'], ['This field is required'])
|
|
236
|
-
})
|
|
237
|
-
|
|
238
|
-
it('returns empty map and isValid=true when everything passes', async () => {
|
|
239
|
-
const schema = [TextField.make('a').required(), TextField.make('b')]
|
|
240
|
-
const errors = await validateSchema(schema, { a: 'hello', b: undefined })
|
|
241
|
-
assert.deepEqual(errors, {})
|
|
242
|
-
assert.equal(isValid(errors), true)
|
|
243
|
-
})
|
|
244
|
-
|
|
245
|
-
it('passes record through to validator ctx', async () => {
|
|
246
|
-
let captured: unknown
|
|
247
|
-
const v = makeValidator((_value, ctx) => { captured = ctx?.record; return null })
|
|
248
|
-
const schema = [TextField.make('x').validate(v)]
|
|
249
|
-
await validateSchema(schema, { x: 'hi' }, { id: 42 })
|
|
250
|
-
assert.deepEqual(captured, { id: 42 })
|
|
251
|
-
})
|
|
252
|
-
|
|
253
|
-
it('readonly fields still validate', async () => {
|
|
254
|
-
const schema = [TextField.make('x').readonly().required()]
|
|
255
|
-
const errors = await validateSchema(schema, { x: '' })
|
|
256
|
-
assert.deepEqual(errors['x'], ['This field is required'])
|
|
257
|
-
})
|
|
258
|
-
|
|
259
|
-
it('awaits async field validators', async () => {
|
|
260
|
-
const v = makeValidator(async _v => {
|
|
261
|
-
await new Promise(r => setTimeout(r, 1))
|
|
262
|
-
return 'async-error'
|
|
263
|
-
})
|
|
264
|
-
const schema = [TextField.make('x').validate(v)]
|
|
265
|
-
const errors = await validateSchema(schema, { x: 'hi' })
|
|
266
|
-
assert.deepEqual(errors['x'], ['async-error'])
|
|
267
|
-
})
|
|
268
|
-
})
|