From 498a231f3f38cd71a383806a3080a55602437a11 Mon Sep 17 00:00:00 2001 From: SimonLab Date: Wed, 5 Oct 2022 16:15:19 +0100 Subject: [PATCH 01/56] Create lists table and list schema Create the lists table --- lib/app/list.ex | 37 +++++++++++ .../migrations/20221005142257_add_list.exs | 31 ++++++++++ test/app/list_test.exs | 62 +++++++++++++++++++ 3 files changed, 130 insertions(+) create mode 100644 lib/app/list.ex create mode 100644 priv/repo/migrations/20221005142257_add_list.exs create mode 100644 test/app/list_test.exs diff --git a/lib/app/list.ex b/lib/app/list.ex new file mode 100644 index 00000000..9c267868 --- /dev/null +++ b/lib/app/list.ex @@ -0,0 +1,37 @@ +defmodule App.List do + use Ecto.Schema + import Ecto.Changeset + alias App.{Item, Person, Repo} + alias __MODULE__ + + schema "lists" do + field :name, :string + + belongs_to :people, Person, references: :person_id, foreign_key: :person_id + many_to_many(:items, Item, join_through: "items_lists") + + timestamps() + end + + @doc false + def changeset(list, attrs \\ %{}) do + list + |> cast(attrs, [:person_id, :name]) + |> validate_required([:person_id, :name]) + |> unique_constraint([:name, :person_id], name: :lists_name_person_id_index) + end + + def create_list(attrs) do + %List{} + |> changeset(attrs) + |> Repo.insert() + end + + def get_list!(id), do: Repo.get!(List, id) + + def update_list(%List{} = list, attrs) do + list + |> List.changeset(attrs) + |> Repo.update() + end +end diff --git a/priv/repo/migrations/20221005142257_add_list.exs b/priv/repo/migrations/20221005142257_add_list.exs new file mode 100644 index 00000000..1b8c2d03 --- /dev/null +++ b/priv/repo/migrations/20221005142257_add_list.exs @@ -0,0 +1,31 @@ +defmodule App.Repo.Migrations.AddList do + use Ecto.Migration + + def change do + execute("CREATE EXTENSION IF NOT EXISTS citext") + + create table(:lists) do + add(:name, :citext) + add(:person_id, references(:people, column: :person_id)) + + timestamps() + end + + create( + unique_index(:lists, [:name, :person_id], + name: :lists_name_person_id_index + ) + ) + + create table(:items_lists, primary_key: false) do + add(:item_id, references(:items, on_delete: :delete_all)) + add(:list_id, references(:tags, on_delete: :delete_all)) + + timestamps() + end + + # create a unique index on item_id and list_id to avoid + # adding an item multiple time to the same list + create(unique_index(:items_lists, [:item_id, :list_id])) + end +end diff --git a/test/app/list_test.exs b/test/app/list_test.exs new file mode 100644 index 00000000..dec4b714 --- /dev/null +++ b/test/app/list_test.exs @@ -0,0 +1,62 @@ +defmodule App.ListTest do + use App.DataCase + alias App.{Person, List} + + setup [:create_person] + + describe "Test constraints and requirements for List schema" do + test "valid list changeset" do + changeset = List.changeset(%List{}, %{person_id: 1, name: "list 1"}) + + assert changeset.valid? + end + + test "invalid list changeset when person_id value missing" do + changeset = List.changeset(%List{}, %{name: "list 1"}) + refute changeset.valid? + end + + test "invalid list changeset when name value missing" do + changeset = List.changeset(%List{}, %{person_id: 1}) + refute changeset.valid? + end + end + + describe "Save list in Postgres" do + @valid_attrs %{person_id: 1, name: "list 1"} + @invalid_attrs %{name: nil} + + test "get_list!/1 returns the list with given id" do + {:ok, list} = List.create_list(@valid_attrs) + assert List.get_list!(list.id).name == list.name + end + + test "create_list/1 with invalid data returns error changeset" do + assert {:error, %Ecto.Changeset{}} = List.create_list(@invalid_attrs) + end + + test "create_list/1 returns invalid changeset when trying to insert a duplicate name" do + assert {:ok, _list} = List.create_list(@valid_attrs) + + assert {:error, _changeset} = List.create_list(@valid_attrs) + end + end + + describe "Update list in Postgres" do + @valid_attrs %{person_id: 1, name: "list 1"} + @valid_update_attrs %{person_id: 1, name: "list 1 updated"} + + test "update_list/2 update the list name" do + assert {:ok, list} = List.create_list(@valid_attrs) + + assert {:ok, list_updated} = List.update_list(list, @valid_update_attrs) + + assert list_updated.name == "list 1 updated" + end + end + + defp create_person(_) do + person = Person.create_person(%{"person_id" => 1, "name" => "guest"}) + %{person: person} + end +end From 13e3c6b9cd195f33204944b7b00f5939da382c02 Mon Sep 17 00:00:00 2001 From: SimonLab Date: Wed, 5 Oct 2022 21:08:21 +0100 Subject: [PATCH 02/56] Add list controller Create list controller to add and edit lists --- lib/app/list.ex | 12 ++++ lib/app_web/controllers/list_controller.ex | 75 +++++++++++++++++++++ lib/app_web/router.ex | 1 + lib/app_web/templates/layout/root.html.heex | 5 +- lib/app_web/templates/list/edit.html.heex | 8 +++ lib/app_web/templates/list/form.html.heex | 14 ++++ lib/app_web/templates/list/index.html.heex | 26 +++++++ lib/app_web/templates/list/new.html.heex | 8 +++ lib/app_web/templates/tag/edit.html.heex | 2 +- lib/app_web/templates/tag/form.html.heex | 4 +- lib/app_web/views/list_view.ex | 3 + 11 files changed, 154 insertions(+), 4 deletions(-) create mode 100644 lib/app_web/controllers/list_controller.ex create mode 100644 lib/app_web/templates/list/edit.html.heex create mode 100644 lib/app_web/templates/list/form.html.heex create mode 100644 lib/app_web/templates/list/index.html.heex create mode 100644 lib/app_web/templates/list/new.html.heex create mode 100644 lib/app_web/views/list_view.ex diff --git a/lib/app/list.ex b/lib/app/list.ex index 9c267868..98bc42f7 100644 --- a/lib/app/list.ex +++ b/lib/app/list.ex @@ -1,6 +1,7 @@ defmodule App.List do use Ecto.Schema import Ecto.Changeset + import Ecto.Query alias App.{Item, Person, Repo} alias __MODULE__ @@ -29,9 +30,20 @@ defmodule App.List do def get_list!(id), do: Repo.get!(List, id) + def list_person_lists(person_id) do + List + |> where(person_id: ^person_id) + |> order_by(:name) + |> Repo.all() + end + def update_list(%List{} = list, attrs) do list |> List.changeset(attrs) |> Repo.update() end + + def delete_list(%List{} = list) do + Repo.delete(list) + end end diff --git a/lib/app_web/controllers/list_controller.ex b/lib/app_web/controllers/list_controller.ex new file mode 100644 index 00000000..0250e0b5 --- /dev/null +++ b/lib/app_web/controllers/list_controller.ex @@ -0,0 +1,75 @@ +defmodule AppWeb.ListController do + use AppWeb, :controller + alias App.List + plug :permission_list when action in [:edit, :update, :delete] + + def index(conn, _params) do + person_id = conn.assigns[:person][:id] || 0 + lists = List.list_person_lists(person_id) + + render(conn, "index.html", lists: lists) + end + + def new(conn, _params) do + changeset = List.changeset(%List{}) + render(conn, "new.html", changeset: changeset) + end + + def create(conn, %{"list" => list_params}) do + person_id = conn.assigns[:person][:id] || 0 + list_params = Map.put(list_params, "person_id", person_id) + + case List.create_list(list_params) do + {:ok, _list} -> + conn + |> put_flash(:info, "List created successfully.") + |> redirect(to: Routes.list_path(conn, :index)) + + {:error, %Ecto.Changeset{} = changeset} -> + render(conn, "new.html", changeset: changeset) + end + end + + def edit(conn, %{"id" => id}) do + list = List.get_list!(id) + changeset = List.changeset(list) + render(conn, "edit.html", list: list, changeset: changeset) + end + + def update(conn, %{"id" => id, "list" => list_params}) do + list = List.get_list!(id) + + case List.update_list(list, list_params) do + {:ok, _list} -> + conn + |> put_flash(:info, "List updated successfully.") + |> redirect(to: Routes.list_path(conn, :index)) + + {:error, %Ecto.Changeset{} = changeset} -> + render(conn, "edit.html", list: list, changeset: changeset) + end + end + + def delete(conn, %{"id" => id}) do + list = List.get_list!(id) + {:ok, _list} = List.delete_list(list) + + conn + |> put_flash(:info, "list deleted successfully.") + |> redirect(to: Routes.list_path(conn, :index)) + end + + defp permission_list(conn, _opts) do + list = List.get_list!(conn.params["id"]) + person_id = conn.assigns[:person][:id] || 0 + + if list.person_id == person_id do + conn + else + conn + |> put_flash(:info, "You can't access that page") + |> redirect(to: "/tags") + |> halt() + end + end +end diff --git a/lib/app_web/router.ex b/lib/app_web/router.ex index 5cd023e6..f07266d1 100644 --- a/lib/app_web/router.ex +++ b/lib/app_web/router.ex @@ -31,6 +31,7 @@ defmodule AppWeb.Router do pipe_through [:check_profile_name] live "/", AppLive resources "/tags", TagController, except: [:show] + resources "/lists", ListController, except: [:show] get "/login", AuthController, :login get "/logout", AuthController, :logout end diff --git a/lib/app_web/templates/layout/root.html.heex b/lib/app_web/templates/layout/root.html.heex index ca048d74..ba2551c8 100644 --- a/lib/app_web/templates/layout/root.html.heex +++ b/lib/app_web/templates/layout/root.html.heex @@ -32,9 +32,12 @@ <%= if @loggedin do %>
- <%= link "tags", to: "/tags", class: "text-white font-bold" %>
+ +
+ <%= link "lists", to: "/lists", class: "text-white font-bold" %> +
diff --git a/lib/app_web/templates/list/edit.html.heex b/lib/app_web/templates/list/edit.html.heex new file mode 100644 index 00000000..314f6033 --- /dev/null +++ b/lib/app_web/templates/list/edit.html.heex @@ -0,0 +1,8 @@ +<.container> +<.h2 class="text-center mt-3">Edit List + +<%= render "form.html", Map.put(assigns, :action, Routes.list_path(@conn, :update, @list)) |> Map.put(:method, "patch") %> + +<.a to={Routes.list_path(@conn, :index)} label="Back to lists" /> + + diff --git a/lib/app_web/templates/list/form.html.heex b/lib/app_web/templates/list/form.html.heex new file mode 100644 index 00000000..96765d03 --- /dev/null +++ b/lib/app_web/templates/list/form.html.heex @@ -0,0 +1,14 @@ + <.form :let={f} for={@changeset} action={@action} method={@method} class="py-3"> + <%= if @changeset.action do %> +
+

Oops, something went wrong! Please check the errors below.

+
+ <% end %> + <.form_field + type="text_input" + form={f} + field={:name} + /> + + <.button type="submit" label="Save" /> + diff --git a/lib/app_web/templates/list/index.html.heex b/lib/app_web/templates/list/index.html.heex new file mode 100644 index 00000000..afb335c3 --- /dev/null +++ b/lib/app_web/templates/list/index.html.heex @@ -0,0 +1,26 @@ +<.h2 class="text-center mt-3">Listing lists +<.container> +<.table> + <.tr> + <.th>Name + + <.th class="w-3"> + <.th class="w-3"> + +<%= for list <- @lists do %> + <.tr> + <.td> + <%= list.name %> + + + <.td> + <%= link "Edit", to: Routes.list_path(@conn, :edit, list) %> + + <.td class="!text-red-500"> + <%= link "Delete", to: Routes.list_path(@conn, :delete, list), method: :delete, data: [confirm: "Are you sure you want to delete this list?"] %> + + +<% end %> + +<.button link_type="a" to={Routes.list_path(@conn, :new)} label="New list" class="my-3"/> + diff --git a/lib/app_web/templates/list/new.html.heex b/lib/app_web/templates/list/new.html.heex new file mode 100644 index 00000000..c46aff85 --- /dev/null +++ b/lib/app_web/templates/list/new.html.heex @@ -0,0 +1,8 @@ +<.container> +<.h2 class="text-center mt-3">New list + +<%= render "form.html", Map.put(assigns, :action, Routes.list_path(@conn, :create)) |> Map.put(:method, "post") %> + +<.a to={Routes.list_path(@conn, :index)} label="Back to lists" /> + + diff --git a/lib/app_web/templates/tag/edit.html.heex b/lib/app_web/templates/tag/edit.html.heex index d9fde6ac..db661fec 100644 --- a/lib/app_web/templates/tag/edit.html.heex +++ b/lib/app_web/templates/tag/edit.html.heex @@ -3,6 +3,6 @@ <%= render "form.html", Map.put(assigns, :action, Routes.tag_path(@conn, :update, @tag)) %> -<.a to={Routes.tag_path(@conn, :index)} class="" label="Back to tags" /> +<.a to={Routes.tag_path(@conn, :index)} label="Back to tags" /> diff --git a/lib/app_web/templates/tag/form.html.heex b/lib/app_web/templates/tag/form.html.heex index 462d61ff..dd938aac 100644 --- a/lib/app_web/templates/tag/form.html.heex +++ b/lib/app_web/templates/tag/form.html.heex @@ -1,4 +1,4 @@ - <.form :let={f} for={@changeset} action={@action} method="patch" class="py-3"> +<.form let={f} for={@changeset} action={@action} method="patch" class="py-3"> <%= if @changeset.action do %>

Oops, something went wrong! Please check the errors below.

