1.7.110 - 2026-05-13
Fixed
- Fixed ungrouped
handle_event/3clauses inMediaBrowserby relocatingcreator_attrs/2helper to private-helpers block
Hygiene
- Routine lockfile updates (
mix.lock) - Precommit:
compile --forcereplaced withcompile --warnings-as-errors --all-warnings, addeddeps.unlock --check-unused, switched fromqualitytoquality.ci(format-check) - Dialyzer: removed 5 unused ignore filters (css_integration, process_scheduled_jobs_worker, duplicate conn_case/data_case, integrations guard_fail)
- Removed stale
:phoenix_kitself-entry frommix.lock
1.7.109 - 2026-05-12
Added
- V114 migration: Integrations storage switched to uuid-only
keycolumn onphoenix_kit_settings(PR #536)- Collapses the per-row
keyfrom the compositeintegration:<provider>:<name>shape to just the row's UUIDv7. Lifts both name restrictions baked into the old shape: the regex[a-zA-Z0-9][a-zA-Z0-9\-_]*and per-provider uniqueness are gone. Any non-empty string (after trim) is now a valid connection name; duplicates within a provider coexist (uuids disambiguate). Names with spaces, punctuation, "My Company Drive (US)" — all allowed add_connection/3: generates UUIDv7 up-front, embeds it in both theuuidandkeycolumns; provider + name live purely invalue_json.rename_connection/3rewrites only the JSONBnamefield — storage key is the row uuid, untouched across renames, so consumer modules pinning to uuid keep working- Read sites (
get_integration_by_uuid/1,list_connections/1,load_all_connections/1) source provider + name from JSONB; the list helpers expose:date_addedso UI callers render "Created N ago" without a second lookup provider:namestring lookups now first-match by case-insensitive name sort (names aren't unique anymore). Read-shim contract preserved for legacymigrate_legacy/0callsiteslog_activitytakes explicit(provider, name)so audit rows carry human-readable names — parsing the key would have stamped a uuid string intometadata.connection- Migration walks every
integration:%-keyed row in a single UPDATE, backfills missingvalue_json -> 'name'/'provider', ensuresmodule = 'integrations', and rewriteskey = uuid::text. Legacy V0-shape keys without:namefold toname = "default".down/1rewrites back to composite shape with-<8-char>suffix from UUIDv7's random tail on duplicate(provider, name)pairs
- Collapses the per-row
- V115 migration:
phoenix_kit_annotationstable for drawn-on-image shapes via the Etcher overlay (PR #537)- Stores rectangle / circle / polygon / freehand shapes tied to a
phoenix_kit_filesrow in image-pixel coordinates. Geometry is JSONB; shape kinds enforced via DB-level CHECK constraint matching Etcher 0.1's four-tool set file_uuidFKON DELETE :delete_all— annotations vanish with their host image.creator_uuidnullable +ON DELETE :nilify_allso user deletion preserves their annotations as anonymous- Discussion threads attach via the existing comments convention: comments anchored to the file (
resource_type = "file",resource_uuid = file_uuid) withmetadata.annotation_uuidcarrying the back-reference. Annotation-rooted comments appear in the file's main thread alongside non-annotated discussion - Indexes:
(file_uuid)for per-file listing, partial(creator_uuid) WHERE creator_uuid IS NOT NULLfor author lookups
- Stores rectangle / circle / polygon / freehand shapes tied to a
- V116 migration: nullable self-FK
parent_uuidonphoenix_kit_entity_data(PR #538)- Each entity-data row can point at another row of the same entity as its parent. System field — always present, optional, never user-removable (does not appear in
entities.fields_definition). Existing rows stayparent_uuid = NULLand become roots; no backfill - No
ON DELETEcascade — parent/child linkage and same-entity scope are managed by thePhoenixKitEntities.EntityDatacontext inside a transaction. A DB-level cascade would bypass the soft-delete machinery and the activity log - Same-entity enforcement is a context-layer responsibility. B-tree index on
(parent_uuid)covers the "list children" query for the WordPress-style indented tree
- Each entity-data row can point at another row of the same entity as its parent. System field — always present, optional, never user-removable (does not appear in
PhoenixKit.Annotationscontext +PhoenixKit.Modules.Storage.EtcherAdapter(PR #537)- Context handles CRUD against
phoenix_kit_annotationspluslist_for_file_with_previews/1that pulls every file comment in a single bulk query and groups bymetadata.annotation_uuidfor the tooltip preview Annotations.delete/1runs comment cascade + annotation row delete in aRepo.transaction/1so a failure between the two doesn't leave the annotation alive with its discussion thread destroyedAnnotation.adapter_writable_fields/0exposes the schema's@cast_fields(minusfile_uuid, which the adapter sets server-side fromtarget_uuid) as the source of truth for the adapter whitelist — the adapter's@schema_keysderives from it so a future schema field can't drift silentlyEtcherAdapterimplements theEtcher.Storagebehaviour, dispatching to the context. Adapter explicitly whitelists payload keys before reachingString.to_existing_atom— guards against forward-compat with Etcher's payload shape growing new client-side keys
- Context handles CRUD against
PhoenixKitWeb.Components.AnnotationComposerLiveComponent (PR #537)- Focused composer for attaching the first comment to a newly-drawn annotation. Explicit Post / Cancel control flow owns the annotation lifecycle: Post commits comment + solidifies annotation, Cancel rolls the annotation back. Communicates with the parent MediaBrowser via LC-to-LC
send_update/2, no host-LV plumbing required - Scope: text + file uploads (image / video / audio / pdf / archive) + Giphy picker. Audio recording (which the full
CommentsComponentsupports) intentionally skipped for v1
- Focused composer for attaching the first comment to a newly-drawn annotation. Explicit Post / Cancel control flow owns the annotation lifecycle: Post commits comment + solidifies annotation, Cancel rolls the annotation back. Communicates with the parent MediaBrowser via LC-to-LC
- MediaBrowser integration with the Etcher overlay (PR #537)
Etcher.layermounted alongsideFresco.viewerin the modal. Newetcher:created/:updated/:deleted/:selectedhandlers wire the JS overlay into the storage backend- Lifecycle:
open_viewer/2preloads annotations + rolls back any pending compose;finalize_annotation_compose/2reloads annotations and pokes the file'sCommentsComponentto refresh;refresh_file_comments/1flips the component'sloaded?to false to trigger a sidebar reload creator_uuidis set server-side from the scope — client-suppliedcreator_uuidin the payload is overridden, preventing author spoofing
IntegrationPickerrewrite (PR #536)- Card subtitle: priority
external_account_id→ masked credential tail (first 8 +…+ last 4 for any ofapi_key/bot_token/access_key;•••for keys under 14 chars) - Age line under subtitle using shared
<.time_ago> - Status badge: distinct label + colour for each of the four canonical statuses (
connected→ green "Connected",error→ red "Auth failed" withvalidation_statustooltip,configured→ yellow "Not tested",disconnected→ grey "Not connected") - Provider icon + display name auto-resolve via
Integrations.Providers.get/1— callers no longer pre-attach a:providerstruct - Provider-name badge hidden when picker is filtered to a single provider (both real callsites do this)
- Click feedback:
phx-click-loadingdims the clicked card + blocks rapid re-clicks during the LV round-trip; daisyUIloading-spinnerswaps in for the status badge during the same window provider_defmemoized in aMap.new(connections, ...)shared betweenfilter_by_search/3and the render path — dropsProviders.get/1calls from 2N to N per render- 33 new component spec tests covering subtitle priority + masked credential + age + provider auto-resolve + status branches + filter-by-provider + search threshold + empty state + deleted-card warning + click-action dispatch
- Card subtitle: priority
<.draggable_list>:sortable_handleattribute (PR #538)- Optional CSS selector (e.g.
".pk-drag-handle") that restricts drag initiation to elements matching the selector inside each item. When set, the item wrapper dropscursor-grabstyling — the caller renders their own handle. Backward-compatible: defaultnilpreserves whole-item drag. Mirrors<.table_default>'s:on_reorder+.pk-drag-handleconvention. JS hook (SortableGrid) already supporteddata-sortable-handle; this PR wires the Elixir-side knob through
- Optional CSS selector (e.g.
- Etcher tooltip JS slot overrides (PR #537)
window.Etcher.tooltipSlots.header/.footer/.bodytranslatemetadata.comment_*keys into the rich tooltip (author header, date · count subheader, thumbnail + quoted text body).window.Etcher = window.Etcher || {}guards against load order
AnnotationComposerPositionJS hook keeping the MediaBrowser's floating annotation-composer popover inside the viewer bounds via re-clamping on mount + updates + window resize- Dep adds:
:fresco ~> 0.1(OpenSeadragon viewer wrapper, now a direct dep since Tessera 0.2 split it out),:etcher ~> 0.1(annotation overlay) - Three post-merge review docs in
dev_docs/pull_requests/2026/:536-integrations-v114-uuid-keys-picker-ux/,537-annotations-v115-etcher-overlay/,538-v116-parent-uuid-draggable-handle/— each with finding disposition tables tracking which items were addressed in follow-up commits and which were deferred to the original PR author
Changed
Integrations.validate_connection/2rescue narrowed to[DBConnection.OwnershipError, Postgrex.Error, Req.TransportError]so genuine logic bugs (KeyError,ArgumentError,MatchError) bubble up to the supervisor instead of being swallowed under a generic "validation failed".validate_credentials/2mirrored the narrowing post-merge for parity (PR #536)Integrations.authenticated_request/4docstring spells out the URL-trust contract: the integration's Bearer token is attached to every request, so callers must pin URLs to a domain allowlist before invoking. Internal callers (OpenRouterClient, OAuth refresh, userinfo) build URLs from the Providers registry which is hardcoded and safe; new callsites taking URLs from elsewhere need their own guard (PR #536)phx-disable-withon Save / Test Connection / Disconnect / Delete buttons onintegration_form.html.heexplus the OAuth Connect Account button. Pre-fix, a double-click + slow network could submit two save requests or spawn parallel HTTP probes (PR #536)- IntegrationForm
create_connection+save_form_with_renameerror branches preserve:new_name+:form_valueson error so a failed:empty_namesubmit doesn't wipe the api_key the operator just typed. Dropped the dead:already_exists/:invalid_nameerror branches (those tuples no longer fire post-V114). Template: removed the now-incorrect "Letters, digits, hyphens, and underscores. Must be unique per provider." name-rules hint (PR #536) PhoenixKit.Users.Permissions.module_label("db")/module_icon/module_descriptionpindbin@core_*maps so the display is correct even when the externalphoenix_kit_dbmodule isn't loaded (PR #536)MediaBrowser.format_date/1strftime format string wrapped ingettext(...)so locales can reorder date components (%d %b %Yfor en-GB / fr / de) without code changesAnnotationComposer.first_error/1routes throughPhoenixKitWeb.Components.Core.Input.translate_error/1— gettext-aware helper that interpolates%{count}and other opts properlyAGENTS.mdCHANGELOG-ownership instruction corrected — entries are written by agents against the bumped@versionheading, matching the project's actual workflow
Fixed
- Post-merge fixes folded from review of PR #536:
- V114 docstring drift after rebase rename: moduledoc references "V113" but the module is V114; "Stamp the table comment with '113'" while code stamps '114'; "post-V113 regression / invariant" in tests and picker comment. All swept to V114
Permissions.module_label("db")was falling through toString.capitalize("db")="Db"when the externalphoenix_kit_dbmodule isn't loaded; test asserts"DB". Folded in to keep the post-rebase baseline green- Test fixture rows in
storage/scope_test.exs,media_browser_scope_test.exs,media_browser_test.exsviolated V113'sphoenix_kit_files_user_or_parent_checkCHECK constraint; eachcreate_file!/1now stampsuser_uuidvia a memoisedensure_user!/0helper IntegrationPicker.filter_by_search/3had a shadowingnamevariable in the inner case pattern match; renamed toprovider_nameString.slicenegative-bound range pinned with explicit step (-4..-1//1) to silence Elixir 1.16+ range-step warning- Credo cleanup:
get_integration/12-branchcondwithtruearm →if/else; inlinePhoenixKit.Settings.Queries.get_setting_by_uuid/1calls aliased asSettingsQueriesinintegrations_test.exs; three test files (storage/scope_test.exs,media_browser_scope_test.exs,media_browser_test.exs) aliasPhoenixKit.Users.Authfor theirensure_user!helpers
- Post-merge fixes folded from review of PR #537:
Annotations.delete/1deleted linked comments outside a transaction — if the comment cascade succeeded and the annotation row delete then failed (FK violation, DB transient), comments were gone but the annotation remained with its discussion permanently destroyed. Wrapped inRepo.transaction/1via an extracteddelete_in_transaction/1helperresource_type = "annotation"claim in three moduledocs (annotation.ex,v115.ex,etcher_adapter.ex) contradicted the actual implementation, which anchors comments to the file (resource_type = "file") withmetadata.annotation_uuid. All docs swept to match realityAnnotations.delete_linked_comments/1barerescue _ -> :okswallowed every exception class including logic bugs. Narrowed to[DBConnection.OwnershipError, Postgrex.Error, ArgumentError]so logic bugs surfaceAnnotationComposer.normalize/1reinvented whatEcto.Changeset.cast/3already does (accepts both atom- and string-keyed maps) AND silently passed the original map through whenString.to_existing_atomfailed, hiding typo'd field names from the user as "geometry: can't be blank" rather than "unknown field" — function deleted- In-repo
Code.ensure_loaded?(PhoenixKit.Annotations)guard inMediaBrowser.load_annotations_for/1was needless defensive code —Annotationsis in the same compilation unit @compile {:no_warn_undefined, [PhoenixKit.Modules.Storage, ...]}would have shadowed legitimate compile errors on a core module rename —Storageremoved from the suppression listAnnotationComposerPosition.destroyedcleanup conditional was a never-falls-through guard — simplified- Etcher slot-preservation JS comment misstated the mechanism — corrected to "PhoenixKit owns the tooltip layout; downstream consumers must load AFTER phoenix_kit.js"
- Credo cleanup:
Annotations.first_attachment_thumbnail/1single-clausewith→case; aliasedAnnotations,Storage,EtcherAdapter,Storage.Fileso six "nested modules could be aliased" findings clear - PhoenixKitComments dialyzer ignores added for
annotations.ex+annotation_composer.ex(optional sibling package, guarded at runtime)
- V114 down SQL collision-suffix source:
substring(uuid::text from 1 for 8)extracted UUIDv7's timestamp prefix — same-millisecond rows produced identical suffixes ⇒ duplicate "uniquified" keys when two+ rows collided on(provider, name). Switched tosubstring(uuid::text from 25 for 8)(random tail, 32 bits of entropy). Mirrored inrun_down!in the V114 test (PR #538 follow-up) - 3-row collision test added to V114 test suite covering N ≥ 3 case (exactly one plain key, N-1 distinct suffixed keys, all keys unique)
- IntegrationPicker click feedback was missing: clicking a card showed no visual response during the 100-500ms LV round-trip — operators would click again and submit a second request. Added
phx-click-loading:opacity-60 phx-click-loading:pointer-events-none+ status-badge → spinner swap during the in-flight window (PR #536)
i18n
- AnnotationComposer user-facing strings wrapped in gettext (flash messages + heex literals + ARIA labels — ~17 strings)
- MediaBrowser
format_datestrftime pattern wrapped in gettext so locales reorder date components - IntegrationPicker status labels (
Connected/Auth failed/Not tested/Not connected) and search placeholder + empty-state strings wrapped in gettext - IntegrationForm flash messages, button labels, name placeholder, danger-zone copy, OAuth step labels, redirect-uri instructions wrapped in gettext
1.7.108 - 2026-05-11
Added
- V112 migration:
phoenix_kit_projects*schema evolution (PR #533)archived_at TIMESTAMP(0)onphoenix_kit_projectsso the admin dashboard can soft-hide projects without flipping a status enum. Mirrors the workspace convention used byphoenix_kit_publishing'sposts.trashed_atandphoenix_kit_files.trashed_at— null = visible, non-null = soft-hidden, with the timestamp doubling as audit metadata. Existingstatus = 'archived'rows backfilled intoarchived_atso the dashboard filters keep working transparentlyphoenix_kit_projects_visible_idxpartial index on(inserted_at DESC) WHERE archived_at IS NULL— one partial covers both project-list and template-list dashboard reads (neither view shows archived rows)translations JSONB NOT NULL DEFAULT '{}'onphoenix_kit_projects,phoenix_kit_project_tasks,phoenix_kit_project_assignmentsfor per-language overrides on user-input content (name / description / title). Primary stays in dedicated columns; JSONB only carries non-primary overridesposition INTEGER NOT NULL DEFAULT 0onphoenix_kit_projectsandphoenix_kit_project_tasksso the drag-and-drop reorder API can persist manual ordering. Existing rows fold into the0bucket and the schema's secondary order-by-inserted_atkicks in until a user actually dragsscheduled_start_dateretyped fromDATEtoTIMESTAMP(0)so scheduled-overdue detection honors time-of-day (a project scheduled for today 09:00 flips to:overdueat 09:01, not at midnight). Column name kept — lying name + honest type beats the churn of renaming every call site- Drops the three remaining unique-name indexes —
phoenix_kit_projects_name_template_index,phoenix_kit_projects_name_project_index,phoenix_kit_project_tasks_title_index. Name uniqueness is now policy, not schema — editing or duplicating names no longer trips a stale index, and future renames don't need migration coordination - Bumps migrator
@current_version111 → 112 — without this V112 was dead code - All steps idempotent (column-existence, index-existence, USING coercion clauses);
down/1reverses each change so a rollback restores the V111 shape.down/1restores the V105/V101 unique indexes first, before dropping any V112 columns, so a duplicate-name conflict at rollback aborts cleanly rather than leaving a half-rolled schema
test/phoenix_kit/migrations/v112_test.exs— pins every V112 addition (archived_at column + type + nullability, visible-index existence AND predicate shape, translations JSONB on the three tables, scheduled_start_date retype, position columns) plus the four dropped indexes and the duplicate-name behavior. The predicate test refutes anyis_templatemention, closing the docs-drift loopdev_docs/pull_requests/2026/533-v112-projects-schema-evolution/CLAUDE_REVIEW.md— post-merge review with finding dispositions- V113 migration: system-managed media flag for Tessera deep-zoom tiles + comments↔files junction (PR #534)
system_managed BOOLEAN NOT NULL DEFAULT falseonphoenix_kit_files— marks internally-generated media (DZI tile pyramids + per-tile chunks) so the MediaBrowser excludes them from user listings and the variant generator skips them (tile chunks don't need small / medium / large — just an"original"FileInstance)parent_file_uuid UUIDnullable FK tophoenix_kit_files(uuid)ON DELETE CASCADE — system-managed tile rows cascade away when their source image is hard-deleteduser_uuiddrops NOT NULL — system-managed rows belong to a parent File, not a user. The DB-level CHECKphoenix_kit_files_user_or_parent_checkenforces "user_uuid IS NOT NULL OR parent_file_uuid IS NOT NULL" so raw inserts can't violate the invariantphoenix_kit_files_system_dedup_indexpartial unique index on(parent_file_uuid, file_name) WHERE system_managed = true— concurrent lazy-generation requests for the same uncached tile dedupe at the DB level via the changeset'sunique_constraint;Storage.store_system_file/3recovers the winner's row on conflict- Two more partial indexes —
phoenix_kit_files_parent_uuid_index(per-source lookup + cascade-cleanup) andphoenix_kit_files_system_managed_index(keeps the MediaBrowser's "user files only" sort cheap as the tile catalog grows) phoenix_kit_comment_mediajunction table letting the comments module attach core File rows to comments with position + caption. Cascade oncomment_uuid, RESTRICT onfile_uuid(a file can't hard-delete while attached). Consumer code lands in a later PR- Bumps migrator
@current_version112 → 113; all DDL idempotent viaIF NOT EXISTS/ DO-blocks
- Deep Zoom Image viewer in MediaBrowser via Tessera (OpenSeadragon wrapper, PR #534)
- New
Tessera.Viewer.viewerreplaces the static<img>in the file modal;tessera_sources/1builds the progressive layer list (medium → optional large → DZI manifest) <Tessera.Storage>adapter (PhoenixKit.Modules.Storage.TesseraAdapter) lands tile writes in the storage pipeline (multi-bucket viaManager.store_file/2+ a system-managed File row viaStorage.store_system_file/3)- Two new public endpoints —
/tiles/:token/:dzi_filenameand/tiles/:token/:files_segment/:level/:tile_filename— generate the DZI manifest and individual tiles lazily on first request, then serve from storage. SignedURLSignertoken in the URL path (not query) so OpenSeadragon's tile-URL derivation preserves it across manifest → tile fetch. The "dzi" variant name is distinct from storage variants so a leaked file-serving token can't grant tile access. Unauthorized requests return 404 to prevent UUID enumeration - Per-
file_uuid:global.set_lockmutex + double-checked locking around the generators serializes concurrent cold-path requests for the same image; different images stay parallel. Lock timeout surfaces as 503 +Retry-Afterheader - Tempfile lifecycles wrapped in
try/aftersoTessera.generate_tile/4orManager.retrieve_file/2exceptions don't leak files intoSystem.tmp_dir!() - New storage setting
storage_tile_generation_enabled(default"false") is the kill switch — when off, MediaBrowser emits no manifest URLs and the tile endpoints return 404
- New
Storage.store_system_file/3— context helper for system-managed media (tiles, manifests). Idempotent: check-then-insert with unique-violation recovery via the new dedup index, so two concurrent writers for the same key both end up with the same rowStoragequery helperexclude_system_managed/1— applied to everylist_*/count_*/ orphan / trash query so system-managed rows are invisible in the MediaBrowser regardless of how the query is composedtest/phoenix_kit/migrations/v113_test.exs— pins every V113 addition (system_managed column shape, parent_file_uuid FK with cascade verification, user_uuid nullability change, three indexes including the partial-unique dedup, CHECK constraint with a raw-SQL test that double-null inserts are rejected, comment_media table + its two indexes)dev_docs/pull_requests/2026/534-media-browser-tessera-tiles/CLAUDE_REVIEW.md— review with finding dispositions; CRITICAL + HIGH items fixed in the same release
Changed
<.translatable_field>inPhoenixKitWeb.Components.MultilangForm— wrapper nowflex flex-col gap-1and base input/textarea classes carryw-full(commits52856738,0412beaf). daisyUI 5's.labelisinline-flexand.input/.textareaareinline-block, so without forcing column direction here label and field sat on the same row. Aligns the multilang form's layout with the regular<.input>core componenttest/phoenix_kit/migrations/v106_test.exs— dropped the "schema state (verified at boot)"describeblock (its assertions pinned V112's drops, not V106's adds — moved toV112Test). File now scoped to V106'sdown/1cross-mode duplicate pre-check, matching its filename
Fixed
- V112
down/1rollback ordering (post-merge review fix, commit0cde04a8). Originaldown/1dropped columns before restoring unique indexes; if post-V112 work introduced any duplicate names (V112's whole purpose), theCREATE UNIQUE INDEXwould raise mid-rollback after columns were already gone. Reordered so index restoration runs first - V112 visible-index predicate docs/code alignment — migrator moduledoc claimed
WHERE archived_at IS NULL AND is_template = falsebut actual index only filtered onarchived_at IS NULL. Docs updated to match the emitted SQL with rationale (one partial covers both visible projects and visible templates) - MediaBrowser i18n — list-view dropdown duplicates of the folder menu (Color heading, delete data-confirm, Delete button) were missed in the original sweep; wrapped so list view matches grid view. Three flash/confirm messages embedded two independent counts in a single
gettextcall which made correct pluralization impossible in Russian (3 forms) and Estonian (2 forms); each composed from twongettextcalls injected into the surroundinggettexttemplate (PR #532)
i18n
- MediaBrowser UI strings wrapped in gettext + Russian and Estonian translations (PRs #532, plus earlier commits
c717c9d0/d44bf95c) /admin/modulespage strings wrapped in gettext + ru/et translations (PR #530)- Core sidebar tabs wired to
PhoenixKitWeb.Gettextfor ru/et (PR #529, commits43e528ac/c5fd5bae) - Sitemap settings UI strings wrapped in gettext (commit
afcc281a) - General Settings widened +
/admin/modules/languagesstrings wrapped (commit24ceec90) - Core
badge.ex+time_display.exstatus strings wrapped (commit1fe0ad78) - 9 new tab-label msgids + complete Estonian translation in
default.po(commit373478f5) - Bare
Activebadge inusers.html.heex:278wrapped (commit8a341e90)
Layout
- Admin settings pages widened for wide screens (commit
3ea70ec1) - Module settings pages — sitemap, storage, referrals — widened for wide screens (commit
d867cb57)
Hygiene
mix formatreflow of long-linegettext/ngettext/put_flashcalls inmedia_browser.exandmedia_browser.html.heex(commit9841ac31). Surfaced whenmix precommitran during V112 review-followup work; no semantic changes- New Hex dep:
{:tessera, "~> 0.1"}— OpenSeadragon wrapper used by the Deep Zoom viewer
1.7.107 - 2026-05-10
Added
- Two opt-in stateless helpers on
PhoenixKitWeb.Components.Core.TableDefault(PR #528)sort_header_cell/1— clickable<th>withhero-chevron-up-mini/-down-miniicon when active, inert label-only<th>whensortattr isnil. Configurableevent(default"toggle_sort"),target,align(:left/:right/:center). Thealignis applied to the<th class>(text-right/text-center) so non-sortable columns honour it consistently with the sortable onessort_header_cell/1emitsaria-sort="ascending"|"descending"|"none"on the<th>when sortable, omitted when inert. Pinned by 4 regression tests covering all three states + omittedsearch_toolbar/1— daisyUIinput-smwithhero-magnifying-glassicon andphx-debounce(default 300ms). Optional<form>wrap whenon_submitis set. Placeholder defaults todgettext("default", "Search...").phx-targetpropagates to both the<form>and<input>so submit-on-Enter retargets correctly when embedded in aLiveComponenttest/phoenix_kit_web/components/core/table_default_test.exs— new directory + 18 component tests; closes part of the standingcore/test-coverage TODO fromCLAUDE.md
change_pageevent handler inPhoenixKitWeb.Live.Users.LiveSessions— pagination on/admin/users/live-sessionswas bound tophx-click="goto_page"with no matchinghandle_event/3clause, so clicking any page number raisedFunctionClauseErrorand crashed the LV (PR #528 follow-up). Renamed the binding tochange_pageand added the handler mirroring the siblingusers.ex:121convention
Changed
PhoenixKitWeb.Live.Users.LiveSessions— collapsed:sort_by+:sort_orderassigns into a single:sort = %{by, dir}map; renamed event"sort_by"→"toggle_sort"with"by"param. First click on a new column sorts ascending (was descending); subsequent clicks toggle.flip_dir/1tightened fromflip_dir(_)catch-all to explicit:descclause so unintended values surface as a crash rather than silent coercion (PR #528)lib/phoenix_kit_web/live/users/live_sessions.html.heexandlib/phoenix_kit_web/live/users/users.html.heex— both call sites of<.search_toolbar>dropped the redundanton_submit="search". The input's debouncedphx-change="search"already covers the same event; keeping both made Enter fire"search"twice (immediate submit + 300ms-later debounced change) (PR #528 follow-up)
Fixed
lib/phoenix_kit_web/live/users/users.html.heex— replaced bare search form (every keystroke hit the server) with<.search_toolbar>carrying the 300msphx-debounce(PR #528)<.search_toolbar>form variant double-boundphx-changeon both<form>and<input>, doubling work per keystroke.phx-changeis now bound only on the<input>; the<form>carriesphx-submitonly.phx-targetnow propagates to both so LiveComponent embedding works end-to-end. Two regression tests pin both behaviours (PR #528, commitdfc91238)
i18n
mix gettext.extract --mergeresync — adds"Search..."msgid +et/rutranslations and surfaces accumulated drift from prior commits where extract wasn't run (PR #528, separate commita7c1d35b)
Hygiene
.gitignore— adds/priv/static/assets/vendor/somix phoenix_kit.installruns against/appitself don't leave an outdated copy of the source JS in tree (PR #528)mix.lock—db_connection2.10.0 → 2.10.1,igniter0.7.9 → 0.8.0 (pulls inex_ast0.11.0 as new transitive). Routine patch bumps
1.7.106 - 2026-05-08
Added
- V111 migration: PDF library tables for the upcoming catalogue PDF subtab (PR #516)
phoenix_kit_cat_pdfs— thin per-upload row.file_uuidFK tophoenix_kit_files(uuid)ON DELETE RESTRICT (catalogue manages the file lifecycle; core prune can't remove files referenced by a live catalogue row). Soft-delete viastatussentinel (active/trashed) +trashed_at. Two uploads of identical content (different filenames) → two rows sharing onephoenix_kit_filesrow + one extractionphoenix_kit_cat_pdf_extractions— keyed byfile_uuidPK. Worker state machine (pending → extracting → extracted | scanned_no_text | failed) +page_count+extracted_at+error_message. Cascades on file hard deletephoenix_kit_cat_pdf_page_contents— content-addressed dedup cache. PK oncontent_hash(SHA-256 hex of normalized page text). Same page text across multiple PDFs is stored once. GIN trigram index lives here so the search index doesn't grow with cross-PDF duplicationphoenix_kit_cat_pdf_pages— composite PK(file_uuid, page_number);content_hashFK to the dedup cache (RESTRICT — orphaned content rows GC'd by a catalogue-side helper, not by FK cascade)- Enables
pg_trgmextension;@current_version110 → 111
PhoenixKit.KnownPackages— live catalog of known external PhoenixKit packages, replacing the previously hardcoded list inModuleRegistry.known_external_packages/0(PR #523)- Fetched on demand from
https://hex.pm/api/packages?search=phoenix_kit_&sort=nameand cached for 10 minutes in an ETS named table (:phoenix_kit_known_packages_cache) - Stale-while-revalidate with cap: on Hex failure, serves cached data up to
:max_stale_age_ms(default 24h); beyond that, drops the cache and falls back to:extra_known_packagesconfig entries only :warninglog on stale-served and empty-cache-extras-only;:errorlog when cache exceeds max stale age — operationally distinct alert levelsLink-header pagination with a 20-page cap (@max_pages) so a malformedLinkheader pointing back to the same page can't loop foreverextra_known_packagesconfig knob — parent apps with private/forked packages declare them inline and they take precedence over Hex entries on thepackagededup key (source: "config"baked in)hex_docs_icon_name: hero-<name>convention — package authors append the marker to their Hex package description and the catalog UI picks it up; default ishero-puzzle-piece
- Fetched on demand from
- Per-module gettext support on Dashboard sidebar labels and tooltips (PR #522)
PhoenixKit.Dashboard.Tabgainsgettext_backend: module() | nil(defaultnil) andgettext_domain: String.t()(default"default") fields, pluslocalized_label/1andlocalized_tooltip/1resolvers that callGettext.dgettext/3when a backend is set and fall back to the raw label otherwisePhoenixKit.Dashboard.Groupgains the same two fields pluslocalized_label/1Tab.divider/1andTab.group_header/1accept the new opts;Tab.new/1round-trips both viaget_attr/2- 14 render sites in
Sidebar,AdminSidebar,TabItemswaptab.label→Tab.localized_label(tab)and equivalents — mechanically uniform, no shape changes - Hot-reload safety via
Map.get/2(not pattern matching) on the new fields — old-shape%Tab{}cached in ETS or:persistent_termfrom before the upgrade falls through as ifgettext_backendwerenilrather than raisingFunctionClauseError. Pinned by an explicitMap.delete(:gettext_backend)regression test guides/per-module-i18n.md— public guide for module developers (setup checklist,mix.exs/ backend /.poflow,dynamic_children/2locale handling, dividers and group headers, tooltips, greenfield template, retrofitting checklist, smoke test pattern, common pitfalls including the hot-reload safety contract)dev_docs/instructions/2026-05-08-per-module-i18n-procedure.md— internal operational procedure capturing every gotcha hit during the Newsletters pilot (skip-worktree on mix.exs, path-dep workflow during local dev, conditional CI skip pattern for graceful degradation)
:per_translation_urlsattr on the threeLanguageSwitchervariants —language_switcher_dropdown/1,language_switcher_buttons/1,language_switcher_inline/1(PR #525)- Each entry is
%{code: <display_code>, url: <full_url>}. Both atom-keyed and string-keyed entries accepted (useful when the list comes from JSON/JSONB rather than Elixir code) - Resolves each language's
base_codeagainst the list viaDialectMapper.extract_base/1so"en-US"and"en"both resolve cleanly. Falls back to the locale-rewrite default when no entry matches OR the matched entry has anilURL (e.g. an unpublished draft) - Useful when a feature module has computed canonical URLs that the simple locale-rewrite default can't reproduce — for example publishing's per-language URL slugs where
/en/blog/my-postand/fr/blog/mon-articlearen't related by segment swap. Passassigns[:phoenix_kit_publishing_translations]from the layout - 7 new tests in
test/phoenix_kit_web/components/core/language_switcher_test.exspin the contract (atom-keyed, string-keyed, full-dialect normalization, per-language fallback, nil/empty/missing-attr pass-through)
- Each entry is
- Drag-handle scoping + sortable feedback infrastructure (PR #525)
<.table_default>emitsdata-sortable-handle=".pk-drag-handle"when@on_reorderis set; only the.pk-drag-handleelement getscursor-grabstyling. Click-to-expand / button-press / text-selection on a card no longer fights with SortableJS drag detectionSortableGridJS hook: newsortable:flashLV→client event handler. The host LV pushes{uuid: "...", status: "ok" | "error"}after eachreorder_itemsattempt; the hook appliespk-sortable-flash-{ok,err}class for ~1.2s, idempotent via reflow trigger. Queries every[data-id]element so table-view + card-view both animate. Defensive status-validation guard — unknown values bail rather than falling into the err-class branch<tr>cell-width preservation viaonChoose/onUnchoose— SortableJS'sforceFallback: true+fallbackOnBody: trueclones the dragged<tr>todocument.body, where it loses its<table>ancestor and<td>s collapse to content width. The hook now snapshots computed widths and pins them inline before the drag preview renders;onUnchooserestores themdata-sortable-handleattr threads to SortableJS'shandleoption for any caller;moved_idalways included in thereorder_itemspayload (was only on cross-container moves) so the LV can push back asortable:flashkeyed to the just-moved row
- MediaBrowser modal viewer becomes the default click target for non-admin / non-select_mode browsers, with read-only image / video / PDF / icon preview, metadata sidebar, Download button, prev/next chevrons (and ←/→ keyboard shortcuts), and Esc / backdrop close (PR #519)
- Mobile-fullscreen layout via
position: fixed; inset: 0— bypasses daisyUI's grid + iOS Safari's 100vh/100dvh quirks. Desktop reverts to95vw × 90vhcentered modal with rounded corners. The!-prefix utility chain on.modal-boxis required because daisyUI v5's defaults win the cascade over plain Tailwind utilities MediaImageZoomJS hook lazy-loads Panzoom 4.6.0 from jsDelivr when the modal opens; image supports wheel/pinch/double-tap zoom and drag-pan. Listener attaches to the parent so the cursor doesn't have to land on the image;destroyedcleanup removes the wheel listener and destroys the Panzoom instance- Bulk-select still reachable — clicking the toolbar's Select button flips
select_modeon, and from then on clicks toggle selection instead of opening the modal
- Mobile-fullscreen layout via
- LiveView login redirect now carries the original request path as
?return_to=(PR #519)- New
login_path_with_return_to/1private helper inPhoenixKitWeb.Users.AuthreadsPhoenix.LiveView.get_connect_info(socket, :uri), encodespath?queryviaURI.encode_www_form/1, and threads it into the redirect target. Wired into the fourredirect_require_loginpaths inon_mounthooks - Trailing-slash self-loop guard:
String.trim_trailing(path, "/")on both sides of the equality check, so/users/log-inand/users/log-in/are treated as the same path and no return-to round-trips back to itself - Pairs with the existing
?return_to=flow inlogin.ex(sanitize_return_to/1→:user_return_tosession →log_in_user/3)
- New
PhoenixKit.ModuleRegistry.get_module_key_for_namespace/1— symmetric with the existingget_by_key/1. Resolves a top-level Elixir namespace string (e.g."PhoenixKitEntities") to the registered plugin'smodule_key/0(PR #521)- Iterates
all_modules/0, matches onModule.split(mod) == [top_namespace](exact, single segment), returns the key string ornilfor unmatched - Reads from
:persistent_termso there's no GenServer roundtrip on the hot path
- Iterates
- Microsoft 365 OAuth tenant override + generic
interpolate_url/3helper inPhoenixKit.Integrations.OAuth— providers can now substitute{key}placeholders inauth_url/token_urlfrom per-rowintegration_data, falling back to a provider-level:url_defaultsmap (PR #516)- Closes the previously hardcoded
/common/Microsoft tenant — single-tenant operators got AADSTS50194 errors. Newtenant_idsetup field withcommondefault; multi-tenant remains the default behavior. Three pinning tests intest/phoenix_kit/integrations/oauth_test.exs - Wired into
authorization_url/5,exchange_code/4,refresh_access_token/2. URLs without{pass through unchanged (zero impact on Google / OpenRouter / Mistral / DeepSeek)
- Closes the previously hardcoded
- "Resolve a LiveView module to its permission key" block-comment on
PhoenixKitWeb.Users.Auth.permission_key_for_admin_view/1documenting the four-step resolution order (static map → custom-tabs →PhoenixKit.Modules.<X>.Web.*namespace → registered-plugin namespace) and the fail-closed nil default
Changed
- DB module extracted from core into the standalone
phoenix_kit_dbHex package (PR #518)- Removed
lib/modules/db/(db.ex,listener.ex,web/{activity,index,show}.{ex,html.heex}) — ~2010 lines across 8 files module_registry.ex— droppedPhoenixKit.Modules.DBfrominternal_modules/0integration.ex— dropped the three hand-registeredlive "/admin/db…"declarations (auto-discovery viaadmin_tabs/0picks them up once the package is installed)modules.html.heex— dropped the hardcoded DB module card; auto-render via<.module_card>based onadmin_tabs/0discoverydev_docs/guides/2026-02-24-module-system-guide.md— movedlib/modules/db/db.exfrom the Internal examples section to External asphoenix_kit_db/, between hello_world and document_creator
- Removed
PhoenixKit.ModuleRegistry.not_installed_packages/0switches fromCode.ensure_loaded?(pkg.module)to OTP-app-name MapSet membership (PR #523) — the more correct semantics, since module-loading state and OTP-app-installed state aren't the same: an extracted-but-not-yet-installed module fragment could passCode.ensure_loaded?but isn't actually a depPhoenixKit.Integrations.OAuth.verify_oauth_state/2missing-state branch tightened from lenient:okto{:error, :state_mismatch}(PR #516, closes a CSRF-relevant gap from PR #511's review NIT #10) — everyconnect_oauthevent saves state viasave_oauth_state/2before redirect post-2026-05, so a missing state at callback time means either bypass or row-mutated-mid-flow; both are CSRF-relevantIntegrationPickerdrops theconn.name == "default"substitution that contradicted PR #511's own moduledoc ("Names are pure user-chosen labels with no system semantics") — always renders the user-chosen name + provider badge (PR #516, closes PR #511 NIT #6)<.file_upload>full_upload/1variant entry-progress label readsUploading… {entry.progress}%instead of the bare percentage —entry.progressis always client→server upload progress per Phoenix LV convention, so the wording is universally accurate (PR #516)LanguageSwitcherresolves the per-language URL once per iteration via inline<% url = ... %>and reuses it forhrefandphx-value-url— halves the per-renderresolve_url/3cost and pins both call sites to the same URL (post-merge triage)KnownPackagesmoduledoc grew an "Operational signals" section enumerating the three log levels (:warningstale-served / no-cache /:errorexceeded max stale age) and what each signals operationally (post-merge triage)
Fixed
- Publishing routing-strategy collision: any host route shaped
/:locale/<literal>/...declared afterphoenix_kit_routes()was silently shadowed by publishing's/:language/:group/*pathcatch-all (PR #524)phoenix_kit_routes/0now emits a publishing-specific dispatch shim whenPhoenixKitPublishing.RouterDispatchis loaded — internal-prefix scope at/<url_prefix>/__phoenix_kit_publishing_dispatchwith/localizedand/rootdiscriminator sub-scopes, plus adef call/2override that callsRouterDispatch.maybe_rewrite/1on every request and only rewrites publishing-bound URLs onto the internal prefix. Host routes get a fair shot at every URLrestore_path/2runs after route binding (via the new:phoenix_kit_publishing_internalpipeline) so canonical-URL generation reads the URL the client sent — without it, publishing'sdefault_language_no_prefixredirect would spin on the internal prefix forever- Compile-time gated on
Code.ensure_loaded?(PhoenixKitPublishing.RouterDispatch); installs without publishing in the dep tree getquote do end(no-op AST). The__mix_recompile__?/0mechanism injected byphoenix_kit_routes/0forces a host-router recompile when publishing is added or removed from deps — handles the dep-cache staleness case - Browser-smoke verified across 8 URL classes: localized + canonical publishing posts, host's
/:locale/services/view/...routes (was 404 pre-fix), admin redirects, plain home, genuine 404s. HTML body sweep confirmed zero leakage of the internal prefix in canonical / og / links / JS / headers
- Custom-role users with explicit plugin permissions (
entities,billing,ai, …) were silently locked out of plugin admin pages becauseinfer_permission_key_from_module/1only resolved the corePhoenixKit.Modules.*namespace. External plugins (PhoenixKitEntities.*,PhoenixKitBilling.*, …) returnednilfrom all three resolution paths, collapsing onto the "no permission" branch inenforce_admin_view_permission/2(PR #521)New
[top | _rest] -> ModuleRegistry.get_module_key_for_namespace(top)clause oninfer_permission_key_from_module/1. Old_ -> nilfallback removed (unreachable post-Module.split/1). Owner / Admin behaviour and the fail-closed default for genuinely unknown views are preservedInitial implementation used
[^top_namespace | _]which matched any registered module whoseModule.splitstarts with the segment; live repro on a parent app showedget_module_key_for_namespace("PhoenixKit") => "db"becausePhoenixKit.Modules.DBhappened to be the first registered module beginning with"PhoenixKit". Tightened to[^top_namespace](exact, single segment) and pinned with a regression testpermission_key_for_admin_view/1exposed as@doc false def(wasdefp) so 4 new unit tests intest/phoenix_kit_web/users/auth_test.exscan exercise the resolution layers without LiveView mounting machinery; 3 new tests intest/phoenix_kit/module_registry_test.exspinget_module_key_for_namespace/1(usesModule.create/3with explicit top-level fixture names to avoid test-module auto-nesting)
PhoenixKitWeb.PagesHTMLremoved (PR #518) — the module had no controller, no routes, no callers. Theembed_templates "pages_html/*"directive pluspages_html/show.html.heexplus theintegration.exdocstring described a markdown-page-rendering feature that was never wired up. Publishing module covers actual CMS-page rendering.ast-grep --lang elixir --pattern 'PhoenixKitWeb.PagesHTML'confirms zero structural references remainMediaBrowserchevron-button positioning: daisyUI's active-state CSS replacestransformwithscale(0.97)on click, which would clobber a-translate-y-1/2on the button itself and make it jump down 50% of its height. Chevron positioning now sits on a wrapper<div>, not the button (PR #519)media_browser.html.heexmodal-viewer leading comment referenced the now-removedviewer={true}attr; rewritten to describe the new default click behaviour (post-merge triage)- Dead
defaults[String.to_atom(key)]fallback inOAuth.interpolate_url/3— no provider inProviders.providers/0ships an atom-keyedurl_defaults, so the path was unreachable. Removed; comment documents that provider authors must use string keys (post-merge triage) KnownPackages.fetch_hex_page/3recursion grew an explicit@max_pages 20cap so a malformedLinkheader pointing back to the same page can no longer loop forever;ensure_table/0rescue now carries an explanatory comment about the:ets.whereis/1→:ets.new/2race window (post-merge triage)KnownPackagestest_helper.exs: theSystem.cmd("psql", ...)DB-existence check nowtry/rescue ErlangErrorso environments wherepsqlisn't on PATH fall through to the connect-direct branch instead of crashing the test boot (PR #523)- Pre-existing credo
Refactor.Applyopportunities on the threeapply/3calls incompile_publishing_routing/1silenced with inline# credo:disable-for-next-lineannotations and an empirically-verified comment explaining why the variable-indirection alternative (mod = ModuleName; mod.fun()) doesn't shield the compiler's static-resolution warning either.mix credo --strictnow reports zero issues across the tree (post-merge triage)
Removed
PhoenixKit.Modules.DBand the entirelib/modules/db/directory — extracted to the standalonephoenix_kit_dbHex package; companion repo TBA (PR #518)PhoenixKitWeb.PagesHTMLand itspages_html/show.html.heextemplate — dead code, never wired up to a controller or route (PR #518)MediaBrowser's:viewerattr — the four-mode click handler collapsed to three modes (select_mode→admin→ modal viewer); pickers reachselect_modevia the toolbar's Select button. The default click action is now the modal viewer, so callers that previously passedviewer={true}see no behaviour change. Callers that depended on the old picker-by-default (noadmin, noviewer→ click toggles selection) need to instruct users to click the toolbar's Select button instead (PR #519)
1.7.105 - 2026-05-05
Added
PhoenixKit.Migration.ensure_current/2— re-runnable analog ofmix ecto.migratefor test helpers and any boot path running against a long-lived database (PR #515)- Passes a fresh wall-clock version (
:os.system_time(:microsecond)) toEcto.Migrator.up/4on every call so Ecto sees a "new" migration each time and invokes the inner runner; PhoenixKit's own marker (the comment on thephoenix_kittable) short-circuits internally if there's nothing new to apply - Forwards
:prefixfrom the Ecto.Migration runner context inside the new privatePhoenixKit.Migration.Runnerwrapper so callers passingprefix: "auth"aren't silently routed to"public" - Microsecond precision keeps the collision and clock-skew windows small enough that an NTP correction would have to rewind the clock by µs at exactly the wrong moment to hide a newly-shipped migration; bigint-safe (Postgres covers ~292 years)
- The
schema_migrationstable accumulates one row per call — cosmetic noise acceptable for the test-DB use case; production migrations viamix ecto.migrate/mix phoenix_kit.updateremain unchanged
- Passes a fresh wall-clock version (
- V110 migration: nullable
language VARCHAR(10)column onphoenix_kit_doc_templatesso each Document Creator template can be tagged with a single locale (PR #515)- Full locale codes (
en-US,et-EE,ja) — matchesPhoenixKit.Module.Languages.get_enabled_languages/0output; lossless, consumers that want bare base codes can derive them viaDialectMapper.dialect_to_base/1 - Existing rows survive without a backfill; the form (landing in
phoenix_kit_document_creatorseparately) pre-selects the project's primary language when creating new templates - Documents intentionally do not get a language column — they inherit from
template_uuid → templates.language @current_version109 → 110; ⚡ LATEST tag moved off V109 onto V110
- Full locale codes (
PhoenixKit.Migration.Runner.runner_opts/1— pure transform of the runner-context prefix into opts threaded toPhoenixKit.Migration.up/1/down/1(PR #515 review follow-up)- Split out of the previous closure-style
runner_opts/0so the prefix-forwarding behaviour can be regression-tested without spinning up a realEcto.Migration.Runnerprocess (which conflicts with the Ecto sandbox) - Three new unit assertions in
test/phoenix_kit/migration_test.exspin the contract:nil → [](drop, sowith_defaults/2's"public"default isn't clobbered),"auth" → [prefix: "auth"], arbitrary tenant prefix forwarded verbatim. If someone "simplifies"runner_optsto always return[], CI now fails
- Split out of the previous closure-style
- "Return contract" section in the
ensure_current/2moduledoc clarifying that failures (advisory-lock contention, migration crashes, connection errors) raise fromEcto.Migrator.up/4rather than being wrapped in{:error, _}(PR #515 review follow-up)
Changed
test/test_helper.exsswitched from path-formEcto.Migrator.run(repo, migrations_path, :up, all: true)toPhoenixKit.Migration.ensure_current/2(PR #515)- Deletes the now-redundant wrapper migration
test/support/postgres/migrations/20260316000000_add_phoenix_kit.exs
- Deletes the now-redundant wrapper migration
AGENTS.mdtest-infra section updated:test_helper.exsis now the canonical migration application point, with a Do not warning against the stale tuple formEcto.Migrator.run(repo, [{0, PhoenixKit.Migration}], :up, all: true)(PR #515)
Fixed
- Documented test-helper migration patterns silently went stale after the first run (PR #515)
- Both the tuple form (
Ecto.Migrator.run(repo, [{0, PhoenixKit.Migration}], :up, all: true), documented indev_docs/migration_cleanup.md) and the path form (used by core's own test_helper via the20260316000000_add_phoenix_kit.exswrapper) hit the same trap: Ecto.Migrator records the version inschema_migrationsafter the first call and filters that entry out of pending on every subsequent boot.PhoenixKit.Migration.up/1was never re-invoked, so newly-shipped Vxxx migrations didn't apply on subsequent boots even though PhoenixKit's own marker was idempotent. Symptom:column ... does not existaftermix deps.update phoenix_kitbrought in new migrations but the test DB stayed at the old marker - Verified empirically — core's own
phoenix_kit_testwas at marker 107 even though Hex 1.7.103 shipped V108 + V109; first boot after switching toensure_current/2advanced the marker through V108 / V109 / V110 correctly
- Both the tuple form (
1.7.104 - 2026-05-04
Changed
- Customer Service module extracted from core into the standalone
phoenix_kit_customer_supportHex package — companion repo: BeamLabEU/phoenix_kit_customer_support (PR #514)- Removed
lib/modules/customer_service/(~6 KLOC, 22 files) andlib/phoenix_kit_web/routes/customer_service.ex; the module is now an external optional dependency module_registry.ex— droppedPhoenixKit.Modules.CustomerServicefrominternal_modules/0, added the correspondingphoenix_kit_customer_supportentry toknown_external_packages/0integration.ex— replaced inline/dashboard/customer-service/ticketsroute blocks withCode.ensure_loaded?(PhoenixKitCustomerSupport.Web.UserList)guards so absent-package = no routes- DB tables (
phoenix_kit_tickets,phoenix_kit_ticket_*) stay in core under their existing names — they're domain-shaped, not module-shaped, and the prior migrations (V35/V51/V53/V58/V72/V74/V75/V77) remain in core's migration history
- Removed
- Renamed "Customer Service" → "Customer Support" across the public surface (PR #514)
- Module:
PhoenixKitCustomerService→PhoenixKitCustomerSupport - OTP app:
:phoenix_kit_customer_service→:phoenix_kit_customer_support - Hex package:
phoenix_kit_customer_service→phoenix_kit_customer_support - Settings keys:
customer_service_*→customer_support_*(7 keys) - URL paths:
/customer-service/*→/customer-support/*(admin + user-facing, both base and locale-prefixed routes) - Permission key:
customer_service→customer_support - Dashboard module card and admin nav target updated to match
- Module:
Added
- V109 migration: rename Customer Service module identifiers in-place so existing installs migrate cleanly (PR #514)
- Renames 7 settings keys from
customer_service_*→customer_support_*inphoenix_kit_settings - Renames
auto_granted_perm:customer_service→auto_granted_perm:customer_support - Renames
phoenix_kit_role_permissions.module_keyfromcustomer_service→customer_support - Idempotent (
IF EXISTSguards on every rename); reversibledown/1for emergency rollback @current_version108 → 109; ⚡ LATEST tag moved off V107 onto V109
- Renames 7 settings keys from
Fixed
PhoenixKit.Users.Auth.anonymize_user_tickets/1was a no-op since the original Tickets → CustomerService rename —Module.concat([PhoenixKit, Modules, Tickets, Ticket])resolved to a never-loaded module so theCode.ensure_loaded?guard always failed and ticket anonymization silently skipped on user deletion. Now points atPhoenixKitCustomerSupport.Ticket(PR #514)- V108 (drag-and-drop position columns, shipped in 1.7.103) was missing from the
lib/phoenix_kit/migrations/postgres.exper-version docstring catalog. Backfilled in this release alongside the V109 entry (PR #514 review) lib/phoenix_kit/migrations/postgres/v109.exrename_role_permission/4carried an unused_prefixarg — the table name is already prefix-qualified at the call site. Trimmed to/3(PR #514 review)
1.7.103 - 2026-05-02
Added
- V107 migration: pin AI endpoints to a specific integration row via
integration_uuid+ add the missing unique index onlower(name)(PR #511)- Nullable
integration_uuid uuidcolumn onphoenix_kit_ai_endpointswith btree index - Backfill maps existing
providerstrings to integration rows: exact"provider:name"matches get the corresponding storage row; bare"provider"gets the most-recently-validatedintegration:provider:*row, tiebreaking onuuid ASC(UUIDv7 time-ordered). Unresolvable endpoints stay NULL - Unique index
phoenix_kit_ai_endpoints_name_index ON (lower(name))— theunique_constraint(:name)declaration in the changeset has been dead code since V34 created this table without the index
- Nullable
- V108 migration:
position integer DEFAULT 0on three admin list surfaces —phoenix_kit_entities,phoenix_kit_cat_catalogues,phoenix_kit_cat_items— so drag-and-drop reordering can persist user-driven order (PR #512) - Strict-UUID Integrations public API (PR #511)
- Write-side APIs now take only the integration row's uuid — no more deriving storage keys from JSONB fields
Integrations.resolve_to_uuid/1— dual-input lookup primitive that accepts a uuid or aprovider:namestring (formigrate_legacy/0callbacks)migrate_legacy/0optional callback onPhoenixKit.Module— each module owns its legacy data shape; core provides primitives. Orchestrated byPhoenixKit.ModuleRegistry.run_all_legacy_migrations/0- Mistral, DeepSeek, and Microsoft 365 added to the built-in providers registry
integration_pickerupdated: no auto-select on single-provider, toggle-to-deselect support
- Drag-and-drop core infrastructure (PR #512)
<.draggable_list>new:draggableboolean attr (defaulttrue) — when false, renders without SortableJS hook and grab-cursor styling<.table_default>new:on_reorder,:reorder_scope,:reorder_group,:item_idattrs — wire the card-view container as a SortableGrid hook target for cross-container dragSortableGridhook (JS):data-sortable-groupfor cross-container drag,readScope/1helper fordata-sortable-scope-*attrs, cross-containeronEnddetection withfrom*scope prefix,try/catchwrappingTableCardViewhook (JS):updated()callback re-applies saved view mode after LV re-renders so card/table toggle survives SortableJS drops
- Media viewer modal on
MediaBrowser(PR #513)- New
viewer={true}attr — clicking a file opens an in-place modal with image/video/PDF/icon preview, metadata sidebar (filename, type, MIME, size, uploaded date), and Download button. Closes via X / Esc / backdrop - Prev/next chevrons and ArrowLeft/ArrowRight keyboard shortcuts step through the current page's files; arrows hide at boundaries
PhoenixKitComments.Web.CommentsComponentembedded in the sidebar when the Comments module is installed and enabled (optional-dep wiring:@compile {:no_warn_undefined}+Code.ensure_loaded?+@dialyzer :nowarn_function)
- New
- Arity-2
dynamic_children_fn@typedoc+ test-only delegate for the admin sidebar dispatcher (PR #506 follow-up in #512)
Changed
handle_event("click_file", …)in MediaBrowser refactored from two-modeif/elseto four-clausecond:select_mode→admin→viewer→ picker default (PR #513)connected_atsemantics clarified in AGENTS.md — rewritten on every successful re-test (not one-shot);last_validated_atrewritten unconditionally on every validation attempt, success or failure- Bumped
leafeditor dependency~> 0.2.10 → ~> 0.2.11and the matching CDN URL (PR #513)
Fixed
- AGENTS.md doc drift:
PhoenixKit.Modules.run_all_legacy_migrations/0corrected toPhoenixKit.ModuleRegistry.run_all_legacy_migrations/0; V107 moduledoc tiebreak clarified asuuid ASCnotinserted_at ASC(PR #511 review) - V107 unique-name index verified with three new integration tests: index exists, duplicate names rejected, case-only differences collide (PR #511 review)
- Media viewer modal:
String.starts_with?/2guarded withis_binary(f.mime_type)so nil mime_type falls through to the icon fallback instead of crashing (PR #513 review) - Media viewer modal: PDF iframe hardened with
sandbox="allow-same-origin"to block embedded JavaScript in same-origin deployments (PR #513 review) <.draggable_list>data-idnow always emitted regardless of:draggableattr so click-to-select handlers and test selectors work in both modes (PR #512 review):reorder_scopeattr doc on<.table_default>now documents the camelCase round-trip (:category_uuid→"categoryUuid"in the LV handler payload) (PR #512 review)
1.7.102 - 2026-04-29
Added
- V105 migration: CRM tables for the upcoming
phoenix_kit_crmplugin (PR #507)phoenix_kit_crm_role_settings— one row per role withenabled BOOLEAN NOT NULL DEFAULT falseso existing roles stay opted out until explicitly enabled. PK onrole_uuid; FK →phoenix_kit_user_roles(uuid)ON DELETE CASCADEphoenix_kit_crm_user_role_view— per-user, per-scope view preferences (column selection, ordering, filters). UUIDv7 PK; unique(user_uuid, scope); index on(user_uuid); FK →phoenix_kit_users(uuid)ON DELETE CASCADE.scopeis a string like"role:<uuid>"or"companies"
- V106 migration: split
phoenix_kit_projects.nameuniqueness across templates and real projects (PR #510)- Replaces V101's single global unique index on
lower(name)with two partial unique indexes:phoenix_kit_projects_name_template_index WHERE is_template = trueandphoenix_kit_projects_name_project_index WHERE is_template = false - Lets a template
"Onboarding"and a real project"Onboarding"coexist, unblockingProjects.create_project_from_template/2for the common reuse-the-template-name path down/1recreates V101's single global index; lossy if a template and a real project share a name post-V106 — resolve duplicates before rolling back
- Replaces V101's single global unique index on
- Legal module i18n — translations across
de/fr/it/plplus refreshedru/es. Newde/fr/it/plPOs created viamix gettext.merge --localewith properPlural-Formsheaders (Germann != 1, Frenchn > 1, Italiann != 1, Polish 3-form rule). Pre-existing non-emptymsgstrvalues preserved (PR #509) lib/phoenix_kit_web/legal_gettext_manifest.ex— re-emits the 50 translatable strings used byphoenix_kit_legalso the gettext extractor (which doesn't walk into deps) records them into core's POT. Never called at runtime; pure extraction target with refresh procedure documented in the moduledoc (PR #509)css_sources/0accepts string entries —@callback css_sources()widened from[atom()]to[atom() | String.t()]. Strings flow throughformat_source/2→source_for_path/1(absolute paths emit@source "<abs>";verbatim, relative get the standard../../prefix); atoms continue to resolve via parent app's mix.exs deps. Lets modules mix OTP-app atoms with literal path strings — first known consumer isphoenix_kit_legal, which ships a path-dep absolute fallback alongside its OTP-app entry so both Hex and path-dep installs work without parent-app toggles. Backwards compatible: existingdef css_sources, do: [:phoenix_kit_my_module]keeps working unchanged (PR #509)
Changed
- Bumped
leafeditor dependency~> 0.2.6 → ~> 0.2.10and the matching CDN URL inpriv/static/assets/phoenix_kit.jsso the runtime loader pulls the same version. Includesmin-width: 0+ toolbar-wrap fixes so the editor stops claiming an unbounded intrinsic width on mount (PR #508) priv/gettext/default.potcleanup — dropped ~900 phantom msgids left over from modules extracted to standalone packages (billing, publishing, entities, etc.) (PR #509)
Fixed
application/pdfuploads in MediaBrowser.determine_file_type/1returned"pdf", but theFilechangeset validatesfile_typeagainst["image", "video", "audio", "document", "archive", "other"]— every PDF upload silently failed validation and never reached any bucket. Now mapsapplication/pdf→"document", matching how form-upload integrations already classify PDFs (PR #507)- V106
COMMENT ON TABLEversion values were off by one (upwrote'105'instead of'106',downwrote'104'instead of'105'). The migration framework reads this comment as the source of truth for the migrated version, so on the incremental V105 → V106 upgrade path the comment never advanced past'105'— V106.up would replay on every deploy and the admin dashboard /mix phoenix_kit.statuswould report a stale version. Fresh installs masked the bug becausehandle_version_recording/4stamps the final version on multi-step runs and overrode V106's bad write. Caught in review of PR #510 and amended in place since V106 had not yet shipped to Hex
1.7.101 - 2026-04-24
Added
- Notifications module — per-user inbox driven by the activity log. When
PhoenixKit.Activity.log/1records an entry withtarget_uuid != actor_uuid, a row is inserted intophoenix_kit_notificationsfor the target user. Independentseen_at/dismissed_atper row, per-user PubSub topic ("phoenix_kit:notifications:<user_uuid>"), global kill-switch vianotifications_enabledsetting (default"true"). Admins still audit via/admin/activityand don't receive notifications (PR #505)- V104 migration:
phoenix_kit_notificationswith UUIDv7 PK, FKs tophoenix_kit_activitiesandphoenix_kit_users(bothON DELETE CASCADE), unique(activity_uuid, recipient_uuid)index, partial(recipient_uuid, inserted_at DESC) WHERE dismissed_at IS NULLindex for the inbox read path PhoenixKit.Notificationspublic API:maybe_create_from_activity/1,list_for_user/2,recent_for_user/2,count_unread/1,mark_seen/2,mark_all_seen/1,dismiss/2,dismiss_all/1,get_notification/2,enabled?/0,retention_days/0,prune/1PhoenixKit.Notifications.Render.render/1— maps action →%{icon, text, link, actor_uuid}; honors metadata overrides (notification_text,notification_icon,notification_link) before falling back to the action lookupPhoenixKit.Notifications.Typesregistry — three core types (account,posts,comments) plus extension point for external modules via the new optionalnotification_types/0callback onPhoenixKit.ModulePhoenixKit.Notifications.Prefs— per-user preferences persisted incustom_fields.notification_preferences(reuses V18 JSONB column; no migration). Fail-open on any ambiguityPhoenixKit.Notifications.PruneWorker— daily Oban cron at"0 4 * * *"; retention vianotifications_retention_days(falls back toactivity_retention_days, default 90)PhoenixKitWeb.Live.NotificationsBell— sticky nested LiveView for the bell + dropdown. Not mounted by default; parent apps render it where they have a user-facing header viaPhoenix.Component.live_render(..., sticky: true, session: %{"user_uuid" => ...}). Badge + recent list refresh live via PubSub- Notification preferences section in
PhoenixKitWeb.Live.Components.UserSettings— one toggle per registered type; unknown submitted keys dropped at the call site notifications_enabledtoggle on/admin/settings
- V104 migration:
- Arity-2
dynamic_children_fnfor admin sidebar tabs — callbacks can now be(scope, locale -> [tab])in addition to the existing(scope -> [tab]). Backwards-compatible extension: the sidebar dispatches on arity, every existing 1-arity callback keeps working unchanged. Lets plugins render locale-aware child labels without readingGettext.get_locale/1at render time (PR #506)
1.7.100 - 2026-04-22
Added
- V103 migration: nullable self-FK
parent_uuidonphoenix_kit_cat_categorieswith b-tree index on(parent_uuid)for arbitrary-depth category trees. Existing rows stayNULLand become roots — no backfill. No DB-levelON DELETEcascade (subtree cascades are owned by the context layer so they go through soft-delete + activity log) (PR #503) scope_folder_idattr onPhoenixKitWeb.Live.Components.MediaSelectorModal— filters the browse query to the given folder plus any files reached viaFolderLink, and assigns newly-uploaded files into that folder (adopt as home if orphan, else add aFolderLink). Plugins scoping the picker to a single domain object (e.g. a catalogue item) pass this after lazy-creating their folder (PR #503)PhoenixKit.Settings.Setting.optional_settings/0accessor exposing@optional_settingsfor invariant tests- Invariant test (
test/phoenix_kit/settings/setting_test.exs) asserting every empty-string default inPhoenixKit.Settings.get_defaults/0is also in@optional_settings, to prevent the class of bug fixed in PR #502 from recurring
Changed
PhoenixKit.Modules.Storage.Filechangesetfile_typeallowlist widened from["image", "video", "document", "archive"]to include"audio"and"other"so non-image/video uploads bucket cleanly (PR #503)MediaSelectorModal.load_files/2refactored into four composablescope_files_by_{user,folder,type,search}helpers — credo cyclomatic-complexity fix from adding the new scope branch (PR #503)
Fixed
- Settings batch save no longer rolls back when
site_icon_file_uuidordefault_tab_titleis left empty on the General Settings form. Both keys added to@optional_settingsinPhoenixKit.Settings.Settingand seeded with empty-string defaults inPhoenixKit.Settings.get_defaults/0(PR #502) MediaSelectorModal.maybe_set_folder/2errors (from thefolder_uuidupdate orFolderLinkinsert) now log a warning viawarn_on_folder_error/3instead of being silently discarded by_ =. Previously, a failed scope assignment after a successful upload left no trace
1.7.99 - 2026-04-20
Added
- V100 migration: staff tables —
phoenix_kit_staff_departments,phoenix_kit_staff_teams,phoenix_kit_staff_people,phoenix_kit_staff_team_memberships(PR #498) - V101 migration: projects tables —
phoenix_kit_project_tasks,phoenix_kit_project_task_dependencies,phoenix_kit_projects,phoenix_kit_project_assignments,phoenix_kit_project_dependencies; polymorphic assignee withCHECK (num_nonnulls(...) <= 1)(PR #498) - V102 migration: smart catalogues + per-catalogue/item discount (PR #500)
phoenix_kit_cat_catalogues.discount_percentage(NOT NULL DEFAULT 0) andkind('standard' | 'smart') columns with CHECK constraintsphoenix_kit_cat_items.discount_percentage,default_value,default_unitoverride columns- new
phoenix_kit_cat_item_catalogue_rulestable with unique(item_uuid, referenced_catalogue_uuid)and ON DELETE CASCADE on both FKs - partial index on
kind = 'smart'
PhoenixKitWeb.Components.MediaBrowser.Embed— one-lineusemacro that injectson_mountupload setup, the"validate"upload-channel stub, and the MediaBrowserhandle_infodelegator (PR #499)- MediaBrowser selection menu with bulk download (staggered
<a download>dispatch viaMediaDragDrophook) (PR #499) - MediaBrowser
adminattr to gate detail-pagepush_navigate— picker mode (default) vs admin mode (PR #499) - MediaBrowser drag-drop file-to-folder move (PR #499)
- MediaBrowser toggleable search bar in the header (PR #499)
- MediaBrowser drag-drop upload at any folder level (PR #499)
- Site icon + default tab title settings, logo moved to main settings page (PR #499)
- MultilangForm debounce flow:
mount_multilang/1attaches a hidden:handle_infohook viaPhoenix.LiveView.attach_hook/4;handle_switch_language/2schedules a 150 ms trailing debounce viaProcess.send_after(timer ref stored insocket.privateto avoid render+diff cycles);switch_lang_js/2toggles skeleton/fieldshiddenclasses client-side at t=0 (PR #500) <.input>gains awrapper_classattr for the outerphx-feedback-fordiv (PR #500)test_load_filters/test_ignore_filtersinmix.exsfor Elixir 1.19mix testhygiene (PR #500)- AGENTS.md: Core Form Components section, Multilang Form Components section, and CHANGELOG-ownership rule (entries written by the maintainer, not agents)
Changed
- MediaBrowser sidebar and content unified into a single card (PR #499)
- Scope-root new-folder form aligned with sibling folder rows (PR #499)
- Core form components (
<.input>,<.select>,<.textarea>,<.checkbox>) now merge theclassattr onto the styled element itself — matches the Phoenix 1.7 generator convention. No in-tree caller used the old wrapper-class behavior; external consumers should switch towrapper_classon<.input>(PR #500) compile.phoenix_kit_css_sourcesemits absolute dep paths verbatim instead of prefixing../../(PR #500)
Fixed
- MediaBrowser list view broken by stale view-toggle CSS (PR #499)
- Credo
AliasUsagewarning insideMediaBrowser.Embed's quoted block silenced (PR #499)
1.7.98 - 2026-04-16
Added
- V99 migration:
trashed_atcolumn onphoenix_kit_fileswith partial index for soft-delete (PR #497) - Media trash bucket: soft-delete files with restore/empty/permanent-delete actions and sidebar count badge
PhoenixKit.Modules.Storage.Workers.PruneTrashJob— daily Oban cron (3 AM) that permanently deletes files older thantrash_retention_days(default 30)- Drag-drop upload: drop device files directly onto the folder content area (
FolderDropUploadJS hook) - URL-param hydration on first mount so reloads don't flash the root view
Fixed
- Scope guard on
restore_selectedin MediaBrowser — a scoped embed could previously restore files outside its scope via a craftedtoggle_selectpayload - Trash view, permanent-delete, and
empty_trashnow respectscope_folder_idvia recursive CTE list_files/1excludes trashed files- Breadcrumb and search bar moved inside card body so padding matches grid/list content
1.7.97 - 2026-04-15
Added
- V97 migration: per-item
markup_percentageoverride on catalogue items (PR #493) - V98 migration:
alternative_formatscolumn on storage dimensions PhoenixKit.Modules.Shared.Components.ImageSet— responsive<picture>component with AVIF/WebP/JPEG<source>entriesPhoenixKit.Modules.Storage.VariantNaming— format-suffix parsing utility- Multi-format variant generation (WebP/AVIF alongside primary format per dimension)
- Variant dimensions and file sizes shown on media detail page
- UUID search support on media page search bar
Changed
- V95 migration made truly idempotent for
folder_uuidcolumn (raw SQLIF NOT EXISTSblock) - Dimensions table format cell renders as
JPEG + WEBP, AVIF(fixed stray" +"separator)
Fixed
- Long text overflow in media detail sidebar
- Missing original file size in variant download buttons
1.7.96 - 2026-04-13
Added
- Sortable languages in admin (drag-and-drop reorder)
- hide_source option on DraggableList component
- Wiggle animation for reorder mode with prefers-reduced-motion support
Changed
- Dedup language codes in reorder, use MapSet for lookup
- Extract wiggle CSS to JS-injected styles with pk- prefix
1.7.95 - 2026-04-11
Added
- V95 migration: media folders and folder links tables
- V96 migration: catalogue_uuid FK on catalogue items for direct catalogue membership
1.7.94 - 2026-04-10
Added
- Media folder system with sidebar, select mode, list/grid view, drag-drop file moving
- Folder colors, inline rename, select folders, and context menus
- Search bar and folder path column to media page
- Media Health page for redundancy monitoring
- Sync with progress tracking, pause/resume/stop controls, real-time sync log
- Media sync moved to Oban worker for persistence and reliability
- Multipart S3 uploads
- Tigris storage provider support
- Test Connection button to bucket form
- Configurable max upload size setting
- Reusable SearchableSelect LiveComponent
- Provider-specific labels for B2, R2, and S3 bucket configuration
- AWS regions dropdown via
aws_regionshex package
Changed
- Rename Storage to Media, use thumbnail variant for media grid
- Redesign media sidebar with proper file explorer conventions
- Migrate bucket and dimensions tables to
table_defaultcomponent - Restyle bucket/dimension forms with card sections and DaisyUI 5 fieldset/legend
- Persist folder tree expand state and sidebar collapsed in localStorage
- Remove LLMText module (preserved in feature/llmtext branch)
- Remove CDN URL field from bucket configuration form
Fixed
- Fix #478: register CSS sources compiler and create stub file on install
- Fix integration activity logs always having nil actor_uuid
- Fix S3 upload failures and false file location records
- Fix folder card hover lag (use
transition-colorsinstead oftransition-all) - Fix folder name truncation, rename animation, and input autofocus
- Fix select mode content jump and improve render performance
- Fix mix tasks:
Routes.pathunavailable,ecto.migrateskips host repo - Fix OAuth users getting signed out after ~1-2 hours
1.7.93 - 2026-04-08
Fixed
- Fix installer to auto-inject PhoenixKitHooks into app.js
- Fix decrypt after legacy integration migration
- Improve OAuth login with remember_me by default
Added
- Activity logging for Integrations (setup, connect, disconnect, token refresh, validation)
- Cookie max_age set to 60 days
Changed
- Simplify integration status:
configuredremoved, nowconnectedordisconnected - Validate connection checks provider exists before credentials
1.7.92 - 2026-04-07
Fixed
- Fix Google token refresh for named integration connections (e.g. google:work)
- Add resolve_provider_lookup_key/2 and resolve_storage_key/2 helpers
Added
- Add V94 migration for Document Creator sync (google_doc_id, status, path, folder_id columns)
1.7.91 - 2026-04-06
Added
- Add centralized Integrations system for external service connections (OAuth, API keys, bot tokens)
- Add AES-256-GCM encryption at rest for stored credentials
- Add OAuth 2.0 CSRF state parameter protection
- Add
required_integrations/0andintegration_providers/0callbacks to PhoenixKit.Module - Add IntegrationPicker reusable component
- Add Integrations admin settings tab
- Add provider registry with Google and OpenRouter built-in
Fixed
- Fix password field overwrite bug when editing integrations
- Fix duplicate line in
maybe_set_userinfo/2 - Consolidate validation logic into Integrations context
- Make validation URL provider-configurable (no more hardcoded Telegram URL)
1.7.90 - 2026-04-04
Added
- Add organization accounts support with person/organization user types
- Add organization invitations system with token-based invite flow
- Add V91 locations migration: location types, locations, and type assignments tables
- Add V92 organization accounts migration (invitations, user type fields)
- Add JS hooks integration for parent app install and update workflows
- Add LLMText module for AI/LLM-friendly content generation
- Add auth logo from settings to admin header
- Add billing tabs component
Changed
- Move tax rate and CountryData from Billing to core PhoenixKit
- Remove hardcoded Billing and E-Commerce module cards in favor of auto-discovery
- Update AGENTS.md with severity level definitions and JS hooks documentation
Fixed
- Fix double sidebar for core modules and improve struct compatibility
- Hide hamburger menu button when sidebar is permanently visible
- Fix token security, gettext, and validation issues in organization invitations
- Fix tax data loss, invitation status guard, and IbanData safety
1.7.88 - 2026-04-02
Changed
- Migrate select elements to daisyUI 5 label wrapper pattern (#472)
Fixed
- Fix negated condition in maintenance toggle flash message
- Fix dialyzer warnings for CSS sources compiler, clean up 6 stale ignore entries
1.7.87 - 2026-03-31
Added
- Add V89 migration: catalogue pricing with base_price and markup_percentage
- Add status_badge component and wrapper_class attr to table_default
- Add inline and auto display modes to table_row_menu
- Add show_toggle attr to table_default and sync TableCardView instances
- Add continent grouping to language switcher for many languages
- Add language system tests, docs, error handling, and group_by_continent option
Changed
- Unify admin and frontend language systems into single source of truth
- Unify status badge components into single status_badge
- Remove deprecated select-bordered class for daisyUI 5 compatibility
- Disable automatic CI triggers, switch to manual-only
Fixed
- Fix language switcher URL generation for prefixed admin paths
- Fix dialyzer warnings in language switcher URL generation
1.7.86 - 2026-03-30
Changed
- Update Shop module references from
PhoenixKit.Modules.ShoptoPhoenixKitEcommercenamespace - Update Billing module references from
PhoenixKit.Modules.BillingtoPhoenixKitBillingnamespace - Restore LayoutWrapper in core module templates (not auto-applied to bundled modules)
- Document
extra_applicationsrequirement for external module auto-discovery
Fixed
- Fix missing
url_pathassign in referrals LiveViews causing runtime crash after LayoutWrapper restoration - Fix media selector modal mobile responsiveness (header overflow, button sizing, padding)
1.7.85 - 2026-03-30
Added
- Add user-scoped media selector for avatar and fix custom fields position bug
Changed
- Remove Pages module from core and clean up all references
- Extract Connections module into external
phoenix_kit_user_connectionspackage - Remove unused Storage alias from user settings
1.7.84 - 2026-03-28
Added
- Add missing cookie consent widget to dashboard layout
- Add user dashboard routes for billing profiles
Changed
- Remove Legal module from core (extracted to
phoenix_kit_legalpackage) - Remove LayoutWrapper from remaining storage and maintenance templates
- Remove duplicate LayoutWrapper from admin module templates
Fixed
- Fix core modules misclassified as external plugin views causing double admin chrome
1.7.83 - 2026-03-27
Added
- Add V88 migration: Publishing schema V2 restructure
- Add user dashboard generator with LiveView templates and standardize layout
- Add
--indexflag to user dashboard generator for overriding default dashboard - Add Estonian to backend languages, fix Chinese code zh-CN → zh
- Add CountryData to core utils for billing extraction
- Add sitemap scheduler startup recovery
Changed
- Extract Comments module into external
phoenix_kit_commentspackage - Remove Shop module from core (extracted to
phoenix_kit_ecommercepackage) - Remove Billing module from core (extracted to
phoenix_kit_billingpackage) - Replace hardcoded external module stats with generic
module_statscallback - Remove hardcoded module cards for extracted packages
- Rename and simplify admin page generator
- Update Leaf dependency to v0.2.6
Fixed
- Fix V88 migration: index prefix and partial re-run safety
- Fix orphan files query to use
publishing_versionstable - Fix post-review issues from PR #453: Shop.Cart guard, consent attrs, language naming
- Fix shop modules: remove billing struct patterns and fix nil clause ordering
- Fix double navbar on comments admin pages
- Fix auth page background breaking footer and page layout
1.7.82 - 2026-03-24
Added
- Add V86 migration: Document Creator tables (headers_footers, templates, documents)
- Add V87 migration: Catalogue tables (manufacturers, suppliers, catalogues, categories, items)
- Add
system_promptfield to AI prompts and AI Playground page - Add database connection check to install and update tasks
- Add AdminEditHelper for universal admin edit links in public views
- Add email provider behaviour, refactor Mailer and UserNotifier
- Add lastmod to sitemap group listings and homepage
- Enrich external module cards with config stats, settings link, and
module_cardcomponent
Changed
- Extract Emails module from core to standalone
phoenix_kit_emailspackage - Extract Publishing module to standalone
phoenix_kit_publishingpackage - Extract Entities module to standalone
phoenix_kit_entitiespackage - Extract AI module to standalone
phoenix_kit_aipackage - Remove hardcoded Emails block from Modules page — now rendered as external package
- Guard all Publishing references behind
Code.ensure_loaded?for external module support - Guard EntityForm render call with
Code.ensure_loaded?check in Pages renderer - Suppress warnings for optional external modules with
@compile :no_warn_undefined - Exclude external module namespaces from Credo alias usage check
- Make module registry and permissions tests count-independent after module extractions
- Document
ensure_compiledvsensure_loaded?choice in integration route collection - Update Leaf dependency to v0.2.5
Fixed
- Fix V86/V87 migrations to use
uuid_generate_v7()instead ofgen_random_uuid() - Fix post-merge issues from Emails extraction
- Fix
extract_admin_links: skip parent tabs, deduplicate paths - Fix
external_plugin_view?to recognizePhoenixKit.Modules.*.Webas external packages - Fix DbConnectionCheck: correct spec, naming, and remove hard exit from status task
- Fix media selector modal z-index to appear above all overlays
- Fix
module_cardto renderhero-*icons properly - Fix cookie consent: dynamic legal links, theme-aware backdrop, daisyUI toggle
1.7.81 - 2026-03-21
Changed
- Extract Posts module to standalone
phoenix_kit_postspackage - Update Comments module to conditionally load Posts handler via
Code.ensure_loaded?/1 - Update scheduled jobs worker with extracted catch-up helpers for optional Posts dispatch
1.7.80 - 2026-03-20
Added
- Add
uuidtype to custom fields system - Add auto-registration of custom field definitions on save with type inference
Changed
- Extract Sync module to standalone
phoenix_kit_syncpackage - Move custom fields domain logic to
CustomFieldsmodule, deduplicate UUID regex, add error logging - Fix permissions table style — replace manual zebra striping with daisyUI
table-zebra, use primary-colored header
Fixed
- Fix avatar upload handling and
custom_fieldspreservation in UserSettings - Fix admin page and user dashboard styles
- Fix plugin reference name in module system guide
1.7.79 - 2026-03-20
Fixed
- Fix UserSettings regressions from PR #436 redesign:
- Restore timezone selector (timezone select, mismatch warning, browser detection)
- Restore Apple OAuth provider icon (
hero-device-phone-mobile) - Restore OAuth-only password warning for users without passwords
- Restore provider email display in connected accounts list
- Fix custom field
selectusing index-based values instead of actual option values (data compatibility break) - Restore all custom field input types (
textarea,number,email,url,date) — were collapsed to plain text - Restore
requiredattribute on custom field inputs - Restore unique
idattributes on password/email form hidden inputs - Restore profile/avatar success and error messages in template
- Fix
shadow-xl→shadow-smfor card styling consistency - Fix divider placement — move out of username field, add "Additional Information" heading for custom fields
- Extract
extract_custom_fields/1andmerge_custom_fields/3helpers to DRY duplicated logic
1.7.78 - 2026-03-18
Added
- Add Tailwind/daisyUI class injection for markdown rendering — replaces inline
<style>block with classes injected during Earmark post-processing (works without@tailwindcss/typographyplugin) - Add blank line preservation in markdown content — intentional double blank lines render as visible spacing
- Add translation worker retry resilience — on retry, already-translated languages are skipped by checking content timestamps against job
inserted_at - Add dynamic timeout scaling for translation worker (~1.5 min per language, minimum 15 minutes)
- Add structured logging with consistent prefixes (
[Sync.Notifier],[Sync.API],[Sync.Connections]) throughout Sync connection flow - Add connection event logging on both sender and receiver sides for debugging
Changed
- Rename Sync "Sender/Receiver" terminology to "Outgoing/Incoming" across UI
- Allow editing incoming Sync connections (previously restricted to outgoing only)
- Remove "with permanent connections" from Sync index page subtitle
- Bump markdown render cache version to v2 to invalidate stale cached HTML
Fixed
- Fix Sync sender URL resolving to
localhost:4000— now checks DBsite_urlsetting before falling back to endpoint config - Fix
auth_token_hashlogged in full — truncate to first 8 characters in Sync connection logs - Fix double
get_our_site_url()call per notification — pass resolved URL instead of recomputing - Fix Sync crash on non-UTF8 binary data — base64-encode raw binaries during serialization, decode on import
- Fix Sync pull error responses silently ignored — add
Logger.errorto all failure paths (401, 404, HTTP errors, offline, invalid response) - Fix Sync completion UI not showing skipped/errored records — track and display per-table import counts with warning state
1.7.77 - 2026-03-17
Added
- Add Open Graph and Twitter Card meta tags for public publishing pages (og:title, og:description, og:image, og:url, og:locale, canonical link, and Twitter Card tags)
- Add
og:site_namemeta tag using project title - Add
resolve_language_key/2helper toLanguageHelpersfor base code to dialect code matching in language maps - Add Tailwind Typography prose overrides using daisyUI theme variables (oklch(--bc), oklch(--p), oklch(--b2), oklch(--b3)) for theme-aware markdown styling
- Add automated scheduled jobs cleanup to prevent table bloat (deletes completed jobs older than 7 days)
- Add
lastmod(last modified) to sitemap entries for SEO — router-discovered routes use beam file mtime, static entries use current date
Changed
- Replace inline markdown CSS with centralized prose overrides in
app.cssusing@layer base(removes 323 lines of duplication) - Update publishing preview template to show full public interface with working language switcher
- Move
MarkdownContentcomponent to use Tailwind prose classes instead of custom inline styles - Extract duplicated
resolve_language_key/2fromlisting.exandhtml.exto sharedLanguageHelpersmodule - Extract
update_post_from_form/3from publishing editor to reduce cyclomatic complexity (from >10 to <10) - Update Leaf content editor dependency from v0.1.0 to v0.2.0
Fixed
- Fix
mix phoenix_kit.statusshowing V01 instead of actual migration version — properly start Repo with parent app config when using--no-start - Fix language map lookup when canonical URL uses base code (e.g., "en" → "en-US" matching)
- Fix
absolute_url/2to use stricter URL protocol checking ("http://" or "https://" instead of just "http") - Fix preview language links to conditionally include version parameter only when version is non-nil
- Fix translation reload showing primary language content instead of translated content
- Fix PubSub subscription mismatch for translation and version events on timestamp-mode posts (slug vs uuid topic mismatch)
- Fix email template seeding failing on fresh install (wrap string fields in i18n maps)
- Fix whitespace in slug format examples on publishing group pages
1.7.76 - 2026-03-16
Fixed
- Fix
mix phoenix_kit.statusport conflict when app is already running (use--no-startto avoid booting the HTTP endpoint) - Add self-healing version comment detection — automatically corrects V83 comment bug where migrations ran but version stayed at V82
1.7.75 - 2026-03-16
Added
- Add
custom_fieldssupport toregistration_changeset/3for atomic user creation with custom metadata - Add entity data view extension documentation and route override pattern
Fixed
- Fix mobile overflow issues in email module UI (queue, blocklist, metrics, template editor)
- Fix early validation in template editor — errors only shown after first user interaction
- Fix Send Test Email modal overflowing on mobile (max-w-4xl → max-w-2xl)
- Fix V83 migration missing
down/1version comment rollback - Fix V83 migration prefix_str inconsistency in version comment
- Fix dialyzer
guard_failwarnings from upstream publishing merge - Fix remaining doc warnings for delegated hidden functions
1.7.74 - 2026-03-16
Fixed
- Remove dead
should_regenerate_cache?/1from Shared module (uncalled function returningtruein every branch) - Remove obsolete
bulk_operation_topictest referencing deleted PubSub function - Fix missing trailing newline in
shared.ex
1.7.73 - 2026-03-13
Changed
- Move module access guards from individual mount functions to centralized
enforce_admin_view_permissionhook - Disabled modules now block all roles (including Owner/Admin) at the
on_mountlevel, covering all ~50 admin LiveViews automatically - Remove per-LiveView
enabled?()mount guards from AI, Entities, Publishing, Sitemap, Billing, Customer Service, Emails, Email Tracking, Legal, Referrals, Shop settings
1.7.72 - 2026-03-13
Added
- Add module access guards — disabled modules now hide action buttons and block mount on settings/endpoints
- Add error flash auto-dismiss after 8 seconds
- Add
enabled?()mount guards to AI, Media, Entities, Publishing, Sitemap endpoints - Add error logging in Legal
list_generated_pagesinstead of silent rescue
Fixed
- Fix Legal module broken connection with DB-backed Publishing (
post.path→post.uuid,updated_at→published_at) - Fix Legal module Configure button guard when module is disabled
- Fix Sitemap RouterDiscovery including routes from disabled modules
- Fix DB.Listener missing
{:eventually, _ref}case for auto_reconnect
Changed
- Remove duplicate enable/disable toggles from 7 module settings pages (Emails, Email Tracking, Legal, Referrals, Billing, Customer Service, Shop)
- Simplify primary_language lookup in Publishing.DBStorage
1.7.71 - 2026-03-12
Fixed
- Fix mixed atom/string key error in
EntityData.maybe_add_position/1when auto-assigning position to string-keyed params - Fix same mixed key error in
EntityData.maybe_add_created_by/1 - Fix
FOR UPDATEwith aggregate function error inEntityData.next_position/1(PostgreSQL0A000 feature_not_supported)
1.7.70 - 2026-03-12
Added
- Add PhoenixKitGlobals component for JavaScript globals injection
- Add metadata JSONB field to comments schema (V82 migration)
- Add reply indicators to admin comments page
- Add test comments seed script for visual verification
- Add admin page generator category index pages with automatic route registration
- Add duplicate validation (ID, URL, label) to admin page generator
- Add compile-time warning for unresolved legacy admin LiveView modules
- Add
phoenix_kit_app_base/0helper to Routes utility
Fixed
- Fix dimension form inputs clearing each other on change
- Fix MarkdownEditor toolbar not working on LiveView navigation
- Fix CommentsComponent crash on post details page (
resource_id→resource_uuid) - Fix credo alias ordering in integration module
- Fix WebP transparency loss in center-crop image processing
- Fix 304 Not Modified support in FileController
Changed
- Update admin page generator to use flat
admin_dashboard_tabsconfig withlive_viewfield - Deprecate legacy
admin_dashboard_categoriesconfig format (warning on use) - Auto-infer LiveView modules from URL paths for legacy admin categories
- Add
attr :rest, :globaltophoenix_kit_globalscomponent
1.7.69 - 2026-03-10
- Add responsive multi-column card grid to
table_defaultcomponent: 1 col on mobile, 2 cols on md, 3 cols on lg breakpoints - Style card view cards with
bg-base-200andshadow-smto visually distinguish them from the page background
1.7.68 - 2026-03-10
- Merge upstream changes: publishing editor rework, AI translation, bulk group actions, V77/V78 migration fixes, scheduled jobs queue, Settings.Queries module, plugin migration callbacks
1.7.67 - 2026-03-10
Breaking Changes (requires manual steps in parent app)
- V79 migration rewritten in-place: drops
phoenix_kit_mailing_*tables, createsphoenix_kit_newsletters_* - Oban queue renamed:
mailing_delivery→newsletters_delivery(updateconfig/config.exs) - Settings keys changed:
mailing_enabled→newsletters_enabled,mailing_default_template→newsletters_default_template,mailing_rate_limit→newsletters_rate_limit - Email template category value changed:
"mailing"→"newsletters"(existing templates need DB update) - URL paths changed:
/admin/mailing/*→/admin/newsletters/*,/mailing/unsubscribe→/newsletters/unsubscribe
Changed
- Rename
PhoenixKit.Modules.Mailing→PhoenixKit.Modules.Newslettersand all submodules - Rename DB tables:
phoenix_kit_mailing_lists/list_members/broadcasts/deliveries→phoenix_kit_newsletters_* - Rename Elixir modules:
Mailing.List,Mailing.Broadcast,Mailing.Delivery,Mailing.ListMember,Mailing.Broadcaster,Mailing.Workers.DeliveryWorker→Newsletters.* - Rename web modules:
Mailing.Web.*→Newsletters.Web.* - Rename route module:
PhoenixKitWeb.Routes.MailingRoutes→NewslettersRoutes - Rename dashboard tabs:
:admin_mailing→:admin_newsletters
1.7.66 - 2026-03-09
- Clean up publishing module: fix UUID routing bugs (slug vs UUID in version creation, PubSub broadcasts, translation status), remove dead code and filesystem path references
- Optimize publishing DB queries: batch loading, ListingCache for dashboard, bulk UPDATE for translation statuses, debounced PubSub updates
- Rework editor to two-column layout with content-first design (title + editor left, metadata right)
- Rework AI translation: integrate AI prompt system, modal UI replacing slidedown, translation progress recovery across page refreshes
- Replace primary language banner with compact tooltip on language switcher
- Add skeleton loading UI for language switching in publishing editor
- Fix collaborative editing: spectator initial sync, lock promotion JS updates, lock expiration timer
- Fix admin sidebar highlighting for publishing group pages
- Fix custom fields card hidden when no field definitions are registered
- Add bulk "Add to Group" action on posts index with dynamic group filter dropdown
1.7.65 - 2026-03-08
- Fix V77 migration crash: role_id column renamed to role_uuid in UUID migration
1.7.64 - 2026-03-08
- Remove legacy filesystem paths from publishing module — strip all
.phkvirtual path references from mapper, editor, listing, and preview - Switch all event handlers and navigation from path-based to UUID-based routing
- Fix timestamp-mode posts returning 404: normalize post_time to zero seconds, with hour:minute-only fallback query for legacy data
- Add collision prevention for same-minute timestamp posts (auto-bump to next minute, max 60 attempts)
- Add unique_constraint on (group_uuid, post_date, post_time) to schema
- Render empty listing page instead of 404 when group exists but has no published posts
- Show Primary Language banner during new post creation
- Add missing PubSub handlers to publishing Index view (version_created, version_live_changed, version_deleted) with catch-all
- Fix primary language migration using removed path field — now uses UUID/slug directly
- Remove cache management UI from listing page (accessible via settings only)
- Add safety guards to URL builders: raise ArgumentError on nil UUID instead of producing broken URLs
1.7.63 - 2026-03-06
- Remove filesystem storage from publishing module — delete Storage, DualWrite, and all storage/* submodules (~7k lines removed)
- Add LanguageHelpers and SlugHelpers as standalone modules, simplify to DB-only throughout
- Fix slug conflict clearing bug:
clear_url_slugs_for_conflictspassed wrong slug to DB cleanup - Fix ngettext interpolation in primary language migration modal (literal
%{count}in UI) - Clean up stale filesystem references in comments, docs, and user-facing strings
- Fix V77/V78 migration crashes when UUID columns are missing (tables created after V56 ran)
- Simplify V77/V78 migrations — remove over-engineered column detection, rely on idempotent patterns
- Fix email tracking bug:
handle_delivery_resultusedget_log!(raises) in a nil-matching branch; addget_log/1non-bang wrapper and remove unused public functions - Add
migration_module/0callback to plugin module system —mix phoenix_kit.updateauto-discovers and runs plugin migrations - Add
Settings.Queriesmodule for database operations - Add dedicated queue and 1-day pruner for scheduled jobs cron worker
- Fix user dashboard navigation links
- Fix ueberauth providers configuration format in installer
1.7.62 - 2026-03-05
- Fix UnicodeConversionError crash in integration plug when response body contains non-UTF8 binary data
- Fix DB browser rendering of raw binary values (e.g. UUID bytes) in table and activity views
- Add V78 migration: backfill missing AI module columns skipped by V41 conditional checks
- Add
reasoning_enabled,reasoning_effort,reasoning_max_tokens,reasoning_excludetophoenix_kit_ai_endpoints - Add
prompt_uuid,prompt_nametophoenix_kit_ai_requestswith index and FK constraint
- Add
1.7.61 - 2026-03-04
- Replace
plug_cowboywithbandit ~> 1.0as HTTP adapter (Phoenix 1.8 default) - Remove stale deps from lock:
cowboy,cowlib,cowboy_telemetry,plug_cowboy,combine,dns_cluster,phoenix_live_dashboard,poolboy,timex,tzdata - Remove deprecated
fetch_live_flashplug - Add audit_log query limits
- Fix atom table exhaustion risk and remove duplicate function
- Update excluded_apps list in route resolver to match current deps
- Update HTML comments to EEx format in templates and components
- Clean up stale dep spec and dead commented-out code
1.7.60 - 2026-03-03
- Remove legacy FS→DB migration modules:
DBImporter,MigrateToDatabaseWorker,ValidateMigrationWorker - Remove
JsIntegrationinstall/update module (JS setup is now manual) - Remove all "Import to DB" / "Migrate to Database" UI buttons from publishing pages
- Remove DB import/migration PubSub broadcast functions and LiveView handlers
- Simplify publishing listing: drop
fs_post_count,needs_import,db_import_in_progressassigns - Move post title field into the editor content column with larger styling
- Simplify editor save button logic (always clickable unless readonly/autosaving)
- Add
enrich_with_db_uuids/2to ListingCache for UUID-based admin links in filesystem mode - Refine Sync module: migrate
connection_idreferences toconnection_uuid - Update publishing README to reflect DB-only storage model
1.7.59 - 2026-03-03
- Fix V75: use CASCADE when dropping
phoenix_kit_id_seq(meta tablephoenix_kit.idDEFAULT depends on it)
1.7.58 - 2026-03-03
- Add V75 migration: fix uuid column defaults and cleanup
- Set
DEFAULT uuid_generate_v7()on 27 tables missing it (Category A tables — V72 rename dropped old sequence DEFAULT) - Fix 4 tables using
gen_random_uuid()(UUIDv4) →uuid_generate_v7()(UUIDv7) - Drop orphaned
phoenix_kit_id_seqsequence
- Set
1.7.57 - 2026-03-03
- Fix V74 migration: skip tables without bigint
id(e.g. publishing tables created with UUID PKs) - Fix V74: use
DROP COLUMN id CASCADEto handle dependent FK constraints in one statement
1.7.56 - 2026-03-03
- Add V74 migration: drop integer
id/_idcolumns, promoteuuidto PK on all tables- Drop all FK constraints referencing integer
idcolumns (dynamic discovery) - Drop ~95 integer FK columns across all tables (sourced from uuid_fk_columns.ex + extras)
- Drop bigint
idPK + promoteuuidto PK on 47 Category B tables - After V74, every PhoenixKit table uses
uuidas its primary key — no integer PKs remain
- Drop all FK constraints referencing integer
- Remove
source: :idfromwebhook_event.exschema (DB column now matches field name)
1.7.55 - 2026-03-03
- Fix scheduled_job.ex
source: :idregression — PR #383 reintroduced mapping to dropped DB column - Add V73 migration: pre-drop prerequisites for Category B UUID migration
- SET NOT NULL on 7 uuid columns (
ai_endpoints,ai_prompts,consent_logs,payment_methods,role_permissions,subscription_types,sync_connections) - CREATE UNIQUE INDEX on 3 tables (
consent_logs,payment_methods,subscription_types) - ALTER INDEX RENAME on 4 indexes to match renamed columns (
post_tag_assignments,post_group_assignments,post_media,file_instances)
- SET NOT NULL on 7 uuid columns (
- Add
RepoHelper.get_pk_column/1— queriespg_indexfor PK column name, falls back to"id" - Fix DB explorer to use dynamic PK column in
fetch_row,table_preview, and notify trigger - Fix Sync API controller to use dynamic PK column in
fetch_filtered_recordsandbuild_where_clause - Fix Sync connection notifier to use dynamic PK column in
insert_recordandbuild_update_clause - Update 4 schema constraint names to match V72 column renames (
post_id→post_uuid,file_id→file_uuid) - Remove dead
:user_idfrom OAuthreplace_all_exceptlist
1.7.54 - 2026-03-03
- Add V72 migration: rename PK column
id→uuidon 30 Category A tables (metadata-only, instant) - Add 4 missing FK constraints:
comments.user_uuid,comments_dislikes.user_uuid,comments_likes.user_uuid,scheduled_jobs.created_by_uuid - Remove
source: :idmapping from 29 Category A Ecto schemas — DB column now matches field name directly
1.7.53 - 2026-03-02
- Add
mix phoenix_kit.doctordiagnostic command — detects migration version vsschema_migrationsdiscrepancies, stale COMMENT tags, and common DB issues - Add
update_modetomix phoenix_kit.update— skips heavy DB components (Oban, cache warmers, settings queries) and caps Ecto pool at 2 during migrations to prevent DB saturation - Run
ecto.migratein-process instead ofSystem.cmdfor better error reporting and reliability - Fix three migration hang root causes: NULL UUIDs causing infinite backfill loop, orphaned FK references blocking constraint creation, varchar uuid columns crashing Ecto schema loader
- Fix migration hang: disable DDL transaction in generated migration wrapper — prevents entire multi-version migration from running in a single transaction holding AccessExclusiveLock
- Fix V50 migration hang: add
lock_timeoutforphoenix_kit_bucketsALTER TABLE and check column existence before ALTER - Fix settings cache race: warm synchronously in
init/1whensync_init: true; fixwarm_critical_datainserting{key, value, nil}3-tuples when TTL is nil - Fix cache
sync_initblocking supervisor for 60s when DB is overloaded - Fix startup DB timeout: defer
Dashboard.Registryinit and reorder supervisor children - Silence cache warmer spam and auto-grant warning when
role_permissionstable doesn't exist yet - Fix
gen.migrationtask: generate UUID primary keys anduser_uuidFK instead of integer-based - Fix broken GDPR anonymization: remove leftover
user_id: nilfromupdate_allcalls - Rename
_id→_uuidacross all remaining application code: billing, shop, storage, entities, sync, emails, tickets, permissions, roles, connections, publishing, and user_notifier - Rename function names:
find_role_by_id→find_role_by_uuid,parse_id→parse_uuid,import_id→import_uuid - Fix Dialyzer warning: remove unreachable pattern match in cache warming
- Fix V56/V63 migration crash:
email_log_uuidbackfill fails withdatatype_mismatchwhenphoenix_kit_email_logs.uuidischaracter varyinginstead of nativeuuidtype - Fix UUIDFKColumns: replace broken Elixir
rescuewith PostgreSQLEXCEPTIONhandler inside DO blocks — prevents outer transaction abort on backfill failure - Add
::uuidexplicit cast in all UUIDFKColumns backfill SQL to handle varchar source columns gracefully - Fix V56: add pre-step to convert varchar
uuidcolumns on all FK source tables to nativeuuidtype beforeUUIDFKColumns.upruns - Fix V63: wrap
matched_email_log_uuidbackfill in DO block with EXCEPTION handler and::uuidcast - Add V70 migration: re-backfills
email_log_uuidandmatched_email_log_uuidfor installs where V56/V63 silently skipped the backfill; resets stale random UUIDs written by the V56 NULL-fill fallback - Add investigation doc:
dev_docs/investigations/2026-03-01-varchar-uuid-migration-bug.md
1.7.52 - 2026-02-28
- Add translatable
titlefield to posts and fix timestamp-mode post handling - Add V69 migration: make role table integer FK columns nullable
- Add
mix precommitalias (compile → format → credo --strict) tomix.exs - Update AGENTS.md with pre-commit instructions, replacing old minimal checklist
- Rename
Scope.user_id→Scope.user_uuidfor consistency - Rename
user_id→user_uuidacross event handlers, templates, and messages - Rename
user_id→user_uuidin emails rate_limiter;log_id→log_uuidin emails interceptor, SQS processor, and sync task - Rename
user_id→aws_user_idin AWS credentials verifier - Rename
resource_id→resource_uuidin scheduled jobs - Rename
_id→_uuidin billing, shop, AI, entities, legal, posts, tickets, storage, scheduled jobs, and permissions - Rename
_id→_uuidacross metadata, forms, helpers, and tests - Replace
DateTime.utc_now()withUtilsDate.utc_now()across codebase - Remove redundant
connection_idparameter from syncconnection_notifier - Fix crash bugs:
.user_idstruct access →.user_uuidin billing events and order_form - Fix UUID field references for webhook_events and post images
- Fix duplicate map keys left from UUID migration in hooks and rate_limiter
- Fix alias ordering Credo violations across 18 files
- Fix timestamp-mode post lookups, migration ordering, and admin UI (PR #376 review follow-up)
- Fix
tab_callback_contextmissing clause; demote double-wrap log to debug - Add dialyzer ignore for unused
tab_callback_contextclause - Update integration guide, making-pages-live guide, dashboard README, and usage-rules to use UUID terminology
1.7.51 - 2026-02-26
- Add V64 migration: fix login crash by replacing
user_idcheck constraint withuser_uuidon user tokens table - Add V65 migration: rename
SubscriptionPlantoSubscriptionType(table, columns, indexes, constraints) - Rename
SubscriptionPlanschema, context functions, events, routes, LiveViews, and workers toSubscriptionType - Add orphaned media file cleanup system
mix phoenix_kit.cleanup_orphaned_filestask with dry-run and--deletemodesDeleteOrphanedFileJobOban worker with 60s delay and orphan re-check before deletion- Orphan filter toggle and "Delete all orphaned" button in Media admin UI
- Add Delete File button with confirmation modal to Media Detail page
- Add secondary language slug uniqueness validation via JSONB query in entity data
- Rename
seed_title_in_datatoseed_translatable_fieldsin entity data form - Unify slug labels to "Slug (URL-friendly identifier)" across entity forms
- Standardize dev_docs file naming convention (
{date}-{kebab-case}-{type}.md) - Fix orphan detection crash: remove references to non-existent
phoenix_kit_shop_variantstable - Fix
String.trim(nil)crash in SQS workers when AWS credentials not configured - Fix default preload
:planto:subscription_typeinlist_subscriptions/1andlist_user_subscriptions/2 - Fix
auth.exstoring integerfile.idinstead offile.uuidfor avatar custom field - Fix
create_subscription/2dead code and key mismatch — now accepts:subscription_type_uuidas preferred key - Fix
change_subscription_type/3reading stalesubscription_type_idinstead ofsubscription_type_uuid - Fix AWS Config returning empty string instead of nil when credentials unconfigured
- Fix email log
Accessbehaviour error when called withEmailLogDatastruct - Fix
Interceptorusing deprecatedlog.idinstead oflog.uuidin header and event - Fix shop billing cascade: check
disable_system()result and log on failure - Replace
defpproxy wrappers withimportin 4 shop LiveViews (cart, catalog, checkout) - Rename
post_idtopost_uuidin 5 private post functions - Remove legacy integer ID function clauses from posts and billing modules
- Remove accidentally committed
.beamfiles, add*.beamto.gitignore - Remove empty legacy
subscription_plan_form.exandsubscription_plans.ex - Add doc notes about performance for
all_admin_tabs/0andget_config/0 - Remove dead
_plugin_session_namevariable from integration routes
1.7.50 - 2026-02-25
- Fix
defp show_dev_notice?CLAUDE.md violation: replace private helper with<.dev_mailbox_notice>Phoenix Component- New component at
lib/phoenix_kit_web/components/core/dev_notice.exwithmessageandclassattrs - Removed from
login.ex,registration.ex,magic_link.ex,forgot_password.ex,dashboard/settings.ex - Updated all corresponding HEEX templates to use
<.dev_mailbox_notice>
- New component at
- Fix duplicate route alias compilation warnings in
phoenix_kit_authenticated_routes/1- Split module-scope routes into
authenticated_live_routes/0andauthenticated_live_locale_routes/0 - Locale variants now use
_localesuffix (e.g.:shop_user_orders_locale)
- Split module-scope routes into
- Fix undeclared
sidebar_after_shopattr inshop_layout/1component - Fix
maybe_redirect_authenticated/1hardcoded"/"redirect — usesigned_in_path(socket)consistently - Fix double
Map.from_structinEmails.Interceptor.create_email_log/2— redundant call removed
1.7.49 - 2026-02-24
- Add V63 migration: UUID companion column safety net round 2
- Add
uuididentity column tophoenix_kit_ai_accounts(missed by V61 due to wrong table name) - Add
account_uuidcompanion tophoenix_kit_ai_requests(backfilled from ai_accounts) - Add
matched_email_log_uuidtophoenix_kit_email_orphaned_events(backfilled from email_logs) - Add
subscription_uuidtophoenix_kit_invoices(backfilled from subscriptions) - Add
variant_uuidtophoenix_kit_shop_cart_items(nullable, no variants table) - Update Invoice, AI Request, and CartItem schemas with new uuid companion fields
- Add
1.7.48 - 2026-02-24
- Add V62 migration: rename 35 UUID-typed FK columns from
_idsuffix to_uuidsuffix- Enforces naming convention:
_id= integer (legacy/deprecated),_uuid= UUID - Groups: Posts module (15 renames), Comments (4), Tickets (6), Storage (3), Publishing (3), Shop (3), Scheduled Jobs (1)
- No data migration — columns already held correct UUID values, pure rename
- All DB operations idempotent (IF EXISTS guards) — safe on installs with optional modules disabled
- Update all Ecto schemas, context files, web files, and tests to use new field names
- Enforces naming convention:
1.7.47 - 2026-02-24
- Fix V13 migration down/0 to use
remove_if_existsinstead ofremovefor idempotency- Fixes "column aws_message_id does not exist" error when rolling back V13
1.7.46 - 2026-02-24
- Add plugin module system with
PhoenixKit.Modulebehaviour,ModuleRegistry, and zero-config auto-discovery- 5 required + 8 optional callbacks with sensible defaults via
use PhoenixKit.Module - Auto-discovers external modules by scanning
.beamfiles for@phoenix_kit_moduleattribute - All 21 internal modules now implement the behaviour, removing 786 lines of hardcoded tab enumeration
- External module admin routes auto-generated at compile time from
admin_tabswithlive_viewfield
- 5 required + 8 optional callbacks with sensible defaults via
- Add live sidebar updates via PubSub when modules are enabled/disabled
- Add server-side authorization on module toggle events (prevents crafted WebSocket bypass)
- Add startup validation: duplicate module keys, permission key mismatches, duplicate tab IDs, missing permission fields
- Add compile-time warnings for route module and LiveView compilation failures
- Standardize AI, Billing, and Shop to use
update_boolean_setting_with_module/3(consistent with all other modules) - Fix billing→shop cascade: shop now disabled after billing toggle succeeds (prevents orphaned state)
- Fix
Tab.permission_granted?/2to handle atom permission keys instead of silently bypassing checks - Fix
static_children/0to catch modulechildren/0failures instead of crashing the supervisor
1.7.45 - 2026-02-23
- Fix auth forms mobile overflow on small screens (px-4 added to all form containers)
- Fix daisyUI v5 compliance: remove deprecated
input-borderedfrom<.input>component and all auth templates - Fix
<.header>hardcodedtext-zinc-*colors replaced with semantictext-base-contentfor dark theme support - Convert forgot_password, reset_password, confirmation, confirmation_instructions to unified card layout
- Add missing
LayoutWrapper.app_layoutwrapper to confirmation form - Fix V40 migration silently skipping V32-V39 tables due to Ecto command buffering
- Root cause:
repo().query()(immediate) couldn't see buffered table creation commands - V31's
flush()was the last flush before V40, creating a clean V31/V32 split - Add
flush()to V40 and V56 to prevent recurrence on new installations
- Root cause:
- Add V61 migration: uuid column safety net for 6 tables missed by V40
- Tables fixed: admin_notes, ai_requests, subscriptions, payment_provider_configs, webhook_events, sync_transfers
- Also adds
created_by_uuidFK column to phoenix_kit_scheduled_jobs
1.7.44 - 2026-02-23
- Add Publishing module: DB storage, public post rendering, and i18n support
- Add unified
admin_page_headercomponent, replace all per-page admin headers - Add try/rescue to all form save handlers to prevent silent data loss on validation errors
- Add skeleton loading placeholders for entity language tab switching
- Add "Update Entity" submit button at top of entity form for quicker saves
- Add responsive card view to entities listing, remove stats/filters
- Memoize
IbanData.all_specs/0with compile-time module attribute for performance - Auto-register built-in comment resource handlers
- Make entity slug translatable and move it into Entity Information section
- Move multilang info alert above language tabs with improved explanation
- Tighten language tab spacing, replace daisyUI tab classes with compact utilities
- Remove hardcoded category column, filter, and bulk action from data navigator
- Fix CommentsComponent crash on post detail page
- Fix Entity update crash from DateTime microseconds in
:utc_datetimefields - Fix
email_templatesschema/migration mismatch breaking fresh installs - Fix locale disappearing from admin URLs on sidebar navigation
- Fix badge component height on mobile devices
- Fix cached plan error spam during migrations with column type changes
- Fix CSS specificity debt and inline styles replaced with Tailwind classes
- Fix mobile responsiveness across admin panel
- Replace remaining
DateTime.utc_now()withUtilsDate.utc_now()in all DB write contexts
1.7.43 - 2026-02-18
- Standardize all schemas to
:utc_datetimeandDateTime.utc_now()across 73 files- Replace
:utc_datetime_usecwith:utc_datetimeandNaiveDateTimewithDateTime - Add V58 migration to convert all timestamp columns across 68 tables from
timestamptotimestamptz - Fix UUID FK backfill to handle NULL UUIDs before applying NOT NULL constraints
- Replace
- Fix DateTime.utc_now() microsecond crashes in 19 files after
:utc_datetimeschema migration- Add
DateTime.truncate(:second)to allDateTime.utc_now()calls in contexts - Affected: settings, billing, shop, emails, referrals, tickets, comments, auth, permissions, roles
- Add
- Fix Language struct Access error on admin modules page and all bracket-access-on-struct bugs
- Add 20 typed structs replacing plain maps across billing, entities, sync, emails, AI, and dashboard
- Billing: CheckoutSession, SetupSession, WebhookEventData, PaymentMethodInfo, ChargeResult, RefundResult, ProviderInfo
- Other: AIModel, FieldType, EmailLogData, LegalFramework, PageType, Group, TableSchema, ColumnInfo, SitemapFile, TimelineEvent, IbanData, SessionFingerprint
- Fix
register_groupsto convert plain maps toGroupstructs, preventing sidebar crashes - Fix CastError in live sessions page by using UUID lookup instead of integer id
- Fix guest checkout flow: relax NOT NULL on legacy integer FK columns, fix transaction error double-wrapping
- Add return_to login redirect support for seamless post-login navigation (e.g., guest checkout)
- Add cart merge on login for guest checkout sessions
- Fix shop module .id to .uuid migration in Storage image lookups and import modules
- Fix hardcoded "PhoenixKit" fallback in admin header project title
- Fix admin sidebar submenu not opening on localized routes
- Fix 2 dialyzer warnings in checkout session and UUID migration
- Add multi-language support for Entities module
- New
Multilangmodule with pure-function helpers for multilang JSONB data - Language tabs in entity form, data form, and data view (adaptive compact mode for >5 languages)
- Override-only storage for secondary languages with ghost-text placeholders
- Lazy re-keying when global primary language changes (recomputes all secondary overrides)
- Translation convenience API:
Entities.set_entity_translation/3,EntityData.set_translation/3,EntityData.set_title_translation/3, and related get/remove functions - Multilang-aware category extraction in data navigator and entity data
- Non-translatable fields (slug, status) separated into their own card
- Required field indicators hidden on secondary language tabs
- Title translations stored as
_titlein JSONB data column (unified with other field translations) - Slug generation disabled on secondary language tabs
- Validation error messages wrapped in gettext for i18n
- 124 pure function tests for Multilang, HtmlSanitizer, FieldTypes, FieldType
- New
- Fix entities multilang review issues
- Unify title storage in JSONB data column, fix rekey logic for primary language changes
- Add
seed_title_in_datafor lazy backwards-compat migration on mount - Replace
String.to_existing_atomwith compile-time@preserve_fieldsmap - Fix 7 remaining issues from PR #341 permissions review
- Add catch-all fallback clauses to Scope functions to prevent FunctionClauseError
- Sort
custom_keys/0explicitly instead of relying on Erlang map ordering
1.7.42 - 2026-02-17
- Use PostgreSQL IF NOT EXISTS / IF EXISTS for UUID column operations
- Replace manual column_exists? checks with native DDL guards in V56 and UUIDFKColumns
- Makes migrations more robust and idempotent
1.7.41 - 2026-02-16
- Fix FK constraint creation crash when UUID target tables lack unique indexes
- Ensure unique indexes on all FK-target uuid columns before adding FK constraints
- Fixes
invalid_foreign_keyerror onphoenix_kit_ai_endpointsand other tables
1.7.40 - 2026-02-16
- Remove redundant mb-4 wrapper div around back buttons in 4 admin pages
- Add V57 migration to repair missing UUID FK columns
- Update language filter to use languages_official instead of languages_spoken
1.7.39 - 2026-02-16
- Complete UUID migration (Pattern 2) across all remaining modules
- Migrate posts, tickets, storage, comments, referrals, and connections schemas to UUID-based user references
- Migrate posts like/dislike/mention functions to accept UUID user identifiers
- Fix stale
.idaccess across posts, storage, tickets, email, connections, and image downloader - Fix ProcessFileJob and media_detail to use user_uuid instead of deprecated user_id
- Replace legacy
.idaccess with.uuidacross mix tasks and admin presence - Remove legacy integer fields from RoleAssignment schema
- Fix 10 Dialyzer warnings across comments, connections, referrals, and shop modules
- Harden permissions system with security and correctness fixes
- Fix security and correctness issues in permissions system
- Add permission edit protection for own role and higher-authority roles
- Add Owner protection to
can_edit_role_permissions/2and standardize UUID usage - Fix edge cases, silent failures, and crash risks in permissions and roles
- Fix dual-write in
set_permissions/3and cross-view PubSub refresh - Fix permissions summary to count only visible keys
- Fix multiple bugs in custom permission keys and admin routing
- Add auto-grant of custom permission keys to Admin role
- Add defensive input validation to custom permission key registration
- Fix
unless/elsetoif/elsefor Credo compliance
- Add gettext i18n to roles and permissions admin UI
- Add Level 1 test suite for permissions, roles, and scope (156 tests)
- Fix responsive header layout across all admin pages
- Add responsive text classes (
text-2xl sm:text-4xl/text-base sm:text-lg) to all page headers - Fix missed responsive text classes in storage, media selector, and publishing pages
- Add responsive text classes (
- Replace dropdown action menus with inline buttons in table rows
- Fix require_module_access plug to check feature_enabled like LiveView on_mount
- Fix admin sidebar wipe when enabling/disabling modules
- Add
get_role_by_uuid/1API and update integration guide - Restore admin edit button in user dropdown and add product links in cart
- Fix selected_ids to use MapSet for O(1) lookups
- Fix Dialyzer CI failure for ExUnit.CaseTemplate test support files
- Fix Credo nesting and Dialyzer MapSet opaque type warnings
- Update Permissions Matrix page title and section labels
1.7.38 - 2026-02-15
- Fix Ecto.ChangeError in entities by using DateTime instead of NaiveDateTime
- Fix infinite recursion risk in category circular reference validation
- Add DateTime inconsistency audit report with phased migration plan
- Add custom permission key auto-registration for admin tabs
- Custom admin tabs with non-built-in permission keys now auto-register with the permission system
- Custom keys appear in the permission matrix and roles popup under "Custom" section
- Owner role automatically gets access to custom permission keys
- Custom LiveView permission enforcement via cached
:persistent_termmapping - New API:
Permissions.register_custom_key/2,unregister_custom_key/1,custom_keys/0,clear_custom_keys/0 - Key validation: format check (
~r/^[a-z][a-z0-9_]*$/),ArgumentErroron built-in key collision
1.7.37 - 2026-02-15
- Fix UUID PR review issues: aliases, dashboard_assigns, and naming issues
- Fix V56 migration: add subscription_plans to uuid column setup lists
- Add admin edit buttons and improve shop catalog UX
- Add registry-driven admin navigation system
- Fix localized field validation in Shop forms
- And bunch of bugs and optimizations
1.7.36 - 2026-02-13
- Add storefront sidebar filters, category grid, and dashboard shop integration
- New
CatalogSidebarcomponent: reusable sidebar with collapsible filter sections and category tree navigation - New
FilterHelpersmodule: filter data loading, URL query string building, price/vendor/metadata filtering - Storefront filter configuration in admin settings: enable/disable filters, edit labels, add metadata option filters
- Auto-discovery of filterable product metadata options (e.g., Size, Color) with one-click filter creation
- Price range filter with min/max inputs and range display
- Vendor and metadata option filters with checkbox selection and active count badges
- Filter state persisted in URL query params for shareable filtered views
- "Show Categories in Shop" setting: displays category card grid above products on main shop page
- Sidebar category navigation always visible in sidebar (decoupled from grid setting)
- Dashboard layout integration: shop filters and categories rendered in dashboard sidebar for authenticated users
sidebar_after_shopslot in dashboard layout for injecting custom sidebar content- Product detail page updated to use shared sidebar and filter context for consistent navigation
- Mobile filter drawer with toggle button and active filter count badge
- Category page filters scoped to category products
- Fix
phx-value-valuecollision on filter checkboxes: renamed tophx-value-valto avoid HTML checkboxvalue="on"overwrite - Known issue: metadata option filters (e.g., Size) may not filter correctly in all cases; needs further investigation
- New
- Add file upload field type to Entities module
- New
filefield type with configurable max entries, file size, and accepted formats FormBuilderrenders file upload UI with drag-and-drop zone (admin entity forms, placeholder)- New
:advancedfield category
- New
- Fix 3 remaining UUID migration bugs in billing forms
- Fix 8 UUID migration bugs found in PR #330 post-merge review
- Add UUIDv7 migration V56 with dual-write support
1.7.35 - 2026-02-12
- Rewrite Sitemap module to sitemapindex architecture with per-module files
/sitemap.xmlnow returns a<sitemapindex>referencing per-module files at/sitemaps/sitemap-{source}.xml- Dual mode support: "Index mode" (per-module files, default) and "Flat mode" (single urlset when Router Discovery enabled)
- New
Sourcebehaviour callbacks:sitemap_filename/0andsub_sitemaps/1for per-group file splitting - New
Generator.generate_all/1andgenerate_module/2with auto-splitting at 50,000 URLs - FileStorage rewrite with
save_module/2,load_module/1,delete_module/1,list_module_files/0 - Cache rewrite supporting
{:module_xml, filename}and{:module_entries, source}keys - Per-module stats stored as JSON in Settings with
get_module_stats/0 - Per-module regeneration via
SchedulerWorker.regenerate_module_now/1(Oban) - Settings UI overhaul: per-module sitemap cards with stats, regeneration buttons, mode indicators
- Publishing source: per-blog sub-sitemaps via
sitemap_publishing_split_by_groupsetting - Entities source: per-entity-type sub-sitemaps
- Static source: login page excluded, registration conditionally included
- Router Discovery default changed to
false(index mode is new default) - Removed "cards" XSL style; added
sitemap-index-minimal.xslandsitemap-index-table.xsl - Sitemap routes no longer go through
:browserpipeline (public XML endpoints)
- Add PDF support for Storage module
- New
PdfProcessormodule usingpoppler-utils(pdftoppm,pdfinfo) - First page rendered to JPEG thumbnail at configurable DPI
- PDF metadata extraction (page count, title, author, creator, creation date)
VariantGeneratorextended for document/PDF MIME types- Media UI: inline PDF viewer on detail page, PDF badges on thumbnails, metadata display
- New system dependency checks for poppler in
Dependenciesmodule
- New
- Fix option price display for options with all-zero modifiers
- New
has_nonzero_modifiers?/1filters out option groups where all price modifiers are zero - Price modifiers displayed as badges on option buttons (e.g., "+$5.00")
- Cart saves all selected specs including non-price-affecting options (e.g., Color)
build_cart_display_name/3includes all selected specs in display name
- New
- Fix category icons fallback to legacy product images
Category.get_image_url/2falls back tofeatured_product.featured_image(legacy URL)- Product detail respects
shop_category_icon_modesetting for category subtab icons - Guard clauses tightened for Storage vs legacy URL handling
- Add ImportConfig filtering at CSV preview stage
- Config filters applied during CSV analysis/preview, not just during import
- Import wizard shows skipped product count with warning badge
- Category creation uses language normalization for consistent JSONB slug keys
- Imported option labels use
_option_slotsmetadata for proper display names
- Fix admin sidebar full-page reload after upstream merge
- Comments and Sync routes merged into main admin
live_session
- Comments and Sync routes merged into main admin
- Add runtime sitemaps directory to gitignore
1.7.34 - 2026-02-11
- Extract Comments into standalone reusable module (V55 migration)
- New
PhoenixKit.Modules.Commentscontext with polymorphicresource_type+resource_idassociations - New tables:
phoenix_kit_comments,phoenix_kit_comments_likes,phoenix_kit_comments_dislikes - Reusable
CommentsComponentLiveComponent that can be embedded in any resource detail page - Threaded comments with configurable max depth and content length
- Like/dislike system with atomic counter cache
- Moderation admin UI at
{prefix}/admin/commentswith filters, search, and bulk actions - Module settings page at
{prefix}/admin/settings/comments - Resource handler callback system for notifying parent modules (e.g., Posts) of comment changes
- "comments" permission key added (25 total permission keys, 20 feature modules)
- Posts module refactored to consume Comments module API instead of inline implementation
- Legacy
phoenix_kit_post_commentstables preserved for backward compatibility
- New
- Add shop enhancements, sitemap sources, and admin navigation fix
- Shop module improvements: product options toggle, import configs, drag-and-drop reordering, catalog language redirects
- Sitemap module: shop source (categories, products, catalog), data source toggles in settings UI
- Admin sidebar seamless navigation (consolidate live_sessions)
- Migration fixes and V54 addition
- Fix preview-to-editor round-trip state and data loss bugs
- Fix 8 bugs in the preview_token handle_params path that had diverged from the other editor entry points as features were added over time
- Merge disk metadata into preview post to prevent silent data loss when saving after a preview round-trip
- Add error logging to enrich_from_disk for observability
- Add module-level permission system for role-based admin access control
- Custom roles can now be granted granular access to specific admin sections and feature modules. Permissions are managed through a new interactive matrix UI, enforced at both route and sidebar level, and update in real-time across all admin tabs via PubSub.
1.7.33 - 2026-02-04
- Add module-level permission system (V53 migration)
phoenix_kit_role_permissionstable with allowlist model (row present = granted)- 24 permission keys: 5 core sections + 19 feature modules
- Owner bypasses all checks; Admin seeded with all 24 keys by default
- Custom roles start with no permissions, assigned via matrix UI or API
PhoenixKit.Users.Permissionscontext for granting, revoking, and querying role permissions- Interactive permission matrix at
{prefix}/admin/users/permissions - Inline permission editor in Roles page with grant/revoke all
- Route-level enforcement via
phoenix_kit_ensure_adminandphoenix_kit_ensure_module_access - Sidebar nav gated per-user based on granted permissions
- Real-time PubSub updates: permission changes reflect across all admin tabs
- Backward compatible: pre-existing Admins retain full access before V53 migration
- Add PubSub events for real-time updates in Tickets and Shop modules
- Tickets.Events module with broadcast for ticket lifecycle (created, updated, status changed, assigned, priority changed)
- Comment and internal note events for ticket discussions
- Shop.Events extension with product, category, inventory events
- LiveViews subscribe to events for real-time UI updates
- Add User Deletion API with GDPR-compliant data handling
- delete_user/2 with cascade delete for related data (tokens, OAuth, billing profiles, carts)
- Anonymization strategy for orders, posts, comments, tickets, email logs, files
- Protection: cannot delete self, cannot delete last Owner
- Admin UI with delete button, confirmation modal, and real-time list updates
- Broadcast :user_deleted event for multi-admin synchronization
- Fix compilation errors in auth.ex (pin operator with dynamic Ecto queries)
- Update core PhoenixKit schemas and Referrals to new UUID standard
- Update Shop module with localized slug support and unified image gallery
- Add PubSub events for Tickets and Shop modules, User Deletion API
- Added support for uuid to referral module
- Add markdown rendering and bucket access types
- Update Sync module to new UUID standard pattern
- Update billing module to use DB-generated UUIDs
- Update entities module to UUID standard matching AI module
1.7.32 - 2026-02-03
Storage Module: Smart file serving with bucket access types (V50 migration)
- Add
access_typefield to buckets: "public", "private", "signed" - Local files are now served directly without temp file copying (performance improvement)
- Public cloud buckets redirect to CDN URL (faster, reduces server load)
- Private cloud buckets proxy files through server (for ACL-protected storage)
- Add retry logic for bucket cache race conditions during file access
⚠️ BREAKING CHANGE: Cloud Bucket Access Type
Cloud buckets (S3, B2, R2) now default to
access_type = "public", which redirects users directly to the bucket's public URL instead of proxying through the server.If you have private/ACL-protected buckets:
- Go to Storage → Buckets → Edit your bucket
- Set "Access Type" to "Private"
- Files will be proxied through the server using credentials (previous behavior)
If you have public buckets (redirect mode):
For redirect to work, your bucket must be publicly accessible:
Enable Public Access in your cloud provider settings:
- AWS S3: Disable "Block all public access" and set bucket policy
- Backblaze B2: Set bucket to "Public"
- Cloudflare R2: Configure public access or use Custom Domain
Configure CORS if serving files cross-origin (required when your site domain differs from bucket domain):
AWS S3 / R2 CORS configuration example:
[ { "AllowedHeaders": ["*"], "AllowedMethods": ["GET", "HEAD"], "AllowedOrigins": ["https://yourdomain.com"], "ExposeHeaders": ["ETag", "Content-Length"], "MaxAgeSeconds": 3600 } ]Replace
https://yourdomain.comwith your actual domain, or use"*"for any origin (less secure but simpler for testing).
See AWS documentation: https://docs.aws.amazon.com/AmazonS3/latest/userguide/enabling-cors-examples.html
- Add
1.7.31 - 2026-01-29
- Refactor publishing module into submodules and improve URL slug handling
- Storage module refactoring:
- Split storage.ex into specialized submodules: Paths, Languages, Slugs, Versions, Deletion, and Helpers for better organization and maintainability
- Move controller logic into submodules: Fallback, Language, Listing, PostFetching, PostRendering, Routing, SlugResolution, Translations
- Move editor logic into submodules: Collaborative, Forms, Helpers, Persistence, Preview, Translation, Versions
- Listing page improvements:
- Show live version's translations and statuses instead of latest version
- Fetch languages from filesystem when version_languages cache is empty
- Fix paths to point to live version files when clicking language buttons
- Add "showing vN" badge that combines with version count display
- Fix public URL to always use post's primary language
- URL slug priority system:
- Directory slugs now have priority over custom url_slugs
- Prevent setting url_slug that conflicts with another post's directory name
- Auto-clear conflicting url_slugs instead of blocking saves
- Show info notice when url_slugs are auto-cleared due to conflicts
- Clear conflicting url_slugs from ALL translations, not just current one
- Clear conflicting custom url_slugs when new post is created
- Storage module refactoring:
1.7.30 - 2026-01-28
- Posts Module
- Add likes and dislikes system for post comments (V48 migration)
- Post body field is no longer required
- User Management
- Add dropdown field type support for user custom fields
- Shop Module (E-commerce)
- Fix JSONB search queries and add defensive guards for robustness
- Fix JSONB localized fields consistency across product/category operations
- Add shop import enhancements with V49 migration
- Fix image migration robustness and catalog display issues
- Add language selection dropdown to CSV import for localized content
- Add variant image mapping support for Shop products
- Add legacy image support for backward-compatible variant mappings
- Bug Fixes
- Fix UUID column error for auth tables during upgrade - Users upgrading from PhoenixKit < 1.7.0 no longer get "column uuid does not exist" error when logging in. Added auth tables (users, tokens, roles, role_assignments) to UUIDRepair module.
1.7.29 - 2026-01-26
- Add primary language improvements and AI translation progress tracking
- Real-time translation progress - Added progress bars to editor and listing pages showing AI translation status
- Primary language improvements - Posts now store their primary language for isolation from global setting changes
- Language handling fixes - Fixed base code to dialect mapping (e.g., en → en-US) across public URLs and editor
- UI polish - Updated language switcher colors, modal text, and added prominent primary language display in editor
- Documentation - Added comprehensive README for the Languages module
1.7.28 - 2026-01-24
- Major improvements to the Publishing module's multi-language workflow: renamed "master" to "primary" terminology, fixed URL routing with locales, added language migration tools, improved cache performance, and fixed several UI/UX issues in settings and admin pages.
- Multi-Language System Improvements
- Rename master to primary terminology - Updated all references from "master language" to "primary language" for consistency and clarity
- Fix language in URL breaking navigation - Resolved issues where locale prefixes in URLs caused routing problems
- Isolate posts from global primary_language changes - Posts now store their own primary language, preventing drift when global settings change
- Add "Translate to This Language" button - Quick translation action for non-primary languages in the editor
- Sort languages in dropdowns - Consistent alphabetical sorting across all language selectors
- Migration Tools
- Add version structure migration UI - Visual indicators and migration buttons throughout the publishing module
- Fix legacy post migration - Resolved "post not found" errors when migrating from legacy to versioned structure
- Handle dual directory structures - Fixed migration when both publishing/ and blogging/ directories exist
- Add primary language migration system - Tools to migrate posts to use isolated primary language settings
- Performance
- Improve listing performance - Read from cache when possible, reducing database/filesystem hits
- Language caching with WebSocket transport - Faster language resolution with proper cache invalidation
- Add Create Group shortcut - Quick access button on publishing overview page
- Settings & Admin UI Fixes
- Fix General settings content language glitch - Resolved weird UI behavior when changing content language
- Fix settings tab highlighting - General and Languages tabs now properly highlight on child pages
- Fix admin header dropdowns - Theme and language dropdowns in admin header now work correctly
- Update Entities module description - Clearer description on the Modules page
- Multi-Language System Improvements
- Updated the languages module added front and backend tabs for languages
- Add localized routes for Shop module
- Add locale-prefixed routes (/:locale/shop/...) for multi-language Shop module support
- Add language validation to only allow enabled languages in URLs
- Add language preview switcher for admin product detail page
1.7.27 - 2026-01-19
- Changed / Added
- Added prefix-aware navigation helpers and dynamic URL prefix support across dashboard, tabs, auth pages, and project home URLs, fixing issues when locale or prefix is nil.
- Introduced comprehensive dashboard branding and theming:
- Configurable branding, title suffix, and logo handling.
- Shared theme controller with daisyUI integration, color scheme guide, and improved theme switcher placement.
- Enhanced dashboard navigation:
- Configurable subtab styling, redirects, highlights, and mobile subtab support.
- Multiple context selectors with dependency support.
- Reserved additional locale path segments for dashboard and users.
- Added context-aware features:
- Context-aware badges with update helpers, guards for nil contexts, and improved preservation during tab refresh.
- Consistent context-aware merge behavior.
- Improved authentication and user setup:
- Added fetch_phoenix_kit_current_user to the auto-setup pipeline.
- Fixed auth pages and titles to use centralized Settings/Config branding.
- Performance and quality improvements:
- Optimized Presence and Config modules to reduce repeated checks and lookups.
- Added dashboard_assigns/1 helper to prevent unnecessary layout re-rendering.
- Fixed hardcoded branding and paths to rely on configuration fallbacks.
- Documentation updates:
- Added guides for dashboard theming, tab path formats, subtab behavior, and context selectors.
- Added prominent built-in features section and reduced overall documentation size.
- Maintenance:
- Fixed Credo/Dialyzer issues, formatting problems, and test failures.
- Cleaned up unused Dialyzer ignores and added ignores for test support files.
1.7.26 - 2026-01-18
- Language switcher fix
1.7.25 - 2026-01-16
- Bug fix - Added check for nil on language_swithcer on log-in page
1.7.24 - 2026-01-15
- Add Shop module with products, categories, cart, and checkout flow
- Add user billing profiles for reusable billing information
- Add payment options selection in checkout (bank transfer, card payment)
- Add user order pages with UUID-based URLs
- Add PubSub broadcasts to Billing module for real-time updates
- Add automatic default currency for orders
- Add Billing and Shop tabs to user dashboard tab system
- Add automatic dashboard tabs refresh when modules are enabled/disabled
- Fix user dashboard layout sidebar height calculation
- Fix OAuth avatar display in admin navigation
1.7.23 - 2026-01-14
- Added user functions, language switcher on login page (also support for Estonian and Russian on login)
- Removed logs spamming about oban jobs
1.7.22 - 2026-01-13
- Add AWS config module with centralized credential management
- Add context selector for multi-tenant dashboard navigation
- Add comprehensive user dashboard tab system with CLI generator
- Consolidate Publishing module into self-contained structure
- Publishing Module: Versioning, AI Translation, Per-Language URLs & Real-time Updates
- Fixed referralcodes to referrals for more universal code
1.7.21 - 2026-01-10
- Publishing Module: Versioning, AI Translation, Per-Language URLs & Real-time Updates
- Fixed referralcodes to referrals for more universal code
- Consolidate OAuth config through Config.UeberAuth abstraction
1.7.20 - 2026-01-09
- Fix user avatar fallback when Gravatar is unavailable
- Fixed issues with phx_kit install
- Add scheduled job cancellation when disabling modules
- Fix race condition in file controller for parallel requests
1.7.19 - 2026-01-07
We are doing code cleanup and refactoring to move forward with more new modules and more features:
- Moved referral_codes module to correct location lib/modules and fixed issue with install not working
- Standardize admin UI styling and add reusable components
- Move Emails module to lib/modules/emails with PhoenixKit.Modules.Emails namespace
- Migrate Entities, AI, and Blogging modules to lib/modules/ with PhoenixKit.Modules namespace
- Updated the javascript usage to not create userspace javascript files
- Move Sitemap and Billing modules to lib/modules/ with consolidated namespace
- Move DB and Sync modules to lib/modules/ with PhoenixKit.Modules namespace
- Moved posts module files to lib/modules folder
- Add DB Explorer module
1.7.18 - 2026-01-03
- Blog Versioning, Caching System, and Complete Programmatic API
- Add Cookie Consent Widget (Legal Module Phase 2)
- Add Legal module improvements and cookie consent enhancements