C# RSA 非对称加密实践

一、前提

背景: 项目的登录页面最初设计为使用前端 MD5 加密用户密码。后端服务通过匹配数据库中存储的 MD5 加密字符串来验证密码的正确性。

加固前 登录接口表单.png

安全漏洞: 安全审计报告指出,攻击者能够在局域网内嗅探网络流量,并捕获请求数据包。由于 MD5 加密算法的安全性不足,攻击者可以轻松解密并还原出明文密码。

密码 md5 加密后被还原.png

加固建议: 报告建议采用更安全的传输加密措施,例如使用 HTTPS 或升级至更安全的加密算法。建议的算法包括不可逆的 Hash 算法结合盐值、安全对称加密算法或非对称加密算法。

实施方案: 为了保证用户体验,决定保留现有的前端 MD5 加密逻辑。在此基础上,前端将对 MD5 加密后的密码字符串进行一次 RSA 加密,然后传输至后端。后端在验证密码前,将先对 RSA 加密的字符串进行解密。

加固后 登录接口表单.png

二、概念

非对称加密算法需要两个密钥:公开密钥(publickey)和私有密钥(privatekey)。公开密钥与私有密钥是一对,如果用公开密钥对数据进行加密,只有用对应的私有密钥才能解密;如果用私有密钥对数据进行加密,那么只有用对应的公开密钥才能解密。

通过分析,当前需求场景为发送者用接收者的公钥加密,接受者用自己的私钥解密。具体原理图如下:


非对称加密原理图.png

三、实践

使用到的工具类和第三方库

具体操作步骤

  1. 后端生成密钥:
using (RSACryptoServiceProvider rsa = new RSACryptoServiceProvider(2048))//2048 是指定密钥加密位数
{
    // 公钥 
    string pubkey = rsa.ToXmlString(false);
    // 私钥 
    string prikey = rsa.ToXmlString(true);
}

注意:\color{red}{生成的密钥是 xml 字符串!}
关于安全保留私钥,微软官方的建议是使用安全密钥容器。为了方便存储和读取,我偷懒把私钥和公钥都保存在了配置文件里。

  1. 前端接收公钥
    前端新增一个获取公钥的接口,加载页面时自动获取 RAS 公钥。
    注意:由于前端\color{red}{jsencrypt 库支持的密钥是 Pkcs8 格式},接口输出的公钥需先在后端将格式 xml 转换为 Pkcs8 格式。进行数据格式转换后端需安装 NuGet 包 XC.RSAUtil 。
    NuGet 程序包上的 XC.RSAUtil.png
//公钥XML格式转Pkcs8格式
RsaKeyConvert.PublicKeyXmlToPem(xmlPublicKeyString);
  1. 前端通过公钥加密
    前端页面引入 jsencrypt.js 库,大家可以根据需要自行选择安装方式,我这里用的是比较简单的文件引入。
<script src="~/Content/js/jsencrypt.min.js"></script>

在登录接口中,使用在步骤 2 获取到的公钥,对登录参数进行一次 RSA 加密后,再传输到后端。

// publickey 公钥
// username 账号
// password 密码 md5 加密后字符串
const encrypt = new JSEncrypt();
encrypt.setPublicKey(publickey);
var encryptedUsername = encrypt.encrypt(username);
var encryptedPassword = encrypt.encrypt(password);
  1. 后端通过私钥解密
    在后端登录验证方法中,读取在步骤 1 时保存于配置文件的私钥,先对登录参数进行一次 RSA 解密成明文,再验证登录信息。
using (var rsa = new RSACryptoServiceProvider(2048))// 2048 是指定密钥加密位数
{
    try
    {
        rsa.FromXmlString(privateKey);// 导入私钥
        var encrypted = Convert.FromBase64String(encryptedText);// encryptedText 加密的密文
        var bytes = rsa.Decrypt(encrypted, false);
        return Encoding.UTF8.GetString(bytes);
    }
    finally
    {
        rsa.PersistKeyInCsp = false;
    }
}

辅助类

为方便使用,以上后端代码统一整理成了一个辅助类 RSAHelper.cs ,使用时注意修改命名空间

using System;
using System.Collections.Generic;
using System.Linq;
using System.Security.Cryptography;
using System.Text;
using XC.RSAUtil;

namespace xxxxxxx.Util
{
    /// <summary>
    /// 描 述:RSA 非对称加密和解密辅助类
    /// </summary>
    public class RSAHelper
    {
        /// <summary>
        ///  RSA 解密文本
        /// </summary>
        /// <param name="encryptedText">加密的密文</param>
        /// <param name="privateKey">私钥</param>
        /// <returns>未加密数据的字符串</returns>
        public static string RSADecrypt(string encryptedText, string privateKey)
        {
            using (var rsa = new RSACryptoServiceProvider(2048))//2048是加密位数
            {
                try
                {
                    rsa.FromXmlString(privateKey);
                    var encrypted = Convert.FromBase64String(encryptedText);
                    var bytes = rsa.Decrypt(encrypted, false);
                    return Encoding.UTF8.GetString(bytes);
                }
                finally
                {
                    rsa.PersistKeyInCsp = false;
                }
            }
        }

        /// <summary>
        /// RSA私钥生成 (XML字符串)
        /// </summary>
        public static ValueTuple<string, string> GenerateKey()
        {
            using (RSACryptoServiceProvider rsa = new RSACryptoServiceProvider(2048))//2048是加密位数
            {
                // 公钥 
                string pubkey = rsa.ToXmlString(false);
                // 私钥 
                string prikey = rsa.ToXmlString(true);
                return new ValueTuple<string, string>(pubkey, prikey);
            }
        }

        /// <summary>
        /// 公钥XML格式转Pkcs8格式
        /// </summary>
        /// <param name="xmlPublicKeyString">xml格式公钥字符串</param>
        /// <returns>Pkcs8格式公钥字符串</returns>
        public static string PublicKeyXmlToPem(string xmlPublicKeyString)
        {
            return RsaKeyConvert.PublicKeyXmlToPem(xmlPublicKeyString);
        }

    }
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容