Rendering champions with win rates
Some checks are pending
ci / docker (push) Waiting to run

This commit is contained in:
Álvaro 2024-05-31 00:45:22 +02:00
parent 520c234a94
commit 22b79f5376
19 changed files with 853 additions and 9 deletions

View File

@ -0,0 +1,44 @@
defmodule LolAnalytics.Dimensions.Champion.ChampionMetadata do
alias LolAnalytics.Dimensions.Champion.ChampionRepo
@champions_data_url "https://ddragon.leagueoflegends.com/cdn/14.11.1/data/en_US/champion.json"
use GenServer
def update_metadata() do
{:ok, %{"data" => data}} = get_champions()
data
|> Enum.each(&save_metadata/1)
end
defp get_champions() do
with {:ok, resp} <- HTTPoison.get(@champions_data_url),
data <- Poison.decode(resp.body) do
data
else
{:error, reason} -> {:error, reason}
end
end
defp save_metadata({champion, info}) do
%{
"image" => %{
"full" => full_image
},
"name" => name,
"key" => key_string
} = info
attrs = %{
image: full_image,
name: name
}
{champion_id, _} = Integer.parse(key_string)
ChampionRepo.update(champion_id, attrs)
info
|> IO.inspect()
end
end

View File

@ -18,6 +18,13 @@ defmodule LolAnalytics.Dimensions.Champion.ChampionRepo do
end end
end end
def update(champion_id, attrs) do
get_or_create(champion_id)
|> ChampionSchema.changeset(attrs)
|> Repo.update()
end
@spec list_champions() :: any()
def list_champions() do def list_champions() do
Repo.all(ChampionSchema) Repo.all(ChampionSchema)
end end

View File

@ -4,12 +4,14 @@ defmodule LolAnalytics.Dimensions.Champion.ChampionSchema do
schema "dim_champion" do schema "dim_champion" do
field :champion_id, :integer field :champion_id, :integer
field :name, :string
field :image, :string
timestamps() timestamps()
end end
def changeset(champion = %__MODULE__{}, attrs \\ %{}) do def changeset(champion = %__MODULE__{}, attrs \\ %{}) do
champion champion
|> cast(attrs, [:champion_id]) |> cast(attrs, [:champion_id, :name, :image])
|> validate_required([:champion_id]) |> validate_required([:champion_id])
end end
end end

View File

@ -1,6 +1,7 @@
defmodule LolAnalytics.Facts.ChampionPlayedGame.ChampionPlayedGameRepo do defmodule LolAnalytics.Facts.ChampionPlayedGame.ChampionPlayedGameRepo do
import Ecto.Query import Ecto.Query
alias LolAnalytics.Dimensions.Champion.ChampionSchema
alias LolAnalytics.Dimensions.Player.PlayerRepo alias LolAnalytics.Dimensions.Player.PlayerRepo
alias LolAnalytics.Dimensions.Champion.ChampionRepo alias LolAnalytics.Dimensions.Champion.ChampionRepo
alias LolAnalytics.Dimensions.Match.MatchRepo alias LolAnalytics.Dimensions.Match.MatchRepo
@ -9,19 +10,70 @@ defmodule LolAnalytics.Facts.ChampionPlayedGame.ChampionPlayedGameRepo do
alias LoLAnalytics.Repo alias LoLAnalytics.Repo
def insert(attrs) do def insert(attrs) do
match = MatchRepo.get_or_create(attrs.match_id) _match = MatchRepo.get_or_create(attrs.match_id)
champion = ChampionRepo.get_or_create(attrs.champion_id) _champion = ChampionRepo.get_or_create(attrs.champion_id)
player = PlayerRepo.get_or_create(attrs.puuid) _player = PlayerRepo.get_or_create(attrs.puuid)
IO.puts(">>>>")
IO.inspect(attrs)
changeset = ChampionPlayedGameSchema.changeset(%ChampionPlayedGameSchema{}, attrs) changeset = ChampionPlayedGameSchema.changeset(%ChampionPlayedGameSchema{}, attrs)
IO.inspect(changeset)
Repo.insert(changeset) Repo.insert(changeset)
# Repo.insert(match)
end end
def list_played_matches() do def list_played_matches() do
Repo.all(ChampionPlayedGameSchema) Repo.all(ChampionPlayedGameSchema)
end end
def get_win_rates2 do
query2 =
from m in ChampionPlayedGameSchema,
join: c in ChampionSchema,
on: c.champion_id == m.champion_id,
select: %{
wins: count(m.is_win),
win_rate:
fragment(
"
((cast(count(CASE WHEN ? THEN 1 END) as float) / cast(count(*) as float)) * 100.0
)",
m.is_win
),
id: m.champion_id,
name: c.name,
image: c.image
},
group_by: [m.champion_id, c.image, c.name]
Repo.all(query2)
end
def get_win_rates() do
query = """
SELECT
(cast(count(CASE WHEN is_win THEN 1 END) as float) / cast(count(*) as float)) * 100.0 as win_rate,
count(CASE WHEN is_win THEN 1 END) as games_won,
count(*) as total_games,
champion_id
FROM fact_champion_played_game
GROUP BY champion_id
ORDER BY win_rate desc;
"""
case Ecto.Adapters.SQL.query(Repo, query, []) do
{:ok, %Postgrex.Result{rows: rows}} ->
rows
|> Enum.map(fn [win_rate, wins, games, champion_id] ->
%{
win_rate: win_rate,
wins: wins,
games: games,
champion_id: champion_id
}
end)
{:error, _exception} ->
:error
end
end
def get_win_rates_by_roles() do
end
end end

