Merge branch 'main' into feature/223_memberfields_settings
All checks were successful
continuous-integration/drone/push Build is passing
All checks were successful
continuous-integration/drone/push Build is passing
This commit is contained in:
commit
922f9f93d0
19 changed files with 2939 additions and 356 deletions
|
|
@ -0,0 +1,87 @@
|
|||
defmodule Mv.Authorization.Checks.HasPermissionIntegrationTest do
|
||||
@moduledoc """
|
||||
Integration tests for HasPermission policy check.
|
||||
|
||||
These tests verify that the filter expressions generated by HasPermission
|
||||
have the correct structure for relationship-based filtering.
|
||||
|
||||
Note: Full integration tests with real queries require resources to have
|
||||
policies that use HasPermission. These tests validate filter expression
|
||||
structure and ensure the relationship paths are correct.
|
||||
"""
|
||||
use ExUnit.Case, async: true
|
||||
|
||||
alias Mv.Authorization.Checks.HasPermission
|
||||
|
||||
# Helper to create mock actor with role
|
||||
defp create_actor_with_role(permission_set_name) do
|
||||
%{
|
||||
id: "user-#{System.unique_integer([:positive])}",
|
||||
role: %{permission_set_name: permission_set_name}
|
||||
}
|
||||
end
|
||||
|
||||
describe "Filter Expression Structure - :linked scope" do
|
||||
test "Member filter uses user.id relationship path" do
|
||||
actor = create_actor_with_role("own_data")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
filter = HasPermission.auto_filter(actor, authorizer, [])
|
||||
|
||||
# Verify filter is not nil (should return a filter for :linked scope)
|
||||
assert not is_nil(filter)
|
||||
|
||||
# The filter should be a valid expression (keyword list or Ash.Expr)
|
||||
# We verify it's not nil and can be used in queries
|
||||
assert is_list(filter) or is_map(filter)
|
||||
end
|
||||
|
||||
test "CustomFieldValue filter uses member.user.id relationship path" do
|
||||
actor = create_actor_with_role("own_data")
|
||||
authorizer = create_authorizer(Mv.Membership.CustomFieldValue, :read)
|
||||
|
||||
filter = HasPermission.auto_filter(actor, authorizer, [])
|
||||
|
||||
# Verify filter is not nil
|
||||
assert not is_nil(filter)
|
||||
|
||||
# The filter should be a valid expression
|
||||
assert is_list(filter) or is_map(filter)
|
||||
end
|
||||
end
|
||||
|
||||
describe "Filter Expression Structure - :own scope" do
|
||||
test "User filter uses id == actor.id" do
|
||||
actor = create_actor_with_role("own_data")
|
||||
authorizer = create_authorizer(Mv.Accounts.User, :read)
|
||||
|
||||
filter = HasPermission.auto_filter(actor, authorizer, [])
|
||||
|
||||
# Verify filter is not nil (should return a filter for :own scope)
|
||||
assert not is_nil(filter)
|
||||
|
||||
# The filter should be a valid expression
|
||||
assert is_list(filter) or is_map(filter)
|
||||
end
|
||||
end
|
||||
|
||||
describe "Filter Expression Structure - :all scope" do
|
||||
test "Admin can read all members without filter" do
|
||||
actor = create_actor_with_role("admin")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
filter = HasPermission.auto_filter(actor, authorizer, [])
|
||||
|
||||
# :all scope should return nil (no filter needed)
|
||||
assert is_nil(filter)
|
||||
end
|
||||
end
|
||||
|
||||
# Helper to create a mock authorizer
|
||||
defp create_authorizer(resource, action) do
|
||||
%Ash.Policy.Authorizer{
|
||||
resource: resource,
|
||||
subject: %{action: %{name: action}}
|
||||
}
|
||||
end
|
||||
end
|
||||
264
test/mv/authorization/checks/has_permission_test.exs
Normal file
264
test/mv/authorization/checks/has_permission_test.exs
Normal file
|
|
@ -0,0 +1,264 @@
|
|||
defmodule Mv.Authorization.Checks.HasPermissionTest do
|
||||
@moduledoc """
|
||||
Tests for the HasPermission Ash Policy Check.
|
||||
|
||||
This check evaluates permissions from the PermissionSets module and applies
|
||||
scope filters to Ash queries.
|
||||
"""
|
||||
use ExUnit.Case, async: true
|
||||
|
||||
alias Mv.Authorization.Checks.HasPermission
|
||||
|
||||
# Helper to create a mock authorizer for strict_check/3
|
||||
defp create_authorizer(resource, action) do
|
||||
%Ash.Policy.Authorizer{
|
||||
resource: resource,
|
||||
subject: %{action: %{name: action}}
|
||||
}
|
||||
end
|
||||
|
||||
# Helper to create actor with role
|
||||
defp create_actor(id, permission_set_name) do
|
||||
%{
|
||||
id: id,
|
||||
role: %{permission_set_name: permission_set_name}
|
||||
}
|
||||
end
|
||||
|
||||
describe "describe/1" do
|
||||
test "returns human-readable description" do
|
||||
description = HasPermission.describe([])
|
||||
assert is_binary(description)
|
||||
assert description =~ "permission"
|
||||
end
|
||||
end
|
||||
|
||||
describe "strict_check/3 - Permission Lookup" do
|
||||
test "admin has permission for all resources/actions" do
|
||||
admin = create_actor("admin-123", "admin")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(admin, authorizer, [])
|
||||
|
||||
assert result == true or result == :unknown
|
||||
end
|
||||
|
||||
test "read_only has read permission for Member" do
|
||||
read_only_user = create_actor("read-only-123", "read_only")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(read_only_user, authorizer, [])
|
||||
|
||||
assert result == true or result == :unknown
|
||||
end
|
||||
|
||||
test "read_only does NOT have create permission for Member" do
|
||||
read_only_user = create_actor("read-only-123", "read_only")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :create)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(read_only_user, authorizer, [])
|
||||
|
||||
assert result == false
|
||||
end
|
||||
|
||||
test "own_data has update permission for User with scope :own" do
|
||||
own_data_user = create_actor("user-123", "own_data")
|
||||
authorizer = create_authorizer(Mv.Accounts.User, :update)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(own_data_user, authorizer, [])
|
||||
|
||||
# Should return :unknown for :own scope (needs filter)
|
||||
assert result == :unknown
|
||||
end
|
||||
end
|
||||
|
||||
describe "strict_check/3 - Scope :all" do
|
||||
test "actor with scope :all can access any record" do
|
||||
admin = create_actor("admin-123", "admin")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(admin, authorizer, [])
|
||||
|
||||
# :all scope should return true (no filter needed)
|
||||
assert result == true
|
||||
end
|
||||
|
||||
test "admin can read all members without filter" do
|
||||
admin = create_actor("admin-123", "admin")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(admin, authorizer, [])
|
||||
|
||||
# Should return true for :all scope
|
||||
assert result == true
|
||||
end
|
||||
end
|
||||
|
||||
describe "strict_check/3 - Scope :own" do
|
||||
test "actor with scope :own returns :unknown (needs filter)" do
|
||||
user = create_actor("user-123", "own_data")
|
||||
authorizer = create_authorizer(Mv.Accounts.User, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(user, authorizer, [])
|
||||
|
||||
# Should return :unknown for :own scope (needs filter via auto_filter)
|
||||
assert result == :unknown
|
||||
end
|
||||
end
|
||||
|
||||
describe "auto_filter/3 - Scope :own" do
|
||||
test "scope :own returns filter expression" do
|
||||
user = create_actor("user-123", "own_data")
|
||||
authorizer = create_authorizer(Mv.Accounts.User, :update)
|
||||
|
||||
filter = HasPermission.auto_filter(user, authorizer, [])
|
||||
|
||||
# Should return a filter expression
|
||||
assert not is_nil(filter)
|
||||
end
|
||||
end
|
||||
|
||||
describe "auto_filter/3 - Scope :linked" do
|
||||
test "scope :linked for Member returns user_id filter" do
|
||||
user = create_actor("user-123", "own_data")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
filter = HasPermission.auto_filter(user, authorizer, [])
|
||||
|
||||
# Should return a filter expression
|
||||
assert not is_nil(filter)
|
||||
end
|
||||
|
||||
test "scope :linked for CustomFieldValue returns member.user_id filter" do
|
||||
user = create_actor("user-123", "own_data")
|
||||
authorizer = create_authorizer(Mv.Membership.CustomFieldValue, :update)
|
||||
|
||||
filter = HasPermission.auto_filter(user, authorizer, [])
|
||||
|
||||
# Should return a filter expression that traverses member relationship
|
||||
assert not is_nil(filter)
|
||||
end
|
||||
end
|
||||
|
||||
describe "strict_check/3 - Error Handling" do
|
||||
test "returns {:ok, false} for nil actor" do
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(nil, authorizer, [])
|
||||
|
||||
assert result == false
|
||||
end
|
||||
|
||||
test "returns {:ok, false} for actor missing role" do
|
||||
actor_without_role = %{id: "user-123"}
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(actor_without_role, authorizer, [])
|
||||
|
||||
assert result == false
|
||||
end
|
||||
|
||||
test "returns {:ok, false} for actor with nil role" do
|
||||
actor_with_nil_role = %{id: "user-123", role: nil}
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(actor_with_nil_role, authorizer, [])
|
||||
|
||||
assert result == false
|
||||
end
|
||||
|
||||
test "returns {:ok, false} for invalid permission_set_name" do
|
||||
actor_with_invalid_permission = %{
|
||||
id: "user-123",
|
||||
role: %{permission_set_name: "invalid_set"}
|
||||
}
|
||||
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(actor_with_invalid_permission, authorizer, [])
|
||||
|
||||
assert result == false
|
||||
end
|
||||
|
||||
test "returns {:ok, false} for no matching permission" do
|
||||
read_only_user = create_actor("read-only-123", "read_only")
|
||||
authorizer = create_authorizer(Mv.Authorization.Role, :create)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(read_only_user, authorizer, [])
|
||||
|
||||
assert result == false
|
||||
end
|
||||
|
||||
test "handles role with nil permission_set_name gracefully" do
|
||||
actor_with_nil_permission_set = %{
|
||||
id: "user-123",
|
||||
role: %{permission_set_name: nil}
|
||||
}
|
||||
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(actor_with_nil_permission_set, authorizer, [])
|
||||
|
||||
assert result == false
|
||||
end
|
||||
end
|
||||
|
||||
describe "strict_check/3 - Logging" do
|
||||
import ExUnit.CaptureLog
|
||||
|
||||
test "logs authorization failure for nil actor" do
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
log =
|
||||
capture_log(fn ->
|
||||
HasPermission.strict_check(nil, authorizer, [])
|
||||
end)
|
||||
|
||||
assert log =~ "Authorization failed" or log == ""
|
||||
end
|
||||
|
||||
test "logs authorization failure for missing role" do
|
||||
actor_without_role = %{id: "user-123"}
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
log =
|
||||
capture_log(fn ->
|
||||
HasPermission.strict_check(actor_without_role, authorizer, [])
|
||||
end)
|
||||
|
||||
assert log =~ "Authorization failed" or log == ""
|
||||
end
|
||||
end
|
||||
|
||||
describe "strict_check/3 - Resource Name Extraction" do
|
||||
test "correctly extracts resource name from nested module" do
|
||||
admin = create_actor("admin-123", "admin")
|
||||
authorizer = create_authorizer(Mv.Membership.Member, :read)
|
||||
|
||||
{:ok, result} = HasPermission.strict_check(admin, authorizer, [])
|
||||
|
||||
# Should work correctly (not crash)
|
||||
assert result == true or result == :unknown or result == false
|
||||
end
|
||||
|
||||
test "works with different resource modules" do
|
||||
admin = create_actor("admin-123", "admin")
|
||||
|
||||
resources = [
|
||||
Mv.Accounts.User,
|
||||
Mv.Membership.Member,
|
||||
Mv.Membership.CustomFieldValue,
|
||||
Mv.Membership.CustomField,
|
||||
Mv.Authorization.Role
|
||||
]
|
||||
|
||||
for resource <- resources do
|
||||
authorizer = create_authorizer(resource, :read)
|
||||
{:ok, result} = HasPermission.strict_check(admin, authorizer, [])
|
||||
|
||||
# Should not crash and should return valid result
|
||||
assert result == true or result == :unknown or result == false
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
219
test/mv_web/authorization_test.exs
Normal file
219
test/mv_web/authorization_test.exs
Normal file
|
|
@ -0,0 +1,219 @@
|
|||
defmodule MvWeb.AuthorizationTest do
|
||||
@moduledoc """
|
||||
Tests for UI-level authorization helpers.
|
||||
"""
|
||||
use ExUnit.Case, async: true
|
||||
|
||||
alias MvWeb.Authorization
|
||||
alias Mv.Membership.Member
|
||||
alias Mv.Accounts.User
|
||||
|
||||
describe "can?/3 with resource atom" do
|
||||
test "returns true when user has permission for resource+action" do
|
||||
admin = %{
|
||||
id: "admin-123",
|
||||
role: %{permission_set_name: "admin"}
|
||||
}
|
||||
|
||||
assert Authorization.can?(admin, :create, Mv.Membership.Member) == true
|
||||
assert Authorization.can?(admin, :read, Mv.Membership.Member) == true
|
||||
assert Authorization.can?(admin, :update, Mv.Membership.Member) == true
|
||||
assert Authorization.can?(admin, :destroy, Mv.Membership.Member) == true
|
||||
end
|
||||
|
||||
test "returns false when user lacks permission" do
|
||||
read_only_user = %{
|
||||
id: "read-only-123",
|
||||
role: %{permission_set_name: "read_only"}
|
||||
}
|
||||
|
||||
assert Authorization.can?(read_only_user, :create, Mv.Membership.Member) == false
|
||||
assert Authorization.can?(read_only_user, :read, Mv.Membership.Member) == true
|
||||
assert Authorization.can?(read_only_user, :update, Mv.Membership.Member) == false
|
||||
assert Authorization.can?(read_only_user, :destroy, Mv.Membership.Member) == false
|
||||
end
|
||||
|
||||
test "returns false for nil user" do
|
||||
assert Authorization.can?(nil, :create, Mv.Membership.Member) == false
|
||||
assert Authorization.can?(nil, :read, Mv.Membership.Member) == false
|
||||
end
|
||||
|
||||
test "admin can manage roles" do
|
||||
admin = %{
|
||||
id: "admin-123",
|
||||
role: %{permission_set_name: "admin"}
|
||||
}
|
||||
|
||||
assert Authorization.can?(admin, :create, Mv.Authorization.Role) == true
|
||||
assert Authorization.can?(admin, :read, Mv.Authorization.Role) == true
|
||||
assert Authorization.can?(admin, :update, Mv.Authorization.Role) == true
|
||||
assert Authorization.can?(admin, :destroy, Mv.Authorization.Role) == true
|
||||
end
|
||||
|
||||
test "non-admin cannot manage roles" do
|
||||
normal_user = %{
|
||||
id: "normal-123",
|
||||
role: %{permission_set_name: "normal_user"}
|
||||
}
|
||||
|
||||
assert Authorization.can?(normal_user, :create, Mv.Authorization.Role) == false
|
||||
assert Authorization.can?(normal_user, :read, Mv.Authorization.Role) == false
|
||||
assert Authorization.can?(normal_user, :update, Mv.Authorization.Role) == false
|
||||
assert Authorization.can?(normal_user, :destroy, Mv.Authorization.Role) == false
|
||||
end
|
||||
end
|
||||
|
||||
describe "can?/3 with record struct - scope :all" do
|
||||
test "admin can update any member" do
|
||||
admin = %{
|
||||
id: "admin-123",
|
||||
role: %{permission_set_name: "admin"}
|
||||
}
|
||||
|
||||
member1 = %Member{id: "member-1", user: %User{id: "other-user"}}
|
||||
member2 = %Member{id: "member-2", user: %User{id: "another-user"}}
|
||||
|
||||
assert Authorization.can?(admin, :update, member1) == true
|
||||
assert Authorization.can?(admin, :update, member2) == true
|
||||
end
|
||||
|
||||
test "normal_user can update any member" do
|
||||
normal_user = %{
|
||||
id: "normal-123",
|
||||
role: %{permission_set_name: "normal_user"}
|
||||
}
|
||||
|
||||
member = %Member{id: "member-1", user: %User{id: "other-user"}}
|
||||
|
||||
assert Authorization.can?(normal_user, :update, member) == true
|
||||
end
|
||||
end
|
||||
|
||||
describe "can?/3 with record struct - scope :own" do
|
||||
test "user can update own User record" do
|
||||
user = %{
|
||||
id: "user-123",
|
||||
role: %{permission_set_name: "own_data"}
|
||||
}
|
||||
|
||||
own_user_record = %User{id: "user-123"}
|
||||
other_user_record = %User{id: "other-user"}
|
||||
|
||||
assert Authorization.can?(user, :update, own_user_record) == true
|
||||
assert Authorization.can?(user, :update, other_user_record) == false
|
||||
end
|
||||
end
|
||||
|
||||
describe "can?/3 with record struct - scope :linked" do
|
||||
test "user can update linked member" do
|
||||
user = %{
|
||||
id: "user-123",
|
||||
role: %{permission_set_name: "own_data"}
|
||||
}
|
||||
|
||||
# Member has_one :user (inverse relationship)
|
||||
linked_member = %Member{id: "member-1", user: %User{id: "user-123"}}
|
||||
unlinked_member = %Member{id: "member-2", user: nil}
|
||||
unlinked_member_other = %Member{id: "member-3", user: %User{id: "other-user"}}
|
||||
|
||||
assert Authorization.can?(user, :update, linked_member) == true
|
||||
assert Authorization.can?(user, :update, unlinked_member) == false
|
||||
assert Authorization.can?(user, :update, unlinked_member_other) == false
|
||||
end
|
||||
|
||||
test "user can update CustomFieldValue of linked member" do
|
||||
user = %{
|
||||
id: "user-123",
|
||||
role: %{permission_set_name: "own_data"}
|
||||
}
|
||||
|
||||
linked_cfv = %Mv.Membership.CustomFieldValue{
|
||||
id: "cfv-1",
|
||||
member: %Member{id: "member-1", user: %User{id: "user-123"}}
|
||||
}
|
||||
|
||||
unlinked_cfv = %Mv.Membership.CustomFieldValue{
|
||||
id: "cfv-2",
|
||||
member: %Member{id: "member-2", user: nil}
|
||||
}
|
||||
|
||||
unlinked_cfv_other = %Mv.Membership.CustomFieldValue{
|
||||
id: "cfv-3",
|
||||
member: %Member{id: "member-3", user: %User{id: "other-user"}}
|
||||
}
|
||||
|
||||
assert Authorization.can?(user, :update, linked_cfv) == true
|
||||
assert Authorization.can?(user, :update, unlinked_cfv) == false
|
||||
assert Authorization.can?(user, :update, unlinked_cfv_other) == false
|
||||
end
|
||||
end
|
||||
|
||||
describe "can_access_page?/2" do
|
||||
test "admin can access all pages via wildcard" do
|
||||
admin = %{
|
||||
id: "admin-123",
|
||||
role: %{permission_set_name: "admin"}
|
||||
}
|
||||
|
||||
assert Authorization.can_access_page?(admin, "/admin/roles") == true
|
||||
assert Authorization.can_access_page?(admin, "/members") == true
|
||||
assert Authorization.can_access_page?(admin, "/any/page") == true
|
||||
end
|
||||
|
||||
test "read_only user can access allowed pages" do
|
||||
read_only_user = %{
|
||||
id: "read-only-123",
|
||||
role: %{permission_set_name: "read_only"}
|
||||
}
|
||||
|
||||
assert Authorization.can_access_page?(read_only_user, "/") == true
|
||||
assert Authorization.can_access_page?(read_only_user, "/members") == true
|
||||
assert Authorization.can_access_page?(read_only_user, "/members/123") == true
|
||||
assert Authorization.can_access_page?(read_only_user, "/admin/roles") == false
|
||||
end
|
||||
|
||||
test "matches dynamic routes correctly" do
|
||||
read_only_user = %{
|
||||
id: "read-only-123",
|
||||
role: %{permission_set_name: "read_only"}
|
||||
}
|
||||
|
||||
assert Authorization.can_access_page?(read_only_user, "/members/123") == true
|
||||
assert Authorization.can_access_page?(read_only_user, "/members/abc") == true
|
||||
assert Authorization.can_access_page?(read_only_user, "/members/123/edit") == false
|
||||
end
|
||||
|
||||
test "returns false for nil user" do
|
||||
assert Authorization.can_access_page?(nil, "/members") == false
|
||||
assert Authorization.can_access_page?(nil, "/admin/roles") == false
|
||||
end
|
||||
end
|
||||
|
||||
describe "error handling" do
|
||||
test "user without role returns false" do
|
||||
user_without_role = %{id: "user-123", role: nil}
|
||||
|
||||
assert Authorization.can?(user_without_role, :create, Mv.Membership.Member) == false
|
||||
assert Authorization.can_access_page?(user_without_role, "/members") == false
|
||||
end
|
||||
|
||||
test "user with invalid permission_set_name returns false" do
|
||||
user_with_invalid_permission = %{
|
||||
id: "user-123",
|
||||
role: %{permission_set_name: "invalid_set"}
|
||||
}
|
||||
|
||||
assert Authorization.can?(user_with_invalid_permission, :create, Mv.Membership.Member) ==
|
||||
false
|
||||
|
||||
assert Authorization.can_access_page?(user_with_invalid_permission, "/members") == false
|
||||
end
|
||||
|
||||
test "handles missing fields gracefully" do
|
||||
user_missing_role = %{id: "user-123"}
|
||||
|
||||
assert Authorization.can?(user_missing_role, :create, Mv.Membership.Member) == false
|
||||
assert Authorization.can_access_page?(user_missing_role, "/members") == false
|
||||
end
|
||||
end
|
||||
end
|
||||
452
test/mv_web/live/role_live_test.exs
Normal file
452
test/mv_web/live/role_live_test.exs
Normal file
|
|
@ -0,0 +1,452 @@
|
|||
defmodule MvWeb.RoleLiveTest do
|
||||
@moduledoc """
|
||||
Tests for role management LiveViews.
|
||||
"""
|
||||
use MvWeb.ConnCase, async: false
|
||||
|
||||
import Phoenix.LiveViewTest
|
||||
|
||||
alias Mv.Authorization
|
||||
alias Mv.Authorization.Role
|
||||
|
||||
# Helper to create a role
|
||||
defp create_role(attrs \\ %{}) do
|
||||
default_attrs = %{
|
||||
name: "Test Role #{System.unique_integer([:positive])}",
|
||||
description: "Test description",
|
||||
permission_set_name: "read_only"
|
||||
}
|
||||
|
||||
attrs = Map.merge(default_attrs, attrs)
|
||||
|
||||
case Authorization.create_role(attrs) do
|
||||
{:ok, role} -> role
|
||||
{:error, error} -> raise "Failed to create role: #{inspect(error)}"
|
||||
end
|
||||
end
|
||||
|
||||
# Helper to create admin user with admin role
|
||||
defp create_admin_user(conn) do
|
||||
# Create admin role
|
||||
admin_role =
|
||||
case Authorization.list_roles() do
|
||||
{:ok, roles} ->
|
||||
case Enum.find(roles, &(&1.name == "Admin")) do
|
||||
nil ->
|
||||
# Create admin role if it doesn't exist
|
||||
create_role(%{
|
||||
name: "Admin",
|
||||
description: "Administrator with full access",
|
||||
permission_set_name: "admin"
|
||||
})
|
||||
|
||||
role ->
|
||||
role
|
||||
end
|
||||
|
||||
_ ->
|
||||
# Create admin role if list_roles fails
|
||||
create_role(%{
|
||||
name: "Admin",
|
||||
description: "Administrator with full access",
|
||||
permission_set_name: "admin"
|
||||
})
|
||||
end
|
||||
|
||||
# Create user
|
||||
{:ok, user} =
|
||||
Mv.Accounts.User
|
||||
|> Ash.Changeset.for_create(:register_with_password, %{
|
||||
email: "admin#{System.unique_integer([:positive])}@mv.local",
|
||||
password: "testpassword123"
|
||||
})
|
||||
|> Ash.create()
|
||||
|
||||
# Assign admin role using manage_relationship
|
||||
{:ok, user} =
|
||||
user
|
||||
|> Ash.Changeset.for_update(:update, %{})
|
||||
|> Ash.Changeset.manage_relationship(:role, admin_role, type: :append_and_remove)
|
||||
|> Ash.update()
|
||||
|
||||
# Load role for authorization checks (must be loaded for can?/3 to work)
|
||||
user_with_role = Ash.load!(user, :role, domain: Mv.Accounts)
|
||||
|
||||
# Store user with role in session for LiveView
|
||||
conn = conn_with_password_user(conn, user_with_role)
|
||||
{conn, user_with_role, admin_role}
|
||||
end
|
||||
|
||||
# Helper to create non-admin user
|
||||
defp create_non_admin_user(conn) do
|
||||
{:ok, user} =
|
||||
Mv.Accounts.User
|
||||
|> Ash.Changeset.for_create(:register_with_password, %{
|
||||
email: "user#{System.unique_integer([:positive])}@mv.local",
|
||||
password: "testpassword123"
|
||||
})
|
||||
|> Ash.create()
|
||||
|
||||
conn = conn_with_password_user(conn, user)
|
||||
{conn, user}
|
||||
end
|
||||
|
||||
describe "index page" do
|
||||
setup %{conn: conn} do
|
||||
{conn, user, _admin_role} = create_admin_user(conn)
|
||||
%{conn: conn, user: user}
|
||||
end
|
||||
|
||||
test "mounts successfully", %{conn: conn} do
|
||||
{:ok, _view, _html} = live(conn, "/admin/roles")
|
||||
end
|
||||
|
||||
test "loads all roles from database", %{conn: conn} do
|
||||
role1 = create_role(%{name: "Role 1"})
|
||||
role2 = create_role(%{name: "Role 2"})
|
||||
|
||||
{:ok, _view, html} = live(conn, "/admin/roles")
|
||||
|
||||
assert html =~ role1.name
|
||||
assert html =~ role2.name
|
||||
end
|
||||
|
||||
test "shows table with role names", %{conn: conn} do
|
||||
role = create_role(%{name: "Test Role"})
|
||||
|
||||
{:ok, _view, html} = live(conn, "/admin/roles")
|
||||
|
||||
assert html =~ role.name
|
||||
assert html =~ role.description
|
||||
assert html =~ role.permission_set_name
|
||||
end
|
||||
|
||||
test "shows system role badge", %{conn: conn} do
|
||||
_system_role =
|
||||
Role
|
||||
|> Ash.Changeset.for_create(:create_role, %{
|
||||
name: "System Role",
|
||||
permission_set_name: "own_data"
|
||||
})
|
||||
|> Ash.Changeset.force_change_attribute(:is_system_role, true)
|
||||
|> Ash.create!()
|
||||
|
||||
{:ok, _view, html} = live(conn, "/admin/roles")
|
||||
|
||||
assert html =~ "System Role" || html =~ "system"
|
||||
end
|
||||
|
||||
test "delete button disabled for system roles", %{conn: conn} do
|
||||
system_role =
|
||||
Role
|
||||
|> Ash.Changeset.for_create(:create_role, %{
|
||||
name: "System Role",
|
||||
permission_set_name: "own_data"
|
||||
})
|
||||
|> Ash.Changeset.force_change_attribute(:is_system_role, true)
|
||||
|> Ash.create!()
|
||||
|
||||
{:ok, view, _html} = live(conn, "/admin/roles")
|
||||
|
||||
assert has_element?(
|
||||
view,
|
||||
"button[phx-click='delete'][phx-value-id='#{system_role.id}'][disabled]"
|
||||
) ||
|
||||
not has_element?(
|
||||
view,
|
||||
"button[phx-click='delete'][phx-value-id='#{system_role.id}']"
|
||||
)
|
||||
end
|
||||
|
||||
test "delete button enabled for non-system roles", %{conn: conn} do
|
||||
role = create_role()
|
||||
|
||||
{:ok, view, html} = live(conn, "/admin/roles")
|
||||
|
||||
# Delete is a link with phx-click containing delete event
|
||||
# Check if delete link exists in HTML (phx-click contains delete and role id)
|
||||
assert (html =~ "phx-click" && html =~ "delete" && html =~ role.id) ||
|
||||
has_element?(view, "a[phx-click*='delete'][phx-value-id='#{role.id}']") ||
|
||||
has_element?(view, "a[aria-label='Delete role']")
|
||||
end
|
||||
|
||||
test "new role button navigates to form", %{conn: conn} do
|
||||
{:ok, view, html} = live(conn, "/admin/roles")
|
||||
|
||||
# Check if button exists (admin should see it)
|
||||
if html =~ "New Role" do
|
||||
{:error, {:live_redirect, %{to: to}}} =
|
||||
view
|
||||
|> element("a[href='/admin/roles/new'], button[href='/admin/roles/new']")
|
||||
|> render_click()
|
||||
|
||||
assert to == "/admin/roles/new"
|
||||
else
|
||||
# If button not visible, user doesn't have permission (expected for non-admin)
|
||||
# This test assumes admin user, so button should be visible
|
||||
flunk("New Role button not found - user may not have admin role loaded")
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe "show page" do
|
||||
setup %{conn: conn} do
|
||||
{conn, user, _admin_role} = create_admin_user(conn)
|
||||
%{conn: conn, user: user}
|
||||
end
|
||||
|
||||
test "mounts with valid role ID", %{conn: conn} do
|
||||
role = create_role()
|
||||
|
||||
{:ok, _view, html} = live(conn, "/admin/roles/#{role.id}")
|
||||
|
||||
assert html =~ role.name
|
||||
assert html =~ role.description
|
||||
assert html =~ role.permission_set_name
|
||||
end
|
||||
|
||||
test "returns 404 for invalid role ID", %{conn: conn} do
|
||||
invalid_id = Ecto.UUID.generate()
|
||||
|
||||
# Should redirect to index with error message
|
||||
# redirect in mount returns {:error, {:redirect, ...}}
|
||||
result = live(conn, "/admin/roles/#{invalid_id}")
|
||||
|
||||
assert match?({:error, {:redirect, %{to: "/admin/roles"}}}, result)
|
||||
end
|
||||
|
||||
test "shows system role badge if is_system_role is true", %{conn: conn} do
|
||||
system_role =
|
||||
Role
|
||||
|> Ash.Changeset.for_create(:create_role, %{
|
||||
name: "System Role",
|
||||
permission_set_name: "own_data"
|
||||
})
|
||||
|> Ash.Changeset.force_change_attribute(:is_system_role, true)
|
||||
|> Ash.create!()
|
||||
|
||||
{:ok, _view, html} = live(conn, "/admin/roles/#{system_role.id}")
|
||||
|
||||
assert html =~ "System Role" || html =~ "system"
|
||||
end
|
||||
end
|
||||
|
||||
describe "form - create" do
|
||||
setup %{conn: conn} do
|
||||
{conn, user, _admin_role} = create_admin_user(conn)
|
||||
%{conn: conn, user: user}
|
||||
end
|
||||
|
||||
test "mounts successfully", %{conn: conn} do
|
||||
{:ok, _view, _html} = live(conn, "/admin/roles/new")
|
||||
end
|
||||
|
||||
test "form dropdown shows all 4 permission sets", %{conn: conn} do
|
||||
{:ok, _view, html} = live(conn, "/admin/roles/new")
|
||||
|
||||
assert html =~ "own_data"
|
||||
assert html =~ "read_only"
|
||||
assert html =~ "normal_user"
|
||||
assert html =~ "admin"
|
||||
end
|
||||
|
||||
test "creates new role with valid data", %{conn: conn} do
|
||||
{:ok, view, _html} = live(conn, "/admin/roles/new")
|
||||
|
||||
attrs = %{
|
||||
"name" => "New Role",
|
||||
"description" => "New description",
|
||||
"permission_set_name" => "read_only"
|
||||
}
|
||||
|
||||
view
|
||||
|> form("#role-form", role: attrs)
|
||||
|> render_submit()
|
||||
|
||||
# Should redirect to index or show page
|
||||
assert_redirect(view, "/admin/roles")
|
||||
end
|
||||
|
||||
test "shows error with invalid permission_set_name", %{conn: conn} do
|
||||
{:ok, view, _html} = live(conn, "/admin/roles/new")
|
||||
|
||||
# Try to submit with empty permission_set_name (invalid)
|
||||
attrs = %{
|
||||
"name" => "New Role",
|
||||
"description" => "New description",
|
||||
"permission_set_name" => ""
|
||||
}
|
||||
|
||||
view
|
||||
|> form("#role-form", role: attrs)
|
||||
|> render_submit()
|
||||
|
||||
# Should show validation error
|
||||
html = render(view)
|
||||
assert html =~ "error" || html =~ "required" || html =~ "Permission Set"
|
||||
end
|
||||
|
||||
test "shows flash message after successful creation", %{conn: conn} do
|
||||
{:ok, view, _html} = live(conn, "/admin/roles/new")
|
||||
|
||||
attrs = %{
|
||||
"name" => "New Role #{System.unique_integer([:positive])}",
|
||||
"description" => "New description",
|
||||
"permission_set_name" => "read_only"
|
||||
}
|
||||
|
||||
view
|
||||
|> form("#role-form", role: attrs)
|
||||
|> render_submit()
|
||||
|
||||
# Should redirect to index
|
||||
assert_redirect(view, "/admin/roles")
|
||||
end
|
||||
end
|
||||
|
||||
describe "form - edit" do
|
||||
setup %{conn: conn} do
|
||||
{conn, user, _admin_role} = create_admin_user(conn)
|
||||
role = create_role()
|
||||
%{conn: conn, user: user, role: role}
|
||||
end
|
||||
|
||||
test "mounts with valid role ID", %{conn: conn, role: role} do
|
||||
{:ok, _view, html} = live(conn, "/admin/roles/#{role.id}/edit")
|
||||
|
||||
assert html =~ role.name
|
||||
end
|
||||
|
||||
test "returns 404 for invalid role ID in edit", %{conn: conn} do
|
||||
invalid_id = Ecto.UUID.generate()
|
||||
|
||||
# Should redirect to index with error message
|
||||
# redirect in mount returns {:error, {:redirect, ...}}
|
||||
result = live(conn, "/admin/roles/#{invalid_id}/edit")
|
||||
|
||||
assert match?({:error, {:redirect, %{to: "/admin/roles"}}}, result)
|
||||
end
|
||||
|
||||
test "updates role name", %{conn: conn, role: role} do
|
||||
{:ok, view, _html} = live(conn, "/admin/roles/#{role.id}/edit?return_to=show")
|
||||
|
||||
attrs = %{
|
||||
"name" => "Updated Role Name",
|
||||
"description" => role.description,
|
||||
"permission_set_name" => role.permission_set_name
|
||||
}
|
||||
|
||||
view
|
||||
|> form("#role-form", role: attrs)
|
||||
|> render_submit()
|
||||
|
||||
assert_redirect(view, "/admin/roles/#{role.id}")
|
||||
|
||||
# Verify update
|
||||
{:ok, updated_role} = Authorization.get_role(role.id)
|
||||
assert updated_role.name == "Updated Role Name"
|
||||
end
|
||||
|
||||
test "updates system role's permission_set_name", %{conn: conn} do
|
||||
system_role =
|
||||
Role
|
||||
|> Ash.Changeset.for_create(:create_role, %{
|
||||
name: "System Role",
|
||||
permission_set_name: "own_data"
|
||||
})
|
||||
|> Ash.Changeset.force_change_attribute(:is_system_role, true)
|
||||
|> Ash.create!()
|
||||
|
||||
{:ok, view, _html} = live(conn, "/admin/roles/#{system_role.id}/edit?return_to=show")
|
||||
|
||||
attrs = %{
|
||||
"name" => system_role.name,
|
||||
"description" => system_role.description,
|
||||
"permission_set_name" => "read_only"
|
||||
}
|
||||
|
||||
view
|
||||
|> form("#role-form", role: attrs)
|
||||
|> render_submit()
|
||||
|
||||
assert_redirect(view, "/admin/roles/#{system_role.id}")
|
||||
|
||||
# Verify update
|
||||
{:ok, updated_role} = Authorization.get_role(system_role.id)
|
||||
assert updated_role.permission_set_name == "read_only"
|
||||
end
|
||||
end
|
||||
|
||||
describe "delete functionality" do
|
||||
setup %{conn: conn} do
|
||||
{conn, user, _admin_role} = create_admin_user(conn)
|
||||
%{conn: conn, user: user}
|
||||
end
|
||||
|
||||
test "deletes non-system role", %{conn: conn} do
|
||||
role = create_role()
|
||||
|
||||
{:ok, view, html} = live(conn, "/admin/roles")
|
||||
|
||||
# Delete is a link - JS.push creates phx-click with value containing id
|
||||
# Verify the role id is in the HTML (in phx-click value)
|
||||
assert html =~ role.id
|
||||
|
||||
# Send delete event directly to avoid selector issues with multiple delete buttons
|
||||
render_click(view, "delete", %{"id" => role.id})
|
||||
|
||||
# Verify deletion by checking database
|
||||
assert {:error, %Ash.Error.Invalid{errors: [%Ash.Error.Query.NotFound{}]}} =
|
||||
Authorization.get_role(role.id)
|
||||
end
|
||||
|
||||
test "fails to delete system role with error message", %{conn: conn} do
|
||||
system_role =
|
||||
Role
|
||||
|> Ash.Changeset.for_create(:create_role, %{
|
||||
name: "System Role",
|
||||
permission_set_name: "own_data"
|
||||
})
|
||||
|> Ash.Changeset.force_change_attribute(:is_system_role, true)
|
||||
|> Ash.create!()
|
||||
|
||||
{:ok, view, html} = live(conn, "/admin/roles")
|
||||
|
||||
# System role delete button should be disabled
|
||||
assert html =~ "disabled" || html =~ "cursor-not-allowed" ||
|
||||
html =~ "System roles cannot be deleted"
|
||||
|
||||
# Try to delete via event (backend check)
|
||||
render_click(view, "delete", %{"id" => system_role.id})
|
||||
|
||||
# Should show error message
|
||||
assert render(view) =~ "System roles cannot be deleted"
|
||||
|
||||
# Role should still exist
|
||||
{:ok, _role} = Authorization.get_role(system_role.id)
|
||||
end
|
||||
end
|
||||
|
||||
describe "authorization" do
|
||||
test "only admin can access /admin/roles", %{conn: conn} do
|
||||
{conn, _user} = create_non_admin_user(conn)
|
||||
|
||||
# Non-admin should be redirected or see error
|
||||
# Note: Authorization is checked via can_access_page? which returns false
|
||||
# The page might still mount but show no content or redirect
|
||||
# For now, we just verify the page doesn't work as expected for non-admin
|
||||
{:ok, _view, html} = live(conn, "/admin/roles")
|
||||
|
||||
# Non-admin should not see "New Role" button (can? returns false)
|
||||
# But the button might still be in HTML, just hidden or disabled
|
||||
# We verify that the page loads but admin features are restricted
|
||||
assert html =~ "Listing Roles" || html =~ "Roles"
|
||||
end
|
||||
|
||||
test "admin can access /admin/roles", %{conn: conn} do
|
||||
{conn, _user, _admin_role} = create_admin_user(conn)
|
||||
|
||||
{:ok, _view, _html} = live(conn, "/admin/roles")
|
||||
end
|
||||
end
|
||||
end
|
||||
Loading…
Add table
Add a link
Reference in a new issue