Changelog
All notable changes to Signals, ordered by version.
0.4.2 — Activities Calendar, iCal Feeds, SSO & Magic-Link Login
Added
Calendar
- Full-page Calendar at
/calendarthat visualises scheduled activities, gated on the existingactivities.accesspermission — no new entity, table, or permission - Three views — Day (staff columns against a vertical hour-axis), Week (seven day-columns against an hour-axis, coloured by owner; the default view), and Month (a seven-column grid of day-cells with event chips)
- Pixel-precise hour-axis placement in Day and Week views; the visible hour window auto-expands to include any event outside working hours so nothing is clipped
- All-day handling computed from the activity's times (starts at
00:00with no end or ending at23:59/next midnight), rendered in a band above the grid and as month chips - Pickable start date defaulting to today and looking forwards, with Today / Previous / Next navigation that steps by the active view
- Owner filter restricting the calendar to selected staff users, with deterministic per-user colours shared across columns, blocks, chips, and avatars
- Owner and participant avatars on each event (owner first, then participants), showing each user's photo where available and coloured initials otherwise
- Unscheduled tray listing activities with no start time, each opening its detail modal so it can be scheduled
- View, date, and filter state persisted in the URL so a calendar link reopens exactly as shared
Modals & Realtime
- Add modal opened from an empty slot (pre-filled owner and start time) or a month day-cell (pre-filled date), reusing the existing activity form
- Detail modal showing an activity's subject, type, status, owner, time, and location, with Complete, Edit, and Delete actions and a link to the full
/activities/{id}page - Edit modal using the same form as create, reached from the detail modal
- All writes flow through the existing
CreateActivity/UpdateActivity/CompleteActivity/DeleteActivityactions — same validation, authorisation, audit logging, andactivity.*webhooks — and the calendar refreshes instantly and locally after every change
iCal Feeds
- Secure global iCal feed at
/calendar/feed.ics(every scheduled activity) and per-user feeds at/calendar/feed/{user}.ics(a single owner's activities) - Feeds secured with Laravel signed URLs (no expiry); a missing or tampered signature returns
403, and the feeds serve without an authenticated session - Feed window covers activities from one year ago with no forward bound; activities without a start time are excluded
- Hand-rolled RFC 5545 output (no new Composer dependency) — one
VEVENTper activity withUID,DTSTAMP,DTSTART/DTENDin UTC,SUMMARY,LOCATION,DESCRIPTION,STATUS,TRANSP, andORGANIZER, with proper text escaping and 75-octet line folding; all-day events emit;VALUE=DATE - Copyable subscribe URLs surfaced in a feed modal on the calendar page (global and own feed for every user; full per-user list for administrators)
Settings
- New Settings → Calendar page at
/settings/calendarshowing the current user's personal feed: a subscribe URL with a copy button, a Download .ics button, and Google/Apple/Outlook subscription help; administrators also see the global feed URL
Navigation
- Calendar opened from the top navigation bar's calendar icon (gated on
activities.access), with Activities relocated into the CRM menu; a "Go to Calendar" command-palette navigation entry is also available
Documentation
- Platform documentation page for the Calendar (views, navigation, owner filter, modals, unscheduled tray, the settings page, and the week-start / working-hours / weekend settings that affect it)
- API documentation page for the calendar feed endpoints, covering the signed-URL security model and its global-only revocation tradeoff
Single Sign-On (SSO)
- SSO login via Google and Microsoft 365 (Azure
commontenant) powered by Laravel Socialite — staff can sign in with their Google or Microsoft account without a Signals password - Auto-link by verified email: on first SSO login the provider's verified email is matched to an existing, active Signals user and the identity is linked automatically; unknown or unverified emails are denied and the user must be invited before using SSO
- Subsequent logins match by stored provider identity (robust to later email changes in Signals)
- Self-hosted credential configuration in Settings → Integrations → Single Sign-On: per-provider enable toggle, Client ID, and Client Secret (stored encrypted at rest; write-only — not re-displayed after save), with the correct callback URL shown for copy-paste into the Google Cloud Console or Azure portal
- Signals Cloud mode: credential fields are hidden in the Integrations UI; only enable/disable toggles are shown (credentials are managed centrally by Signals Cloud)
- Per-role SSO enforcement in Settings → Security → SSO Enforcement: admins can require specific roles to use SSO; password login for enforced roles is blocked with an inline guidance message and SSO buttons; the Owner role is always exempt (break-glass — cannot be selected)
- 2FA still applies: SSO replaces only the password step; users with 2FA enabled are redirected to the existing one-time-code challenge after a successful OAuth callback
- Allowed email domains allow-list (Settings → Integrations → Single Sign-On): when set, only users whose IdP email domain is on the list may sign in or auto-link — enforced on both the auto-link and existing-link paths — strongly recommended with Microsoft's multitenant
commonsetting to prevent cross-tenant account takeover; an empty list permits any domain - Client secrets are now write-only end-to-end: they are never loaded back into the Integrations form, never rendered to the browser, and a blank secret field on save keeps the existing stored secret (enter a new value to rotate)
- Routes:
GET /auth/google/redirect,GET /auth/google/callback,GET /auth/microsoft/redirect,GET /auth/microsoft/callback; unknown providers return404 oauth_identitiestable andOAuthIdentitymodel recording the(provider, provider_id, user_id, email)link
Authentication — Password Manager & Accessibility
- One-time-code 2FA input now auto-submits when six digits are entered, removing the need to press Enter or click a button
/.well-known/change-passwordwell-known URL pointing to the profile password-change page, so browsers and password managers can locate the change-password flow automatically- Invitation acceptance page includes a username field pre-filled with the invited email, giving password managers the anchor they need to save the credential against the right account on first use
- Login and password forms now carry canonical
autocomplete="username"/autocomplete="current-password"/autocomplete="new-password"attributes throughout the auth flow for consistent password-manager integration
Documentation
- Platform documentation page for Single Sign-On — covers enabling Google and Microsoft (self-hosted and Signals Cloud), callback URL registration, auto-link matching, per-role enforcement, and 2FA interaction
Magic-Link Login
- Passwordless login for existing users via a single-use link emailed to their address — an opt-in alternative to password login, enabled in Settings → Security → Magic-Link Login (off by default)
- Links are valid for 15 minutes and can only be consumed once; requesting a new link invalidates any prior unconsumed link for that account
- 2FA is not bypassed — users with 2FA enabled (or enforced) are redirected to the existing one-time-code challenge after the link is consumed
- Anti-enumeration: Signals always returns the same neutral response ("if an account exists, we've emailed a link") regardless of whether the email is registered, so no information about account existence is leaked; the email is dispatched asynchronously so response timing cannot be used to infer an account's existence
- Blocked for SSO-enforced roles at both the request step and the consume step, so a policy change after issue invalidates the outstanding link; the Owner role is always exempt
- Tokens are stored as a SHA-256 hash only — the plaintext travels only in the email and is compared with
hash_equals; request and consume endpoints are rate-limited per email and per IP - Platform documentation page for Magic-Link Login — covers the login flow, enabling the feature, anti-enumeration behaviour, SSO interaction, and the security model
Changed
Activities
- Activity type is now a user-editable list of values — the seeded "Activity Type" list (Task, Call, Fax, Email, Meeting, Note, Letter), manageable in Settings → List Names — replacing the previous fixed enum. The change applies across the Activities pages, the Calendar create/edit/detail modals, and the API, where
type_idnow references alist_valuesid and responses includeactivity_type_name - Activity participants now resolve to the linked user's name and avatar (with photo where available) on the activity page, in the calendar, and in the owner/participant pickers; the activity page lists the owner first followed by participants
- Activities API operations now publish full OpenAPI request/response schemas — the paginated list envelope, the single-resource envelope, and the create/update request bodies
Notes
- The calendar reuses the existing Activities write path and
activities.*permissions; no new permissions, webhook events, or database tables were introduced - Feed URLs are not individually revocable — rotating the application's
APP_KEYinvalidates all existing feed URLs at once
0.4.1 — Member Merge, Anonymise & Docs Overhaul
Added
Members — Merge & Anonymise
MergeMemberaction — transfers relationships, contact details, custom field values, and memberships from a secondary member to a primary member of the same type, then archives the secondaryAnonymiseMemberaction — erases a member's personally identifiable information (name, description, icon, emails, phones, addresses, links); irreversible; users cannot anonymise their own recordPOST /api/v1/members/{member}/mergeAPI endpoint — requiresmembers:writeability andmembers.deletepermission; returns the updated primary member under thememberkey; 422 on self-merge or membership-type mismatchPOST /api/v1/members/{member}/anonymiseAPI endpoint — same auth requirements; 422 if the authenticated user targets their own recordmember.mergedandmember.anonymisedwebhook events — both registered inWebhookService::EVENTSand dispatched by their respective actions
Navigation & UI Permission Gating
- CRM mega menu in the application header is now gated behind
members.access— hidden entirely for users without access - Command palette "Members" navigation entry and "New Member" create entry gated behind
members.view/members.createpermissions respectively - Dashboard "Add Member" quick action is now a live link gated behind
members.create
Removed
- Admin Modules page (
/admin/settings/modules) — the page's toggle cards did not actually enable or disable functionality. Module selection now lives solely in setup feature profiles (FeatureProfile::modules()); storedsettings('modules.*')values are unaffected
Documentation
docs/api/webhooks.md— Events table expanded from 7 to 39 events, grouped by domain (Members, Products, Stock, Activities, Rate Definitions, Tax, Users, Roles, Settings)docs/platform/members.md— updated bulk actions (archive + conditional merge), row actions (View/Edit/Archive/Restore), archive filter chips, column toggle/export, merge modal, and anonymise; detail page tabs corrected from 7 to 8 (Overview, Information, Contacts, Activities, Opportunities, Movements, Invoices, Files)docs/getting-started/seeders.md— expanded from 4 to 16 seeders; added CountrySeeder, ListOfValuesSeeder, CurrencySeeder, TaxClassSeeder, TaxRateSeeder, RevenueGroupSeeder, CostGroupSeeder, ProductGroupSeeder, EmailTemplateSeeder, NotificationTypeSeeder, ViewSeeder, RateDefinitionPresetSeeder, ProductSeeder, and ActivitySeeder; corrected DatabaseSeeder run order; addedsignals:seed-demo/signals:clear-demodemo data commands; updated idempotency notesdocs/api/members.md— documentedPOST /mergeandPOST /anonymiseendpoints including request body, response shape, and 422 error cases
0.4.0 — Rate Engine
Added
Rate Engine
- Composable rate engine — a rate definition combines a calculation strategy, a base period, and optional modifiers to describe how a charge is calculated over a rental window
- Three calculation strategies — Period-based (charge per chargeable unit), Fixed (flat charge), and Hybrid (fixed initial charge plus per-unit charge thereafter)
- Five base periods — half-hourly, hourly, daily, weekly, and monthly, each converting elapsed time into chargeable units
- Multiplier modifier — tiered duration multipliers that scale the unit price as a rental lengthens, with the final tier inheriting forward
- Factor modifier — quantity ranges that scale the per-unit subtotal, with an open-ended final range
- Rate breakdown value objects — lossless integer-minor-unit arithmetic via Brick Money, rounded once at assembly, with structured line items and applied-modifier records
- Time options for period and hybrid strategies — clock vs business-hours day type, business hours, rental days per week, leeway minutes, and first/last day cutoffs
- 11 rate definition presets replicating industry-standard RMS engine types (Daily Rate, Daily Multiplier and Factor, Hourly Rate, Hourly Multiplier and Factor, Half Hourly Rate, Weekly Rate, Monthly Rate, Monthly Multiplier and Factor, Fixed Rate, Fixed Rate and Factor, Fixed Rate and Subs Days)
- Presets seeded on install and re-runnable from the Database Seeders admin panel
- Rate resolver — resolves the highest-priority product rate by store, transaction type, date validity, and priority, with per-product tagged caching invalidated on write
RateTransactionTypeenum (rental, sale, service) for product rates, distinct from the int-backed stock transaction type
Config Schema System
- Reusable
App\Support\ConfigSchemasubsystem for dynamic, schema-driven configuration forms — Field hierarchy (Text, Number, Decimal, Toggle, Select, Time), GroupField, RepeaterField, Schema, and Section - Conditional field visibility evaluated server-side, validation rule generation for visible fields, and sanitisation that strips hidden and disabled-modifier values
Persistence
rate_definitionstable — strategy, base period, JSONB enabled modifiers and config, preset slug, and self-referential clone trackingproduct_ratestable — integer minor-unit price, transaction type, currency, optional store scope, validity dates, and priorityRateDefinitionandProductRatemodels with factories, schema definitions, and aProduct.rates()relationship
Actions & DTOs
- Seven rate actions — Create/Update/Delete/Duplicate RateDefinition and Create/Update/Delete ProductRate — each authorising via gate, wrapping writes in a transaction, firing audit events, and dispatching webhooks
- Rate DTOs for input validation and API serialisation, including
RateBreakdownData(decimal-string money, structured line items) - Config validation on create/update against the composed schema; disabled-modifier configs stripped on update
- Non-blocking product rate overlap detection surfaced as a warning
API
- Rate definition endpoints — full CRUD plus duplicate, with Ransack filtering by strategy, base period, and preset
- Nested product rate endpoints under
products/{product}/rates, with anoverlapping_rate_idswarning in the response meta POST products/{product}/calculate_rate— returns a rate breakdown, or a zero-priced breakdown withmeta.resolved: falsewhen no rate is configured- Rate engine metadata endpoints —
rate_engine/strategies,/modifiers,/presets, and/schemafor external form builders rates:read/rates:writeSanctum abilities and therates.*permission group
Admin & Catalogue UI
- Rate Definitions admin panel under Admin → Pricing — list of presets and custom definitions with in-use counts, schema-driven create/edit form with preset picker and From Scratch option, and duplicate support
- Server-driven config form — choosing a strategy constrains base periods, enabling a modifier reveals its tier/range table, and fields show/hide based on other values
- Config-schema Blade field partials styled to Signals conventions, including bordered repeater row cards with add/remove/reorder controls
- Product Rates tab on the product detail page with a dedicated rate assignment form, modal-confirmed removal, and an overlap warning banner
- New Pricing group in the admin sidebar, landing grid, and group switcher
Documentation
- Platform documentation page for Rate Definitions
- API documentation page for rate definitions, product rates, calculation, and metadata, including the RMS rate engine mapping table
Changed
AppServiceProviderregisters the rate engine registry as a singleton with the three strategies and two modifiersPermissionSeederadds therates.*permission group, assigned to the Admin and Operations Manager rolesCompleteSetupandDatabaseSeederseed the rate definition presets on install
0.3.0 — Products, Activities & Inventory
Added
Products Module
- Product model with
ProductTypeenum (Rental, Sale, Service, Loss & Damage),StockMethodenum (Bulk, Serialised), andHasSchemaintegration - Products list page with type filter chips (Rental/Sale/Service), archive filter (Active/Archived/All), per-type counts, and data table with column sorting, filtering, and bulk selection
- Product detail page — 3-column layout with description, quick stats, product group, tax & revenue (left), product details, pricing, activity timeline placeholder (center), product image, key attributes, tags (right)
- Product create and edit forms — 2-column layout with basic info, identification, stock, pricing, tax & revenue, options, and custom fields panel
- Product merge — select two products, side-by-side comparison modal, migrates stock levels, accessories, attachments, and custom fields to primary; archives secondary
- Product icon upload on show and edit pages
- Product tabs — Overview, Stock, Accessories, Custom Fields, Activities, Files
- Product API endpoints — full CRUD with Ransack filtering, custom view support, and RMS-compatible response shape
- Product policies for view, create, edit, and delete permissions
- Product column registry for custom views with 11 columns and 6 defaults
- Custom views for products — All Products, Rental Products, Sale Products, Active Products, Inactive Products
Product Groups
- Product group model with hierarchical structure and product count
- Product groups list page with data table, search, and product count column
- Product group detail page — sidebar with group details, main area with products-in-group data table
- Product group create and edit forms
- Product group API endpoints — full CRUD with Ransack filtering
- Product group policy for view, create, edit, and delete permissions
- Delete product group action with audit trail and webhook dispatch
- Navigation — mega menu link, command palette commands
Stock Levels
- Stock level model with
StockCategoryenum (Bulk, Serialised) andHasSchemaintegration - Stock levels list page with data table and search
- Stock level detail page — 2-column layout with details, quantities (with available calculation), dates, and transactions panel
- Stock level tab on product pages — scoped data table showing stock for the product
- Stock level API endpoints — full CRUD with Ransack filtering and custom view support
- Stock level policies and column registry
- Custom views — All Stock Levels, Serialised Stock, Bulk Stock
Stock Transactions
- Stock transaction model with
TransactionTypeenum — 11 types: Opening Balance, Increase, Decrease, Buy, Find, Write Off, Sell, Return, Make, Transfer Out, Transfer In - Signed quantity calculation via
quantitySign()— negative for reductions (Decrease, WriteOff, Sell, TransferOut), positive for additions quantity_moveaccessor on model — signed quantity for display- Manual transaction creation restricted to Buy, Find, Write Off, Sell, Make (system types like Opening, Transfer are API-only)
- Inline transaction form on stock level show page — type selector, quantity, date, description
- Automatic
quantity_heldupdate on stock level when transactions are created - Stock transaction API endpoints — RMS-compatible nested routes (
products/{product}/stock_levels/{stock_level}/stock_transactions) with index, show, and store - RMS-compatible response shape with
transaction_type,transaction_type_name,quantity,quantity_move,manualflags
Activities (CRM)
- Activity model with polymorphic
regardingrelationship (Member, Product, StockLevel),owned_byuser, andparticipantsmany-to-many with members ActivityTypeenum — Task (1001), Call (1002), Fax (1003), Email (1004), Meeting (1005), Note (1006), Letter (1007) — mapped to industry-standardtype_idinteger codesActivityStatusenum — Scheduled (2001), Completed (2002), Cancelled (2003), Held (2004)ActivityPriorityenum — Low (0), Normal (1), High (2)TimeStatusenum — Free (0), Busy (1) for calendar blocking- Activities list page with type filter chips (Task/Call/Meeting/Email/Note), status filter chips (Scheduled/Completed/Cancelled/Held), per-type and per-status counts
- Activity detail page with badges (type, status, priority), details panel, regarding link, participants list, quick actions (Complete, Edit, Delete)
- Activity create and edit forms — 2-column layout with basic info, classification, schedule (left), assignment and entity picker (right)
- Member activities tab — scoped data table with "New Activity" button pre-filling regarding_type=Member
- Product activities tab — scoped data table with "New Activity" button pre-filling regarding_type=Product
- Stock level activities tab — scoped data table with "New Activity" button pre-filling regarding_type=StockLevel
- Complete activity action — sets status to Completed,
completed=true, fires audit event - Activity API endpoints — full CRUD plus
POST /activities/{id}/complete, Ransack filtering, custom view support - RMS-compatible API responses —
type_id,status_id,activity_type_name,activity_status_name,time_status_name,participantsarray, nestedregardingandownerobjects - Activity permissions — access, view, create, edit, delete, complete — assigned to Admin, Operations Manager, Sales, and Read Only roles
- Activity column registry for custom views with 11 columns and 7 defaults
- Custom views — All Activities, Scheduled Activities, Completed Activities
- RMS type mapping via
Activity::resolveRegardingType()andActivity::shortRegardingType()— stores full class names in DB, exposes short names (Member, Product, StockLevel) in API responses
Accessories
- Accessory model linking products to other products with quantity
- Accessories tab on product pages
- Accessories API endpoints — index, store, destroy nested under products
Navigation & Search
- Activities mega menu link under CRM > Engagement with active state
- Product Groups mega menu link under Resources with active state
- Command palette — Activities, Product Groups, Stock Levels navigation commands; New Activity, New Product, New Product Group create commands
- Global search — activities searchable by subject (permission-gated), results displayed in command palette with calendar icon
- Search results for products, stock levels, and product groups in command palette
Schema & Services
- Schema definitions registered on 14 Phase 1 models (Address, Country, Currency, ExchangeRate, Email, Phone, Link, Attachment, User, ActionLog, Webhook, CustomView, TaxRate, TaxRule)
- Two-tier caching on SchemaRegistry — L1 in-memory + L2 Redis with tag-based invalidation
- Schema Discovery API —
GET /api/v1/schemaandGET /api/v1/schema/{model}for introspecting field metadata - VisibilityRuleEvaluator — 11 operators (equals, not_equals, contains, in, gt, lt, etc.) with AND logic
- AutoNumberService — atomic increment, preview, and reset for custom field auto-numbering
- Formatter::money() and moneyDecimal() — brick/money string-based formatting without float conversion
- EnforceSessionTimeout middleware — settings-driven idle session expiry
Security
- SecuritySettings enforcement for login rate limiting (max_login_attempts, lockout_duration)
- FileService hardened — S3 path changed to entity-organised structure, scan_status default changed to pending, Storage::put() failure checks added
- Icon thumbnail size corrected from 400px to 150px
Changed
- Member activities tab replaced placeholder with scoped data table and action buttons
- Custom view description column added to custom_views table
- Attachment scan_status default changed from
cleantopending - AppServiceProvider cleanup — removed unused Relation import after morph map evaluation
- Search controller expanded to cover products, stock levels, product groups, and activities alongside members
- Command palette expanded from 22 to 30+ static commands with product, activity, and product group entries
Quality
- 227 new/modified files, 17,200+ lines of code added
- 130+ new tests across enums, models, policies, actions, API controllers, column registries, Livewire components, and search
- PHPStan level 6 — zero errors across all new code
- Pint formatting enforced on all changes
- Documentation pages for Activities (platform + API)
0.2.0 — People & Places
Added
Members Module
- Members module — universal entity for contacts, organisations, venues, and users
- Members list page with search, type filter chips, column sorting, column filtering, bulk selection, and archive/restore
- Member detail page — CRM-style 3-column layout with left sidebar (avatar, quick actions, key contacts, account details), center content (stat cards with sparklines, AI recommendations, activity timeline), and right sidebar (customer health score, health factors)
- Member create and edit forms — 2-column layout with membership type, status, description, locale, currency, tax class, and conditional org/contact sections
- Contact detail management — addresses, emails, phones, and links with primary flags and type classification
- Member relationships — link contacts to organisations with relationship type labels
- Member archive and restore — replace delete with soft-archive, restore from archived view, archive filter chips (Active/Archived/All)
- Member merge — select two same-type members, side-by-side comparison modal, migrates all polymorphic relations (addresses, emails, phones, links, attachments), relationships, custom fields, and memberships to primary; archives secondary
- Member profile icon in page header — visible on all member pages, white box with border and shadow, initials fallback
- Phone country code —
country_codecolumn on phones table, flag emoji picker with searchable country dropdown and dial code display, international format on information page - Searchable country combobox on address form — replaces plain
<select>withx-signals.comboboxfor 246 countries - what3words geocoding on address form — forward lookup (three words to coordinates), Nominatim geocode from address fields, interactive Leaflet.js map with draggable marker
- Command palette with live member search — triggered by
/orCmd+K, 22 static commands plus live search with type badges and status indicators - Keyboard shortcuts on member pages —
eto edit,nto open New dropdown - Member API endpoints — full CRUD for members, addresses, emails, phones, links, and relationships
- RMS-compatible member API responses — active flag, membership object, type-specific fields, icon object, child/parent members
- Auto-create User-type member on user creation via InviteUser action
- BackfillUserMembers command for existing users
Custom Fields
- Custom field groups — create, edit, reorder, and delete groups that organise custom fields
- Custom fields admin — 16 field types (Text, TextArea, Integer, Decimal, Boolean, Date, DateTime, Time, Select, MultiSelect, URL, Email, Phone, Colour, Currency, Percentage, RichText) with validation rules, default values, and list-backed dropdowns
- Custom field values on member detail pages, grouped by field group
- Custom field enforcement — default values applied on creation, is_required validated before persistence, is_searchable and is_active enforced in queries
- Custom field multi-value support for multi-select fields
- Auto-number sequences for custom field auto-numbering
- Default
custom_fieldsin member API responses — RMS-compatible, no explicit?include=customFieldValuesrequired - Custom field API endpoints — CRUD for groups and definitions
Custom Views
- Custom views system — saved list configs with columns, filters, sort order, and visibility levels (personal, shared, role-restricted)
- View builder Livewire component — create and edit custom views with column picker, filter builder, and sort configuration
- Column registry — schema-driven column definitions for DataTable, custom view validation, and API field selection
- DataTable custom views integration — view selector dropdown, filter/sort application from saved views
- Custom views API endpoints — full CRUD
Multi-Currency
- Currencies table (ISO 4217) with CurrencyService for conversion
- Exchange rates with effective dates — create, update, delete via admin and API
- Currency and exchange rate API endpoints
- Currency seeder with ISO 4217 data
File Attachments
- Polymorphic
attachmentstable with S3 signed URLs and virus scanning support - File service — S3/public disk abstraction with signed URL generation, icon upload, and thumbnail creation
- Icon upload Livewire component
- File upload modal on member pages
- Attachment API endpoints
Reference Data & Lists
- Lists (reference data) — create and manage named lists with hierarchical values, system/non-system flags, and active/inactive status
- Built-in system lists seeded on install: AddressType, EmailType, PhoneType, LinkType
- Additional list categories: Lawful Basis Type, Location Type, Rating, Invoice Term, Locale, Currency
- Countries admin page — browse, search, and toggle active status
- List API endpoints — CRUD for list names and list values
- Countries API endpoints — read-only index and show
Tax
- TaxCalculator service — resolves tax rules by organisation + product tax class matrix with fallback to defaults, bcmath precision, currency-aware decimal formatting
- Tax classes — product and organisation tax classifications with default designation
- Tax rate and tax rule API endpoints — full CRUD with Ransack filtering
- Tax rate and tax rule admin pages — create, edit, delete with priority ordering
- Tax class API endpoints — CRUD for product and organisation tax classes
Authorization & Security
- Model policies for 10 domain models — Member, Store, CustomField, CustomFieldGroup, ListName, OrganisationTaxClass, ProductTaxClass, Webhook, ActionLog, EmailTemplate
- RoleLevel enum — hierarchical role ordering for authorization comparisons
- Policy traits —
AuthorizesByPermissionandChecksStoreAccessextracted from 8 policies - Store scoping via
StoreScopeglobal scope andHasStoreScopingtrait — request-scoped via Context facade, Octane-safe - Cost visibility trait —
HasCostVisibilitywith fail-closed design, explicit user parameter for queue contexts - Blade permission directives —
@area,@action,@costsfor template-level authorization
Schema & Query Engine
- Field Registry and Schema Engine —
HasSchemacontract,SchemaBuilder,SchemaRegistryfor unified field metadata across core, computed, and custom fields - Ransack filter enhancements —
matches(regex),start,end,present,blankpredicates; relationship filtering; custom field EAV filtering viacf.prefix; per-token rate limits - Reusable
applyIncludes()in FiltersQueries trait with$defaultIncludesand$allowedIncludesproperties
Admin & Settings
- Admin Integrations settings page — encrypted API key storage for what3words, Google Maps, and future third-party services
- Feature profiles — Full, Lite, Warehouse, Services presets for module configuration
- Setup wizard improvements — infrastructure checks, branding step, module selection, feature profile presets
- Admin seeder management page — run seeders from the admin panel
- Getting started checklist on dashboard — tracks setup completion progress
Infrastructure
- Reusable DataTable Livewire component — configurable columns, sorting, filtering, pagination, row selection with shift-click range, bulk actions, and event-driven refresh
- CustomFieldCopier service — copies custom field values between entities when field name + type match
- Permission registry
validate()method for reusable permission validation - Action log export job — async CSV export with date validation, failure notifications, and retry configuration
- Social promo pages — LinkedIn (1200x627) and Instagram (1080x1080) format
Quality
- PHPStan upgraded from level 5 to level 6 — 168 generic type annotation fixes, zero errors across 618 files
- 7 new migrations: currencies, exchange rates, attachments, custom views, custom view roles, user view preferences, phone country code
- 100+ new tests across actions, services, Livewire components, DTOs, and pre-existing coverage gaps
- Documentation pages for Members, Custom Fields, Lists, Tax Classes, Countries, Currencies, Exchange Rates, Attachments, and Custom Views
Changed
- Member show page redesigned as 3-column CRM-style layout with stat cards, AI recommendations, and activity timeline
- Member form redesigned as 2-column layout with conditional org/contact sections and list-value dropdowns
- Members list bulk action changed from delete to archive with confirmation modal
- Row actions changed from delete to archive/restore based on member state
- Admin sidebar expanded with Data section, Tax section, and Integrations under Preferences
- Demo data seeder updated with sample members, contact details, custom fields, lists, and tax classes
- RansackFilter uses grammar-wrapped column quoting for SQL safety
- CustomFieldSerializer uses batch loading for ListOfValues fields (eliminates N+1 queries)
- ExportActionLog validates date filters and notifies users on failure
- AnonymiseMember uses
refresh()instead offresh()for null safety - PermissionRegistry centralises permission validation (removed duplication from CreateRole/UpdateRole)
- UpdateMemberData membership_type changed from
?stringto?MembershipTypeenum - MergeMember action wraps all operations in DB::transaction with orphan cleanup
- ArchiveMember action wrapped in DB::transaction for atomic state changes
- What3WordsService adds HTTP timeouts (10s) and logging on all failure paths
- MergeModal catches exceptions with user-friendly flash messages instead of raw 500 errors
Fixed
- SQL injection vector in RansackFilter
matchespredicate — column names now safely quoted via query grammar - Relation filter column names validated against
[a-z_][a-z0-9_]*pattern to prevent injection - ILIKE wildcard characters escaped in SearchController to prevent pattern injection
- TaxResult decimal accessors now respect per-currency minor unit exponents (JPY, BHD, KWD, etc.)
- TaxCalculator uses pure bcmath chain — eliminated float intermediate that could cause precision loss
- StoreScope replaced static mutable state with request-scoped Context facade (Octane-safe)
- HasCostVisibility throws LogicException when
$costColumnsproperty is missing (fail-closed instead of fail-open) - BackfillUserMembers wrapped in DB::transaction with per-user error handling
- Gate::authorize('members.view') permission check added to SearchController
- RestoreMember guards against restoring non-deleted members
- Migration logs warnings for unresolvable ListOfValues during type code realignment
0.1.0 — Initial Alpha
Added
- Infrastructure install wizard (
signals:install) with interactive configuration for PostgreSQL, Redis, S3, and Reverb - Web-based setup wizard for company details, stores, feature profiles, branding, and admin account creation
- CLI setup command (
signals:setup) as terminal alternative to web wizard - Status command (
signals:status) for checking infrastructure connection health - Livewire authentication system with login, forgot password, email verification, and profile settings
- Two-factor authentication (TOTP) with recovery codes, manageable from Settings → Profile
- Composite dashboard with branded design system and KPI cards
- Documentation system with markdown rendering, full-text search, and three-column layout
- API documentation auto-generated via Scramble at
/docs/api - Demo data seeding (
signals:seed-demo) for evaluation and testing - Landing page with technical blueprint aesthetic
- Non-interactive mode for CI/CD deployments
- System admin panel with settings management (Company, Stores, Branding, Modules)
- User management with invite, deactivate, reactivate, password reset, and ownership transfer
- User edit page with role assignment and authorization checks
- Role management with system roles (Admin, Manager, Operator, Viewer) and custom role creation
- Role edit page with multi-user assignment
- Permissions reference page showing all registered permissions grouped by domain
- Security settings page for password policies, session timeouts, login lockout, and 2FA enforcement
- Email settings page for SMTP configuration with connection testing
- Database seeders admin page showing seeder status with run controls
- Permission and role seeders integrated into default
DatabaseSeeder - Branding colours (primary and accent) applied across the UI via CSS custom properties
- Invitation system with signed URLs and accept-invitation flow
- REST API (v1) with Sanctum bearer token authentication and scoped abilities (
resource:action) - API token management — create, list, and revoke personal access tokens with granular ability scoping
- Ransack-compatible query filtering engine with 18 predicates (
_eq,_cont,_lt,_in, etc.) and sort support - API endpoints for users, roles, settings, action logs, and system health
- Webhook system — register URLs with event subscriptions, HMAC-SHA256 signed delivery, exponential backoff retry
- Webhook management API — create, update, delete, toggle, and view delivery logs
- Automatic webhook dispatch on user, role, and settings changes
- Auto-disable webhooks after 18 consecutive delivery failures
- ForceJsonResponse middleware ensuring all API responses return JSON
- EnsureActiveUser middleware blocking deactivated users from API access
- Rate limiting on API routes (60 requests/minute per user)
- API documentation pages for authentication, webhooks, and overview
Changed
- Moved
symfony/yamlandlivewire/voltto production dependencies for documentation rendering - Updated documentation URL to
docs.signals.rent - Header navigation bar and mobile sidebar use brand primary colour from settings
- Search input and focus states in header use brand colour tokens instead of hardcoded values
- Auth views updated to use signals component library (
s-*classes) - Component library expanded with 17 new components and
s-prefix convention