@@ -12,4 +12,4 @@ <.form_field type="color_input" form={f} field={:color}/> <.button type="submit" label="Save" /> - + diff --git a/lib/app_web/views/list_view.ex b/lib/app_web/views/list_view.ex new file mode 100644 index 00000000..89a80c28 --- /dev/null +++ b/lib/app_web/views/list_view.ex @@ -0,0 +1,3 @@ +defmodule AppWeb.ListView do + use AppWeb, :view +end From 3c3e83c51035dca138649cbe3a8ca310b1e02523 Mon Sep 17 00:00:00 2001 From: SimonLab Date: Wed, 5 Oct 2022 22:17:17 +0100 Subject: [PATCH 03/56] Add item endpoint Create item controller to link items to lists --- lib/app/item.ex | 5 ++++- lib/app_web/controllers/item_controller.ex | 15 +++++++++++++++ lib/app_web/live/app_live.html.heex | 4 ++++ lib/app_web/router.ex | 3 +++ lib/app_web/templates/item/edit.html.heex | 11 +++++++++++ lib/app_web/views/item_view.ex | 3 +++ 6 files changed, 40 insertions(+), 1 deletion(-) create mode 100644 lib/app_web/controllers/item_controller.ex create mode 100644 lib/app_web/templates/item/edit.html.heex create mode 100644 lib/app_web/views/item_view.ex diff --git a/lib/app/item.ex b/lib/app/item.ex index 5fdb282d..a156a2c5 100644 --- a/lib/app/item.ex +++ b/lib/app/item.ex @@ -3,6 +3,7 @@ defmodule App.Item do import Ecto.Changeset import Ecto.Query alias App.{Repo, Tag, ItemTag, Person} + alias App.List, as: L alias __MODULE__ schema "items" do @@ -11,12 +12,13 @@ defmodule App.Item do belongs_to :people, Person, references: :person_id, foreign_key: :person_id many_to_many(:tags, Tag, join_through: ItemTag, on_replace: :delete) + many_to_many(:lists, L, join_through: "items_lists", on_replace: :delete) timestamps() end @doc false - def changeset(item, attrs) do + def changeset(item, attrs \\ %{}) do item |> cast(attrs, [:person_id, :status, :text]) |> validate_required([:text, :person_id]) @@ -69,6 +71,7 @@ defmodule App.Item do Item |> Repo.get!(id) |> Repo.preload(tags: from(t in Tag, order_by: t.text)) + |> Repo.preload(lists: from(l in L, order_by: l.name)) end @doc """ diff --git a/lib/app_web/controllers/item_controller.ex b/lib/app_web/controllers/item_controller.ex new file mode 100644 index 00000000..505e8a58 --- /dev/null +++ b/lib/app_web/controllers/item_controller.ex @@ -0,0 +1,15 @@ +defmodule AppWeb.ItemController do + use AppWeb, :controller + alias App.{Item, List} + # plug :permission_tag when action in [:edit, :update, :delete] + + def edit(conn, %{"id" => id}) do + person_id = conn.assigns[:person][:id] || 0 + + item = Item.get_item!(id) + lists = List.list_person_lists(person_id) + + changeset = Item.changeset(item) + render(conn, "edit.html", item: item, lists: lists, changeset: changeset) + end +end diff --git a/lib/app_web/live/app_live.html.heex b/lib/app_web/live/app_live.html.heex index d013a74a..bb9f87f2 100644 --- a/lib/app_web/live/app_live.html.heex +++ b/lib/app_web/live/app_live.html.heex @@ -258,6 +258,10 @@ <%= live_patch tag.text, to: Routes.live_path(@socket, AppWeb.AppLive, %{filter_by: @filter, filter_by_tag: tag.text}), style: "background-color:#{tag.color}", class: " text-white font-bold py-1 px-2 rounded-full" %> <% end %>
+ +
+ <.a to={Routes.item_path(@socket, :edit, item.id)} class="" label="Edit lists" /> +
<% end %> <% end %> diff --git a/lib/app_web/router.ex b/lib/app_web/router.ex index f07266d1..706e3934 100644 --- a/lib/app_web/router.ex +++ b/lib/app_web/router.ex @@ -34,6 +34,9 @@ defmodule AppWeb.Router do resources "/lists", ListController, except: [:show] get "/login", AuthController, :login get "/logout", AuthController, :logout + + # edit item lists + resources "/items", ItemController, only: [:edit, :update] end # assign to conn the loggedin value used in templates diff --git a/lib/app_web/templates/item/edit.html.heex b/lib/app_web/templates/item/edit.html.heex new file mode 100644 index 00000000..a6f71f13 --- /dev/null +++ b/lib/app_web/templates/item/edit.html.heex @@ -0,0 +1,11 @@ +edit item + +<.form let={f} for={@changeset} action="" method="patch" class="py-3"> + <%= if @changeset.action do %> +
+

Oops, something went wrong! Please check the errors below.

+
+ <% end %> + + <.button type="submit" label="Save" /> + diff --git a/lib/app_web/views/item_view.ex b/lib/app_web/views/item_view.ex new file mode 100644 index 00000000..84240e42 --- /dev/null +++ b/lib/app_web/views/item_view.ex @@ -0,0 +1,3 @@ +defmodule AppWeb.ItemView do + use AppWeb, :view +end From ffe8d64486a57547da25648cf363f58277c08354 Mon Sep 17 00:00:00 2001 From: SimonLab Date: Fri, 7 Oct 2022 11:55:01 +0100 Subject: [PATCH 04/56] Display lists under tags - Add/Remove list linked to an item - Display lists as badges on main page --- lib/app/item.ex | 28 +++++++++++++++-- lib/app/list.ex | 4 +++ lib/app_web/controllers/item_controller.ex | 35 ++++++++++++++++++++-- lib/app_web/live/app_live.html.heex | 11 ++++++- lib/app_web/templates/item/edit.html.heex | 17 +++++++++-- lib/item_list.ex | 12 ++++++++ 6 files changed, 97 insertions(+), 10 deletions(-) create mode 100644 lib/item_list.ex diff --git a/lib/app/item.ex b/lib/app/item.ex index a156a2c5..f52e4aa3 100644 --- a/lib/app/item.ex +++ b/lib/app/item.ex @@ -2,17 +2,18 @@ defmodule App.Item do use Ecto.Schema import Ecto.Changeset import Ecto.Query - alias App.{Repo, Tag, ItemTag, Person} + alias App.{Repo, Tag, ItemList, ItemTag, Person} alias App.List, as: L alias __MODULE__ schema "items" do field :status, :integer field :text, :string + field :item_lists, {:array, :string}, virtual: true belongs_to :people, Person, references: :person_id, foreign_key: :person_id many_to_many(:tags, Tag, join_through: ItemTag, on_replace: :delete) - many_to_many(:lists, L, join_through: "items_lists", on_replace: :delete) + many_to_many(:lists, L, join_through: ItemList, on_replace: :delete) timestamps() end @@ -20,7 +21,7 @@ defmodule App.Item do @doc false def changeset(item, attrs \\ %{}) do item - |> cast(attrs, [:person_id, :status, :text]) + |> cast(attrs, [:person_id, :status, :text, :item_lists]) |> validate_required([:text, :person_id]) end @@ -29,6 +30,15 @@ defmodule App.Item do |> put_assoc(:tags, Tag.parse_and_create_tags(attrs)) end + def changeset_with_lists(item, list_ids) do + # get list based on ids + lists = Repo.all(from l in L, where: l.id in ^list_ids) + + item + |> change() + |> put_assoc(:lists, lists) + end + @doc """ Creates an `item`. @@ -95,6 +105,7 @@ defmodule App.Item do |> where(person_id: ^person_id) |> Repo.all() |> Repo.preload(tags: from(t in Tag, order_by: t.text)) + |> Repo.preload(lists: from(l in L, order_by: l.name)) end @doc """ @@ -124,6 +135,12 @@ defmodule App.Item do |> Repo.update() end + def update_item_with_lists(%Item{} = item, list_ids) do + item + |> Item.changeset_with_lists(list_ids) + |> Repo.update() + end + def delete_item(id) do get_item!(id) |> Item.changeset(%{status: 6}) @@ -165,6 +182,11 @@ defmodule App.Item do accumulate_item_timers(values) |> Enum.map(fn t -> Map.put(t, :tags, items_tags[t.id].tags) + # Map.put(t, :lists, items_tags[t.id].lists) + end) + |> Enum.map(fn t -> + # Map.put(t, :tags, items_tags[t.id].tags) + Map.put(t, :lists, items_tags[t.id].lists) end) end diff --git a/lib/app/list.ex b/lib/app/list.ex index 98bc42f7..b0fdd70b 100644 --- a/lib/app/list.ex +++ b/lib/app/list.ex @@ -30,6 +30,10 @@ defmodule App.List do def get_list!(id), do: Repo.get!(List, id) + def get_lists_from_ids(ids) do + Repo.all(from l in List, where: l.id in ^ids) + end + def list_person_lists(person_id) do List |> where(person_id: ^person_id) diff --git a/lib/app_web/controllers/item_controller.ex b/lib/app_web/controllers/item_controller.ex index 505e8a58..f8ad1b05 100644 --- a/lib/app_web/controllers/item_controller.ex +++ b/lib/app_web/controllers/item_controller.ex @@ -7,9 +7,38 @@ defmodule AppWeb.ItemController do person_id = conn.assigns[:person][:id] || 0 item = Item.get_item!(id) - lists = List.list_person_lists(person_id) + lists = List.list_person_lists(person_id) |> Enum.map(&{&1.name, &1.id}) + selected_list_id = Enum.map(item.lists, & &1.id) - changeset = Item.changeset(item) - render(conn, "edit.html", item: item, lists: lists, changeset: changeset) + changeset = Item.changeset(item, %{item_lists: selected_list_id}) + + render(conn, "edit.html", + item: item, + lists: lists, + changeset: changeset + ) + end + + def update(conn, %{"id" => id} = params) do + person_id = conn.assigns[:person][:id] || 0 + item = Item.get_item!(id) + + list_ids = + case params["item"]["item_lists"] do + "" -> [] + ids -> ids + end + + case Item.update_item_with_lists(item, list_ids) do + {:ok, _item} -> + conn + |> put_flash(:info, "Item's list updated successfully.") + |> redirect(to: "/") + + {:error, %Ecto.Changeset{} = changeset} -> + lists = List.list_person_lists(person_id) |> Enum.map(&{&1.name, &1.id}) + + render(conn, "edit.html", item: item, lists: lists, changeset: changeset) + end end end diff --git a/lib/app_web/live/app_live.html.heex b/lib/app_web/live/app_live.html.heex index bb9f87f2..b44f39e5 100644 --- a/lib/app_web/live/app_live.html.heex +++ b/lib/app_web/live/app_live.html.heex @@ -253,14 +253,23 @@ <%= if item.id != @editing do %> +
<%= for tag <- item.tags do %> <%= live_patch tag.text, to: Routes.live_path(@socket, AppWeb.AppLive, %{filter_by: @filter, filter_by_tag: tag.text}), style: "background-color:#{tag.color}", class: " text-white font-bold py-1 px-2 rounded-full" %> <% end %>
+
+ <%= for list <- item.lists do %> + <.badge color="info" label={list.name} /> + <% end %> +
+ <.a to={Routes.item_path(@socket, :edit, item.id)} class="" label="Edit lists" /> +
+
+
- <.a to={Routes.item_path(@socket, :edit, item.id)} class="" label="Edit lists" />
<% end %> <% end %> diff --git a/lib/app_web/templates/item/edit.html.heex b/lib/app_web/templates/item/edit.html.heex index a6f71f13..5fd0e9e4 100644 --- a/lib/app_web/templates/item/edit.html.heex +++ b/lib/app_web/templates/item/edit.html.heex @@ -1,11 +1,22 @@ -edit item +<.container> +<.h2 class="text-center mt-3">Edit Item's lists -<.form let={f} for={@changeset} action="" method="patch" class="py-3"> +<.form let={f} for={@changeset} action={Routes.item_path(@conn, :update, @item)} method="patch" class="py-3"> <%= if @changeset.action do %>

Oops, something went wrong! Please check the errors below.

<% end %> - + + <.form_field + type="checkbox_group" + form={f} + field={:item_lists} + label="Lists" + options={@lists} + /> <.button type="submit" label="Save" /> + +<.a to={Routes.list_path(@conn, :new)} class="mt-2" label="Create a list" /> + diff --git a/lib/item_list.ex b/lib/item_list.ex new file mode 100644 index 00000000..12fc06ef --- /dev/null +++ b/lib/item_list.ex @@ -0,0 +1,12 @@ +defmodule App.ItemList do + use Ecto.Schema + alias App.{Item, List} + + @primary_key false + schema "items_lists" do + belongs_to(:item, Item) + belongs_to(:list, List) + + timestamps() + end +end From 74e73ab3694bf27b11c3c9868cccca105ec0fba2 Mon Sep 17 00:00:00 2001 From: SimonLab Date: Fri, 7 Oct 2022 20:56:19 +0100 Subject: [PATCH 05/56] Insert or get person Check if person exist before insert --- lib/app/item.ex | 4 ++-- lib/app/list.ex | 4 ++-- lib/app/person.ex | 5 +++++ lib/app_web/router.ex | 6 ++++-- lib/{item_list.ex => list_item.ex} | 4 ++-- priv/repo/migrations/20221005142257_add_list.exs | 4 ++-- priv/repo/seeds.exs | 2 ++ test/app_web/controllers/tag_controller_test.exs | 2 +- 8 files changed, 20 insertions(+), 11 deletions(-) rename lib/{item_list.ex => list_item.ex} (74%) diff --git a/lib/app/item.ex b/lib/app/item.ex index f52e4aa3..448d621a 100644 --- a/lib/app/item.ex +++ b/lib/app/item.ex @@ -2,7 +2,7 @@ defmodule App.Item do use Ecto.Schema import Ecto.Changeset import Ecto.Query - alias App.{Repo, Tag, ItemList, ItemTag, Person} + alias App.{Repo, Tag, ListItem, ItemTag, Person} alias App.List, as: L alias __MODULE__ @@ -13,7 +13,7 @@ defmodule App.Item do belongs_to :people, Person, references: :person_id, foreign_key: :person_id many_to_many(:tags, Tag, join_through: ItemTag, on_replace: :delete) - many_to_many(:lists, L, join_through: ItemList, on_replace: :delete) + many_to_many(:lists, L, join_through: ListItem, on_replace: :delete) timestamps() end diff --git a/lib/app/list.ex b/lib/app/list.ex index b0fdd70b..8374ed23 100644 --- a/lib/app/list.ex +++ b/lib/app/list.ex @@ -2,14 +2,14 @@ defmodule App.List do use Ecto.Schema import Ecto.Changeset import Ecto.Query - alias App.{Item, Person, Repo} + alias App.{Item, ListItem, Person, Repo} alias __MODULE__ schema "lists" do field :name, :string belongs_to :people, Person, references: :person_id, foreign_key: :person_id - many_to_many(:items, Item, join_through: "items_lists") + many_to_many(:items, Item, join_through: ListItem) timestamps() end diff --git a/lib/app/person.ex b/lib/app/person.ex index c01f311c..c46b2164 100644 --- a/lib/app/person.ex +++ b/lib/app/person.ex @@ -36,4 +36,9 @@ defmodule App.Person do |> Person.changeset(attrs) |> Repo.update() end + + def get_or_insert(person_id) do + Repo.get_by(Person, person_id: person_id) || + Repo.insert!(%Person{person_id: person_id}) + end end diff --git a/lib/app_web/router.ex b/lib/app_web/router.ex index 706e3934..a6ed78d7 100644 --- a/lib/app_web/router.ex +++ b/lib/app_web/router.ex @@ -1,6 +1,6 @@ defmodule AppWeb.Router do use AppWeb, :router - alias App.Person + alias App.{Person, Repo} pipeline :browser do plug :accepts, ["html"] @@ -52,7 +52,9 @@ defmodule AppWeb.Router do # add name to their profile for sharing items feature defp profile_name(conn, _opts) do person_id = conn.assigns[:person][:id] || 0 - person = Person.get_person!(person_id) + # person = Person.get_person!(person_id) + + person = Person.get_or_insert(person_id) if is_nil(person.name) do conn diff --git a/lib/item_list.ex b/lib/list_item.ex similarity index 74% rename from lib/item_list.ex rename to lib/list_item.ex index 12fc06ef..0b8e0e9e 100644 --- a/lib/item_list.ex +++ b/lib/list_item.ex @@ -1,9 +1,9 @@ -defmodule App.ItemList do +defmodule App.ListItem do use Ecto.Schema alias App.{Item, List} @primary_key false - schema "items_lists" do + schema "lists_items" do belongs_to(:item, Item) belongs_to(:list, List) diff --git a/priv/repo/migrations/20221005142257_add_list.exs b/priv/repo/migrations/20221005142257_add_list.exs index 1b8c2d03..71a3fa4e 100644 --- a/priv/repo/migrations/20221005142257_add_list.exs +++ b/priv/repo/migrations/20221005142257_add_list.exs @@ -17,7 +17,7 @@ defmodule App.Repo.Migrations.AddList do ) ) - create table(:items_lists, primary_key: false) do + create table(:lists_items, primary_key: false) do add(:item_id, references(:items, on_delete: :delete_all)) add(:list_id, references(:tags, on_delete: :delete_all)) @@ -26,6 +26,6 @@ defmodule App.Repo.Migrations.AddList do # create a unique index on item_id and list_id to avoid # adding an item multiple time to the same list - create(unique_index(:items_lists, [:item_id, :list_id])) + create(unique_index(:lists_items, [:item_id, :list_id])) end end diff --git a/priv/repo/seeds.exs b/priv/repo/seeds.exs index 77e790ed..bd927863 100644 --- a/priv/repo/seeds.exs +++ b/priv/repo/seeds.exs @@ -1,3 +1,5 @@ +alias App.{Person, Repo} + if not Envar.is_set?("AUTH_API_KEY") do Envar.load(".env") end diff --git a/test/app_web/controllers/tag_controller_test.exs b/test/app_web/controllers/tag_controller_test.exs index 96c5cd0c..a8fac12f 100644 --- a/test/app_web/controllers/tag_controller_test.exs +++ b/test/app_web/controllers/tag_controller_test.exs @@ -95,7 +95,7 @@ defmodule AppWeb.TagControllerTest do defp create_person(_) do person = Person.create_person(%{"person_id" => 0, "name" => "guest"}) - person = Person.create_person(%{"person_id" => 1, "name" => "Person1"}) + _person = Person.create_person(%{"person_id" => 1, "name" => "Person1"}) %{person: person} end end From a3269b7bd597ef26391eb447556a08bd2e29879a Mon Sep 17 00:00:00 2001 From: SimonLab Date: Fri, 7 Oct 2022 22:44:27 +0100 Subject: [PATCH 06/56] Add tests for lists Add schemas and controllers tests --- lib/app_web/controllers/list_controller.ex | 2 +- priv/repo/seeds.exs | 2 - test/app/item_test.exs | 3 +- test/app/list_test.exs | 23 +++- test/app/person_test.exs | 9 ++ .../controllers/item_controller_test.exs | 66 ++++++++++ .../controllers/list_controller_test.exs | 118 ++++++++++++++++++ 7 files changed, 218 insertions(+), 5 deletions(-) create mode 100644 test/app_web/controllers/item_controller_test.exs create mode 100644 test/app_web/controllers/list_controller_test.exs diff --git a/lib/app_web/controllers/list_controller.ex b/lib/app_web/controllers/list_controller.ex index 0250e0b5..9c3150b3 100644 --- a/lib/app_web/controllers/list_controller.ex +++ b/lib/app_web/controllers/list_controller.ex @@ -68,7 +68,7 @@ defmodule AppWeb.ListController do else conn |> put_flash(:info, "You can't access that page") - |> redirect(to: "/tags") + |> redirect(to: "/lists") |> halt() end end diff --git a/priv/repo/seeds.exs b/priv/repo/seeds.exs index bd927863..77e790ed 100644 --- a/priv/repo/seeds.exs +++ b/priv/repo/seeds.exs @@ -1,5 +1,3 @@ -alias App.{Person, Repo} - if not Envar.is_set?("AUTH_API_KEY") do Envar.load(".env") end diff --git a/test/app/item_test.exs b/test/app/item_test.exs index 61a9770d..4ca5abf0 100644 --- a/test/app/item_test.exs +++ b/test/app/item_test.exs @@ -1,6 +1,7 @@ defmodule App.ItemTest do - use App.DataCase + use App.DataCase, async: false alias App.{Item, Person, Timer} + alias App.List, as: L setup [:create_person] diff --git a/test/app/list_test.exs b/test/app/list_test.exs index dec4b714..51cb93db 100644 --- a/test/app/list_test.exs +++ b/test/app/list_test.exs @@ -1,5 +1,5 @@ defmodule App.ListTest do - use App.DataCase + use App.DataCase, async: false alias App.{Person, List} setup [:create_person] @@ -40,6 +40,18 @@ defmodule App.ListTest do assert {:error, _changeset} = List.create_list(@valid_attrs) end + + test "get lists from ids" do + {:ok, _list} = List.create_list(@valid_attrs) + assert lists = List.get_lists_from_ids([1, 2, 3]) + assert length(lists) == 1 + end + + test "get the lists for a person" do + {:ok, _list} = List.create_list(@valid_attrs) + assert lists = List.list_person_lists(1) + assert length(lists) == 1 + end end describe "Update list in Postgres" do @@ -55,6 +67,15 @@ defmodule App.ListTest do end end + describe "Delete list in Postgres" do + @valid_attrs %{person_id: 1, name: "list 1"} + + test "delet the list" do + assert {:ok, list} = List.create_list(@valid_attrs) + assert {:ok, _} = List.delete_list(list) + end + end + defp create_person(_) do person = Person.create_person(%{"person_id" => 1, "name" => "guest"}) %{person: person} diff --git a/test/app/person_test.exs b/test/app/person_test.exs index d1c53ed0..95185d9b 100644 --- a/test/app/person_test.exs +++ b/test/app/person_test.exs @@ -39,6 +39,15 @@ defmodule App.PersonTest do assert {:error, _changeset} = Person.create_person(%{person_id: 2, name: "person 1"}) end + + test "get_or_insert preson" do + assert person = Person.get_or_insert(0) + assert person.person_id == 0 + + {:ok, person} = Person.create_person(@valid_attrs) + assert get_person = Person.get_or_insert(1) + assert get_person.name == person.name + end end describe "Update person in Postgres" do diff --git a/test/app_web/controllers/item_controller_test.exs b/test/app_web/controllers/item_controller_test.exs new file mode 100644 index 00000000..2a78337c --- /dev/null +++ b/test/app_web/controllers/item_controller_test.exs @@ -0,0 +1,66 @@ +defmodule AppWeb.ItemControllerTest do + use AppWeb.ConnCase + + alias App.{List, Person, Item} + + setup [:create_person] + + @create_attrs %{name: "list1", person_id: 1} + @create_attrs_item %{text: "item1", person_id: 1} + + def fixture(:list) do + {:ok, list} = List.create_list(@create_attrs) + list + end + + def fixture(:item) do + {:ok, item} = Item.create_item(@create_attrs_item) + item + end + + describe "edit item" do + setup [:create_item, :create_list] + + test "renders form for editing chosen item's lists", %{ + conn: conn, + list: list, + item: item + } do + conn = + conn + |> assign(:person, %{id: 1}) + |> get(Routes.item_path(conn, :edit, item)) + + assert html_response(conn, 200) =~ "Edit Item's lists" + end + end + + # describe "update item's list" do + # setup [:create_item, :create_list] + # + # test "redirects to index when data is valid", %{conn: conn, list: list} do + # conn = + # conn + # |> assign(:person, %{id: 1}) + # |> put(Routes.list_path(conn, :update, list), list: @update_attrs) + # + # assert redirected_to(conn) == Routes.list_path(conn, :index) + # end + # end + + defp create_list(_) do + list = fixture(:list) + %{list: list} + end + + defp create_item(_) do + item = fixture(:item) + %{item: item} + end + + defp create_person(_) do + person = Person.create_person(%{"person_id" => 0, "name" => "guest"}) + _person1 = Person.create_person(%{"person_id" => 1, "name" => "person1"}) + %{person: person} + end +end diff --git a/test/app_web/controllers/list_controller_test.exs b/test/app_web/controllers/list_controller_test.exs new file mode 100644 index 00000000..9dab06a5 --- /dev/null +++ b/test/app_web/controllers/list_controller_test.exs @@ -0,0 +1,118 @@ +defmodule AppWeb.ListControllerTest do + use AppWeb.ConnCase + + alias App.{List, Person} + + setup [:create_person] + + @create_attrs %{name: "list1", person_id: 1} + @update_attrs %{name: "list1 updated"} + @invalid_attrs %{name: nil, person_id: 1} + + def fixture(:list) do + {:ok, list} = List.create_list(@create_attrs) + list + end + + describe "index" do + test "lists all lists", %{conn: conn} do + conn = get(conn, Routes.list_path(conn, :index)) + assert html_response(conn, 200) =~ "Listing lists" + end + end + + describe "new list" do + test "Display new list form", %{conn: conn} do + conn = get(conn, Routes.list_path(conn, :new)) + assert html_response(conn, 200) =~ "New list" + end + end + + describe "create list" do + test "redirects to index when data is valid", %{conn: conn} do + conn = + conn + |> assign(:person, %{id: 1}) + |> post(Routes.list_path(conn, :create), list: @create_attrs) + + assert redirected_to(conn) == Routes.list_path(conn, :index) + end + + test "renders errors when data is invalid", %{conn: conn} do + conn = + conn + |> assign(:person, %{id: 1}) + |> post(Routes.list_path(conn, :create), list: @invalid_attrs) + + assert html_response(conn, 200) =~ "New list" + end + end + + describe "edit list" do + setup [:create_list] + + test "renders form for editing chosen list", %{conn: conn, list: list} do + conn = + conn + |> assign(:person, %{id: 1}) + |> get(Routes.list_path(conn, :edit, list)) + + assert html_response(conn, 200) =~ "Edit List" + end + + test "redirect to index when missing permission to edit the list", %{ + conn: conn, + list: list + } do + conn = get(conn, Routes.list_path(conn, :edit, list)) + + assert redirected_to(conn) == Routes.list_path(conn, :index) + end + end + + describe "update list" do + setup [:create_list] + + test "redirects to index when data is valid", %{conn: conn, list: list} do + conn = + conn + |> assign(:person, %{id: 1}) + |> put(Routes.list_path(conn, :update, list), list: @update_attrs) + + assert redirected_to(conn) == Routes.list_path(conn, :index) + end + + test "renders errors when data is invalid", %{conn: conn, list: list} do + conn = + conn + |> assign(:person, %{id: 1}) + |> put(Routes.list_path(conn, :update, list), list: @invalid_attrs) + + assert html_response(conn, 200) =~ "Edit List" + end + end + + describe "delete list" do + setup [:create_list] + + test "deletes chosen list", %{conn: conn, list: list} do + conn = + conn + |> assign(:person, %{id: 1}) + |> delete(Routes.list_path(conn, :delete, list)) + + assert redirected_to(conn) == Routes.list_path(conn, :index) + end + end + + defp create_list(_) do + list = fixture(:list) + %{list: list} + end + + defp create_person(_) do + person = Person.create_person(%{"person_id" => 0, "name" => "guest"}) + _person1 = Person.create_person(%{"person_id" => 1, "name" => "person1"}) + %{person: person} + end +end From 68f556225d568e2e3e90ff8092f8041f8fb0b6c5 Mon Sep 17 00:00:00 2001 From: SimonLab Date: Sat, 8 Oct 2022 15:32:11 +0100 Subject: [PATCH 07/56] Add item controller test for lists - Add test for item controller managing lists - Fix https://github.com/dwyl/mvp/issues/168 --- lib/app/item.ex | 3 - lib/app_web/controllers/item_controller.ex | 15 ++--- lib/app_web/router.ex | 3 +- .../migrations/20221005142257_add_list.exs | 2 +- test/app/item_test.exs | 1 - test/app/list_test.exs | 4 +- .../controllers/item_controller_test.exs | 55 +++++++++++++------ 7 files changed, 47 insertions(+), 36 deletions(-) diff --git a/lib/app/item.ex b/lib/app/item.ex index 448d621a..13496cc5 100644 --- a/lib/app/item.ex +++ b/lib/app/item.ex @@ -31,7 +31,6 @@ defmodule App.Item do end def changeset_with_lists(item, list_ids) do - # get list based on ids lists = Repo.all(from l in L, where: l.id in ^list_ids) item @@ -182,10 +181,8 @@ defmodule App.Item do accumulate_item_timers(values) |> Enum.map(fn t -> Map.put(t, :tags, items_tags[t.id].tags) - # Map.put(t, :lists, items_tags[t.id].lists) end) |> Enum.map(fn t -> - # Map.put(t, :tags, items_tags[t.id].tags) Map.put(t, :lists, items_tags[t.id].lists) end) end diff --git a/lib/app_web/controllers/item_controller.ex b/lib/app_web/controllers/item_controller.ex index f8ad1b05..e693c9ee 100644 --- a/lib/app_web/controllers/item_controller.ex +++ b/lib/app_web/controllers/item_controller.ex @@ -20,7 +20,6 @@ defmodule AppWeb.ItemController do end def update(conn, %{"id" => id} = params) do - person_id = conn.assigns[:person][:id] || 0 item = Item.get_item!(id) list_ids = @@ -29,16 +28,10 @@ defmodule AppWeb.ItemController do ids -> ids end - case Item.update_item_with_lists(item, list_ids) do - {:ok, _item} -> - conn - |> put_flash(:info, "Item's list updated successfully.") - |> redirect(to: "/") - - {:error, %Ecto.Changeset{} = changeset} -> - lists = List.list_person_lists(person_id) |> Enum.map(&{&1.name, &1.id}) + {:ok, _item} = Item.update_item_with_lists(item, list_ids) - render(conn, "edit.html", item: item, lists: lists, changeset: changeset) - end + conn + |> put_flash(:info, "Item's list updated successfully.") + |> redirect(to: "/") end end diff --git a/lib/app_web/router.ex b/lib/app_web/router.ex index a6ed78d7..5b925068 100644 --- a/lib/app_web/router.ex +++ b/lib/app_web/router.ex @@ -1,6 +1,6 @@ defmodule AppWeb.Router do use AppWeb, :router - alias App.{Person, Repo} + alias App.Person pipeline :browser do plug :accepts, ["html"] @@ -52,7 +52,6 @@ defmodule AppWeb.Router do # add name to their profile for sharing items feature defp profile_name(conn, _opts) do person_id = conn.assigns[:person][:id] || 0 - # person = Person.get_person!(person_id) person = Person.get_or_insert(person_id) diff --git a/priv/repo/migrations/20221005142257_add_list.exs b/priv/repo/migrations/20221005142257_add_list.exs index 71a3fa4e..58cc1316 100644 --- a/priv/repo/migrations/20221005142257_add_list.exs +++ b/priv/repo/migrations/20221005142257_add_list.exs @@ -19,7 +19,7 @@ defmodule App.Repo.Migrations.AddList do create table(:lists_items, primary_key: false) do add(:item_id, references(:items, on_delete: :delete_all)) - add(:list_id, references(:tags, on_delete: :delete_all)) + add(:list_id, references(:lists, on_delete: :delete_all)) timestamps() end diff --git a/test/app/item_test.exs b/test/app/item_test.exs index 4ca5abf0..d1f9152e 100644 --- a/test/app/item_test.exs +++ b/test/app/item_test.exs @@ -1,7 +1,6 @@ defmodule App.ItemTest do use App.DataCase, async: false alias App.{Item, Person, Timer} - alias App.List, as: L setup [:create_person] diff --git a/test/app/list_test.exs b/test/app/list_test.exs index 51cb93db..acfb3b1a 100644 --- a/test/app/list_test.exs +++ b/test/app/list_test.exs @@ -42,8 +42,8 @@ defmodule App.ListTest do end test "get lists from ids" do - {:ok, _list} = List.create_list(@valid_attrs) - assert lists = List.get_lists_from_ids([1, 2, 3]) + {:ok, list} = List.create_list(@valid_attrs) + assert lists = List.get_lists_from_ids([list.id]) assert length(lists) == 1 end diff --git a/test/app_web/controllers/item_controller_test.exs b/test/app_web/controllers/item_controller_test.exs index 2a78337c..20ae0eba 100644 --- a/test/app_web/controllers/item_controller_test.exs +++ b/test/app_web/controllers/item_controller_test.exs @@ -18,12 +18,11 @@ defmodule AppWeb.ItemControllerTest do item end - describe "edit item" do + describe "edit item's list" do setup [:create_item, :create_list] - test "renders form for editing chosen item's lists", %{ + test "renders form for editing item's lists", %{ conn: conn, - list: list, item: item } do conn = @@ -31,22 +30,46 @@ defmodule AppWeb.ItemControllerTest do |> assign(:person, %{id: 1}) |> get(Routes.item_path(conn, :edit, item)) - assert html_response(conn, 200) =~ "Edit Item's lists" + assert html_response(conn, 200) =~ "Edit Item" end end - # describe "update item's list" do - # setup [:create_item, :create_list] - # - # test "redirects to index when data is valid", %{conn: conn, list: list} do - # conn = - # conn - # |> assign(:person, %{id: 1}) - # |> put(Routes.list_path(conn, :update, list), list: @update_attrs) - # - # assert redirected_to(conn) == Routes.list_path(conn, :index) - # end - # end + describe "update item's list" do + setup [:create_item, :create_list] + + test "Add a list to an item", %{ + conn: conn, + list: list, + item: item + } do + update_params = %{item_lists: [list.id]} + + conn = + conn + |> assign(:person, %{id: 1}) + |> put(Routes.item_path(conn, :update, item), item: update_params) + + assert length(Item.get_item!(item.id).lists) == 1 + + assert redirected_to(conn) == "/" + end + + test "Remove list from item", %{ + conn: conn, + item: item + } do + update_params = %{item_lists: ""} + + conn = + conn + |> assign(:person, %{id: 1}) + |> put(Routes.item_path(conn, :update, item), item: update_params) + + assert length(Item.get_item!(item.id).lists) == 0 + + assert redirected_to(conn) == "/" + end + end defp create_list(_) do list = fixture(:list) From c11fb5c5fc1985a143cbfbbe73491a0cf338ed56 Mon Sep 17 00:00:00 2001 From: nelsonic Date: Tue, 8 Nov 2022 08:42:05 +0000 Subject: [PATCH 08/56] tidy BUILDIT.md --- BUILDIT.md | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/BUILDIT.md b/BUILDIT.md index b85893f5..144a910f 100644 --- a/BUILDIT.md +++ b/BUILDIT.md @@ -28,11 +28,11 @@ and _then_ built the UI.
We were able to do this because we had a good idea of which functions we were going to need.
If you are reading through this -and scratching your head + wondering where a particular function will be used, simply scroll down to the UI section where (_hopefully_) it will all be clear. - +mii At the end of each step, remember to run the tests: @@ -86,6 +86,11 @@ With that in place, let's get building! - [9. Update the `LiveView` Template](#9-update-the-liveview-template) - [10. Filter Items](#10-filter-items) - [11. Tags](#11-tags) + - [11.1 Migrations](#111-migrations) + - [11.2 Schemas](#112-schemas) + - [11.3 Test tags with Iex](#113-test-tags-with-iex) + - [11.4 Testing Schemas](#114-testing-schemas) + - [11.4 Items-Tags association](#114--items-tags-association) - [12. Run the _Finished_ MVP App!](#12-run-the-finished-mvp-app) - [12.1 Run the Tests](#121-run-the-tests) - [12.2 Run The App](#122-run-the-app) @@ -482,7 +487,7 @@ A `timer` is associated with an `item` to track how long it takes to ***complete***. + `id`: `Int` - + `item_id` (Foreign Key `item.id`) + + `item_id` (Foreign Key `item.id`)mvvvvyvyvy + `start`: `NaiveDateTime` - start time for the timer + `stop`: `NaiveDateTime` - stop time for the timer + `inserted_at`: `NaiveDateTime` - record insertion time From bd65456d623f706974373d047edf67640144fe2b Mon Sep 17 00:00:00 2001 From: nelsonic Date: Fri, 8 Sep 2023 10:18:14 +0100 Subject: [PATCH 09/56] remove Person & Profile code #118 --- BUILDIT.md | 2 +- lib/app/item.ex | 28 +---- lib/app/{list.ex => list.ex_BAK} | 0 lib/app/{person.ex => person.ex_BAK} | 0 lib/app/tag.ex | 4 +- ...m_controller.ex => item_controller.ex_BAK} | 0 ...t_controller.ex => list_controller.ex_BAK} | 0 ...ontroller.ex => profile_controller.ex_BAK} | 0 lib/app_web/live/app_live.html.heex | 11 -- lib/app_web/router.ex | 61 ++++------ .../{edit.html.heex => edit.html.heex_BAK} | 0 lib/app_web/templates/layout/root.html.heex | 2 - lib/app_web/templates/profile/edit.html.heex | 19 --- .../views/{item_view.ex => item_view.ex_BAK} | 0 lib/app_web/views/profile_view.ex | 3 - lib/{list_item.ex => list_item.ex_BAK} | 0 mix.exs | 57 ++++++--- mix.lock | 110 ++++++++++-------- .../migrations/20221004195054_add_person.exs | 30 ----- .../migrations/20221005142257_add_list.exs | 31 ----- test/app/item_test.exs | 11 +- test/app/{list_test.exs => list_test.exs_BAK} | 0 test/app/person_test.exs | 66 ----------- test/app/tag_test.exs | 9 +- test/app/timer_test.exs | 12 +- .../controllers/item_controller_test.exs | 89 -------------- ..._test.exs => list_controller_test.exs_BAK} | 0 .../controllers/profile_controller_test.exs | 92 --------------- .../controllers/tag_controller_test.exs | 14 +-- test/app_web/live/app_live_test.exs | 26 +---- 30 files changed, 140 insertions(+), 537 deletions(-) rename lib/app/{list.ex => list.ex_BAK} (100%) rename lib/app/{person.ex => person.ex_BAK} (100%) rename lib/app_web/controllers/{item_controller.ex => item_controller.ex_BAK} (100%) rename lib/app_web/controllers/{list_controller.ex => list_controller.ex_BAK} (100%) rename lib/app_web/controllers/{profile_controller.ex => profile_controller.ex_BAK} (100%) rename lib/app_web/templates/item/{edit.html.heex => edit.html.heex_BAK} (100%) delete mode 100644 lib/app_web/templates/profile/edit.html.heex rename lib/app_web/views/{item_view.ex => item_view.ex_BAK} (100%) delete mode 100644 lib/app_web/views/profile_view.ex rename lib/{list_item.ex => list_item.ex_BAK} (100%) delete mode 100644 priv/repo/migrations/20221004195054_add_person.exs delete mode 100644 priv/repo/migrations/20221005142257_add_list.exs rename test/app/{list_test.exs => list_test.exs_BAK} (100%) delete mode 100644 test/app/person_test.exs delete mode 100644 test/app_web/controllers/item_controller_test.exs rename test/app_web/controllers/{list_controller_test.exs => list_controller_test.exs_BAK} (100%) delete mode 100644 test/app_web/controllers/profile_controller_test.exs diff --git a/BUILDIT.md b/BUILDIT.md index 144a910f..87df02e5 100644 --- a/BUILDIT.md +++ b/BUILDIT.md @@ -487,7 +487,7 @@ A `timer` is associated with an `item` to track how long it takes to ***complete***. + `id`: `Int` - + `item_id` (Foreign Key `item.id`)mvvvvyvyvy + + `item_id` (Foreign Key `item.id`) + `start`: `NaiveDateTime` - start time for the timer + `stop`: `NaiveDateTime` - stop time for the timer + `inserted_at`: `NaiveDateTime` - record insertion time diff --git a/lib/app/item.ex b/lib/app/item.ex index 13496cc5..1da167c0 100644 --- a/lib/app/item.ex +++ b/lib/app/item.ex @@ -2,18 +2,15 @@ defmodule App.Item do use Ecto.Schema import Ecto.Changeset import Ecto.Query - alias App.{Repo, Tag, ListItem, ItemTag, Person} - alias App.List, as: L + alias App.{Repo, Tag, ItemTag} alias __MODULE__ schema "items" do field :status, :integer field :text, :string - field :item_lists, {:array, :string}, virtual: true + field :person_id, :integer - belongs_to :people, Person, references: :person_id, foreign_key: :person_id many_to_many(:tags, Tag, join_through: ItemTag, on_replace: :delete) - many_to_many(:lists, L, join_through: ListItem, on_replace: :delete) timestamps() end @@ -21,7 +18,7 @@ defmodule App.Item do @doc false def changeset(item, attrs \\ %{}) do item - |> cast(attrs, [:person_id, :status, :text, :item_lists]) + |> cast(attrs, [:person_id, :status, :text]) |> validate_required([:text, :person_id]) end @@ -30,14 +27,6 @@ defmodule App.Item do |> put_assoc(:tags, Tag.parse_and_create_tags(attrs)) end - def changeset_with_lists(item, list_ids) do - lists = Repo.all(from l in L, where: l.id in ^list_ids) - - item - |> change() - |> put_assoc(:lists, lists) - end - @doc """ Creates an `item`. @@ -80,7 +69,6 @@ defmodule App.Item do Item |> Repo.get!(id) |> Repo.preload(tags: from(t in Tag, order_by: t.text)) - |> Repo.preload(lists: from(l in L, order_by: l.name)) end @doc """ @@ -104,7 +92,6 @@ defmodule App.Item do |> where(person_id: ^person_id) |> Repo.all() |> Repo.preload(tags: from(t in Tag, order_by: t.text)) - |> Repo.preload(lists: from(l in L, order_by: l.name)) end @doc """ @@ -134,12 +121,6 @@ defmodule App.Item do |> Repo.update() end - def update_item_with_lists(%Item{} = item, list_ids) do - item - |> Item.changeset_with_lists(list_ids) - |> Repo.update() - end - def delete_item(id) do get_item!(id) |> Item.changeset(%{status: 6}) @@ -182,9 +163,6 @@ defmodule App.Item do |> Enum.map(fn t -> Map.put(t, :tags, items_tags[t.id].tags) end) - |> Enum.map(fn t -> - Map.put(t, :lists, items_tags[t.id].lists) - end) end @doc """ diff --git a/lib/app/list.ex b/lib/app/list.ex_BAK similarity index 100% rename from lib/app/list.ex rename to lib/app/list.ex_BAK diff --git a/lib/app/person.ex b/lib/app/person.ex_BAK similarity index 100% rename from lib/app/person.ex rename to lib/app/person.ex_BAK diff --git a/lib/app/tag.ex b/lib/app/tag.ex index a331a31a..1b9f69a1 100644 --- a/lib/app/tag.ex +++ b/lib/app/tag.ex @@ -2,14 +2,14 @@ defmodule App.Tag do use Ecto.Schema import Ecto.Changeset import Ecto.Query - alias App.{Item, ItemTag, Person, Repo} + alias App.{Item, ItemTag, Repo} alias __MODULE__ schema "tags" do field :text, :string field :color, :string + field :person_id, :integer - belongs_to :people, Person, references: :person_id, foreign_key: :person_id many_to_many(:items, Item, join_through: ItemTag) timestamps() end diff --git a/lib/app_web/controllers/item_controller.ex b/lib/app_web/controllers/item_controller.ex_BAK similarity index 100% rename from lib/app_web/controllers/item_controller.ex rename to lib/app_web/controllers/item_controller.ex_BAK diff --git a/lib/app_web/controllers/list_controller.ex b/lib/app_web/controllers/list_controller.ex_BAK similarity index 100% rename from lib/app_web/controllers/list_controller.ex rename to lib/app_web/controllers/list_controller.ex_BAK diff --git a/lib/app_web/controllers/profile_controller.ex b/lib/app_web/controllers/profile_controller.ex_BAK similarity index 100% rename from lib/app_web/controllers/profile_controller.ex rename to lib/app_web/controllers/profile_controller.ex_BAK diff --git a/lib/app_web/live/app_live.html.heex b/lib/app_web/live/app_live.html.heex index b44f39e5..0a20672f 100644 --- a/lib/app_web/live/app_live.html.heex +++ b/lib/app_web/live/app_live.html.heex @@ -259,18 +259,7 @@ <%= live_patch tag.text, to: Routes.live_path(@socket, AppWeb.AppLive, %{filter_by: @filter, filter_by_tag: tag.text}), style: "background-color:#{tag.color}", class: " text-white font-bold py-1 px-2 rounded-full" %> <% end %>
- -
- <%= for list <- item.lists do %> - <.badge color="info" label={list.name} /> - <% end %> -
- <.a to={Routes.item_path(@socket, :edit, item.id)} class="" label="Edit lists" /> -
-
-
-
<% end %> <% end %> diff --git a/lib/app_web/router.ex b/lib/app_web/router.ex index 5b925068..768f97e1 100644 --- a/lib/app_web/router.ex +++ b/lib/app_web/router.ex @@ -1,6 +1,5 @@ defmodule AppWeb.Router do use AppWeb, :router - alias App.Person pipeline :browser do plug :accepts, ["html"] @@ -15,53 +14,37 @@ defmodule AppWeb.Router do scope "/", AppWeb do pipe_through :browser get "/init", InitController, :index + get "/login", AuthController, :login end - pipeline :authOptional, do: plug(AuthPlugOptional) - pipeline :verify_loggedin, do: plug(:loggedin) - pipeline :check_profile_name, do: plug(:profile_name) + pipeline :authOptional do + plug(AuthPlugOptional) + end scope "/", AppWeb do - pipe_through [:browser, :authOptional, :verify_loggedin] - - resources "/profile", ProfileController, - only: [:edit, :update], - param: "person_id" - - pipe_through [:check_profile_name] + pipe_through [:browser, :authOptional] live "/", AppLive - resources "/tags", TagController, except: [:show] - resources "/lists", ListController, except: [:show] - get "/login", AuthController, :login get "/logout", AuthController, :logout - - # edit item lists - resources "/items", ItemController, only: [:edit, :update] + # live "/stats", StatsLive + resources "/tags", TagController, except: [:show] + # resources "/lists", ListController, except: [:show] end - # assign to conn the loggedin value used in templates - defp loggedin(conn, _opts) do - if not is_nil(conn.assigns[:jwt]) do - assign(conn, :loggedin, true) - else - assign(conn, :loggedin, false) - end - end + # pipeline :api do + # plug :accepts, ["json"] + # plug :fetch_session + # end - # Redirect to edit profile to force user to - # add name to their profile for sharing items feature - defp profile_name(conn, _opts) do - person_id = conn.assigns[:person][:id] || 0 + # scope "/api", API, as: :api do + # pipe_through [:api, :authOptional] - person = Person.get_or_insert(person_id) + # resources "/items", Item, only: [:create, :update, :show] - if is_nil(person.name) do - conn - |> put_flash(:info, "Add a name to your profile to allow sharing items") - |> redirect(to: "/profile/#{person.person_id}/edit") - |> halt() - else - conn - end - end + # resources "/items/:item_id/timers", Timer, + # only: [:create, :update, :show, :index] + + # put "/timers/:id", Timer, :stop + + # resources "/tags", Tag, only: [:create, :update, :show] + # end end diff --git a/lib/app_web/templates/item/edit.html.heex b/lib/app_web/templates/item/edit.html.heex_BAK similarity index 100% rename from lib/app_web/templates/item/edit.html.heex rename to lib/app_web/templates/item/edit.html.heex_BAK diff --git a/lib/app_web/templates/layout/root.html.heex b/lib/app_web/templates/layout/root.html.heex index ba2551c8..38b31732 100644 --- a/lib/app_web/templates/layout/root.html.heex +++ b/lib/app_web/templates/layout/root.html.heex @@ -16,9 +16,7 @@
<%= if @loggedin do %> - avatar image - <% else %>

Hi Friend!

<% end %> diff --git a/lib/app_web/templates/profile/edit.html.heex b/lib/app_web/templates/profile/edit.html.heex deleted file mode 100644 index d06b7b6a..00000000 --- a/lib/app_web/templates/profile/edit.html.heex +++ /dev/null @@ -1,19 +0,0 @@ -<.container> -<.h2 class="text-center mt-3">Edit Profile -<.p>Please make sure a name is defined as it is used to share items -<.form :let={f} for={@changeset} action={Routes.profile_path(@conn, :update, @profile)} method="patch" class="py-3"> - <%= if @changeset.action do %> -
-

Oops, something went wrong! Please check the errors below.

-
- <% end %> - <.form_field - type="text_input" - form={f} - field={:name} - /> - - <.button type="submit" label="Save" /> - - - diff --git a/lib/app_web/views/item_view.ex b/lib/app_web/views/item_view.ex_BAK similarity index 100% rename from lib/app_web/views/item_view.ex rename to lib/app_web/views/item_view.ex_BAK diff --git a/lib/app_web/views/profile_view.ex b/lib/app_web/views/profile_view.ex deleted file mode 100644 index f42fb05e..00000000 --- a/lib/app_web/views/profile_view.ex +++ /dev/null @@ -1,3 +0,0 @@ -defmodule AppWeb.ProfileView do - use AppWeb, :view -end diff --git a/lib/list_item.ex b/lib/list_item.ex_BAK similarity index 100% rename from lib/list_item.ex rename to lib/list_item.ex_BAK diff --git a/mix.exs b/mix.exs index 9ea3dc9b..f0bbf0c0 100644 --- a/mix.exs +++ b/mix.exs @@ -5,9 +5,8 @@ defmodule App.MixProject do [ app: :app, version: "1.0.0", - elixir: "~> 1.12", + elixir: "~> 1.14", elixirc_paths: elixirc_paths(Mix.env()), - compilers: Mix.compilers(), start_permanent: Mix.env() == :prod, aliases: aliases(), deps: deps(), @@ -41,13 +40,15 @@ defmodule App.MixProject do # Type `mix help deps` for examples and options. defp deps do [ - {:phoenix, "~> 1.6.10"}, + # Phoenix deps: + {:phoenix, "~> 1.7.0"}, {:phoenix_ecto, "~> 4.4"}, {:ecto_sql, "~> 3.6"}, {:postgrex, ">= 0.0.0"}, {:phoenix_html, "~> 3.0"}, {:phoenix_live_reload, "~> 1.2", only: :dev}, - {:phoenix_live_view, "~> 0.18.0"}, + {:phoenix_live_view, "~> 0.18.3"}, + {:phoenix_view, "~> 2.0"}, {:floki, ">= 0.30.0", only: :test}, {:esbuild, "~> 0.4", runtime: Mix.env() == :dev}, {:telemetry_metrics, "~> 0.6"}, @@ -55,30 +56,50 @@ defmodule App.MixProject do {:jason, "~> 1.2"}, {:plug_cowboy, "~> 2.5"}, - # Check/get Environment Variables: https://github.com/dwyl/envar - {:envar, "~> 1.0.8"}, # Auth with ONE Environment Variableβ„’: github.com/dwyl/auth_plug - {:auth_plug, "~> 1.4.14"}, - # Easily Encrypt Senstive Data: github.com/dwyl/fields - {:fields, "~> 2.9.1"}, + {:auth_plug, "~> 1.5.1"}, + + # Check/get Environment Variables: github.com/dwyl/envar + {:envar, "~> 1.1.0", override: true}, + + # Universally Unique Deterministic Content IDs: github.com/dwyl/cid + {:excid, "~> 1.0.1"}, + + # Easily Encrypt Sensitive Data: github.com/dwyl/fields + {:fields, "~> 2.10.3"}, + + # Database changes tracking: + # github.com/dwyl/phoenix-papertrail-demo + {:paper_trail, "~> 1.0.0"}, + + # Time string parsing: github.com/bitwalker/timex + {:timex, "~> 3.7"}, + # Useful functions: github.com/dwyl/useful - {:useful, "~> 1.0.8", override: true}, - # https://github.com/dwyl/useful/issues/17 + {:useful, "~> 1.12.1", override: true}, + + # See: github.com/dwyl/useful/issues/17 {:atomic_map, "~> 0.9.3"}, + + # Decimal precision: github.com/ericmj/decimal + {:decimal, "~> 2.0"}, + # Statuses: github.com/dwyl/statuses {:statuses, "~> 1.1.1"}, # create docs on localhost by running "mix docs" - {:ex_doc, "~> 0.28.4", only: :dev, runtime: false}, - # Track test coverage - {:excoveralls, "~> 0.14.5", only: [:test, :dev]}, - # git pre-commit hook runs tests before allowing commits - {:pre_commit, "~> 0.3.4"}, - {:credo, "~> 1.6.4", only: [:dev, :test], runtime: false}, + {:ex_doc, "~> 0.29", only: :dev, runtime: false}, + # Track test coverage: github.com/parroty/excoveralls + {:excoveralls, "~> 0.15", only: [:test, :dev]}, + + # Git pre-commit hook runs tests before allowing commits: + # github.com/dwyl/elixir-pre-commit + {:pre_commit, "~> 0.3.4", only: :dev}, + {:credo, "~> 1.7.0", only: [:dev, :test], runtime: false}, # Ref: github.com/dwyl/learn-tailwind {:tailwind, "~> 0.1", runtime: Mix.env() == :dev}, - {:petal_components, "~> 0.18"} + {:petal_components, "~> 1.0"} ] end diff --git a/mix.lock b/mix.lock index 5c3f6e68..d1dd34d3 100644 --- a/mix.lock +++ b/mix.lock @@ -1,67 +1,77 @@ %{ "argon2_elixir": {:hex, :argon2_elixir, "3.0.0", "fd4405f593e77b525a5c667282172dd32772d7c4fa58cdecdaae79d2713b6c5f", [:make, :mix], [{:comeonin, "~> 5.3", [hex: :comeonin, repo: "hexpm", optional: false]}, {:elixir_make, "~> 0.6", [hex: :elixir_make, repo: "hexpm", optional: false]}], "hexpm", "8b753b270af557d51ba13fcdebc0f0ab27a2a6792df72fd5a6cf9cfaffcedc57"}, "atomic_map": {:hex, :atomic_map, "0.9.3", "3c7f1302e0590164732d08ca999708efbb2cd768abf2911cf140280ce2dc499d", [:mix], [], "hexpm", "c237babf301bd2435bd85b96cffc973022b4cbb7721537059ee0dd3bb74938d2"}, - "auth_plug": {:hex, :auth_plug, "1.4.14", "bc201bfa26778f3fa2651b0b20e78a500602c3330bc6cd457ba8cb2decd3ca55", [:mix], [{:envar, "~> 1.0.8", [hex: :envar, repo: "hexpm", optional: false]}, {:httpoison, "~> 1.8.0", [hex: :httpoison, repo: "hexpm", optional: false]}, {:jason, "~> 1.3", [hex: :jason, repo: "hexpm", optional: false]}, {:joken, "~> 2.5.0", [hex: :joken, repo: "hexpm", optional: false]}, {:plug, "~> 1.13.4", [hex: :plug, repo: "hexpm", optional: false]}, {:useful, "~> 1.0.8", [hex: :useful, repo: "hexpm", optional: false]}], "hexpm", "60cc03b87327b2a63d38eefb503948247564b5fb99e5b6af93fc48842855dda1"}, + "auth_plug": {:hex, :auth_plug, "1.5.2", "3e330cd9c231a29dc9e0524ac4bf81444168dc3ab09bfe789048d88a06e4908a", [:mix], [{:envar, "~> 1.1.0", [hex: :envar, repo: "hexpm", optional: false]}, {:httpoison, "~> 1.8.0", [hex: :httpoison, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}, {:joken, "~> 2.5", [hex: :joken, repo: "hexpm", optional: false]}, {:plug, "~> 1.10", [hex: :plug, repo: "hexpm", optional: false]}, {:useful, "~> 1.10.0", [hex: :useful, repo: "hexpm", optional: false]}], "hexpm", "326bedcedce560de37d3606bb85f7124b8e29c7ad892d4d8fe04a3ba14d18b06"}, + "b58": {:hex, :b58, "1.0.3", "d300d6ae5a3de956a54b9e8220e924e4fee1a349de983df2340fe61e0e464202", [:mix], [], "hexpm", "af62a98a8661fd89978cf3a3a4b5b2ebe82209de6ac6164f0b112e36af72fc59"}, "bunt": {:hex, :bunt, "0.2.1", "e2d4792f7bc0ced7583ab54922808919518d0e57ee162901a16a1b6664ef3b14", [:mix], [], "hexpm", "a330bfb4245239787b15005e66ae6845c9cd524a288f0d141c148b02603777a5"}, - "castore": {:hex, :castore, "0.1.18", "deb5b9ab02400561b6f5708f3e7660fc35ca2d51bfc6a940d2f513f89c2975fc", [:mix], [], "hexpm", "61bbaf6452b782ef80b33cdb45701afbcf0a918a45ebe7e73f1130d661e66a06"}, - "certifi": {:hex, :certifi, "2.9.0", "6f2a475689dd47f19fb74334859d460a2dc4e3252a3324bd2111b8f0429e7e21", [:rebar3], [], "hexpm", "266da46bdb06d6c6d35fde799bcb28d36d985d424ad7c08b5bb48f5b5cdd4641"}, - "comeonin": {:hex, :comeonin, "5.3.3", "2c564dac95a35650e9b6acfe6d2952083d8a08e4a89b93a481acb552b325892e", [:mix], [], "hexpm", "3e38c9c2cb080828116597ca8807bb482618a315bfafd98c90bc22a821cc84df"}, - "connection": {:hex, :connection, "1.1.0", "ff2a49c4b75b6fb3e674bfc5536451607270aac754ffd1bdfe175abe4a6d7a68", [:mix], [], "hexpm", "722c1eb0a418fbe91ba7bd59a47e28008a189d47e37e0e7bb85585a016b2869c"}, - "cowboy": {:hex, :cowboy, "2.9.0", "865dd8b6607e14cf03282e10e934023a1bd8be6f6bacf921a7e2a96d800cd452", [:make, :rebar3], [{:cowlib, "2.11.0", [hex: :cowlib, repo: "hexpm", optional: false]}, {:ranch, "1.8.0", [hex: :ranch, repo: "hexpm", optional: false]}], "hexpm", "2c729f934b4e1aa149aff882f57c6372c15399a20d54f65c8d67bef583021bde"}, + "castore": {:hex, :castore, "1.0.3", "7130ba6d24c8424014194676d608cb989f62ef8039efd50ff4b3f33286d06db8", [:mix], [], "hexpm", "680ab01ef5d15b161ed6a95449fac5c6b8f60055677a8e79acf01b27baa4390b"}, + "certifi": {:hex, :certifi, "2.12.0", "2d1cca2ec95f59643862af91f001478c9863c2ac9cb6e2f89780bfd8de987329", [:rebar3], [], "hexpm", "ee68d85df22e554040cdb4be100f33873ac6051387baf6a8f6ce82272340ff1c"}, + "combine": {:hex, :combine, "0.10.0", "eff8224eeb56498a2af13011d142c5e7997a80c8f5b97c499f84c841032e429f", [:mix], [], "hexpm", "1b1dbc1790073076580d0d1d64e42eae2366583e7aecd455d1215b0d16f2451b"}, + "comeonin": {:hex, :comeonin, "5.4.0", "246a56ca3f41d404380fc6465650ddaa532c7f98be4bda1b4656b3a37cc13abe", [:mix], [], "hexpm", "796393a9e50d01999d56b7b8420ab0481a7538d0caf80919da493b4a6e51faf1"}, + "cowboy": {:hex, :cowboy, "2.10.0", "ff9ffeff91dae4ae270dd975642997afe2a1179d94b1887863e43f681a203e26", [:make, :rebar3], [{:cowlib, "2.12.1", [hex: :cowlib, repo: "hexpm", optional: false]}, {:ranch, "1.8.0", [hex: :ranch, repo: "hexpm", optional: false]}], "hexpm", "3afdccb7183cc6f143cb14d3cf51fa00e53db9ec80cdcd525482f5e99bc41d6b"}, "cowboy_telemetry": {:hex, :cowboy_telemetry, "0.4.0", "f239f68b588efa7707abce16a84d0d2acf3a0f50571f8bb7f56a15865aae820c", [:rebar3], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "7d98bac1ee4565d31b62d59f8823dfd8356a169e7fcbb83831b8a5397404c9de"}, - "cowlib": {:hex, :cowlib, "2.11.0", "0b9ff9c346629256c42ebe1eeb769a83c6cb771a6ee5960bd110ab0b9b872063", [:make, :rebar3], [], "hexpm", "2b3e9da0b21c4565751a6d4901c20d1b4cc25cbb7fd50d91d2ab6dd287bc86a9"}, - "credo": {:hex, :credo, "1.6.7", "323f5734350fd23a456f2688b9430e7d517afb313fbd38671b8a4449798a7854", [:mix], [{:bunt, "~> 0.2.1", [hex: :bunt, repo: "hexpm", optional: false]}, {:file_system, "~> 0.2.8", [hex: :file_system, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "41e110bfb007f7eda7f897c10bf019ceab9a0b269ce79f015d54b0dcf4fc7dd3"}, - "db_connection": {:hex, :db_connection, "2.4.2", "f92e79aff2375299a16bcb069a14ee8615c3414863a6fef93156aee8e86c2ff3", [:mix], [{:connection, "~> 1.0", [hex: :connection, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "4fe53ca91b99f55ea249693a0229356a08f4d1a7931d8ffa79289b145fe83668"}, - "decimal": {:hex, :decimal, "2.0.0", "a78296e617b0f5dd4c6caf57c714431347912ffb1d0842e998e9792b5642d697", [:mix], [], "hexpm", "34666e9c55dea81013e77d9d87370fe6cb6291d1ef32f46a1600230b1d44f577"}, - "earmark_parser": {:hex, :earmark_parser, "1.4.28", "0bf6546eb7cd6185ae086cbc5d20cd6dbb4b428aad14c02c49f7b554484b4586", [:mix], [], "hexpm", "501cef12286a3231dc80c81352a9453decf9586977f917a96e619293132743fb"}, - "ecto": {:hex, :ecto, "3.9.0", "7c74fc0d950a700eb7019057ff32d047ed7f19b57c1b2ca260cf0e565829101d", [:mix], [{:decimal, "~> 1.6 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "fed5ebc5831378b916afd0b5852a0c5bb3e7390665cc2b0ec8ab0c712495b73d"}, - "ecto_sql": {:hex, :ecto_sql, "3.9.0", "2bb21210a2a13317e098a420a8c1cc58b0c3421ab8e3acfa96417dab7817918c", [:mix], [{:db_connection, "~> 2.5 or ~> 2.4.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:ecto, "~> 3.9.0", [hex: :ecto, repo: "hexpm", optional: false]}, {:myxql, "~> 0.6.0", [hex: :myxql, repo: "hexpm", optional: true]}, {:postgrex, "~> 0.16.0 or ~> 1.0", [hex: :postgrex, repo: "hexpm", optional: true]}, {:tds, "~> 2.1.1 or ~> 2.2", [hex: :tds, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.0 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "a8f3f720073b8b1ac4c978be25fa7960ed7fd44997420c304a4a2e200b596453"}, - "elixir_make": {:hex, :elixir_make, "0.6.3", "bc07d53221216838d79e03a8019d0839786703129599e9619f4ab74c8c096eac", [:mix], [], "hexpm", "f5cbd651c5678bcaabdbb7857658ee106b12509cd976c2c2fca99688e1daf716"}, - "envar": {:hex, :envar, "1.0.9", "b51976b00035efd254c3f51ee7f3cf2e22f91350ef104da393d1d71286eb4fdc", [:mix], [], "hexpm", "bfc3a73f97910c744e0d9e53722ad2d1f73bbb392d2dd1cac63e0af27776fde3"}, - "esbuild": {:hex, :esbuild, "0.5.0", "d5bb08ff049d7880ee3609ed5c4b864bd2f46445ea40b16b4acead724fb4c4a3", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "f183a0b332d963c4cfaf585477695ea59eef9a6f2204fdd0efa00e099694ffe5"}, - "ex_doc": {:hex, :ex_doc, "0.28.5", "3e52a6d2130ce74d096859e477b97080c156d0926701c13870a4e1f752363279", [:mix], [{:earmark_parser, "~> 1.4.19", [hex: :earmark_parser, repo: "hexpm", optional: false]}, {:makeup_elixir, "~> 0.14", [hex: :makeup_elixir, repo: "hexpm", optional: false]}, {:makeup_erlang, "~> 0.1", [hex: :makeup_erlang, repo: "hexpm", optional: false]}], "hexpm", "d2c4b07133113e9aa3e9ba27efb9088ba900e9e51caa383919676afdf09ab181"}, - "excoveralls": {:hex, :excoveralls, "0.14.6", "610e921e25b180a8538229ef547957f7e04bd3d3e9a55c7c5b7d24354abbba70", [:mix], [{:hackney, "~> 1.16", [hex: :hackney, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "0eceddaa9785cfcefbf3cd37812705f9d8ad34a758e513bb975b081dce4eb11e"}, - "fields": {:hex, :fields, "2.9.1", "af8ce8e90e0e33df3ca173adec0839f34778c229211e35f5258971fe57ee40ff", [:mix], [{:argon2_elixir, "~> 3.0.0", [hex: :argon2_elixir, repo: "hexpm", optional: false]}, {:ecto, "~> 3.8", [hex: :ecto, repo: "hexpm", optional: false]}, {:envar, "~> 1.0.8", [hex: :envar, repo: "hexpm", optional: false]}, {:html_sanitize_ex, "~> 1.4.2", [hex: :html_sanitize_ex, repo: "hexpm", optional: false]}], "hexpm", "caa388160938fc5180c32bec66514c87ebae3a4029c9371f42e9ccb0093ad7f5"}, + "cowlib": {:hex, :cowlib, "2.12.1", "a9fa9a625f1d2025fe6b462cb865881329b5caff8f1854d1cbc9f9533f00e1e1", [:make, :rebar3], [], "hexpm", "163b73f6367a7341b33c794c4e88e7dbfe6498ac42dcd69ef44c5bc5507c8db0"}, + "credo": {:hex, :credo, "1.7.0", "6119bee47272e85995598ee04f2ebbed3e947678dee048d10b5feca139435f75", [:mix], [{:bunt, "~> 0.2.1", [hex: :bunt, repo: "hexpm", optional: false]}, {:file_system, "~> 0.2.8", [hex: :file_system, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "6839fcf63d1f0d1c0f450abc8564a57c43d644077ab96f2934563e68b8a769d7"}, + "db_connection": {:hex, :db_connection, "2.5.0", "bb6d4f30d35ded97b29fe80d8bd6f928a1912ca1ff110831edcd238a1973652c", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "c92d5ba26cd69ead1ff7582dbb860adeedfff39774105a4f1c92cbb654b55aa2"}, + "decimal": {:hex, :decimal, "2.1.1", "5611dca5d4b2c3dd497dec8f68751f1f1a54755e8ed2a966c2633cf885973ad6", [:mix], [], "hexpm", "53cfe5f497ed0e7771ae1a475575603d77425099ba5faef9394932b35020ffcc"}, + "earmark_parser": {:hex, :earmark_parser, "1.4.33", "3c3fd9673bb5dcc9edc28dd90f50c87ce506d1f71b70e3de69aa8154bc695d44", [:mix], [], "hexpm", "2d526833729b59b9fdb85785078697c72ac5e5066350663e5be6a1182da61b8f"}, + "ecto": {:hex, :ecto, "3.10.3", "eb2ae2eecd210b4eb8bece1217b297ad4ff824b4384c0e3fdd28aaf96edd6135", [:mix], [{:decimal, "~> 1.6 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "44bec74e2364d491d70f7e42cd0d690922659d329f6465e89feb8a34e8cd3433"}, + "ecto_sql": {:hex, :ecto_sql, "3.10.2", "6b98b46534b5c2f8b8b5f03f126e75e2a73c64f3c071149d32987a5378b0fdbd", [:mix], [{:db_connection, "~> 2.4.1 or ~> 2.5", [hex: :db_connection, repo: "hexpm", optional: false]}, {:ecto, "~> 3.10.0", [hex: :ecto, repo: "hexpm", optional: false]}, {:myxql, "~> 0.6.0", [hex: :myxql, repo: "hexpm", optional: true]}, {:postgrex, "~> 0.16.0 or ~> 0.17.0 or ~> 1.0", [hex: :postgrex, repo: "hexpm", optional: true]}, {:tds, "~> 2.1.1 or ~> 2.2", [hex: :tds, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.0 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "68c018debca57cb9235e3889affdaec7a10616a4e3a80c99fa1d01fdafaa9007"}, + "elixir_make": {:hex, :elixir_make, "0.7.7", "7128c60c2476019ed978210c245badf08b03dbec4f24d05790ef791da11aa17c", [:mix], [{:castore, "~> 0.1 or ~> 1.0", [hex: :castore, repo: "hexpm", optional: true]}], "hexpm", "5bc19fff950fad52bbe5f211b12db9ec82c6b34a9647da0c2224b8b8464c7e6c"}, + "envar": {:hex, :envar, "1.1.0", "105bcac5a03800a1eb21e2c7e229edc687359b0cc184150ec1380db5928c115c", [:mix], [], "hexpm", "97028ab4a040a5c19e613fdf46a41cf51c6e848d99077e525b338e21d2993320"}, + "esbuild": {:hex, :esbuild, "0.7.1", "fa0947e8c3c3c2f86c9bf7e791a0a385007ccd42b86885e8e893bdb6631f5169", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "66661cdf70b1378ee4dc16573fcee67750b59761b2605a0207c267ab9d19f13c"}, + "ex_doc": {:hex, :ex_doc, "0.30.6", "5f8b54854b240a2b55c9734c4b1d0dd7bdd41f71a095d42a70445c03cf05a281", [:mix], [{:earmark_parser, "~> 1.4.31", [hex: :earmark_parser, repo: "hexpm", optional: false]}, {:makeup_elixir, "~> 0.14", [hex: :makeup_elixir, repo: "hexpm", optional: false]}, {:makeup_erlang, "~> 0.1", [hex: :makeup_erlang, repo: "hexpm", optional: false]}], "hexpm", "bd48f2ddacf4e482c727f9293d9498e0881597eae6ddc3d9562bd7923375109f"}, + "ex_multihash": {:hex, :ex_multihash, "2.0.0", "7fb36f842a2ec1c6bbba550f28fcd16d3c62981781b9466c9c1975c43d7db43c", [:mix], [], "hexpm", "66a08a86a1ba00d95736c595d7975696e5691308cdf7770c50b0f84a2a1172b0"}, + "excid": {:hex, :excid, "1.0.1", "cc47c05ea64c1e5d93a90fb232d8d94d115dd65718547fa73145b6b8ab80f00d", [:mix], [{:b58, "~> 1.0.3", [hex: :b58, repo: "hexpm", optional: false]}, {:ex_multihash, "~> 2.0", [hex: :ex_multihash, repo: "hexpm", optional: false]}, {:jason, "~> 1.1", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "63c1814d000c604ffaefa79bb1dadb9c23365f15ce5dc19275441c5e859a90e5"}, + "excoveralls": {:hex, :excoveralls, "0.17.1", "83fa7906ef23aa7fc8ad7ee469c357a63b1b3d55dd701ff5b9ce1f72442b2874", [:mix], [{:castore, "~> 1.0", [hex: :castore, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "95bc6fda953e84c60f14da4a198880336205464e75383ec0f570180567985ae0"}, + "expo": {:hex, :expo, "0.4.1", "1c61d18a5df197dfda38861673d392e642649a9cef7694d2f97a587b2cfb319b", [:mix], [], "hexpm", "2ff7ba7a798c8c543c12550fa0e2cbc81b95d4974c65855d8d15ba7b37a1ce47"}, + "fields": {:hex, :fields, "2.10.3", "2683d8fdfd582869b459c88c693c4e15e2247961869719e03213286764b82093", [:mix], [{:argon2_elixir, "~> 3.0.0", [hex: :argon2_elixir, repo: "hexpm", optional: false]}, {:ecto, "~> 3.3", [hex: :ecto, repo: "hexpm", optional: false]}, {:envar, "~> 1.0.8", [hex: :envar, repo: "hexpm", optional: false]}, {:html_sanitize_ex, "~> 1.4.2", [hex: :html_sanitize_ex, repo: "hexpm", optional: false]}], "hexpm", "d68567e175fb6d3be04cdf795fc6ed94973c66706ca3f87e2ec6251159b4b965"}, "file_system": {:hex, :file_system, "0.2.10", "fb082005a9cd1711c05b5248710f8826b02d7d1784e7c3451f9c1231d4fc162d", [:mix], [], "hexpm", "41195edbfb562a593726eda3b3e8b103a309b733ad25f3d642ba49696bf715dc"}, - "floki": {:hex, :floki, "0.33.1", "f20f1eb471e726342b45ccb68edb9486729e7df94da403936ea94a794f072781", [:mix], [{:html_entities, "~> 0.5.0", [hex: :html_entities, repo: "hexpm", optional: false]}], "hexpm", "461035fd125f13fdf30f243c85a0b1e50afbec876cbf1ceefe6fddd2e6d712c6"}, - "hackney": {:hex, :hackney, "1.18.1", "f48bf88f521f2a229fc7bae88cf4f85adc9cd9bcf23b5dc8eb6a1788c662c4f6", [:rebar3], [{:certifi, "~>2.9.0", [hex: :certifi, repo: "hexpm", optional: false]}, {:idna, "~>6.1.0", [hex: :idna, repo: "hexpm", optional: false]}, {:metrics, "~>1.0.0", [hex: :metrics, repo: "hexpm", optional: false]}, {:mimerl, "~>1.1", [hex: :mimerl, repo: "hexpm", optional: false]}, {:parse_trans, "3.3.1", [hex: :parse_trans, repo: "hexpm", optional: false]}, {:ssl_verify_fun, "~>1.1.0", [hex: :ssl_verify_fun, repo: "hexpm", optional: false]}, {:unicode_util_compat, "~>0.7.0", [hex: :unicode_util_compat, repo: "hexpm", optional: false]}], "hexpm", "a4ecdaff44297e9b5894ae499e9a070ea1888c84afdd1fd9b7b2bc384950128e"}, - "heroicons": {:hex, :heroicons, "0.5.0", "4ae690b2808246cd67f8a0e1a8b81f22ed35ec4ba25039c87f02e6caa5e07142", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.18.0", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}], "hexpm", "cc3633d161c096f98a1fa6ca2f1361cd75da6804457c33116c8388c16d5ca141"}, - "html_entities": {:hex, :html_entities, "0.5.2", "9e47e70598da7de2a9ff6af8758399251db6dbb7eebe2b013f2bbd2515895c3c", [:mix], [], "hexpm", "c53ba390403485615623b9531e97696f076ed415e8d8058b1dbaa28181f4fdcc"}, - "html_sanitize_ex": {:hex, :html_sanitize_ex, "1.4.2", "c479398b6de798c03eb5d04a0a9a9159d73508f83f6590a00b8eacba3619cf4c", [:mix], [{:mochiweb, "~> 2.15", [hex: :mochiweb, repo: "hexpm", optional: false]}], "hexpm", "aef6c28585d06a9109ad591507e508854c5559561f950bbaea773900dd369b0e"}, + "floki": {:hex, :floki, "0.34.3", "5e2dcaec5d7c228ce5b1d3501502e308b2d79eb655e4191751a1fe491c37feac", [:mix], [], "hexpm", "9577440eea5b97924b4bf3c7ea55f7b8b6dce589f9b28b096cc294a8dc342341"}, + "gettext": {:hex, :gettext, "0.23.1", "821e619a240e6000db2fc16a574ef68b3bd7fe0167ccc264a81563cc93e67a31", [:mix], [{:expo, "~> 0.4.0", [hex: :expo, repo: "hexpm", optional: false]}], "hexpm", "19d744a36b809d810d610b57c27b934425859d158ebd56561bc41f7eeb8795db"}, + "hackney": {:hex, :hackney, "1.18.2", "d7ff544ddae5e1cb49e9cf7fa4e356d7f41b283989a1c304bfc47a8cc1cf966f", [:rebar3], [{:certifi, "~> 2.12.0", [hex: :certifi, repo: "hexpm", optional: false]}, {:idna, "~> 6.1.0", [hex: :idna, repo: "hexpm", optional: false]}, {:metrics, "~> 1.0.0", [hex: :metrics, repo: "hexpm", optional: false]}, {:mimerl, "~> 1.1", [hex: :mimerl, repo: "hexpm", optional: false]}, {:parse_trans, "3.4.1", [hex: :parse_trans, repo: "hexpm", optional: false]}, {:ssl_verify_fun, "~> 1.1.0", [hex: :ssl_verify_fun, repo: "hexpm", optional: false]}, {:unicode_util_compat, "~> 0.7.0", [hex: :unicode_util_compat, repo: "hexpm", optional: false]}], "hexpm", "af94d5c9f97857db257090a4a10e5426ecb6f4918aa5cc666798566ae14b65fd"}, + "heroicons": {:hex, :heroicons, "0.5.3", "ee8ae8335303df3b18f2cc07f46e1cb6e761ba4cf2c901623fbe9a28c0bc51dd", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:phoenix_live_view, ">= 0.18.2", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}], "hexpm", "a210037e8a09ac17e2a0a0779d729e89c821c944434c3baa7edfc1f5b32f3502"}, + "html_sanitize_ex": {:hex, :html_sanitize_ex, "1.4.3", "67b3d9fa8691b727317e0cc96b9b3093be00ee45419ffb221cdeee88e75d1360", [:mix], [{:mochiweb, "~> 2.15 or ~> 3.1", [hex: :mochiweb, repo: "hexpm", optional: false]}], "hexpm", "87748d3c4afe949c7c6eb7150c958c2bcba43fc5b2a02686af30e636b74bccb7"}, "httpoison": {:hex, :httpoison, "1.8.2", "9eb9c63ae289296a544842ef816a85d881d4a31f518a0fec089aaa744beae290", [:mix], [{:hackney, "~> 1.17", [hex: :hackney, repo: "hexpm", optional: false]}], "hexpm", "2bb350d26972e30c96e2ca74a1aaf8293d61d0742ff17f01e0279fef11599921"}, - "idna": {:hex, :idna, "6.1.1", "8a63070e9f7d0c62eb9d9fcb360a7de382448200fbbd1b106cc96d3d8099df8d", [:rebar3], [{:unicode_util_compat, "~>0.7.0", [hex: :unicode_util_compat, repo: "hexpm", optional: false]}], "hexpm", "92376eb7894412ed19ac475e4a86f7b413c1b9fbb5bd16dccd57934157944cea"}, - "jason": {:hex, :jason, "1.4.0", "e855647bc964a44e2f67df589ccf49105ae039d4179db7f6271dfd3843dc27e6", [:mix], [{:decimal, "~> 1.0 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: true]}], "hexpm", "79a3791085b2a0f743ca04cec0f7be26443738779d09302e01318f97bdb82121"}, - "joken": {:hex, :joken, "2.5.0", "09be497d804b8115eb6f07615cef2e60c2a1008fb89dc0aef0d4c4b4609b99aa", [:mix], [{:jose, "~> 1.11.2", [hex: :jose, repo: "hexpm", optional: false]}], "hexpm", "22b25c89617c5ed8ca7b31026340a25ea0f9ca7160f9706b79be9ed81fdf74e7"}, - "jose": {:hex, :jose, "1.11.2", "f4c018ccf4fdce22c71e44d471f15f723cb3efab5d909ab2ba202b5bf35557b3", [:mix, :rebar3], [], "hexpm", "98143fbc48d55f3a18daba82d34fe48959d44538e9697c08f34200fa5f0947d2"}, + "idna": {:hex, :idna, "6.1.1", "8a63070e9f7d0c62eb9d9fcb360a7de382448200fbbd1b106cc96d3d8099df8d", [:rebar3], [{:unicode_util_compat, "~> 0.7.0", [hex: :unicode_util_compat, repo: "hexpm", optional: false]}], "hexpm", "92376eb7894412ed19ac475e4a86f7b413c1b9fbb5bd16dccd57934157944cea"}, + "jason": {:hex, :jason, "1.4.1", "af1504e35f629ddcdd6addb3513c3853991f694921b1b9368b0bd32beb9f1b63", [:mix], [{:decimal, "~> 1.0 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: true]}], "hexpm", "fbb01ecdfd565b56261302f7e1fcc27c4fb8f32d56eab74db621fc154604a7a1"}, + "joken": {:hex, :joken, "2.6.0", "b9dd9b6d52e3e6fcb6c65e151ad38bf4bc286382b5b6f97079c47ade6b1bcc6a", [:mix], [{:jose, "~> 1.11.5", [hex: :jose, repo: "hexpm", optional: false]}], "hexpm", "5a95b05a71cd0b54abd35378aeb1d487a23a52c324fa7efdffc512b655b5aaa7"}, + "jose": {:hex, :jose, "1.11.6", "613fda82552128aa6fb804682e3a616f4bc15565a048dabd05b1ebd5827ed965", [:mix, :rebar3], [], "hexpm", "6275cb75504f9c1e60eeacb771adfeee4905a9e182103aa59b53fed651ff9738"}, "makeup": {:hex, :makeup, "1.1.0", "6b67c8bc2882a6b6a445859952a602afc1a41c2e08379ca057c0f525366fc3ca", [:mix], [{:nimble_parsec, "~> 1.2.2 or ~> 1.3", [hex: :nimble_parsec, repo: "hexpm", optional: false]}], "hexpm", "0a45ed501f4a8897f580eabf99a2e5234ea3e75a4373c8a52824f6e873be57a6"}, - "makeup_elixir": {:hex, :makeup_elixir, "0.16.0", "f8c570a0d33f8039513fbccaf7108c5d750f47d8defd44088371191b76492b0b", [:mix], [{:makeup, "~> 1.0", [hex: :makeup, repo: "hexpm", optional: false]}, {:nimble_parsec, "~> 1.2.3", [hex: :nimble_parsec, repo: "hexpm", optional: false]}], "hexpm", "28b2cbdc13960a46ae9a8858c4bebdec3c9a6d7b4b9e7f4ed1502f8159f338e7"}, - "makeup_erlang": {:hex, :makeup_erlang, "0.1.1", "3fcb7f09eb9d98dc4d208f49cc955a34218fc41ff6b84df7c75b3e6e533cc65f", [:mix], [{:makeup, "~> 1.0", [hex: :makeup, repo: "hexpm", optional: false]}], "hexpm", "174d0809e98a4ef0b3309256cbf97101c6ec01c4ab0b23e926a9e17df2077cbb"}, + "makeup_elixir": {:hex, :makeup_elixir, "0.16.1", "cc9e3ca312f1cfeccc572b37a09980287e243648108384b97ff2b76e505c3555", [:mix], [{:makeup, "~> 1.0", [hex: :makeup, repo: "hexpm", optional: false]}, {:nimble_parsec, "~> 1.2.3 or ~> 1.3", [hex: :nimble_parsec, repo: "hexpm", optional: false]}], "hexpm", "e127a341ad1b209bd80f7bd1620a15693a9908ed780c3b763bccf7d200c767c6"}, + "makeup_erlang": {:hex, :makeup_erlang, "0.1.2", "ad87296a092a46e03b7e9b0be7631ddcf64c790fa68a9ef5323b6cbb36affc72", [:mix], [{:makeup, "~> 1.0", [hex: :makeup, repo: "hexpm", optional: false]}], "hexpm", "f3f5a1ca93ce6e092d92b6d9c049bcda58a3b617a8d888f8e7231c85630e8108"}, "metrics": {:hex, :metrics, "1.0.1", "25f094dea2cda98213cecc3aeff09e940299d950904393b2a29d191c346a8486", [:rebar3], [], "hexpm", "69b09adddc4f74a40716ae54d140f93beb0fb8978d8636eaded0c31b6f099f16"}, - "mime": {:hex, :mime, "2.0.3", "3676436d3d1f7b81b5a2d2bd8405f412c677558c81b1c92be58c00562bb59095", [:mix], [], "hexpm", "27a30bf0db44d25eecba73755acf4068cbfe26a4372f9eb3e4ea3a45956bff6b"}, + "mime": {:hex, :mime, "2.0.5", "dc34c8efd439abe6ae0343edbb8556f4d63f178594894720607772a041b04b02", [:mix], [], "hexpm", "da0d64a365c45bc9935cc5c8a7fc5e49a0e0f9932a761c55d6c52b142780a05c"}, "mimerl": {:hex, :mimerl, "1.2.0", "67e2d3f571088d5cfd3e550c383094b47159f3eee8ffa08e64106cdf5e981be3", [:rebar3], [], "hexpm", "f278585650aa581986264638ebf698f8bb19df297f66ad91b18910dfc6e19323"}, - "mochiweb": {:hex, :mochiweb, "2.22.0", "f104d6747c01a330c38613561977e565b788b9170055c5241ac9dd6e4617cba5", [:rebar3], [], "hexpm", "cbbd1fd315d283c576d1c8a13e0738f6dafb63dc840611249608697502a07655"}, - "nimble_parsec": {:hex, :nimble_parsec, "1.2.3", "244836e6e3f1200c7f30cb56733fd808744eca61fd182f731eac4af635cc6d0b", [:mix], [], "hexpm", "c8d789e39b9131acf7b99291e93dae60ab48ef14a7ee9d58c6964f59efb570b0"}, - "parse_trans": {:hex, :parse_trans, "3.3.1", "16328ab840cc09919bd10dab29e431da3af9e9e7e7e6f0089dd5a2d2820011d8", [:rebar3], [], "hexpm", "07cd9577885f56362d414e8c4c4e6bdf10d43a8767abb92d24cbe8b24c54888b"}, - "petal_components": {:hex, :petal_components, "0.18.3", "5f48103d6fae8a891160c019a24c722b332e6c3dff28ad2c08e613b91a343b5d", [:mix], [{:heroicons, "~> 0.5.0", [hex: :heroicons, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.6", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_ecto, "~> 4.4", [hex: :phoenix_ecto, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.18.0", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}], "hexpm", "a1acd47e478ff1a292d6b46e5e9a36f112657601fa3e36843d67c159de57c0e8"}, - "phoenix": {:hex, :phoenix, "1.6.13", "5b3152907afdb8d3a6cdafb4b149e8aa7aabbf1422fd9f7ef4c2a67ead57d24a", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix_pubsub, "~> 2.0", [hex: :phoenix_pubsub, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 1.0", [hex: :phoenix_view, repo: "hexpm", optional: false]}, {:plug, "~> 1.10", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.2", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:plug_crypto, "~> 1.2", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "13d8806c31176e2066da4df2d7443c144211305c506ed110ad4044335b90171d"}, - "phoenix_ecto": {:hex, :phoenix_ecto, "4.4.0", "0672ed4e4808b3fbed494dded89958e22fb882de47a97634c0b13e7b0b5f7720", [:mix], [{:ecto, "~> 3.3", [hex: :ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 2.14.2 or ~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:plug, "~> 1.9", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "09864e558ed31ee00bd48fcc1d4fc58ae9678c9e81649075431e69dbabb43cc1"}, - "phoenix_html": {:hex, :phoenix_html, "3.2.0", "1c1219d4b6cb22ac72f12f73dc5fad6c7563104d083f711c3fcd8551a1f4ae11", [:mix], [{:plug, "~> 1.5", [hex: :plug, repo: "hexpm", optional: true]}], "hexpm", "36ec97ba56d25c0136ef1992c37957e4246b649d620958a1f9fa86165f8bc54f"}, - "phoenix_live_reload": {:hex, :phoenix_live_reload, "1.3.3", "3a53772a6118d5679bf50fc1670505a290e32a1d195df9e069d8c53ab040c054", [:mix], [{:file_system, "~> 0.2.1 or ~> 0.3", [hex: :file_system, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}], "hexpm", "766796676e5f558dbae5d1bdb066849673e956005e3730dfd5affd7a6da4abac"}, - "phoenix_live_view": {:hex, :phoenix_live_view, "0.18.2", "635cf07de947235deb030cd6b776c71a3b790ab04cebf526aa8c879fe17c7784", [:mix], [{:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.6 or ~> 1.7", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.1", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "da287a77327e996cc166e4c440c3ad5ab33ccdb151b91c793209b39ebbce5b75"}, - "phoenix_pubsub": {:hex, :phoenix_pubsub, "2.1.1", "ba04e489ef03763bf28a17eb2eaddc2c20c6d217e2150a61e3298b0f4c2012b5", [:mix], [], "hexpm", "81367c6d1eea5878ad726be80808eb5a787a23dee699f96e72b1109c57cdd8d9"}, - "phoenix_view": {:hex, :phoenix_view, "1.1.2", "1b82764a065fb41051637872c7bd07ed2fdb6f5c3bd89684d4dca6e10115c95a", [:mix], [{:phoenix_html, "~> 2.14.2 or ~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}], "hexpm", "7ae90ad27b09091266f6adbb61e1d2516a7c3d7062c6789d46a7554ec40f3a56"}, - "plug": {:hex, :plug, "1.13.6", "187beb6b67c6cec50503e940f0434ea4692b19384d47e5fdfd701e93cadb4cc2", [:mix], [{:mime, "~> 1.0 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:plug_crypto, "~> 1.1.1 or ~> 1.2", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.3 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "02b9c6b9955bce92c829f31d6284bf53c591ca63c4fb9ff81dfd0418667a34ff"}, - "plug_cowboy": {:hex, :plug_cowboy, "2.5.2", "62894ccd601cf9597e2c23911ff12798a8a18d237e9739f58a6b04e4988899fe", [:mix], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:cowboy_telemetry, "~> 0.3", [hex: :cowboy_telemetry, repo: "hexpm", optional: false]}, {:plug, "~> 1.7", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "ea6e87f774c8608d60c8d34022a7d073bd7680a0a013f049fc62bf35efea1044"}, - "plug_crypto": {:hex, :plug_crypto, "1.2.3", "8f77d13aeb32bfd9e654cb68f0af517b371fb34c56c9f2b58fe3df1235c1251a", [:mix], [], "hexpm", "b5672099c6ad5c202c45f5a403f21a3411247f164e4a8fab056e5cd8a290f4a2"}, - "postgrex": {:hex, :postgrex, "0.16.5", "fcc4035cc90e23933c5d69a9cd686e329469446ef7abba2cf70f08e2c4b69810", [:mix], [{:connection, "~> 1.1", [hex: :connection, repo: "hexpm", optional: false]}, {:db_connection, "~> 2.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:decimal, "~> 1.5 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:table, "~> 0.1.0", [hex: :table, repo: "hexpm", optional: true]}], "hexpm", "edead639dc6e882618c01d8fc891214c481ab9a3788dfe38dd5e37fd1d5fb2e8"}, + "mochiweb": {:hex, :mochiweb, "3.2.0", "51ba1e51ebc949ae5ec985c57bef8e4ece8f6900104fbec35585fe1e1a40c5ff", [:rebar3], [], "hexpm", "11c56bade414c41d0437922d006325ae8705f179ea437e1f40d0bb54cf60fbe8"}, + "nimble_parsec": {:hex, :nimble_parsec, "1.3.1", "2c54013ecf170e249e9291ed0a62e5832f70a476c61da16f6aac6dca0189f2af", [:mix], [], "hexpm", "2682e3c0b2eb58d90c6375fc0cc30bc7be06f365bf72608804fb9cffa5e1b167"}, + "paper_trail": {:hex, :paper_trail, "1.0.0", "afe08da1b6c07bdd300bfc087a23e22c8706f8649182d3bd549e32427fea7850", [:mix], [{:ecto, ">= 3.9.2", [hex: :ecto, repo: "hexpm", optional: false]}, {:ecto_sql, ">= 3.9.1", [hex: :ecto_sql, repo: "hexpm", optional: false]}], "hexpm", "bbfe0c85f898a478efb508ce340a90acdac6c3b6715470dbc60f99d4ab23af56"}, + "parse_trans": {:hex, :parse_trans, "3.4.1", "6e6aa8167cb44cc8f39441d05193be6e6f4e7c2946cb2759f015f8c56b76e5ff", [:rebar3], [], "hexpm", "620a406ce75dada827b82e453c19cf06776be266f5a67cff34e1ef2cbb60e49a"}, + "petal_components": {:hex, :petal_components, "1.2.9", "7eede7eee60edd3dd3958059a44ce74f6f80ec5ecfd4e508f15e0b6422592dbe", [:mix], [{:heroicons, "~> 0.5.0", [hex: :heroicons, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.6", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_ecto, "~> 4.4", [hex: :phoenix_ecto, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.18.3", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}], "hexpm", "c91a489c9230408fb7f66c931d4bc14367e0ee0a8f904cd70ff16883eb988f0c"}, + "phoenix": {:hex, :phoenix, "1.7.7", "4cc501d4d823015007ba3cdd9c41ecaaf2ffb619d6fb283199fa8ddba89191e0", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix_pubsub, "~> 2.1", [hex: :phoenix_pubsub, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.6", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:plug_crypto, "~> 1.2", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:websock_adapter, "~> 0.5.3", [hex: :websock_adapter, repo: "hexpm", optional: false]}], "hexpm", "8966e15c395e5e37591b6ed0bd2ae7f48e961f0f60ac4c733f9566b519453085"}, + "phoenix_ecto": {:hex, :phoenix_ecto, "4.4.2", "b21bd01fdeffcfe2fab49e4942aa938b6d3e89e93a480d4aee58085560a0bc0d", [:mix], [{:ecto, "~> 3.5", [hex: :ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 2.14.2 or ~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:plug, "~> 1.9", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "70242edd4601d50b69273b057ecf7b684644c19ee750989fd555625ae4ce8f5d"}, + "phoenix_html": {:hex, :phoenix_html, "3.3.2", "d6ce982c6d8247d2fc0defe625255c721fb8d5f1942c5ac051f6177bffa5973f", [:mix], [{:plug, "~> 1.5", [hex: :plug, repo: "hexpm", optional: true]}], "hexpm", "44adaf8e667c1c20fb9d284b6b0fa8dc7946ce29e81ce621860aa7e96de9a11d"}, + "phoenix_live_reload": {:hex, :phoenix_live_reload, "1.4.1", "2aff698f5e47369decde4357ba91fc9c37c6487a512b41732818f2204a8ef1d3", [:mix], [{:file_system, "~> 0.2.1 or ~> 0.3", [hex: :file_system, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}], "hexpm", "9bffb834e7ddf08467fe54ae58b5785507aaba6255568ae22b4d46e2bb3615ab"}, + "phoenix_live_view": {:hex, :phoenix_live_view, "0.18.18", "1f38fbd7c363723f19aad1a04b5490ff3a178e37daaf6999594d5f34796c47fc", [:mix], [{:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.6.15 or ~> 1.7.0", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.3", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "a5810d0472f3189ede6d2a95bda7f31c6113156b91784a3426cb0ab6a6d85214"}, + "phoenix_pubsub": {:hex, :phoenix_pubsub, "2.1.3", "3168d78ba41835aecad272d5e8cd51aa87a7ac9eb836eabc42f6e57538e3731d", [:mix], [], "hexpm", "bba06bc1dcfd8cb086759f0edc94a8ba2bc8896d5331a1e2c2902bf8e36ee502"}, + "phoenix_template": {:hex, :phoenix_template, "1.0.3", "32de561eefcefa951aead30a1f94f1b5f0379bc9e340bb5c667f65f1edfa4326", [:mix], [{:phoenix_html, "~> 2.14.2 or ~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}], "hexpm", "16f4b6588a4152f3cc057b9d0c0ba7e82ee23afa65543da535313ad8d25d8e2c"}, + "phoenix_view": {:hex, :phoenix_view, "2.0.2", "6bd4d2fd595ef80d33b439ede6a19326b78f0f1d8d62b9a318e3d9c1af351098", [:mix], [{:phoenix_html, "~> 2.14.2 or ~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}], "hexpm", "a929e7230ea5c7ee0e149ffcf44ce7cf7f4b6d2bfe1752dd7c084cdff152d36f"}, + "plug": {:hex, :plug, "1.14.2", "cff7d4ec45b4ae176a227acd94a7ab536d9b37b942c8e8fa6dfc0fff98ff4d80", [:mix], [{:mime, "~> 1.0 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:plug_crypto, "~> 1.1.1 or ~> 1.2", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.3 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "842fc50187e13cf4ac3b253d47d9474ed6c296a8732752835ce4a86acdf68d13"}, + "plug_cowboy": {:hex, :plug_cowboy, "2.6.1", "9a3bbfceeb65eff5f39dab529e5cd79137ac36e913c02067dba3963a26efe9b2", [:mix], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:cowboy_telemetry, "~> 0.3", [hex: :cowboy_telemetry, repo: "hexpm", optional: false]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "de36e1a21f451a18b790f37765db198075c25875c64834bcc82d90b309eb6613"}, + "plug_crypto": {:hex, :plug_crypto, "1.2.5", "918772575e48e81e455818229bf719d4ab4181fcbf7f85b68a35620f78d89ced", [:mix], [], "hexpm", "26549a1d6345e2172eb1c233866756ae44a9609bd33ee6f99147ab3fd87fd842"}, + "postgrex": {:hex, :postgrex, "0.17.3", "c92cda8de2033a7585dae8c61b1d420a1a1322421df84da9a82a6764580c503d", [:mix], [{:db_connection, "~> 2.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:decimal, "~> 1.5 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:table, "~> 0.1.0", [hex: :table, repo: "hexpm", optional: true]}], "hexpm", "946cf46935a4fdca7a81448be76ba3503cff082df42c6ec1ff16a4bdfbfb098d"}, "pre_commit": {:hex, :pre_commit, "0.3.4", "e2850f80be8090d50ad8019ef2426039307ff5dfbe70c736ad0d4d401facf304", [:mix], [], "hexpm", "16f684ba4f1fed1cba6b19e082b0f8d696e6f1c679285fedf442296617ba5f4e"}, "ranch": {:hex, :ranch, "1.8.0", "8c7a100a139fd57f17327b6413e4167ac559fbc04ca7448e9be9057311597a1d", [:make, :rebar3], [], "hexpm", "49fbcfd3682fab1f5d109351b61257676da1a2fdbe295904176d5e521a2ddfe5"}, - "ssl_verify_fun": {:hex, :ssl_verify_fun, "1.1.6", "cf344f5692c82d2cd7554f5ec8fd961548d4fd09e7d22f5b62482e5aeaebd4b0", [:make, :mix, :rebar3], [], "hexpm", "bdb0d2471f453c88ff3908e7686f86f9be327d065cc1ec16fa4540197ea04680"}, + "ssl_verify_fun": {:hex, :ssl_verify_fun, "1.1.7", "354c321cf377240c7b8716899e182ce4890c5938111a1296add3ec74cf1715df", [:make, :mix, :rebar3], [], "hexpm", "fe4c190e8f37401d30167c8c405eda19469f34577987c76dde613e838bbc67f8"}, "statuses": {:hex, :statuses, "1.1.1", "bb52e1d7d3d16624c592a8a7712ddd03ea19f11cf8df54d8604c20d220b358e9", [:mix], [], "hexpm", "9dbc3839547401153dce11b8ab1bc406339e18d4c67eb0449c38749e26ba2e27"}, - "tailwind": {:hex, :tailwind, "0.1.9", "25ba09d42f7bfabe170eb67683a76d6ec2061952dc9bd263a52a99ba3d24bd4d", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "9213f87709c458aaec313bb5f2df2b4d2cedc2b630e4ae821bf3c54c47a56d0b"}, - "telemetry": {:hex, :telemetry, "1.1.0", "a589817034a27eab11144ad24d5c0f9fab1f58173274b1e9bae7074af9cbee51", [:rebar3], [], "hexpm", "b727b2a1f75614774cff2d7565b64d0dfa5bd52ba517f16543e6fc7efcc0df48"}, + "tailwind": {:hex, :tailwind, "0.2.1", "83d8eadbe71a8e8f67861fe7f8d51658ecfb258387123afe4d9dc194eddc36b0", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "e8a13f6107c95f73e58ed1b4221744e1eb5a093cd1da244432067e19c8c9a277"}, + "telemetry": {:hex, :telemetry, "1.2.1", "68fdfe8d8f05a8428483a97d7aab2f268aaff24b49e0f599faa091f1d4e7f61c", [:rebar3], [], "hexpm", "dad9ce9d8effc621708f99eac538ef1cbe05d6a874dd741de2e689c47feafed5"}, "telemetry_metrics": {:hex, :telemetry_metrics, "0.6.1", "315d9163a1d4660aedc3fee73f33f1d355dcc76c5c3ab3d59e76e3edf80eef1f", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "7be9e0871c41732c233be71e4be11b96e56177bf15dde64a8ac9ce72ac9834c6"}, "telemetry_poller": {:hex, :telemetry_poller, "1.0.0", "db91bb424e07f2bb6e73926fcafbfcbcb295f0193e0a00e825e589a0a47e8453", [:rebar3], [{:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "b3a24eafd66c3f42da30fc3ca7dda1e9d546c12250a2d60d7b81d264fbec4f6e"}, + "timex": {:hex, :timex, "3.7.11", "bb95cb4eb1d06e27346325de506bcc6c30f9c6dea40d1ebe390b262fad1862d1", [:mix], [{:combine, "~> 0.10", [hex: :combine, repo: "hexpm", optional: false]}, {:gettext, "~> 0.20", [hex: :gettext, repo: "hexpm", optional: false]}, {:tzdata, "~> 1.1", [hex: :tzdata, repo: "hexpm", optional: false]}], "hexpm", "8b9024f7efbabaf9bd7aa04f65cf8dcd7c9818ca5737677c7b76acbc6a94d1aa"}, + "tzdata": {:hex, :tzdata, "1.1.1", "20c8043476dfda8504952d00adac41c6eda23912278add38edc140ae0c5bcc46", [:mix], [{:hackney, "~> 1.17", [hex: :hackney, repo: "hexpm", optional: false]}], "hexpm", "a69cec8352eafcd2e198dea28a34113b60fdc6cb57eb5ad65c10292a6ba89787"}, "unicode_util_compat": {:hex, :unicode_util_compat, "0.7.0", "bc84380c9ab48177092f43ac89e4dfa2c6d62b40b8bd132b1059ecc7232f9a78", [:rebar3], [], "hexpm", "25eee6d67df61960cf6a794239566599b09e17e668d3700247bc498638152521"}, - "useful": {:hex, :useful, "1.0.8", "795a5bf94567e4a1b374621577acabc80ea80634b634095237f98e40e64e9d24", [:mix], [], "hexpm", "947ae0ba2b3c06bcfd8994e95e29f4cc13287aab81b481ae6abb9077fc9c1ad5"}, + "useful": {:hex, :useful, "1.12.1", "972d68f402226d3c82aa4496ac3977eee2a3f3f05bdc781e849e9941e5ac3b8d", [:mix], [{:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "897554c252c765be2aaec9ba6bc7a77e6bc101a70b40cc1c09416a6be707f2cf"}, + "websock": {:hex, :websock, "0.5.3", "2f69a6ebe810328555b6fe5c831a851f485e303a7c8ce6c5f675abeb20ebdadc", [:mix], [], "hexpm", "6105453d7fac22c712ad66fab1d45abdf049868f253cf719b625151460b8b453"}, + "websock_adapter": {:hex, :websock_adapter, "0.5.4", "7af8408e7ed9d56578539594d1ee7d8461e2dd5c3f57b0f2a5352d610ddde757", [:mix], [{:bandit, ">= 0.6.0", [hex: :bandit, repo: "hexpm", optional: true]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.6", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:websock, "~> 0.5", [hex: :websock, repo: "hexpm", optional: false]}], "hexpm", "d2c238c79c52cbe223fcdae22ca0bb5007a735b9e933870e241fce66afb4f4ab"}, } diff --git a/priv/repo/migrations/20221004195054_add_person.exs b/priv/repo/migrations/20221004195054_add_person.exs deleted file mode 100644 index 9eade4e3..00000000 --- a/priv/repo/migrations/20221004195054_add_person.exs +++ /dev/null @@ -1,30 +0,0 @@ -defmodule App.Repo.Migrations.AddPerson do - use Ecto.Migration - - def change do - execute("CREATE EXTENSION IF NOT EXISTS citext") - - create table(:people, primary_key: false) do - add(:person_id, :integer, primary_key: true) - add(:name, :citext) - - timestamps() - end - - # insert new people rows by select unique person_id from items and tags - execute( - "INSERT INTO people(person_id, inserted_at, updated_at) - SELECT DISTINCT i.person_id, now(), now() FROM items as i UNION SELECT DISTINCT t.person_id, now(), now() from tags as t;" - ) - - create(unique_index(:people, [:name], name: :people_name_index)) - - alter table(:items) do - modify(:person_id, references(:people, column: :person_id)) - end - - alter table(:tags) do - modify(:person_id, references(:people, column: :person_id)) - end - end -end diff --git a/priv/repo/migrations/20221005142257_add_list.exs b/priv/repo/migrations/20221005142257_add_list.exs deleted file mode 100644 index 58cc1316..00000000 --- a/priv/repo/migrations/20221005142257_add_list.exs +++ /dev/null @@ -1,31 +0,0 @@ -defmodule App.Repo.Migrations.AddList do - use Ecto.Migration - - def change do - execute("CREATE EXTENSION IF NOT EXISTS citext") - - create table(:lists) do - add(:name, :citext) - add(:person_id, references(:people, column: :person_id)) - - timestamps() - end - - create( - unique_index(:lists, [:name, :person_id], - name: :lists_name_person_id_index - ) - ) - - create table(:lists_items, primary_key: false) do - add(:item_id, references(:items, on_delete: :delete_all)) - add(:list_id, references(:lists, on_delete: :delete_all)) - - timestamps() - end - - # create a unique index on item_id and list_id to avoid - # adding an item multiple time to the same list - create(unique_index(:lists_items, [:item_id, :list_id])) - end -end diff --git a/test/app/item_test.exs b/test/app/item_test.exs index d1f9152e..b44189bf 100644 --- a/test/app/item_test.exs +++ b/test/app/item_test.exs @@ -1,8 +1,6 @@ defmodule App.ItemTest do - use App.DataCase, async: false - alias App.{Item, Person, Timer} - - setup [:create_person] + use App.DataCase, async: true + alias App.{Item, Timer} describe "items" do @valid_attrs %{text: "some text", person_id: 1, status: 2} @@ -153,9 +151,4 @@ defmodule App.ItemTest do assert length(item_timers) > 0 end end - - defp create_person(_) do - person = Person.create_person(%{"person_id" => 1, "name" => "guest"}) - %{person: person} - end end diff --git a/test/app/list_test.exs b/test/app/list_test.exs_BAK similarity index 100% rename from test/app/list_test.exs rename to test/app/list_test.exs_BAK diff --git a/test/app/person_test.exs b/test/app/person_test.exs deleted file mode 100644 index 95185d9b..00000000 --- a/test/app/person_test.exs +++ /dev/null @@ -1,66 +0,0 @@ -defmodule App.PersonTest do - use App.DataCase - alias App.Person - - describe "Test constraints and requirements for Person schema" do - test "valid person changeset" do - changeset = Person.changeset(%Person{}, %{person_id: 1, name: "person1"}) - - assert changeset.valid? - end - - test "invalid person changeset when name value missing" do - changeset = Person.changeset(%Person{}, %{person_id: 1, name: ""}) - refute changeset.valid? - end - - test "invalid person changeset when person_id value missing" do - changeset = Person.changeset(%Person{}, %{name: "person1"}) - refute changeset.valid? - end - end - - describe "Save person in Postgres" do - @valid_attrs %{person_id: 1, name: "person 1"} - @invalid_attrs %{name: nil} - - test "get_person!/1 returns the person with given id" do - {:ok, person} = Person.create_person(@valid_attrs) - assert Person.get_person!(person.person_id).name == person.name - end - - test "create_person/1 with invalid data returns error changeset" do - assert {:error, %Ecto.Changeset{}} = Person.create_person(@invalid_attrs) - end - - test "create_person/1 returns invalid changeset when trying to insert a duplicate name" do - assert {:ok, _person} = Person.create_person(@valid_attrs) - - assert {:error, _changeset} = - Person.create_person(%{person_id: 2, name: "person 1"}) - end - - test "get_or_insert preson" do - assert person = Person.get_or_insert(0) - assert person.person_id == 0 - - {:ok, person} = Person.create_person(@valid_attrs) - assert get_person = Person.get_or_insert(1) - assert get_person.name == person.name - end - end - - describe "Update person in Postgres" do - @valid_attrs %{person_id: 1, name: "person 1"} - @valid_update_attrs %{person_id: 1, name: "person 1 updated"} - - test "create_person/1 returns invalid changeset when trying to insert a duplicate name" do - assert {:ok, person} = Person.create_person(@valid_attrs) - - assert {:ok, person_updated} = - Person.update_person(person, @valid_update_attrs) - - assert person_updated.name == "person 1 updated" - end - end -end diff --git a/test/app/tag_test.exs b/test/app/tag_test.exs index 7b414ddf..3a0e0b24 100644 --- a/test/app/tag_test.exs +++ b/test/app/tag_test.exs @@ -1,8 +1,6 @@ defmodule App.TagTest do use App.DataCase - alias App.{Person, Tag} - - setup [:create_person] + alias App.Tag describe "Test constraints and requirements for Tag schema" do test "valid tag changeset" do @@ -56,9 +54,4 @@ defmodule App.TagTest do assert length(tags) == 0 end end - - defp create_person(_) do - person = Person.create_person(%{"person_id" => 1, "name" => "guest"}) - %{person: person} - end end diff --git a/test/app/timer_test.exs b/test/app/timer_test.exs index 28b63ef0..1e7844e9 100644 --- a/test/app/timer_test.exs +++ b/test/app/timer_test.exs @@ -1,8 +1,6 @@ defmodule App.TimerTest do use App.DataCase - alias App.{Item, Person, Timer} - - setup [:create_person] + alias App.{Item, Timer} describe "timers" do @valid_item_attrs %{text: "some text", person_id: 1} @@ -69,8 +67,8 @@ defmodule App.TimerTest do end end - defp create_person(_) do - person = Person.create_person(%{"person_id" => 1, "name" => "guest"}) - %{person: person} - end + # defp create_person(_) do + # person = Person.create_person(%{"person_id" => 1, "name" => "guest"}) + # %{person: person} + # end end diff --git a/test/app_web/controllers/item_controller_test.exs b/test/app_web/controllers/item_controller_test.exs deleted file mode 100644 index 20ae0eba..00000000 --- a/test/app_web/controllers/item_controller_test.exs +++ /dev/null @@ -1,89 +0,0 @@ -defmodule AppWeb.ItemControllerTest do - use AppWeb.ConnCase - - alias App.{List, Person, Item} - - setup [:create_person] - - @create_attrs %{name: "list1", person_id: 1} - @create_attrs_item %{text: "item1", person_id: 1} - - def fixture(:list) do - {:ok, list} = List.create_list(@create_attrs) - list - end - - def fixture(:item) do - {:ok, item} = Item.create_item(@create_attrs_item) - item - end - - describe "edit item's list" do - setup [:create_item, :create_list] - - test "renders form for editing item's lists", %{ - conn: conn, - item: item - } do - conn = - conn - |> assign(:person, %{id: 1}) - |> get(Routes.item_path(conn, :edit, item)) - - assert html_response(conn, 200) =~ "Edit Item" - end - end - - describe "update item's list" do - setup [:create_item, :create_list] - - test "Add a list to an item", %{ - conn: conn, - list: list, - item: item - } do - update_params = %{item_lists: [list.id]} - - conn = - conn - |> assign(:person, %{id: 1}) - |> put(Routes.item_path(conn, :update, item), item: update_params) - - assert length(Item.get_item!(item.id).lists) == 1 - - assert redirected_to(conn) == "/" - end - - test "Remove list from item", %{ - conn: conn, - item: item - } do - update_params = %{item_lists: ""} - - conn = - conn - |> assign(:person, %{id: 1}) - |> put(Routes.item_path(conn, :update, item), item: update_params) - - assert length(Item.get_item!(item.id).lists) == 0 - - assert redirected_to(conn) == "/" - end - end - - defp create_list(_) do - list = fixture(:list) - %{list: list} - end - - defp create_item(_) do - item = fixture(:item) - %{item: item} - end - - defp create_person(_) do - person = Person.create_person(%{"person_id" => 0, "name" => "guest"}) - _person1 = Person.create_person(%{"person_id" => 1, "name" => "person1"}) - %{person: person} - end -end diff --git a/test/app_web/controllers/list_controller_test.exs b/test/app_web/controllers/list_controller_test.exs_BAK similarity index 100% rename from test/app_web/controllers/list_controller_test.exs rename to test/app_web/controllers/list_controller_test.exs_BAK diff --git a/test/app_web/controllers/profile_controller_test.exs b/test/app_web/controllers/profile_controller_test.exs deleted file mode 100644 index 05c2cfdd..00000000 --- a/test/app_web/controllers/profile_controller_test.exs +++ /dev/null @@ -1,92 +0,0 @@ -defmodule AppWeb.ProfileControllerTest do - use AppWeb.ConnCase - - alias App.{Person, Repo} - - setup [:create_profile] - - @create_attrs %{person_id: 1, name: "person 1"} - @update_attrs %{person_id: 1, name: "person 1 udpated"} - @invalid_attrs %{person_id: 1, name: ""} - - describe "edit profile" do - test "renders form for editing chosen profile", %{ - conn: conn, - person: profile - } do - conn = - conn - |> assign(:person, %{id: 1}) - |> get(Routes.profile_path(conn, :edit, profile)) - - assert html_response(conn, 200) =~ "Edit Profile" - end - - test "redirect to index when missing permission to edit the profile", %{ - conn: conn, - person: profile - } do - conn = get(conn, Routes.profile_path(conn, :edit, profile)) - - assert redirected_to(conn) == "/" - end - end - - describe "update profile" do - test "redirects to home page when data is valid", %{ - conn: conn, - person: person - } do - conn = - conn - |> assign(:person, %{id: 1}) - |> put(Routes.profile_path(conn, :update, person), person: @update_attrs) - - assert redirected_to(conn) == "/" - end - - test "renders errors when data is invalid", %{conn: conn, person: person} do - conn = - conn - |> assign(:person, %{id: 1}) - |> put(Routes.profile_path(conn, :update, person), - person: @invalid_attrs - ) - - assert html_response(conn, 200) =~ "Edit Profile" - end - end - - describe "redirect to edit profile when profile name not define" do - setup :create_profile_with_no_name - - test "redirect to edit profile", %{conn: conn, person: person} do - conn = - conn - |> assign(:person, %{id: person.person_id}) - |> get("/") - - assert redirected_to(conn) == "/profile/#{person.person_id}/edit" - end - end - - def fixture(:person) do - {:ok, person} = Person.create_person(@create_attrs) - person - end - - def fixture(:person_no_name) do - {:ok, person} = Repo.insert(%Person{person_id: 404}) - person - end - - defp create_profile(_) do - person = fixture(:person) - %{person: person} - end - - defp create_profile_with_no_name(_) do - person = fixture(:person_no_name) - %{person: person} - end -end diff --git a/test/app_web/controllers/tag_controller_test.exs b/test/app_web/controllers/tag_controller_test.exs index a8fac12f..6f9bae2b 100644 --- a/test/app_web/controllers/tag_controller_test.exs +++ b/test/app_web/controllers/tag_controller_test.exs @@ -1,9 +1,7 @@ defmodule AppWeb.TagControllerTest do use AppWeb.ConnCase - alias App.{Tag, Person} - - setup [:create_person] + alias App.Tag @create_attrs %{text: "tag1", person_id: 1, color: "#FCA5A5"} @update_attrs %{text: "tag1 updated", color: "#F87171"} @@ -20,14 +18,14 @@ defmodule AppWeb.TagControllerTest do assert html_response(conn, 200) =~ "Listing Tags" end - test "lists all tags and display logout button", %{conn: conn} do + test "lists all tagz", %{conn: conn} do conn = conn |> assign(:jwt, "jwt value...") |> assign(:person, %{id: 1, picture: ""}) |> get(Routes.tag_path(conn, :index)) - assert html_response(conn, 200) =~ "logout" + assert html_response(conn, 200) =~ "Tags" end end @@ -92,10 +90,4 @@ defmodule AppWeb.TagControllerTest do tag = fixture(:tag) %{tag: tag} end - - defp create_person(_) do - person = Person.create_person(%{"person_id" => 0, "name" => "guest"}) - _person = Person.create_person(%{"person_id" => 1, "name" => "Person1"}) - %{person: person} - end end diff --git a/test/app_web/live/app_live_test.exs b/test/app_web/live/app_live_test.exs index fd8bec89..a39055c2 100644 --- a/test/app_web/live/app_live_test.exs +++ b/test/app_web/live/app_live_test.exs @@ -4,8 +4,6 @@ defmodule AppWeb.AppLiveTest do import Phoenix.LiveViewTest alias Phoenix.Socket.Broadcast - setup [:create_person] - test "disconnected and connected render", %{conn: conn} do {:ok, page_live, disconnected_html} = live(conn, "/") assert disconnected_html =~ "done" @@ -214,21 +212,6 @@ defmodule AppWeb.AppLiveTest do assert render(view) end - test "Logout link displayed when loggedin", %{conn: conn} do - data = %{ - email: "test@dwyl.com", - givenName: "Alex", - picture: "this", - auth_provider: "GitHub", - id: 0 - } - - jwt = AuthPlug.Token.generate_jwt!(data) - - conn = get(conn, "/?jwt=#{jwt}") - assert html_response(conn, 200) =~ "logout" - end - test "get /logout with valid JWT", %{conn: conn} do data = %{ email: "test@dwyl.com", @@ -249,9 +232,9 @@ defmodule AppWeb.AppLiveTest do assert "/" = redirected_to(conn, 302) end - test "test login link redirect to auth.dwyl.com", %{conn: conn} do + test "test login link redirect to authdemo.fly.dev", %{conn: conn} do conn = get(conn, "/login") - assert redirected_to(conn, 302) =~ "auth.dwyl.com" + assert redirected_to(conn, 302) =~ "authdemo.fly.dev" end test "tags_to_string/1" do @@ -260,9 +243,4 @@ defmodule AppWeb.AppLiveTest do %Tag{text: "Elixir"} ]) == "Learn, Elixir" end - - defp create_person(_) do - person = Person.create_person(%{"person_id" => 0, "name" => "guest"}) - %{person: person} - end end From 42392dcec9689ec7c46f67ae87ccf8fff3d591c8 Mon Sep 17 00:00:00 2001 From: nelsonic Date: Fri, 8 Sep 2023 10:22:31 +0100 Subject: [PATCH 10/56] revert changes in BUILDT.md https://github.com/dwyl/mvp/pull/165#issuecomment-1709925736 --- BUILDIT.md | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/BUILDIT.md b/BUILDIT.md index 87df02e5..6ba75453 100644 --- a/BUILDIT.md +++ b/BUILDIT.md @@ -28,11 +28,10 @@ and _then_ built the UI.
We were able to do this because we had a good idea of which functions we were going to need.
If you are reading through this - +and scratching your head wondering where a particular function will be used, simply scroll down to the UI section where (_hopefully_) it will all be clear. -mii At the end of each step, remember to run the tests: From 6a233f87d4cd5cf45ef3d19d91777942387f88ab Mon Sep 17 00:00:00 2001 From: nelsonic Date: Fri, 8 Sep 2023 15:53:34 +0100 Subject: [PATCH 11/56] merge main https://github.com/dwyl/mvp/pull/165#issuecomment-1711355895 --- BUILDIT.md | 3323 ++++++++++++++++- lib/app/item.ex | 182 +- lib/app/person.ex | 6 + lib/app/person.ex_BAK | 44 - lib/app/tag.ex | 28 +- lib/app_web/live/app_live.html.heex | 852 +++-- lib/app_web/router.ex | 28 +- lib/app_web/templates/layout/root.html.heex | 229 +- lib/app_web/templates/tag/edit.html.heex | 18 +- lib/app_web/templates/tag/form.html.heex | 24 +- mix.lock | 16 +- test/app/item_test.exs | 109 +- test/app/person_test.exs | 29 + test/app/tag_test.exs | 18 +- test/app/timer_test.exs | 70 +- .../controllers/tag_controller_test.exs | 45 +- test/app_web/live/app_live_test.exs | 716 +++- 17 files changed, 5124 insertions(+), 613 deletions(-) create mode 100644 lib/app/person.ex delete mode 100644 lib/app/person.ex_BAK create mode 100644 test/app/person_test.exs diff --git a/BUILDIT.md b/BUILDIT.md index 6ba75453..cbf58391 100644 --- a/BUILDIT.md +++ b/BUILDIT.md @@ -1,8 +1,7 @@
# Build Log πŸ‘©β€πŸ’» -![GitHub Workflow Status](https://img.shields.io/github/workflow/status/dwyl/mvp/Elixir%20CI?label=build&style=flat-square) - +![GitHub Workflow Status](https://img.shields.io/github/actions/workflow/status/dwyl/mvp/ci.yml?label=build&style=flat-square&branch=main) This is a log of the steps taken @@ -28,10 +27,11 @@ and _then_ built the UI.
We were able to do this because we had a good idea of which functions we were going to need.
If you are reading through this -and scratching your head +and scratching your head wondering where a particular function will be used, simply scroll down to the UI section where (_hopefully_) it will all be clear. + At the end of each step, remember to run the tests: @@ -46,6 +46,11 @@ We suggest keeping two terminal tabs/windows running one for the server `mix phx.server` and the other for the tests. That way you can also see the UI as you progress. +We created a *separate* +document detailing the implementation of the `API`. +Please see: +[`API.md`](./API.md). + With that in place, let's get building! - [Build Log πŸ‘©β€πŸ’»](#build-log-) @@ -90,9 +95,34 @@ With that in place, let's get building! - [11.3 Test tags with Iex](#113-test-tags-with-iex) - [11.4 Testing Schemas](#114-testing-schemas) - [11.4 Items-Tags association](#114--items-tags-association) -- [12. Run the _Finished_ MVP App!](#12-run-the-finished-mvp-app) - - [12.1 Run the Tests](#121-run-the-tests) - - [12.2 Run The App](#122-run-the-app) +- [12. Editing timers](#12-editing-timers) + - [12.1 Parsing DateTime strings](#121-parsing-datetime-strings) + - [12.2 Persisting update in database](#122-persisting-update-in-database) + - [12.3 Adding event handler in `app_live.ex`](#123-adding-event-handler-in-app_liveex) + - [12.4 Updating timer changeset list on `timer.ex`](#124-updating-timer-changeset-list-on-timerex) + - [12.5 Updating the UI](#125-updating-the-ui) + - [12.6 Updating the tests and going back to 100% coverage](#126-updating-the-tests-and-going-back-to-100-coverage) +- [13. Tracking changes of `items` in database](#13-tracking-changes-of-items-in-database) + - [13.1 Setting up](#131-setting-up) + - [13.2 Changing database transactions on `item` insert and update](#132-changing-database-transactions-on-item-insert-and-update) + - [13.3 Fixing tests](#133-fixing-tests) + - [13.4 Checking the changes using `DBEaver`](#134-checking-the-changes-using-dbeaver) +- [14. Adding a dashboard to track metrics](#14-adding-a-dashboard-to-track-metrics) + - [14.1 Adding new `LiveView` page in `/stats`](#141-adding-new-liveview-page-in-stats) + - [14.2 Fetching counter of `timers` and `items` for each `person`](#142-fetching-counter-of-timers-and-items-for-each-person) + - [14.3 Building the Stats Page](#143-building-the-stats-page) + - [14.4 Broadcasting to `stats` channel](#144-broadcasting-to-stats-channel) + - [14.5 Adding tests](#145-adding-tests) +- [15. `People` in Different Timezones 🌐](#15-people-in-different-timezones-) + - [15.1 Getting the `person`'s Timezone](#151-getting-the-persons-timezone) + - [15.2 Changing how the timer datetime is displayed](#152-changing-how-the-timer-datetime-is-displayed) + - [15.3 Persisting the adjusted timezone](#153-persisting-the-adjusted-timezone) + - [15.4 Adding test](#154-adding-test) +- [16. `Lists`](#16-lists) +- [17. Reordering `items` Using Drag \& Drop](#17-reordering-items-using-drag--drop) +- [18. Run the _Finished_ MVP App!](#18-run-the-finished-mvp-app) + - [18.1 Run the Tests](#181-run-the-tests) + - [18.2 Run The App](#182-run-the-app) - [Thanks!](#thanks) @@ -125,6 +155,7 @@ to reduce complexity/dependencies. Run the `Phoenix` app with the command: ```sh +cd app mix phx.server ``` @@ -172,14 +203,12 @@ That tells us everything is working as expected. πŸš€ If you prefer to see **test coverage** - we certainly do - then you will need to add a few lines to the [`mix.exs`](https://github.com/dwyl/app-mvp/blob/main/mix.exs) -file and -create a -[`coveralls.json`](https://github.com/dwyl/app-mvp/blob/main/coveralls.json) -file to exclude `Phoenix` files from `excoveralls` checking. -Add alias (shortcuts) in `mix.exs` `defp aliases do` list. - -e.g: `mix c` runs `mix coveralls.html` -see: [**`commits/d6ab5ef`**](https://github.com/dwyl/app-mvp/pull/90/commits/d6ab5ef7c2be5dcad7d060e782393ae29c94a526) ... +file. You need to do two things: +*add a [`coveralls.json`](https://github.com/dwyl/app-mvp/blob/main/coveralls.json) +file (copy the contents)* +and _change `mix.exs` file according to +[**`commit #d6ab5ef7c`**](https://github.com/dwyl/app-mvp/pull/90/commits/d6ab5ef7c2be5dcad7d060e782393ae29c94a526) +(add alias in `defp aliases do` list with the `mix c` command)_. This is just standard `Phoenix` project setup for us, so we don't duplicate any of the steps here.
@@ -203,14 +232,17 @@ You should see output similar to the following: ## 1.3 Setup `Tailwind` -As we're using **`Tailwind CSS`** -for the **UI** in this project -we need to enable it. +With the release of +[**`Phoenix 1.7`**](https://phoenixframework.org/blog/phoenix-1.7-final-released), +**`Tailwind CSS`** is automatically built-in. +So if you are starting the project with the new +`phx.new` generator, +you don't need to follow any steps. -We are not duplicating the instructions here, +Otherwise, please refer to: [Tailwind in Phoenix](https://github.com/dwyl/learn-tailwind#part-2-tailwind-in-phoenix). -Should only take **`~1 minute`**. +Setting up should only take **`~1 minute`**. By the end of this step you should have **`Tailwind`** working. When you visit @@ -299,7 +331,7 @@ end Now if you refresh the page you should see the following: -![liveveiw-page-with-tailwind-style](https://user-images.githubusercontent.com/194400/176137805-34467c88-add2-487f-9593-931d0314df62.png) +liveveiw-page-with-tailwind-style ## 1.6 Update Tests @@ -350,7 +382,9 @@ defmodule AppWeb.AppLiveTest do end ``` -Save the file +Save the file, +delete the `test/app_web/controllers/page_controller_test.exs` +(it has the failing test on an unused component) and re-run the tests: `mix test` You should see output similar to the following: @@ -375,7 +409,6 @@ we can delete the default files created by `Phoenix`: lib/app_web/views/page_view.ex lib/app_web/controllers/page_controller.ex lib/app_web/templates/page/index.html.heex -test/app_web/controllers/page_controller_test.exs ``` With those files deleted, @@ -473,7 +506,7 @@ for what we are building later on. > **Note**: The keen-eyed observer (with PostgreSQL experience) will have noticed that `items.person_id` is an `Integer` (`int4`) data type. -This means we are _limted_ to **`2147483647` people** using the MVP. +This means we are _limited_ to **`2147483647` people** using the MVP. See: [/datatype-numeric.html](https://www.postgresql.org/docs/current/datatype-numeric.html) We aren't expecting more than @@ -792,7 +825,7 @@ defmodule App.TimerTest do {:ok, timer} = Timer.start(%{item_id: item.id, person_id: 1, start: seven_seconds_ago}) - #Β stop the timer based on it's item_id + # stop the timer based on it's item_id Timer.stop_timer_for_item_id(item.id) stopped_timer = Timer.get_timer!(timer.id) @@ -1284,18 +1317,7 @@ defmodule AppWeb.AuthController do import Phoenix.LiveView, only: [assign_new: 3] def on_mount(:default, _params, %{"jwt" => jwt} = _session, socket) do - - claims = jwt - |> AuthPlug.Token.verify_jwt!() - |> AuthPlug.Helpers.strip_struct_metadata() - |> Useful.atomize_map_keys() - - socket = - socket - |> assign_new(:person, fn -> claims end) - |> assign_new(:loggedin, fn -> true end) - - {:cont, socket} + {:cont, AuthPlug.assign_jwt_to_socket(socket, &Phoenix.Component.assign_new/3, jwt)} end def on_mount(:default, _params, _session, socket) do @@ -1773,6 +1795,17 @@ file and replace the contents with the following: + <%= live_title_tag assigns[:page_title] || "dwyl mvp"%> <%= render "icons.html" %> @@ -1781,33 +1814,140 @@ file and replace the contents with the following: - -