sm2 js加密,java服务器端解密

有些敏感数据,为了保证数据的安全传输,在向浏览器发送的时刻就要被加密,例如密码,但是密码一般是不可逆的,服务器不需要解密它,直接存储即可。所以密码一般是使用md5或sha 哈希一下直接存到数据库了。但是有些是需要服务器重新解密的,比如手机号,身份证号等。客户端加密之后,需要服务器解密出来才可以。如果使用对称算法,那么对称密钥也会泄漏,所以可以使用非对称算法,rsa或sm2,而目前大多数系统都要求使用国密的sm2非对称算法。

前端的sm2加密操作,可以参考这个项目:https://gitee.com/wgslucky/SM2

服务器端解密操作:

复制代码
public static String decrypt(String cipherData, String privateKey) throws InvalidCipherTextException {
        byte[] cipherDataByte = Hex.decode(cipherData);
        // 刚才的私钥Hex,先还原私钥
        BigInteger privateKeyD = new BigInteger(privateKey, 16);
        X9ECParameters sm2ECParameters = GMNamedCurves.getByName("sm2p256v1");
        ECDomainParameters domainParameters =
            new ECDomainParameters(sm2ECParameters.getCurve(), sm2ECParameters.getG(), sm2ECParameters.getN());
        ECPrivateKeyParameters privateKeyParameters = new ECPrivateKeyParameters(privateKeyD, domainParameters);

        // 用私钥解密,这里的Model要与客户端的保持一致。
        SM2Engine sm2Engine = new SM2Engine(SM2Engine.Mode.C1C2C3);

        sm2Engine.init(false, privateKeyParameters);

        // processBlock得到Base64格式,记得解码
        byte[] arrayOfBytes = Base64.decodeBase64(sm2Engine.processBlock(cipherDataByte, 0, cipherDataByte.length));

        // 得到明文:SM2 Encryption Test
        String data = new String(arrayOfBytes);
        return data;
    }

需要注意的是,这里面使用公钥,私钥参数都是十六进制的字段串,可以使用下面的方法生成一对十六进制的密钥对:

复制代码
    public static void createKey() throws NoSuchAlgorithmException {
        X9ECParameters sm2ECParameters = GMNamedCurves.getByName("sm2p256v1");
        ECDomainParameters domainParameters =
            new ECDomainParameters(sm2ECParameters.getCurve(), sm2ECParameters.getG(), sm2ECParameters.getN());
        ECKeyPairGenerator keyPairGenerator = new ECKeyPairGenerator();
        keyPairGenerator.init(new ECKeyGenerationParameters(domainParameters, SecureRandom.getInstance("SHA1PRNG")));
        AsymmetricCipherKeyPair asymmetricCipherKeyPair = keyPairGenerator.generateKeyPair();

        // 私钥,16进制格式,自己保存,格式如a2081b5b81fbea0b6b973a3ab6dbbbc65b1164488bf22d8ae2ff0b8260f64853
        BigInteger privatekey = ((ECPrivateKeyParameters)asymmetricCipherKeyPair.getPrivate()).getD();
        String privateKeyHex = privatekey.toString(16);
        System.out.println(privateKeyHex);
        System.out.println("------------------");
        // 公钥,16进制格式,发给前端,格式如04813d4d97ad31bd9d18d785f337f683233099d5abed09cb397152d50ac28cc0ba43711960e811d90453db5f5a9518d660858a8d0c57e359a8bf83427760ebcbba
        ECPoint ecPoint = ((ECPublicKeyParameters)asymmetricCipherKeyPair.getPublic()).getQ();
        String publicKeyHex = Hex.toHexString(ecPoint.getEncoded(false));
        System.out.println(publicKeyHex);
    }

这样就可以直接实现sm2 js加密,java服务器端解密了,不过这只是针对小数据而已,如果数据量比较大,建议采用对称算法 + sm2非对称算法联合的方法,使用对称算法加密数据,而使用sm2加密对称算法的密钥。因为sm2的公钥是公开的,只有私钥可以解密,所以是安全的。

相关推荐
程序员老邢5 分钟前
【产品底稿 04】商助慧 V1.1 里程碑:爬虫入库 + MySQL + Milvus 全链路打通
java·爬虫·mysql·ai·springboot·milvus
2601_950703947 分钟前
Java安全编程与静态分析实战
java
唐叔在学习8 分钟前
Python移动端应用消息提醒开发实践
开发语言·python
好家伙VCC8 分钟前
**发散创新:基于Python与OpenCV的视频流帧级分析实战**在当前人工智能与计算机视觉飞速发展的背景下
java·人工智能·python·计算机视觉
SimonKing9 分钟前
大V说’AI替代不了你’,但现实是——用AI的人正在替代你
java·后端·程序员
暴力求解9 分钟前
C++ ---string类(三)
开发语言·c++
Pocker_Spades_A15 分钟前
Python快速入门专业版(五十七)——POST请求与模拟登录:从表单分析到实战(以测试网站为例)
开发语言·python
Cobyte18 分钟前
4.响应式系统基础:从发布订阅模式的角度理解 Vue3 的数据响应式原理
前端·javascript·vue.js
晓得迷路了21 分钟前
栗子前端技术周刊第 124 期 - ESLint v10.2.0、React Native 0.85、Node.js 25.9.0...
前端·javascript·eslint
一叶龙洲25 分钟前
Java中使用模板引擎(FreeMarker / Velocity) + Word XML导出复杂Word
xml·java·word