View File

@ -0,0 +1,10 @@
defmodule LoLAnalytics.Repo.Migrations.ChampionDimName do
use Ecto.Migration
def change do
alter table("dim_champion") do
add :name, :string
add :image, :string
end
end
end

View File

@ -0,0 +1,90 @@
defmodule LoLAnalyticsWeb.ChampionLive.FormComponent do
use LoLAnalyticsWeb, :live_component
alias LoLAnalytics.Accounts
@impl true
def render(assigns) do
~H"""
<div>
<.header>
<%= @title %>
<:subtitle>Use this form to manage champion records in your database.</:subtitle>
</.header>
<.simple_form
for={@form}
id="champion-form"
phx-target={@myself}
phx-change="validate"
phx-submit="save"
>
<:actions>
<.button phx-disable-with="Saving...">Save Champion</.button>
</:actions>
</.simple_form>
</div>
"""
end
@impl true
def update(%{champion: champion} = assigns, socket) do
changeset = Accounts.change_champion(champion)
{:ok,
socket
|> assign(assigns)
|> assign_form(changeset)}
end
@impl true
def handle_event("validate", %{"champion" => champion_params}, socket) do
changeset =
socket.assigns.champion
|> Accounts.change_champion(champion_params)
|> Map.put(:action, :validate)
{:noreply, assign_form(socket, changeset)}
end
def handle_event("save", %{"champion" => champion_params}, socket) do
save_champion(socket, socket.assigns.action, champion_params)
end
defp save_champion(socket, :edit, champion_params) do
case Accounts.update_champion(socket.assigns.champion, champion_params) do
{:ok, champion} ->
notify_parent({:saved, champion})
{:noreply,
socket
|> put_flash(:info, "Champion updated successfully")
|> push_patch(to: socket.assigns.patch)}
{:error, %Ecto.Changeset{} = changeset} ->
{:noreply, assign_form(socket, changeset)}
end
end
defp save_champion(socket, :new, champion_params) do
case Accounts.create_champion(champion_params) do
{:ok, champion} ->
notify_parent({:saved, champion})
{:noreply,
socket
|> put_flash(:info, "Champion created successfully")
|> push_patch(to: socket.assigns.patch)}
{:error, %Ecto.Changeset{} = changeset} ->
{:noreply, assign_form(socket, changeset)}
end
end
defp assign_form(socket, %Ecto.Changeset{} = changeset) do
assign(socket, :form, to_form(changeset))
end
defp notify_parent(msg), do: send(self(), {__MODULE__, msg})
end

View File

