一個不錯的基於Spring boot+Security+Redis+MySql實現權限登錄和反爬蟲的腳手架

介紹

一個基於Spring boot 2.4.2、JDK 1.8、Security、防惡意請求技術實現的前後端分離的腳手架,可以爲開發人員省去前期框架調研和搭建的成本。源碼在碼雲上。

軟件架構

  1. Spring boot 2.4.2
  2. JDK 1.8
  3. Security
  4. kk-anti-reptile(反爬蟲接口防刷組件)
  5. Redis(登錄用戶信息)
  6. MySQL
  7. MyBatis

安裝教程

  1. 克隆 本代碼到本地
  2. 修改application.yml配置,包含Mysql數據庫信息、Redis連接信息、kk-anti-reptile配置信息(不修改也可以,默認,詳細參數見:kk-anti-reptile
  3. 將項目other/SQL/中的腳本放到mysql中執行,初始化表結構
  4. 啓動項目
  5. 使用測試工具或者前端連接請求

使用說明

本示例在TestWebSecurity類中配置了Security前端登錄接口和退出登錄接口,如想自定義接口,請修改如下圖圈中的配置

application.yml 配置文件中kk-anti-reptile 部分配置如下:

# 惡意請求
anti:
  reptile:
    manager:
    # 激活惡意請求/反爬蟲配置
      enabled: true
    # 需要攔截的請求,多個以“,”分隔,或者直接在controller方法上添加@AntiReptile
    #  include-urls: ^/admin/.*$
      ua-rule:
      #允許linux系統訪問
        allowed-linux: true
        # 全局攔截,默認爲false,爲false時,需要配置上面的include-urls
      global-filter-mode: true

詳細參數請參見kk-anti-reptile

請求示例

登錄接口

請求鏈接

http://localhost:8099/auth/login

請求參數

username:admin password:1234

注意,此處需要使用表單提交,postman中如下:

響應參數

登錄成功返回參數如下:

{
    "code": "00000",
    "msg": "操作成功!",
    "data": {
        "token": "3aa72e4a-74b5-4542-9e56-e7f31ec5ce09",
        "permissions": [],
        "loginTime": 1611284216123
    }
}

其中,token的值就是前端需要在每次請求中在請求頭部傳遞到後端的參數

退出登錄接口

請求鏈接

http://localhost:8099/auth/logout

請求參數

請求頭部需要放置如下參數:

X-token:token的值

Postman中如下:

響應參數

{
    "code": "00000",
    "msg": "您已經退出登錄!"
}

正常接口

正常業務邏輯中調用的接口傳參

請求鏈接

http://localhost:8099/test/listOrientations

請求參數

header中必須帶有X-token參數,其他業務參數請寫在請求體中

Postman中如下:

正常響應參數(沒有超過反爬蟲預警)

{
    "code": "00000",
    "msg": "ddd"
}

被攔截的響應參數

當使用postman多次請求,請求超過閾值,將會返回驗證碼頁面(如果前端JQ請求,請參照kk-anti-reptile處理方式正常顯示驗證碼)




	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes">
	<title>kk-anti-reptile驗證</title>
	<script>
		function getXhr() {
            var xhr = null;
            try {
                xhr = new XMLHttpRequest();
            } catch (e) {
                try {
                    xhr = new ActiveXObject("Msxml2.XMLHTTP");
                } catch (e) {
                    xhr = new ActiveXObject("Microsoft.XMLHTTP");
                }
            }
            return xhr;
        }

        function refresh() {
            var xhr = getXhr();
            var verifyId = document.getElementById("verifyId").value;
            var baseUrl = document.getElementById("baseUrl").value;
            xhr.onreadystatechange = function () {
                if (xhr.readyState == 4 && xhr.status == 200) {
                    var verifyObj = JSON.parse(xhr.responseText);
                    document.getElementById("verifyId").value = verifyObj.verifyId;
                    document.getElementById("verifyImg").src = verifyObj.verifyImgStr;
                }
            }
            xhr.open("POST", baseUrl + "/kk-anti-reptile/refresh?verifyId="+verifyId, "true");
            xhr.send();
        }

        function validate() {
            var elements = document.getElementById("verifyFrom");
            var formData = new FormData();
            for(var i = 0; i < elements.length; i++) {
                formData.append(elements[i].name, elements[i].value);
            }
            var baseUrl = document.getElementById("baseUrl").value;
            var xhr = getXhr();
            xhr.onreadystatechange = function () {
                if (xhr.readyState == 4 && xhr.status == 200) {
                    var obj = JSON.parse(xhr.responseText);
                    if (obj.result == true) {
                        closeThisWindows();
                    } else {
                        alert("驗證碼填寫錯誤")
                    }
                }
            }
            xhr.open("POST", baseUrl + "/kk-anti-reptile/validate", "true");
            xhr.send(formData);
        }

        function closeThisWindows() {
            if (navigator.userAgent.indexOf("MSIE") > 0) {
                if (navigator.userAgent.indexOf("MSIE 6.0") > 0) {
                    window.opener = null;
                    window.close();
                } else {
                    window.open('', '_top');
                    window.top.close();
                }
            } else if (navigator.userAgent.indexOf("Firefox") > 0) {
                window.location.href = 'about:blank';
            } else if (navigator.userAgent.indexOf("AppleWebKit") > 0) {
                window.location.href = 'about:blank';
                window.close();
            } else {
                window.opener = null;
                window.open('', '_self', '');
                window.close();
            }
        }
	</script>
	<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
	<meta name="viewport" content="width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no">
	<title>普通驗證碼</title>

	<style>
		* {
			box-sizing: border-box;
			margin: 0;
			padding: 0;
		}

		form {
			width: 240px;
			margin: 100px auto;
			padding: 20px;
		}

		input[type="text"] {
			margin: 10px 0;
			padding: 0 4px;
			width: 100%;
			height: 32px;
			border: 1px solid #c3c3c3;
			border-radius: 4px;
		}

		input[type="button"] {
			width: 100%;
			height: 32px;
			color: #fff;
			background-color: #40a9ff;
			border-color: #40a9ff;
			border-radius: 4px;
			outline: 0;
			cursor: pointer;
			text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.12);
			box-shadow: 0 2px 0 rgba(0, 0, 0, 0.045);
			border-style: none;
		}

		.img-wrapper {
			display: flex;
			align-items: center;
		}

		.img-wrapper img {
			width: 130px;
			height: 48px;
		}

		.img-wrapper a {
			text-decoration: none;
			color: #1890ff;
		}
	</style>




	<form id="verifyFrom" method="post" action="">
		<input type="hidden" id="baseUrl" name="baseUrl">
		<input type="hidden" id="verifyId" name="verifyId" value="d4eee0b1-b416-4231-96d8-8af6f70ad2c2">
		<input type="hidden" id="realRequestUri" name="realRequestUri" value="/test/listOrientations">
		<span>操作頻繁,請輸入驗證碼</span>
		<div class="img-wrapper">
			<img id="verifyImg" src=""> &nbsp;&nbsp;&nbsp;&nbsp;
			<a href="javascript:void(0);" onclick="refresh()">刷新</a>
		</div>
		<input type="text" id="result" name="result">
		<br>
		<input type="button" value="確認" onclick="validate()">
</form>


讚賞我

  1. 如果您覺得這個不錯,可以打賞下作者,您的贊同是作者不懈努力的源泉

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章