目前代码片段仅用于密码的加密

后端spirntboot部分

// RsaUtils.java
package com.carolin_violet.travel_system.utils;

import org.apache.commons.codec.binary.Base64;
import org.junit.Test;

import javax.crypto.Cipher;
import java.io.ByteArrayOutputStream;
import java.nio.charset.StandardCharsets;
import java.security.*;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;
import java.util.HashMap;


/**
 * @ClassName RsaUtils
 * @Description RSA工具类
 * @Author zj
 * @Date 2022/9/2 16:20
 * @Version 1.0
 */
public class RsaUtils {

    @Test
    public void test() {
        String cipher = "fSoBJLZLpgVxY4FW8EMBxekh730u0aRh+gZHf9S2rXKYwinEPP4fds+fNtjcw1t0K3cf+HC0fQUF+bES/1EN9WHuRDwSVpUWJDg8iGiULCh7R/l4qwzpnj0pqhWPvQWtqzyMoS8RX2ugG455nivPq/RUPVWCfaViRfbDwFBWDLk=";
        String privateKey = "MIICdgIBADANBgkqhkiG9w0BAQEFAASCAmAwggJcAgEAAoGBALNe8nXBhfJ93lDcWwCFli1/ORmf1AlvA2bdE/CzFAiLhcRnRDMI4+hh2Kw8cUj5tNxl16TpcxHjwvvmGC4MZsaNxP8+CDYy1i73P2qEBceg7Libb7cSHa0E6Pulg/8BuUgeedrk+NEwNnqsW2B6SH2OUzMegRC05fG5/FnAyQlhAgMBAAECgYA//sIWjUMS7yFgTB4To5WfCeL/Z2ypKPhXvMyedeXR3qWNGyLT6noVsMMzQCVS2fRTJeq4JaouWcp5QPCjjT0lRCL1REEIOZBOJ+8JlrX1USLx5JisUFrUAxugraEo/sMIRWn6vskcSayVch8QkeT/tf8rpoYGIJk+BFgI1LjaIQJBAO7vAF3bFfLcl6tyFk2oOGoyb+9fb4AcN6Trk9C0ngCtT5JIVqyEsIwrub8NPZU7rQksCpwAO/sHrbAk9NMmDjUCQQDALtEvVNIBNjo1Y1+q13BZQbTqOTejy6zMJN3dgUlmV8SD2QzUJOnjpM9ZeMM3JbdInpbf/GPAbCGC2DxBxOP9AkA9qxwqHUjN9ReEXZ7aDkf5SOlRHA24QBAwlckB/gwl2pWGCPut7VVv6D05zV4OrWdczawzVXAlu3ad1vwFn3XdAkBm4ymem1xqxatYFcGL5xcA0gqXMta03D/AcyRMiADRx78vjy4tDS4ib4OKtPpeNdTwDedcNvT4eWtfb4Ci/VPJAkEAky3CktQRsXwmaYWd7rGuOT9Uln1ejMh+YXv+nfdeYzDy153PhMttiuXwqHkr/o3h8UMCzCycxKkgRwrsfDjkYA==";
        String decryptData = null;
        try {
            decryptData = decrypt(cipher, getPrivateKey(privateKey));
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("解密后内容 => " + decryptData + "\n");
    }


    public static void main(String[] args) {
        try {

            // 生成密钥对
            //KeyPair keyPair = getKeyPair();
            //String privateKey = new String(Base64.encodeBase64(keyPair.getPrivate().getEncoded()));
            //String publicKey = new String(Base64.encodeBase64(keyPair.getPublic().getEncoded()));
            //System.out.println("私钥 => " + privateKey + "\n");
            //System.out.println("公钥 =>" + publicKey + "\n");

            HashMap<String, String> keyPairMap = getKeyPairMap();
            String privateKey = keyPairMap.get("privateKey");
            String publicKey =  keyPairMap.get("publicKey");
            System.out.println("私钥 => " + privateKey + "\n");
            System.out.println("公钥 =>" + publicKey + "\n");

            // RSA加密
            //String data = "123456";
            String data = "Zj123456!";
            String encryptData = encrypt(data, getPublicKey(publicKey));
            System.out.println("加密后内容 => " + encryptData + "\n");
            // RSA解密
            String decryptData = decrypt(encryptData, getPrivateKey(privateKey));
            System.out.println("解密后内容 => " + decryptData + "\n");


            // RSA签名
            String sign = sign(data, getPrivateKey(privateKey));
            // RSA验签
            boolean result = verify(data, getPublicKey(publicKey), sign);
            System.out.println("验签结果 => " + result + "\n");

        } catch (Exception e) {
            e.printStackTrace();
            System.err.println("RSA加解密异常");
        }
    }

    /**
     * RSA最大加密明文大小
     */
    private static final int MAX_ENCRYPT_BLOCK = 117;
    /**
     * RSA最大解密密文大小
     */
    private static final int MAX_DECRYPT_BLOCK = 128;

    private static final String ALGORITHM_NAME = "RSA";
    private static final String MD5_RSA = "MD5withRSA";

    /**
     * 获取密钥对
     */
    public static KeyPair getKeyPair() throws Exception {
        KeyPairGenerator generator = KeyPairGenerator.getInstance(ALGORITHM_NAME);
        generator.initialize(1024);
        return generator.generateKeyPair();
    }
    /**
     * 获取base64加密后密钥对
     */
    public static HashMap<String, String> getKeyPairMap() throws Exception {
        KeyPairGenerator generator = KeyPairGenerator.getInstance(ALGORITHM_NAME);
        generator.initialize(1024);
        KeyPair keyPair = generator.generateKeyPair();
        String privateKey = new String(Base64.encodeBase64(keyPair.getPrivate().getEncoded()));
        String publicKey = new String(Base64.encodeBase64(keyPair.getPublic().getEncoded()));
        HashMap<String, String> keyMap = new HashMap<>();
        keyMap.put("privateKey", privateKey);
        keyMap.put("publicKey", publicKey);
        return keyMap;
    }

    /**
     * 获取公钥
     *
     * @param publicKey base64加密的公钥字符串
     */
    public static PublicKey getPublicKey(String publicKey) throws Exception {
        byte[] decodedKey = Base64.decodeBase64(publicKey.getBytes());
        X509EncodedKeySpec keySpec = new X509EncodedKeySpec(decodedKey);
        KeyFactory keyFactory = KeyFactory.getInstance(ALGORITHM_NAME);
        return keyFactory.generatePublic(keySpec);
    }

    /**
     * RSA加密
     *
     * @param data      待加密数据
     * @param publicKey 公钥
     */
    public static String encrypt(String data, PublicKey publicKey) throws Exception {
        Cipher cipher = Cipher.getInstance(ALGORITHM_NAME);
        cipher.init(Cipher.ENCRYPT_MODE, publicKey);
        int inputLen = data.getBytes().length;
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        int offset = 0;
        byte[] cache;
        int i = 0;
        // 对数据分段加密
        while (inputLen - offset > 0) {
            if (inputLen - offset > MAX_ENCRYPT_BLOCK) {
                cache = cipher.doFinal(data.getBytes(), offset, MAX_ENCRYPT_BLOCK);
            } else {
                cache = cipher.doFinal(data.getBytes(), offset, inputLen - offset);
            }
            out.write(cache, 0, cache.length);
            i++;
            offset = i * MAX_ENCRYPT_BLOCK;
        }
        byte[] encryptedData = out.toByteArray();
        out.close();
        // 获取加密内容使用base64进行编码,并以UTF-8为标准转化成字符串
        // 加密后的字符串
        return new String(Base64.encodeBase64(encryptedData));
    }

    /**
     * 获取私钥
     *
     * @param privateKey base64加密的私钥字符串
     */
    public static PrivateKey getPrivateKey(String privateKey) throws Exception {
        byte[] decodedKey = Base64.decodeBase64(privateKey.getBytes());
        PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(decodedKey);
        KeyFactory keyFactory = KeyFactory.getInstance(ALGORITHM_NAME);
        return keyFactory.generatePrivate(keySpec);
    }

    /**
     * RSA解密
     *
     * @param data       待解密数据
     * @param privateKey 私钥
     */
    public static String decrypt(String data, PrivateKey privateKey) throws Exception {
        Cipher cipher = Cipher.getInstance(ALGORITHM_NAME);
        cipher.init(Cipher.DECRYPT_MODE, privateKey);
        byte[] dataBytes = Base64.decodeBase64(data);
        int inputLen = dataBytes.length;
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        int offset = 0;
        byte[] cache;
        int i = 0;
        // 对数据分段解密
        while (inputLen - offset > 0) {
            if (inputLen - offset > MAX_DECRYPT_BLOCK) {
                cache = cipher.doFinal(dataBytes, offset, MAX_DECRYPT_BLOCK);
            } else {
                cache = cipher.doFinal(dataBytes, offset, inputLen - offset);
            }
            out.write(cache, 0, cache.length);
            i++;
            offset = i * MAX_DECRYPT_BLOCK;
        }
        byte[] decryptedData = out.toByteArray();
        out.close();
        // 解密后的内容
        return new String(decryptedData, StandardCharsets.UTF_8);
    }


    /**
     * 签名
     *
     * @param data       待签名数据
     * @param privateKey 私钥
     */
    public static String sign(String data, PrivateKey privateKey) throws Exception {
        byte[] keyBytes = privateKey.getEncoded();
        PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(keyBytes);
        KeyFactory keyFactory = KeyFactory.getInstance(ALGORITHM_NAME);
        PrivateKey key = keyFactory.generatePrivate(keySpec);
        Signature signature = Signature.getInstance(MD5_RSA);
        signature.initSign(key);
        signature.update(data.getBytes());
        return new String(Base64.encodeBase64(signature.sign()));
    }

    /**
     * 验签
     *
     * @param srcData   原始字符串
     * @param publicKey 公钥
     * @param sign      签名
     */
    public static boolean verify(String srcData, PublicKey publicKey, String sign) throws Exception {
        byte[] keyBytes = publicKey.getEncoded();
        X509EncodedKeySpec keySpec = new X509EncodedKeySpec(keyBytes);
        KeyFactory keyFactory = KeyFactory.getInstance(ALGORITHM_NAME);
        PublicKey key = keyFactory.generatePublic(keySpec);
        Signature signature = Signature.getInstance(MD5_RSA);
        signature.initVerify(key);
        signature.update(srcData.getBytes());
        return signature.verify(Base64.decodeBase64(sign.getBytes()));
    }

}


// SecurityController
package com.carolin_violet.travel_system.controller;

import com.carolin_violet.travel_system.utils.R;
import com.carolin_violet.travel_system.utils.RsaUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletRequest;
import java.util.HashMap;
import java.util.concurrent.TimeUnit;

/**
 * @ClassName SecurityController
 * @Description 生成RSA公私钥对
 * @Author zj
 * @Date 2022/9/2 17:09
 * @Version 1.0
 */
@RestController
@RequestMapping("/travel_system/security")
@CrossOrigin
public class SecurityController {

