tcp_chat_server.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. /*
  2. * tcp chat server
  3. *
  4. * @build make examples
  5. * @server bin/tcp_chat_server 1234
  6. * @clients bin/nc 127.0.0.1 1234
  7. * nc 127.0.0.1 1234
  8. * telnet 127.0.0.1 1234
  9. */
  10. #include "hloop.h"
  11. #include "hsocket.h"
  12. #include "hbase.h"
  13. #include "list.h"
  14. unpack_setting_t unpack_setting;
  15. // hloop_create_tcp_server
  16. // on_accept => join
  17. // on_recv => broadcast
  18. // on_close => leave
  19. typedef struct chatroom_s {
  20. hloop_t* loop;
  21. hio_t* listenio;
  22. int roomid;
  23. struct list_head conns;
  24. } chatroom_t;
  25. typedef struct connection_s {
  26. hio_t* connio;
  27. char addr[SOCKADDR_STRLEN];
  28. struct list_node node;
  29. } connection_t;
  30. static chatroom_t s_chatroom;
  31. static void join(chatroom_t* room, connection_t* conn);
  32. static void leave(chatroom_t* room, connection_t* conn);
  33. static void broadcast(chatroom_t* room, const char* msg, int msglen);
  34. void join(chatroom_t* room, connection_t* conn) {
  35. list_add(&conn->node, &room->conns);
  36. char msg[256] = {0};
  37. int msglen = 0;
  38. struct list_node* node;
  39. connection_t* cur;
  40. msglen = snprintf(msg, sizeof(msg), "room[%06d] clients:\r\n", room->roomid);
  41. hio_write(conn->connio, msg, msglen);
  42. list_for_each (node, &room->conns) {
  43. cur = list_entry(node, connection_t, node);
  44. msglen = snprintf(msg, sizeof(msg), "[%s]\r\n", cur->addr);
  45. hio_write(conn->connio, msg, msglen);
  46. }
  47. hio_write(conn->connio, "\r\n", 2);
  48. msglen = snprintf(msg, sizeof(msg), "client[%s] join room[%06d]\r\n", conn->addr, room->roomid);
  49. broadcast(room, msg, msglen);
  50. }
  51. void leave(chatroom_t* room, connection_t* conn) {
  52. list_del(&conn->node);
  53. char msg[256] = {0};
  54. int msglen = snprintf(msg, sizeof(msg), "client[%s] leave room[%d]\r\n", conn->addr, room->roomid);
  55. broadcast(room, msg, msglen);
  56. }
  57. void broadcast(chatroom_t* room, const char* msg, int msglen) {
  58. printf("> %.*s", msglen, msg);
  59. struct list_node* node;
  60. connection_t* conn;
  61. list_for_each (node, &room->conns) {
  62. conn = list_entry(node, connection_t, node);
  63. hio_write(conn->connio, msg, msglen);
  64. }
  65. }
  66. static void on_close(hio_t* io) {
  67. printf("on_close fd=%d error=%d\n", hio_fd(io), hio_error(io));
  68. connection_t* conn = (connection_t*)hevent_userdata(io);
  69. if (conn) {
  70. hevent_set_userdata(io, NULL);
  71. leave(&s_chatroom, conn);
  72. HV_FREE(conn);
  73. }
  74. }
  75. static void on_recv(hio_t* io, void* buf, int readbytes) {
  76. printf("on_recv fd=%d readbytes=%d\n", hio_fd(io), readbytes);
  77. char localaddrstr[SOCKADDR_STRLEN] = {0};
  78. char peeraddrstr[SOCKADDR_STRLEN] = {0};
  79. printf("[%s] <=> [%s]\n",
  80. SOCKADDR_STR(hio_localaddr(io), localaddrstr),
  81. SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
  82. printf("< %.*s", readbytes, (char*)buf);
  83. // broadcast
  84. connection_t* conn = (connection_t*)hevent_userdata(io);
  85. assert(conn != NULL);
  86. char msg[256] = {0};
  87. int msglen = snprintf(msg, sizeof(msg), "client[%s] say: %.*s", conn->addr, readbytes, (char*)buf);
  88. broadcast(&s_chatroom, msg, msglen);
  89. }
  90. static void on_accept(hio_t* io) {
  91. printf("on_accept connfd=%d\n", hio_fd(io));
  92. char localaddrstr[SOCKADDR_STRLEN] = {0};
  93. char peeraddrstr[SOCKADDR_STRLEN] = {0};
  94. printf("accept connfd=%d [%s] <= [%s]\n", hio_fd(io),
  95. SOCKADDR_STR(hio_localaddr(io), localaddrstr),
  96. SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
  97. hio_setcb_close(io, on_close);
  98. hio_setcb_read(io, on_recv);
  99. hio_set_unpack(io, &unpack_setting);
  100. hio_read_start(io);
  101. // free on_close
  102. connection_t* conn = NULL;
  103. HV_ALLOC_SIZEOF(conn);
  104. conn->connio = io;
  105. strcpy(conn->addr, peeraddrstr);
  106. hevent_set_userdata(io, conn);
  107. join(&s_chatroom, conn);
  108. }
  109. int main(int argc, char** argv) {
  110. if (argc < 2) {
  111. printf("Usage: %s port\n", argv[0]);
  112. return -10;
  113. }
  114. int port = atoi(argv[1]);
  115. memset(&unpack_setting, 0, sizeof(unpack_setting_t));
  116. unpack_setting.package_max_length = DEFAULT_PACKAGE_MAX_LENGTH;
  117. unpack_setting.mode = UNPACK_BY_DELIMITER;
  118. unpack_setting.delimiter[0] = '\n';
  119. unpack_setting.delimiter_bytes = 1;
  120. hloop_t* loop = hloop_new(0);
  121. hio_t* listenio = hloop_create_tcp_server(loop, "0.0.0.0", port, on_accept);
  122. if (listenio == NULL) {
  123. return -20;
  124. }
  125. printf("listenfd=%d\n", hio_fd(listenio));
  126. s_chatroom.loop = loop;
  127. s_chatroom.listenio = listenio;
  128. s_chatroom.roomid = hv_rand(100000, 999999);
  129. list_init(&s_chatroom.conns);
  130. hloop_run(loop);
  131. hloop_free(&loop);
  132. return 0;
  133. }