Architecture
Module Structure
| Module | Artifact | Description |
|---|---|---|
security-core | security-core | Generic, framework-neutral security concepts and decision logic |
security-vaadin | security-vaadin | Vaadin adapter — view and navigation security |
security-rest | security-rest | Framework-light REST adapter — request and handler security |
security-standalone | security-standalone | Core-Java adapter — dynamic-proxy Secured.wrap(…) for CLI / desktop / batch / embedded apps |
demo-rest-shared | demo-rest-shared | Transport-level constants + tiny JSON helper, shared between the REST server and any client |
demo-vaadin | demo-vaadin | Standalone Vaadin demo (WAR) — auth runs in-JVM |
demo-rest | demo-rest | Runnable REST reference: JDK-only HTTP server + CLI client |
demo-vaadin-rest-client | demo-vaadin-rest-client | Vaadin demo where demo-rest is the authoritative backend; UI talks to it through one encapsulated Java client |
demo-standalone | demo-standalone | Interactive Core-Java CLI demonstrating Secured.wrap(…) against an in-memory library service |
Dependency Rules
security-core -> (no project deps)
security-vaadin -> security-core
security-rest -> security-core
security-standalone -> security-core
demo-rest-shared -> (no project deps; transport-only)
demo-vaadin -> security-core, security-vaadin
demo-rest -> security-core, security-rest, demo-rest-shared
demo-vaadin-rest-client -> security-core, security-vaadin, demo-rest-shared
(test scope only: demo-rest)
demo-standalone -> security-core, security-standalonesecurity-core has no Vaadin, Servlet, or REST-framework dependencies.
None of the three adapters (security-vaadin, security-rest,
security-standalone) depend on each other — they all sit on top of
security-core and can be used independently.
Package layout
security-core packages are organised by concern, not by adapter:
com.svenruppert.vaadin.security
├── action/ — ActionAuthorizationService, ActionPermission
├── audit/ — SecurityAuditService + 16 sealed AuditEvent types + sinks
├── authentication/ — AuthenticationService, PasswordHasher, Pbkdf2PasswordHasher
├── authorization/ — annotations, evaluators, scanner, AuthorizationDecision
├── bootstrap/ — first-run admin setup
├── bruteforce/ — LoginAttemptPolicy + InMemory default
├── logout/ — LogoutService, SubjectSessionRegistry, LogoutListener
└── session/ — SessionPolicy + TimeoutSessionPolicy + SessionDecisionEach adapter adds one small wiring package on top:
com.svenruppert.vaadin.security.standalone — Secured, StandaloneLoginFlow,
ThreadLocalSubjectStore (3 classes)
com.svenruppert.vaadin.security.logout.vaadin — VaadinLogoutService + Gateway
com.svenruppert.vaadin.security.session.vaadin— SessionLifetimeListener
com.svenruppert.vaadin.security.rest — RestRequest/Response, filters,
BearerTokenExtractor, …Core rule
Library modules do not define project-specific permissions. Concrete roles, permissions, and business operations belong to consuming applications or demo modules.
Decision Model
The library uses two decision types:
| Type | Module | Purpose |
|---|---|---|
AuthorizationDecision | security-core | Adapter-neutral: Granted / Unauthenticated / Forbidden |
AccessDecision | security-core | Vaadin-oriented (legacy, kept for backward compatibility) |
Adapters map these to framework-specific behavior:
security-vaadin→ navigation: continue, reroute to login, or reroute to error.security-rest→ HTTP status:200/handler,401, or403.
In addition, two sealed decision hierarchies cover the production-hardening SPIs:
LoginAttemptDecision = Allowed | LockedOut(Duration, int)— see Brute-Force ProtectionSessionDecision = Continue | RequireLogin | Invalidate(String reason, String loginRoute)— see Session PolicySessionPolicyDecision = Active | IdleTimeout | AbsoluteLifetimeExceeded— pure-query result ofSessionPolicy.evaluate(SessionMetadata)
SecurityServiceResolver — one resolver for all SPIs
SecurityAuditService audit = SecurityServiceResolver.auditService();
LogoutService logout = SecurityServiceResolver.logoutService();
LoginAttemptPolicy bruteFor = SecurityServiceResolver.loginAttemptPolicy();
SessionPolicy<MyUser> session = SecurityServiceResolver.sessionPolicy();
ActionAuthorizationService<MyUser> action = SecurityServiceResolver.actionService();
PasswordHasher hasher = SecurityServiceResolver.passwordHasher();Covers all eight SPIs (Authentication / Authorization / Audit / Action
/ LoginAttempt / Session / PasswordHasher / Logout). Strict accessors
throw IllegalStateException for missing services; find…() returns
Optional; set…(…) is a programmatic test seam.
Annotation-Driven Protection
SecurityAnnotationScanner scans classes, methods, or any
AnnotatedElement for restriction annotations meta-annotated with
@SecurityAnnotation. Both adapters use the same scanner.
Generic annotations (in security-core):
@RequiresRole({"ROLE_ADMIN"})→RequiresRoleEvaluator@RequiresPermission("document:delete")→RequiresPermissionEvaluator@ProtectedBy(...)→ProtectedByEvaluator
Project-specific annotations are encouraged for Vaadin views (e.g.
@VisibleFor).
Three Authorization Patterns
The library distinguishes three intent-explicit call shapes:
// Pattern A — UX hint. Hide the button if the user can't use it.
if (PermissionGuard.hasPermission(user, "document:delete")) {
layout.add(deleteButton);
}
// Pattern B — Server-side guard. Throws AccessDeniedException.
public void handleDelete() {
PermissionGuard.requirePermission(user, "document:delete");
documentService.delete(...);
}
// Pattern C — Audited action check. ActionAuthorizationService SPI.
public void handleDelete() {
actionService.requireAllowed(user, ActionPermission.of("document:delete"));
// ActionDenied audit event is emitted automatically on denial
documentService.delete(...);
}Hiding a button is never the security boundary — it’s a usability hint. The real check happens at the route, the handler, or the service call. The three patterns make the intent explicit at the call site so reviewers can tell the difference at a glance.
The two-tier setup in demo-vaadin-rest-client takes this further:
PermissionGuard runs locally against the cached RemoteUser purely
for UX, while the REST backend is the authoritative decision point.
Clients never make local authorization decisions that the server hasn’t
sanctioned.
Two-tier reference architecture
demo-vaadin-rest-client shows how to wire demo-rest as the
authoritative backend behind a Vaadin UI:
- Vaadin code sees no REST calls. All
views/andsecurity/code is free ofjava.net.http.*,URI,HttpClient, JSON, or endpoint paths. The contract isDemoBackendClient;HttpDemoBackendClientis the only class with transport knowledge. - REST server is authoritative. Mutating clicks call the server.
200 / 401 / 403decides. LocalPermissionGuardchecks against the cached subject are UX hints only. - Bootstrap goes over REST. The Vaadin
/setupview callsPOST /api/bootstrap/admin— no in-JVM admin logic.
demo-rest-shared provides the transport-level constants
(DemoEndpoints) and a tiny JSON helper, shared between the REST
server and any client. It has no project-specific code.
Reusable security building blocks
| Type | Module / package | Purpose |
|---|---|---|
SecurityServiceResolver | security-core/.../authorization/api | Central SPI cache for all eight services. |
PermissionGuard, AccessDeniedException | security-core/.../authorization/api | Stateless hasPermission / requirePermission (and role variants). |
AuthenticationService<T,U> | security-core/.../authentication | SPI: credential validation + subject loading. |
PasswordHasher, PasswordHash, Pbkdf2PasswordHasher | security-core/.../authentication | Hash + verify + needsRehash drift detection. Demos rehash transparently on login. |
LogoutService, LogoutScope, SubjectId, SubjectSessionRegistry, InMemorySubjectSessionRegistry, LogoutListener, SubjectClearingLogoutService | security-core/.../logout | Multi-session logout. See Logout Flows. |
VaadinLogoutService | security-vaadin | Registers as a LogoutListener; invalidates Vaadin + HTTP sessions, redirects browser. |
LoginAttemptPolicy, LoginAttemptDecision, InMemoryLoginAttemptPolicy, LoginAttemptConfiguration[Loader] | security-core/.../bruteforce | Pluggable login throttling. See Brute-Force Protection. |
SessionPolicy<U>, SessionDecision, SessionMetadata, TimeoutSessionPolicy | security-core/.../session | Idle / absolute lifetime + session-id rotation. See Session Policy. |
SecurityAuditService, sealed AuditEvent (16 record types), RingBufferAuditSink, LoggingAuditSink, CompositeAuditService, DefaultCompositeAuditService | security-core/.../audit | Typed publish/query pipeline. Powers the Vaadin /audit route and the REST GET /api/audit endpoint. See Security Audit. |
ActionAuthorizationService<U>, ActionPermission, StaticActionAuthorizationService | security-core/.../action | Stable SPI for isAllowed/requireAllowed with auto-audit on denial. |
StaticRolePermissionMapping, RolePermissionResolver | …/authorization/api/permissions | Immutable role → permissions map; permission merge across roles. |
SecuredOperationDescriptor, SecuredOperationRegistry, OperationVisibilityService | …/authorization/api/operations | Generic operation discovery with subject-aware filtering. |
BootstrapConfigurationLoader, BootstrapStatus | security-core/.../bootstrap | Centralised sysprop+env+default loading; leak-safe status snapshot. |
RestHeaders, BearerTokenExtractor | security-rest | Case-insensitive header lookup and Bearer-token parsing. |
RestAuthenticationFilter, RestAuthorizationFilter | security-rest | 401-only and full 401/403 filters. The authorization filter additionally consults SessionPolicy.evaluate(...) when subject metadata is available. |
BodyRestRequest | security-rest | Body-capable RestRequest. Avoids concrete-class casts. |
BootstrapRestStatusMapper | security-rest | InitialAdminCreationResult → HTTP status + stable error code. |
Secured.wrap(Class<T>, T), Secured.requireAllowed(Class<?>, String) | security-standalone | Dynamic-proxy enforcement of @RequiresRole / @RequiresPermission on any interface — no framework needed. See Standalone Integration. |
StandaloneLoginFlow<T,U>, LoginResult<U>, ThreadLocalSubjectStore | security-standalone | Login lifecycle for CLI / desktop / batch apps; integrates with the same brute-force + audit SPIs as the framework adapters. |
Quality — Mutation Coverage
Library modules carry the production guarantee:
| Module | Line Coverage | Mutation Coverage | Test Strength |
|---|---|---|---|
security-core | 82 % (1113 / 1351) | 79 % (497 / 629) ¹ | 91 % (497 / 547) |
security-vaadin | 91 % (440 / 486) | 90 % (163 / 182) | 93 % (163 / 175) |
security-rest | 93 % (154 / 166) | 95 % (57 / 60) | 95 % (57 / 60) |
security-standalone | 86 % (113 / 131) | 98 % (44 / 45) | 98 % (44 / 45) |
¹ security-core was measured at 86 % in 00.51; the percentage dropped
in 00.60 because the audit pipeline, LoginAttemptPolicy, SessionPolicy,
ActionAuthorizationService and the refactored LogoutService are now
inside the scope. The absolute killed-mutant count rose from 254 → 497.
Demo modules are exercised end-to-end through the library tests; their direct mutation numbers are reported for transparency:
| Module | Mutation Coverage |
|---|---|
demo-standalone | 86 % |
demo-vaadin | 70 % |
demo-rest | 49 % |
demo-vaadin-rest-client | 10 % |
Test strength = killed mutations / covered mutations. A 91% test strength means: of every 100 mutations the tests reached, 91 were detected. Line coverage tells you whether code runs; test strength tells you whether the assertions catch bugs.
Reports are generated with Pitest via
mvn -P mutation verify.
Stable vs. Experimental API
Stable: role-based access, REST adapter contracts, SecuritySubject,
AccessContext, AuthorizationDecision, scanner, LogoutService,
LoginAttemptPolicy, SessionPolicy, SecurityAuditService,
ActionAuthorizationService, PasswordHasher, SecurityServiceResolver.
Experimental (marked with @ExperimentalSecurityApi):
permission-based access types — PermissionBasedAccessEvaluator,
PermissionName, HasPermissions, PermissionAuthorizationService.
May change in incompatible ways in future releases.
Project-specific permissions live in applications
Library modules contain no concrete business permissions. Examples like
document:read belong in demo-rest. Real applications define their own
catalog (e.g. shortlink:create, audit:read) inside the consuming
project.