@pilotiq/pilotiq 0.24.1 → 0.24.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +33 -0
- package/boost/guidelines.md +566 -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/package.json +6 -1
- 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,95 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Broadcast-backed notifications — server-side push helpers.
|
|
3
|
-
*
|
|
4
|
-
* Phase 2 of the notifications plan. Pairs with `database.ts` (Phase 1):
|
|
5
|
-
* `database.ts` writes a row, `broadcast.ts` pushes the same payload to
|
|
6
|
-
* the recipient's private WebSocket channel so the bell-icon dropdown
|
|
7
|
-
* can refetch immediately instead of waiting on its polling interval.
|
|
8
|
-
*
|
|
9
|
-
* `@rudderjs/broadcast` is a runtime soft-import — pilotiq has no hard
|
|
10
|
-
* dependency on it. When the host app hasn't installed broadcast (or
|
|
11
|
-
* the provider hasn't booted), `push()` resolves to a clean no-op so
|
|
12
|
-
* apps that opt into the database surface only stay quiet.
|
|
13
|
-
*
|
|
14
|
-
* Channel naming convention: `private-pilotiq-notifications.${userId}`.
|
|
15
|
-
* Auth callbacks register at panel boot (`registerBroadcastAuth` in
|
|
16
|
-
* `routes.ts`) and gate subscriptions on
|
|
17
|
-
* `pilotiq.resolveUser(req).id === channel.userId`.
|
|
18
|
-
*/
|
|
19
|
-
|
|
20
|
-
import type { DatabaseNotificationMeta } from './database.js'
|
|
21
|
-
|
|
22
|
-
/** Internal cached snapshot of `@rudderjs/broadcast`'s public surface.
|
|
23
|
-
* We only need the `broadcast` fn from the package — auth registration
|
|
24
|
-
* goes through the same module via a separate path in `routes.ts`. */
|
|
25
|
-
interface BroadcastModule {
|
|
26
|
-
broadcast(channel: string, event: string, data: unknown): void
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
let _testModule: BroadcastModule | null | 'unset' = 'unset'
|
|
30
|
-
|
|
31
|
-
/** Soft-resolve the broadcast module. Returns `null` when the package
|
|
32
|
-
* isn't installed OR when `BroadcastingProvider` hasn't booted (the
|
|
33
|
-
* `broadcast()` fn is a no-op until then, but we still want to import
|
|
34
|
-
* the module in case the provider is wired up later in the request
|
|
35
|
-
* lifecycle). */
|
|
36
|
-
async function loadBroadcast(): Promise<BroadcastModule | null> {
|
|
37
|
-
if (_testModule !== 'unset') return _testModule
|
|
38
|
-
const moduleName = '@rudderjs/broadcast'
|
|
39
|
-
try {
|
|
40
|
-
const mod = await import(/* @vite-ignore */ moduleName) as Partial<BroadcastModule>
|
|
41
|
-
if (typeof mod.broadcast !== 'function') return null
|
|
42
|
-
return mod as BroadcastModule
|
|
43
|
-
} catch {
|
|
44
|
-
return null
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
/** Build the private channel name for a recipient. Mirrors the constant
|
|
49
|
-
* in `routes.ts` so the boot-time auth registration and the runtime
|
|
50
|
-
* push agree on the shape. Exported so the client renderer's
|
|
51
|
-
* channel-name calculation can import the same helper. */
|
|
52
|
-
export function notificationChannel(userId: string | number): string {
|
|
53
|
-
return `private-pilotiq-notifications.${String(userId)}`
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
/** Standard event name we push for new-row events. The bell client
|
|
57
|
-
* subscribes to this event and re-fetches the list when it fires. */
|
|
58
|
-
export const NOTIFICATION_CREATED_EVENT = 'notification.created'
|
|
59
|
-
|
|
60
|
-
export interface PushOptions {
|
|
61
|
-
/** Recipient identity — typically the user.id used as `notifiable_id`. */
|
|
62
|
-
recipientId: string | number
|
|
63
|
-
/** Optional override for the channel name (advanced). Defaults to
|
|
64
|
-
* `notificationChannel(recipientId)`. */
|
|
65
|
-
channel?: string
|
|
66
|
-
/** Optional override for the event name. Defaults to
|
|
67
|
-
* `NOTIFICATION_CREATED_EVENT`. */
|
|
68
|
-
event?: string
|
|
69
|
-
/** The payload to push. Typically the same `DatabaseNotificationMeta`
|
|
70
|
-
* shape returned by the list endpoint, plus any free-form keys. */
|
|
71
|
-
payload: DatabaseNotificationMeta | Record<string, unknown>
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
/** Push a notification event to the recipient's private channel. Soft-fails
|
|
75
|
-
* when `@rudderjs/broadcast` isn't installed — apps that haven't enabled
|
|
76
|
-
* broadcast still work; the bell falls back to polling. */
|
|
77
|
-
export async function push(opts: PushOptions): Promise<{ ok: boolean }> {
|
|
78
|
-
const mod = await loadBroadcast()
|
|
79
|
-
if (!mod) return { ok: false }
|
|
80
|
-
const channel = opts.channel ?? notificationChannel(opts.recipientId)
|
|
81
|
-
const event = opts.event ?? NOTIFICATION_CREATED_EVENT
|
|
82
|
-
try {
|
|
83
|
-
mod.broadcast(channel, event, opts.payload)
|
|
84
|
-
return { ok: true }
|
|
85
|
-
} catch {
|
|
86
|
-
return { ok: false }
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
/** @internal — test seam. Inject a fake broadcast module (or `null` to
|
|
91
|
-
* mimic "package not installed"). Pass `undefined` to clear and fall
|
|
92
|
-
* back to the dynamic import. */
|
|
93
|
-
export function _setTestBroadcast(mod: BroadcastModule | null | undefined): void {
|
|
94
|
-
_testModule = mod === undefined ? 'unset' : mod
|
|
95
|
-
}
|
|
@@ -1,383 +0,0 @@
|
|
|
1
|
-
import { describe, it, beforeEach, afterEach } from 'node:test'
|
|
2
|
-
import assert from 'node:assert/strict'
|
|
3
|
-
|
|
4
|
-
import {
|
|
5
|
-
listForUser,
|
|
6
|
-
unreadCount,
|
|
7
|
-
markAsRead,
|
|
8
|
-
markAsUnread,
|
|
9
|
-
markAllAsRead,
|
|
10
|
-
persist,
|
|
11
|
-
_setTestAdapter,
|
|
12
|
-
_internal,
|
|
13
|
-
} from './database.js'
|
|
14
|
-
import { Notification } from './Notification.js'
|
|
15
|
-
|
|
16
|
-
// ─── Fake ORM adapter ──────────────────────────────────────
|
|
17
|
-
|
|
18
|
-
interface Row { [k: string]: unknown }
|
|
19
|
-
interface FakeStore {
|
|
20
|
-
rows: Row[]
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
function makeFakeAdapter() {
|
|
24
|
-
const store: FakeStore = { rows: [] }
|
|
25
|
-
|
|
26
|
-
const buildQB = (_table: string) => {
|
|
27
|
-
const filters: Array<(r: Row) => boolean> = []
|
|
28
|
-
let order: { column: string; dir: 'ASC' | 'DESC' } | null = null
|
|
29
|
-
|
|
30
|
-
const apply = () => {
|
|
31
|
-
let out = store.rows.slice()
|
|
32
|
-
for (const f of filters) out = out.filter(f)
|
|
33
|
-
if (order) {
|
|
34
|
-
const { column, dir } = order
|
|
35
|
-
out.sort((a, b) => {
|
|
36
|
-
const av = (a[column] ?? '') as string
|
|
37
|
-
const bv = (b[column] ?? '') as string
|
|
38
|
-
const cmp = av < bv ? -1 : av > bv ? 1 : 0
|
|
39
|
-
return dir === 'DESC' ? -cmp : cmp
|
|
40
|
-
})
|
|
41
|
-
}
|
|
42
|
-
return out
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
const qb: any = {
|
|
46
|
-
// database.ts only uses the 2-arg `where(col, value)` shape; null
|
|
47
|
-
// is a valid value (matches `read_at IS NULL`).
|
|
48
|
-
where: (col: string, value: unknown) => {
|
|
49
|
-
filters.push(r => r[col] === value)
|
|
50
|
-
return qb
|
|
51
|
-
},
|
|
52
|
-
orderBy: (column: string, dir: 'ASC' | 'DESC' = 'ASC') => {
|
|
53
|
-
order = { column, dir }
|
|
54
|
-
return qb
|
|
55
|
-
},
|
|
56
|
-
paginate: async (_page: number, perPage = 25) => {
|
|
57
|
-
const out = apply()
|
|
58
|
-
return { data: out.slice(0, perPage), total: out.length }
|
|
59
|
-
},
|
|
60
|
-
count: async () => apply().length,
|
|
61
|
-
updateAll: async (data: Row) => {
|
|
62
|
-
const matching = apply()
|
|
63
|
-
for (const r of matching) {
|
|
64
|
-
for (const k of Object.keys(data)) r[k] = data[k]
|
|
65
|
-
}
|
|
66
|
-
return matching.length
|
|
67
|
-
},
|
|
68
|
-
create: async (data: Row) => {
|
|
69
|
-
store.rows.push({ ...data })
|
|
70
|
-
return { ...data }
|
|
71
|
-
},
|
|
72
|
-
}
|
|
73
|
-
return qb
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
const adapter = {
|
|
77
|
-
query: <T>(table: string) => buildQB(table),
|
|
78
|
-
}
|
|
79
|
-
return { adapter, store }
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
// ─── Helpers ───────────────────────────────────────────────
|
|
83
|
-
|
|
84
|
-
function seed(store: { rows: Row[] }, ...rows: Partial<Row>[]) {
|
|
85
|
-
for (const r of rows) {
|
|
86
|
-
store.rows.push({
|
|
87
|
-
id: `row-${store.rows.length + 1}`,
|
|
88
|
-
notifiable_id: '1',
|
|
89
|
-
notifiable_type: 'users',
|
|
90
|
-
type: 'PilotiqNotification',
|
|
91
|
-
data: JSON.stringify({ title: 'Test' }),
|
|
92
|
-
read_at: null,
|
|
93
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
94
|
-
updated_at: new Date(2026, 0, 1).toISOString(),
|
|
95
|
-
...r,
|
|
96
|
-
})
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
// ─── Tests ─────────────────────────────────────────────────
|
|
101
|
-
|
|
102
|
-
describe('notifications/database — _internal.rowToMeta', () => {
|
|
103
|
-
it('parses JSON data column + maps known keys', () => {
|
|
104
|
-
const meta = _internal.rowToMeta({
|
|
105
|
-
id: 'r1',
|
|
106
|
-
notifiable_id: '1',
|
|
107
|
-
notifiable_type: 'users',
|
|
108
|
-
type: 'X',
|
|
109
|
-
data: JSON.stringify({
|
|
110
|
-
title: 'Hi',
|
|
111
|
-
body: 'b',
|
|
112
|
-
icon: 'check',
|
|
113
|
-
url: '/x',
|
|
114
|
-
type: 'success',
|
|
115
|
-
irrelevant: 'ignored',
|
|
116
|
-
}),
|
|
117
|
-
read_at: null,
|
|
118
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
119
|
-
updated_at: new Date(2026, 0, 1).toISOString(),
|
|
120
|
-
})
|
|
121
|
-
assert.equal(meta.id, 'r1')
|
|
122
|
-
assert.equal(meta.title, 'Hi')
|
|
123
|
-
assert.equal(meta.body, 'b')
|
|
124
|
-
assert.equal(meta.icon, 'check')
|
|
125
|
-
assert.equal(meta.url, '/x')
|
|
126
|
-
assert.equal(meta.type, 'success')
|
|
127
|
-
assert.equal(meta.readAt, undefined)
|
|
128
|
-
})
|
|
129
|
-
|
|
130
|
-
it('rowToMeta surfaces readAt when present', () => {
|
|
131
|
-
const at = new Date(2026, 0, 2).toISOString()
|
|
132
|
-
const meta = _internal.rowToMeta({
|
|
133
|
-
id: 'r1',
|
|
134
|
-
notifiable_id: '1',
|
|
135
|
-
notifiable_type: 'users',
|
|
136
|
-
type: 'X',
|
|
137
|
-
data: '{}',
|
|
138
|
-
read_at: at,
|
|
139
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
140
|
-
updated_at: at,
|
|
141
|
-
})
|
|
142
|
-
assert.equal(meta.readAt, at)
|
|
143
|
-
})
|
|
144
|
-
|
|
145
|
-
it('rowToMeta tolerates malformed JSON in data column', () => {
|
|
146
|
-
const meta = _internal.rowToMeta({
|
|
147
|
-
id: 'r1',
|
|
148
|
-
notifiable_id: '1',
|
|
149
|
-
notifiable_type: 'users',
|
|
150
|
-
type: 'X',
|
|
151
|
-
data: 'not json',
|
|
152
|
-
read_at: null,
|
|
153
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
154
|
-
updated_at: new Date(2026, 0, 1).toISOString(),
|
|
155
|
-
})
|
|
156
|
-
assert.equal(meta.title, '') // missing title becomes empty string
|
|
157
|
-
})
|
|
158
|
-
|
|
159
|
-
it('rowToMeta drops unknown type values', () => {
|
|
160
|
-
const meta = _internal.rowToMeta({
|
|
161
|
-
id: 'r1',
|
|
162
|
-
notifiable_id: '1',
|
|
163
|
-
notifiable_type: 'users',
|
|
164
|
-
type: 'X',
|
|
165
|
-
data: JSON.stringify({ title: 'Hi', type: 'critical' }),
|
|
166
|
-
read_at: null,
|
|
167
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
168
|
-
updated_at: new Date(2026, 0, 1).toISOString(),
|
|
169
|
-
})
|
|
170
|
-
assert.equal(meta.type, undefined)
|
|
171
|
-
})
|
|
172
|
-
|
|
173
|
-
it('parses a valid actions array', () => {
|
|
174
|
-
const meta = _internal.rowToMeta({
|
|
175
|
-
id: 'r1',
|
|
176
|
-
notifiable_id: '1',
|
|
177
|
-
notifiable_type: 'users',
|
|
178
|
-
type: 'X',
|
|
179
|
-
data: JSON.stringify({
|
|
180
|
-
title: 'Hi',
|
|
181
|
-
actions: [
|
|
182
|
-
{ name: 'view', label: 'View', url: '/p/1', markAsRead: true },
|
|
183
|
-
{ name: 'archive', label: 'Archive', handler: 'archive-project', payload: { projectId: 1 } },
|
|
184
|
-
],
|
|
185
|
-
}),
|
|
186
|
-
read_at: null,
|
|
187
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
188
|
-
updated_at: new Date(2026, 0, 1).toISOString(),
|
|
189
|
-
})
|
|
190
|
-
assert.equal(meta.actions?.length, 2)
|
|
191
|
-
assert.equal(meta.actions?.[0]?.url, '/p/1')
|
|
192
|
-
assert.equal(meta.actions?.[0]?.markAsRead, true)
|
|
193
|
-
assert.equal(meta.actions?.[1]?.handler, 'archive-project')
|
|
194
|
-
assert.deepEqual(meta.actions?.[1]?.payload, { projectId: 1 })
|
|
195
|
-
})
|
|
196
|
-
|
|
197
|
-
it('drops actions that fail validation (non-string name, missing dispatch, etc)', async () => {
|
|
198
|
-
// Suppress console.warn during the test — we know we're feeding bad data.
|
|
199
|
-
const orig = console.warn
|
|
200
|
-
console.warn = () => {}
|
|
201
|
-
try {
|
|
202
|
-
const meta = _internal.rowToMeta({
|
|
203
|
-
id: 'r1',
|
|
204
|
-
notifiable_id: '1',
|
|
205
|
-
notifiable_type: 'users',
|
|
206
|
-
type: 'X',
|
|
207
|
-
data: JSON.stringify({
|
|
208
|
-
title: 'Hi',
|
|
209
|
-
actions: [
|
|
210
|
-
{ name: 'good', label: 'Good', url: '/x' },
|
|
211
|
-
{ name: 'bad-no-dispatch', label: 'Bad' },
|
|
212
|
-
{ /* no name */ label: 'Also bad', url: '/y' },
|
|
213
|
-
{ name: 'two-modes', label: 'Bad 2', url: '/y', post: '/z' },
|
|
214
|
-
],
|
|
215
|
-
}),
|
|
216
|
-
read_at: null,
|
|
217
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
218
|
-
updated_at: new Date(2026, 0, 1).toISOString(),
|
|
219
|
-
})
|
|
220
|
-
assert.equal(meta.actions?.length, 1)
|
|
221
|
-
assert.equal(meta.actions?.[0]?.name, 'good')
|
|
222
|
-
} finally {
|
|
223
|
-
console.warn = orig
|
|
224
|
-
}
|
|
225
|
-
})
|
|
226
|
-
|
|
227
|
-
it('omits the actions key when none present', () => {
|
|
228
|
-
const meta = _internal.rowToMeta({
|
|
229
|
-
id: 'r1',
|
|
230
|
-
notifiable_id: '1',
|
|
231
|
-
notifiable_type: 'users',
|
|
232
|
-
type: 'X',
|
|
233
|
-
data: JSON.stringify({ title: 'Hi' }),
|
|
234
|
-
read_at: null,
|
|
235
|
-
created_at: new Date(2026, 0, 1).toISOString(),
|
|
236
|
-
updated_at: new Date(2026, 0, 1).toISOString(),
|
|
237
|
-
})
|
|
238
|
-
assert.equal(meta.actions, undefined)
|
|
239
|
-
})
|
|
240
|
-
})
|
|
241
|
-
|
|
242
|
-
describe('notifications/database — store unavailable', () => {
|
|
243
|
-
beforeEach(() => _setTestAdapter(null))
|
|
244
|
-
afterEach(() => _setTestAdapter(undefined))
|
|
245
|
-
|
|
246
|
-
it('listForUser returns an empty result', async () => {
|
|
247
|
-
const r = await listForUser({ notifiableType: 'users', notifiableId: '1' })
|
|
248
|
-
assert.deepEqual(r, { notifications: [], unreadCount: 0 })
|
|
249
|
-
})
|
|
250
|
-
|
|
251
|
-
it('unreadCount returns 0', async () => {
|
|
252
|
-
assert.equal(await unreadCount({ notifiableType: 'users', notifiableId: '1' }), 0)
|
|
253
|
-
})
|
|
254
|
-
|
|
255
|
-
it('markAsRead / markAsUnread return false', async () => {
|
|
256
|
-
assert.equal(await markAsRead('1', { notifiableType: 'users', notifiableId: '1' }), false)
|
|
257
|
-
assert.equal(await markAsUnread('1', { notifiableType: 'users', notifiableId: '1' }), false)
|
|
258
|
-
})
|
|
259
|
-
|
|
260
|
-
it('markAllAsRead returns 0', async () => {
|
|
261
|
-
assert.equal(await markAllAsRead({ notifiableType: 'users', notifiableId: '1' }), 0)
|
|
262
|
-
})
|
|
263
|
-
|
|
264
|
-
it('persist throws — caller asked to write but no adapter', async () => {
|
|
265
|
-
await assert.rejects(
|
|
266
|
-
persist({ notifiableType: 'users', notifiableId: '1', data: { title: 't' } as any }),
|
|
267
|
-
/no @rudderjs\/orm adapter/i,
|
|
268
|
-
)
|
|
269
|
-
})
|
|
270
|
-
})
|
|
271
|
-
|
|
272
|
-
describe('notifications/database — store wired', () => {
|
|
273
|
-
let store: { rows: Row[] }
|
|
274
|
-
|
|
275
|
-
beforeEach(() => {
|
|
276
|
-
const { adapter, store: s } = makeFakeAdapter()
|
|
277
|
-
store = s
|
|
278
|
-
_setTestAdapter(adapter)
|
|
279
|
-
})
|
|
280
|
-
afterEach(() => _setTestAdapter(undefined))
|
|
281
|
-
|
|
282
|
-
it('persist inserts a row + returns id', async () => {
|
|
283
|
-
const r = await persist({
|
|
284
|
-
notifiableType: 'users',
|
|
285
|
-
notifiableId: '7',
|
|
286
|
-
data: { id: 'placeholder', title: 'Saved', type: 'success' } as any,
|
|
287
|
-
})
|
|
288
|
-
assert.match(r.id, /^pn_/)
|
|
289
|
-
assert.equal(store.rows.length, 1)
|
|
290
|
-
const row = store.rows[0]!
|
|
291
|
-
assert.equal(row['notifiable_id'], '7')
|
|
292
|
-
assert.equal(row['notifiable_type'], 'users')
|
|
293
|
-
assert.equal(row['read_at'], null)
|
|
294
|
-
assert.match(row['data'] as string, /Saved/)
|
|
295
|
-
})
|
|
296
|
-
|
|
297
|
-
it('Notification.sendToDatabase round-trips through persist', async () => {
|
|
298
|
-
const { id } = await Notification.make('Hi')
|
|
299
|
-
.body('b').success().url('/x')
|
|
300
|
-
.sendToDatabase({ id: 42 })
|
|
301
|
-
assert.match(id, /^pn_/)
|
|
302
|
-
assert.equal(store.rows.length, 1)
|
|
303
|
-
const row = store.rows[0]!
|
|
304
|
-
const data = JSON.parse(row['data'] as string)
|
|
305
|
-
assert.equal(data.title, 'Hi')
|
|
306
|
-
assert.equal(data.body, 'b')
|
|
307
|
-
assert.equal(data.url, '/x')
|
|
308
|
-
assert.equal(data.type, 'success')
|
|
309
|
-
// recipient id coerced to string
|
|
310
|
-
assert.equal(row['notifiable_id'], '42')
|
|
311
|
-
})
|
|
312
|
-
|
|
313
|
-
it('listForUser scopes by notifiable + returns unread count', async () => {
|
|
314
|
-
seed(store,
|
|
315
|
-
{ id: 'a', notifiable_id: '1', read_at: null, data: JSON.stringify({ title: 'unread-1' }) },
|
|
316
|
-
{ id: 'b', notifiable_id: '1', read_at: new Date().toISOString(), data: JSON.stringify({ title: 'read-1' }) },
|
|
317
|
-
{ id: 'c', notifiable_id: '2', read_at: null, data: JSON.stringify({ title: 'someone-else' }) },
|
|
318
|
-
)
|
|
319
|
-
const r = await listForUser({ notifiableType: 'users', notifiableId: '1' })
|
|
320
|
-
assert.equal(r.notifications.length, 2)
|
|
321
|
-
assert.equal(r.unreadCount, 1)
|
|
322
|
-
const titles = r.notifications.map(n => n.title).sort()
|
|
323
|
-
assert.deepEqual(titles, ['read-1', 'unread-1'])
|
|
324
|
-
})
|
|
325
|
-
|
|
326
|
-
it('listForUser unreadOnly filters out read rows', async () => {
|
|
327
|
-
seed(store,
|
|
328
|
-
{ id: 'a', notifiable_id: '1', read_at: null, data: JSON.stringify({ title: 'A' }) },
|
|
329
|
-
{ id: 'b', notifiable_id: '1', read_at: new Date().toISOString(), data: JSON.stringify({ title: 'B' }) },
|
|
330
|
-
)
|
|
331
|
-
const r = await listForUser({ notifiableType: 'users', notifiableId: '1', unreadOnly: true })
|
|
332
|
-
assert.equal(r.notifications.length, 1)
|
|
333
|
-
assert.equal(r.notifications[0]!.title, 'A')
|
|
334
|
-
})
|
|
335
|
-
|
|
336
|
-
it('markAsRead stamps read_at', async () => {
|
|
337
|
-
seed(store, { id: 'a', notifiable_id: '1', read_at: null })
|
|
338
|
-
const updated = await markAsRead('a', { notifiableType: 'users', notifiableId: '1' })
|
|
339
|
-
assert.equal(updated, true)
|
|
340
|
-
assert.notEqual(store.rows[0]!['read_at'], null)
|
|
341
|
-
})
|
|
342
|
-
|
|
343
|
-
it('markAsRead refuses cross-user access', async () => {
|
|
344
|
-
seed(store, { id: 'a', notifiable_id: '1', read_at: null })
|
|
345
|
-
const updated = await markAsRead('a', { notifiableType: 'users', notifiableId: '999' })
|
|
346
|
-
assert.equal(updated, false)
|
|
347
|
-
assert.equal(store.rows[0]!['read_at'], null)
|
|
348
|
-
})
|
|
349
|
-
|
|
350
|
-
it('markAsUnread clears read_at', async () => {
|
|
351
|
-
seed(store, { id: 'a', notifiable_id: '1', read_at: new Date().toISOString() })
|
|
352
|
-
const updated = await markAsUnread('a', { notifiableType: 'users', notifiableId: '1' })
|
|
353
|
-
assert.equal(updated, true)
|
|
354
|
-
assert.equal(store.rows[0]!['read_at'], null)
|
|
355
|
-
})
|
|
356
|
-
|
|
357
|
-
it('markAllAsRead stamps every unread row + leaves read rows alone', async () => {
|
|
358
|
-
const earlier = new Date(2025, 0, 1).toISOString()
|
|
359
|
-
seed(store,
|
|
360
|
-
{ id: 'a', notifiable_id: '1', read_at: null },
|
|
361
|
-
{ id: 'b', notifiable_id: '1', read_at: null },
|
|
362
|
-
{ id: 'c', notifiable_id: '1', read_at: earlier },
|
|
363
|
-
{ id: 'd', notifiable_id: '2', read_at: null },
|
|
364
|
-
)
|
|
365
|
-
const n = await markAllAsRead({ notifiableType: 'users', notifiableId: '1' })
|
|
366
|
-
assert.equal(n, 2)
|
|
367
|
-
assert.notEqual(store.rows[0]!['read_at'], null)
|
|
368
|
-
assert.notEqual(store.rows[1]!['read_at'], null)
|
|
369
|
-
assert.equal(store.rows[2]!['read_at'], earlier) // unchanged
|
|
370
|
-
assert.equal(store.rows[3]!['read_at'], null) // someone else's row
|
|
371
|
-
})
|
|
372
|
-
|
|
373
|
-
it('unreadCount returns only unread rows for the notifiable', async () => {
|
|
374
|
-
seed(store,
|
|
375
|
-
{ id: 'a', notifiable_id: '1', read_at: null },
|
|
376
|
-
{ id: 'b', notifiable_id: '1', read_at: null },
|
|
377
|
-
{ id: 'c', notifiable_id: '1', read_at: new Date().toISOString() },
|
|
378
|
-
{ id: 'd', notifiable_id: '2', read_at: null },
|
|
379
|
-
)
|
|
380
|
-
const count = await unreadCount({ notifiableType: 'users', notifiableId: '1' })
|
|
381
|
-
assert.equal(count, 2)
|
|
382
|
-
})
|
|
383
|
-
})
|