    @Autowired
    private RedisTemplate<String, Object> redisTemplate;

    // 获取RSA公钥
    @GetMapping("publicKey")
    public R getPublicKey(HttpServletRequest httpServletRequest) {
        try {
            String publicKey = (String) redisTemplate.opsForValue().get(httpServletRequest.getRemoteAddr() + "-publicKey:");
            if (publicKey == null) {
                // 生成base64编码后的公私密钥对
                HashMap<String, String> keyPairMap = RsaUtils.getKeyPairMap();
                String privateKey = keyPairMap.get("privateKey");
                publicKey =  keyPairMap.get("publicKey");
                redisTemplate.opsForValue().set(httpServletRequest.getRemoteAddr() + "-publicKey:", privateKey, 1, TimeUnit.MINUTES);
                redisTemplate.opsForValue().set(httpServletRequest.getRemoteAddr() + "-privateKey:", privateKey, 1, TimeUnit.MINUTES);
            }
            return R.ok().data("publicKey", publicKey);
        }catch (Exception e) {
            e.printStackTrace();
            return R.error().message("秘钥生成失败");
        }
    }
}

// TokenLoginFilter.java
package com.carolin_violet.travel_system.filter;

import com.carolin_violet.travel_system.bean.security.LoginUser;
import com.carolin_violet.travel_system.bean.security.SecurityUser;
import com.carolin_violet.travel_system.security.TokenManager;
import com.carolin_violet.travel_system.utils.R;
import com.carolin_violet.travel_system.utils.ResponseUtil;
import com.carolin_violet.travel_system.utils.RsaUtils;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.security.authentication.AuthenticationManager;
import org.springframework.security.authentication.UsernamePasswordAuthenticationToken;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.AuthenticationException;
import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter;
import org.springframework.security.web.util.matcher.AntPathRequestMatcher;

import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.ArrayList;

/**
 * @ClassName JwtLoginFilter
 * @Description 验证登录账户并返回token
 * @Author zj
 * @Date 2022/5/24 10:03
 * @Version 1.0
 */
public class TokenLoginFilter extends UsernamePasswordAuthenticationFilter {

