xs_openssl.h 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. /* copyright (c) 2022 - 2023 grunfink / MIT license */
  2. #ifndef _XS_OPENSSL_H
  3. #define _XS_OPENSSL_H
  4. xs_str *_xs_digest(const xs_val *input, int size, const char *digest, int as_hex);
  5. #define xs_md5_hex(input, size) _xs_digest(input, size, "md5", 1)
  6. #define xs_sha1_hex(input, size) _xs_digest(input, size, "sha1", 1)
  7. #define xs_sha256_hex(input, size) _xs_digest(input, size, "sha256", 1)
  8. #define xs_sha256_base64(input, size) _xs_digest(input, size, "sha256", 0)
  9. xs_dict *xs_evp_genkey(int bits);
  10. xs_str *xs_evp_sign(const char *secret, const char *mem, int size);
  11. int xs_evp_verify(const char *pubkey, const char *mem, int size, const char *b64sig);
  12. #ifdef XS_IMPLEMENTATION
  13. #include "openssl/rsa.h"
  14. #include "openssl/pem.h"
  15. #include "openssl/evp.h"
  16. xs_str *_xs_digest(const xs_val *input, int size, const char *digest, int as_hex)
  17. /* generic function for generating and encoding digests */
  18. {
  19. const EVP_MD *md;
  20. if ((md = EVP_get_digestbyname(digest)) == NULL)
  21. return NULL;
  22. unsigned char output[1024];
  23. unsigned int out_size;
  24. EVP_MD_CTX *mdctx;
  25. mdctx = EVP_MD_CTX_new();
  26. EVP_DigestInit_ex(mdctx, md, NULL);
  27. EVP_DigestUpdate(mdctx, input, size);
  28. EVP_DigestFinal_ex(mdctx, output, &out_size);
  29. EVP_MD_CTX_free(mdctx);
  30. return as_hex ? xs_hex_enc ((char *)output, out_size) :
  31. xs_base64_enc((char *)output, out_size);
  32. }
  33. xs_dict *xs_evp_genkey(int bits)
  34. /* generates an RSA keypair using the EVP interface */
  35. {
  36. xs_dict *keypair = NULL;
  37. EVP_PKEY_CTX *ctx;
  38. EVP_PKEY *pkey = NULL;
  39. if ((ctx = EVP_PKEY_CTX_new_id(EVP_PKEY_RSA, NULL)) == NULL)
  40. goto end;
  41. if (EVP_PKEY_keygen_init(ctx) <= 0 ||
  42. EVP_PKEY_CTX_set_rsa_keygen_bits(ctx, bits) <= 0 ||
  43. EVP_PKEY_keygen(ctx, &pkey) <= 0)
  44. goto end;
  45. BIO *bs = BIO_new(BIO_s_mem());
  46. BIO *bp = BIO_new(BIO_s_mem());
  47. BUF_MEM *sptr;
  48. BUF_MEM *pptr;
  49. PEM_write_bio_PrivateKey(bs, pkey, NULL, NULL, 0, 0, NULL);
  50. BIO_get_mem_ptr(bs, &sptr);
  51. PEM_write_bio_PUBKEY(bp, pkey);
  52. BIO_get_mem_ptr(bp, &pptr);
  53. keypair = xs_dict_new();
  54. keypair = xs_dict_append(keypair, "secret", sptr->data);
  55. keypair = xs_dict_append(keypair, "public", pptr->data);
  56. BIO_free(bs);
  57. BIO_free(bp);
  58. end:
  59. return keypair;
  60. }
  61. xs_str *xs_evp_sign(const char *secret, const char *mem, int size)
  62. /* signs a memory block (secret is in PEM format) */
  63. {
  64. xs_str *signature = NULL;
  65. BIO *b;
  66. unsigned char *sig;
  67. unsigned int sig_len;
  68. EVP_PKEY *pkey;
  69. EVP_MD_CTX *mdctx;
  70. const EVP_MD *md;
  71. /* un-PEM the key */
  72. b = BIO_new_mem_buf(secret, strlen(secret));
  73. pkey = PEM_read_bio_PrivateKey(b, NULL, NULL, NULL);
  74. /* I've learnt all these magical incantations by watching
  75. the Python module code and the OpenSSL manual pages */
  76. /* Well, "learnt" may be an overstatement */
  77. md = EVP_get_digestbyname("sha256");
  78. mdctx = EVP_MD_CTX_new();
  79. sig_len = EVP_PKEY_size(pkey);
  80. sig = xs_realloc(NULL, sig_len);
  81. EVP_SignInit(mdctx, md);
  82. EVP_SignUpdate(mdctx, mem, size);
  83. if (EVP_SignFinal(mdctx, sig, &sig_len, pkey) == 1)
  84. signature = xs_base64_enc((char *)sig, sig_len);
  85. EVP_MD_CTX_free(mdctx);
  86. EVP_PKEY_free(pkey);
  87. BIO_free(b);
  88. xs_free(sig);
  89. return signature;
  90. }
  91. int xs_evp_verify(const char *pubkey, const char *mem, int size, const char *b64sig)
  92. /* verifies a base64 block, returns non-zero on ok */
  93. {
  94. int r = 0;
  95. BIO *b;
  96. EVP_PKEY *pkey;
  97. EVP_MD_CTX *mdctx;
  98. const EVP_MD *md;
  99. /* un-PEM the key */
  100. b = BIO_new_mem_buf(pubkey, strlen(pubkey));
  101. pkey = PEM_read_bio_PUBKEY(b, NULL, NULL, NULL);
  102. md = EVP_get_digestbyname("sha256");
  103. mdctx = EVP_MD_CTX_new();
  104. if (pkey != NULL) {
  105. xs *sig = NULL;
  106. int s_size;
  107. /* de-base64 */
  108. sig = xs_base64_dec(b64sig, &s_size);
  109. if (sig != NULL) {
  110. EVP_VerifyInit(mdctx, md);
  111. EVP_VerifyUpdate(mdctx, mem, size);
  112. r = EVP_VerifyFinal(mdctx, (unsigned char *)sig, s_size, pkey);
  113. }
  114. }
  115. EVP_MD_CTX_free(mdctx);
  116. EVP_PKEY_free(pkey);
  117. BIO_free(b);
  118. return r;
  119. }
  120. #endif /* XS_IMPLEMENTATION */
  121. #endif /* _XS_OPENSSL_H */