1
0
mirror of https://github.com/PurpleI2P/i2pd.git synced 2025-01-18 20:59:57 +00:00
i2pd/libi2pd/Family.cpp

192 lines
5.1 KiB
C++
Raw Normal View History

/*
* Copyright (c) 2013-2020, The PurpleI2P Project
*
* This file is part of Purple i2pd project and licensed under BSD3
*
* See full license text in LICENSE file at top of project tree
*/
#include <string.h>
2016-05-11 16:02:26 -04:00
#include <openssl/evp.h>
#include <openssl/ssl.h>
2016-05-11 15:12:38 -04:00
#include "Crypto.h"
#include "FS.h"
2016-02-18 15:57:43 -05:00
#include "Log.h"
#include "Family.h"
#include "Config.h"
2016-02-18 15:57:43 -05:00
namespace i2p
{
namespace data
{
Families::Families ()
{
}
Families::~Families ()
{
}
void Families::LoadCertificate (const std::string& filename)
{
2017-01-23 16:22:48 -05:00
SSL_CTX * ctx = SSL_CTX_new (TLS_method ());
2018-01-06 11:48:51 +08:00
int ret = SSL_CTX_use_certificate_file (ctx, filename.c_str (), SSL_FILETYPE_PEM);
2016-02-18 15:57:43 -05:00
if (ret)
2018-01-06 11:48:51 +08:00
{
2016-02-18 15:57:43 -05:00
SSL * ssl = SSL_new (ctx);
X509 * cert = SSL_get_certificate (ssl);
if (cert)
2018-01-06 11:48:51 +08:00
{
std::shared_ptr<i2p::crypto::Verifier> verifier;
2016-02-18 15:57:43 -05:00
// extract issuer name
char name[100];
X509_NAME_oneline (X509_get_issuer_name(cert), name, 100);
2016-02-20 08:33:13 -05:00
char * cn = strstr (name, "CN=");
if (cn)
2018-01-06 11:48:51 +08:00
{
2016-02-20 08:33:13 -05:00
cn += 3;
char * family = strstr (cn, ".family");
if (family) family[0] = 0;
2018-01-06 11:48:51 +08:00
}
2016-02-18 15:57:43 -05:00
auto pkey = X509_get_pubkey (cert);
2016-11-10 12:51:39 -05:00
int keyType = EVP_PKEY_base_id (pkey);
2016-02-18 15:57:43 -05:00
switch (keyType)
{
case EVP_PKEY_DSA:
// TODO:
break;
case EVP_PKEY_EC:
{
EC_KEY * ecKey = EVP_PKEY_get1_EC_KEY (pkey);
if (ecKey)
{
auto group = EC_KEY_get0_group (ecKey);
if (group)
{
int curve = EC_GROUP_get_curve_name (group);
if (curve == NID_X9_62_prime256v1)
{
uint8_t signingKey[64];
BIGNUM * x = BN_new(), * y = BN_new();
EC_POINT_get_affine_coordinates_GFp (group,
EC_KEY_get0_public_key (ecKey), x, y, NULL);
i2p::crypto::bn2buf (x, signingKey, 32);
i2p::crypto::bn2buf (y, signingKey + 32, 32);
BN_free (x); BN_free (y);
2019-01-01 17:00:37 -05:00
verifier = std::make_shared<i2p::crypto::ECDSAP256Verifier>();
verifier->SetPublicKey (signingKey);
2018-01-06 11:48:51 +08:00
}
else
LogPrint (eLogWarning, "Family: elliptic curve ", curve, " is not supported");
}
EC_KEY_free (ecKey);
}
2016-02-18 15:57:43 -05:00
break;
}
default:
LogPrint (eLogWarning, "Family: Certificate key type ", keyType, " is not supported");
}
2016-02-18 16:28:43 -05:00
EVP_PKEY_free (pkey);
2016-02-20 08:33:13 -05:00
if (verifier && cn)
m_SigningKeys[cn] = verifier;
2018-01-06 11:48:51 +08:00
}
SSL_free (ssl);
}
2016-02-18 15:57:43 -05:00
else
LogPrint (eLogError, "Family: Can't open certificate file ", filename);
2018-01-06 11:48:51 +08:00
SSL_CTX_free (ctx);
2016-02-18 15:57:43 -05:00
}
void Families::LoadCertificates ()
{
std::string certDir = i2p::fs::GetCertsDir() + i2p::fs::dirSep + "family";
std::vector<std::string> files;
2016-02-18 15:57:43 -05:00
int numCertificates = 0;
if (!i2p::fs::ReadDir(certDir, files)) {
2016-05-26 00:00:00 +00:00
LogPrint(eLogWarning, "Family: Can't load family certificates from ", certDir);
return;
}
for (const std::string & file : files) {
if (file.compare(file.size() - 4, 4, ".crt") != 0) {
LogPrint(eLogWarning, "Family: ignoring file ", file);
continue;
}
LoadCertificate (file);
numCertificates++;
2018-01-06 11:48:51 +08:00
}
LogPrint (eLogInfo, "Family: ", numCertificates, " certificates loaded");
2016-02-18 15:57:43 -05:00
}
2018-01-06 11:48:51 +08:00
bool Families::VerifyFamily (const std::string& family, const IdentHash& ident,
const char * signature, const char * key)
{
2020-05-01 14:30:56 -04:00
uint8_t buf[100], signatureBuf[64];
size_t len = family.length (), signatureLen = strlen (signature);
2020-05-01 14:30:56 -04:00
if (len + 32 > 100)
2016-09-16 16:18:50 -04:00
{
LogPrint (eLogError, "Family: ", family, " is too long");
return false;
2018-01-06 11:48:51 +08:00
}
2016-09-16 16:18:50 -04:00
memcpy (buf, family.c_str (), len);
memcpy (buf + len, (const uint8_t *)ident, 32);
2018-01-06 11:48:51 +08:00
len += 32;
Base64ToByteStream (signature, signatureLen, signatureBuf, 64);
auto it = m_SigningKeys.find (family);
if (it != m_SigningKeys.end ())
return it->second->Verify (buf, len, signatureBuf);
// TODO: process key
return true;
}
2016-02-20 20:20:19 -05:00
std::string CreateFamilySignature (const std::string& family, const IdentHash& ident)
{
auto filename = i2p::fs::DataDirPath("family", (family + ".key"));
2016-02-20 20:20:19 -05:00
std::string sig;
2017-01-23 16:22:48 -05:00
SSL_CTX * ctx = SSL_CTX_new (TLS_method ());
2018-01-06 11:48:51 +08:00
int ret = SSL_CTX_use_PrivateKey_file (ctx, filename.c_str (), SSL_FILETYPE_PEM);
2016-02-20 20:20:19 -05:00
if (ret)
{
SSL * ssl = SSL_new (ctx);
EVP_PKEY * pkey = SSL_get_privatekey (ssl);
EC_KEY * ecKey = EVP_PKEY_get1_EC_KEY (pkey);
if (ecKey)
{
auto group = EC_KEY_get0_group (ecKey);
if (group)
{
int curve = EC_GROUP_get_curve_name (group);
if (curve == NID_X9_62_prime256v1)
{
uint8_t signingPrivateKey[32], buf[50], signature[64];
i2p::crypto::bn2buf (EC_KEY_get0_private_key (ecKey), signingPrivateKey, 32);
i2p::crypto::ECDSAP256Signer signer (signingPrivateKey);
size_t len = family.length ();
memcpy (buf, family.c_str (), len);
memcpy (buf + len, (const uint8_t *)ident, 32);
len += 32;
signer.Sign (buf, len, signature);
len = Base64EncodingBufferSize (64);
char * b64 = new char[len+1];
len = ByteStreamToBase64 (signature, 64, b64, len);
b64[len] = 0;
sig = b64;
delete[] b64;
}
else
LogPrint (eLogWarning, "Family: elliptic curve ", curve, " is not supported");
2018-01-06 11:48:51 +08:00
}
}
SSL_free (ssl);
}
2016-02-20 20:20:19 -05:00
else
LogPrint (eLogError, "Family: Can't open keys file: ", filename);
2018-01-06 11:48:51 +08:00
SSL_CTX_free (ctx);
2016-02-20 20:20:19 -05:00
return sig;
2018-01-06 11:48:51 +08:00
}
2016-02-18 15:57:43 -05:00
}
}