在64位macOS上,Delphi FMX dcpcrypt错误结果

时间:2019-07-31 20:45:52

标签: delphi firemonkey

我正在使用Delphi 10.3.2 我正在尝试将DCPCrypt单元与firemonkey框架一起使用来加密字符串。 它可以在Win32,Win64和macOS 32目标上100%运行,结果始终相同。但是当我针对macOS 64进行编译时,结果会有所不同。

这是使用的代码:

function EncodeAES(code:ansistring; key:ansistring):string;
var
    s,u:ansistring;
  enc: TEncoding;
  k,iv, Data, Crypt: TBytes;
  Cipher: TDCP_rijndael;
begin
    u:='';
    enc:=TEncoding.ANSI;
    Data := enc.GetBytes(code); // VMpuXJGbUNOv
    k:=enc.GetBytes(key);   // kj3214ed)k32nre2
    iv:=enc.GetBytes(u);
    Cipher:=TDCP_rijndael.Create(nil);
    Cipher.Init(K[0], 128, @iv[0]);
    Crypt:=Copy(Data, 0, Length(Data));
    BytePadding(Crypt, Cipher.BlockSize, pmPKCS7);
    Cipher.EncryptECB(Crypt[0], Crypt[0]);
    Cipher.Free;
    s:=tencoding.ANSI.GetString(crypt);
    result:=StringToHex(s);
end;

这是BytePadding函数:

procedure BytePadding(var Data: TBytes; BlockSize: integer; PaddingMode: TPaddingMode);
var
    I, DataBlocks, DataLength, PaddingStart, PaddingCount: integer;
begin
    BlockSize := BlockSize div 8;
    if PaddingMode in [pmZeroPadding, pmRandomPadding] then
        if Length(Data) mod BlockSize = 0 then Exit;
    DataBlocks := (Length(Data) div BlockSize) + 1;
    DataLength := DataBlocks * BlockSize;
    PaddingCount := DataLength - Length(Data);
    if PaddingMode in [pmANSIX923, pmISO10126, pmPKCS7] then
        if PaddingCount > $FF then Exit;
    PaddingStart := Length(Data);
    SetLength(Data, DataLength);
    case PaddingMode of
        pmZeroPadding, pmANSIX923, pmISO7816: // fill with $00 bytes
            FillChar(Data[PaddingStart], PaddingCount, 0);
        pmPKCS7: // fill with PaddingCount bytes
            FillChar(Data[PaddingStart], PaddingCount, PaddingCount);
        pmRandomPadding, pmISO10126: // fill with random bytes
            for I := PaddingStart to DataLength-1 do Data[I] := Random($FF);
    end;
    case PaddingMode of
        pmANSIX923, pmISO10126:
            Data[DataLength-1] := PaddingCount; // set end-marker with number of bytes added
        pmISO7816:
            Data[PaddingStart] := $80; // set fixed end-markder $80
    end;
end;

我将函数调用为:

procedure TForm1.BtnClick(Sender: TObject);
var
    s:string;
begin
    s:=EncodeAES('VMpuXJGbUNOv','kj3214ed)k32nre2');
end;

使用Win32 / Win64 / macOS 32的结果(正确): E32A9DE47CC60BDB70CA27885128D17A

使用macOS 64的结果(错误): CF622155545E485AC3A083E8A0478493

我在做什么错了?

1 个答案:

答案 0 :(得分:5)

加密对二进制数据而非文本数据进行操作。处理文本时,必须在加密之前将字符编码为字节,然后在解密后将字节解码为字符。这意味着在加密之前和解密之后使用相同的字符编码。您正在尝试进行这种编码,但没有考虑TEncoding.ANSI在OS平台之间,甚至在使用同一OS平台的不同机器之间均不可移植的事实。为了获得更好的可移植性,您需要使用一致的编码,例如TEncoding.UTF8

此外,TEncoding仅与UnicodeString一起使用,因此将TEncoding.GetBytes()TEncoding.GetString()AnsiString一起使用,将使用 RTL的 ANSI定义,而不是您的,如果您的字符串中包含任何非ASCII字符,则会生成您可能不希望的字节。

最好将EncodeAES()用于所有字符串处理,而忘记(Unicode)String甚至存在。尽管Linux之类的平台主要是UTF-8系统,但是Delphi的默认AnsiString在所有平台上都使用UTF-16。如果要编码ANSI字符串,请在调用代码要使用(Unicode)stringRawByteString等时使用UTF8String避免任何隐式转换。按原样加密8位字符,而无需使用{{ 1}}。 AnsiString(N)仅用于TEncoding数据。

最后,您的TEncoding函数不应将加密的字节解码为UnicodeString,而只是将其转换为十六进制字符串。您应该按原样对加密的字节进行十六进制编码。

尝试以下方法:

EncodeAES()