acl/lib_fiber/samples-c++1x/client/client.cpp

161 lines
3.7 KiB
C++

#include "stdafx.h"
#include <stdio.h>
#include <stdlib.h>
//////////////////////////////////////////////////////////////////////////////
static size_t do_echo(acl::socket_stream& conn, int count, bool readable) {
acl::string data("hello world\r\n");
acl::string buf;
size_t i;
for (i = 0; i < (size_t) count; i++) {
if (conn.write(data) == -1) {
printf("client write error %s\r\n", acl::last_serror());
break;
}
if (readable) {
struct timeval begin, end;
gettimeofday(&begin, NULL);
int ret = acl_readable(conn.sock_handle());
gettimeofday(&end, NULL);
double cost = acl::stamp_sub(end, begin);
if (ret == 0) {
printf("not readable, cost=%.2f\r\n", cost);
} else if (ret == 1) {
printf("readable, cost=%.2f\r\n", cost);
} else {
printf("readable error\r\n");
}
}
if (!conn.gets(buf, false)) {
printf("client read error %s\r\n", acl::last_serror());
break;
}
if (i == 0) {
printf("gets: %s\r\n", buf.c_str());
}
}
printf(">>>count=%zd<<<\r\n", i);
return i;
}
static size_t start(const char* addr, int count, bool readable,
int conn_timeout, int rw_timeout) {
if (conn_timeout > 0) {
acl::fiber::set_non_blocking(true);
} else {
acl::fiber::set_non_blocking(false);
}
acl::socket_stream conn;
if (!conn.open(addr, conn_timeout, rw_timeout)) {
printf("connect %s error %s\r\n", addr, acl::last_serror());
return 0;
}
printf(">>>connect %s ok, conn_timeout=%d, rw_timeout=%d\r\n",
addr, conn_timeout, rw_timeout);
struct sockaddr_in in;
int len = sizeof(in);
#if defined(_WIN32) || defined(_WIN64)
if (getpeername(conn.sock_handle(), (struct sockaddr*) &in, &len) < 0) {
#else
if (getpeername(conn.sock_handle(), (struct sockaddr*) &in,
(socklen_t*) &len) < 0) {
#endif
printf(">>>getpeername error: %s\r\n", acl::last_serror());
}
return do_echo(conn, count, readable);
}
//////////////////////////////////////////////////////////////////////////////
static void usage(const char* procname) {
printf("usage: %s -h [help]\r\n"
" -e event_type[kernel|select|poll, default: kernel]\r\n"
" -s server_addr[default: 127.0.0.1:9000]\r\n"
" -c fiber_count[default: 1]\r\n"
" -n count[default: 100]\r\n"
" -R [if call readable? default: false]\r\n"
" -t connect_timeout[default: 10 seconds]\r\n"
" -r rw_timeout[default: 10 seconds]\r\n"
, procname);
}
int main(int argc, char *argv[]) {
int ch, nfiber = 1, count = 100, conn_timeout = 10, rw_timeout = 10;
bool readable = false;
acl::string addr = "127.0.0.1:9000", event_type("kernel");
acl::acl_cpp_init();
acl::log::stdout_open(true);
acl::fiber::stdout_open(true);
while ((ch = getopt(argc, argv, "he:s:c:n:Rt:r:")) > 0) {
switch (ch) {
case 'h':
usage(argv[0]);
return 0;
case 'e':
event_type = optarg;
break;
case 's':
addr = optarg;
break;
case 'c':
nfiber = atoi(optarg);
break;
case 'n':
count = atoi(optarg);
break;
case 'R':
readable = true;
break;
case 't':
conn_timeout = atoi(optarg);
break;
case 'r':
rw_timeout = atoi(optarg);
break;
default:
break;
}
}
acl::fiber_event_t type;
if (event_type == "select") {
type = acl::FIBER_EVENT_T_SELECT;
} else if (event_type == "poll") {
type = acl::FIBER_EVENT_T_POLL;
} else {
type = acl::FIBER_EVENT_T_KERNEL;
}
long long total = 0;
struct timeval begin;
gettimeofday(&begin, NULL);
for (int i = 0; i < nfiber; i++) {
go[&] {
total += start(addr, count, readable, conn_timeout, rw_timeout);
};
}
acl::fiber::schedule_with(type);
struct timeval end;
gettimeofday(&end, NULL);
double cost = acl::stamp_sub(end, begin);
printf("Total count=%lld, cost=%.2f ms, speed=%.2f\r\n",
total, cost, (total * 1000) / (cost > 0 ? cost : 0.1));
return 0;
}