10#include <gnutls/gnutls.h>
11#include <gnutls/abstract.h>
12#include <gnutls/x509.h>
13#include <gnutls/ocsp.h>
33class OPENDHT_PUBLIC CryptoException :
public std::runtime_error
36 explicit CryptoException(
const std::string& str)
37 : std::runtime_error(str) {};
38 explicit CryptoException(
const char* str)
39 : std::runtime_error(str) {};
40 CryptoException(
const CryptoException& e)
noexcept =
default;
41 CryptoException& operator=(
const CryptoException&)
noexcept =
default;
47class OPENDHT_PUBLIC DecryptError :
public CryptoException
50 explicit DecryptError(
const std::string& str)
51 : CryptoException(str) {};
52 explicit DecryptError(
const char* str)
53 : CryptoException(str) {};
54 DecryptError(
const DecryptError& e)
noexcept =
default;
55 DecryptError& operator=(
const DecryptError&)
noexcept =
default;
62using Identity = std::pair<std::shared_ptr<PrivateKey>, std::shared_ptr<Certificate>>;
67struct OPENDHT_PUBLIC PublicKey
80 PublicKey(
const Blob& pk)
81 : PublicKey(pk.data(), pk.size())
84 :
PublicKey((const uint8_t*) pk.data(), pk.size())
86 PublicKey(PublicKey&& o) noexcept
93 explicit operator bool()
const {
return pk; }
94 bool operator==(
const PublicKey& o)
const {
return pk == o.pk || getLongId() == o.getLongId(); }
95 bool operator!=(
const PublicKey& o)
const {
return !(*
this == o); }
97 PublicKey& operator=(PublicKey&& o)
noexcept;
109 bool checkSignature(
const uint8_t* data,
size_t data_len,
const uint8_t* signature,
size_t signature_len)
const;
110 inline bool checkSignature(
const Blob& data,
const Blob& signature)
const
112 return checkSignature(data.data(), data.size(), signature.data(), signature.size());
115 Blob encrypt(
const uint8_t* data,
size_t data_len)
const;
116 inline Blob encrypt(
const Blob& data)
const {
return encrypt(data.data(), data.size()); }
117 inline Blob encrypt(std::string_view data)
const {
return encrypt((
const uint8_t*) data.data(), data.size()); }
118 void pack(Blob& b)
const;
119 int pack(uint8_t* out,
size_t* out_len)
const;
120 void unpack(
const uint8_t* dat,
size_t dat_size);
122 std::string toString()
const;
124 template<
typename Packer>
125 void msgpack_pack(Packer& p)
const
129 p.pack_bin(b.size());
130 p.pack_bin_body((
const char*) b.data(), b.size());
133 void msgpack_unpack(
const msgpack::object& o);
135 gnutls_digest_algorithm_t getPreferredDigest()
const;
137 gnutls_pubkey_t pk {
nullptr};
140 mutable InfoHash cachedId_ {};
141 mutable PkId cachedLongId_ {};
142 mutable std::atomic_bool idCached_ {
false};
143 mutable std::atomic_bool longIdCached_ {
false};
145 PublicKey(
const PublicKey&) =
delete;
146 PublicKey& operator=(
const PublicKey&) =
delete;
147 void encryptBloc(
const uint8_t* src,
size_t src_size, uint8_t* dst,
size_t dst_size)
const;
153struct OPENDHT_PUBLIC PrivateKey
163 PrivateKey(PrivateKey&& o)
noexcept;
164 PrivateKey& operator=(PrivateKey&& o)
noexcept;
166 PrivateKey(
const uint8_t* src,
size_t src_size,
const char* password =
nullptr);
167 PrivateKey(
const Blob& src,
const std::string& password = {})
168 :
PrivateKey(src.data(), src.size(), password.c_str())
170 PrivateKey(std::string_view src,
const std::string& password = {})
171 : PrivateKey((const uint8_t*) src.data(), src.size(), password.c_str())
175 explicit operator bool()
const {
return key; }
177 const PublicKey& getPublicKey()
const;
178 const std::shared_ptr<PublicKey>& getSharedPublicKey()
const;
180 int serialize(uint8_t* out,
size_t* out_len,
const std::string& password = {})
const;
181 Blob serialize(
const std::string& password = {})
const;
187 Blob sign(
const uint8_t* data,
size_t data_len)
const;
188 inline Blob sign(std::string_view dat)
const {
return sign((
const uint8_t*) dat.data(), dat.size()); }
189 inline Blob sign(
const Blob& dat)
const {
return sign(dat.data(), dat.size()); }
206 static PrivateKey
generate(
unsigned key_length = 4096, gnutls_pk_algorithm_t algo = GNUTLS_PK_RSA);
207 static PrivateKey generateEC();
209 gnutls_privkey_t key {};
210 gnutls_x509_privkey_t x509_key {};
213 PrivateKey(
const PrivateKey&) =
delete;
214 PrivateKey& operator=(
const PrivateKey&) =
delete;
215 Blob decryptBloc(
const uint8_t* src,
size_t src_size)
const;
217 mutable std::mutex publicKeyMutex_ {};
218 mutable std::shared_ptr<PublicKey> publicKey_ {};
221class OPENDHT_PUBLIC RevocationList
223 using clock = std::chrono::system_clock;
224 using time_point = clock::time_point;
225 using duration = clock::duration;
229 RevocationList(
const Blob& b);
230 RevocationList(RevocationList&& o) noexcept
237 RevocationList& operator=(RevocationList&& o)
244 void pack(
Blob& b)
const;
245 void unpack(
const uint8_t* dat,
size_t dat_size);
246 Blob getPacked()
const
253 template<
typename Packer>
254 void msgpack_pack(Packer& p)
const
256 Blob b = getPacked();
257 p.pack_bin(b.size());
258 p.pack_bin_body((
const char*) b.data(), b.size());
261 void msgpack_unpack(
const msgpack::object& o);
263 void revoke(
const Certificate& crt, time_point t = time_point::min());
272 void sign(
const Identity&
id) { sign(*
id.first, *
id.second); }
274 bool isSignedBy(
const Certificate& issuer)
const;
276 std::string toString()
const;
289 time_point getUpdateTime()
const;
290 time_point getNextUpdateTime()
const;
292 gnutls_x509_crl_t get() {
return crl; }
293 gnutls_x509_crl_t getCopy()
const
298 gnutls_x509_crl_t ret = copy.crl;
304 gnutls_x509_crl_t crl {};
305 RevocationList(
const RevocationList&) =
delete;
306 RevocationList& operator=(
const RevocationList&) =
delete;
309enum class NameType { UNKNOWN = 0, RFC822, DNS, URI, IP };
311class OPENDHT_PUBLIC CertificateRequest
314 CertificateRequest();
315 CertificateRequest(
const uint8_t* data,
size_t size);
316 CertificateRequest(std::string_view src)
317 : CertificateRequest((
const uint8_t*) src.data(), src.size())
319 CertificateRequest(
const Blob& data)
320 : CertificateRequest(data.data(), data.size())
323 CertificateRequest(CertificateRequest&& o) noexcept
324 : request(std::move(o.request))
328 CertificateRequest& operator=(CertificateRequest&& o)
noexcept;
330 ~CertificateRequest();
332 void setName(
const std::string& name);
333 void setUID(
const std::string& name);
334 void setAltName(NameType type,
const std::string& name);
336 std::string getName()
const;
337 std::string getUID()
const;
339 void sign(
const PrivateKey& key,
const std::string& password = {});
344 std::string toString()
const;
346 gnutls_x509_crq_t get()
const {
return request; }
349 CertificateRequest(
const CertificateRequest& o) =
delete;
350 CertificateRequest& operator=(
const CertificateRequest& o) =
delete;
351 gnutls_x509_crq_t request {
nullptr};
354class OPENDHT_PUBLIC OcspRequest
357 OcspRequest(gnutls_ocsp_req_t r)
360 OcspRequest(
const uint8_t* dat_ptr,
size_t dat_size);
361 OcspRequest(std::string_view dat)
362 : OcspRequest((
const uint8_t*) dat.data(), dat.size())
369 std::string toString(
const bool compact =
true)
const;
372 Blob getNonce()
const;
375 gnutls_ocsp_req_t request;
378class OPENDHT_PUBLIC OcspResponse
381 OcspResponse(
const uint8_t* dat_ptr,
size_t dat_size);
382 OcspResponse(std::string_view response)
383 : OcspResponse((
const uint8_t*) response.data(), response.size())
391 std::string toString(
const bool compact =
true)
const;
398 gnutls_ocsp_cert_status_t getCertificateStatus()
const;
405 gnutls_ocsp_cert_status_t verifyDirect(
const Certificate& crt,
const Blob& nonce);
408 gnutls_ocsp_resp_t response;
411struct OPENDHT_PUBLIC Certificate
413 Certificate()
noexcept {}
424 , issuer(std::move(o.issuer))
425 , publicKey_(std::move(o.publicKey_))
435 Certificate(
const uint8_t* dat,
size_t dat_size)
438 unpack(dat, dat_size);
441 :
Certificate((const uint8_t*) pem.data(), pem.size())
448 template<
typename Iterator>
458 template<
typename Iterator>
459 Certificate(
const std::vector<std::pair<Iterator, Iterator>>& certs)
467 void pack(
Blob& b)
const;
468 void unpack(
const uint8_t* dat,
size_t dat_size);
469 Blob getPacked()
const
484 template<
typename Iterator>
485 void unpack(
const Iterator& begin,
const Iterator& end)
487 std::shared_ptr<Certificate> tmp_subject {};
488 std::shared_ptr<Certificate> first {};
489 for (Iterator icrt = begin; icrt < end; ++icrt) {
490 auto tmp_crt = std::make_shared<Certificate>(*icrt);
492 tmp_subject->issuer = tmp_crt;
493 tmp_subject = std::move(tmp_crt);
497 *
this = first ? std::move(*first) : Certificate();
511 template<
typename Iterator>
512 void unpack(
const std::vector<std::pair<Iterator, Iterator>>& certs)
514 std::shared_ptr<Certificate> tmp_issuer;
516 for (
auto li = certs.rbegin(); li != certs.rend(); ++li) {
518 gnutls_x509_crt_init(&tmp_crt.cert);
519 const gnutls_datum_t crt_dt {(uint8_t*) &(*li->first), (unsigned) (li->second - li->first)};
520 int err = gnutls_x509_crt_import(tmp_crt.cert, &crt_dt, GNUTLS_X509_FMT_PEM);
521 if (err != GNUTLS_E_SUCCESS)
522 err = gnutls_x509_crt_import(tmp_crt.cert, &crt_dt, GNUTLS_X509_FMT_DER);
523 if (err != GNUTLS_E_SUCCESS)
524 throw CryptoException(std::string(
"Could not read certificate - ") + gnutls_strerror(err));
525 tmp_crt.issuer = tmp_issuer;
526 tmp_issuer = std::make_shared<Certificate>(std::move(tmp_crt));
528 *
this = tmp_issuer ? std::move(*tmp_issuer) : Certificate();
531 template<
typename Packer>
532 void msgpack_pack(Packer& p)
const
536 p.pack_bin(b.size());
537 p.pack_bin_body((
const char*) b.data(), b.size());
540 void msgpack_unpack(
const msgpack::object& o);
542 explicit operator bool()
const {
return cert; }
543 const PublicKey& getPublicKey()
const;
544 const std::shared_ptr<PublicKey>& getSharedPublicKey()
const;
551 Blob getSerialNumber()
const;
572 std::vector<std::pair<NameType, std::string>>
getAltNames()
const;
574 std::chrono::system_clock::time_point getActivation()
const;
575 std::chrono::system_clock::time_point getExpiration()
const;
589 std::string print()
const;
608 static Certificate generate(
const PrivateKey& key,
609 const std::string& name =
"dhtnode",
610 const Identity& ca = {},
612 int64_t validity = 0);
615 gnutls_x509_crt_t getCopy()
const
620 gnutls_x509_crt_t ret = copy.cert;
625 std::vector<gnutls_x509_crt_t> getChain(
bool copy =
false)
const
629 std::vector<gnutls_x509_crt_t> crts;
630 for (
auto c =
this; c; c = c->issuer.get())
631 crts.emplace_back(copy ? c->getCopy() : c->cert);
635 std::pair<std::vector<gnutls_x509_crt_t>, std::vector<gnutls_x509_crl_t>> getChainWithRevocations(
636 bool copy =
false)
const
640 std::vector<gnutls_x509_crt_t> crts;
641 std::vector<gnutls_x509_crl_t> crls;
642 for (
auto c =
this; c; c = c->issuer.get()) {
643 crts.emplace_back(copy ? c->getCopy() : c->cert);
644 crls.reserve(crls.size() + c->revocation_lists.size());
645 for (
const auto& crl : c->revocation_lists)
646 crls.emplace_back(copy ? crl->getCopy() : crl->get());
651 gnutls_digest_algorithm_t getPreferredDigest()
const;
658 std::pair<std::string, Blob> generateOcspRequest(gnutls_x509_crt_t& issuer);
666 gnutls_x509_crt_t cert {
nullptr};
667 std::shared_ptr<Certificate> issuer {};
668 std::shared_ptr<OcspResponse> ocspResponse;
671 Certificate(
const Certificate&) =
delete;
672 Certificate& operator=(
const Certificate&) =
delete;
673 mutable InfoHash cachedId_ {};
674 mutable PkId cachedLongId_ {};
675 mutable std::atomic_bool idCached_ {
false};
676 mutable std::atomic_bool longIdCached_ {
false};
680 bool operator()(
const std::shared_ptr<RevocationList>& lhs,
const std::shared_ptr<RevocationList>& rhs)
const
682 return lhs->getNumber() < rhs->getNumber();
686 std::set<std::shared_ptr<RevocationList>, crlNumberCmp> revocation_lists;
688 mutable std::mutex publicKeyMutex_ {};
689 mutable std::shared_ptr<PublicKey> publicKey_ {};
692struct OPENDHT_PUBLIC TrustList
698 bool hasError()
const {
return ret < 0; }
699 bool isValid()
const {
return !hasError() and !(result & GNUTLS_CERT_INVALID); }
700 explicit operator bool()
const {
return isValid(); }
701 OPENDHT_PUBLIC std::string toString()
const;
702 OPENDHT_PUBLIC
friend std::ostream& operator<<(std::ostream& s,
const VerifyResult& h);
706 TrustList(TrustList&& o) noexcept
707 : trust(std::move(o.trust))
715 void remove(
const Certificate& crt,
bool parents =
true);
721 gnutls_x509_trust_list_t trust {
nullptr};
731OPENDHT_PUBLIC Identity
generateIdentity(
const std::string& name,
const Identity& ca,
unsigned key_length,
bool is_ca);
732OPENDHT_PUBLIC Identity
generateIdentity(
const std::string& name =
"dhtnode",
733 const Identity& ca = {},
734 unsigned key_length = 4096);
736OPENDHT_PUBLIC Identity generateEcIdentity(
const std::string& name,
const Identity& ca,
bool is_ca);
737OPENDHT_PUBLIC Identity generateEcIdentity(
const std::string& name =
"dhtnode",
const Identity& ca = {});
739OPENDHT_PUBLIC
void saveIdentity(
const Identity&
id,
const std::string& path,
const std::string& privkey_password = {});
740OPENDHT_PUBLIC Identity loadIdentity(
const std::string& path,
const std::string& privkey_password = {});
750OPENDHT_PUBLIC
Blob hash(
const Blob& data,
size_t hash_length = 512 / 8);
752OPENDHT_PUBLIC
void hash(
const uint8_t* data,
size_t data_length, uint8_t*
hash,
size_t hash_length);
767OPENDHT_PUBLIC
inline Blob
770 return aesEncrypt(data.data(), data.size(), key);
788OPENDHT_PUBLIC
inline Blob
791 return aesDecrypt(data.data(), data.size(), key);
793OPENDHT_PUBLIC
inline Blob
796 return aesDecrypt((uint8_t*) data.data(), data.size(), key);
799OPENDHT_PUBLIC
Blob aesDecrypt(
const uint8_t* data,
size_t data_length, std::string_view password);
800OPENDHT_PUBLIC
inline Blob
803 return aesDecrypt(data.data(), data.size(), password);
805OPENDHT_PUBLIC
inline Blob
806aesDecrypt(std::string_view data, std::string_view password)
808 return aesDecrypt((uint8_t*) data.data(), data.size(), password);
814OPENDHT_PUBLIC
Blob aesGetKey(
const uint8_t* data,
size_t data_length, std::string_view password);
817 return aesGetKey(data.data(), data.size(), password);
826OPENDHT_PUBLIC std::string_view
aesGetEncrypted(
const uint8_t* data,
size_t data_length);
844 return aesBuildEncrypted((
const uint8_t*) encryptedData.data(), encryptedData.size(), salt);
858 const uint8_t* payload,
859 size_t payload_length,
861 const Blob& salt = {});
std::string getIssuerUID() const
void sign(const PrivateKey &, const Certificate &, duration validity_period={})
std::string getIssuerName() const
OPENDHT_PUBLIC Blob aesEncrypt(const uint8_t *data, size_t data_length, const Blob &key)
OPENDHT_PUBLIC Blob hash(const Blob &data, size_t hash_length=512/8)
OPENDHT_PUBLIC Blob aesBuildEncrypted(const uint8_t *encryptedData, size_t data_length, const Blob &salt)
OPENDHT_PUBLIC Identity generateIdentity(const std::string &name, const Identity &ca, unsigned key_length, bool is_ca)
OPENDHT_PUBLIC Blob stretchKey(std::string_view password, Blob &salt, size_t key_length=512/8)
OPENDHT_PUBLIC std::string_view aesGetEncrypted(const uint8_t *data, size_t data_length)
OPENDHT_PUBLIC Blob webPushEncrypt(const Blob &p256dh, const Blob &auth, const uint8_t *payload, size_t payload_length, const PrivateKey &privKey={}, const Blob &salt={})
OPENDHT_PUBLIC Blob aesGetKey(const uint8_t *data, size_t data_length, std::string_view password)
OPENDHT_PUBLIC Blob aesGetSalt(const uint8_t *data, size_t data_length)
OPENDHT_PUBLIC Blob aesDecrypt(const uint8_t *data, size_t data_length, const Blob &key)
std::vector< uint8_t > Blob
std::string getIssuerUID() const
Certificate(const std::vector< std::pair< Iterator, Iterator > > &certs)
void unpack(const Iterator &begin, const Iterator &end)
void setValidity(const Identity &ca, int64_t validity)
Certificate(const Blob &crt)
void revoke(const PrivateKey &, const Certificate &)
const PkId & getLongId() const
const InfoHash & getId() const
Certificate(const Iterator &begin, const Iterator &end)
std::string getIssuerDN() const
std::vector< std::shared_ptr< RevocationList > > getRevocationLists() const
std::string getIssuerName() const
std::string getUID() const
void unpack(const std::vector< std::pair< Iterator, Iterator > > &certs)
std::string toString(bool chain=true) const
std::string getDN() const
void addRevocationList(RevocationList &&)
std::string getName() const
std::vector< std::pair< NameType, std::string > > getAltNames() const
Certificate(gnutls_x509_crt_t crt) noexcept
Blob sign(const uint8_t *data, size_t data_len) const
Blob decrypt(const uint8_t *cypher, size_t cypher_len) const
PrivateKey(gnutls_x509_privkey_t k)
static PrivateKey generate(unsigned key_length=4096, gnutls_pk_algorithm_t algo=GNUTLS_PK_RSA)
const InfoHash & getId() const
PublicKey(const uint8_t *dat, size_t dat_size)
PublicKey(gnutls_pubkey_t k)
const PkId & getLongId() const