vue + jwt + redis 实现验证码功能

3 篇文章 0 订阅

实现基本思路

登录界面向后台请求验证码,后台呢就 先调用随机函数生成验证码,并且更具验证码生成一张图片,以 base64 字符串的形式传到前台,这时我们还要生成jwt令牌做为请求验证码客户端的区分。我们先将验证码信息存入reids。key是 jwt令牌的值,value就是验证码了。并且将令牌放入到响应头。传给客户端。当客户端提交的时候将保持的jwt令牌放入请求头带过来。后端根据前端传过来的 jwt令牌去redis中获取数据,将验证码拿到后和现有的验证码进行比较。看看是否相等。这就是实现验证码的基本思路了。更多细节我将一起去看代码吧

后端代码

UserController
请求验证码图片和用户登录都是在这里了

@Controller
@RequestMapping("/user")
public class UserController {
    private static final String VERIFICATION_CODE = "verificationCode_";

    @Autowired
    private UserService userService;

    @Autowired
    private RedisTemplate redisTemplate;

    @RequestMapping("/login")
    @ResponseBody
    public R login(HttpServletRequest request, HttpServletResponse response){
        //获取用户输入的验证码
        String userVerificationCode = request.getParameter("verificationCode");
        //获取验证码jwt令牌
        String userJwt = request.getHeader(JwtUtils.JWT_VERIFICATION_KEY);
        //获取到保存在redis中的验证码
        Object redisVerificationCode =  redisTemplate.opsForValue().get(VERIFICATION_CODE + userJwt) ;

        if(StringUtils.isEmpty(redisVerificationCode)){
            return R.error("你的验证码已超时");
        }

        if(!redisVerificationCode.toString().equalsIgnoreCase(userVerificationCode)){
            return R.error("验证码错误");
        }

        User user = userService.login(MapUtil.conversion(request));
        //判断是否登录成功
        if(user != null){
            Map<String,Object> map=new HashMap<String, Object>();
            map.put("User", user);
            //这是颁发用户登录成功的jwt令牌
            String jwt= JwtUtils.createJwt(map, JwtUtils.JWT_WEB_TTL);
            response.setHeader(JwtUtils.JWT_HEADER_KEY, jwt);
            return R.ok(user);
        }else {
            return R.okFalse("密码或账户错误");
        }
        
    }




    /**生成图片验证码*/
    @RequestMapping("/verificationCode")
    @ResponseBody
    public String verificationCode(HttpServletRequest req, HttpServletResponse resp) throws IOException {
        //生成验证码随机数
        String word = VerifyCodeUtil.produceNumAndChar(4);
//        获取用户的jwt令牌
        String userVerificationJwt = req.getHeader(JwtUtils.JWT_VERIFICATION_KEY);
        //验证码令牌
        Claims claims = JwtUtils.validateJwtToken(userVerificationJwt);
        if(claims == null){
            //如果用户令牌过期那么对应存放在redis中的数据也要清空
            if(!StringUtils.isEmpty(userVerificationJwt)){
                redisTemplate.expire(VERIFICATION_CODE + userVerificationJwt, 1, TimeUnit.DAYS);
            }
            userVerificationJwt =  JwtUtils.createJwt(new HashMap<String, Object>() ,JwtUtils.JWT_WEB_TTL);
            //将jwt令牌放入 response head中
            resp.setHeader(JwtUtils.JWT_VERIFICATION_KEY, userVerificationJwt);
        }
        //刷新缓存,更新验证码
        redisTemplate.opsForValue().set(VERIFICATION_CODE + userVerificationJwt , word,60, TimeUnit.SECONDS);
        //生成图片
        String code = "data:image/png;base64," + ImageUtil.createImageWithVerifyCode(word, 116,40);;
        return code;
    }
}

VerifyCodeUtil

生成验证码随机数工具类

public class VerifyCodeUtil {


    /**生成N位数字和字母混合的验证码
     * @param  num 验证码位数
     * @return code 生成的验证码字符串*/
    public static String produceNumAndChar(int num){
        Random random = new Random();
        String code = "";
        String ch = "ABCDEFGHIJKLMNPQRSTUVWXYZ";
        String n = "123456789";
        for(int i=0;i<num;i++){
            int flag = random.nextInt(2);
            if(flag==0){//数字
                code+=n.charAt(random.nextInt(n.length()));
            }else{//字母
                code+=ch.charAt(random.nextInt(ch.length()));
            }
        }
        return code;
    }
}

ImageUtil
生成 base64 格式生成的验证码图片 的工具类

public class ImageUtil {

