C# алгоритм шифрования AES до Python3 - PullRequest
3 голосов
/ 13 января 2020

Здравствуйте, мне нужно передать алгоритм шифрования C # в python, но я не могу получить тот же результат в конечном ха sh, кто-нибудь знает, скажите мне, что я делаю неправильно?

Это код шифра C# AES:

using System;
using System.Security.Cryptography;
using System.Text;
using System.IO;

public class Program
{
    public static void Main()
    {
        string data = "leandro";
        string encrypt = Encrypt(data);
        Console.WriteLine(encrypt);

    }

    static readonly char[] padding = { '=' };
    private const string EncryptionKey = "teste123";
    public static string Encrypt(string clearText)
    {

        byte[] clearBytes = Encoding.Unicode.GetBytes(clearText);
        Console.WriteLine(clearBytes);
        using (Aes encryptor = Aes.Create())
        {
            Rfc2898DeriveBytes pdb = new Rfc2898DeriveBytes(EncryptionKey, new byte[] { 0x49, 0x76, 0x61, 0x6e, 0x20, 0x4d, 0x65, 0x64, 0x76, 0x65, 0x64, 0x65, 0x76 });
            encryptor.Key = pdb.GetBytes(32);
            encryptor.IV = pdb.GetBytes(16);
            using (MemoryStream ms = new MemoryStream())
            {
                using (CryptoStream cs = new CryptoStream(ms, encryptor.CreateEncryptor(), CryptoStreamMode.Write))
                {
                    cs.Write(clearBytes, 0, clearBytes.Length);
                    cs.Close();
                }
                clearText = Convert.ToBase64String(ms.ToArray()).TrimEnd(padding).Replace('+', '-').Replace('/', '_');
            }
        }
        return clearText;

    }

}

Выход : DTyK3ABF4337NRNHPoTliQ

А это моя python версия:

import base64

from Crypto.Cipher import AES
from Crypto.Protocol.KDF import PBKDF2



class AESCipher(object):

    def __init__(self, key, interactions=1000):
        self.bs = AES.block_size
        self.key = key
        self.interactions = interactions

    def encrypt(self, raw):
        raw = self._pad(raw)
        nbytes = [0x49, 0x76, 0x61, 0x6e, 0x20, 0x4d, 0x65, 0x64, 0x76,
                  0x65, 0x64, 0x65, 0x76]
        salt = bytes(nbytes)
        keyiv = PBKDF2(self.key, salt, 48, self.interactions)
        key = keyiv[:32]
        iv = keyiv[32:48]
        cipher = AES.new(key, AES.MODE_CBC, iv)
        enc = base64.b64encode(iv + cipher.encrypt(raw.encode('utf-16le')))
        return self._base64_url_safe(str(enc, "utf-8"))


    def _pad(self, s):
        return s + (self.bs - len(s) % self.bs) * \
            chr(self.bs - len(s) % self.bs)

    def _base64_url_safe(self, s):
        return s.replace('+', '-').replace('/', '_').replace('=', '')

    @staticmethod
    def _unpad(s):
        return s[:-ord(s[len(s) - 1:])]


enc = AESCipher("teste123")
dec = enc.encrypt("leandro")
print(dec)

Выход : LJTFEn0vmz8IvqFZJ87k8lI8DPh8-oIOSIxmS5NE4D0

1 Ответ

3 голосов
/ 14 января 2020

Вы используете Encoding.Unicode.GetBytes(clearText) , который возвращает UTF-16LE , в то время как Python (более разумно) по умолчанию UTF-8 для raw.encode(). Я бы использовал Encoding.UTF8 для вашего C# кода.

Как уже упоминалось в комментариях, Python также добавляет IV перед шифротекстом, тогда как * Код 1026 * просто выполняет шифрование и вычисляет IV во время дешифрования (поэтому его не нужно хранить).

Вот программа Python, которая делает то же самое:

import base64

from Crypto.Cipher import AES
from Crypto.Protocol.KDF import PBKDF2
from Crypto.Util.Padding import pad


class AESCipher(object):

    def __init__(self, key, interactions=1000):
        self.bs = AES.block_size
        self.key = key
        self.interactions = interactions

    def encrypt(self, raw):
        nbytes = [0x49, 0x76, 0x61, 0x6e, 0x20, 0x4d, 0x65, 0x64, 0x76,
                  0x65, 0x64, 0x65, 0x76]

        salt = bytes(nbytes)
        keyiv = PBKDF2(self.key, salt, 48, self.interactions)
        key = keyiv[:32]
        iv = keyiv[32:48]

        cipher = AES.new(key, AES.MODE_CBC, iv)

        encoded = raw.encode('utf-16le')
        encodedpad = pad(encoded, self.bs)

        ct = cipher.encrypt(encodedpad)

        cip = base64.b64encode(ct)
        return self._base64_url_safe(str(cip, "utf-8"))

    def _base64_url_safe(self, s):
        return s.replace('+', '-').replace('/', '_').replace('=', '')

enc = AESCipher("teste123")
dec = enc.encrypt("leandro")
print(dec)

Прежде чем кричать Эврика, пожалуйста, поймите, что зашифрованный текст, который создает код C#, не защищен целостностью и не аутентифицирован. Более того, он уязвим к атакам с отступом oracle, если они присутствуют на приемнике. Атаки с отступом oracle чрезвычайно эффективны, и вы потеряете полную конфиденциальность сообщения, если они будут применены.

Кроме того, если соль не случайна, то ключ и IV также не случайны. Это, в свою очередь, означает, что зашифрованный текст такой же случайный, как и обычный текст. Другими словами, он пропускает данные, если встречаются одни и те же блоки открытого текста. Поэтому я надеюсь, что неслучайная соль только для целей тестирования.

В конце концов, правильное шифрование не означает, что ваше решение безопасно.

...