這次講講如何限制用戶登錄嘗試次數,防止壞人多次嘗試,惡意暴力破解密碼的情況出現,要限制用戶登錄嘗試次數,必然要對用戶名密碼驗證失敗做記錄,Shiro中用戶名密碼的驗證交給了CredentialsMatcher 所以在CredentialsMatcher里面檢查,記錄登錄次數是最簡單的做法。當登錄失敗次數達到限制,修改數據庫中的狀態字段,并返回前臺錯誤信息。
因為之前的博客都是用的明文,這里就不對密碼進行加密了,如果有需要加密,將自定義密碼比較器從SimpleCredentialsMatcher改為HashedCredentialsMatcher 然后將對應的配置項打開就可以。
說在前面
非常抱歉,因為我之前整合的時候,只是注意功能,而沒有注意細節,導致在登錄失敗之后,再次轉發到 post方法/login 也就是真正的登錄方法,導致 再次登錄,然后導致下面密碼錯誤3次之后 就 鎖定 我設置的是5次.
所以將shiroConfig中的值改為shiroFilterFactoryBean.setLoginUrl("/");具體參考源代碼。
另外 還需要將 自定義ShiroRealm 中 密碼對比注銷掉, 將密碼對比 交給 底層的 密碼比較器才可以 鎖定用戶,否則將 永遠報密碼錯誤。,具體代碼 如下:
修改登錄方法改為登錄之后,重定向到/index

限制登錄次數
自定義RetryLimitHashedCredentialsMatcher繼承SimpleCredentialsMatcher
package com.springboot.test.shiro.config.shiro;
import java.util.concurrent.atomic.AtomicInteger;
import com.springboot.test.shiro.modules.user.dao.UserMapper;
import com.springboot.test.shiro.modules.user.dao.entity.User;
import org.apache.log4j.Logger;
import org.apache.shiro.authc.AuthenticationInfo;
import org.apache.shiro.authc.AuthenticationToken;
import org.apache.shiro.authc.LockedAccountException;
import org.apache.shiro.authc.credential.SimpleCredentialsMatcher;
import org.apache.shiro.cache.Cache;
import org.apache.shiro.cache.CacheManager;
import org.springframework.beans.factory.annotation.Autowired;
/**
* @author: WangSaiChao
* @date: 2018/5/25
* @description: 登陸次數限制
*/
public class RetryLimitHashedCredentialsMatcher extends SimpleCredentialsMatcher {
private static final Logger logger = Logger.getLogger(RetryLimitHashedCredentialsMatcher.class);
@Autowired
private UserMapper userMapper;
private Cache<String, AtomicInteger> passwordRetryCache;
public RetryLimitHashedCredentialsMatcher(CacheManager cacheManager) {
passwordRetryCache = cacheManager.getCache("passwordRetryCache");
}
@Override
public boolean doCredentialsMatch(AuthenticationToken token, AuthenticationInfo info) {
//獲取用戶名
String username = (String)token.getPrincipal();
//獲取用戶登錄次數
AtomicInteger retryCount = passwordRetryCache.get(username);
if (retryCount == null) {
//如果用戶沒有登陸過,登陸次數加1 并放入緩存
retryCount = new AtomicInteger(0);
passwordRetryCache.put(username, retryCount);
}
if (retryCount.incrementAndGet() > 5) {
//如果用戶登陸失敗次數大于5次 拋出鎖定用戶異常 并修改數據庫字段
User user = userMapper.findByUserName(username);
if (user != null && "0".equals(user.getState())){
//數據庫字段 默認為 0 就是正常狀態 所以 要改為1
//修改數據庫的狀態字段為鎖定
user.setState("1");
userMapper.update(user);
}
logger.info("鎖定用戶" + user.getUsername());
//拋出用戶鎖定異常
throw new LockedAccountException();
}
//判斷用戶賬號和密碼是否正確
boolean matches = super.doCredentialsMatch(token, info);
if (matches) {
//如果正確,從緩存中將用戶登錄計數 清除
passwordRetryCache.remove(username);
}
return matches;
}
/**
* 根據用戶名 解鎖用戶
* @param username
* @return
*/
public void unlockAccount(String username){
User user = userMapper.findByUserName(username);
if (user != null){
//修改數據庫的狀態字段為鎖定
user.setState("0");
userMapper.update(user);
passwordRetryCache.remove(username);
}
}
}
在shiroConfig中配置該bean
/**
* 配置密碼比較器
* @return
*/
@Bean("credentialsMatcher")
public RetryLimitHashedCredentialsMatcher retryLimitHashedCredentialsMatcher(){
RetryLimitHashedCredentialsMatcher retryLimitHashedCredentialsMatcher = new RetryLimitHashedCredentialsMatcher(ehCacheManager());
//如果密碼加密,可以打開下面配置
//加密算法的名稱
//retryLimitHashedCredentialsMatcher.setHashAlgorithmName("MD5");
//配置加密的次數
//retryLimitHashedCredentialsMatcher.setHashIterations(1024);
//是否存儲為16進制
//retryLimitHashedCredentialsMatcher.setStoredCredentialsHexEncoded(true);
return retryLimitHashedCredentialsMatcher;
}
在shiroRealm中配置密碼比較器
/**
* 身份認證realm; (這個需要自己寫,賬號密碼校驗;權限等)
* @return
*/
@Bean
public ShiroRealm shiroRealm(){
ShiroRealm shiroRealm = new ShiroRealm();
......
//配置自定義密碼比較器
shiroRealm.setCredentialsMatcher(retryLimitHashedCredentialsMatcher());
return shiroRealm;
}
在ehcache-shiro.xml添加緩存項
<!-- 登錄失敗次數緩存 注意 timeToLiveSeconds 設置為300秒 也就是5分鐘 可以根據自己的需求更改 --> <cache name="passwordRetryCache" maxEntriesLocalHeap="2000" eternal="false" timeToIdleSeconds="0" timeToLiveSeconds="300" overflowToDisk="false" statistics="true"> </cache>
在LoginController中添加解除admin用戶限制方法
/**
* 解除admin 用戶的限制登錄
* 寫死的 方便測試
* @return
*/
@RequestMapping("/unlockAccount")
public String unlockAccount(Model model){
model.addAttribute("msg","用戶解鎖成功");
retryLimitHashedCredentialsMatcher.unlockAccount("admin");
return "login";
}
注意:為了方便測試,記得將 unlockAccount 權限改為 任何人可訪問。
在login.html頁面 添加 解鎖admin用戶的按鈕
<a href="/unlockAccount" rel="external nofollow" >解鎖admin用戶</a></button>
測試結果

總結
以上所述是小編給大家介紹的springboot整合shiro-登錄失敗次數限制,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對億速云網站的支持!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。