9. 数据访问控制
到目前为止,我们都是用 admin 用户登录应用程序,对 UI 和数据具有完全的访问权限。这最后一节中,我们将为 HR 经理和员工设置不同访问应用程序的权限。
员工的资源角色
创建资源角色
在 Jmix 工具窗口中,点击 New()→ Resource Role:
data:image/s3,"s3://crabby-images/9c750/9c750a8b46933e2833c8ed5ed6dbecdd27bebf15" alt="employee role 1"
New Resource Role 弹窗中,Role name 字段输入 Employee
并在 Security scope 下拉框选择 UI
:
data:image/s3,"s3://crabby-images/fa48f/fa48f1841f3431c730ae61d8a959c4e8b79868a2" alt="employee role 2"
点击 OK。
Studio 会创建并打开一个带注解的接口:
package com.company.onboarding.security;
import io.jmix.security.role.annotation.ResourceRole;
@ResourceRole(name = "Employee", code = "employee", scope = "UI")
public interface EmployeeRole {
}
UI 权限范围(scope)仅在用户通过 UI 登录系统后应用。如果同一个用户是通过 REST API 登录,则不会使用这个角色。建议为 API 创建一组不同的角色,许可范围相对少一些。
|
切换至 User Interface 标签页定义界面许可。在菜单树中,选中 MyOnboardingScreen
,然后在右侧勾选 Allow 复选框:
data:image/s3,"s3://crabby-images/d8795/d87951cb58c1945125792ffc0d21403cbd79b3d3" alt="employee role 3"
然后切换至 Entities 标签页并选择下列权限:
data:image/s3,"s3://crabby-images/8bf7f/8bf7f797d1a180400d65a296da6c7298f0ceecaf" alt="employee role 4"
员工需要读取 Step
、User
和 UserStep
实体以便在 UI 进行查看,还需要能更新 User
和 UserStep
,标记步骤完成。
切回 Text 标签页。可以看到 Studio 在接口内生成了几个带注解的方法,分别对应于刚才我们设置的权限:
@ResourceRole(name = "Employee", code = "employee", scope = "UI")
public interface EmployeeRole {
@MenuPolicy(menuIds = "MyOnboardingScreen")
@ScreenPolicy(screenIds = "MyOnboardingScreen")
void screens();
@EntityAttributePolicy(entityClass = Step.class,
attributes = "*",
action = EntityAttributePolicyAction.VIEW)
@EntityPolicy(entityClass = Step.class,
actions = EntityPolicyAction.READ)
void step();
@EntityAttributePolicy(entityClass = User.class,
attributes = "*",
action = EntityAttributePolicyAction.VIEW)
@EntityPolicy(entityClass = User.class,
actions = {EntityPolicyAction.READ, EntityPolicyAction.UPDATE})
void user();
@EntityAttributePolicy(entityClass = UserStep.class,
attributes = "*", action = EntityAttributePolicyAction.VIEW)
@EntityPolicy(entityClass = UserStep.class,
actions = {EntityPolicyAction.READ, EntityPolicyAction.UPDATE})
void userStep();
}
按下 Ctrl/Cmd+S 保存修改然后切换至运行中的程序。打开 Administration → Resource roles 界面,可以在列表中看到新角色:
data:image/s3,"s3://crabby-images/ce4ba/ce4baf7c1255829fc6bee8748485462146258ce3" alt="employee role 5"
分配角色
现在我们可以将角色分配给用户。打开 Users 浏览界面并创建一个新用户 bob
。选择该用户并点击 Role assignments 按钮:
data:image/s3,"s3://crabby-images/9c197/9c197e918c9a2a0dfd36817c9eb0511d359b8b5b" alt="assign role 1"
在 Role assignments 界面中,点击 Resource permissions 中的 Add 按钮。
弹出的 Select resource roles 对话框中,选择 Employee
和 UI: minimal access
角色(使用 Ctrl/Cmd+单击
):
data:image/s3,"s3://crabby-images/52a0e/52a0ee46c7d0d23ada859aef3798bb2f3f14ff8e" alt="assign role 2"
点击 Select。会在 Resource permissions 面板展示选择的角色:
data:image/s3,"s3://crabby-images/9dfc0/9dfc0d6e6de4525f95931ee28b1c23591d28054b" alt="assign role 3"
点击 OK 保存分配的角色。
用户需要 UI: minimal access 角色用来登录应用程序 UI。可以通过 Resource roles 界面打开该角色或者在 IDE 中搜索 UiMinimalRole 类查看角色内容。
|
使用左下角用户名右边的按钮登出系统:
data:image/s3,"s3://crabby-images/fab95/fab95c648ea825bd1adac25c6317fcdba5e827b5" alt="assign role 4"
用 bob
登录。则在主菜单中仅能看到 My onboarding
界面:
data:image/s3,"s3://crabby-images/91088/9108847aa868f30147539d51e1d19ea72fbdb0c3" alt="assign role 5"
HR 经理的资源角色
在 Jmix 工具窗口中,点击 New()→ Role。
New Role 弹窗中,Role name 字段输入 HR Manager
,设置 Role code 为 hr-manager
,并在 Security scope 下拉框选择 UI
:
data:image/s3,"s3://crabby-images/eade9/eade980dd1cd0b2c5a5019fef847e067a02a4ae0" alt="manager role 1"
点击 OK。
Studio 会创建并打开一个带注解的接口:
package com.company.onboarding.security;
import io.jmix.security.role.annotation.ResourceRole;
@ResourceRole(name = "HR Manager", code = "hr-manager", scope = "UI")
public interface HRManagerRole {
}
切换至 User Interface 标签页并允许 User.browse
和 User.edit
界面(可以用顶部的搜索栏进行查找):
data:image/s3,"s3://crabby-images/b73e1/b73e1f7b123ff09eec3a8b2b716359e68ea29fed" alt="manager role 2"
切换至 Entities 标签页,赋予对 Department
和 Step
的只读权限,User
和 UserStep
的所有权限:
data:image/s3,"s3://crabby-images/f068b/f068b4b3c1550dbccc1caf4f98c8e843a3cd79dd" alt="manager role 3"
切回 Text 标签页,查看 Studio 生成的带注解方法:
@ResourceRole(name = "HR Manager", code = "hr-manager", scope = "UI")
public interface HRManagerRole {
@MenuPolicy(menuIds = "User.browse")
@ScreenPolicy(screenIds = {"User.browse", "User.edit"})
void screens();
@EntityAttributePolicy(entityClass = Department.class,
attributes = "*",
action = EntityAttributePolicyAction.VIEW)
@EntityPolicy(entityClass = Department.class,
actions = EntityPolicyAction.READ)
void department();
@EntityAttributePolicy(entityClass = Step.class,
attributes = "*",
action = EntityAttributePolicyAction.VIEW)
@EntityPolicy(entityClass = Step.class,
actions = EntityPolicyAction.READ)
void step();
@EntityAttributePolicy(entityClass = User.class,
attributes = "*",
action = EntityAttributePolicyAction.MODIFY)
@EntityPolicy(entityClass = User.class,
actions = EntityPolicyAction.ALL)
void user();
@EntityAttributePolicy(entityClass = UserStep.class,
attributes = "*",
action = EntityAttributePolicyAction.MODIFY)
@EntityPolicy(entityClass = UserStep.class,
actions = EntityPolicyAction.ALL)
void userStep();
}
按下 Ctrl/Cmd+S 保存修改然后切换至运行中的程序。以 admin
登录。打开 Administration → Resource roles 界面,确保列表中存在新创建的 HR Manager
角色。
创建一个新用户,比如 alice
。
通过与 前一小节 一样的方法,给 alice
分配 HR Mnager
和 UI: minimal access
角色。
然后以 alice
的账号登录。将可以打开 Users
界面并能管理用户和入职步骤:
data:image/s3,"s3://crabby-images/cfcbe/cfcbe055214fa3ed3f65e736f9e917d30b5ab444" alt="manager role 4"
HR 经理的行级角色
此时,HR 经理可以创建用户、为用户分配任意部门并能查看所有部门的用户。
本小节中,我们将引入一个 行级角色(row-level role),用于限制 HR 经理对部门和其他用户的访问权限。他们将只能看到并分配他们自己的部门(也就是部门中他们作为 hrManager
的那些)。
在 Jmix 工具窗口中,点击 New()→ Row-level Role:
data:image/s3,"s3://crabby-images/0292f/0292f30a870b560fb3b580bba6e0e246a668075b" alt="rl role 1"
在 New Row-level Role 弹窗中输入:
-
Role name:
HR manager’s departments and users
-
Role code:
hr-manager-rl
-
Class:
com.company.onboarding.security.HrManagerRlRole
data:image/s3,"s3://crabby-images/a0ab6/a0ab68252e56fb44c231dda87db53492b09a9f86" alt="rl role 2"
点击 OK。
Studio 会创建并打开一个带注解的接口:
package com.company.onboarding.security;
import io.jmix.security.role.annotation.RowLevelRole;
@RowLevelRole(
name = "HR manager's departments and users",
code = "hr-manager-rl")
public interface HrManagerRlRole {
}
在顶部的操作面板中点击 Add Policy → JPQL Policy:
data:image/s3,"s3://crabby-images/35a22/35a2295f8c35d2c30301498b7850a9af1325c71d" alt="rl role 3"
然后在 Add JPQL Policy 弹窗中,输入:
-
Entity:
Department
-
Where clause:
{E}.hrManager.id = :current_user_id
data:image/s3,"s3://crabby-images/9032e/9032ea3a5c25a18217d368d15448ad179a9d967d" alt="rl role 3 1"
点击 OK。
再次点击 Add Policy → JPQL Policy,并输入:
-
Entity:
User
-
Where clause:
{E}.department.hrManager.id = :current_user_id
点击 OK。
HrManagerRlRole
接口代码如下:
package com.company.onboarding.security;
import com.company.onboarding.entity.Department;
import com.company.onboarding.entity.User;
import io.jmix.security.role.annotation.JpqlRowLevelPolicy;
import io.jmix.security.role.annotation.RowLevelRole;
@RowLevelRole( (1)
name = "HR manager's departments and users",
code = "hr-manager-rl")
public interface HrManagerRlRole {
@JpqlRowLevelPolicy( (2)
entityClass = Department.class, (3)
where = "{E}.hrManager.id = :current_user_id") (4)
void department();
@JpqlRowLevelPolicy(
entityClass = User.class,
where = "{E}.department.hrManager.id = :current_user_id")
void user();
}
1 | @RowLevelRole 注解表示这个接口定义的是一个行级角色。 |
2 | @JpqlRowLevelPolicy 定义当读取实体时,在数据库层面将使用的一个策略。 |
3 | 策略应用的实体类。 |
4 | 执行实体的每个 JPQL 查询语句时添加的 where 子句。语句中使用 {E} 而非实体的别名。:current_user_id 是预定义的参数,由框架设置为当前登录用户的 id。 |
按下 Ctrl/Cmd+S 保存修改然后切换至运行中的程序。用 admin
登录。打开 Administration → Row-level roles 界面,确保列表存在新添加的 HR manager’s departments and users
角色。
为用户 alice
打开 Role assignments 界面并在 Row-level constraints 表格中添加刚才创建的角色:
data:image/s3,"s3://crabby-images/71269/71269aca12713d730e152291a150988b691982c4" alt="rl role 4"
点击 OK 保存角色修改。
将 alice
设置为一个部门的 HR 经理:
data:image/s3,"s3://crabby-images/21275/2127547b686e246d0ccbb0e518fbf9e94c5b20b5" alt="rl role 5"
以 alice
用户登录。
在 Users 浏览界面,只能看到同一部门的用户了:
data:image/s3,"s3://crabby-images/c532e/c532e451ee17f3c1ac0500f2ab01fa9ff48a794b" alt="rl role 6"
并且 alice
只能将本部门分配给其他用户:
data:image/s3,"s3://crabby-images/10087/10087ce2080b569158f1a14af7b3e21f3193e596" alt="rl role 7"