@ -0,0 +1,80 @@
defmodule LoLAnalyticsWeb.ChampionLive.Index do
alias ElixirLS.LanguageServer.Providers.Completion.Reducers.Struct
use LoLAnalyticsWeb, :live_view
defstruct id: "", win_rate: 0, wins: 0, image: "", name: ""
@roles ["all", "TOP", "JUNGLE", "MIDDLE", "BOTTOM", "UTILITY"]
@impl true
def mount(_params, _session, socket) do
champs = LolAnalytics.Facts.ChampionPlayedGame.ChampionPlayedGameRepo.get_win_rates2()
mapped =
champs
|> Enum.map(fn champ ->
Kernel.struct!(%__MODULE__{}, champ)
end)
|> Enum.sort(&(&1.win_rate >= &2.win_rate))
{:ok,
stream(
socket,
:champions,
mapped
)
|> assign(:form, to_form(%{"name" => ""}))}
end
def handle_event("filter", params, socket) do
%{"name" => query_name} = params
champs =
LolAnalytics.Facts.ChampionPlayedGame.ChampionPlayedGameRepo.get_win_rates2()
|> Enum.filter(fn %{name: name} ->
String.downcase(name) |> String.contains?(query_name)
end)
|> Enum.map(fn champ ->
Kernel.struct!(%__MODULE__{}, champ)
end)
|> Enum.sort(&(&1.win_rate >= &2.win_rate))
IO.inspect(champs)
{:noreply,
stream(
socket,
:champions,
champs
)}
end
def handle_event("filter_by", %{"name" => name}, socket) do
{:noreply, assign(socket, %{filter_by: name})}
end
@impl true
def handle_params(params, _url, socket) do
{:noreply, apply_action(socket, socket.assigns.live_action, params)}
end
defp apply_action(socket, :index, _params) do
socket
|> assign(:page_title, "Listing Champions")
# |> assign(:champion, nil)
end
@impl true
def handle_info({LoLAnalyticsWeb.ChampionLive.FormComponent, {:saved, champion}}, socket) do
{:noreply, stream_insert(socket, :champions, champion)}
end
# @impl true
# def handle_event("delete", %{"id" => id}, socket) do
# champion = Accounts.get_champion!(id)
# {:ok, _} = Accounts.delete_champion(champion)
# {:noreply, stream_delete(socket, :champions, champion)}
# end
end

View File

@ -0,0 +1,66 @@
<.header>
Listing Champions
</.header>
<h1>Champions</h1>
<.form for={@form} phx-change="filter" phx-submit="save">
<.input type="text" field={@form["name"]} />
<.input type="checkbox" field={@form["all_roles"]} />
<div class="flex flex-col justify-between">
<div class="flex flex-row gap-2">
<.input type="checkbox" field={@form["top_role"]} />
<p>Top</p>
</div>
<div class="flex flex-row">
<.input type="checkbox" field={@form["middle_role"]} />
<p>Middle</p>
</div>
<div class="flex flex-row">
<.input type="checkbox" field={@form["jungle_role"]} />
<p>Jungle</p>
</div>
<div class="flex flex-row">
<.input type="checkbox" field={@form["bottom_role"]} />
<p>Bottom</p>
</div>
<div class="flex flex-row">
<.input type="checkbox" field={@form["utility_role"]} />
<p>Utility</p>
</div>
</div>
<button>Save</button>
</.form>
<div id="champions" class="grid grid-cols-4 gap-4">
<%= for {_, champion} <- @streams.champions do %>
<div class="flex flex-col">
<img src={"https://ddragon.leagueoflegends.com/cdn/14.11.1/img/champion/#{champion.image}"} />
<div class="flex-auto flex-col">
<p><%= champion.name %></p>
<p><%= champion.id %></p>
<p><%= champion.win_rate %>%</p>
</div>
</div>
<div class="sr-only">
<.link navigate={~p"/champions/#{champion}"}>Show</.link>
</div>
<% end %>
</div>
<.modal
:if={@live_action in [:new, :edit]}
id="champion-modal"
show
on_cancel={JS.patch(~p"/champions")}
>
<.live_component
module={LoLAnalyticsWeb.ChampionLive.FormComponent}
id={@champion.champion_id || :new}
title={@page_title}
action={@live_action}
champion={@champion}
patch={~p"/champions"}
/>
</.modal>

View File

@ -0,0 +1,21 @@
defmodule LoLAnalyticsWeb.ChampionLive.Show do
use LoLAnalyticsWeb, :live_view
alias LoLAnalytics.Accounts
@impl true
def mount(_params, _session, socket) do
{:ok, socket}
end
@impl true
def handle_params(%{"id" => id}, _, socket) do
{:noreply,
socket
|> assign(:page_title, page_title(socket.assigns.live_action))
|> assign(:champion, Accounts.get_champion!(id))}
end
defp page_title(:show), do: "Show Champion"
defp page_title(:edit), do: "Edit Champion"
end

View File

