#include #include #include "lib_acl.h" #include "../../util.h" #include "acl_cpp/lib_acl.hpp" typedef struct { acl::aio_handle* handle; char addr[64]; int connect_timeout; int read_timeout; int nopen_limit; int nopen_total; int nwrite_limit; int nwrite_total; int nread_total; int id_begin; int dlen; } IO_CTX; static bool connect_server(acl::sslbase_conf* ssl_conf, IO_CTX* ctx, int id); /** * 客户端异步连接流回调函数类 */ class client_io_callback : public acl::aio_open_callback { public: /** * 构造函数 * @param client {aio_socket_stream*} 异步连接流 * @param ssl_conf {acl::sslbase_conf*} 非空时指定 SSL 连接方式 * @param ctx {IO_CTX*} * @param id {int} 本流的ID号 */ client_io_callback(acl::aio_socket_stream* client, acl::sslbase_conf* ssl_conf, IO_CTX* ctx, int id) : client_(client) , ssl_conf_(ssl_conf) , ctx_(ctx) , nwrite_(0) , nread_(0) , id_(id) { dlen_ = ctx->dlen; buff_ = (char*) malloc(dlen_); memset(buff_, 'x', dlen_); buff_[dlen_ - 1] = '\n'; buff_[dlen_ - 2] = '\r'; } ~client_io_callback() { std::cout << ">>>ID: " << id_ << ", io_callback deleted now!" << std::endl; free(buff_); } /** * 基类虚函数, 当异步流读到所要求的数据时调用此回调函数 * @param data {char*} 读到的数据地址 * @param len {int} 读到的数据长度 * @return {bool} 返回给调用者 true 表示继续,否则表示需要关闭异步流 */ bool read_callback(char*, int len) { nread_ += len; ctx_->nread_total++; if (nwrite_ < 100 || nwrite_ % 1000 == 0) { char buf[256]; acl::safe_snprintf(buf, sizeof(buf), "current len: %d, total_len: %d, nwrite: %d", len, nread_, nwrite_); acl::meter_time(__FILE__, __LINE__, buf); } write_line(); return true; } /** * 基类虚函数, 当异步流写成功时调用此回调函数 * @return {bool} 返回给调用者 true 表示继续,否则表示需要关闭异步流 */ bool write_callback() { ctx_->nwrite_total++; nwrite_++; return true; } /** * 基类虚函数, 当该异步流关闭时调用此回调函数 */ void close_callback() { if (!client_->is_opened()) { std::cout << "Id: " << id_ << " connect " << ctx_->addr << " error: " << acl::last_serror(); // 如果是第一次连接就失败,则退出 if (ctx_->nopen_total == 0) { std::cout << ", first connect error, quit"; /* 获得异步引擎句柄,并设置为退出状态 */ client_->get_handle().stop(); } std::cout << std::endl; delete this; return; } /* 获得异步引擎中受监控的异步流个数 */ int nleft = client_->get_handle().length(); if (ctx_->nopen_total == ctx_->nopen_limit && nleft == 1) { std::cout << "Id: " << id_ << " stop now! nstream: " << nleft << std::endl; /* 获得异步引擎句柄,并设置为退出状态 */ client_->get_handle().stop(); } // 必须在此处删除该动态分配的回调类对象以防止内存泄露 delete this; } /** * 基类虚函数,当异步流超时时调用此函数 * @return {bool} 返回给调用者 true 表示继续,否则表示需要关闭异步流 */ bool timeout_callback() { std::cout << "Connect " << ctx_->addr << " Timeout ..." << std::endl; client_->close(); return false; } bool read_wakeup() { // 取得之前通过 setup_hook 注册的 SSL IO句柄 acl::sslbase_io* hook = (acl::sslbase_io*) client_->get_hook(); if (hook == NULL) { std::cout << "get hook error"<< std::endl; return false; } // 尝试进行 SSL 握手 if (!hook->handshake()) { logger_error("ssl handshake failed"); return false; } // SSL 握手还未完成,等待本函数再次被触发 if (!hook->handshake_ok()) { return true; } // 如果 SSL 握手已经成功,则开始读数据 printf("ssl handshake ok\r\n"); // 由 reactor 模式转为 proactor 模式,从而取消 // read_wakeup 回调过程 client_->disable_read(); // 开始与服务端的读写过程 return begin_run(); } /** * 基类虚函数, 当异步连接成功后调用此函数 * @return {bool} 返回给调用者 true 表示继续,否则表示需要关闭异步流 */ bool open_callback() { // 连接成功,设置IO读写回调函数 client_->add_read_callback(this); client_->add_write_callback(this); ctx_->nopen_total++; acl::assert_(id_ > 0); if (ctx_->nopen_total < ctx_->nopen_limit) { // 开始进行下一个连接过程 if (!connect_server(ssl_conf_, ctx_, id_ + 1)) { std::cout << "connect error!" << std::endl; } } // 设置 SSL 方式 if (ssl_conf_) { return setup_ssl(*ssl_conf_); } // 开始与服务端的读写过程 else { return begin_run(); } } private: acl::aio_socket_stream* client_; acl::sslbase_conf* ssl_conf_; IO_CTX* ctx_; int nwrite_; int nread_; int id_; char *buff_; int dlen_; bool setup_ssl(acl::sslbase_conf& ssl_conf) { acl::sslbase_io* ssl = ssl_conf.create(true); // 将 SSL IO 过程注册至异步流中 if (client_->setup_hook(ssl) == ssl) { std::cout << "open ssl error!" << std::endl; ssl->destroy(); return false; } // 开始 SSL 握手过程 if (!ssl->handshake()) { client_->remove_hook(); ssl->destroy(); return false; } // 开始异步 SSL 握手过程,满足可读条件时将触发 read_wakeup client_->read_wait(10); return true; } bool begin_run(void) { write_line(); // 异步从服务器读取数据 //client_->read(); client_->gets(10, false); return true; } void write_line(void) { // 异步向服务器发送数据 client_->write(buff_, dlen_); if (nwrite_ >= ctx_->nwrite_limit) { client_->close(); } } }; static bool connect_server(acl::sslbase_conf* ssl_conf, IO_CTX* ctx, int id) { // 开始异步连接远程服务器 acl::aio_socket_stream* stream = acl::aio_socket_stream::open( ctx->handle, ctx->addr, ctx->connect_timeout); if (stream == NULL) { std::cout << "connect " << ctx->addr << " error!" << std::endl; std::cout << "stoping ..." << std::endl; if (id == 0) { ctx->handle->stop(); } return false; } acl_non_blocking(stream->sock_handle(), ACL_BLOCKING); // 创建连接后的回调函数类 client_io_callback* callback = new client_io_callback(stream, ssl_conf, ctx, id); // 添加连接成功的回调函数类 stream->add_open_callback(callback); // 添加连接失败后回调函数类 stream->add_close_callback(callback); // 添加连接超时的回调函数类 stream->add_timeout_callback(callback); return true; } static void usage(const char* procname) { printf("usage: %s -h[help] -l server_addr \r\n" " -d path_to_polarssl\r\n" " -c nconnect\r\n" " -n io_max\r\n" " -k[use kernel event: epoll/kqueue/devpoll\r\n" " -I check_fds_inter\r\n" " -M delay_ms\r\n" " -t connect_timeout\r\n" " -S[use_ssl]\r\n" " -L data_len[default: 8193]\n", procname); } int main(int argc, char* argv[]) { bool use_kernel = false, use_ssl = false; IO_CTX ctx; acl::string libpath("../libpolarssl.so"); acl::sslbase_conf* ssl_conf = NULL; int ch; int check_fds_inter = 10, delay_ms = 100; memset(&ctx, 0, sizeof(ctx)); ctx.connect_timeout = 5; ctx.nopen_limit = 1; ctx.id_begin = 1; ctx.nwrite_limit = 1; ctx.dlen = 8193; acl::safe_snprintf(ctx.addr, sizeof(ctx.addr), "127.0.0.1:9800"); while ((ch = getopt(argc, argv, "hd:c:n:kl:t:SL:I:M:")) > 0) { switch (ch) { case 'c': ctx.nopen_limit = atoi(optarg); if (ctx.nopen_limit <= 0) { ctx.nopen_limit = 10; } break; case 'd': libpath = optarg; break; case 'n': ctx.nwrite_limit = atoi(optarg); if (ctx.nwrite_limit <= 0) { ctx.nwrite_limit = 10; } break; case 'h': usage(argv[0]); return 0; case 'k': use_kernel = true; break; case 'l': acl::safe_snprintf(ctx.addr, sizeof(ctx.addr), "%s", optarg); break; case 't': ctx.connect_timeout = atoi(optarg); break; case 'S': use_ssl = true; break; case 'L': ctx.dlen = atoi(optarg); break; case 'I': check_fds_inter = atoi(optarg); break; case 'M': delay_ms = atoi(optarg); break; default: break; } } acl::meter_time(__FUNCTION__, __LINE__, "-----BEGIN-----"); acl::acl_cpp_init(); acl::log::stdout_open(true); acl::aio_handle handle(use_kernel ? acl::ENGINE_KERNEL : acl::ENGINE_SELECT); handle.set_check_inter(check_fds_inter); int delay_sec = delay_ms / 1000; int delay_usec = (delay_ms - delay_sec * 1000) * 1000; handle.set_delay_sec(delay_sec); handle.set_delay_usec(delay_usec); printf(">>>delay_sec: %d, delay_usec: %d, check_fds_inter: %d\r\n", delay_sec, delay_usec, check_fds_inter); printf("Enter any key to continue ...\r\n"); getchar(); if (use_ssl) { if (libpath.find("mbedtls") != NULL) { const std::vector& libs = libpath.split2(";"); if (libs.size() != 3) { printf("invalid libpath=%s\r\n", libpath.c_str()); return 1; } acl::mbedtls_conf::set_libpath(libs[0], libs[1], libs[2]); if (acl::mbedtls_conf::load()) { ssl_conf = new acl::mbedtls_conf(false); } else { printf("load %s error\r\n", libpath.c_str()); } } else if (libpath.find("polarssl") != NULL) { acl::polarssl_conf::set_libpath(libpath); if (acl::polarssl_conf::load()) { ssl_conf = new acl::polarssl_conf; } else { printf("load %s error\r\n", libpath.c_str()); } } } ctx.handle = &handle; if (!connect_server(ssl_conf, &ctx, ctx.id_begin)) { std::cout << "enter any key to exit." << std::endl; getchar(); return 1; } std::cout << "Connect " << ctx.addr << " ..." << std::endl; struct timeval begin; gettimeofday(&begin, NULL); while (true) { // 如果返回 false 则表示不再继续,需要退出 if (!handle.check()) { break; } } struct timeval end; gettimeofday(&end, NULL); double spent = util::stamp_sub(&end, &begin); printf("total open: %d, total write: %d, total read: %d," " spent: %.2f ms, speed: %.2f\r\n", ctx.nopen_total, ctx.nwrite_total, ctx.nread_total, spent, (ctx.nread_total * 1000) / (spent > 1 ? spent : 1)); delete ssl_conf; return 0; }