首页 > 其他分享 >Base64编码、解码 C语言例子(使用OpenSSL库)----亲测OK

Base64编码、解码 C语言例子(使用OpenSSL库)----亲测OK

时间:2023-11-20 12:11:06浏览次数:53  
标签:BIO base64 OK Base64 OpenSSL len str xxxx data

摘自:https://www.dandelioncloud.cn/article/details/1498198300963708930

 

// Base64Util.h
#ifndef __BASE64_UTIL_H__
#define __BASE64_UTIL_H__



#ifdef __cplusplus             //告诉编译器,这部分代码按C语言的格式进行编译,而不是C++的
extern "C"{
#endif


string UTIL_base64_encode(const unsigned char *data, unsigned int data_len);
int UTIL_base64_decode(string base64_str, unsigned char **output, unsigned int *out_len);
string hex_2_string(unsigned char *data, unsigned int data_len);


#ifdef __cplusplus             //告诉编译器,这部分代码按C语言的格式进行编译,而不是C++的
}
#endif

#endif /* __BASE64_UTIL_H__ */
//Base64Util.cpp
#include <string>
using namespace std;

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <openssl/bio.h>
#include <openssl/evp.h>
//#include "xxxx_platform_common.h"
//#include "liblicense_log.h"
#include "Base64Util.h"

#define DEBUG_BASE64UTIL (1)

#define print_ln(log_level, fmt, ...) do {printf("(%s|%d)" fmt "\r\n", __func__, __LINE__, ##__VA_ARGS__); fflush(stdout);} while(0)
#define xxxx_free(a) do {if(a) {free((void *)a); (a) = NULL;}} while(0)

/*****************************************************************

    Base64编码

    参数 : data, 输入参数, 输入数据
           data_len, 输入参数, 输入数据长度

     return : 长度>0, 成功, 返回 base64 字符串
              长度=0, 失败

*****************************************************************/
string UTIL_base64_encode(const unsigned char *data, unsigned int data_len)
{
    BIO *bio = NULL;
    BIO *b64 = NULL;
    char *buffer = NULL;
    int buf_len = 0;
    string base64_str = "";

    if (NULL == data || 0 >= data_len) {
        xxxx_print_ln(xxxx_ERROR, "wrong parameter.");
        return "";
    }

    // 创建Base64编码的BIO
    b64 = BIO_new(BIO_f_base64());
    BIO_set_flags(b64, BIO_FLAGS_BASE64_NO_NL); // 去掉所有换行符

    bio = BIO_new(BIO_s_mem());
    bio = BIO_push(b64, bio);


    // 写入数据并刷新BIO
    BIO_write(bio, data, (int)data_len);
    (void)BIO_flush(bio);

    // 读取Base64编码后的数据
    buf_len = BIO_get_mem_data(bio, &buffer);
    if (0 >= buf_len) {
        base64_str = "";
        goto __clean;
    }

    buffer[buf_len] = '\0';

    base64_str = (NULL == buffer) ? "" : buffer;

__clean:
    BIO_free_all(bio); // 内部已经 释放buffer

    return base64_str;
}

/*****************************************************************

    Base64解码

    参数 : base64_str, 输入参数, base64 字符串
           output, 输出参数, 输出数据(请调用者自行free)
           out_len, 输出参数, 输出数据长度

     return :  0, 成功
              -1, 失败

*****************************************************************/
int UTIL_base64_decode(string base64_str, unsigned char **output, unsigned int *out_len)
{
    BIO *bio = NULL;
    BIO *b64 = NULL;
    char *buffer = NULL;
    int buf_len = 0;
    int decoded_len = 0;
    int ret = 0;

    if (NULL == output || NULL == out_len) {
        xxxx_print_ln(xxxx_ERROR, "wrong parameter.");
        return -1;
    }

    // 创建Base64解码的BIO
    b64 = BIO_new(BIO_f_base64());
    BIO_set_flags(b64, BIO_FLAGS_BASE64_NO_NL); // 没有换行符

    bio = BIO_new_mem_buf(base64_str.c_str(), base64_str.length());
    bio = BIO_push(b64, bio);


    // 读取Base64解码后的数据
    buf_len = base64_str.length() * 3 / 4;
    buffer = (char *)calloc(1, buf_len + 1);
    if (NULL == buffer) {
        ret = -1;
        xxxx_print_ln(xxxx_ERROR, "malloc Failed(size=%u).", buf_len + 1);
        goto __Failed;
    }

    decoded_len = BIO_read(b64, buffer, base64_str.length());
    if (0 >= decoded_len) {
        ret = -1;
        goto __Failed;
    }

    buffer[decoded_len] = '\0';
    *output = (unsigned char *)buffer;
    *out_len = decoded_len;

    ret = 0;

#if DEBUG_BASE64UTIL
    xxxx_print_ln(xxxx_DEBUG, "decoded_len=%d, buf_len=%d, buffer=%s", decoded_len, buf_len, buffer);
#endif

//__Success:
    BIO_free_all(bio);

    return ret;

__Failed:
    BIO_free_all(bio);
    xxxx_free(buffer);

    return -1;
}