    /**
     * 根据指定的随机数 生成验证码图片 转 base64
     * @param word 要生存的验证码随机字符串
     * @param width 图片宽度
     * @param height 图片高度
     * @return base64 格式生成的验证码图片
     * @throws IOException
     */
    public static String createImageWithVerifyCode(String word, int width, int height) throws IOException {
        String png_base64="";
        //绘制内存中的图片
        BufferedImage bufferedImage = new BufferedImage(width,height,BufferedImage.TYPE_INT_RGB);
        //得到画图对象
        Graphics graphics = bufferedImage.getGraphics();
        //绘制图片前指定一个颜色
        graphics.setColor(getRandColor(160,200));
        graphics.fillRect(0,0,width,height);
        //绘制边框
        graphics.setColor(Color.white);
        graphics.drawRect(0, 0, width - 1, height - 1);
        // 步骤四 四个随机数字
        Graphics2D graphics2d = (Graphics2D) graphics;
        graphics2d.setFont(new Font("宋体", Font.BOLD, 18));
        Random random = new Random();
        // 定义x坐标
        int x = 10;
        for (int i = 0; i < word.length(); i++) {
            // 随机颜色
            graphics2d.setColor(new Color(20 + random.nextInt(110), 20 + random.nextInt(110), 20 + random.nextInt(110)));
            // 旋转 -30 --- 30度
            int jiaodu = random.nextInt(60) - 30;
            // 换算弧度
            double theta = jiaodu * Math.PI / 180;
            // 获得字母数字
            char c = word.charAt(i);
            //将c 输出到图片
            graphics2d.rotate(theta, x, 20);
            graphics2d.drawString(String.valueOf(c), x, 20);
            graphics2d.rotate(-theta, x, 20);
            x += 30;
        }
        // 绘制干扰线
        graphics.setColor(getRandColor(160, 200));
        int x1;
        int x2;
        int y1;
        int y2;
        for (int i = 0; i < 30; i++) {
            x1 = random.nextInt(width);
            x2 = random.nextInt(12);
            y1 = random.nextInt(height);
            y2 = random.nextInt(12);
            graphics.drawLine(x1, y1, x1 + x2, x2 + y2);
        }
        graphics.dispose();// 释放资源
        ByteArrayOutputStream baos = new ByteArrayOutputStream();//io流
        ImageIO.write(bufferedImage, "png", baos);//写入流中
        byte[] bytes = baos.toByteArray();//转换成字节
        BASE64Encoder encoder = new BASE64Encoder();
        png_base64 = encoder.encodeBuffer(bytes).trim();
        png_base64 = png_base64.replaceAll("\n", "").replaceAll("\r", "");//删除 \r\n
        return png_base64;
    }



    /**设置随机颜色*/
    private static Color getRandColor(int fc, int bc) {
        // 取其随机颜色
        Random random = new Random();
        if (fc > 255) {
            fc = 255;
        }
        if (bc > 255) {
            bc = 255;
        }
        int r = fc + random.nextInt(bc - fc);
        int g = fc + random.nextInt(bc - fc);
        int b = fc + random.nextInt(bc - fc);
        return new Color(r, g, b);
    }

}

CorsFilter
在此拦截器中要允许客户端 处理和和发送 verificationJwt 这个头

public class CorsFilter implements Filter {

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
            throws IOException, ServletException {
        System.out.println("进入跨域过滤器");
        HttpServletResponse httpResponse = (HttpServletResponse) servletResponse;
        HttpServletRequest httpRequest=(HttpServletRequest) servletRequest;
        // Access-Control-Allow-Origin就是我们需要设置的域名
        // Access-Control-Allow-Headers跨域允许包含的头。
        // Access-Control-Allow-Methods是允许的请求方式
        httpResponse.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名
        httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");

        //允许客户端发送一个新的请求头
        httpResponse.setHeader("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept, jwt, verificationJwt");
        //允许客户端处理一个新的响应头jwt
        httpResponse.setHeader("Access-Control-Expose-Headers", "jwt");
        httpResponse.setHeader("Access-Control-Expose-Headers", "verificationJwt");

        //axios和ajax会发送两次请求,第一次提交方式为:option直接返回即可
        if("OPTIONS".equals(httpRequest.getMethod())) {
            return ;
        }
        filterChain.doFilter(servletRequest, servletResponse);
    }

    @Override
    public void destroy() {

    }
}

后台到这里就over了。下面我们看看前台代码

前台

首先要在state.js中定义一个用于保持验证码jwt的属性

verificationJwt:null, //这是用来保存用户等登录验证码jwt身份识别的

get/set 配置我就不展示了

在 action.js中添加 验证码图片获取的请求路径映射变量

'VERIFICATION':'/user/verificationCode',  //获取验证码

在http.js中要的拦截器就要在响应的时候如果有 verificationJwt 这个响应头 就要保存。并且在下次请求的时候带上这个 verificationJwt放入请求头都 。

