@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
package/src/Page.ts
DELETED
|
@@ -1,208 +0,0 @@
|
|
|
1
|
-
import type { Element } from './schema/Element.js'
|
|
2
|
-
import type { SchemaContext, SchemaDefinition } from './schema/resolveSchema.js'
|
|
3
|
-
import type { ResourceClass, NavigationBadgeColor, NavigationBadgeHandler } from './Resource.js'
|
|
4
|
-
import type { ClusterClass } from './Cluster.js'
|
|
5
|
-
import { type IconValue, serializeIcon } from './icons/types.js'
|
|
6
|
-
|
|
7
|
-
/**
|
|
8
|
-
* Per-custom-page collab configuration. Custom pages have no record id
|
|
9
|
-
* to derive a room from, so `room` is required and must be supplied as
|
|
10
|
-
* a literal string (e.g. `'team-settings'`). The plugin reads this off
|
|
11
|
-
* the panel-level `pageCollab` map and uses it to seed the page's
|
|
12
|
-
* Y.Doc.
|
|
13
|
-
*
|
|
14
|
-
* room — literal room identifier; namespaced internally so two
|
|
15
|
-
* pages with the same `room` literal collide deliberately.
|
|
16
|
-
* presence — when false, suppress the awareness layer (focus chips,
|
|
17
|
-
* cursor positions) while keeping value-sync. Defaults to true.
|
|
18
|
-
*
|
|
19
|
-
* Field-level `.collab(false)` always wins per field. Resource-bound
|
|
20
|
-
* default pages (List/Create/Edit/View) ignore this — record-scoped
|
|
21
|
-
* collab is governed by `Resource.collab`.
|
|
22
|
-
*/
|
|
23
|
-
export interface PageCollabConfig {
|
|
24
|
-
room: string
|
|
25
|
-
presence: boolean
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
/** Raw shape accepted by `static collab` before normalization. Unlike
|
|
29
|
-
* `Resource.collab` there is no `true` shorthand — `room` is required. */
|
|
30
|
-
export type PageCollabInput = {
|
|
31
|
-
room: string
|
|
32
|
-
presence?: boolean
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
/**
|
|
36
|
-
* Discriminator the framework uses for default rendering, route generation,
|
|
37
|
-
* and breadcrumbs. `'custom'` is for standalone Pages that don't belong to
|
|
38
|
-
* a Resource. `'record'` is for record-scoped sub-pages declared under a
|
|
39
|
-
* Resource's `pages().record` map. The other four modes correspond to the
|
|
40
|
-
* standard resource page roles.
|
|
41
|
-
*/
|
|
42
|
-
export type PageMode = 'list' | 'create' | 'edit' | 'view' | 'record' | 'custom'
|
|
43
|
-
|
|
44
|
-
export interface PageMeta {
|
|
45
|
-
slug: string
|
|
46
|
-
label: string
|
|
47
|
-
/** Serialized: string registry key, `{ class }` manifest reference, or undefined. */
|
|
48
|
-
icon: string | { class: string } | undefined
|
|
49
|
-
mode: PageMode
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
export class Page {
|
|
53
|
-
/** URL slug (e.g. 'analytics'). Derived from class name if not set. */
|
|
54
|
-
static slug?: string
|
|
55
|
-
|
|
56
|
-
/** Sidebar label (e.g. 'Analytics'). Derived from class name if not set. */
|
|
57
|
-
static label?: string
|
|
58
|
-
|
|
59
|
-
/** Optional sidebar icon. Either a kebab-case name resolved through
|
|
60
|
-
* `registerIcons()` (e.g., `'newspaper'`), or a React component
|
|
61
|
-
* reference (e.g., `import { Newspaper } from 'lucide-react'`).
|
|
62
|
-
* Component refs serialize via the build-time `_components.ts`
|
|
63
|
-
* manifest emitted by the Pilotiq Vite plugin. */
|
|
64
|
-
static icon?: IconValue
|
|
65
|
-
|
|
66
|
-
// ─── Plan #9: navigation metadata ──────────────────────────
|
|
67
|
-
// Mirrors Resource's nav fields (minus `recordTitleAttribute`).
|
|
68
|
-
static navigationGroup: string | undefined = undefined
|
|
69
|
-
static navigationSort: number | undefined = undefined
|
|
70
|
-
static navigationLabel: string | undefined = undefined
|
|
71
|
-
static navigationIcon: IconValue = undefined
|
|
72
|
-
static navigationBadge: NavigationBadgeHandler | undefined = undefined
|
|
73
|
-
static navigationBadgeColor: NavigationBadgeColor = 'default'
|
|
74
|
-
static navigationParentItem: string | undefined = undefined
|
|
75
|
-
|
|
76
|
-
/**
|
|
77
|
-
* Cluster this page belongs to. When set, the page's URL gains the
|
|
78
|
-
* cluster's slug as a prefix segment and the page nests under the
|
|
79
|
-
* cluster's nav entry. The referenced class must be registered via
|
|
80
|
-
* `Pilotiq.clusters([…])`.
|
|
81
|
-
*/
|
|
82
|
-
static cluster?: ClusterClass
|
|
83
|
-
|
|
84
|
-
/** Stored schema definition. */
|
|
85
|
-
protected static _schemaDef?: SchemaDefinition
|
|
86
|
-
|
|
87
|
-
/**
|
|
88
|
-
* Define the page content using a stored schema definition.
|
|
89
|
-
*
|
|
90
|
-
* @example
|
|
91
|
-
* static {
|
|
92
|
-
* this.define(async (ctx) => [
|
|
93
|
-
* Heading.make('Analytics'),
|
|
94
|
-
* ])
|
|
95
|
-
* }
|
|
96
|
-
*/
|
|
97
|
-
static define(def: SchemaDefinition): typeof Page {
|
|
98
|
-
this._schemaDef = def
|
|
99
|
-
return this
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
/**
|
|
103
|
-
* Return the page's schema elements.
|
|
104
|
-
* Override this method for full control, or use define() for inline definitions.
|
|
105
|
-
*/
|
|
106
|
-
static schema(_ctx?: SchemaContext): Element[] | Promise<Element[]> {
|
|
107
|
-
if (!this._schemaDef) return []
|
|
108
|
-
return typeof this._schemaDef === 'function'
|
|
109
|
-
? this._schemaDef(_ctx ?? {})
|
|
110
|
-
: this._schemaDef
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
static getSlug(): string {
|
|
114
|
-
if (this.slug) return this.slug
|
|
115
|
-
return this.name.replace(/Page$/, '').toLowerCase()
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
static getLabel(): string {
|
|
119
|
-
if (this.label) return this.label
|
|
120
|
-
const name = this.name.replace(/Page$/, '')
|
|
121
|
-
return name.replace(/([A-Z])/g, ' $1').trim()
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
/** Sidebar label: `navigationLabel` override falls through to `getLabel()`. */
|
|
125
|
-
static getNavigationLabel(): string {
|
|
126
|
-
return this.navigationLabel ?? this.getLabel()
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
/** Sidebar icon: `navigationIcon` override falls through to `icon`. */
|
|
130
|
-
static getNavigationIcon(): IconValue {
|
|
131
|
-
return this.navigationIcon ?? this.icon
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
static hasSchema(): boolean {
|
|
135
|
-
return this._schemaDef !== undefined || this.schema !== Page.schema
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
// ─── Realtime collab opt-in ────────────────────────────────
|
|
139
|
-
// Per-page opt-in for custom Pages. Resource-bound default pages
|
|
140
|
-
// (List/Create/Edit/View) ignore this — record-scoped collab is
|
|
141
|
-
// governed by `Resource.collab`. Set on a custom Page subclass to
|
|
142
|
-
// mount the plugin-registered custom-page wrapper around its content
|
|
143
|
-
// area at runtime.
|
|
144
|
-
|
|
145
|
-
/** Enable collab on this custom page. `room` is required (no recordId
|
|
146
|
-
* to derive one from). Omitting `static collab` keeps the page
|
|
147
|
-
* collab-free even when the `@pilotiq-pro/collab` plugin is installed. */
|
|
148
|
-
static collab: PageCollabInput | null = null
|
|
149
|
-
|
|
150
|
-
/**
|
|
151
|
-
* Normalize `static collab` into the canonical wire shape, or return
|
|
152
|
-
* `null` when the page has not opted in. Centralizes the `presence`
|
|
153
|
-
* default. Resource-bound default pages always return `null` here —
|
|
154
|
-
* use `Resource.collab` for those.
|
|
155
|
-
*
|
|
156
|
-
* Result lands on `panelInfo().pageCollab[slug]`; the gate reads it to
|
|
157
|
-
* decide whether to mount the custom-page wrapper.
|
|
158
|
-
*/
|
|
159
|
-
static getResolvedCollabConfig(): PageCollabConfig | null {
|
|
160
|
-
if (this.getMode() !== 'custom') return null
|
|
161
|
-
const raw = this.collab
|
|
162
|
-
if (raw === null || raw === undefined) return null
|
|
163
|
-
if (typeof raw.room !== 'string' || raw.room.length === 0) return null
|
|
164
|
-
return {
|
|
165
|
-
room: raw.room,
|
|
166
|
-
presence: raw.presence ?? true,
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
/** Plan #10: authorization. Custom pages get a single `canAccess` gate
|
|
171
|
-
* (no per-record predicates — pages are too freeform to assume a
|
|
172
|
-
* record concept). Resource-bound default page subclasses can still
|
|
173
|
-
* read their owning resource's predicates via `getResource()`.
|
|
174
|
-
*
|
|
175
|
-
* Record sub-pages (declared under `ResourcePages.record`) receive
|
|
176
|
-
* the loaded parent record as the second argument so subclasses can
|
|
177
|
-
* gate on record state in addition to user. The parameter is
|
|
178
|
-
* optional so existing custom-page subclasses with `canAccess(user)`
|
|
179
|
-
* keep type-checking. Record-aware sub-pages typically override as
|
|
180
|
-
* `canAccess(user, record)`. */
|
|
181
|
-
static async canAccess(_user: unknown, _record?: unknown): Promise<boolean> { return true }
|
|
182
|
-
|
|
183
|
-
/**
|
|
184
|
-
* Optional back-reference to the owning Resource. Auto-generated default
|
|
185
|
-
* pages set this; user subclasses may override for breadcrumb / title
|
|
186
|
-
* resolution. Standalone custom pages return undefined.
|
|
187
|
-
*/
|
|
188
|
-
static getResource(): ResourceClass | undefined { return undefined }
|
|
189
|
-
|
|
190
|
-
/**
|
|
191
|
-
* Mode discriminator. Default `'custom'` (standalone page). Resource-bound
|
|
192
|
-
* pages override to one of `'list' | 'create' | 'edit' | 'view'`.
|
|
193
|
-
*/
|
|
194
|
-
static getMode(): PageMode { return 'custom' }
|
|
195
|
-
|
|
196
|
-
/** @internal */
|
|
197
|
-
static toMeta(): PageMeta {
|
|
198
|
-
// Serialize via the icon-system helper so component-typed icons
|
|
199
|
-
// ship as `{ class: <name> }` rather than the raw forwardRef
|
|
200
|
-
// object — viewProps must be JSON-serializable.
|
|
201
|
-
return {
|
|
202
|
-
slug: this.getSlug(),
|
|
203
|
-
label: this.getLabel(),
|
|
204
|
-
icon: serializeIcon(this.icon, this.name),
|
|
205
|
-
mode: this.getMode(),
|
|
206
|
-
}
|
|
207
|
-
}
|
|
208
|
-
}
|
package/src/Pilotiq.perf.test.ts
DELETED
|
@@ -1,252 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Phase 5 perf sweep — covers the four hot-path changes that landed
|
|
3
|
-
* 2026-05-22:
|
|
4
|
-
*
|
|
5
|
-
* - 5b Per-user navigation-badge TTL cache (`Pilotiq.navigationBadgeTtl`)
|
|
6
|
-
* - 5c Map-based slug lookup (`Pilotiq.findResource/findGlobal/findPage`)
|
|
7
|
-
* - 5a Chunked import (`importFactory.runImport` honors `concurrency`)
|
|
8
|
-
*
|
|
9
|
-
* 5d (`policyGate`) is exercised indirectly by the existing routes /
|
|
10
|
-
* authorization tests — its contract is identical to the prior
|
|
11
|
-
* serial pair, just parallelized; no behavior change to assert here.
|
|
12
|
-
*/
|
|
13
|
-
import { describe, it } from 'node:test'
|
|
14
|
-
import assert from 'node:assert/strict'
|
|
15
|
-
|
|
16
|
-
import { Pilotiq } from './Pilotiq.js'
|
|
17
|
-
import { Resource } from './Resource.js'
|
|
18
|
-
import { Global } from './Global.js'
|
|
19
|
-
import { Page } from './Page.js'
|
|
20
|
-
import { runImport } from './actions/importFactory.js'
|
|
21
|
-
|
|
22
|
-
// ─── Fixtures ─────────────────────────────────────────────────
|
|
23
|
-
|
|
24
|
-
class Articles extends Resource {
|
|
25
|
-
static override slug = 'articles'
|
|
26
|
-
static override label = 'Articles'
|
|
27
|
-
}
|
|
28
|
-
class Comments extends Resource {
|
|
29
|
-
static override slug = 'comments'
|
|
30
|
-
static override label = 'Comments'
|
|
31
|
-
}
|
|
32
|
-
class Settings extends Global {
|
|
33
|
-
static override slug = 'settings'
|
|
34
|
-
static override label = 'Settings'
|
|
35
|
-
}
|
|
36
|
-
class Branding extends Global {
|
|
37
|
-
static override slug = 'branding'
|
|
38
|
-
static override label = 'Branding'
|
|
39
|
-
}
|
|
40
|
-
class Reports extends Page {
|
|
41
|
-
static override slug = 'reports'
|
|
42
|
-
static override label = 'Reports'
|
|
43
|
-
}
|
|
44
|
-
class Health extends Page {
|
|
45
|
-
static override slug = 'health'
|
|
46
|
-
static override label = 'Health'
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
// ─── 5c — Map-based slug lookup ───────────────────────────────
|
|
50
|
-
|
|
51
|
-
describe('Pilotiq.find{Resource,Global,Page}() — Plan 5c', () => {
|
|
52
|
-
it('returns the matching class by slug', () => {
|
|
53
|
-
const p = Pilotiq.make('admin')
|
|
54
|
-
.resources([Articles, Comments])
|
|
55
|
-
.globals([Settings, Branding])
|
|
56
|
-
.pages([Reports, Health])
|
|
57
|
-
assert.equal(p.findResource('articles'), Articles)
|
|
58
|
-
assert.equal(p.findResource('comments'), Comments)
|
|
59
|
-
assert.equal(p.findGlobal('settings'), Settings)
|
|
60
|
-
assert.equal(p.findGlobal('branding'), Branding)
|
|
61
|
-
assert.equal(p.findPage('reports'), Reports)
|
|
62
|
-
assert.equal(p.findPage('health'), Health)
|
|
63
|
-
})
|
|
64
|
-
|
|
65
|
-
it('returns undefined for unknown slugs', () => {
|
|
66
|
-
const p = Pilotiq.make('admin').resources([Articles])
|
|
67
|
-
assert.equal(p.findResource('nope'), undefined)
|
|
68
|
-
assert.equal(p.findGlobal('nope'), undefined)
|
|
69
|
-
assert.equal(p.findPage('nope'), undefined)
|
|
70
|
-
})
|
|
71
|
-
|
|
72
|
-
it('invalidates the cache when .resources() is reassigned', () => {
|
|
73
|
-
const p = Pilotiq.make('admin').resources([Articles])
|
|
74
|
-
assert.equal(p.findResource('articles'), Articles)
|
|
75
|
-
assert.equal(p.findResource('comments'), undefined)
|
|
76
|
-
p.resources([Articles, Comments])
|
|
77
|
-
assert.equal(p.findResource('comments'), Comments)
|
|
78
|
-
})
|
|
79
|
-
|
|
80
|
-
it('invalidates the page cache when .pages() is reassigned', () => {
|
|
81
|
-
const p = Pilotiq.make('admin').pages([Reports])
|
|
82
|
-
assert.equal(p.findPage('reports'), Reports)
|
|
83
|
-
assert.equal(p.findPage('health'), undefined)
|
|
84
|
-
p.pages([Reports, Health])
|
|
85
|
-
assert.equal(p.findPage('health'), Health)
|
|
86
|
-
})
|
|
87
|
-
|
|
88
|
-
it('invalidates the page cache when .dashboard()/.profile() auto-append', () => {
|
|
89
|
-
class Dash extends Page {
|
|
90
|
-
static override slug = 'dash'
|
|
91
|
-
static override label = 'Dashboard'
|
|
92
|
-
}
|
|
93
|
-
const p = Pilotiq.make('admin')
|
|
94
|
-
assert.equal(p.findPage('dash'), undefined)
|
|
95
|
-
p.dashboard(Dash)
|
|
96
|
-
assert.equal(p.findPage('dash'), Dash)
|
|
97
|
-
})
|
|
98
|
-
})
|
|
99
|
-
|
|
100
|
-
// ─── 5b — Navigation badge TTL cache ──────────────────────────
|
|
101
|
-
|
|
102
|
-
describe('Pilotiq.navigationBadgeTtl() + resolveNavigationBadge() — Plan 5b', () => {
|
|
103
|
-
it('default TTL is 30s', () => {
|
|
104
|
-
const p = Pilotiq.make('admin')
|
|
105
|
-
assert.equal(p.getNavigationBadgeTtl(), 30_000)
|
|
106
|
-
})
|
|
107
|
-
|
|
108
|
-
it('navigationBadgeTtl(ms) overrides; clamps negatives to 0', () => {
|
|
109
|
-
const p = Pilotiq.make('admin').navigationBadgeTtl(5_000)
|
|
110
|
-
assert.equal(p.getNavigationBadgeTtl(), 5_000)
|
|
111
|
-
p.navigationBadgeTtl(-1)
|
|
112
|
-
assert.equal(p.getNavigationBadgeTtl(), 0)
|
|
113
|
-
})
|
|
114
|
-
|
|
115
|
-
it('navigationBadgeTtl(null) restores the default', () => {
|
|
116
|
-
const p = Pilotiq.make('admin').navigationBadgeTtl(1_000)
|
|
117
|
-
p.navigationBadgeTtl(null)
|
|
118
|
-
assert.equal(p.getNavigationBadgeTtl(), 30_000)
|
|
119
|
-
})
|
|
120
|
-
|
|
121
|
-
it('resolveNavigationBadge caches within TTL, busts on user change', async () => {
|
|
122
|
-
const p = Pilotiq.make('admin')
|
|
123
|
-
let calls = 0
|
|
124
|
-
const resolver = async () => { calls++; return String(calls) }
|
|
125
|
-
|
|
126
|
-
// First call: miss → resolver fires → returns '1'.
|
|
127
|
-
assert.equal(await p.resolveNavigationBadge('Articles', { id: 1 }, resolver), '1')
|
|
128
|
-
assert.equal(calls, 1)
|
|
129
|
-
// Same user + owner: hit → no new call → still '1'.
|
|
130
|
-
assert.equal(await p.resolveNavigationBadge('Articles', { id: 1 }, resolver), '1')
|
|
131
|
-
assert.equal(calls, 1)
|
|
132
|
-
// Different user: miss → resolver fires again.
|
|
133
|
-
assert.equal(await p.resolveNavigationBadge('Articles', { id: 2 }, resolver), '2')
|
|
134
|
-
assert.equal(calls, 2)
|
|
135
|
-
// Different owner, same user: separate cache slot.
|
|
136
|
-
assert.equal(await p.resolveNavigationBadge('Comments', { id: 1 }, resolver), '3')
|
|
137
|
-
assert.equal(calls, 3)
|
|
138
|
-
})
|
|
139
|
-
|
|
140
|
-
it('TTL of 0 disables caching entirely', async () => {
|
|
141
|
-
const p = Pilotiq.make('admin').navigationBadgeTtl(0)
|
|
142
|
-
let calls = 0
|
|
143
|
-
const resolver = async () => { calls++; return 'x' }
|
|
144
|
-
await p.resolveNavigationBadge('Articles', { id: 1 }, resolver)
|
|
145
|
-
await p.resolveNavigationBadge('Articles', { id: 1 }, resolver)
|
|
146
|
-
assert.equal(calls, 2)
|
|
147
|
-
})
|
|
148
|
-
|
|
149
|
-
it('caches undefined results (no need to keep re-resolving "no badge")', async () => {
|
|
150
|
-
const p = Pilotiq.make('admin')
|
|
151
|
-
let calls = 0
|
|
152
|
-
const resolver = async () => { calls++; return undefined }
|
|
153
|
-
assert.equal(await p.resolveNavigationBadge('Articles', null, resolver), undefined)
|
|
154
|
-
assert.equal(await p.resolveNavigationBadge('Articles', null, resolver), undefined)
|
|
155
|
-
assert.equal(calls, 1)
|
|
156
|
-
})
|
|
157
|
-
|
|
158
|
-
it('navigationBadgeTtl(ms) clears the cache', async () => {
|
|
159
|
-
const p = Pilotiq.make('admin')
|
|
160
|
-
let calls = 0
|
|
161
|
-
const resolver = async () => { calls++; return 'x' }
|
|
162
|
-
await p.resolveNavigationBadge('A', null, resolver)
|
|
163
|
-
assert.equal(calls, 1)
|
|
164
|
-
p.navigationBadgeTtl(60_000)
|
|
165
|
-
await p.resolveNavigationBadge('A', null, resolver)
|
|
166
|
-
assert.equal(calls, 2)
|
|
167
|
-
})
|
|
168
|
-
|
|
169
|
-
it('anonymous users share one cache slot', async () => {
|
|
170
|
-
const p = Pilotiq.make('admin')
|
|
171
|
-
let calls = 0
|
|
172
|
-
const resolver = async () => { calls++; return 'x' }
|
|
173
|
-
await p.resolveNavigationBadge('A', null, resolver)
|
|
174
|
-
await p.resolveNavigationBadge('A', undefined, resolver)
|
|
175
|
-
assert.equal(calls, 1)
|
|
176
|
-
})
|
|
177
|
-
|
|
178
|
-
it('falls back to JSON.stringify when user has no .id', async () => {
|
|
179
|
-
const p = Pilotiq.make('admin')
|
|
180
|
-
let calls = 0
|
|
181
|
-
const resolver = async () => { calls++; return 'x' }
|
|
182
|
-
await p.resolveNavigationBadge('A', { role: 'editor' }, resolver)
|
|
183
|
-
await p.resolveNavigationBadge('A', { role: 'editor' }, resolver)
|
|
184
|
-
assert.equal(calls, 1) // same JSON shape → cache hit
|
|
185
|
-
await p.resolveNavigationBadge('A', { role: 'admin' }, resolver)
|
|
186
|
-
assert.equal(calls, 2) // different JSON → miss
|
|
187
|
-
})
|
|
188
|
-
})
|
|
189
|
-
|
|
190
|
-
// ─── 5a — Chunked importFactory.runImport ─────────────────────
|
|
191
|
-
|
|
192
|
-
describe('importFactory.runImport — Plan 5a chunking', () => {
|
|
193
|
-
it('runs rows in chunks of `concurrency` and aggregates counts', async () => {
|
|
194
|
-
const created: string[] = []
|
|
195
|
-
let maxInFlight = 0
|
|
196
|
-
let inFlight = 0
|
|
197
|
-
const M = {
|
|
198
|
-
async create(row: { id: string }) {
|
|
199
|
-
inFlight++; if (inFlight > maxInFlight) maxInFlight = inFlight
|
|
200
|
-
await new Promise(r => setTimeout(r, 5))
|
|
201
|
-
inFlight--
|
|
202
|
-
created.push(row.id)
|
|
203
|
-
},
|
|
204
|
-
// unused for create-mode tests but the type wants them present
|
|
205
|
-
query() { return { where() { return { paginate: async () => ({ data: [] }) } } } },
|
|
206
|
-
async update() {},
|
|
207
|
-
}
|
|
208
|
-
const rows = Array.from({ length: 25 }, (_, i) => ({ id: `r${i}` }))
|
|
209
|
-
const summary = await runImport(rows, M, 'create', { concurrency: 5 }, { request: undefined })
|
|
210
|
-
assert.equal(summary.created, 25)
|
|
211
|
-
assert.equal(summary.errors.length, 0)
|
|
212
|
-
// With concurrency=5 we should see at least 4 in-flight at peak.
|
|
213
|
-
assert.ok(maxInFlight >= 4, `expected >=4 concurrent, saw ${maxInFlight}`)
|
|
214
|
-
// Never exceed the cap.
|
|
215
|
-
assert.ok(maxInFlight <= 5, `expected <=5 concurrent, saw ${maxInFlight}`)
|
|
216
|
-
})
|
|
217
|
-
|
|
218
|
-
it('preserves original-row indices in error messages despite chunking', async () => {
|
|
219
|
-
const M = {
|
|
220
|
-
async create(row: { id: string }) {
|
|
221
|
-
if (row.id === 'r2') throw new Error('boom')
|
|
222
|
-
},
|
|
223
|
-
query() { return { where() { return { paginate: async () => ({ data: [] }) } } } },
|
|
224
|
-
async update() {},
|
|
225
|
-
}
|
|
226
|
-
const rows = [{ id: 'r0' }, { id: 'r1' }, { id: 'r2' }, { id: 'r3' }]
|
|
227
|
-
const summary = await runImport(rows, M, 'create', { concurrency: 4 }, { request: undefined })
|
|
228
|
-
assert.equal(summary.created, 3)
|
|
229
|
-
assert.equal(summary.skipped, 1)
|
|
230
|
-
assert.equal(summary.errors.length, 1)
|
|
231
|
-
assert.equal(summary.errors[0]?.row, 3) // 1-based, original index 2 → row 3
|
|
232
|
-
assert.match(summary.errors[0]?.message ?? '', /boom/)
|
|
233
|
-
})
|
|
234
|
-
|
|
235
|
-
it('defaults to concurrency 10 when unset', async () => {
|
|
236
|
-
let maxInFlight = 0
|
|
237
|
-
let inFlight = 0
|
|
238
|
-
const M = {
|
|
239
|
-
async create() {
|
|
240
|
-
inFlight++; if (inFlight > maxInFlight) maxInFlight = inFlight
|
|
241
|
-
await new Promise(r => setTimeout(r, 3))
|
|
242
|
-
inFlight--
|
|
243
|
-
},
|
|
244
|
-
query() { return { where() { return { paginate: async () => ({ data: [] }) } } } },
|
|
245
|
-
async update() {},
|
|
246
|
-
}
|
|
247
|
-
const rows = Array.from({ length: 30 }, () => ({}))
|
|
248
|
-
await runImport(rows, M, 'create', {}, { request: undefined })
|
|
249
|
-
assert.ok(maxInFlight <= 10, `expected <=10 concurrent, saw ${maxInFlight}`)
|
|
250
|
-
assert.ok(maxInFlight >= 5, `expected >=5 concurrent under default, saw ${maxInFlight}`)
|
|
251
|
-
})
|
|
252
|
-
})
|
package/src/Pilotiq.test.ts
DELETED
|
@@ -1,129 +0,0 @@
|
|
|
1
|
-
import { describe, it } from 'node:test'
|
|
2
|
-
import assert from 'node:assert/strict'
|
|
3
|
-
import type { Router } from '@rudderjs/router'
|
|
4
|
-
import { Pilotiq, type PilotiqPlugin } from './Pilotiq.js'
|
|
5
|
-
import { registerPilotiqRoutes } from './routes.js'
|
|
6
|
-
|
|
7
|
-
describe('Pilotiq plugins', () => {
|
|
8
|
-
it('.use() invokes register() with the panel and records the plugin', () => {
|
|
9
|
-
const seen: Array<{ name: string; same: boolean }> = []
|
|
10
|
-
const p1: PilotiqPlugin = {
|
|
11
|
-
name: 'p1',
|
|
12
|
-
register(panel) { seen.push({ name: 'p1', same: panel instanceof Pilotiq }) },
|
|
13
|
-
}
|
|
14
|
-
const panel = Pilotiq.make('Admin').use(p1)
|
|
15
|
-
assert.deepEqual(seen, [{ name: 'p1', same: true }])
|
|
16
|
-
const installed = panel.getPlugins()
|
|
17
|
-
assert.equal(installed.length, 1)
|
|
18
|
-
assert.equal(installed[0]?.name, 'p1')
|
|
19
|
-
})
|
|
20
|
-
|
|
21
|
-
it('.plugins([…]) runs each plugin in array order', () => {
|
|
22
|
-
const order: string[] = []
|
|
23
|
-
const make = (name: string): PilotiqPlugin => ({
|
|
24
|
-
name,
|
|
25
|
-
register() { order.push(name) },
|
|
26
|
-
})
|
|
27
|
-
const panel = Pilotiq.make('Admin').plugins([make('a'), make('b'), make('c')])
|
|
28
|
-
assert.deepEqual(order, ['a', 'b', 'c'])
|
|
29
|
-
assert.deepEqual(panel.getPlugins().map((p) => p.name), ['a', 'b', 'c'])
|
|
30
|
-
})
|
|
31
|
-
|
|
32
|
-
it('.plugins([]) is a no-op', () => {
|
|
33
|
-
const panel = Pilotiq.make('Admin').plugins([])
|
|
34
|
-
assert.equal(panel.getPlugins().length, 0)
|
|
35
|
-
})
|
|
36
|
-
|
|
37
|
-
it('.plugins([…]) composes with .use()', () => {
|
|
38
|
-
const log: string[] = []
|
|
39
|
-
const mk = (n: string): PilotiqPlugin => ({ name: n, register() { log.push(n) } })
|
|
40
|
-
const panel = Pilotiq.make('Admin')
|
|
41
|
-
.use(mk('first'))
|
|
42
|
-
.plugins([mk('second'), mk('third')])
|
|
43
|
-
.use(mk('fourth'))
|
|
44
|
-
assert.deepEqual(log, ['first', 'second', 'third', 'fourth'])
|
|
45
|
-
assert.deepEqual(panel.getPlugins().map((p) => p.name), [
|
|
46
|
-
'first', 'second', 'third', 'fourth',
|
|
47
|
-
])
|
|
48
|
-
})
|
|
49
|
-
})
|
|
50
|
-
|
|
51
|
-
// Minimal stub Router — `registerPilotiqRoutes` only calls `get` / `post`
|
|
52
|
-
// / `put` / `delete` on the router; route bodies don't fire here. The
|
|
53
|
-
// hook tests don't care about the route surface, only that the plugin's
|
|
54
|
-
// `registerRoutes` callback fires with the right arguments.
|
|
55
|
-
function makeStubRouter(): Router & { _calls: Array<{ method: string; path: string }> } {
|
|
56
|
-
const calls: Array<{ method: string; path: string }> = []
|
|
57
|
-
const noop = (path: string): void => { calls.push({ method: 'unknown', path }) }
|
|
58
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
59
|
-
const stub: any = {
|
|
60
|
-
get: (path: string) => noop(path),
|
|
61
|
-
post: (path: string) => noop(path),
|
|
62
|
-
put: (path: string) => noop(path),
|
|
63
|
-
delete: (path: string) => noop(path),
|
|
64
|
-
patch: (path: string) => noop(path),
|
|
65
|
-
// `router.group(opts, fn)` runs `fn()` synchronously inside its
|
|
66
|
-
// scope. Stub mirrors that — `Pilotiq.guard()` middleware doesn't
|
|
67
|
-
// touch the stub, only `fn()` matters.
|
|
68
|
-
group: (_opts: unknown, fn: () => void) => { fn() },
|
|
69
|
-
_calls: calls,
|
|
70
|
-
}
|
|
71
|
-
return stub
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
describe('PilotiqPlugin.registerRoutes hook', () => {
|
|
75
|
-
it('fires once per plugin during registerPilotiqRoutes, in registration order', () => {
|
|
76
|
-
const order: string[] = []
|
|
77
|
-
const router = makeStubRouter()
|
|
78
|
-
const mk = (n: string): PilotiqPlugin => ({
|
|
79
|
-
name: n,
|
|
80
|
-
register() { /* no-op */ },
|
|
81
|
-
registerRoutes(r, p) {
|
|
82
|
-
assert.equal(r, router)
|
|
83
|
-
assert.ok(p instanceof Pilotiq)
|
|
84
|
-
order.push(n)
|
|
85
|
-
},
|
|
86
|
-
})
|
|
87
|
-
const panel = Pilotiq.make('Admin').plugins([mk('a'), mk('b'), mk('c')])
|
|
88
|
-
registerPilotiqRoutes(router, panel)
|
|
89
|
-
assert.deepEqual(order, ['a', 'b', 'c'])
|
|
90
|
-
})
|
|
91
|
-
|
|
92
|
-
it('skips plugins that do not implement the hook', () => {
|
|
93
|
-
const order: string[] = []
|
|
94
|
-
const router = makeStubRouter()
|
|
95
|
-
const withHook: PilotiqPlugin = {
|
|
96
|
-
name: 'with-hook',
|
|
97
|
-
register() { /* no-op */ },
|
|
98
|
-
registerRoutes() { order.push('with-hook') },
|
|
99
|
-
}
|
|
100
|
-
const withoutHook: PilotiqPlugin = {
|
|
101
|
-
name: 'without-hook',
|
|
102
|
-
register() { /* no-op */ },
|
|
103
|
-
}
|
|
104
|
-
const panel = Pilotiq.make('Admin').plugins([withoutHook, withHook])
|
|
105
|
-
// No throw expected — the optional hook is skipped on plugins that
|
|
106
|
-
// omit it, so a "config-only" plugin remains a one-method object.
|
|
107
|
-
registerPilotiqRoutes(router, panel)
|
|
108
|
-
assert.deepEqual(order, ['with-hook'])
|
|
109
|
-
})
|
|
110
|
-
|
|
111
|
-
it('runs after core routes — at least one core call has fired before the first hook', () => {
|
|
112
|
-
const router = makeStubRouter()
|
|
113
|
-
let coreCallsAtHookTime = -1
|
|
114
|
-
const probe: PilotiqPlugin = {
|
|
115
|
-
name: 'probe',
|
|
116
|
-
register() { /* no-op */ },
|
|
117
|
-
registerRoutes() {
|
|
118
|
-
coreCallsAtHookTime = router._calls.length
|
|
119
|
-
},
|
|
120
|
-
}
|
|
121
|
-
// The dashboard route alone (`GET ${base}`) registers unconditionally
|
|
122
|
-
// — so any panel produces at least one core router call before the
|
|
123
|
-
// plugin hooks fire. Empty-resources panel is enough to cover this.
|
|
124
|
-
const panel = Pilotiq.make('Admin').plugins([probe])
|
|
125
|
-
registerPilotiqRoutes(router, panel)
|
|
126
|
-
assert.ok(coreCallsAtHookTime > 0,
|
|
127
|
-
`expected core routes to register before the hook, got ${coreCallsAtHookTime} calls`)
|
|
128
|
-
})
|
|
129
|
-
})
|