From 720a43a38ca86fc2b7799045e877c028751d1b7b Mon Sep 17 00:00:00 2001 From: carla Date: Mon, 12 Jan 2026 17:36:15 +0100 Subject: [PATCH 1/4] feat: added csv templates --- docs/csv-member-import-v1.md | 20 ++++++++++++++++++++ lib/mv_web.ex | 2 +- priv/static/templates/member_import_de.csv | 2 ++ priv/static/templates/member_import_en.csv | 2 ++ 4 files changed, 25 insertions(+), 1 deletion(-) create mode 100644 priv/static/templates/member_import_de.csv create mode 100644 priv/static/templates/member_import_en.csv diff --git a/docs/csv-member-import-v1.md b/docs/csv-member-import-v1.md index 2bdbe69..bc8f99f 100644 --- a/docs/csv-member-import-v1.md +++ b/docs/csv-member-import-v1.md @@ -191,6 +191,26 @@ A **basic CSV member import feature** that allows administrators to upload a CSV - `/templates/member_import_de.csv` - In LiveView, link them using Phoenix static path helpers (e.g. `~p` or `Routes.static_path/2`, depending on Phoenix version). +**Example Usage in LiveView Templates:** + +```heex + +<.link href={~p"/templates/member_import_en.csv"} download> + <%= gettext("Download English Template") %> + + +<.link href={~p"/templates/member_import_de.csv"} download> + <%= gettext("Download German Template") %> + + + +<.link href={Routes.static_path(MvWeb.Endpoint, "/templates/member_import_en.csv")} download> + <%= gettext("Download English Template") %> + +``` + +**Note:** The `templates` directory must be included in `MvWeb.static_paths()` (configured in `lib/mv_web.ex`) for the files to be served. + ### File Limits - **Max file size:** 10 MB diff --git a/lib/mv_web.ex b/lib/mv_web.ex index 8589be1..08a3c23 100644 --- a/lib/mv_web.ex +++ b/lib/mv_web.ex @@ -17,7 +17,7 @@ defmodule MvWeb do those modules here. """ - def static_paths, do: ~w(assets fonts images favicon.ico robots.txt) + def static_paths, do: ~w(assets fonts images favicon.ico robots.txt templates) def router do quote do diff --git a/priv/static/templates/member_import_de.csv b/priv/static/templates/member_import_de.csv new file mode 100644 index 0000000..3bcbeb5 --- /dev/null +++ b/priv/static/templates/member_import_de.csv @@ -0,0 +1,2 @@ +Vorname;Nachname;E-Mail;Straße;PLZ;Stadt +Max;Mustermann;max.mustermann@example.com;Hauptstraße;10115;Berlin diff --git a/priv/static/templates/member_import_en.csv b/priv/static/templates/member_import_en.csv new file mode 100644 index 0000000..d4e67f3 --- /dev/null +++ b/priv/static/templates/member_import_en.csv @@ -0,0 +1,2 @@ +first_name;last_name;email;street;postal_code;city +John;Doe;john.doe@example.com;Main Street;12345;Berlin From 35895ac7fde89a0591a7de37597dc51bd5d8735f Mon Sep 17 00:00:00 2001 From: carla Date: Tue, 13 Jan 2026 10:48:44 +0100 Subject: [PATCH 2/4] fix tests --- lib/mv_web/live/member_live/show.ex | 20 +-- .../member_live/index_display_name_test.exs | 141 ------------------ test/mv_web/member_live/show_test.exs | 24 ++- 3 files changed, 28 insertions(+), 157 deletions(-) delete mode 100644 test/mv_web/member_live/index_display_name_test.exs diff --git a/lib/mv_web/live/member_live/show.ex b/lib/mv_web/live/member_live/show.ex index 997cb1a..5aa4d93 100644 --- a/lib/mv_web/live/member_live/show.ex +++ b/lib/mv_web/live/member_live/show.ex @@ -148,9 +148,9 @@ defmodule MvWeb.MemberLive.Show do <%!-- Custom Fields Section --%> - <%= if Enum.any?(@custom_fields) do %> + <%= if is_list(@custom_fields) && Enum.any?(@custom_fields) do %>
- <.section_box title={gettext("Custom Fields")}> + <.section_box title={gettext("Additional Data Fields")}>
<%= for custom_field <- @custom_fields do %> <% cfv = find_custom_field_value(@member.custom_field_values, custom_field.id) %> @@ -233,13 +233,15 @@ defmodule MvWeb.MemberLive.Show do @impl true def handle_params(%{"id" => id}, _, socket) do - # Load custom fields once using assign_new to avoid repeated queries - socket = - assign_new(socket, :custom_fields, fn -> - Mv.Membership.CustomField - |> Ash.Query.sort(name: :asc) - |> Ash.read!() - end) + # Load custom fields for display + # Note: Each page load starts a new LiveView process, so caching with + # assign_new is not necessary here (mount creates a fresh socket each time) + custom_fields = + Mv.Membership.CustomField + |> Ash.Query.sort(name: :asc) + |> Ash.read!() + + socket = assign(socket, :custom_fields, custom_fields) query = Mv.Membership.Member diff --git a/test/mv_web/member_live/index_display_name_test.exs b/test/mv_web/member_live/index_display_name_test.exs deleted file mode 100644 index 7a11235..0000000 --- a/test/mv_web/member_live/index_display_name_test.exs +++ /dev/null @@ -1,141 +0,0 @@ -defmodule MvWeb.Helpers.MemberHelpersTest do - @moduledoc """ - Tests for the display_name/1 helper function in MemberHelpers. - """ - use Mv.DataCase, async: true - - alias Mv.Membership.Member - alias MvWeb.Helpers.MemberHelpers - - describe "display_name/1" do - test "returns full name when both first_name and last_name are present" do - member = %Member{ - first_name: "John", - last_name: "Doe", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "John Doe" - end - - test "returns email when both first_name and last_name are nil" do - member = %Member{ - first_name: nil, - last_name: nil, - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "john@example.com" - end - - test "returns first_name only when last_name is nil" do - member = %Member{ - first_name: "John", - last_name: nil, - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "John" - end - - test "returns last_name only when first_name is nil" do - member = %Member{ - first_name: nil, - last_name: "Doe", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "Doe" - end - - test "returns email when first_name and last_name are empty strings" do - member = %Member{ - first_name: "", - last_name: "", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "john@example.com" - end - - test "returns email when first_name and last_name are whitespace only" do - member = %Member{ - first_name: " ", - last_name: " \t ", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "john@example.com" - end - - test "trims whitespace from name parts" do - member = %Member{ - first_name: " John ", - last_name: " Doe ", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "John Doe" - end - - test "handles one empty string and one nil" do - member = %Member{ - first_name: "", - last_name: nil, - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "john@example.com" - end - - test "handles one nil and one empty string" do - member = %Member{ - first_name: nil, - last_name: "", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "john@example.com" - end - - test "handles one whitespace and one nil" do - member = %Member{ - first_name: " ", - last_name: nil, - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "john@example.com" - end - - test "handles one valid name and one whitespace" do - member = %Member{ - first_name: "John", - last_name: " ", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "John" - end - - test "handles member with only first_name containing whitespace" do - member = %Member{ - first_name: " John ", - last_name: nil, - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "John" - end - - test "handles member with only last_name containing whitespace" do - member = %Member{ - first_name: nil, - last_name: " Doe ", - email: "john@example.com" - } - - assert MemberHelpers.display_name(member) == "Doe" - end - end -end diff --git a/test/mv_web/member_live/show_test.exs b/test/mv_web/member_live/show_test.exs index 1e04559..33505ec 100644 --- a/test/mv_web/member_live/show_test.exs +++ b/test/mv_web/member_live/show_test.exs @@ -7,13 +7,13 @@ defmodule MvWeb.MemberLive.ShowTest do - Custom Fields section visibility (Issue #282 regression test) - Custom field values formatting - ## Note on async: false - Tests use `async: false` (not `async: true`) to prevent PostgreSQL deadlocks - when creating members and custom fields concurrently. This is intentional and - documented here to avoid confusion in commit messages. + ## Note on async + Tests can run with `async: true` because: + - Each test explicitly manages its own custom fields (creates/deletes as needed) + - The SQL Sandbox provides proper isolation between parallel tests + - Custom field cleanup in tests ensures no interference between tests """ - # async: false to prevent PostgreSQL deadlocks when creating members and custom fields - use MvWeb.ConnCase, async: false + use MvWeb.ConnCase, async: true import Phoenix.LiveViewTest require Ash.Query use Gettext, backend: MvWeb.Gettext @@ -113,11 +113,21 @@ defmodule MvWeb.MemberLive.ShowTest do conn: conn, member: member } do + # Ensure no custom fields exist for this test + # This ensures test isolation even if previous tests created custom fields + existing_custom_fields = Ash.read!(CustomField) + for cf <- existing_custom_fields do + Ash.destroy!(cf) + end + + # Verify no custom fields exist + assert Ash.read!(CustomField) == [] + conn = conn_with_oidc_user(conn) {:ok, _view, html} = live(conn, ~p"/members/#{member}") # Custom Fields section should NOT be visible - refute html =~ gettext("Custom Fields") + refute html =~ gettext("Additional Data Fields") end end From 6fe75db56da6c9564227e12f2675242daa51ba11 Mon Sep 17 00:00:00 2001 From: carla Date: Tue, 13 Jan 2026 10:50:33 +0100 Subject: [PATCH 3/4] formatting --- test/mv_web/member_live/show_test.exs | 1 + 1 file changed, 1 insertion(+) diff --git a/test/mv_web/member_live/show_test.exs b/test/mv_web/member_live/show_test.exs index 33505ec..fdcfebb 100644 --- a/test/mv_web/member_live/show_test.exs +++ b/test/mv_web/member_live/show_test.exs @@ -116,6 +116,7 @@ defmodule MvWeb.MemberLive.ShowTest do # Ensure no custom fields exist for this test # This ensures test isolation even if previous tests created custom fields existing_custom_fields = Ash.read!(CustomField) + for cf <- existing_custom_fields do Ash.destroy!(cf) end From 469c4c0c1d8fd47e64f16df95a0d88805d52b36e Mon Sep 17 00:00:00 2001 From: carla Date: Tue, 13 Jan 2026 10:55:09 +0100 Subject: [PATCH 4/4] i18n: update translations --- priv/gettext/de/LC_MESSAGES/default.po | 6 +++++- priv/gettext/default.pot | 6 +++++- priv/gettext/en/LC_MESSAGES/default.po | 6 +++++- 3 files changed, 15 insertions(+), 3 deletions(-) diff --git a/priv/gettext/de/LC_MESSAGES/default.po b/priv/gettext/de/LC_MESSAGES/default.po index abf6f8d..182a428 100644 --- a/priv/gettext/de/LC_MESSAGES/default.po +++ b/priv/gettext/de/LC_MESSAGES/default.po @@ -633,7 +633,6 @@ msgstr "Bitte wähle zuerst ein Benutzerdefiniertes Feld" #: lib/mv_web/components/layouts/sidebar.ex #: lib/mv_web/live/member_live/form.ex -#: lib/mv_web/live/member_live/show.ex #, elixir-autogen, elixir-format msgid "Custom Fields" msgstr "Benutzerdefinierte Felder" @@ -2059,6 +2058,11 @@ msgstr "" msgid "read_only - Read access to all data" msgstr "" +#: lib/mv_web/live/member_live/show.ex +#, elixir-autogen, elixir-format +msgid "Additional Data Fields" +msgstr "Zusätzliche Datenfelder" + #~ #: lib/mv_web/live/custom_field_live/show.ex #~ #, elixir-autogen, elixir-format #~ msgid "Auto-generated identifier (immutable)" diff --git a/priv/gettext/default.pot b/priv/gettext/default.pot index f7f0f0e..1be771a 100644 --- a/priv/gettext/default.pot +++ b/priv/gettext/default.pot @@ -634,7 +634,6 @@ msgstr "" #: lib/mv_web/components/layouts/sidebar.ex #: lib/mv_web/live/member_live/form.ex -#: lib/mv_web/live/member_live/show.ex #, elixir-autogen, elixir-format msgid "Custom Fields" msgstr "" @@ -2059,3 +2058,8 @@ msgstr "" #, elixir-autogen, elixir-format msgid "read_only - Read access to all data" msgstr "" + +#: lib/mv_web/live/member_live/show.ex +#, elixir-autogen, elixir-format +msgid "Additional Data Fields" +msgstr "" diff --git a/priv/gettext/en/LC_MESSAGES/default.po b/priv/gettext/en/LC_MESSAGES/default.po index 034392e..8b01f61 100644 --- a/priv/gettext/en/LC_MESSAGES/default.po +++ b/priv/gettext/en/LC_MESSAGES/default.po @@ -634,7 +634,6 @@ msgstr "" #: lib/mv_web/components/layouts/sidebar.ex #: lib/mv_web/live/member_live/form.ex -#: lib/mv_web/live/member_live/show.ex #, elixir-autogen, elixir-format, fuzzy msgid "Custom Fields" msgstr "" @@ -2060,6 +2059,11 @@ msgstr "" msgid "read_only - Read access to all data" msgstr "" +#: lib/mv_web/live/member_live/show.ex +#, elixir-autogen, elixir-format +msgid "Additional Data Fields" +msgstr "" + #~ #: lib/mv_web/live/components/payment_filter_component.ex #~ #, elixir-autogen, elixir-format #~ msgid "All payment statuses"