// 请求拦截器
axios.interceptors.request.use(function(config) {
  //设置验证码jwt令牌
  let verificationJwt = window.vm.$store.getters.getVerificationJwt;
  if(verificationJwt){
    config.headers['verificationJwt'] = verificationJwt;
  }

  //这是登录后设置的jwt令牌
  let jwt=window.vm.$store.getters.getJwt;
  if(jwt){
    config.headers['jwt']=jwt;
  }
  return config;
}, function(error) {
  return Promise.reject(error);
});

// 响应拦截器
axios.interceptors.response.use(function(response) {
  //保存验证码jwt令牌
  let verificationjwt=response.headers['verificationjwt'];
  if(verificationjwt){
    window.vm.$store.commit('setVerificationJwt',{
      verificationJwt:verificationjwt
    });
  }
// 保存登录成功的jwt令牌
  let jwt=response.headers['jwt'];
  if(jwt){
    window.vm.$store.commit('setJwt',{
      jwt:jwt
    });
  }
  return response;
}, function(error) {
  return Promise.reject(error);
});

下面是登录界面,就直接贴代码啦:

<template>
  <div class="login-wrap">
    <el-form class="login-container">
      <h1 class="title">用户登录</h1>
      <el-form-item label="">
        <el-input type="text" v-model="userName" placeholder="请输入登录账号" autocomplete="off"></el-input>
      </el-form-item>
      <el-form-item label="">
        <el-input type="password" v-model="userPwd" placeholder="请输入登录密码" autocomplete="off"></el-input>
      </el-form-item>
      <el-form-item label="">
        <el-row>
          <el-col :span="16">
            <el-input type="text" v-model="verificationCode" placeholder="请输入验证码" autocomplete="off"></el-input>
          </el-col>
          <el-col :span="8">
            <img id="img" :src="verificationCodeSrc" width="116px" height="40px" @click="changeVerificationCode" >
          </el-col>
        </el-row>
      </el-form-item>
      <el-form-item>
        <el-button type="primary" style="width: 100%;" @click="doSubmit">登  录</el-button>
      </el-form-item>
      <el-row style="text-align: center; margin-top: -15;">
        <el-link type="primary">忘记密码</el-link>
        <el-link type="primary" @click="gotoRegister">用户注册</el-link>
      </el-row>
    </el-form>
  </div>
</template>


<script>
    export default {
        name: 'Login',
        data: function() {
            return {
                userName: null,
                userPwd: null,
                verificationCode:null,
                verificationCodeSrc:null
            }
        },
        methods: {
            gotoRegister:function(){
                this.$router.push('/Register');
            },
            doSubmit: function() {
                let params = {
                    userName: this.userName,
                    userPwd: this.userPwd,
                    verificationCode: this.verificationCode
                };
                let url = this.axios.urls.SYSTEM_USER_DOLOGIN;

                this.axios.post(url, params).then(resp => {
                    if(resp.data.success) {
                        //提示登录成功
                        this.$message({
                            message: "登录成功",
                            type: 'success'
                        });
                        //跳转路由
                        this.$router.push({
                            path:'/Main'
                        })
                        //这是将用户信息保持下来
                        let user=resp.data.data
                        this.$store.dispatch('setUserAsync',{
                            user:user
                        });
                    }else{
                        this.$message({
                            message: "账户或密码错误",
                            type: 'error'
                        });
                    }
                }).catch(resp => {
                    this.$message({
                        message: "请求异常",
                        type: 'error'
                    });
                });
            },
            //更新验证码
            changeVerificationCode(){
                let url = this.axios.urls.VERIFICATION;
                this.axios.post(url, {}).then(resp => {
                    this.verificationCodeSrc = resp.data;
                }).catch(resp => {
                    console.log(resp);
                });

            }
        }
        ,
        created() {
            let url = this.axios.urls.VERIFICATION;
            this.axios.post(url, {}).then(resp => {
                this.verificationCodeSrc = resp.data;
            }).catch(resp => {
                console.log(resp);
            });
        }
    }
</script>

<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped>
  .login-wrap {
    box-sizing: border-box;
    width: 100%;
    height: 100%;
    padding-top: 10%;
    background-image: url();
    /* background-color: #112346; */
    background-repeat: no-repeat;
    background-position: center right;
    background-size: 100%;
  }

  .login-container {
    border-radius: 10px;
    margin: 0px auto;
    width: 350px;
    padding: 30px 35px 15px 35px;
    background: #fff;
    border: 1px solid #eaeaea;
    text-align: left;
    box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
  }

  .title {
    margin: 0px auto 40px auto;
    text-align: center;
    color: #505458;
  }
</style>

展示效果

前台:
在这里插入图片描述
redis中
在这里插入图片描述
over

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值