大家好,我是 V 哥。Apache Shiro 是一個強大且靈活的 Java 安全框架,專注于提供認證、授權、會話管理和加密功能。它常用于保護 Java 應用的訪問控制,特別是在 Web 應用中。相比于 Spring Security,Shiro 的設計更簡潔,適合輕量級應用,并且在許多方面具有更好的易用性和擴展性,今天 V 哥就來聊聊 Shiro 安全框架。
按照慣例,和 V 哥一起來了解一下 Shiro 的核心概念:
SecurityUtils.getSubject()
獲取當前的 Subject。它代表了用戶的身份信息和權限數(shù)據(jù)。subject.login(token)
。在實際應用中,通常通過用戶名和密碼的組合進行認證,但 Shiro 也支持其他方式(如 OAuth2、JWT 等)。subject.hasRole
或 subject.isPermitted
方法,開發(fā)者可以檢查用戶的角色和權限。V 哥總結幾點Shiro 的主要功能和優(yōu)勢,這個在面試時吹牛逼用得到。
光講概念不是 V 哥風格,接下來,通過一個典型的 Shiro 應用來了解一下如何使用,包含配置 SecurityManager、配置 Realm、進行認證和授權等步驟。
shiro.ini
文件配置 Shiro,也可以通過代碼進行配置。 [main]
# 配置 SecurityManager
securityManager = org.apache.shiro.mgt.DefaultSecurityManager
# 配置 Realm
myRealm = com.wg.MyCustomRealm
securityManager.realms = $myRealm
自定義 Realm 通過繼承 AuthorizingRealm
并實現(xiàn) doGetAuthenticationInfo
和 doGetAuthorizationInfo
方法來提供用戶和權限數(shù)據(jù)。
public class MyCustomRealm extends AuthorizingRealm {
@Override
protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken token) throws AuthenticationException {
// 獲取用戶名和密碼等信息,查詢數(shù)據(jù)庫進行認證
return new SimpleAuthenticationInfo(username, password, getName());
}
@Override
protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals) {
// 獲取用戶角色和權限信息
SimpleAuthorizationInfo info = new SimpleAuthorizationInfo();
info.addRole("admin");
info.addStringPermission("user:read");
return info;
}
}
Subject currentUser = SecurityUtils.getSubject();
if (!currentUser.isAuthenticated()) {
UsernamePasswordToken token = new UsernamePasswordToken("username", "password");
try {
currentUser.login(token);
System.out.println("認證成功");
} catch (AuthenticationException ae) {
System.out.println("認證失敗");
}
}
// 檢查權限
if (currentUser.hasRole("admin")) {
//用輸出模擬一下哈
System.out.println("用戶擁有 admin 角色");
}
if (currentUser.isPermitted("user:read")) {
//用輸出模擬一下哈
System.out.println("用戶具有 user:read 權限");
}
通過這個簡單的案例學習,咱們可以了解 Shiro 的基本使用,但這不是全部,聽V哥繼續(xù)慢慢道來。
這點很重要,強調(diào)一下哈,Shiro 適合需要簡潔易用、安全控制要求靈活的 Java 應用,如中小型 Web 應用、桌面應用、分布式微服務等。對于大型企業(yè)應用或需要集成多種認證方式(如 OAuth2、JWT 等)的項目,Spring Security 可能會更合適。
要在微服務架構中實現(xiàn)基于 Apache Shiro 的安全認證和授權,比如一個訂單管理系統(tǒng)為例。這個系統(tǒng)包含兩個主要服務:
咱們來看一下,這個應該怎么設計呢?
在這個場景中,我們需要以下幾項核心功能:
+------------------+ +---------------------+
| 用戶服務 | | 訂單服務 |
| | | |
| 用戶注冊、登錄 | | 查看、創(chuàng)建、刪除訂單|
+------------------+ +---------------------+
| |
|----用戶 Token ------|
在 pom.xml
文件中,添加 Shiro、JWT 和 Spring Data JPA 等依賴:
<dependency>
<groupId>org.apache.shiro</groupId>
<artifactId>shiro-spring</artifactId>
<version>1.8.0</version>
</dependency>
<dependency>
<groupId>io.jsonwebtoken</groupId>
<artifactId>jjwt</artifactId>
<version>0.9.1</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
</dependency>
使用 Shiro 的自定義 JWT 過濾器實現(xiàn)無狀態(tài)認證,通過 Token 驗證用戶。
public class JwtFilter extends BasicHttpAuthenticationFilter {
@Override
protected boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) {
HttpServletRequest httpServletRequest = (HttpServletRequest) request;
String token = httpServletRequest.getHeader("Authorization");
if (StringUtils.isBlank(token)) {
return false;
}
try {
// 解析 JWT token
JwtToken jwtToken = new JwtToken(token);
getSubject(request, response).login(jwtToken);
return true;
} catch (Exception e) {
return false;
}
}
}
自定義 Realm
,從數(shù)據(jù)庫獲取用戶和角色信息,并使用 JWT Token 進行無狀態(tài)認證。
public class JwtRealm extends AuthorizingRealm {
@Override
public boolean supports(AuthenticationToken token) {
return token instanceof JwtToken;
}
@Override
protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken token) throws AuthenticationException {
String jwtToken = (String) token.getPrincipal();
// 驗證 Token
String username = JwtUtil.getUsernameFromToken(jwtToken);
if (username == null) {
throw new AuthenticationException("Token 無效");
}
// 查詢用戶
User user = userService.findByUsername(username);
if (user == null) {
throw new AuthenticationException("用戶不存在");
}
return new SimpleAuthenticationInfo(jwtToken, jwtToken, getName());
}
@Override
protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals) {
String username = JwtUtil.getUsernameFromToken(principals.toString());
SimpleAuthorizationInfo authorizationInfo = new SimpleAuthorizationInfo();
User user = userService.findByUsername(username);
// 添加角色和權限
authorizationInfo.addRole(user.getRole());
authorizationInfo.addStringPermission(user.getPermission());
return authorizationInfo;
}
}
編寫一個工具類,用于生成和解析 JWT Token。
public class JwtUtil {
//這里替換一下你自己的secret_key
private static final String SECRET_KEY = "這里打碼了";
public static String generateToken(String username) {
return Jwts.builder()
.setSubject(username)
.setIssuedAt(new Date())
.setExpiration(new Date(System.currentTimeMillis() + 3600000)) // 1 hour
.signWith(SignatureAlgorithm.HS256, SECRET_KEY)
.compact();
}
public static String getUsernameFromToken(String token) {
Claims claims = Jwts.parser()
.setSigningKey(SECRET_KEY)
.parseClaimsJws(token)
.getBody();
return claims.getSubject();
}
public static boolean isTokenExpired(String token) {
Claims claims = Jwts.parser()
.setSigningKey(SECRET_KEY)
.parseClaimsJws(token)
.getBody();
return claims.getExpiration().before(new Date());
}
}
用戶服務提供注冊和登錄 API。
@RestController
@RequestMapping("/user")
public class UserController {
@PostMapping("/register")
public ResponseEntity<?> register(@RequestBody User user) {
userService.save(user);
return ResponseEntity.ok("用戶注冊成功");
}
@PostMapping("/login")
public ResponseEntity<?> login(@RequestBody User user) {
User dbUser = userService.findByUsername(user.getUsername());
if (dbUser != null && dbUser.getPassword().equals(user.getPassword())) {
String token = JwtUtil.generateToken(user.getUsername());
return ResponseEntity.ok(token);
}
return ResponseEntity.status(HttpStatus.UNAUTHORIZED).body("登錄失敗");
}
}
訂單服務在操作訂單時會驗證用戶的角色和權限。
@RestController
@RequestMapping("/order")
public class OrderController {
@GetMapping("/{orderId}")
public ResponseEntity<?> getOrder(@PathVariable Long orderId) {
Subject currentUser = SecurityUtils.getSubject();
if (currentUser.isPermitted("order:read")) {
// 查詢訂單
return ResponseEntity.ok("訂單詳情");
}
return ResponseEntity.status(HttpStatus.FORBIDDEN).body("無權限查看訂單");
}
@DeleteMapping("/{orderId}")
public ResponseEntity<?> deleteOrder(@PathVariable Long orderId) {
Subject currentUser = SecurityUtils.getSubject();
if (currentUser.hasRole("admin")) {
// 刪除訂單
return ResponseEntity.ok("訂單已刪除");
}
return ResponseEntity.status(HttpStatus.FORBIDDEN).body("無權限刪除訂單");
}
}
這個案例中咱們通過如何使用 Shiro、JWT 和 Spring Boot 來構建一個無狀態(tài)的微服務認證授權機制。通過 Shiro 實現(xiàn)用戶認證和權限控制,使用 JWT 實現(xiàn)無狀態(tài) Token 驗證。在輕量級的分布式微服務應用中,是不是使用 Shiro 感覺更加清爽呢,歡迎評論區(qū)一起討論,關注威哥愛編程,愛上Java,一輩子。
更多建議: