ich vorbei bin in folgenden (der Digest/Hash SHA1):OpenSSL kann nicht ECDSA Signatur C++ überprüfen, überprüft C# richtig
hash = HexToBytes("9E712647173B435CF691537A76C2F1423E4A18ED");
signature = Base64ToBytes("ASLQ3wguSDkJCfFWE3kvBfp7BDNjdajl2ezIetR6DsiacFVASvEAw9v6S3IM0LnaqAV2BTe7eBcRmef/qb2/Hw==");
pubKey16 = "04C2D0A868C35F475208B6C33A58D4AC275190F1A9D5804456FF07C42605716EF748FB4FD246163E851DBE9A942569741F54341A7C85F394B20777AB7FE526096A";//Actual key lacks 04 at front but I'm guessing OpenSSL needs this?
Um diese Funktion:
int Misc::verify_signature(unsigned char* hash, std::vector<unsigned char> signature, char* cPubKey16) {
printf("Signature length: %d\n", signature.size());
int function_status = -1;
EC_KEY *eckey = NULL;
EC_POINT *pub_key;
const EC_GROUP *ecgroup;
SSL_library_init();
SSL_load_error_strings();
std::string pubKeyS(cPubKey16);
std::vector<unsigned char> pubKeyVC = Misc::hexToBytes(pubKeyS);
const unsigned char* pubKeyVCp = pubKeyVC.data();
const unsigned char** pubKeyVCpp = &pubKeyVCp;
//NID_secp256k1 is not r1 which is what .NET uses
eckey = EC_KEY_new_by_curve_name(NID_X9_62_prime256v1);
//Load our public key
eckey = o2i_ECPublicKey(&eckey, pubKeyVCpp, pubKeyVC.size());
if (!EC_KEY_check_key(eckey)) {
printf("EC_KEY_check_key failed:\n");
printf("%s\n", ERR_error_string(ERR_get_error(), NULL));
}
else {
printf("Public key verified OK\n");
}
//Create the properly formatted signature
ECDSA_SIG* ec_sig = ECDSA_SIG_new();
//Split signature into R and S value
//Set R
if (NULL == BN_bin2bn(&signature[0], 32, (ec_sig->r))) {
printf("Failed to set R value in EC Signature\n");
function_status = -1;
}
printf("post r :%s\n", BN_bn2hex(ec_sig->r));
//Set S
if (NULL == BN_bin2bn(&signature[0] + 32, 32, (ec_sig->s))) {
printf("Failed to set S value in EC Signature\n");
function_status = -1;
}
printf("post s :%s\n", BN_bn2hex(ec_sig->s));
//Encode the signature
int sig_size = i2d_ECDSA_SIG(ec_sig, NULL);
unsigned char *sig_bytes =(unsigned char *) malloc(sig_size);
unsigned char *p;
printf("Orig Sig Size: %d\n", sig_size);
p = sig_bytes;
int new_sig_size = i2d_ECDSA_SIG(ec_sig, &p);
printf("New Sig Size: %d\n", new_sig_size);
int verify_status = ECDSA_do_verify(hash, 20, ec_sig, eckey);
printf("Verify status: %d\n", verify_status);
const int verify_success = 1;
if (verify_success != verify_status)
{
if(verify_status==-1)handleErrors();
printf("Failed to verify EC Signature\n");
function_status = -1;
}
else
{
printf("Verifed EC Signature\n");
function_status = 1;
}
//EC_GROUP_free(ecgroup);//Might fail as Ecgroup is constant TODO
EC_KEY_free(eckey);
return function_status;
}
Aber ich kann nicht überprüfen die Signatur in OpenSSL (verify_success ist 0), obwohl die exakt gleichen Daten in C# erfolgreich verifiziert werden.
Irgendwelche Ideen, warum oder was ich falsch mache?
Der öffentliche Schlüssel in C# ist:
4543533120000000C2D0A868C35F475208B6C33A58D4AC275190F1A9D5804456FF07C42605716EF748FB4FD246163E851DBE9A942569741F54341A7C85F394B20777AB7FE526096A
Ich gehe davon aus, dass 4543533120000000 .NET spezielle Sachen i so nur 04 davon an den Rest vorgehängten.
Hier ist der C# -Code verwendet, um die Signatur zu verifizieren und es tut so erfolgreich (SHA1 von Datenbytes ist über beide Programme identisch) ..
HashAlgorithm hashMan2 = new SHA1Managed();
byte[] dataBytes = hashMan2.ComputeHash(Encoding.ASCII.GetBytes("H4sIAAAAAAAEADPQMQBCQzBJDsSm0xCMDTFUYYpQAjFNAIsAAOvFhT3RAAAA"));
String sig = "ASLQ3wguSDkJCfFWE3kvBfp7BDNjdajl2ezIetR6DsiacFVASvEAw9v6S3IM0LnaqAV2BTe7eBcRmef/qb2/Hw==";
byte[] readPublicKey2 = Convert.FromBase64String("RUNTMSAAAADC0Khow19HUgi2wzpY1KwnUZDxqdWARFb/B8QmBXFu90j7T9JGFj6FHb6alCVpdB9UNBp8hfOUsgd3q3/lJglq");
Console.WriteLine("Public key file is read as:");
Console.WriteLine(Convert.ToBase64String(readPublicKey2));
using (ECDsaCng ecsdKey = new ECDsaCng(CngKey.Import(readPublicKey2, CngKeyBlobFormat.EccPublicBlob)))
{
if (ecsdKey.VerifyData(dataBytes, Convert.FromBase64String(sig)))
{
Console.WriteLine("Data and Signature have been verified.");
}
else
{
Console.WriteLine("Data and Signature could not be verified!");
}
}
Jede geschätzt Hilfe.
Überprüfen Sie die Versionshinweise. –
Ich habe, konnte nichts zu beachten als auch eine neue Methode Handler, die keine Konvertierung zu und von ECDSA_SIG, https://www.openssl.org/news/changelog.html – Akumaburn
Sie könnten versuchen, die Bytes in die Schlüssel- und Signatur-Integer * Werte * der Koordinaten, X, Y und R, S (eventuell auch mischen, 3 Möglichkeiten). C# verwendet häufig eine Codierung mit wenig Endian. –