241 lines
6.2 KiB
C
241 lines
6.2 KiB
C
|
/*
|
||
|
* network client
|
||
|
*
|
||
|
* @build: make examples
|
||
|
* @server bin/httpd -s restart -d
|
||
|
* @usage: bin/nc 127.0.0.1 8080
|
||
|
* > GET / HTTP/1.1
|
||
|
* > Connection: close
|
||
|
* > [Enter]
|
||
|
* > GET / HTTP/1.1
|
||
|
* > Connection: keep-alive
|
||
|
* > [Enter]
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* @test udp client
|
||
|
* @build ./configure && make examples
|
||
|
* @client bin/nc -u 127.0.0.1 1234
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* @test ssl client
|
||
|
* @build ./configure --with-openssl && make clean && make
|
||
|
* @client bin/nc -s 127.0.0.1 1234
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* @test kcp client
|
||
|
* @build ./configure --with-kcp && make clean && make
|
||
|
* @client bin/nc -k 127.0.0.1 1234
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include "hloop.h"
|
||
|
#include "hbase.h"
|
||
|
#include "hsocket.h"
|
||
|
#include "hssl.h"
|
||
|
|
||
|
#define RECV_BUFSIZE 8192
|
||
|
static char recvbuf[RECV_BUFSIZE];
|
||
|
|
||
|
static char protocol = 't';
|
||
|
static const char* protocolname = "tcp";
|
||
|
// for stdin
|
||
|
static hio_t* stdinio = NULL;
|
||
|
// for socket
|
||
|
static hio_t* sockio = NULL;
|
||
|
|
||
|
static int verbose = 0;
|
||
|
|
||
|
static void send_heartbeat(hio_t* io) {
|
||
|
static char buf[] = "PING\r\n";
|
||
|
// printf("send_heartbeat %s", buf);
|
||
|
hio_write(io, buf, 6);
|
||
|
}
|
||
|
|
||
|
static void on_recv(hio_t* io, void* buf, int readbytes) {
|
||
|
// printf("on_recv fd=%d readbytes=%d\n", hio_fd(io), readbytes);
|
||
|
if (verbose) {
|
||
|
char localaddrstr[SOCKADDR_STRLEN] = {0};
|
||
|
char peeraddrstr[SOCKADDR_STRLEN] = {0};
|
||
|
printf("[%s] <=> [%s]\n",
|
||
|
SOCKADDR_STR(hio_localaddr(io), localaddrstr),
|
||
|
SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
|
||
|
}
|
||
|
printf("%.*s", readbytes, (char*)buf);
|
||
|
|
||
|
// test hio_set_readbuf in hread_cb
|
||
|
#if 0
|
||
|
static int total_readbytes = 0;
|
||
|
total_readbytes += readbytes;
|
||
|
if (total_readbytes >= RECV_BUFSIZE) {
|
||
|
total_readbytes = 0;
|
||
|
}
|
||
|
hio_set_readbuf(io, recvbuf + total_readbytes, RECV_BUFSIZE - total_readbytes);
|
||
|
printf("%.*s", total_readbytes, recvbuf);
|
||
|
#endif
|
||
|
|
||
|
fflush(stdout);
|
||
|
}
|
||
|
|
||
|
static void on_stdin(hio_t* io, void* buf, int readbytes) {
|
||
|
// printf("on_stdin fd=%d readbytes=%d\n", hio_fd(io), readbytes);
|
||
|
// printf("> %s\n", buf);
|
||
|
|
||
|
char* str = (char*)buf;
|
||
|
|
||
|
// test hio_read_start/hio_read_stop/hio_close/hloop_stop
|
||
|
#if 1
|
||
|
if (strncmp(str, "start", 5) == 0) {
|
||
|
printf("call hio_read_start\n");
|
||
|
hio_read_start(sockio);
|
||
|
return;
|
||
|
}
|
||
|
else if (strncmp(str, "stop", 4) == 0) {
|
||
|
printf("call hio_read_stop\n");
|
||
|
hio_read_stop(sockio);
|
||
|
return;
|
||
|
}
|
||
|
else if (strncmp(str, "close", 5) == 0) {
|
||
|
printf("call hio_close\n");
|
||
|
hio_close(sockio);
|
||
|
return;
|
||
|
}
|
||
|
else if (strncmp(str, "quit", 4) == 0) {
|
||
|
printf("call hloop_stop\n");
|
||
|
hloop_stop(hevent_loop(io));
|
||
|
return;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
// CR|LF => CRLF for test most protocols
|
||
|
char eol = str[readbytes-1];
|
||
|
if (eol == '\n' || eol == '\r') {
|
||
|
if (readbytes > 1 && str[readbytes-2] == '\r' && eol == '\n') {
|
||
|
// have been CRLF
|
||
|
}
|
||
|
else {
|
||
|
++readbytes;
|
||
|
str[readbytes - 2] = '\r';
|
||
|
str[readbytes - 1] = '\n';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hio_write(sockio, buf, readbytes);
|
||
|
|
||
|
if (strncmp(str, "CLOSE", 5) == 0) {
|
||
|
printf("call hio_close\n");
|
||
|
hio_close(sockio);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void on_close(hio_t* io) {
|
||
|
// printf("on_close fd=%d error=%d\n", hio_fd(io), hio_error(io));
|
||
|
hio_del(stdinio, HV_READ);
|
||
|
}
|
||
|
|
||
|
static void on_connect(hio_t* io) {
|
||
|
// printf("on_connect fd=%d\n", hio_fd(io));
|
||
|
if (verbose) {
|
||
|
char localaddrstr[SOCKADDR_STRLEN] = {0};
|
||
|
char peeraddrstr[SOCKADDR_STRLEN] = {0};
|
||
|
printf("connect connfd=%d [%s] => [%s]\n", hio_fd(io),
|
||
|
SOCKADDR_STR(hio_localaddr(io), localaddrstr),
|
||
|
SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
|
||
|
}
|
||
|
|
||
|
hio_read_start(io);
|
||
|
// uncomment to test heartbeat
|
||
|
// hio_set_heartbeat(sockio, 3000, send_heartbeat);
|
||
|
}
|
||
|
|
||
|
int main(int argc, char** argv) {
|
||
|
if (argc < 3) {
|
||
|
printf("\
|
||
|
Usage: nc [-tusk] host port\n\
|
||
|
Options:\n\
|
||
|
-t Use tcp protocol (default)\n\
|
||
|
-u Use udp protocol\n\
|
||
|
-s Use ssl protocol\n\
|
||
|
-k Use kcp protocol\n\
|
||
|
Examples: nc 127.0.0.1 80\n\
|
||
|
nc -u 127.0.0.1 80\n");
|
||
|
return -10;
|
||
|
}
|
||
|
|
||
|
int index = 1;
|
||
|
if (argv[1][0] == '-') {
|
||
|
protocol = argv[1][1];
|
||
|
switch(protocol) {
|
||
|
case 't': protocolname = "tcp"; break;
|
||
|
case 'u': protocolname = "udp"; break;
|
||
|
case 's': protocolname = "ssl"; break;
|
||
|
case 'k': protocolname = "kcp"; break;
|
||
|
default: fprintf(stderr, "Unsupported protocol '%c'\n", protocol); exit(1);
|
||
|
}
|
||
|
++index;
|
||
|
}
|
||
|
const char* host = argv[index++];
|
||
|
int port = atoi(argv[index++]);
|
||
|
if (verbose) {
|
||
|
printf("%s %s %d\n", protocolname, host, port);
|
||
|
}
|
||
|
|
||
|
HV_MEMCHECK;
|
||
|
|
||
|
hloop_t* loop = hloop_new(HLOOP_FLAG_QUIT_WHEN_NO_ACTIVE_EVENTS);
|
||
|
|
||
|
// stdin use default readbuf
|
||
|
stdinio = hread(loop, 0, NULL, 0, on_stdin);
|
||
|
if (stdinio == NULL) {
|
||
|
return -20;
|
||
|
}
|
||
|
|
||
|
// socket
|
||
|
if (protocol == 't' || protocol == 's') {
|
||
|
// tcp
|
||
|
sockio = hloop_create_tcp_client(loop, host, port, on_connect, on_close);
|
||
|
if (sockio == NULL) {
|
||
|
return -20;
|
||
|
}
|
||
|
if (protocol == 's') {
|
||
|
if (strcmp(hssl_backend(), "nossl") == 0) {
|
||
|
fprintf(stderr, "Please recompile WITH_SSL!\n");
|
||
|
exit(1);
|
||
|
}
|
||
|
hio_enable_ssl(sockio);
|
||
|
}
|
||
|
}
|
||
|
else if (protocol == 'u' || protocol == 'k') {
|
||
|
// udp
|
||
|
sockio = hloop_create_udp_client(loop, host, port);
|
||
|
if (sockio == NULL) {
|
||
|
return -20;
|
||
|
}
|
||
|
if (protocol == 'k') {
|
||
|
#if WITH_KCP
|
||
|
static kcp_setting_t s_kcp_setting;
|
||
|
kcp_setting_init_with_normal_mode(&s_kcp_setting);
|
||
|
s_kcp_setting.conv = hv_rand(1, 999999);
|
||
|
hio_set_kcp(sockio, &s_kcp_setting);
|
||
|
#else
|
||
|
fprintf(stderr, "Please recompile WITH_KCP!\n");
|
||
|
exit(1);
|
||
|
#endif
|
||
|
}
|
||
|
hio_read(sockio);
|
||
|
}
|
||
|
// printf("sockfd=%d\n", hio_fd(sockio));
|
||
|
hio_setcb_close(sockio, on_close);
|
||
|
hio_setcb_read(sockio, on_recv);
|
||
|
hio_set_readbuf(sockio, recvbuf, RECV_BUFSIZE);
|
||
|
|
||
|
hloop_run(loop);
|
||
|
hloop_free(&loop);
|
||
|
|
||
|
return 0;
|
||
|
}
|