@ -0,0 +1,25 @@
<.header>
Champion <%= @champion.id %>
<:subtitle>This is a champion record from your database.</:subtitle>
<:actions>
<.link patch={~p"/champions/#{@champion}/show/edit"} phx-click={JS.push_focus()}>
<.button>Edit champion</.button>
</.link>
</:actions>
</.header>
<.list>
</.list>
<.back navigate={~p"/champions"}>Back to champions</.back>
<.modal :if={@live_action == :edit} id="champion-modal" show on_cancel={JS.patch(~p"/champions/#{@champion}")}>
<.live_component
module={LoLAnalyticsWeb.ChampionLive.FormComponent}
id={@champion.id}
title={@page_title}
action={@live_action}
champion={@champion}
patch={~p"/champions/#{@champion}"}
/>
</.modal>

View File

@ -0,0 +1,90 @@
defmodule LoLAnalyticsWeb.RoleLive.FormComponent do
use LoLAnalyticsWeb, :live_component
alias LoLAnalytics.Accounts
@impl true
def render(assigns) do
~H"""
<div>
<.header>
<%= @title %>
<:subtitle>Use this form to manage role records in your database.</:subtitle>
</.header>
<.simple_form
for={@form}
id="role-form"
phx-target={@myself}
phx-change="validate"
phx-submit="save"
>
<:actions>
<.button phx-disable-with="Saving...">Save Role</.button>
</:actions>
</.simple_form>
</div>
"""
end
@impl true
def update(%{role: role} = assigns, socket) do
changeset = Accounts.change_role(role)
{:ok,
socket
|> assign(assigns)
|> assign_form(changeset)}
end
@impl true
def handle_event("validate", %{"role" => role_params}, socket) do
changeset =
socket.assigns.role
|> Accounts.change_role(role_params)
|> Map.put(:action, :validate)
{:noreply, assign_form(socket, changeset)}
end
def handle_event("save", %{"role" => role_params}, socket) do
save_role(socket, socket.assigns.action, role_params)
end
defp save_role(socket, :edit, role_params) do
case Accounts.update_role(socket.assigns.role, role_params) do
{:ok, role} ->
notify_parent({:saved, role})
{:noreply,
socket
|> put_flash(:info, "Role updated successfully")
|> push_patch(to: socket.assigns.patch)}
{:error, %Ecto.Changeset{} = changeset} ->
{:noreply, assign_form(socket, changeset)}
end
end
defp save_role(socket, :new, role_params) do
case Accounts.create_role(role_params) do
{:ok, role} ->
notify_parent({:saved, role})
{:noreply,
socket
|> put_flash(:info, "Role created successfully")
|> push_patch(to: socket.assigns.patch)}
{:error, %Ecto.Changeset{} = changeset} ->
{:noreply, assign_form(socket, changeset)}
end
end
defp assign_form(socket, %Ecto.Changeset{} = changeset) do
assign(socket, :form, to_form(changeset))
end
defp notify_parent(msg), do: send(self(), {__MODULE__, msg})
end

View File

@ -0,0 +1,37 @@
defmodule LoLAnalyticsWeb.RoleLive.Index do
use LoLAnalyticsWeb, :live_view
alias LoLAnalytics.Accounts
alias LoLAnalytics.Accounts.Role
@roles ["ALL", "TOP", "MIDDLE", "JUNGLE", "UTILITY", "BOTTOM"]
@impl true
def mount(_params, _session, socket) do
{:ok, stream(socket, :roles, @roles)}
end
@impl true
def handle_params(params, _url, socket) do
{:noreply, apply_action(socket, socket.assigns.live_action, params)}
end
defp apply_action(socket, :index, _params) do
socket
|> assign(:page_title, "Listing Roles")
|> assign(:role, nil)
end
@impl true
def handle_info({LoLAnalyticsWeb.RoleLive.FormComponent, {:saved, role}}, socket) do
{:noreply, stream_insert(socket, :roles, role)}
end
# @impl true
# def handle_event("delete", %{"id" => id}, socket) do
# role = Accounts.get_role!(id)
# {:ok, _} = Accounts.delete_role(role)
# {:noreply, stream_delete(socket, :roles, role)}
# end
end

View File

@ -0,0 +1,40 @@
<.header>
Listing Roles
<:actions>
<.link patch={~p"/roles/new"}>
<.button>New Role</.button>
</.link>
</:actions>
</.header>
<.table
id="roles"
rows={@streams.roles}
row_click={fn {_id, role} -> JS.navigate(~p"/roles/#{role}") end}
>
<:action :let={{_id, role}}>
<div class="sr-only">
<.link navigate={~p"/roles/#{role}"}>Show</.link>
</div>
<.link patch={~p"/roles/#{role}/edit"}>Edit</.link>
</:action>
<:action :let={{id, role}}>
<.link
phx-click={JS.push("delete", value: %{id: role.id}) |> hide("##{id}")}
data-confirm="Are you sure?"
>
Delete
</.link>
</:action>
</.table>
<.modal :if={@live_action in [:new, :edit]} id="role-modal" show on_cancel={JS.patch(~p"/roles")}>
<.live_component
module={LoLAnalyticsWeb.RoleLive.FormComponent}
id={@role.id || :new}
title={@page_title}
action={@live_action}
role={@role}
patch={~p"/roles"}
/>
</.modal>