    private AuthenticationManager authenticationManager;
    private TokenManager tokenManager;
    private RedisTemplate redisTemplate;

    public TokenLoginFilter(AuthenticationManager authenticationManager, TokenManager tokenManager, RedisTemplate redisTemplate) {
        this.authenticationManager = authenticationManager;
        this.tokenManager = tokenManager;
        this.redisTemplate = redisTemplate;
        this.setPostOnly(false); // 关闭登录只允许 post
        // 设置登陆路径,并且post请求
        this.setRequiresAuthenticationRequestMatcher(new AntPathRequestMatcher("/travel_system/login","POST"));
    }

    //这个方法是用来去尝试验证用户的
    @Override
    public Authentication attemptAuthentication(HttpServletRequest request, HttpServletResponse response) throws AuthenticationException {

        try {
            LoginUser loginUser = new ObjectMapper().readValue(request.getInputStream(), LoginUser.class);
            String privateKey = (String) redisTemplate.opsForValue().get(request.getRemoteAddr() + "-privateKey:");
            String decryptPassword = RsaUtils.decrypt(loginUser.getPassword(), RsaUtils.getPrivateKey(privateKey));
            return authenticationManager.authenticate(new UsernamePasswordAuthenticationToken(loginUser.getUsername(), decryptPassword, new ArrayList<>()));
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * 登录成功后调用的方法
     * @param req
     * @param res
     * @param chain
     * @param auth
     * @throws IOException
     * @throws ServletException
     */
    @Override
    protected void successfulAuthentication(HttpServletRequest req, HttpServletResponse res, FilterChain chain,
                                            Authentication auth) throws IOException, ServletException {
        logger.info("=================登录成功=============");
        SecurityUser securityUser = (SecurityUser) auth.getPrincipal();
        // 根据用户名生产token
        String token = tokenManager.createToken(securityUser.getUsername());
        // 将权限存入redis缓存
        redisTemplate.opsForValue().set("securityUser:"+ securityUser.getUsername(), securityUser);
        redisTemplate.opsForValue().set("token:"+ securityUser.getUsername(), token);
        ResponseUtil.out(res, R.ok().data("token", token));
    }

    /**
     * 登录失败
     * @param request
     * @param response
     * @param e
     * @throws IOException
     * @throws ServletException
     */
    @Override
    protected void unsuccessfulAuthentication(HttpServletRequest request, HttpServletResponse response,
                                              AuthenticationException e) throws IOException, ServletException {
        ResponseUtil.out(response, R.error().message("账号或密码错误"));
    }
}

前端vue部分

// user.js
import request from '@/utils/request'

// 获取RSA公钥,后端根据ip判断唯一
export function getPublickey() {
  return request({
    url: `/travel_system/security/publicKey`,
    method: 'get'
  })
}

secure.js
import JSEncrypt from 'jsencrypt'
import { getPublickey } from '@/api/user';

export function encryptPassword(content) {
  return new Promise((resolve, reject) => {
    getPublickey().then(res => {
      const publicKey = res.data.publicKey
      const encrypt = new JSEncrypt()      // 创建加密对象实例
      encrypt.setPublicKey(publicKey)       // 设置公钥
      const cipher = encrypt.encrypt(content)      // 加密数据
      resolve(cipher)
    }).catch(err => {
      reject(err)
    })
  })
}


// 需要的文件中引入加密即可
import { encryptPassword } from '@/utils/secure';

 let cipher =  await encryptPassword(data.password)

Q.E.D.


一个二次元web开发咸鱼