Custom Implementation Example
本指南讲解如何构建一个完整的自定义安全实现,使用基于会话的身份验证。您将学习四个核心接口如何通过从头开始实现它们相互协作。
大多数应用程序应使用 Spring Security
Spring Security 集成 会自动配置此处显示的所有内容。仅在您有特定需求或未使用 Spring Boot 时构建自定义安全。
您将构建 的内容
一个包含四个类的工作安全系统:
- SecurityConfiguration - 定义安全行为和重定向位置
- SecurityContext - 通过 HTTP 会话跟踪谁已登录
- SecurityManager - 协调安全检查并提供登录/注销功能
- SecurityRegistrar - 在应用启动时将所有内容连接在一起
此示例使用基于会话的存储,但您也可以使用数据库查询、LDAP 或其他身份验证后端来实现相同的接口。
各组件如何协作
流程:
SecurityRegistrar在启动时运行,创建管理器,注册评估器,并附加观察者SecurityManager协调所有操作 - 提供上下文和配置给评估器SecurityContext通过读取 HTTP 会话来回答“谁已登录?”SecurityConfiguration回答“重定向到哪里?”用于登录和访问被拒绝页面Evaluators使用上下文和配置做出访问决策
第一步:定义安全配置
配置告诉安全系统如何执行和在哪里重定 向用户:
SecurityConfiguration.java
package com.securityplain.security;
import com.webforj.router.history.Location;
import com.webforj.router.security.RouteSecurityConfiguration;
import java.util.Optional;
/**
* 应用程序的安全配置。
*
* <p>
* 定义在需要身份验证或访问被拒绝时重定向用户的位置。
* </p>
*/
public class SecurityConfiguration implements RouteSecurityConfiguration {
@Override
public boolean isEnabled() {
return true;
}
@Override
public boolean isSecureByDefault() {
return false;
}
@Override
public Optional<Location> getAuthenticationLocation() {
return Optional.of(new Location("/login"));
}
@Override
public Optional<Location> getDenyLocation() {
return Optional.of(new Location("/access-denied"));
}
}
isEnabled() = true- 安全处于活动状态isSecureByDefault() = false- 路由为公共路由,除非有注释(使用true可默认要求所有路由身份验证)/login- 未认证用户的去处/access-denied- 有权限但未认证用户的去处
第二步:实现安全上下文
上下文跟踪谁已登录。此实现使用 HTTP 会话存储用户信息:
工作原理:
isAuthenticated()检查会话中是否存在用户主体getPrincipal()从会话存储中检索用户名hasRole()检查用户的角色集合是否包含指定的角色getAttribute()/setAttribute()管理自定义安全属性Environment.getSessionAccessor()提供线程安全的会话访问
第三步:创建安全管理器
管理器协调安全决策。它扩展了 AbstractRouteSecurityManager,后者处理评估器链和访问拒绝:
工作原理:
- 扩展
AbstractRouteSecurityManager以继承评估器链逻辑 - 提供
getConfiguration()和getSecurityContext()的实现 - 增加
login()以验证用户并将凭据存储在会话中 - 增加
logout()以清除会话并重定向到登录页面 - 使用
SessionObjectTable进行简单的会话存储 - 将自身存储在
ObjectTable中,以便于全局访问
第四步:在启动时连接所有组件
注册器在应用启动时连接所有组件:
SecurityRegistrar.java
package com.securityplain.security;
import com.webforj.App;
import com.webforj.AppLifecycleListener;
import com.webforj.annotation.AppListenerPriority;
import com.webforj.router.Router;
import com.webforj.router.security.RouteSecurityObserver;
import com.webforj.router.security.evaluator.AnonymousAccessEvaluator;
import com.webforj.router.security.evaluator.DenyAllEvaluator;
import com.webforj.router.security.evaluator.PermitAllEvaluator;
import com.webforj.router.security.evaluator.RolesAllowedEvaluator;
/**
* 在应用启动时注册路由安全组件。
*
* <p>
* 与路由设置安全管理器和评估器。
* </p>
*/
@AppListenerPriority(1)
public class SecurityRegistrar implements AppLifecycleListener {
/**
* {@inheritDoc}
*/
@Override
public void onWillRun(App app) {
// 创建安全管理器
SecurityManager securityManager = new SecurityManager();
securityManager.saveCurrent(securityManager);
// 按优先级注册内置评估器
securityManager.registerEvaluator(new DenyAllEvaluator(), 0);
securityManager.registerEvaluator(new AnonymousAccessEvaluator(), 1);
securityManager.registerEvaluator(new PermitAllEvaluator(), 2);
securityManager.registerEvaluator(new RolesAllowedEvaluator(), 3);
// 创建安全观察者并附加到路由
RouteSecurityObserver securityObserver = new RouteSecurityObserver(securityManager);
Router router = Router.getCurrent();
if (router != null) {
router.getRenderer().addObserver(securityObserver);
}
}
}
注册监听器:
在 src/main/resources/META-INF/services/com.webforj.AppLifecycleListener 中创建:
com.securityplain.security.SecurityRegistrar
这将注册您的 AppLifecycleListener,以便在应用启动时运行。
工作原理:
- 运行较早(
@AppListenerPriority(1)),以在路由加载之前设置安全 - 创建安全管理器并全局存储
- 按优先级顺序注册内置评估器(较小的数字先运行)
- 创建会拦截导航的观察者
- 将观察者附加到路由,以便自动进行安全检查
在此运行之后,所有导航的安全性将处于活动状态。
使用您的实现
创建登录视图
以下视图使用 Login 组件。
LoginView.java
package com.securityplain.views;
import com.securityplain.security.SecurityManager;
import com.webforj.component.Composite;
import com.webforj.component.login.Login;
import com.webforj.router.Router;
import com.webforj.router.annotation.FrameTitle;
import com.webforj.router.annotation.Route;
import com.webforj.router.history.Location;
import com.webforj.router.security.annotation.AnonymousAccess;
@Route("/login")
@FrameTitle("登录")
@AnonymousAccess
public class LoginView extends Composite<Login> {
private Login self = getBoundComponent();
public LoginView() {
self.onSubmit(e -> {
var result = SecurityManager.getCurrent().login(
e.getUsername(), e.getPassword()
);
if (result.isGranted()) {
Router.getCurrent().navigate(new Location("/"));
} else {
self.setError(true);
self.setEnabled(true);
}
});
self.whenAttached().thenAccept(c -> self.open());
}
}