C# Sign Data with RSA using BouncyCastle

前端 未结 2 1009
北荒
北荒 2020-12-23 15:28

Does anyone know of a simple tutorial or sample code of how to sign data in c# using bouncy castle. In Java there are tons of tutorials and samples. I can\'t find a single e

相关标签:
2条回答
  • 2020-12-23 15:40

    Okay I could not find any documentation on how to do this. But I ended up figuring it out. I am pasting the full code here so hopefully it can help someone in the future.

    This class will calculate a RSA signature with a sha1 hash for the provided string and verify it as well.

    using System;
    using System.IO;
    using System.Text;
    using Org.BouncyCastle.Asn1;
    using Org.BouncyCastle.Crypto;
    using Org.BouncyCastle.Crypto.Parameters;
    using Org.BouncyCastle.OpenSsl;
    using Org.BouncyCastle.Security;
    using Org.BouncyCastle.Utilities.Encoders;
    
    namespace API.Crypto
    {
        public class RsaSha1Signing
        {
            private RsaKeyParameters MakeKey(String modulusHexString, String exponentHexString, bool isPrivateKey)
            {
                var modulus = new Org.BouncyCastle.Math.BigInteger(modulusHexString, 16);
                var exponent = new Org.BouncyCastle.Math.BigInteger(exponentHexString, 16);
    
                return new RsaKeyParameters(isPrivateKey, modulus, exponent);
            }
    
            public String Sign(String data, String privateModulusHexString, String privateExponentHexString)
            {
                /* Make the key */
                RsaKeyParameters key = MakeKey(privateModulusHexString, privateExponentHexString, true);
    
                /* Init alg */
                ISigner sig = SignerUtilities.GetSigner("SHA1withRSA");
    
                /* Populate key */
                sig.Init(true, key);
    
                /* Get the bytes to be signed from the string */
                var bytes = Encoding.UTF8.GetBytes(data);
    
                /* Calc the signature */
                sig.BlockUpdate(bytes, 0, bytes.Length);
                byte[] signature = sig.GenerateSignature();
    
                /* Base 64 encode the sig so its 8-bit clean */
                var signedString = Convert.ToBase64String(signature);
    
                return signedString;
            }
    
            public bool Verify(String data, String expectedSignature, String publicModulusHexString, String publicExponentHexString)
            {
                /* Make the key */
                RsaKeyParameters key = MakeKey(publicModulusHexString, publicExponentHexString, false);
    
                /* Init alg */
                ISigner signer = SignerUtilities.GetSigner("SHA1withRSA");
    
                /* Populate key */
                signer.Init(false, key);
    
                /* Get the signature into bytes */
                var expectedSig = Convert.FromBase64String(expectedSignature);
    
                /* Get the bytes to be signed from the string */
                var msgBytes = Encoding.UTF8.GetBytes(data);
    
                /* Calculate the signature and see if it matches */
                signer.BlockUpdate(msgBytes, 0, msgBytes.Length);
                return signer.VerifySignature(expectedSig);
            }
        }
    }
    
    0 讨论(0)
  • 2020-12-23 16:00

    Look at Bouncy Castle web site. There is archive with sources and examples. http://www.bouncycastle.org/csharp/download/bccrypto-net-1.7-src-ext.zip

    As a examples there are a lot of NUnit tests. Below is code of method to encrypt data byte array using RSA algorithm as a sample, but in Bouncy Castle sources and tests you can find more samples.

        public static byte[] Encrypt(byte[] data, AsymmetricKeyParameter key)
        {
            RsaEngine e = new RsaEngine();
            e.Init(true, key);
            int blockSize = e.GetInputBlockSize();
            List<byte> output = new List<byte>();
    
            for (int chunkPosition = 0; chunkPosition < data.Length; chunkPosition += blockSize)
            {
                int chunkSize = Math.Min(blockSize, data.Length - (chunkPosition * blockSize));
                output.AddRange(e.ProcessBlock(data, chunkPosition, chunkSize));
            }
            return output.ToArray();
        }
    
    0 讨论(0)
提交回复
热议问题