string hex_2_string(unsigned char *data, unsigned int data_len)
{
    string str = "";
    size_t malloc_len = 2 * data_len + 4;
    char *tmp_str = NULL;

    if (NULL == data || 0 >= data_len) {
        xxxx_print_ln(xxxx_ERROR, "wrong parameter.");
        return "";
    }

    tmp_str = (char *)calloc(1, malloc_len);
    if (NULL == tmp_str) {
        xxxx_print_ln(xxxx_ERROR, "malloc Failed(size=%lu).", malloc_len);
        return "";
    }

    for (unsigned int index = 0; index < data_len; index++) {
        snprintf(tmp_str + strlen(tmp_str), malloc_len - strlen(tmp_str) - 1, "%02x", data[index]);
    }


    str = tmp_str;

    xxxx_free(tmp_str);

    return str;
}

#if DEBUG_BASE64UTIL
// 主函数测试Base64编码和解码函数
int main() {
    const char *str = "Hello, World!中华人民";
    string base64_str = "";
    unsigned char *output = NULL;
    int out_len = 0;
    int ret = 0;

    base64_str = UTIL_base64_encode((const unsigned char *)str, strlen(str));
    ret = UTIL_base64_decode(base64_str, &output, &out_len);

    xxxx_print_ln(xxxx_DEBUG, "base64=%s", base64_str.c_str());
    xxxx_print_ln(xxxx_DEBUG, "ret=%d, out_len=%d, str=%s", ret, out_len, output);

    xxxx_free(output);

    return 0;
}
#endif
g++ Base64Util.cpp -lcrypto

 

标签:BIO,base64,OK,Base64,OpenSSL,len,str,xxxx,data
From: https://www.cnblogs.com/LiuYanYGZ/p/17843648.html

相关文章

  • 关于token的生成格式--Bearer头部说明
    1.Bearer头部:好处在于可以让请求方和服务方都快速而准确地识别Token的传递方式,使得身份验证更加规范化和通用化,便于开发和维护。但并没有更安全,且具体使用须前后一致。2.带Bearer头部的生成和解密如下:publicStringcreateTokenByBao(StringuserId){Datedate=newDate......
  • openssl做HMAC实例(C++)----自测OK
    摘自:https://blog.csdn.net/mijichui2153/article/details/1047414601、HMAC简介(1)MAC(MessageAuthenticationCode,消息认证码算法),可以将其认为是含有秘钥的散列(Hash)函数算法;即兼容了MD和SHA算法,并在此基础上加上了秘钥。因此MAC算法也经常被称作HMAC算法。当然HMAC就是“基......
  • openssl做HMAC实例(C++)原文
    摘自:https://blog.csdn.net/mijichui2153/article/details/1047414601、HMAC简介(1)MAC(MessageAuthenticationCode,消息认证码算法),可以将其认为是含有秘钥的散列(Hash)函数算法;即兼容了MD和SHA算法,并在此基础上加上了秘钥。因此MAC算法也经常被称作HMAC算法。当然HMAC就是“基......
  • GreatSQL社区与Amazon、Facebook、Tencent共同被MySQL致谢
    一、来自MySQL官方的感谢在2023-10-25MySQL官方发布的8.2版本ReleaseNotes中,GreatSQL社区核心开发者RichardDang和HaoLu,分别收到了来自MySQL官方的贡献感谢,与Amazon、Facebook(Meta)、Tencent等一并出现在感谢清单中。详见:MySQL8.2ReleaseNotes/Chang......
  • Docker部署cook | 跟着我一起用NAS来学习做菜吧
    一、介绍该项目是疫情期间大佬开发者开发的,项目初衷是方便特殊时期隔离在家而材料有限的小伙伴,因此菜谱材料会尽量限制在特定范围内,但很多都是家常菜,并且你能搜到的每一种菜都配有完整的视频教学。就算拿到今天对于我这种不怎么会做菜的人来说,这个项目真的泰裤辣。二、搭建cook......
  • web03(过滤器,监听器,cookie)
    过滤器(Filter)作用:在访问到正式资源之前进行过滤(请求到达Servlet之前、Servlet处理之后以及响应返回客户端之前进行干预);解决中文乱码问题:publicvoiddoFilter(ServletRequestrequest,ServletResponseresponse,FilterChainchain)throwsIOException,Servle......
  • cmd常用命令(十)nslookup
    nslookup查看默认dnsC:\Users\admin>nslookup默认服务器:UnKnownAddress:192.168.51.51nslookup-debug8.8.8.8C:\Users\admin>nslookup-debug8.8.8.8------------Gotanswer:HEADER:opcode=QUERY,id=1,rcode=NXDOMAINheaderflag......
  • 什么是 Microsoft Outlook 的 Addin
    Outlook的Addin,又被称为“Outlook插件”,是一种可以扩展和增强Outlook功能的软件工具。插件可以深度集成到Outlook用户界面中,并在用户需要时提供自定义的功能和服务。例如,一个插件可能会提供特定的电子邮件管理功能,如排序、标记、归档或者搜索,或者提供与第三方服务的集成,如......
  • nanomq 轻量快速的mqtt broker
    nanomq是emqx团队开源的iot边缘mqttbroker,有点是轻量,性能好,同时官方也提供了与Mosquitto对比的介绍nanomq很多功能上与emqx是类似的,包含了rule,自定义auth,gatway协议,基于关系型数据库的存储一张官方的集成玩法一些特点快速 相比mosquitto有10倍的提升轻量 比较小跨......
  • 【nest入门】[bug记录]在jupyter notebook中使用conda环境(linux)
    NEST在安装引导https://nest-simulator.readthedocs.io/en/stable/installation/user.html#user-install中提供了conda的方式,如下图所示:如上图中步骤第1步,进行了condacreate--namenest36-cconda-forgenest-simulator后,在jupyter中想要切换到所创建的名为ENVNAME的con......