spring boot从零搭建登录注册功能并进行所有接口验证 [TOC]
目前大多项目是前后端分离。在后台接口服务开发过程中,往往我们需要先搭建一个基础服务,比如登录注册功能、自动对所有的接口进行token的安全校验等,这样可以防范安全问题的出现。并且这样后续的同事可以只关注业务代码的开发,不需要关心基础架构服务的实现。
这次我准备搭建一个简单的后台服务,用的是spring boot + mysql + mybatis
。
1、搭建Spring boot项目 首先我们使用IDEA自带的初始化项目功能,创建一个Spring boot项目,如图:
或者在线生成,点击进入
pom.xml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 <?xml version="1.0" encoding="UTF-8"?> <project xmlns ="http://maven.apache.org/POM/4.0.0" xmlns:xsi ="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation ="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd" > <modelVersion > 4.0.0</modelVersion > <parent > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-parent</artifactId > <version > 2.1.6.RELEASE</version > </parent > <groupId > com.zz</groupId > <artifactId > rest-api</artifactId > <version > 0.0.1-SNAPSHOT</version > <name > rest-api</name > <description > Demo project for Spring Boot</description > <properties > <java.version > 1.8</java.version > </properties > <dependencies > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-web</artifactId > </dependency > <dependency > <groupId > com.baomidou</groupId > <artifactId > mybatis-plus-boot-starter</artifactId > <version > 3.1.0</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-test</artifactId > <scope > test</scope > </dependency > <dependency > <groupId > com.jayway.jsonpath</groupId > <artifactId > json-path</artifactId > <scope > test</scope > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-autoconfigure</artifactId > <version > 2.1.6.RELEASE</version > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-data-rest</artifactId > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-data-mongodb</artifactId > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-configuration-processor</artifactId > <optional > true</optional > </dependency > <dependency > <groupId > com.google.code.gson</groupId > <artifactId > gson</artifactId > <version > 2.8.5</version > <scope > compile</scope > </dependency > <dependency > <groupId > mysql</groupId > <artifactId > mysql-connector-java</artifactId > </dependency > <dependency > <groupId > commons-io</groupId > <artifactId > commons-io</artifactId > <version > 2.6</version > </dependency > <dependency > <groupId > io.jsonwebtoken</groupId > <artifactId > jjwt</artifactId > <version > 0.9.1</version > </dependency > <dependency > <groupId > com.auth0</groupId > <artifactId > java-jwt</artifactId > <version > 3.4.0</version > </dependency > <dependency > <groupId > commons-codec</groupId > <artifactId > commons-codec</artifactId > </dependency > <dependency > <groupId > com.github.terran4j</groupId > <artifactId > terran4j-commons-api2doc</artifactId > <version > 1.0.2</version > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-devtools</artifactId > <optional > true</optional > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-thymeleaf</artifactId > </dependency > <dependency > <groupId > com.itextpdf.tool</groupId > <artifactId > xmlworker</artifactId > <version > 5.5.10</version > </dependency > <dependency > <groupId > org.apache.poi</groupId > <artifactId > poi-ooxml</artifactId > <version > 3.15</version > </dependency > <dependency > <groupId > org.apache.poi</groupId > <artifactId > poi-scratchpad</artifactId > <version > 3.15</version > </dependency > </dependencies > <build > <plugins > <plugin > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-maven-plugin</artifactId > </plugin > </plugins > </build > </project >
maven更改为国内阿里云镜像,这样比较快
settings.xml:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 <settings xmlns ="http://maven.apache.org/SETTINGS/1.0.0" xmlns:xsi ="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation ="http://maven.apache.org/SETTINGS/1.0.0 https://maven.apache.org/xsd/settings-1.0.0.xsd" > <mirrors > <mirror > <id > alimaven</id > <name > aliyun maven</name > <url > http://maven.aliyun.com/nexus/content/groups/public/</url > <mirrorOf > central</mirrorOf > </mirror > </mirrors > </settings >
Mybatis 推荐插件如下:
application.yml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 spring: jpa: show-sql: true hibernate: ddl-auto: update servlet: multipart: max-file-size: 10MB max-request-size: 10MB profiles: active: dev mvc: static-path-pattern: /** resources: static-locations: file:/Users/wz/projects/blog/uploadFile/,classpath:/static/,classpath:/resources/,classpath:/file/,classpath:/templates/ mybatis-plus: mapper-locations: classpath:/mapper/*.xml type-aliases-package: com.zz.entity my: tokenURL: "55555" authURL: "88888"
application-dev.yml
1 2 3 4 5 6 7 8 9 spring: datasource: driver-class-name: com.mysql.cj.jdbc.Driver url: jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8 username: root password: ****** server: port: 8080
大概目录结构如下
搭建细节不再赘述;
2、读取自定义配置文件
com.zz.config.MyConfiguration
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 package com.zz.config;import org.springframework.boot.context.properties.ConfigurationProperties;import org.springframework.context.annotation.Configuration;@Configuration @ConfigurationProperties (prefix = "my" )public class MyConfiguration { private String tokenURL; private String authURL; public String getAuthURL () { return this .authURL; } public void setAuthURL (String authURL) { this .authURL = authURL; } public String getTokenURL () { return this .tokenURL; } public void setTokenURL (String tokenURL) { this .tokenURL = tokenURL; } }
3、web服务配置
com.zz.config.MyConfiguration
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 package com.zz.config;import com.zz.common.interceptor.AuthenticationInterceptor;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.web.servlet.config.annotation.InterceptorRegistry;import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;@Configuration public class WebConfig implements WebMvcConfigurer { @Override public void addInterceptors (InterceptorRegistry registry) { registry.addInterceptor(authenticationInterceptor()) .addPathPatterns("/**" ); } @Bean public AuthenticationInterceptor authenticationInterceptor () { return new AuthenticationInterceptor(); } }
4、自定义返回统一的实体类Response
com.zz.model.Response
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 package com.zz.model;public class Response { private int code; private String msg; private Object data; public Object getData () { return data; } public void setData (Object data) { this .data = data; } public int getCode () { return code; } public void setCode (int code) { this .code = code; } public String getMsg () { return msg; } public void setMsg (String msg) { this .msg = msg; } }
5、Utils公共方法类
com.zz.utils.HttpUtils
获取Request、 Response、session
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 package com.zz.utils;import org.springframework.web.context.request.RequestContextHolder;import org.springframework.web.context.request.ServletRequestAttributes;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import javax.servlet.http.HttpSession;public class HttpUtils { public static HttpServletRequest getRequest () { ServletRequestAttributes requestAttributes = (ServletRequestAttributes)RequestContextHolder.getRequestAttributes(); if (requestAttributes == null ) return null ; return requestAttributes.getRequest(); } public static HttpServletResponse getResponse () { ServletRequestAttributes requestAttributes = (ServletRequestAttributes)RequestContextHolder.getRequestAttributes(); if (requestAttributes == null ) return null ; return requestAttributes.getResponse(); } public static HttpSession getSession () { HttpServletRequest request = getRequest(); if (request == null ) return null ; return request.getSession(); } }
com.zz.utils.JWTUtils
JWT 生成token, 验证token
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 package com.zz.utils;import com.zz.entity.User;import io.jsonwebtoken.*;import org.apache.commons.codec.binary.Base64;import javax.crypto.SecretKey;import javax.crypto.spec.SecretKeySpec;import java.util.Date;import java.util.HashMap;import java.util.Map;import java.util.UUID;public class JWTUtils { private static final String SECRETKEY = "KJHUhjjJYgYUllVbXhKDHXhkSyHjlNiVkYzWTBac1Yxkjhuad" ; private static long expirationDate = 2 * 60 * 60 ; private static SecretKey generalKey (String stringKey) { byte [] encodedKey = Base64.decodeBase64(stringKey); return new SecretKeySpec(encodedKey, 0 , encodedKey.length, "AES" ); } public static String createToken (User user) { SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256; long nowMillis = System.currentTimeMillis(); Date now = new Date(nowMillis); Map<String, Object> claims = new HashMap<>(); claims.put("userId" , user.getUserId()); claims.put("userName" , user.getUserName()); claims.put("password" , user.getPassword()); SecretKey key = generalKey(SECRETKEY + user.getPassword()); HashMap<String, Object> storeInfo = new HashMap<String, Object>(); storeInfo.put("userId" , user.getUserId()); storeInfo.put("userName" , user.getUserName()); String subject = storeInfo.toString(); JwtBuilder builder = Jwts.builder() .setClaims(claims) .setId(UUID.randomUUID().toString()) .setIssuedAt(now) .setSubject(subject) .signWith(signatureAlgorithm, key); if (expirationDate >= 0 ) { long expMillis = nowMillis + expirationDate * 1000 ; Date exp = new Date(expMillis); builder.setExpiration(exp); } return builder.compact(); } public static Claims parseToken (String token, User user) { SecretKey key = generalKey(SECRETKEY + user.getPassword()); Claims claims = Jwts.parser() .setSigningKey(key) .parseClaimsJws(token).getBody(); return claims; } public static Boolean verify (String token, User user) { Claims claims = parseToken(token, user); return claims.get("password" ).equals(user.getPassword()); } }
6、查询实体类 query 所有的服务查询都采用统一的各自的实体类
比如:
com.zz.query.UserQuery
用户查询实体
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 package com.zz.query;public class UserQuery { private String userName; private String password; private long userId; private boolean showPassword; public boolean isShowPassword () { return showPassword; } public void setShowPassword (boolean showPassword) { this .showPassword = showPassword; } public long getUserId () { return userId; } public void setUserId (long userId) { this .userId = userId; } public String getUserName () { return userName; } public void setUserName (String userName) { this .userName = userName; } public String getPassword () { return password; } public void setPassword (String password) { this .password = password; } }
7、查询后返回实体类 所有的服务查询返回都采用统一的各自的实体类
比如:
com.zz.entity.User
用户数据返回实体
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 package com.zz.entity;public class User { private long userId; private String userName; private String token; private String password; public String getPassword () { return password; } public void setPassword (String password) { this .password = password; } public String getToken () { return token; } public void setToken (String token) { this .token = token; } public long getUserId () { return userId; } public void setUserId (long userId) { this .userId = userId; } public String getUserName () { return userName; } public void setUserName (String userName) { this .userName = userName; } }
8 、接口实现三层架构 我们这采取的是三层架构:controller —> service —> mapper;
如果我们要写一个User类接口,先声明一个UserController路由控制层,然后这个里调用UserService实现类方法,然后再调用mapper持久层去CRUD(mysql增查删改)。
9、开始搭建注册用户功能 基础搭建先暂停,开始实质业务的推演;
mysql的连接就不多说啦;
让我们开始实现之旅吧;
com.zz.newController.UserController
用户注册
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 package com.zz.newController;import com.auth0.jwt.JWT;import com.auth0.jwt.interfaces.DecodedJWT;import com.zz.common.annotation.PassToken;import com.zz.common.base.BaseApplicationController;import com.zz.entity.User;import com.zz.model.Response;import com.zz.query.UserQuery;import com.zz.service.UserService;import com.zz.utils.JWTUtils;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.web.bind.annotation.*;import java.util.Map;@RestController @RequestMapping ("/user" )public class UserController { @Autowired private UserService userService; @PostMapping ("/add" ) @PassToken public Response addUser (@RequestParam String userName, @RequestParam String password, Response response) { UserQuery query = new UserQuery(); User userData = null ; query.setUserName(userName); query.setPassword(password); int result; String message = "" ; UserQuery findUserQuery = new UserQuery(); findUserQuery.setUserName(userName); User existUser = this .userService.findUserByName(findUserQuery); if (existUser == null ) { try { result = this .userService.addUser(query); message = "success" ; } catch (Exception e) { result = 0 ; message = "error" ; e.printStackTrace(); } if (result == 1 ) { userData = this .userService.findUserByName(findUserQuery); String token = null ; User currentUser = new User(); if (userData != null ) { currentUser.setUserId(userData.getUserId()); currentUser.setUserName(userData.getUserName()); currentUser.setPassword(password); token = JWTUtils.createToken(currentUser); } if (token != null ) { userData.setToken(token); } } } else { message = "用户已经存在" ; } response.setData(userData); response.setMsg(message); return response; } }
com.zz.service.UserService
Interface 用户接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 package com.zz.service;import com.zz.entity.User;import com.zz.query.UserQuery;import java.util.List;import java.util.Map;public interface UserService { int addUser (UserQuery query) ; }
com.zz.service.impl.UserServiceImpl
用户接口实现类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 package com.zz.service.impl;import com.zz.entity.User;import com.zz.mapper.UserMapper;import com.zz.query.UserQuery;import com.zz.service.UserService;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Service;import java.util.List;import java.util.Map;@Service public class UserServiceImpl implements UserService { @Autowired private UserMapper userMapper; @Override public int addUser (UserQuery query) { return this .userMapper.insert(query); } }
com.zz.mapper.UserMapper
mapper
1 2 3 4 5 6 7 8 9 10 11 12 package com.zz.mapper;import com.zz.entity.User;import com.zz.query.UserQuery;import java.util.List;public interface UserMapper { int insert (UserQuery query) ; }
resources/mapper/UserMapper.xml
前后名字一定对应
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" > <mapper namespace ="com.zz.mapper.UserMapper" > <resultMap id ="BaseResult" type ="com.zz.entity.User" > <id column ="user_id" property ="userId" > </id > <id column ="user_name" property ="userName" > </id > </resultMap > <sql id ="base" > user_id, user_name <if test ="showPassword" > , password </if > </sql > <sql id ="base_condition" > <where > <if test ="userName!=null and userName!=''" > user_name=#{userName} </if > <if test ="password!=null and password!=''" > and password=#{password} </if > </where > </sql > <insert id ="insert" > INSERT INTO user( user_name, password ) VALUES ( #{userName}, #{password} ) </insert > </mapper >
到此,整个接口书写过程已全部完成,这就是在当前架构下写一个接口的全部过程。
10、搭建web实例 ——注册用户 由于我们在配置文件里已经配置静态资源的路径,所以我们可以在resources里面写一个不分离的we b实例进行访问。
resources/static/regist.html
注册页面
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 <!DOCTYPE html > <html lang ="en" > <head > <meta charset ="UTF-8" > <meta name ="viewport" content ="width=device-width,initial-scale=1,user-scalable=0" > <title > 注册用户</title > <link rel ="stylesheet" href ="css/regist.css" /> <link rel ="stylesheet" href ="https://res.wx.qq.com/open/libs/weui/2.1.3/weui.min.css" > </head > <body > <div class ="container" > <div class ="page form_page js_show" > <div class ="weui-form" > <div class ="weui-form__text-area" > <h2 class ="weui-form__title" > 注册新用户</h2 > </div > <div class ="weui-form__control-area" > <div class ="weui-cells__group weui-cells__group_form" > <div class ="weui-cells weui-cells_form" > <div class ="weui-cell" > <div class ="weui-cell__hd" > <label class ="weui-label" > 用户名</label > </div > <div class ="weui-cell__bd" > <input id ="js_input——user" class ="weui-input" placeholder ="请输入要设置的用户名" > </div > </div > <div class ="weui-cell" > <div class ="weui-cell__hd" > <label class ="weui-label" > 密码</label > </div > <div class ="weui-cell__bd" > <input id ="js_input——pwd" type ="password" class ="weui-input" placeholder ="请输入要设置的密码" > </div > </div > <div class ="weui-cell" > <div class ="weui-cell__hd" > <label class ="weui-label" > 确认密码</label > </div > <div class ="weui-cell__bd" > <input id ="js_input——pwd2" type ="password" class ="weui-input" placeholder ="请再次输入设置的密码" type ="number" pattern ="[0-9]*" > </div > </div > </div > </div > </div > <div class ="weui-form__opr-area" > <a class ="weui-btn weui-btn_primary" href ="javascript:" id ="submit" > 确定</a > </div > <div class ="weui-form__extra-area" > <div class ="weui-footer" > <p class ="weui-footer__text" > Copyright © 2019 alex wong</p > </div > </div > </div > <div id ="js_toast" style ="display: none;" > <div class ="weui-mask_transparent" > </div > <div class ="weui-toast" > <i class ="weui-icon-success-no-circle weui-icon_toast" > </i > <p class ="weui-toast__content" > 已完成</p > </div > </div > </div > </div > </body > <script src ="js/md5.js" > </script > <script src ="js/utils.js" > </script > <script src ="js/dataService.js" > </script > <script type ="text/javascript" src ="https://res.wx.qq.com/open/libs/weuijs/1.2.1/weui.min.js" > </script > <script src ="js/regist.js" > </script > </html >
static/js/dataService.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 const APIURL = '/' ;window .dataService = { get : (url, params = {}) => { const searchArr = []; Object .keys(params).forEach(n => { searchArr.push(`${n} =${params[n]} ` ); }); const searchStr = searchArr.length ? '?' + searchArr.join('&' ) : '' ; const token = utils.getCookie('token' ); return fetch(APIURL + url + searchStr, { method: 'GET' , headers: { token } }).then(res => res.json()); }, post: (url, params = {} ) => { const formData = new FormData(); Object .keys(params).forEach(n => { formData.append(n, params[n]); }); const token = utils.getCookie('token' ); return fetch(APIURL + url, { method: 'POST' , headers: { token }, body: formData }).then(res => res.json()); }, addUser(params) { return this .post('user/add' , params); }, login(params) { return this .post('user/login' , params); }, getUserInfo(params) { return this .get('user/info' , params); }, };
static/js/utils.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 window .utils = { generateMd5(userName, password) { const salt = "1qaz2wsx3edc4rfv5tgb6yhn7ujm8ik9ol0p@!." ; const asciStr = userName + salt + password; const asciArr = asciStr.split('' ); const asciResult = []; asciArr.forEach(n => { asciResult.push(n.charCodeAt()); }); const ascireusltStr = asciResult.join(salt); return hex_md5(ascireusltStr); }, setCookie(name, value) { var time = 2 * 60 * 60 * 1000 ; var exp = new Date (); exp.setTime(exp.getTime() + time); document .cookie = name + "=" + escape (value) + ";expires=" + exp.toGMTString(); }, getCookie(name) { var arr, reg = new RegExp ("(^| )" + name + "=([^;]*)(;|$)" ); if (arr = document .cookie.match(reg)) return unescape (arr[2 ]); else return null ; } };
static/js/regist.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 const userNameInput = document .getElementById("js_input——user" );const passwordInput = document .getElementById("js_input——pwd" );const passwordConfirmInput = document .getElementById("js_input——pwd2" );const submitBtn = document .getElementById("submit" );submitBtn.onclick = () => { const userName = userNameInput.value; const password = passwordInput.value; const confirmPassword = passwordConfirmInput.value; if (!userName) { weui.topTips('用户姓名不能为空' ); return ; } else if (!password) { weui.topTips('用户密码不能为空' ); return ; } else if (confirmPassword !== password) { weui.topTips('前后密码不一致,请重试' ); return ; } const newPassword = utils.generateMd5(userName, password); dataService.addUser({ userName, password: newPassword, }).then(res => { const {code, data, msg} = res; if (!data) { weui.topTips(msg); } else { weui.topTips(`注册成功,欢迎 ${data.userName} ` ); window .location.href = location.origin + '/login.html' ; } }) };
效果如图:
增加一些基本的校验
用户密码加密传输,并验证新用户是否已经注册
mysql 查看下用户表
11、后端-用户登录功能 按上面第9步骤所述,下面的添加内容,请直接添加到上述服务中,不再全部展示代码。
com.zz.newController.UserController
首先要判断用户是否存在,如果存在,返回基本信息并返回用户凭证token
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 @PostMapping ("/login" ) @PassToken public Response login (@RequestParam String userName, @RequestParam String password, Response response) { UserQuery query = new UserQuery(); query.setUserName(userName); query.setPassword(password); try { User existUser = this .userService.findUserByName(query); String token = null ; User currentUser = new User(); if (existUser != null ) { currentUser.setUserId(existUser.getUserId()); currentUser.setUserName(existUser.getUserName()); currentUser.setPassword(password); token = JWTUtils.createToken(currentUser); if (token != null ) { existUser.setToken(token); } response.setMsg("success" ); response.setData(existUser); } else { response.setMsg("登录失败,请检查用户名和密码" ); response.setData(null ); } } catch (Exception e) { response.setMsg("login failed" ); response.setData(null ); e.printStackTrace(); } return response; }
com.zz.service.UserService
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 package com.zz.service;import com.zz.entity.User;import com.zz.query.UserQuery;import java.util.List;import java.util.Map;public interface UserService { int addUser (UserQuery query) ; User findUserById (UserQuery query) ; User findUserByName (UserQuery query) ; }
com.zz.service.impl.UserServiceImpl
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 package com.zz.service.impl;import com.zz.entity.User;import com.zz.mapper.UserMapper;import com.zz.query.UserQuery;import com.zz.service.UserService;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Service;import java.util.List;import java.util.Map;@Service public class UserServiceImpl implements UserService { @Autowired private UserMapper userMapper; @Override public int addUser (UserQuery query) { return this .userMapper.insert(query); } @Override public User findUserById (UserQuery query) { return this .userMapper.findUserById(query); } @Override public User findUserByName (UserQuery query) { return this .userMapper.findUserByName(query); } @Override public List<User> findAllUser (UserQuery query) { return this .userMapper.findAllUser(query); } }
com.zz.mapper.UserMapper
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 package com.zz.mapper;import com.zz.entity.User;import com.zz.query.UserQuery;import java.util.List;public interface UserMapper { int insert (UserQuery query) ; User findUserById (UserQuery query) ; User findUserByName (UserQuery query) ; List<User> findAllUser (UserQuery query) ; }
mapper/UserMapper.xml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" > <mapper namespace="com.zz.mapper.UserMapper"> <resultMap id="BaseResult" type="com.zz.entity.User"> <id column="user_id" property="userId"></id> <id column="user_name" property="userName"></id> </resultMap> <sql id="base"> user_id, user_name <if test="showPassword"> , password </if> </sql> <sql id="base_condition"> <where> <if test="userName!=null and userName!=''"> user_name=#{userName} </if> <if test="password!=null and password!=''"> and password=#{password} </if> </where> </sql> <!-- 查询所有user --> <select id="findAllUser" resultMap="BaseResult"> select <include refid="base"/> from user </select> <!-- 查询user --> <select id="findUserById" resultMap="BaseResult"> select <include refid="base"/> from user where user_id = </select> <select id="findUserByName" resultMap="BaseResult"> select <include refid="base"/> from user <include refid="base_condition"/> </select> <insert id="insert"> INSERT INTO user( user_name, password ) VALUES ( ) </insert> </mapper>
12、搭建web实例 ——登录用户
static/login.html
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 <!DOCTYPE html > <html lang ="en" > <head > <meta charset ="UTF-8" > <meta name ="viewport" content ="width=device-width,initial-scale=1,user-scalable=0" > <title > login</title > <link rel ="stylesheet" href ="css/regist.css" /> <link rel ="stylesheet" href ="https://res.wx.qq.com/open/libs/weui/2.1.3/weui.min.css" > </head > <body > <div class ="container" > <div class ="page form_page js_show" > <div class ="weui-form" > <div class ="weui-form__text-area" > <h2 class ="weui-form__title" > 登录</h2 > </div > <div class ="weui-form__control-area" > <div class ="weui-cells__group weui-cells__group_form" > <div class ="weui-cells weui-cells_form" > <div class ="weui-cell" > <div class ="weui-cell__hd" > <label class ="weui-label" > 用户名</label > </div > <div class ="weui-cell__bd" > <input id ="js_input——user" class ="weui-input" placeholder ="请输入用户名" > </div > </div > <div class ="weui-cell" > <div class ="weui-cell__hd" > <label class ="weui-label" > 密码</label > </div > <div class ="weui-cell__bd" > <input id ="js_input——pwd" type ="password" class ="weui-input" placeholder ="请输入密码" > </div > </div > </div > </div > </div > <div class ="weui-form__opr-area" > <a class ="weui-btn weui-btn_primary" href ="javascript:" id ="submit" > 确定</a > </div > <div class ="weui-form__extra-area" > <div class ="weui-footer" > <p class ="weui-footer__text" > Copyright © 2019 alex wong</p > </div > </div > </div > <div id ="js_toast" style ="display: none;" > <div class ="weui-mask_transparent" > </div > <div class ="weui-toast" > <i class ="weui-icon-success-no-circle weui-icon_toast" > </i > <p class ="weui-toast__content" > 已完成</p > </div > </div > </div > </div > </body > <script src ="js/md5.js" > </script > <script src ="js/utils.js" > </script > <script src ="js/dataService.js" > </script > <script type ="text/javascript" src ="https://res.wx.qq.com/open/libs/weuijs/1.2.1/weui.min.js" > </script > <script src ="js/login.js" > </script > </html >
static/js/login.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 const userNameInput = document .getElementById("js_input——user" );const passwordInput = document .getElementById("js_input——pwd" );const submitBtn = document .getElementById("submit" );submitBtn.onclick = () => { const userName = userNameInput.value; const password = passwordInput.value; if (!userName) { weui.topTips('用户姓名不能为空' ); return ; } else if (!password) { weui.topTips('用户密码不能为空' ); return ; } const newPassword = utils.generateMd5(userName, password); dataService.login({ userName, password: newPassword, }).then(res => { const {code, data, msg} = res; if (!data) { weui.topTips(msg); } else { weui.topTips(`登录成功,欢迎 ${data.userName} ` ); utils.setCookie('token' , data.token); location.href = location.origin + '/home.html' ; } }) };
登录接口返回用户凭证token,后续用来校验用户接口,增加安全性。
13、增加自定义注解和拦截器 在常规的业务开发中,切记不可把接口服务暴露给任何人都可以访问,不然别人可以任意查看或者修改你的数据,这是很严重的事情。除了常规从网段IP方面限制固定客户端IP的范围,接口本身也要增加安全验证,这个时候我们就需要用到之前生成的用户凭证token;
问题是我们如果自定义控制,哪些接口是需要经过验证,哪些接口是不需要通过验证的呢?有人可能会说,直接全部验证不就可以啦,何苦纠结。但是在真实的业务中,有些接口是不能强制校验的,比如一些用户分享到微信的那种接口,是不能增加验证,否则分享的页面无法正常显示。
所以我们可以自定义注解@PassToken, 添加这个注解的接口,就可以不用进行token验证了。
com.zz.common.annotation.PassToken
PassToken 注解
1 2 3 4 5 6 7 8 9 10 11 12 13 package com.zz.common.annotation;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;@Target ({ElementType.METHOD, ElementType.TYPE})@Retention (RetentionPolicy.RUNTIME)public @interface PassToken { boolean required () default true ; }
添加拦截器
com.zz.common.interceptor.AuthenticationInterceptor
在发送请求的时候,在请求头里面加token, 然后验证的时候token从头部获取
如果没有token, 进行无token提示;
如果存在,就用 JWT 校验 token 是否存在,并且校验用户密码是否正确。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 package com.zz.common.interceptor;import com.auth0.jwt.JWT;import com.auth0.jwt.exceptions.JWTDecodeException;import com.mongodb.util.JSON;import com.zz.common.annotation.PassToken;import com.zz.common.base.BaseApplicationController;import com.zz.entity.User;import com.zz.model.Response;import com.zz.query.UserQuery;import com.zz.service.UserService;import com.zz.utils.JWTUtils;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.boot.configurationprocessor.json.JSONException;import org.springframework.boot.configurationprocessor.json.JSONObject;import org.springframework.web.method.HandlerMethod;import org.springframework.web.servlet.HandlerInterceptor;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import java.lang.reflect.Method;public class AuthenticationInterceptor implements HandlerInterceptor { @Autowired private UserService userService; public JSONObject getJsonObject (int code, String message) throws JSONException { JSONObject jsonObject = new JSONObject(); jsonObject.put("msg" , message); jsonObject.put("code" , code); return jsonObject; } @Override public boolean preHandle (HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object object) throws Exception { String token = BaseApplicationController.getToken(); if (!(object instanceof HandlerMethod)) { return true ; } HandlerMethod handlerMethod = (HandlerMethod) object; Method method = handlerMethod.getMethod(); if (method.isAnnotationPresent(PassToken.class )) { PassToken passToken = method.getAnnotation(PassToken.class ) ; if (passToken.required()) { return true ; } } httpServletResponse.setContentType("application/json;charset=UTF-8" ); if (token == null || token.equals("null" )) { JSONObject jsonObject = getJsonObject(403 , "无token,请重新登录" ); httpServletResponse.getWriter().write(jsonObject.toString()); return false ; } long userId; try { userId = BaseApplicationController.getCurrentUserId(); } catch (JWTDecodeException j) { JSONObject jsonObject = getJsonObject(500 , "访问异常, token不正确,请重新登录" ); httpServletResponse.getWriter().write(jsonObject.toString()); return false ; } UserQuery query = new UserQuery(); query.setUserId(userId); query.setShowPassword(Boolean.TRUE); User user = userService.findUserById(query); if (user == null ) { JSONObject jsonObject = getJsonObject(500 , "用户不存在,请重新登录" ); httpServletResponse.getWriter().write(jsonObject.toString()); return false ; } Boolean verify = JWTUtils.verify(token, user); if (!verify) { JSONObject jsonObject = getJsonObject(500 , "非法访问,请重新登录" ); httpServletResponse.getWriter().write(jsonObject.toString()); return false ; } return true ; } }
下面让我们实例看下效果:
com.zz.newController.UserController
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 package com.zz.newController;import com.auth0.jwt.JWT;import com.auth0.jwt.interfaces.DecodedJWT;import com.zz.common.annotation.PassToken;import com.zz.common.base.BaseApplicationController;import com.zz.entity.User;import com.zz.model.Response;import com.zz.query.UserQuery;import com.zz.service.UserService;import com.zz.utils.JWTUtils;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.web.bind.annotation.*;import java.util.Map;@RestController @RequestMapping ("/user" )public class UserController { @Autowired private UserService userService; @PostMapping ("/add" ) @PassToken public Response addUser (@RequestParam String userName, @RequestParam String password, Response response) { UserQuery query = new UserQuery(); User userData = null ; query.setUserName(userName); query.setPassword(password); int result; String message = "" ; UserQuery findUserQuery = new UserQuery(); findUserQuery.setUserName(userName); User existUser = this .userService.findUserByName(findUserQuery); if (existUser == null ) { try { result = this .userService.addUser(query); message = "success" ; } catch (Exception e) { result = 0 ; message = "error" ; e.printStackTrace(); } if (result == 1 ) { userData = this .userService.findUserByName(findUserQuery); String token = null ; User currentUser = new User(); if (userData != null ) { currentUser.setUserId(userData.getUserId()); currentUser.setUserName(userData.getUserName()); currentUser.setPassword(password); token = JWTUtils.createToken(currentUser); } if (token != null ) { userData.setToken(token); } } } else { message = "用户已经存在" ; } response.setData(userData); response.setMsg(message); return response; } @PostMapping ("/login" ) @PassToken public Response login (@RequestParam String userName, @RequestParam String password, Response response) { UserQuery query = new UserQuery(); query.setUserName(userName); query.setPassword(password); try { User existUser = this .userService.findUserByName(query); String token = null ; User currentUser = new User(); if (existUser != null ) { currentUser.setUserId(existUser.getUserId()); currentUser.setUserName(existUser.getUserName()); currentUser.setPassword(password); token = JWTUtils.createToken(currentUser); if (token != null ) { existUser.setToken(token); } response.setMsg("success" ); response.setData(existUser); } else { response.setMsg("登录失败,请检查用户名和密码" ); response.setData(null ); } } catch (Exception e) { response.setMsg("login failed" ); response.setData(null ); e.printStackTrace(); } return response; } @GetMapping ("/info" ) public Response getUserInfo (Response response) { String token = BaseApplicationController.getToken(); User userData2 = BaseApplicationController.getCurrentUser(); Map<String, Object> headerData = BaseApplicationController.getHeader(); if (token != null && !token.equals("null" )) { User userData = new User(); DecodedJWT claims = JWT.decode(token); userData.setUserName(claims.getClaim("userName" ).asString()); userData.setUserId(claims.getClaim("userId" ).asLong()); response.setData(userData); response.setMsg("success" ); } else { response.setMsg("token不存在" ); } return response; } }
我们新写了一个接口,获取用户信息,如上,其余代码不再赘述;
成功获取用户信息
删除token
Token 故意改错
到此,验证过程完美成功;
14、总结 以上过程,只是一个简单服务的搭建,真实的服务还需要更多配置,比如XSS配置防止XSS攻击等。好了,本篇文章到此为止,如果有哪里描述得不清楚,请多多包涵。
打赏