View File

@ -0,0 +1,21 @@
defmodule LoLAnalyticsWeb.RoleLive.Show do
use LoLAnalyticsWeb, :live_view
alias LoLAnalytics.Accounts
@impl true
def mount(_params, _session, socket) do
{:ok, socket}
end
@impl true
def handle_params(%{"id" => id}, _, socket) do
{:noreply,
socket
|> assign(:page_title, page_title(socket.assigns.live_action))
|> assign(:role, Accounts.get_role!(id))}
end
defp page_title(:show), do: "Show Role"
defp page_title(:edit), do: "Edit Role"
end

View File

@ -0,0 +1,25 @@
<.header>
Role <%= @role.id %>
<:subtitle>This is a role record from your database.</:subtitle>
<:actions>
<.link patch={~p"/roles/#{@role}/show/edit"} phx-click={JS.push_focus()}>
<.button>Edit role</.button>
</.link>
</:actions>
</.header>
<.list>
</.list>
<.back navigate={~p"/roles"}>Back to roles</.back>
<.modal :if={@live_action == :edit} id="role-modal" show on_cancel={JS.patch(~p"/roles/#{@role}")}>
<.live_component
module={LoLAnalyticsWeb.RoleLive.FormComponent}
id={@role.id}
title={@page_title}
action={@live_action}
role={@role}
patch={~p"/roles/#{@role}"}
/>
</.modal>

View File

@ -18,6 +18,12 @@ defmodule LoLAnalyticsWeb.Router do
pipe_through :browser pipe_through :browser
get "/", PageController, :home get "/", PageController, :home
live "/champions", ChampionLive.Index, :index
live "/champions/new", ChampionLive.Index, :new
live "/champions/:id/edit", ChampionLive.Index, :edit
live "/champions/:id", ChampionLive.Show, :show
live "/champions/:id/show/edit", ChampionLive.Show, :edit
end end
# Other scopes may use custom stacks. # Other scopes may use custom stacks.

View File

@ -0,0 +1,108 @@
defmodule LoLAnalyticsWeb.ChampionLiveTest do
use LoLAnalyticsWeb.ConnCase
import Phoenix.LiveViewTest
import LoLAnalytics.AccountsFixtures
@create_attrs %{}
@update_attrs %{}
@invalid_attrs %{}
defp create_champion(_) do
champion = champion_fixture()
%{champion: champion}
end
describe "Index" do
setup [:create_champion]
test "lists all champions", %{conn: conn} do
{:ok, _index_live, html} = live(conn, ~p"/champions")
assert html =~ "Listing Champions"
end
test "saves new champion", %{conn: conn} do
{:ok, index_live, _html} = live(conn, ~p"/champions")
assert index_live |> element("a", "New Champion") |> render_click() =~
"New Champion"
assert_patch(index_live, ~p"/champions/new")
assert index_live
|> form("#champion-form", champion: @invalid_attrs)
|> render_change() =~ "can&#39;t be blank"
assert index_live
|> form("#champion-form", champion: @create_attrs)
|> render_submit()
assert_patch(index_live, ~p"/champions")
html = render(index_live)
assert html =~ "Champion created successfully"
end
test "updates champion in listing", %{conn: conn, champion: champion} do
{:ok, index_live, _html} = live(conn, ~p"/champions")
assert index_live |> element("#champions-#{champion.id} a", "Edit") |> render_click() =~
"Edit Champion"
assert_patch(index_live, ~p"/champions/#{champion}/edit")
assert index_live
|> form("#champion-form", champion: @invalid_attrs)
|> render_change() =~ "can&#39;t be blank"
assert index_live
|> form("#champion-form", champion: @update_attrs)
|> render_submit()
assert_patch(index_live, ~p"/champions")
html = render(index_live)
assert html =~ "Champion updated successfully"
end
test "deletes champion in listing", %{conn: conn, champion: champion} do
{:ok, index_live, _html} = live(conn, ~p"/champions")
assert index_live |> element("#champions-#{champion.id} a", "Delete") |> render_click()
refute has_element?(index_live, "#champions-#{champion.id}")
end
end
describe "Show" do
setup [:create_champion]
test "displays champion", %{conn: conn, champion: champion} do
{:ok, _show_live, html} = live(conn, ~p"/champions/#{champion}")
assert html =~ "Show Champion"
end
test "updates champion within modal", %{conn: conn, champion: champion} do
{:ok, show_live, _html} = live(conn, ~p"/champions/#{champion}")
assert show_live |> element("a", "Edit") |> render_click() =~
"Edit Champion"
assert_patch(show_live, ~p"/champions/#{champion}/show/edit")
assert show_live
|> form("#champion-form", champion: @invalid_attrs)
|> render_change() =~ "can&#39;t be blank"
assert show_live
|> form("#champion-form", champion: @update_attrs)
|> render_submit()
assert_patch(show_live, ~p"/champions/#{champion}")
html = render(show_live)
assert html =~ "Champion updated successfully"
end
end
end

