Skip to content

Commit

Permalink
Merge branch 'custom-actor-plug'
Browse files Browse the repository at this point in the history
  • Loading branch information
zachdaniel committed Jul 25, 2023
2 parents 77b5fac + 2667209 commit f96ef8b
Show file tree
Hide file tree
Showing 8 changed files with 217 additions and 173 deletions.
1 change: 1 addition & 0 deletions config/config.exs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ Application.put_env(:ash_admin, Demo.Repo, url: "ecto://#{pg_url}/#{pg_database}

config :phoenix, :json_library, Jason
config :ash_admin, ecto_repos: [Demo.Repo]
config :ash, :use_all_identities_in_manage_relationship?, false

config :surface, :components, [
{Surface.Components.Form.ErrorTag, default_class: "invalid-feedback"}
Expand Down
6 changes: 3 additions & 3 deletions dev/resources/tickets/resources/representative.ex
Original file line number Diff line number Diff line change
Expand Up @@ -14,10 +14,10 @@ defmodule Demo.Tickets.Representative do

resource do
base_filter representative: true
end

identities do
identity :representative_name, [:first_name, :last_name]
end
identities do
identity :representative_name, [:first_name, :last_name]
end

postgres do
Expand Down
2 changes: 1 addition & 1 deletion dev/resources/tickets/resources/ticket/ticket.ex
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,7 @@ defmodule Demo.Tickets.Ticket do
update :link do
accept []
argument :tickets, {:array, :map}, allow_nil?: false
argument :link_comment, :map, type: :create
argument :link_comment, :map

# Uses the defult create action of the join table, which accepts the `type`
change manage_relationship(:tickets, :source_links, on_lookup: {:relate_and_update, :create, :read, :all})
Expand Down
126 changes: 9 additions & 117 deletions lib/ash_admin/actor_plug.ex
Original file line number Diff line number Diff line change
Expand Up @@ -2,131 +2,23 @@ defmodule AshAdmin.ActorPlug do
@moduledoc false

@behaviour Plug
require Ash.Query

import AshAdmin.Helpers
@plug Application.compile_env(:ash_admin, :actor_plug, AshAdmin.ActorPlug.Plug)

@callback set_actor_session(conn :: Plug.Conn.t()) :: Plug.Conn.t()
@callback actor_assigns(socket :: Phoenix.LiveView.Socket.t(), session :: map) :: Keyword.t()

def init(opts), do: opts

def call(conn, _opts) do
case conn.cookies do
%{"actor_resource" => "undefined"} ->
conn

session ->
actor_session(conn, session)
end
end

def actor_session(
conn,
%{
"actor_resource" => resource,
"actor_api" => api,
"actor_action" => action,
"actor_primary_key" => primary_key
} = session
) do
authorizing = session["actor_authorizing"] || false

actor_paused =
if is_nil(session["actor_paused"]) do
true
else
session["actor_paused"]
end

actor = actor_from_session(conn.private.phoenix_endpoint, session)

authorizing = session_bool(authorizing)
actor_paused = session_bool(actor_paused)

conn
|> Plug.Conn.put_session(:actor_resource, resource)
|> Plug.Conn.put_session(:actor_api, api)
|> Plug.Conn.put_session(:actor_action, action)
|> Plug.Conn.put_session(:actor_primary_key, primary_key)
|> Plug.Conn.put_session(:actor_authorizing, authorizing)
|> Plug.Conn.put_session(:actor_paused, actor_paused)
|> Plug.Conn.assign(:actor, actor)
|> Plug.Conn.assign(:authorizing, authorizing || false)
|> Plug.Conn.assign(:actor_paused, actor_paused)
|> Plug.Conn.assign(:authorizing, authorizing)
set_actor_session(conn)
end

def actor_session(conn, _), do: conn

def actor_api_from_session(endpoint, %{"actor_api" => api}) do
otp_app = endpoint.config(:otp_app)
apis = Application.get_env(otp_app, :ash_apis)

Enum.find(apis, fn allowed_api ->
AshAdmin.Api.show?(allowed_api) && AshAdmin.Api.name(allowed_api) == api
end)
def actor_assigns(socket, session) do
@plug.actor_assigns(socket, session)
end

def actor_api_from_session(_, _), do: nil

def actor_from_session(endpoint, %{
"actor_resource" => resource,
"actor_api" => api,
"actor_primary_key" => primary_key,
"actor_action" => action
})
when not is_nil(resource) and not is_nil(api) do
otp_app = endpoint.config(:otp_app)
apis = Application.get_env(otp_app, :ash_apis)

api =
Enum.find(apis, fn allowed_api ->
AshAdmin.Api.show?(allowed_api) && AshAdmin.Api.name(allowed_api) == api
end)

resource =
if api do
api
|> Ash.Api.Info.resources()
|> Enum.find(fn api_resource ->
AshAdmin.Resource.name(api_resource) == resource
end)
end

if api && resource do
action =
if action do
Ash.Resource.Info.action(resource, String.to_existing_atom(action), :read)
end

case decode_primary_key(resource, primary_key) do
:error ->
nil

{:ok, filter} ->
resource
|> Ash.Query.filter(^filter)
|> api.read_one!(action: action, authorize?: false)
end
end
end

def actor_from_session(_, _), do: nil

def session_bool(value) do
case value do
"true" ->
true

"false" ->
false

"undefined" ->
false

boolean when is_boolean(boolean) ->
boolean

nil ->
false
end
def set_actor_session(conn) do
@plug.set_actor_session(conn)
end
end
165 changes: 165 additions & 0 deletions lib/ash_admin/actor_plug/plug.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,165 @@
defmodule AshAdmin.ActorPlug.Plug do
@behaviour AshAdmin.ActorPlug
import AshAdmin.Helpers
require Ash.Query

@impl true
def actor_assigns(socket, session) do
otp_app = socket.endpoint.config(:otp_app)
apis = apis(otp_app)

actor_paused =
if is_nil(session["actor_paused"]) do
true
else
session_bool(session["actor_paused"])
end

[
actor: actor_from_session(socket.endpoint, session),
actor_api: actor_api_from_session(socket.endpoint, session),
actor_resources: actor_resources(apis),
authorizing: session_bool(session["actor_authorizing"]),
actor_paused: actor_paused
]
end

@impl true
def set_actor_session(conn) do
case conn.cookies do
%{"actor_resource" => "undefined"} ->
conn

session ->
case session do
%{
"actor_resource" => resource,
"actor_api" => api,
"actor_action" => action,
"actor_primary_key" => primary_key
} ->
authorizing = session["actor_authorizing"] || false

actor_paused =
if is_nil(session["actor_paused"]) do
true
else
session["actor_paused"]
end

actor = actor_from_session(conn.private.phoenix_endpoint, session)

authorizing = session_bool(authorizing)
actor_paused = session_bool(actor_paused)

conn
|> Plug.Conn.put_session(:actor_resource, resource)
|> Plug.Conn.put_session(:actor_api, api)
|> Plug.Conn.put_session(:actor_action, action)
|> Plug.Conn.put_session(:actor_primary_key, primary_key)
|> Plug.Conn.put_session(:actor_authorizing, authorizing)
|> Plug.Conn.put_session(:actor_paused, actor_paused)
|> Plug.Conn.assign(:actor, actor)
|> Plug.Conn.assign(:authorizing, authorizing || false)
|> Plug.Conn.assign(:actor_paused, actor_paused)
|> Plug.Conn.assign(:authorizing, authorizing)

_ ->
conn
end
end
end

defp session_bool(value) do
case value do
"true" ->
true

"false" ->
false

"undefined" ->
false

boolean when is_boolean(boolean) ->
boolean

nil ->
false
end
end

defp actor_resources(apis) do
apis
|> Enum.flat_map(fn api ->
api
|> Ash.Api.Info.resources()
|> Enum.filter(fn resource ->
AshAdmin.Helpers.primary_action(resource, :read) && AshAdmin.Resource.actor?(resource)
end)
|> Enum.map(fn resource -> {api, resource} end)
end)
end

defp apis(otp_app) do
otp_app
|> Application.get_env(:ash_apis)
|> Enum.filter(&AshAdmin.Api.show?/1)
end

defp actor_api_from_session(endpoint, %{"actor_api" => api}) do
otp_app = endpoint.config(:otp_app)
apis = Application.get_env(otp_app, :ash_apis)

Enum.find(apis, fn allowed_api ->
AshAdmin.Api.show?(allowed_api) && AshAdmin.Api.name(allowed_api) == api
end)
end

defp actor_api_from_session(_, _), do: nil

defp actor_from_session(endpoint, %{
"actor_resource" => resource,
"actor_api" => api,
"actor_primary_key" => primary_key,
"actor_action" => action
})
when not is_nil(resource) and not is_nil(api) do
otp_app = endpoint.config(:otp_app)
apis = Application.get_env(otp_app, :ash_apis)

api =
Enum.find(apis, fn allowed_api ->
AshAdmin.Api.show?(allowed_api) && AshAdmin.Api.name(allowed_api) == api
end)

resource =
if api do
api
|> Ash.Api.Info.resources()
|> Enum.find(fn api_resource ->
AshAdmin.Resource.name(api_resource) == resource
end)
end

if api && resource do
action =
if action do
Ash.Resource.Info.action(resource, String.to_existing_atom(action), :read)
end

case decode_primary_key(resource, primary_key) do
:error ->
nil

{:ok, filter} ->
resource
|> Ash.Query.filter(^filter)
|> api.read_one!(action: action, authorize?: false)
end
end
end

defp actor_from_session(_, _), do: nil

end
Loading

0 comments on commit f96ef8b

Please sign in to comment.