ruby_cms 0.2.1.1 → 1.0.0
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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +31 -75
- data/README.md +293 -158
- data/Rakefile +0 -18
- data/SYNC_RULES.md +84 -0
- data/exe/ruby_cms +13 -18
- data/lib/generators/ruby_cms/admin_page_generator.rb +116 -60
- data/lib/generators/ruby_cms/templates/admin.html.erb +35 -17
- data/lib/generators/ruby_cms/templates/admin_page/_admin_table_content.html.erb.tt +25 -0
- data/lib/generators/ruby_cms/templates/admin_page/_form.html.erb.tt +36 -0
- data/lib/generators/ruby_cms/templates/admin_page/_row.html.erb.tt +20 -0
- data/lib/generators/ruby_cms/templates/admin_page/controller.rb.tt +74 -2
- data/lib/generators/ruby_cms/templates/admin_page/edit.html.erb.tt +14 -0
- data/lib/generators/ruby_cms/templates/admin_page/index.html.erb.tt +21 -10
- data/lib/generators/ruby_cms/templates/admin_page/new.html.erb.tt +14 -0
- data/lib/generators/ruby_cms/templates/assets/tailwind/application.css +225 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/accordion/accordion.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/accordion/accordion_content.rb +21 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/accordion/accordion_default_content.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/accordion/accordion_default_trigger.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/accordion/accordion_icon.rb +38 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/accordion/accordion_item.rb +28 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/accordion/accordion_trigger.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/admin_page/admin_page.rb +31 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/admin_page/admin_page_header.rb +65 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/admin_page/admin_resource_card.rb +28 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/admin_page/admin_table_content.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/alert/alert.rb +36 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/alert/alert_description.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/alert/alert_title.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/aspect_ratio/aspect_ratio.rb +33 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/avatar/avatar.rb +31 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/avatar/avatar_fallback.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/avatar/avatar_image.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/badge/badge.rb +60 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/base.rb +29 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/box/box.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/breadcrumb/breadcrumb.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/breadcrumb/breadcrumb_ellipsis.rb +39 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/breadcrumb/breadcrumb_item.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/breadcrumb/breadcrumb_link.rb +22 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/breadcrumb/breadcrumb_list.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/breadcrumb/breadcrumb_page.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/breadcrumb/breadcrumb_separator.rb +38 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_button.rb +82 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table.rb +133 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_actions.rb +104 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_body.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_checkbox_cell.rb +40 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_checkbox_head.rb +34 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_delete_modal.rb +43 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_header.rb +42 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_pagination.rb +115 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_action_table_row.rb +77 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/bulk_action_table/bulk_actions.rb +86 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/button/button.rb +113 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar.rb +39 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar_body.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar_days.rb +104 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar_header.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar_next.rb +43 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar_prev.rb +43 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar_title.rb +27 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/calendar/calendar_weekdays.rb +33 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/card/card.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/card/card_content.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/card/card_description.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/card/card_footer.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/card/card_header.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/card/card_title.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/carousel/carousel.rb +44 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/carousel/carousel_content.rb +23 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/carousel/carousel_item.rb +23 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/carousel/carousel_next.rb +48 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/carousel/carousel_previous.rb +49 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/checkbox/checkbox.rb +29 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/checkbox/checkbox_group.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/clipboard/clipboard.rb +42 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/clipboard/clipboard_popover.rb +40 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/clipboard/clipboard_source.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/clipboard/clipboard_trigger.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/codeblock/codeblock.rb +102 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/collapsible/collapsible.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/collapsible/collapsible_content.rb +18 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/collapsible/collapsible_trigger.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_checkbox.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_empty_state.rb +21 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_item.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_list.rb +18 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_list_group.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_popover.rb +30 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_radio.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_search_input.rb +53 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_toggle_all_checkbox.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/combobox/combobox_trigger.rb +57 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command.rb +9 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_dialog.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_dialog_content.rb +48 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_dialog_trigger.rb +29 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_empty.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_group.rb +40 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_input.rb +56 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_item.rb +32 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/command/command_list.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/container/container.rb +33 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/container/container_section.rb +31 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/context_menu/context_menu.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/context_menu/context_menu_content.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/context_menu/context_menu_item.rb +66 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/context_menu/context_menu_label.rb +24 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/context_menu/context_menu_separator.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/context_menu/context_menu_trigger.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table.rb +29 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_bulk_actions.rb +18 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_column_toggle.rb +62 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_expand_toggle.rb +53 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_form.rb +39 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_kaminari_adapter.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_manual_adapter.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_pagination.rb +100 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_pagination_bar.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_pagy_adapter.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_per_page_select.rb +35 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_row_checkbox.rb +30 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_search.rb +57 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_select_all_checkbox.rb +21 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_selection_summary.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_sort_head.rb +112 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/data_table/data_table_toolbar.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog_content.rb +78 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog_description.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog_footer.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog_header.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog_middle.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog_title.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dialog/dialog_trigger.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dropdown_menu/dropdown_menu.rb +35 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dropdown_menu/dropdown_menu_content.rb +44 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dropdown_menu/dropdown_menu_item.rb +28 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dropdown_menu/dropdown_menu_label.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dropdown_menu/dropdown_menu_separator.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/dropdown_menu/dropdown_menu_trigger.rb +18 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/empty_state/empty_state.rb +190 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/form/form.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/form/form_field.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/form/form_field_error.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/form/form_field_hint.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/form/form_field_label.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/hover_card/hover_card.rb +27 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/hover_card/hover_card_content.rb +22 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/hover_card/hover_card_trigger.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/input/input.rb +34 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/link/link.rb +106 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/masked_input/masked_input.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal_action.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal_cancel.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal_content.rb +68 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal_description.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal_footer.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal_header.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/modal/modal_title.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/native_select/native_select.rb +39 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/native_select/native_select_group.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/native_select/native_select_icon.rb +39 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/native_select/native_select_option.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/pagination/pagination.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/pagination/pagination_content.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/pagination/pagination_ellipsis.rb +42 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/pagination/pagination_item.rb +28 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/popover/popover.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/popover/popover_content.rb +27 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/popover/popover_trigger.rb +20 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/progress/progress.rb +37 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/radio_button/radio_button.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select.rb +23 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select_content.rb +32 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select_group.rb +15 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select_input.rb +22 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select_item.rb +52 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select_label.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select_trigger.rb +54 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/select/select_value.rb +27 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/separator/separator.rb +38 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet_content.rb +77 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet_description.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet_footer.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet_header.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet_middle.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet_title.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/sheet/sheet_trigger.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/shortcut_key/shortcut_key.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/skeleton/skeleton.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/switch/switch.rb +24 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table_body.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table_caption.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table_cell.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table_footer.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table_head.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table_header.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/table/table_row.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/tabs/tabs.rb +25 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/tabs/tabs_content.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/tabs/tabs_list.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/tabs/tabs_trigger.rb +28 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/textarea/textarea.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/theme_toggle/set_dark_mode.rb +16 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/theme_toggle/set_light_mode.rb +16 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/theme_toggle/theme_toggle.rb +9 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/tooltip/tooltip.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/tooltip/tooltip_content.rb +26 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/tooltip/tooltip_trigger.rb +19 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/typography/heading.rb +60 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/typography/inline_code.rb +17 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/typography/inline_link.rb +22 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/typography/text.rb +53 -0
- data/lib/generators/ruby_cms/templates/components/ruby_ui/typography/typography_blockquote.rb +17 -0
- data/lib/generators/ruby_cms/templates/{ruby_cms.rb → config/initializers/admin.rb} +15 -9
- data/lib/generators/ruby_cms/templates/config/initializers/admin_dashboard.rb +43 -0
- data/lib/generators/ruby_cms/templates/config/initializers/pagy.rb +7 -0
- data/lib/generators/ruby_cms/templates/config/initializers/ruby_cms_core.rb +25 -0
- data/lib/generators/ruby_cms/templates/config/initializers/ruby_cms_custom_settings.rb +35 -0
- data/lib/generators/ruby_cms/templates/config/initializers/webauthn.rb +28 -0
- data/lib/generators/ruby_cms/templates/config/locales/en.yml +277 -0
- data/lib/generators/ruby_cms/templates/config/locales/nl.yml +329 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/analytics_controller.rb +211 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/application_controller.rb +186 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/audit_log_entries_controller.rb +69 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/commands_controller.rb +146 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/content_block_versions_controller.rb +96 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/content_blocks_controller.rb +405 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/dashboard_controller.rb +114 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/invitations_controller.rb +64 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/locale_controller.rb +19 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/media_assets_controller.rb +186 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/notifications_controller.rb +37 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/passkey_credentials_controller.rb +17 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/passkey_registrations_controller.rb +50 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/permissions_controller.rb +132 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/redirects_controller.rb +175 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/security_controller.rb +146 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/settings_controller.rb +298 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/system_health_controller.rb +11 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/trash_controller.rb +61 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/user_permissions_controller.rb +91 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/users_controller.rb +220 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/visitor_errors_controller.rb +114 -0
- data/lib/generators/ruby_cms/templates/controllers/admin/visual_editor_controller.rb +384 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/.keep +0 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/admin_bulk_actions.rb +59 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/admin_pagination.rb +106 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/admin_turbo_table.rb +46 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/audit_loggable.rb +28 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/authentication.rb +63 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/page_tracking.rb +45 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/security_monitoring.rb +102 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/sudo_mode.rb +29 -0
- data/lib/generators/ruby_cms/templates/controllers/concerns/visitor_error_capture.rb +35 -0
- data/lib/generators/ruby_cms/templates/controllers/errors_controller.rb +36 -0
- data/lib/generators/ruby_cms/templates/controllers/passkey_sessions_controller.rb +69 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20251121085414_create_email_blocklists.rb +12 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20251121161828_create_ip_blocklists.rb +12 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20251205120000_add_analytics_indices.rb +48 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20251216064753_backfill_security_fields_in_ahoy_events.rb +49 -0
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260125000001_create_ruby_cms_permissions.rb +2 -2
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260125000002_create_ruby_cms_user_permissions.rb +2 -2
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260125000003_create_ruby_cms_content_blocks.rb +3 -3
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260125000010_add_indexes_to_ruby_cms_tables.rb +1 -1
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260127000001_add_locale_to_ruby_cms_content_blocks.rb +8 -6
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260129000001_create_ruby_cms_visitor_errors.rb +4 -4
- data/lib/generators/ruby_cms/templates/db/migrate/20260130000001_add_referer_and_query_to_ruby_cms_visitor_errors.rb +10 -0
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260130000002_create_ruby_cms_preferences.rb +2 -2
- data/lib/generators/ruby_cms/templates/db/migrate/20260130000003_add_category_to_ruby_cms_preferences.rb +10 -0
- data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260328000001_create_content_block_versions.rb +2 -2
- data/lib/generators/ruby_cms/templates/db/migrate/20260525120000_create_audit_log_entries.rb +24 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260525130000_create_redirects.rb +28 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260525140000_create_media_assets.rb +25 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260525150000_create_invitations.rb +22 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260525160000_create_command_runs.rb +22 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260525190000_add_state_to_visitor_errors.rb +20 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260527165017_add_discarded_at_to_visitor_errors.rb +10 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260527165728_add_discarded_at_to_cms_tables.rb +12 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260605151935_create_passkey_credentials.rb +15 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260605151936_add_webauthn_id_to_users.rb +22 -0
- data/lib/generators/ruby_cms/templates/db/migrate/20260606000001_add_authenticated_at_to_sessions.rb +14 -0
- data/lib/generators/ruby_cms/templates/helpers/admin/admin_page_helper.rb +9 -0
- data/lib/generators/ruby_cms/templates/helpers/admin/dashboard_helper.rb +16 -0
- data/lib/generators/ruby_cms/templates/helpers/admin/media_assets_helper.rb +38 -0
- data/lib/generators/ruby_cms/templates/helpers/admin/relative_time_helper.rb +37 -0
- data/lib/generators/ruby_cms/templates/helpers/admin/ui_helper.rb +48 -0
- data/lib/generators/ruby_cms/templates/helpers/cms_application_helpers.rb +70 -0
- data/lib/generators/ruby_cms/templates/helpers/content_blocks_helper.rb +382 -0
- data/lib/generators/ruby_cms/templates/helpers/settings_helper.rb +192 -0
- data/lib/generators/ruby_cms/templates/javascript/admin.js +6 -0
- data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/admin}/admin_commands_controller.js +2 -2
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/autosearch_controller.js +18 -0
- data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/admin}/clickable_row_controller.js +11 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/clipboard_controller.js +34 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/command_palette_controller.js +189 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/commands_controller.js +300 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/content_block_history_controller.js +167 -0
- data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/admin}/flash_messages_controller.js +1 -1
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/inline_rich_editor_controller.js +157 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/media_selection_controller.js +76 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/media_uploader_controller.js +23 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/mobile_menu_controller.js +125 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/nav_order_sortable_controller.js +296 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/nav_shortcuts_controller.js +47 -0
- data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/admin}/page_preview_controller.js +64 -3
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/permission_matrix_controller.js +49 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/search_shortcut_controller.js +24 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/settings_rail_search_controller.js +17 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/shortcuts_overlay_controller.js +39 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/sidebar_collapse_controller.js +135 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/sidebar_nav_persist_controller.js +48 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/sidebar_toggle_trigger_controller.js +10 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/topbar_controller.js +88 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/visual_editor_controller.js +721 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/admin/visual_editor_page_select_controller.js +174 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/application.js +10 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/index.js +2 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/accordion_controller.js +97 -0
- data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui}/bulk_action_table_controller.js +86 -233
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/calendar_controller.js +249 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/calendar_input_controller.js +8 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/carousel_controller.js +60 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/checkbox_group_controller.js +21 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/clipboard_controller.js +54 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/collapsible_controller.js +47 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/combobox_controller.js +190 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/command_controller.js +138 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/context_menu_controller.js +144 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/data_table_column_visibility_controller.js +14 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/data_table_controller.js +109 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/data_table_search_controller.js +62 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/dialog_controller.js +32 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/dropdown_menu_controller.js +149 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/form_field_controller.js +61 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/hover_card_controller.js +153 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/masked_input_controller.js +9 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/modal_controller.js +45 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/popover_controller.js +107 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/select_controller.js +192 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/select_item_controller.js +11 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/sheet_content_controller.js +7 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/sheet_controller.js +9 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/tabs_controller.js +67 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/theme_toggle_controller.js +30 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/ruby_ui/tooltip_controller.js +38 -0
- data/lib/generators/ruby_cms/templates/javascript/controllers/webauthn_controller.js +153 -0
- data/lib/generators/ruby_cms/templates/lib/ruby_cms/cli.rb +170 -0
- data/lib/generators/ruby_cms/templates/lib/ruby_cms/commands_registry.rb +76 -0
- data/lib/{ruby_cms → generators/ruby_cms/templates/lib/ruby_cms}/content_blocks_grouping.rb +2 -6
- data/lib/{ruby_cms → generators/ruby_cms/templates/lib/ruby_cms}/content_blocks_sync.rb +6 -8
- data/lib/{ruby_cms → generators/ruby_cms/templates/lib/ruby_cms}/dashboard_blocks.rb +5 -5
- data/lib/{ruby_cms → generators/ruby_cms/templates/lib/ruby_cms}/icons.rb +41 -10
- data/lib/{ruby_cms → generators/ruby_cms/templates/lib/ruby_cms}/settings.rb +60 -10
- data/lib/{ruby_cms → generators/ruby_cms/templates/lib/ruby_cms}/settings_registry.rb +79 -3
- data/lib/generators/ruby_cms/templates/lib/ruby_cms.rb +342 -0
- data/lib/{tasks → generators/ruby_cms/templates/lib/tasks}/admin.rake +16 -16
- data/lib/generators/ruby_cms/templates/lib/tasks/ruby_cms.rake +158 -0
- data/lib/generators/ruby_cms/templates/models/audit_log_entry.rb +56 -0
- data/lib/generators/ruby_cms/templates/models/command_run.rb +29 -0
- data/lib/generators/ruby_cms/templates/models/concerns/content_block/searchable.rb +18 -0
- data/{app → lib/generators/ruby_cms/templates}/models/concerns/content_block/versionable.rb +1 -1
- data/lib/generators/ruby_cms/templates/models/concerns/ransackable.rb +32 -0
- data/{app → lib/generators/ruby_cms/templates}/models/content_block.rb +13 -5
- data/lib/generators/ruby_cms/templates/models/email_blocklist.rb +21 -0
- data/lib/generators/ruby_cms/templates/models/invitation.rb +86 -0
- data/lib/generators/ruby_cms/templates/models/ip_blocklist.rb +34 -0
- data/lib/generators/ruby_cms/templates/models/media_asset.rb +90 -0
- data/lib/generators/ruby_cms/templates/models/passkey_credential.rb +6 -0
- data/lib/generators/ruby_cms/templates/models/permission.rb +71 -0
- data/lib/generators/ruby_cms/templates/models/permittable.rb +69 -0
- data/lib/generators/ruby_cms/templates/models/preference.rb +109 -0
- data/lib/generators/ruby_cms/templates/models/redirect.rb +100 -0
- data/lib/generators/ruby_cms/templates/models/user_permission.rb +25 -0
- data/lib/generators/ruby_cms/templates/models/visitor_error.rb +157 -0
- data/lib/generators/ruby_cms/templates/nav/analytics.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/audit_log.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/commands.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/content_blocks.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/core.rb +13 -0
- data/lib/generators/ruby_cms/templates/nav/media.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/permissions.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/redirects.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/security.rb +4 -0
- data/lib/generators/ruby_cms/templates/nav/system_health.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/trash.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/users.rb +3 -0
- data/lib/generators/ruby_cms/templates/nav/visual_editor.rb +3 -0
- data/lib/generators/ruby_cms/templates/routes/analytics.rb +5 -0
- data/lib/generators/ruby_cms/templates/routes/audit_log.rb +3 -0
- data/lib/generators/ruby_cms/templates/routes/auth.public.rb +4 -0
- data/lib/generators/ruby_cms/templates/routes/auth.rb +6 -0
- data/lib/generators/ruby_cms/templates/routes/commands.rb +4 -0
- data/lib/generators/ruby_cms/templates/routes/content_blocks.rb +12 -0
- data/lib/generators/ruby_cms/templates/routes/core.public.rb +5 -0
- data/lib/generators/ruby_cms/templates/routes/core.rb +16 -0
- data/lib/generators/ruby_cms/templates/routes/media.rb +9 -0
- data/lib/generators/ruby_cms/templates/routes/passkeys.public.rb +4 -0
- data/lib/generators/ruby_cms/templates/routes/passkeys.rb +5 -0
- data/lib/generators/ruby_cms/templates/routes/permissions.rb +5 -0
- data/lib/generators/ruby_cms/templates/routes/redirects.rb +10 -0
- data/lib/generators/ruby_cms/templates/routes/security.rb +20 -0
- data/lib/generators/ruby_cms/templates/routes/system_health.rb +3 -0
- data/lib/generators/ruby_cms/templates/routes/trash.rb +5 -0
- data/lib/generators/ruby_cms/templates/routes/users.rb +10 -0
- data/lib/generators/ruby_cms/templates/routes/visual_editor.rb +5 -0
- data/lib/generators/ruby_cms/templates/services/admin/health_check.rb +362 -0
- data/lib/generators/ruby_cms/templates/services/analytics/report.rb +503 -0
- data/lib/generators/ruby_cms/templates/services/analytics_service.rb +305 -0
- data/lib/generators/ruby_cms/templates/services/audit_log.rb +60 -0
- data/lib/generators/ruby_cms/templates/services/command_runner.rb +79 -0
- data/lib/generators/ruby_cms/templates/services/security_service.rb +203 -0
- data/lib/generators/ruby_cms/templates/services/security_tracker.rb +90 -0
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/index.html.erb +61 -50
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/page_details.html.erb +15 -10
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_browser_device.html.erb +2 -2
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_conversions.html.erb +3 -3
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_daily_activity_chart.html.erb +18 -17
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_exit_pages.html.erb +4 -4
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_hourly_activity_chart.html.erb +18 -17
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_landing_pages.html.erb +3 -3
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_os_stats.html.erb +2 -2
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_popular_pages.html.erb +4 -4
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_recent_activity.html.erb +4 -4
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_top_referrers.html.erb +3 -3
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_top_visitors.html.erb +4 -4
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_utm_sources.html.erb +2 -2
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/visitor_details.html.erb +15 -10
- data/lib/generators/ruby_cms/templates/views/admin/audit_log_entries/_admin_table_content.html.erb +55 -0
- data/lib/generators/ruby_cms/templates/views/admin/audit_log_entries/_row.html.erb +84 -0
- data/lib/generators/ruby_cms/templates/views/admin/audit_log_entries/index.html.erb +33 -0
- data/lib/generators/ruby_cms/templates/views/admin/audit_log_entries/show.html.erb +89 -0
- data/lib/generators/ruby_cms/templates/views/admin/commands/index.html.erb +393 -0
- data/lib/generators/ruby_cms/templates/views/admin/content_block_versions/index.html.erb +63 -0
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/content_block_versions/show.html.erb +2 -2
- data/lib/generators/ruby_cms/templates/views/admin/content_blocks/_audits.html.erb +49 -0
- data/lib/generators/ruby_cms/templates/views/admin/content_blocks/_form.html.erb +79 -0
- data/lib/generators/ruby_cms/templates/views/admin/content_blocks/_rich_field.html.erb +44 -0
- data/lib/generators/ruby_cms/templates/views/admin/content_blocks/_row.html.erb +60 -0
- data/lib/generators/ruby_cms/templates/views/admin/content_blocks/index.html.erb +84 -0
- data/lib/generators/ruby_cms/templates/views/admin/content_blocks/new.html.erb +18 -0
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/content_blocks/show.html.erb +67 -61
- data/lib/generators/ruby_cms/templates/views/admin/dashboard/_gmail_token_status.html.erb +69 -0
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/dashboard/blocks/_analytics_overview.html.erb +11 -11
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/dashboard/blocks/_content_blocks_stats.html.erb +3 -3
- data/lib/generators/ruby_cms/templates/views/admin/dashboard/blocks/_notifications.html.erb +50 -0
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/dashboard/blocks/_permissions_stats.html.erb +3 -3
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/dashboard/blocks/_quick_actions.html.erb +13 -13
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/dashboard/blocks/_recent_errors.html.erb +8 -8
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/dashboard/blocks/_users_stats.html.erb +3 -3
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/dashboard/blocks/_visitor_errors_stats.html.erb +3 -3
- data/lib/generators/ruby_cms/templates/views/admin/dashboard/index.html.erb +34 -0
- data/lib/generators/ruby_cms/templates/views/admin/media_assets/_admin_table_content.html.erb +107 -0
- data/lib/generators/ruby_cms/templates/views/admin/media_assets/_form.html.erb +33 -0
- data/lib/generators/ruby_cms/templates/views/admin/media_assets/_grid.html.erb +58 -0
- data/lib/generators/ruby_cms/templates/views/admin/media_assets/_list.html.erb +43 -0
- data/lib/generators/ruby_cms/templates/views/admin/media_assets/edit.html.erb +13 -0
- data/lib/generators/ruby_cms/templates/views/admin/media_assets/index.html.erb +36 -0
- data/lib/generators/ruby_cms/templates/views/admin/media_assets/show.html.erb +97 -0
- data/lib/generators/ruby_cms/templates/views/admin/passkey_credentials/index.html.erb +30 -0
- data/lib/generators/ruby_cms/templates/views/admin/passkey_registrations/new.html.erb +23 -0
- data/lib/generators/ruby_cms/templates/views/admin/permissions/_admin_table_content.html.erb +45 -0
- data/lib/generators/ruby_cms/templates/views/admin/permissions/_matrix.html.erb +75 -0
- data/lib/generators/ruby_cms/templates/views/admin/permissions/_row.html.erb +62 -0
- data/lib/generators/ruby_cms/templates/views/admin/permissions/index.html.erb +65 -0
- data/lib/generators/ruby_cms/templates/views/admin/redirects/_admin_table_content.html.erb +77 -0
- data/lib/generators/ruby_cms/templates/views/admin/redirects/_form.html.erb +97 -0
- data/lib/generators/ruby_cms/templates/views/admin/redirects/_row.html.erb +82 -0
- data/lib/generators/ruby_cms/templates/views/admin/redirects/edit.html.erb +21 -0
- data/lib/generators/ruby_cms/templates/views/admin/redirects/index.html.erb +40 -0
- data/lib/generators/ruby_cms/templates/views/admin/redirects/new.html.erb +12 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/index.html.erb +88 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/ip_blocklist_content.html.erb +3 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/partials/_events_by_type.html.erb +24 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/partials/_ip_blocklist.html.erb +37 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/partials/_ip_blocklist_content.html.erb +41 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/partials/_ip_management.html.erb +11 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/partials/_recent_events.html.erb +59 -0
- data/lib/generators/ruby_cms/templates/views/admin/security/partials/_top_threat_ips.html.erb +21 -0
- data/lib/generators/ruby_cms/templates/views/admin/settings/index.html.erb +509 -0
- data/lib/generators/ruby_cms/templates/views/admin/shared/_add_button.html.erb +15 -0
- data/lib/generators/ruby_cms/templates/views/admin/shared/_data_table.html.erb +255 -0
- data/lib/generators/ruby_cms/templates/views/admin/shared/_turbo_pagination.html.erb +53 -0
- data/lib/generators/ruby_cms/templates/views/admin/system_health/index.html.erb +248 -0
- data/lib/generators/ruby_cms/templates/views/admin/trash/index.html.erb +58 -0
- data/lib/generators/ruby_cms/templates/views/admin/user_permissions/_row.html.erb +29 -0
- data/lib/generators/ruby_cms/templates/views/admin/user_permissions/index.html.erb +78 -0
- data/lib/generators/ruby_cms/templates/views/admin/users/_admin_table_content.html.erb +50 -0
- data/lib/generators/ruby_cms/templates/views/admin/users/_invite_form.html.erb +43 -0
- data/lib/generators/ruby_cms/templates/views/admin/users/_invites_panel.html.erb +91 -0
- data/lib/generators/ruby_cms/templates/views/admin/users/_row.html.erb +109 -0
- data/lib/generators/ruby_cms/templates/views/admin/users/index.html.erb +43 -0
- data/lib/generators/ruby_cms/templates/views/admin/users/show.html.erb +343 -0
- data/lib/generators/ruby_cms/templates/views/admin/visitor_errors/_admin_table_content.html.erb +61 -0
- data/lib/generators/ruby_cms/templates/views/admin/visitor_errors/_row.html.erb +90 -0
- data/lib/generators/ruby_cms/templates/views/admin/visitor_errors/index.html.erb +18 -0
- data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/visitor_errors/show.html.erb +4 -4
- data/lib/generators/ruby_cms/templates/views/admin/visual_editor/index.html.erb +383 -0
- data/{app/views/layouts/ruby_cms → lib/generators/ruby_cms/templates/views/layouts/admin}/_admin_flash_messages.html.erb +3 -3
- data/{app/views/layouts/ruby_cms → lib/generators/ruby_cms/templates/views/layouts/admin}/_admin_sidebar.html.erb +47 -48
- data/lib/generators/ruby_cms/templates/views/layouts/admin/_command_palette.html.erb +65 -0
- data/lib/generators/ruby_cms/templates/views/layouts/admin/_shortcuts_overlay.html.erb +81 -0
- data/lib/generators/ruby_cms/templates/views/layouts/admin/_topbar_widgets.html.erb +9 -0
- data/lib/generators/ruby_cms/templates/views/layouts/admin/minimal.html.erb +71 -0
- data/lib/generators/ruby_cms/templates/views/layouts/admin/topbar/_bell.html.erb +81 -0
- data/lib/generators/ruby_cms/templates/views/layouts/admin/topbar/_profile_menu.html.erb +86 -0
- data/lib/generators/ruby_cms/templates/views/layouts/admin/topbar/_search_trigger.html.erb +9 -0
- data/lib/generators/ruby_cms/templates/views/layouts/admin/topbar/_shortcuts_button.html.erb +8 -0
- data/lib/generators/ruby_cms/templates/views/shared/_maintenance_banner.html.erb +10 -0
- data/lib/ruby_cms/app_wiring.rb +56 -0
- data/lib/ruby_cms/auth_wiring.rb +64 -0
- data/lib/ruby_cms/cli.rb +181 -121
- data/lib/ruby_cms/excludes.rb +41 -0
- data/lib/ruby_cms/file_installer.rb +41 -0
- data/lib/ruby_cms/gem_setup.rb +84 -0
- data/lib/ruby_cms/helper_wiring.rb +32 -0
- data/lib/ruby_cms/importmap_wiring.rb +41 -0
- data/lib/ruby_cms/installer.rb +128 -0
- data/lib/ruby_cms/lockfile.rb +49 -0
- data/lib/ruby_cms/manifest.rb +88 -0
- data/lib/ruby_cms/manifest_data.rb +233 -0
- data/lib/ruby_cms/migration_installer.rb +64 -0
- data/lib/ruby_cms/nav_assembler.rb +43 -0
- data/lib/ruby_cms/passkey_wiring.rb +42 -0
- data/lib/ruby_cms/path_map.rb +26 -0
- data/lib/ruby_cms/routes_assembler.rb +103 -0
- data/lib/ruby_cms/syncer.rb +81 -0
- data/lib/ruby_cms/updater.rb +76 -0
- data/lib/ruby_cms/version.rb +1 -1
- data/lib/ruby_cms.rb +4 -328
- metadata +606 -170
- data/.cursor/dhh.mdc +0 -698
- data/app/components/ruby_cms/admin/admin_page/admin_table_content.rb +0 -32
- data/app/components/ruby_cms/admin/admin_page.rb +0 -345
- data/app/components/ruby_cms/admin/admin_page_header.rb +0 -79
- data/app/components/ruby_cms/admin/admin_resource_card.rb +0 -55
- data/app/components/ruby_cms/admin/base_component.rb +0 -78
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table.rb +0 -157
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_actions.rb +0 -127
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_body.rb +0 -15
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_checkbox_cell.rb +0 -43
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_checkbox_head.rb +0 -35
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_delete_modal.rb +0 -174
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_header.rb +0 -59
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_header_bar.rb +0 -169
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_pagination.rb +0 -192
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_action_table_row.rb +0 -96
- data/app/components/ruby_cms/admin/bulk_action_table/bulk_actions.rb +0 -196
- data/app/controllers/concerns/ruby_cms/admin_pagination.rb +0 -120
- data/app/controllers/concerns/ruby_cms/admin_turbo_table.rb +0 -68
- data/app/controllers/concerns/ruby_cms/page_tracking.rb +0 -67
- data/app/controllers/concerns/ruby_cms/visitor_error_capture.rb +0 -39
- data/app/controllers/ruby_cms/admin/analytics_controller.rb +0 -213
- data/app/controllers/ruby_cms/admin/base_controller.rb +0 -132
- data/app/controllers/ruby_cms/admin/commands_controller.rb +0 -122
- data/app/controllers/ruby_cms/admin/content_block_versions_controller.rb +0 -62
- data/app/controllers/ruby_cms/admin/content_blocks_controller.rb +0 -391
- data/app/controllers/ruby_cms/admin/dashboard_controller.rb +0 -89
- data/app/controllers/ruby_cms/admin/locale_controller.rb +0 -21
- data/app/controllers/ruby_cms/admin/permissions_controller.rb +0 -66
- data/app/controllers/ruby_cms/admin/settings_controller.rb +0 -223
- data/app/controllers/ruby_cms/admin/user_permissions_controller.rb +0 -77
- data/app/controllers/ruby_cms/admin/users_controller.rb +0 -107
- data/app/controllers/ruby_cms/admin/visitor_errors_controller.rb +0 -89
- data/app/controllers/ruby_cms/admin/visual_editor_controller.rb +0 -345
- data/app/controllers/ruby_cms/errors_controller.rb +0 -35
- data/app/helpers/ruby_cms/admin/admin_page_helper.rb +0 -21
- data/app/helpers/ruby_cms/admin/bulk_action_table_helper.rb +0 -159
- data/app/helpers/ruby_cms/admin/dashboard_helper.rb +0 -20
- data/app/helpers/ruby_cms/application_helper.rb +0 -49
- data/app/helpers/ruby_cms/bulk_action_table_helper.rb +0 -151
- data/app/helpers/ruby_cms/content_blocks_helper.rb +0 -399
- data/app/helpers/ruby_cms/settings_helper.rb +0 -172
- data/app/javascript/controllers/ruby_cms/content_block_history_controller.js +0 -91
- data/app/javascript/controllers/ruby_cms/index.js +0 -117
- data/app/javascript/controllers/ruby_cms/mobile_menu_controller.js +0 -55
- data/app/javascript/controllers/ruby_cms/nav_order_sortable_controller.js +0 -192
- data/app/javascript/controllers/ruby_cms/visual_editor_controller.js +0 -325
- data/app/models/concerns/content_block/searchable.rb +0 -22
- data/app/models/ruby_cms/content_block.rb +0 -8
- data/app/models/ruby_cms/permission.rb +0 -57
- data/app/models/ruby_cms/permittable.rb +0 -37
- data/app/models/ruby_cms/preference.rb +0 -111
- data/app/models/ruby_cms/user_permission.rb +0 -12
- data/app/models/ruby_cms/visitor_error.rb +0 -109
- data/app/services/ruby_cms/analytics/report.rb +0 -512
- data/app/services/ruby_cms/command_runner.rb +0 -42
- data/app/services/ruby_cms/security_tracker.rb +0 -92
- data/app/views/admin/content_block_versions/index.html.erb +0 -52
- data/app/views/admin/content_block_versions/show.html.erb +0 -37
- data/app/views/layouts/ruby_cms/admin.html.erb +0 -77
- data/app/views/layouts/ruby_cms/minimal.html.erb +0 -181
- data/app/views/ruby_cms/_tailwind_safelist.html.erb +0 -2
- data/app/views/ruby_cms/admin/commands/index.html.erb +0 -104
- data/app/views/ruby_cms/admin/content_block_versions/index.html.erb +0 -52
- data/app/views/ruby_cms/admin/content_blocks/_form.html.erb +0 -92
- data/app/views/ruby_cms/admin/content_blocks/_row.html.erb +0 -25
- data/app/views/ruby_cms/admin/content_blocks/index.html.erb +0 -62
- data/app/views/ruby_cms/admin/content_blocks/new.html.erb +0 -10
- data/app/views/ruby_cms/admin/dashboard/index.html.erb +0 -31
- data/app/views/ruby_cms/admin/permissions/_row.html.erb +0 -11
- data/app/views/ruby_cms/admin/permissions/index.html.erb +0 -62
- data/app/views/ruby_cms/admin/settings/index.html.erb +0 -268
- data/app/views/ruby_cms/admin/shared/_bulk_action_table_index.html.erb +0 -56
- data/app/views/ruby_cms/admin/user_permissions/index.html.erb +0 -85
- data/app/views/ruby_cms/admin/users/_row.html.erb +0 -17
- data/app/views/ruby_cms/admin/users/index.html.erb +0 -70
- data/app/views/ruby_cms/admin/visitor_errors/_row.html.erb +0 -35
- data/app/views/ruby_cms/admin/visitor_errors/index.html.erb +0 -57
- data/app/views/ruby_cms/admin/visual_editor/index.html.erb +0 -157
- data/app/views/ruby_cms/errors/not_found.html.erb +0 -92
- data/config/database.yml +0 -6
- data/config/importmap.rb +0 -40
- data/config/locales/en.yml +0 -198
- data/config/locales/nl.yml +0 -156
- data/config/routes.rb +0 -76
- data/db/migrate/20260130000001_add_referer_and_query_to_ruby_cms_visitor_errors.rb +0 -8
- data/db/migrate/20260130000003_add_category_to_ruby_cms_preferences.rb +0 -8
- data/docs/assets/ruby_cms_logo.png +0 -0
- data/lib/generators/ruby_cms/install_generator.rb +0 -1159
- data/lib/ruby_cms/app_integration.rb +0 -82
- data/lib/ruby_cms/commands_registry.rb +0 -40
- data/lib/ruby_cms/css_compiler.rb +0 -35
- data/lib/ruby_cms/engine/admin_permissions.rb +0 -69
- data/lib/ruby_cms/engine/content_blocks_tasks.rb +0 -66
- data/lib/ruby_cms/engine/css.rb +0 -14
- data/lib/ruby_cms/engine/dashboard_registration.rb +0 -66
- data/lib/ruby_cms/engine/navigation_registration.rb +0 -90
- data/lib/ruby_cms/engine.rb +0 -273
- /data/{app → lib/generators/ruby_cms/templates}/assets/images/ruby_cms/logo.png +0 -0
- /data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260211000001_add_ruby_cms_analytics_fields_to_ahoy_events.rb +0 -0
- /data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260212000001_use_unprefixed_cms_tables.rb +0 -0
- /data/{db → lib/generators/ruby_cms/templates/db}/migrate/20260409000001_add_analytics_performance_indexes.rb +0 -0
- /data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/admin}/auto_save_preference_controller.js +0 -0
- /data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/admin}/locale_tabs_controller.js +0 -0
- /data/{app/javascript/controllers/ruby_cms → lib/generators/ruby_cms/templates/javascript/controllers/admin}/toggle_controller.js +0 -0
- /data/{app → lib/generators/ruby_cms/templates}/models/concerns/content_block/publishable.rb +0 -0
- /data/{app → lib/generators/ruby_cms/templates}/models/content_block_version.rb +0 -0
- /data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_back_button.html.erb +0 -0
- /data/{app/views/ruby_cms → lib/generators/ruby_cms/templates/views}/admin/analytics/partials/_security_alert.html.erb +0 -0
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
# Include in controllers to track page views via Ahoy.
|
|
4
|
+
# Requires Ahoy to be installed (via RubyCMS install generator).
|
|
5
|
+
#
|
|
6
|
+
# Usage:
|
|
7
|
+
# class PagesController < ApplicationController
|
|
8
|
+
# include PageTracking
|
|
9
|
+
# end
|
|
10
|
+
#
|
|
11
|
+
# Sets @page_name to controller_name by default. Override in actions:
|
|
12
|
+
# @page_name = "custom_page_name"
|
|
13
|
+
#
|
|
14
|
+
module PageTracking
|
|
15
|
+
extend ActiveSupport::Concern
|
|
16
|
+
|
|
17
|
+
included do
|
|
18
|
+
before_action :set_page_name
|
|
19
|
+
after_action :track_page_view
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
private
|
|
23
|
+
|
|
24
|
+
def set_page_name
|
|
25
|
+
@page_name = controller_name if @page_name.blank?
|
|
26
|
+
end
|
|
27
|
+
|
|
28
|
+
def track_page_view
|
|
29
|
+
return unless should_track_page_view?
|
|
30
|
+
|
|
31
|
+
ahoy.track "page_view",
|
|
32
|
+
page_name: @page_name,
|
|
33
|
+
request_path: request.path
|
|
34
|
+
rescue StandardError => e
|
|
35
|
+
Rails.logger.error "[RubyCMS] Failed to track page view: #{e.message}"
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
def should_track_page_view?
|
|
39
|
+
return false if @page_name.blank?
|
|
40
|
+
return false if request.path.start_with?("/admin")
|
|
41
|
+
return false if request.headers["Turbo-Frame"].present?
|
|
42
|
+
|
|
43
|
+
true
|
|
44
|
+
end
|
|
45
|
+
end
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
module SecurityMonitoring
|
|
4
|
+
extend ActiveSupport::Concern
|
|
5
|
+
|
|
6
|
+
included do
|
|
7
|
+
before_action :log_request_info, unless: -> { skip_security_monitoring? }
|
|
8
|
+
end
|
|
9
|
+
|
|
10
|
+
private
|
|
11
|
+
|
|
12
|
+
def skip_security_monitoring?
|
|
13
|
+
path = request.path
|
|
14
|
+
return true if [ "/", "/up" ].include?(path)
|
|
15
|
+
return true if path.start_with?("/assets", "/packs", "/favicon", "/robots.txt", "/service-worker", "/manifest", "/pwa", "/sitemap")
|
|
16
|
+
return true if path.match?(/\.(css|js|map|png|jpe?g|gif|svg|ico|webp|txt)\z/i)
|
|
17
|
+
|
|
18
|
+
false
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
def log_request_info
|
|
22
|
+
log_suspicious_user_agent if suspicious_user_agent?
|
|
23
|
+
log_unusual_request_pattern if unusual_request_pattern?
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
def suspicious_user_agent?
|
|
27
|
+
ua = request.user_agent.to_s.downcase
|
|
28
|
+
suspicious_patterns = %w[nikto sqlmap nmap burp crawler bot]
|
|
29
|
+
suspicious_patterns.any? { |p| ua.include?(p) }
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
def unusual_request_pattern?
|
|
33
|
+
path = request.path.to_s.downcase
|
|
34
|
+
return false if path == "/"
|
|
35
|
+
|
|
36
|
+
query = request.query_string.to_s.downcase
|
|
37
|
+
suspicious_patterns = %w[wp-admin wp-login.php .php .env .git/HEAD /etc/passwd <script> union select]
|
|
38
|
+
suspicious_patterns.any? { |p| path.include?(p) || query.include?(p) }
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
def log_suspicious_user_agent
|
|
42
|
+
return if recently_logged?("suspicious_user_agent")
|
|
43
|
+
return if IpBlocklist.blocked?(request.remote_ip)
|
|
44
|
+
|
|
45
|
+
Rails.logger.warn "Suspicious User-Agent detected: #{request.user_agent} from IP: #{request.remote_ip}"
|
|
46
|
+
|
|
47
|
+
SecurityTracker.track(
|
|
48
|
+
"suspicious_user_agent",
|
|
49
|
+
description: "Suspicious user agent detected",
|
|
50
|
+
request: request
|
|
51
|
+
)
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
def log_unusual_request_pattern
|
|
55
|
+
return if recently_logged?("unusual_request_pattern")
|
|
56
|
+
return if IpBlocklist.blocked?(request.remote_ip)
|
|
57
|
+
|
|
58
|
+
Rails.logger.warn "Unusual request pattern detected: #{request.method} #{request.path} from IP: #{request.remote_ip}"
|
|
59
|
+
|
|
60
|
+
SecurityTracker.track(
|
|
61
|
+
"unusual_request_pattern",
|
|
62
|
+
description: "Unusual request pattern detected",
|
|
63
|
+
request: request
|
|
64
|
+
)
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
def log_security_event(event_type, description, user_id: nil)
|
|
68
|
+
user = user_id ? User.find_by(id: user_id) : nil
|
|
69
|
+
SecurityTracker.track(
|
|
70
|
+
event_type,
|
|
71
|
+
description:,
|
|
72
|
+
user:,
|
|
73
|
+
request:
|
|
74
|
+
)
|
|
75
|
+
AuditLog.record(
|
|
76
|
+
action: audit_action_for(event_type),
|
|
77
|
+
actor: user || :system,
|
|
78
|
+
target: "Session",
|
|
79
|
+
summary: description,
|
|
80
|
+
request: request,
|
|
81
|
+
meta: { event_type: }
|
|
82
|
+
)
|
|
83
|
+
end
|
|
84
|
+
|
|
85
|
+
def audit_action_for(event_type)
|
|
86
|
+
case event_type.to_s
|
|
87
|
+
when "successful_login" then :login_success
|
|
88
|
+
when "failed_login" then :login_failed
|
|
89
|
+
when "unauthorized_admin_attempt" then :login_failed_unauthorized
|
|
90
|
+
when "logout" then :login_logout
|
|
91
|
+
else :"session_#{event_type}"
|
|
92
|
+
end
|
|
93
|
+
end
|
|
94
|
+
|
|
95
|
+
def recently_logged?(type, window: 5.minutes)
|
|
96
|
+
key = "secev:#{type}:#{request.remote_ip}"
|
|
97
|
+
return true if Rails.cache.exist?(key)
|
|
98
|
+
|
|
99
|
+
Rails.cache.write(key, true, expires_in: window)
|
|
100
|
+
false
|
|
101
|
+
end
|
|
102
|
+
end
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
# Requires a recent authentication before sensitive actions such as managing passkeys.
|
|
4
|
+
module SudoMode
|
|
5
|
+
extend ActiveSupport::Concern
|
|
6
|
+
|
|
7
|
+
# How long a fresh sign-in keeps sudo mode "active" before re-auth is required.
|
|
8
|
+
SUDO_WINDOW = 30.minutes
|
|
9
|
+
|
|
10
|
+
class_methods do
|
|
11
|
+
def require_recent_authentication(**options)
|
|
12
|
+
before_action :require_recent_authentication, **options
|
|
13
|
+
end
|
|
14
|
+
end
|
|
15
|
+
|
|
16
|
+
private
|
|
17
|
+
|
|
18
|
+
def require_recent_authentication
|
|
19
|
+
return if sudo_active?
|
|
20
|
+
|
|
21
|
+
session[:return_to_after_authenticating] = request.url
|
|
22
|
+
redirect_to new_session_path, alert: "Confirm your password to continue."
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
def sudo_active?
|
|
26
|
+
authenticated_at = Current.session&.try(:authenticated_at)
|
|
27
|
+
authenticated_at.present? && authenticated_at > SUDO_WINDOW.ago
|
|
28
|
+
end
|
|
29
|
+
end
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
# Include in ApplicationController to capture public-site errors to VisitorError.
|
|
4
|
+
# Skips admin controllers (paths under /admin) and development environment by default.
|
|
5
|
+
#
|
|
6
|
+
# Usage in ApplicationController:
|
|
7
|
+
# include VisitorErrorCapture
|
|
8
|
+
# rescue_from StandardError, with: :handle_visitor_error
|
|
9
|
+
#
|
|
10
|
+
module VisitorErrorCapture
|
|
11
|
+
extend ActiveSupport::Concern
|
|
12
|
+
|
|
13
|
+
class_methods do
|
|
14
|
+
def install(controller_class)
|
|
15
|
+
controller_class.include VisitorErrorCapture
|
|
16
|
+
controller_class.rescue_from StandardError, with: :handle_visitor_error
|
|
17
|
+
end
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
private
|
|
21
|
+
|
|
22
|
+
def handle_visitor_error(exception)
|
|
23
|
+
return if skip_visitor_error_capture?
|
|
24
|
+
|
|
25
|
+
VisitorError.log_error(exception, request)
|
|
26
|
+
ensure
|
|
27
|
+
raise exception
|
|
28
|
+
end
|
|
29
|
+
|
|
30
|
+
def skip_visitor_error_capture?
|
|
31
|
+
return true if Rails.env.development?
|
|
32
|
+
|
|
33
|
+
false
|
|
34
|
+
end
|
|
35
|
+
end
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
# Handles 404 errors from catch-all routes.
|
|
4
|
+
# Add this to the BOTTOM of your routes.rb to capture routing errors:
|
|
5
|
+
#
|
|
6
|
+
# match "*path", to: "errors#not_found", via: :all,
|
|
7
|
+
# constraints: ->(req) { !req.path.start_with?("/rails/") }
|
|
8
|
+
#
|
|
9
|
+
class ErrorsController < ApplicationController
|
|
10
|
+
allow_unauthenticated_access only: :not_found
|
|
11
|
+
skip_before_action :verify_authenticity_token, only: :not_found
|
|
12
|
+
|
|
13
|
+
def not_found
|
|
14
|
+
# Log the routing error to VisitorError (skips in development)
|
|
15
|
+
VisitorError.log_routing_error(request)
|
|
16
|
+
|
|
17
|
+
respond_to do |format|
|
|
18
|
+
format.html do
|
|
19
|
+
render_html_not_found
|
|
20
|
+
end
|
|
21
|
+
format.json { render json: { error: "Not found" }, status: :not_found }
|
|
22
|
+
format.any { head :not_found }
|
|
23
|
+
end
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
private
|
|
27
|
+
|
|
28
|
+
def render_html_not_found
|
|
29
|
+
static_not_found = Rails.public_path.join("404.html")
|
|
30
|
+
if static_not_found.exist?
|
|
31
|
+
render file: static_not_found, status: :not_found, layout: false
|
|
32
|
+
else
|
|
33
|
+
render :not_found, status: :not_found, layout: false
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
end
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
class PasskeySessionsController < ApplicationController
|
|
4
|
+
allow_unauthenticated_access only: %i[new create]
|
|
5
|
+
# Never let a proxy/browser/service-worker cache the challenge options or result.
|
|
6
|
+
before_action { response.headers["Cache-Control"] = "no-store" }
|
|
7
|
+
|
|
8
|
+
# POST /passkey_sessions/new — issue usernameless (discoverable) request options.
|
|
9
|
+
def new
|
|
10
|
+
get_options = WebAuthn::Credential.options_for_get(user_verification: "preferred")
|
|
11
|
+
session[:passkey_authentication_challenge] = get_options.challenge
|
|
12
|
+
render json: get_options
|
|
13
|
+
end
|
|
14
|
+
|
|
15
|
+
# POST /passkey_sessions — verify the assertion and start a session.
|
|
16
|
+
def create
|
|
17
|
+
raw = params.to_unsafe_h[:credential]
|
|
18
|
+
return render(json: { error: "Missing credential" }, status: :bad_request) if raw.blank?
|
|
19
|
+
|
|
20
|
+
webauthn_credential = WebAuthn::Credential.from_get(raw)
|
|
21
|
+
stored = PasskeyCredential.find_by!(external_id: webauthn_credential.id)
|
|
22
|
+
challenge = session.delete(:passkey_authentication_challenge)
|
|
23
|
+
|
|
24
|
+
verify_assertion(webauthn_credential, stored, challenge)
|
|
25
|
+
apply_sign_count(webauthn_credential, stored)
|
|
26
|
+
stored.update!(last_used_at: Time.current)
|
|
27
|
+
|
|
28
|
+
start_new_session_for(stored.user)
|
|
29
|
+
render json: { ok: true, redirect: after_authentication_url }
|
|
30
|
+
rescue ActiveRecord::RecordNotFound, ActiveRecord::RecordInvalid, WebAuthn::Error, ArgumentError, TypeError => e
|
|
31
|
+
Rails.logger.warn("[passkey] authentication failed: #{e.class}: #{e.message}")
|
|
32
|
+
render json: { error: "Authentication failed" }, status: :unprocessable_entity
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
private
|
|
36
|
+
|
|
37
|
+
def verify_assertion(webauthn_credential, stored, challenge)
|
|
38
|
+
webauthn_credential.verify(
|
|
39
|
+
challenge,
|
|
40
|
+
public_key: stored.public_key,
|
|
41
|
+
sign_count: stored.sign_count
|
|
42
|
+
)
|
|
43
|
+
rescue WebAuthn::SignCountVerificationError
|
|
44
|
+
# Signature already verified before the count check. Synced passkeys can report a
|
|
45
|
+
# non-increasing counter — log/alert but do NOT block. apply_sign_count handles it.
|
|
46
|
+
Rails.logger.warn("[passkey] sign_count not increasing for credential #{stored.id}")
|
|
47
|
+
end
|
|
48
|
+
|
|
49
|
+
# Lenient sign_count policy (see spec). Never raises.
|
|
50
|
+
def apply_sign_count(webauthn_credential, stored)
|
|
51
|
+
new_count = webauthn_credential.sign_count
|
|
52
|
+
if new_count > stored.sign_count
|
|
53
|
+
stored.update!(sign_count: new_count)
|
|
54
|
+
elsif new_count.zero? && stored.sign_count.zero?
|
|
55
|
+
# authenticator does not implement counters — nothing to do
|
|
56
|
+
else
|
|
57
|
+
notify_suspicious_sign_count(stored, new_count)
|
|
58
|
+
end
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
def notify_suspicious_sign_count(stored, new_count)
|
|
62
|
+
Rails.logger.warn(
|
|
63
|
+
"[passkey] suspicious sign_count for user=#{stored.user_id} " \
|
|
64
|
+
"credential=#{stored.id} stored=#{stored.sign_count} got=#{new_count}"
|
|
65
|
+
)
|
|
66
|
+
# Surface via the existing Noticed pipeline without blocking login.
|
|
67
|
+
# (Logging is the guaranteed minimum.)
|
|
68
|
+
end
|
|
69
|
+
end
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
class CreateEmailBlocklists < ActiveRecord::Migration[8.1]
|
|
2
|
+
def change
|
|
3
|
+
create_table :email_blocklists, if_not_exists: true do |t|
|
|
4
|
+
t.string :email, null: false
|
|
5
|
+
t.text :note
|
|
6
|
+
|
|
7
|
+
t.timestamps
|
|
8
|
+
end
|
|
9
|
+
|
|
10
|
+
add_index :email_blocklists, :email, unique: true, if_not_exists: true
|
|
11
|
+
end
|
|
12
|
+
end
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
class CreateIpBlocklists < ActiveRecord::Migration[8.1]
|
|
2
|
+
def change
|
|
3
|
+
create_table :ip_blocklists, if_not_exists: true do |t|
|
|
4
|
+
t.string :ip_address, null: false
|
|
5
|
+
t.text :note
|
|
6
|
+
|
|
7
|
+
t.timestamps
|
|
8
|
+
end
|
|
9
|
+
|
|
10
|
+
add_index :ip_blocklists, :ip_address, unique: true, if_not_exists: true
|
|
11
|
+
end
|
|
12
|
+
end
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
class AddAnalyticsIndices < ActiveRecord::Migration[8.0]
|
|
4
|
+
def change
|
|
5
|
+
# Ahoy visits indices for analytics queries
|
|
6
|
+
add_index :ahoy_visits, :started_at, if_not_exists: true
|
|
7
|
+
add_index :ahoy_visits, :ip, if_not_exists: true
|
|
8
|
+
add_index :ahoy_visits, :referrer, if_not_exists: true
|
|
9
|
+
|
|
10
|
+
# Ahoy events indices
|
|
11
|
+
add_index :ahoy_events, :time, if_not_exists: true
|
|
12
|
+
|
|
13
|
+
# Convert properties column from text to jsonb for better PostgreSQL performance
|
|
14
|
+
# This enables GIN indexing and native JSON queries
|
|
15
|
+
# Modern Ahoy (5.x) already creates `properties` as jsonb; only convert when an
|
|
16
|
+
# older schema left it as text, otherwise the `= ''` comparison fails on jsonb.
|
|
17
|
+
reversible do |dir|
|
|
18
|
+
dir.up do
|
|
19
|
+
properties_col = columns(:ahoy_events).find { |c| c.name == "properties" }
|
|
20
|
+
if properties_col && properties_col.sql_type == "text"
|
|
21
|
+
execute <<-SQL
|
|
22
|
+
ALTER TABLE ahoy_events
|
|
23
|
+
ALTER COLUMN properties TYPE jsonb
|
|
24
|
+
USING CASE
|
|
25
|
+
WHEN properties IS NULL THEN NULL
|
|
26
|
+
WHEN properties = '' THEN '{}'::jsonb
|
|
27
|
+
ELSE properties::jsonb
|
|
28
|
+
END
|
|
29
|
+
SQL
|
|
30
|
+
end
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
dir.down do
|
|
34
|
+
properties_col = columns(:ahoy_events).find { |c| c.name == "properties" }
|
|
35
|
+
if properties_col && properties_col.sql_type == "jsonb"
|
|
36
|
+
execute <<-SQL
|
|
37
|
+
ALTER TABLE ahoy_events
|
|
38
|
+
ALTER COLUMN properties TYPE text
|
|
39
|
+
USING properties::text
|
|
40
|
+
SQL
|
|
41
|
+
end
|
|
42
|
+
end
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
# GIN index for JSONB properties (PostgreSQL specific)
|
|
46
|
+
add_index :ahoy_events, :properties, using: :gin, if_not_exists: true
|
|
47
|
+
end
|
|
48
|
+
end
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
class BackfillSecurityFieldsInAhoyEvents < ActiveRecord::Migration[8.0]
|
|
2
|
+
def up
|
|
3
|
+
say_with_time "Backfilling Ahoy::Event columns from properties and clearing properties..." do
|
|
4
|
+
Ahoy::Event.find_in_batches(batch_size: 1_000) do |events|
|
|
5
|
+
events.each do |event|
|
|
6
|
+
props =
|
|
7
|
+
begin
|
|
8
|
+
case event.properties
|
|
9
|
+
when String
|
|
10
|
+
JSON.parse(event.properties)
|
|
11
|
+
when Hash
|
|
12
|
+
event.properties
|
|
13
|
+
else
|
|
14
|
+
{}
|
|
15
|
+
end
|
|
16
|
+
rescue JSON::ParserError
|
|
17
|
+
{}
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
updates = {}
|
|
21
|
+
|
|
22
|
+
# Universal mapping (works for both security + page_view events)
|
|
23
|
+
ip_from_props = props["ip_address"] || props["ip"]
|
|
24
|
+
ua_from_props = props["user_agent"]
|
|
25
|
+
path_from_props = props["request_path"] || props["path"]
|
|
26
|
+
desc_from_props = props["description"]
|
|
27
|
+
page_name_from_props = props["page_name"]
|
|
28
|
+
|
|
29
|
+
updates[:ip_address] = ip_from_props if event.ip_address.blank? && ip_from_props.present?
|
|
30
|
+
updates[:user_agent] = ua_from_props if event.user_agent.blank? && ua_from_props.present?
|
|
31
|
+
updates[:request_path] = path_from_props if event.request_path.blank? && path_from_props.present?
|
|
32
|
+
|
|
33
|
+
updates[:description] = desc_from_props if event.respond_to?(:description) && event.description.blank? && desc_from_props.present?
|
|
34
|
+
# Page views: copy page_name into its new column (if present)
|
|
35
|
+
updates[:page_name] = page_name_from_props if event.respond_to?(:page_name) && event.page_name.blank? && page_name_from_props.present?
|
|
36
|
+
|
|
37
|
+
# Stop using properties going forward: clear it once we’ve extracted whatever we can
|
|
38
|
+
updates[:properties] = nil if event.properties.present?
|
|
39
|
+
|
|
40
|
+
event.update_columns(updates) if updates.any?
|
|
41
|
+
end
|
|
42
|
+
end
|
|
43
|
+
end
|
|
44
|
+
end
|
|
45
|
+
|
|
46
|
+
def down
|
|
47
|
+
# no-op (we intentionally do not restore properties)
|
|
48
|
+
end
|
|
49
|
+
end
|
|
@@ -2,13 +2,13 @@
|
|
|
2
2
|
|
|
3
3
|
class CreateRubyCmsPermissions < ActiveRecord::Migration[7.1]
|
|
4
4
|
def change
|
|
5
|
-
create_table :ruby_cms_permissions do |t|
|
|
5
|
+
create_table :ruby_cms_permissions, if_not_exists: true do |t|
|
|
6
6
|
t.string :key, null: false
|
|
7
7
|
t.string :name
|
|
8
8
|
|
|
9
9
|
t.timestamps
|
|
10
10
|
end
|
|
11
11
|
|
|
12
|
-
add_index :ruby_cms_permissions, :key, unique: true
|
|
12
|
+
add_index :ruby_cms_permissions, :key, unique: true, if_not_exists: true
|
|
13
13
|
end
|
|
14
14
|
end
|
|
@@ -2,13 +2,13 @@
|
|
|
2
2
|
|
|
3
3
|
class CreateRubyCmsUserPermissions < ActiveRecord::Migration[7.1]
|
|
4
4
|
def change
|
|
5
|
-
create_table :ruby_cms_user_permissions do |t|
|
|
5
|
+
create_table :ruby_cms_user_permissions, if_not_exists: true do |t|
|
|
6
6
|
t.references :user, null: false, foreign_key: false
|
|
7
7
|
t.references :permission, null: false, foreign_key: { to_table: :ruby_cms_permissions }
|
|
8
8
|
|
|
9
9
|
t.timestamps
|
|
10
10
|
end
|
|
11
11
|
|
|
12
|
-
add_index :ruby_cms_user_permissions, %i[user_id permission_id], unique: true
|
|
12
|
+
add_index :ruby_cms_user_permissions, %i[user_id permission_id], unique: true, if_not_exists: true
|
|
13
13
|
end
|
|
14
14
|
end
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
class CreateRubyCmsContentBlocks < ActiveRecord::Migration[7.1]
|
|
4
4
|
def change
|
|
5
|
-
create_table :ruby_cms_content_blocks do |t|
|
|
5
|
+
create_table :ruby_cms_content_blocks, if_not_exists: true do |t|
|
|
6
6
|
t.string :key, null: false
|
|
7
7
|
t.string :title
|
|
8
8
|
t.text :content
|
|
@@ -13,7 +13,7 @@ class CreateRubyCmsContentBlocks < ActiveRecord::Migration[7.1]
|
|
|
13
13
|
t.timestamps
|
|
14
14
|
end
|
|
15
15
|
|
|
16
|
-
add_index :ruby_cms_content_blocks, :key, unique: true
|
|
17
|
-
add_index :ruby_cms_content_blocks, %i[published content_type]
|
|
16
|
+
add_index :ruby_cms_content_blocks, :key, unique: true, if_not_exists: true
|
|
17
|
+
add_index :ruby_cms_content_blocks, %i[published content_type], if_not_exists: true
|
|
18
18
|
end
|
|
19
19
|
end
|
|
@@ -4,6 +4,6 @@ class AddIndexesToRubyCmsTables < ActiveRecord::Migration[7.1]
|
|
|
4
4
|
def change
|
|
5
5
|
return if index_exists?(:ruby_cms_content_blocks, :updated_by_id)
|
|
6
6
|
|
|
7
|
-
add_index :ruby_cms_content_blocks, :updated_by_id
|
|
7
|
+
add_index :ruby_cms_content_blocks, :updated_by_id, if_not_exists: true
|
|
8
8
|
end
|
|
9
9
|
end
|
|
@@ -2,20 +2,22 @@
|
|
|
2
2
|
|
|
3
3
|
class AddLocaleToRubyCmsContentBlocks < ActiveRecord::Migration[7.1]
|
|
4
4
|
def change
|
|
5
|
-
|
|
5
|
+
return unless table_exists?(:ruby_cms_content_blocks)
|
|
6
|
+
|
|
7
|
+
unless column_exists?(:ruby_cms_content_blocks, :locale)
|
|
8
|
+
add_column :ruby_cms_content_blocks, :locale, :string, default: "en", null: false
|
|
9
|
+
end
|
|
6
10
|
|
|
7
11
|
# Remove old unique index on key
|
|
8
12
|
remove_index :ruby_cms_content_blocks, :key if index_exists?(:ruby_cms_content_blocks, :key)
|
|
9
13
|
|
|
10
14
|
# Add new composite unique index on key + locale
|
|
11
|
-
add_index :ruby_cms_content_blocks, %i[key locale], unique: true
|
|
15
|
+
add_index :ruby_cms_content_blocks, %i[key locale], unique: true, if_not_exists: true
|
|
12
16
|
|
|
13
17
|
# Add index for locale queries
|
|
14
|
-
add_index :ruby_cms_content_blocks, :locale
|
|
18
|
+
add_index :ruby_cms_content_blocks, :locale, if_not_exists: true
|
|
15
19
|
|
|
16
|
-
# Migrate existing records to default locale
|
|
17
|
-
# This ensures existing content blocks get the default locale
|
|
18
|
-
# Use execute to avoid model loading issues during migration
|
|
20
|
+
# Migrate existing records to default locale (idempotent — only touches blanks)
|
|
19
21
|
reversible do |dir|
|
|
20
22
|
dir.up do
|
|
21
23
|
default_locale = begin
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
class CreateRubyCmsVisitorErrors < ActiveRecord::Migration[7.1]
|
|
4
4
|
def change
|
|
5
|
-
create_table :ruby_cms_visitor_errors do |t|
|
|
5
|
+
create_table :ruby_cms_visitor_errors, if_not_exists: true do |t|
|
|
6
6
|
t.string :error_class, null: false
|
|
7
7
|
t.text :error_message, null: false
|
|
8
8
|
t.string :request_path, null: false
|
|
@@ -17,8 +17,8 @@ class CreateRubyCmsVisitorErrors < ActiveRecord::Migration[7.1]
|
|
|
17
17
|
t.timestamps
|
|
18
18
|
end
|
|
19
19
|
|
|
20
|
-
add_index :ruby_cms_visitor_errors, :created_at
|
|
21
|
-
add_index :ruby_cms_visitor_errors, :request_path
|
|
22
|
-
add_index :ruby_cms_visitor_errors, :resolved
|
|
20
|
+
add_index :ruby_cms_visitor_errors, :created_at, if_not_exists: true
|
|
21
|
+
add_index :ruby_cms_visitor_errors, :request_path, if_not_exists: true
|
|
22
|
+
add_index :ruby_cms_visitor_errors, :resolved, if_not_exists: true
|
|
23
23
|
end
|
|
24
24
|
end
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
class AddRefererAndQueryToRubyCmsVisitorErrors < ActiveRecord::Migration[7.1]
|
|
4
|
+
def change
|
|
5
|
+
return unless table_exists?(:ruby_cms_visitor_errors)
|
|
6
|
+
|
|
7
|
+
add_column :ruby_cms_visitor_errors, :referer, :string unless column_exists?(:ruby_cms_visitor_errors, :referer)
|
|
8
|
+
add_column :ruby_cms_visitor_errors, :query_string, :string unless column_exists?(:ruby_cms_visitor_errors, :query_string)
|
|
9
|
+
end
|
|
10
|
+
end
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
class CreateRubyCmsPreferences < ActiveRecord::Migration[7.1]
|
|
4
4
|
def change
|
|
5
|
-
create_table :ruby_cms_preferences do |t|
|
|
5
|
+
create_table :ruby_cms_preferences, if_not_exists: true do |t|
|
|
6
6
|
t.string :key, null: false
|
|
7
7
|
t.text :value
|
|
8
8
|
t.string :value_type, default: "string", null: false
|
|
@@ -11,6 +11,6 @@ class CreateRubyCmsPreferences < ActiveRecord::Migration[7.1]
|
|
|
11
11
|
t.timestamps
|
|
12
12
|
end
|
|
13
13
|
|
|
14
|
-
add_index :ruby_cms_preferences, :key, unique: true
|
|
14
|
+
add_index :ruby_cms_preferences, :key, unique: true, if_not_exists: true
|
|
15
15
|
end
|
|
16
16
|
end
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
class AddCategoryToRubyCmsPreferences < ActiveRecord::Migration[7.1]
|
|
4
|
+
def change
|
|
5
|
+
return unless table_exists?(:ruby_cms_preferences)
|
|
6
|
+
|
|
7
|
+
add_column :ruby_cms_preferences, :category, :string, default: "general" unless column_exists?(:ruby_cms_preferences, :category)
|
|
8
|
+
add_index :ruby_cms_preferences, :category, if_not_exists: true
|
|
9
|
+
end
|
|
10
|
+
end
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
class CreateContentBlockVersions < ActiveRecord::Migration[7.1]
|
|
4
4
|
def change
|
|
5
|
-
create_table :content_block_versions do |t|
|
|
5
|
+
create_table :content_block_versions, if_not_exists: true do |t|
|
|
6
6
|
t.references :content_block, null: false, foreign_key: true, index: true
|
|
7
7
|
t.references :user, null: true, foreign_key: true
|
|
8
8
|
t.integer :version_number, null: false
|
|
@@ -17,6 +17,6 @@ class CreateContentBlockVersions < ActiveRecord::Migration[7.1]
|
|
|
17
17
|
end
|
|
18
18
|
|
|
19
19
|
add_index :content_block_versions, %i[content_block_id version_number],
|
|
20
|
-
unique: true, name: "idx_cb_versions_on_block_and_number"
|
|
20
|
+
unique: true, name: "idx_cb_versions_on_block_and_number", if_not_exists: true
|
|
21
21
|
end
|
|
22
22
|
end
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
class CreateAuditLogEntries < ActiveRecord::Migration[8.0]
|
|
4
|
+
def change
|
|
5
|
+
create_table :audit_log_entries, if_not_exists: true do |t|
|
|
6
|
+
t.string :actor_email
|
|
7
|
+
t.bigint :actor_id
|
|
8
|
+
t.string :action, null: false
|
|
9
|
+
t.string :target_type
|
|
10
|
+
t.bigint :target_id
|
|
11
|
+
t.string :target_label
|
|
12
|
+
t.string :summary, limit: 500
|
|
13
|
+
t.string :ip, limit: 64
|
|
14
|
+
t.string :user_agent, limit: 500
|
|
15
|
+
t.jsonb :metadata, null: false, default: {}
|
|
16
|
+
t.datetime :created_at, null: false
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
add_index :audit_log_entries, :created_at, order: { created_at: :desc }, if_not_exists: true
|
|
20
|
+
add_index :audit_log_entries, %i[actor_email created_at], if_not_exists: true
|
|
21
|
+
add_index :audit_log_entries, :action, if_not_exists: true
|
|
22
|
+
add_index :audit_log_entries, %i[target_type target_id], if_not_exists: true
|
|
23
|
+
end
|
|
24
|
+
end
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
class CreateRedirects < ActiveRecord::Migration[8.1]
|
|
2
|
+
def change
|
|
3
|
+
# citext gives us case-insensitive uniqueness on source_path without
|
|
4
|
+
# juggling functional indexes in app code.
|
|
5
|
+
enable_extension "citext" unless extension_enabled?("citext")
|
|
6
|
+
|
|
7
|
+
create_table :redirects, if_not_exists: true do |t|
|
|
8
|
+
t.citext :source_path, null: false
|
|
9
|
+
t.string :target_path
|
|
10
|
+
t.integer :status_code, null: false, default: 301
|
|
11
|
+
t.boolean :enabled, null: false, default: true
|
|
12
|
+
t.string :note
|
|
13
|
+
t.bigint :hits, null: false, default: 0
|
|
14
|
+
t.datetime :last_hit_at
|
|
15
|
+
|
|
16
|
+
t.timestamps
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
add_index :redirects, :source_path, unique: true, if_not_exists: true
|
|
20
|
+
add_index :redirects, :enabled, if_not_exists: true
|
|
21
|
+
add_index :redirects, :last_hit_at, if_not_exists: true
|
|
22
|
+
add_index :redirects, :hits, if_not_exists: true
|
|
23
|
+
|
|
24
|
+
add_check_constraint :redirects,
|
|
25
|
+
"status_code IN (301, 302, 410)",
|
|
26
|
+
name: "redirects_status_code_check"
|
|
27
|
+
end
|
|
28
|
+
end
|