View File

@ -0,0 +1,108 @@
defmodule LoLAnalyticsWeb.RoleLiveTest do
use LoLAnalyticsWeb.ConnCase
import Phoenix.LiveViewTest
import LoLAnalytics.AccountsFixtures
@create_attrs %{}
@update_attrs %{}
@invalid_attrs %{}
defp create_role(_) do
role = role_fixture()
%{role: role}
end
describe "Index" do
setup [:create_role]
test "lists all roles", %{conn: conn} do
{:ok, _index_live, html} = live(conn, ~p"/roles")
assert html =~ "Listing Roles"
end
test "saves new role", %{conn: conn} do
{:ok, index_live, _html} = live(conn, ~p"/roles")
assert index_live |> element("a", "New Role") |> render_click() =~
"New Role"
assert_patch(index_live, ~p"/roles/new")
assert index_live
|> form("#role-form", role: @invalid_attrs)
|> render_change() =~ "can&#39;t be blank"
assert index_live
|> form("#role-form", role: @create_attrs)
|> render_submit()
assert_patch(index_live, ~p"/roles")
html = render(index_live)
assert html =~ "Role created successfully"
end
test "updates role in listing", %{conn: conn, role: role} do
{:ok, index_live, _html} = live(conn, ~p"/roles")
assert index_live |> element("#roles-#{role.id} a", "Edit") |> render_click() =~
"Edit Role"
assert_patch(index_live, ~p"/roles/#{role}/edit")
assert index_live
|> form("#role-form", role: @invalid_attrs)
|> render_change() =~ "can&#39;t be blank"
assert index_live
|> form("#role-form", role: @update_attrs)
|> render_submit()
assert_patch(index_live, ~p"/roles")
html = render(index_live)
assert html =~ "Role updated successfully"
end
test "deletes role in listing", %{conn: conn, role: role} do
{:ok, index_live, _html} = live(conn, ~p"/roles")
assert index_live |> element("#roles-#{role.id} a", "Delete") |> render_click()
refute has_element?(index_live, "#roles-#{role.id}")
end
end
describe "Show" do
setup [:create_role]
test "displays role", %{conn: conn, role: role} do
{:ok, _show_live, html} = live(conn, ~p"/roles/#{role}")
assert html =~ "Show Role"
end
test "updates role within modal", %{conn: conn, role: role} do
{:ok, show_live, _html} = live(conn, ~p"/roles/#{role}")
assert show_live |> element("a", "Edit") |> render_click() =~
"Edit Role"
assert_patch(show_live, ~p"/roles/#{role}/show/edit")
assert show_live
|> form("#role-form", role: @invalid_attrs)
|> render_change() =~ "can&#39;t be blank"
assert show_live
|> form("#role-form", role: @update_attrs)
|> render_submit()
assert_patch(show_live, ~p"/roles/#{role}")
html = render(show_live)
assert html =~ "Role updated successfully"
end
end
end

View File

@ -7,4 +7,16 @@ SELECT
FROM fact_champion_played_game FROM fact_champion_played_game
GROUP BY champion_id GROUP BY champion_id
ORDER BY win_rate desc; ORDER BY win_rate desc;
```
```
SELECT
(cast(count(CASE WHEN is_win THEN 1 END) as float) / cast(count(*) as float)) * 100.0 as win_rate,
count(CASE WHEN is_win THEN 1 END) as games_won,
count(*) as total_games,
champion_id,
team_position
FROM fact_champion_played_game
GROUP BY champion_id, team_position
ORDER BY win_rate desc;
``` ```