Bundles the prior autonomous-session output that was sitting unstaged: - Em-dash sweep across src/ + tests/ (en-dash/em-dash to hyphen, ~2280 instances) - country-flag-icons rollout (CountryFlag component, replaces emoji glyphs that never rendered on Windows; lazy-loads the 3x2 SVG index as a single chunk after the per-subpath dynamic-import approach silently failed in webpack) - Admin IA Phase 1+2: 7-domain regroup, 41 to 38 pages, /admin/berths index, redirects (ocr to ai, reports to dashboard, invitations to users), docs/admin-ia-proposal.md - Per-template email tester (registry + endpoint + UI on Email admin page) - Cancel-document mode picker (delete-from-Documenso vs keep-for-audit) - Dashboard PDF report: 25 widgets, SVG charts, date-range picker, 11 resolvers - Customize-widgets per-region sortables at xl+ (charts/rails/feed); single flat sortable below xl when the layout stacks; per-viewport saved orders - Audit doc updates capturing each shipped item - Lint fixes: react-compiler immutability in DonutChart (reduce instead of let-reassign), set-state-in-effect disables in CountryFlag and UploadForSigning preview-bytes effect, unused 'confirm' destructures in interest contract + reservation tabs, unescaped apostrophe in test-template card copy
32 lines
1.4 KiB
SQL
32 lines
1.4 KiB
SQL
-- 0054_user_profiles_username.sql
|
|
-- ----------------------------------------------------------------------------
|
|
-- Optional username as a sign-in alternative to email. Stored alongside the
|
|
-- canonical first/last name on user_profiles so the rest of the auth/profile
|
|
-- code keeps a single place to look. The Better-Auth `user` table stays the
|
|
-- source of truth for email + password; the username is a thin alias the
|
|
-- login form looks up to resolve to the matching email before delegating
|
|
-- to better-auth's email/password flow.
|
|
--
|
|
-- Constraints (enforced application-side AND in SQL):
|
|
-- - 2..30 characters
|
|
-- - lowercase letters, digits, dot, underscore, hyphen
|
|
-- - case-insensitive uniqueness per install (no per-port scoping -
|
|
-- reps move between ports and a global username keeps URLs stable)
|
|
--
|
|
-- The column is nullable; existing users keep email-only sign-in until they
|
|
-- pick one.
|
|
|
|
ALTER TABLE user_profiles
|
|
ADD COLUMN IF NOT EXISTS username TEXT;
|
|
|
|
-- Shape check at the DB level catches anything that slipped past the API
|
|
-- (raw SQL inserts in tests, scripts, etc.).
|
|
ALTER TABLE user_profiles
|
|
ADD CONSTRAINT chk_user_profiles_username_shape
|
|
CHECK (username IS NULL OR username ~ '^[a-z0-9._-]{2,30}$');
|
|
|
|
-- Case-insensitive uniqueness. Partial so multiple NULLs are allowed.
|
|
CREATE UNIQUE INDEX IF NOT EXISTS idx_user_profiles_username_unique
|
|
ON user_profiles (LOWER(username))
|
|
WHERE username IS NOT NULL;
|