asio 使用 openssl 示例
#include <boost/asio.hpp>
#include <boost/asio/ssl.hpp>
#include <boost/algorithm/string.hpp>
#include <boost/lexical_cast.hpp>
#define OPENSSL_NO_DEPRECATED
#include <openssl/ssl.h>
#include <wincrypt.h>
#pragma comment(lib, "Crypt32.lib")
#include <iostream>
#include <fstream>
#include <string_view>
void fail(const char* what, const boost::system::error_code& ec) {
std::cout << what << ": " << ec.message() << std::endl;
}
namespace asio = boost::asio;
using tcp = asio::ip::tcp;
int add_ca(asio::ssl::context& ssl_ctx, boost::system::error_code& ec)
{
BIO* bio = BIO_new_fp(stdout, BIO_NOCLOSE);
if (!bio) {
return -1;
}
auto cert_store = CertOpenSystemStore(NULL, L"ROOT");
if (!cert_store) {
return -1;
}
const unsigned char* encoded_cert = nullptr;
int i = 0;
for (PCCERT_CONTEXT cert = nullptr; cert = CertEnumCertificatesInStore(cert_store, cert);) {
X509* x = d2i_X509(NULL, (const unsigned char**)&cert->pbCertEncoded, cert->cbCertEncoded);
if (x) {
BIO* bio_mem = BIO_new(BIO_s_mem());
PEM_write_bio_X509(bio_mem, x);
const char* data = nullptr;
auto len = BIO_get_mem_data(bio_mem, &data);
ssl_ctx.add_certificate_authority(asio::buffer(data, len), ec);
BIO_free(bio_mem);
if (ec) {
return -1;
}
}
++i;
}
BIO_printf(bio, "cert sum: %d\n", i);
CertCloseStore(cert_store, CERT_CLOSE_STORE_FORCE_FLAG);
return 0;
}
int main()
{
boost::system::error_code ec;
asio::io_context io_ctx;
tcp::resolver resolver(io_ctx);
auto results = resolver.resolve("www.baidu.com", "https", ec);
if (ec) {
fail("resolver.resolve", ec);
return -1;
}
asio::ssl::context ssl_ctx(asio::ssl::context::method::sslv23_client);
add_ca(ssl_ctx, ec);
if (ec) {
fail("add_ca", ec);
return -1;
}
asio::ssl::stream<tcp::socket> s(io_ctx, ssl_ctx);
s.set_verify_mode(asio::ssl::verify_peer | asio::ssl::verify_client_once);
asio::connect(s.lowest_layer(), results, ec);
if (ec) {
fail("asio::connect", ec);
return -1;
}
s.handshake(s.client, ec);
if (ec) {
fail("s.handshake", ec);
return -1;
}
const char* req = "GET / HTTP/1.1\r\nHost: www.baidu.com\r\nAccept: text/html\r\n\r\n";
asio::write(s, asio::buffer(req, std::strlen(req)), ec);
if (ec) {
fail("asio::write", ec);
return -1;
}
// Header
asio::streambuf streambuf;
std::size_t content_length = 0;
constexpr std::string_view delimiter = "\r\n";
constexpr std::string_view header_name = "Content-Length";
while (true) {
std::size_t len = asio::read_until(s, streambuf, delimiter, ec);
if (ec) {
fail("asio::read_until", ec);
return -1;
}
std::string_view header(asio::buffer_cast<const char*>(streambuf.data()), len - delimiter.size());
std::cout << header << std::endl;
if (!header.empty()) {
if (boost::algorithm::istarts_with(header, header_name)) {
bool has_optional_space = header[header_name.size() + 1] == ' ';
std::string_view header_value(header.cbegin() + header_name.size() + 1 + has_optional_space, header.cend());
content_length = boost::lexical_cast<std::size_t>(header_value);
}
}
streambuf.consume(len);
if (header.empty()) {
break;
}
}
// Body
std::cout << "content_length: " << content_length << std::endl;
std::size_t len = asio::read(s, streambuf.prepare(content_length - streambuf.size()), ec);
if (ec) {
fail("asio::read", ec);
return -1;
}
streambuf.commit(len);
std::string_view body(asio::buffer_cast<const char*>(streambuf.data()), streambuf.size());
std::system("chcp 65001");
std::cout << body;
streambuf.consume(streambuf.size());
s.shutdown(ec);
if (ec) {
fail("s.shutdown", ec);
return -1;
}
s.lowest_layer().shutdown(tcp::socket::shutdown_both, ec);
if (ec) {
fail("s.lowest_layer().shutdown", ec);
return -1;
}
return 0;
}
asio 使用 openssl 示例的更多相关文章
- asio 广播代码示例
代码网络收集 修改了一个编译的小问题 客户端 // Client.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include < ...
- 一个简单的 openssl 示例
////生成一个私钥////$key=openssl_pkey_new();openssl_pkey_export($key,$out);//等于下面写入的内容////将私钥写入一个文件////ope ...
- Muduo 网络编程示例之零:前言
陈硕 (giantchen_AT_gmail)Blog.csdn.net/Solstice Muduo 全系列文章列表: http://blog.csdn.net/Solstice/category/ ...
- asio kcp源码分析
asio kcp代码走读 (1)kcp_client_wrap类 a 提供方法接口如下: send_msg kcp_client_.send_msg(msg); stop //等待工作线程退出 set ...
- wazuh官方安装指南(中文译版本)
安装Wazuh服务器 Wazuh服务器可以安装在任何类型的Unix操作系统上.最常见安装在Linux上.如果可以为您的系统提供自动化脚本,则安装过程会更容易,但是,从源码构建和安装也非常简单. 通 ...
- CentOS6.7上安装nginx1.8.0
主题: CentOS6.7上安装nginx1.8.0 环境准备: 1.gcc-c++ 示例:yum install gcc-c++ 安装:gcc-c++ gcc-c++编译工具 2.PCRE(Perl ...
- OpenSSL密码算法库: MD5示例小程序
OpenSSL http://www.openssl.org/ OpenSSL整个软件包大概可以分成三个主要的功能部分:密码算法库.SSL协议库以及应用程序.OpenSSL 的密码算法库包含多种加密算 ...
- openssl AES加密算法API的使用示例
openssl为用户提供了丰富的指令,同时也提供了供编程调用的API,本文以使用128位aes算法的ecb模式进行加密和解密验证,如下所示 第一种方法,直接使用aes算法提供的api进行调用,代码如下 ...
- OpenSSl 加密解密 示例(终于有编程实践了)
OPenSSl的加密主要有三个重要的函数.看懂下面的代码就基本上知道该如何使用openssL来加密了. 不过注意,要先将libssl.so.1.0和libcrypto.so.1.0文件复制到执行的文件 ...
- VC环境下编译OpenSSL(仅仅是个示例,网上还有许多相关文章)
VC环境OpenSSL安装以及编程过程 SSL就是Secure Sockets Layer,是一种安全套接字协议,详情请参考链接中的介绍. 配置过程中需要生成一些mak文件,这些生成代码用perl脚本 ...
随机推荐
- CISC与RISC
- JavaScript 生产者消费者模型
因为node使用单线程的方式实现,所以,在此使用定时器timer取代线程thread来实现生产者消费者模型. 1 var sigintCount = 0; 2 var productArray = [ ...
- IDEA 激活码全家桶 webStorm亲测可用【更新日期2021.11.30】
3MRUAPM31O-eyJsaWNlbnNlSWQiOiIzTVJVQVBNMzFPIiwibGljZW5zZWVOYW1lIjoi5rC45LmF5r+A5rS7IHd3d8K3YWppaHVvw ...
- docker 设计及源码分析
1.dockerd 是一个长期运行的守护进程(docker daemon).负责管理 docker 容器的生命周期.镜像和存储等.实际还是通过grpc 的协议调用 containerd 的 api 接 ...
- 操作系统大作业:在Linux环境下模拟实现简单命令解释器(代码部分)
好家伙 1. 题目要求 一. 课程设计(大作业)目的 熟悉Linux编程环境,加强对Linux命令的理解及函数的运用,完成一个操作系统的部分系统的设计过程.编码.调试,锻炼实际应用能力. 二. ...
- Python——第二章:字典的循环、嵌套、"解构"(解包)
字典进阶操作 -- 循环和嵌套 字典的循环 我们先看直接打印字典的样子,会分别对每对key:value进行打印,并使用,分隔他们 dic = { "赵四": "特别能歪嘴 ...
- 1.elasticsearch运行
在docker中运行elasticsearch.kibana 一.MacOs 首先需要安装doceker,提供两种方式,选一种方便的就好 1.命令行安装方式 安装命令行 xcode-select -- ...
- BFS(二)转动转盘锁
对应 LeetCode 752.转动转盘锁 ### 问题定义 你有一个带有四个圆形拨轮的转盘锁.每个拨轮都有10个数字: '0', '1', '2', '3', '4', '5', '6', '7', ...
- Ubuntu 终端如何分割多个窗口
sudo apt-get install terminator 查看 ~/.config(隐藏文件夹 ctrl + h 即可看见) 下是否有 terminator 文件夹 如果没有手动创建一个 然后在 ...
- Java 在PPT中添加文本、图片超链接
本文介绍通过Java程序在PPT幻灯片中添加超链接的方法,可以给文本或者图片设置超链接,设置超链接时,可设置包括网页链接.邮件地址链接.幻灯片跳转链接等不同指向对象的链接.文中方法使用了免费版PPT类 ...