CClientSession.cpp 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. #include "CClientSession.h"
  2. #include "CServer.h"
  3. #include <sqlite3/sqlite3.h>
  4. #include "PosMessage.h"
  5. CClientSession::~CClientSession()
  6. {
  7. }
  8. void CClientSession::start()
  9. {
  10. m_pos_message.clear();
  11. boost::asio::async_read(socket_, boost::asio::buffer(&m_pos_message, sizeof(PosMessage)),
  12. boost::bind(&CClientSession::handle_read, this,
  13. boost::asio::placeholders::error,
  14. boost::asio::placeholders::bytes_transferred));
  15. }
  16. /*
  17. *服务器端主动关闭,才能调用这个函数
  18. **/
  19. void CClientSession::stop()
  20. {
  21. //先给客户端发送一个下线通知
  22. rapidjson::Document doc;
  23. doc.SetObject();
  24. rapidjson::Document::AllocatorType& allocator = doc.GetAllocator(); //获取分配器
  25. std::string timestamp = to_string(time(NULL));
  26. doc.AddMember("msg_type", "0", allocator);
  27. doc.AddMember("timestamp", StringRef(timestamp.c_str(), timestamp.length()), allocator);
  28. rapidjson::StringBuffer buffer;
  29. rapidjson::Writer<StringBuffer> writer(buffer);
  30. doc.Accept(writer);
  31. //返回给接入层的消息
  32. std::string login_out_msg = buffer.GetString();
  33. send_message(login_out_msg);
  34. //然后关闭socket
  35. socket_.close();
  36. //注意这里不能调用delete this,因为后面会异步返回,要判断状态,如果这里delete了,后面就会内存越界
  37. }
  38. /*
  39. *同步发送消息,返回true表示发送成功,返回false表示发送失败
  40. **/
  41. bool CClientSession::send_message(std::string msg)
  42. {
  43. try
  44. {
  45. PosMessage new_msg;
  46. new_msg.m_length = (int)msg.length();
  47. memcpy(new_msg.data, msg.c_str(), msg.length());
  48. new_msg.data[new_msg.m_length] = '\0';
  49. socket_.write_some(boost::asio::buffer(&new_msg, sizeof(PosMessage)));
  50. }
  51. catch (const std::exception& e)
  52. {
  53. std::string err = e.what();
  54. //注意这里是同步发送消息,这里不要处理socket的关闭和delete操作,让后面的异步读错误处理
  55. return false;
  56. }
  57. return true;
  58. }
  59. void CClientSession::SetServer(CServer* server)
  60. {
  61. m_server = server;
  62. }
  63. void CClientSession::handle_read(const boost::system::error_code& error,
  64. size_t bytes_transferred)
  65. {
  66. if(!error)
  67. {
  68. //读到客户端发过来的内容,进行绑定
  69. std::string msg = m_pos_message.data;
  70. rapidjson::Document document;
  71. document.Parse(msg.c_str());
  72. if(!document.IsObject())
  73. {
  74. LOG_ERROR("message 非法! msg:" << msg.c_str());
  75. LOG_ERROR("msg length:" << m_pos_message.m_length);
  76. LOG_ERROR("ip:" << socket_.remote_endpoint().address().to_string());
  77. //处理完了,重新读取
  78. m_pos_message.clear();
  79. boost::asio::async_read(socket_, boost::asio::buffer(&m_pos_message, sizeof(PosMessage)),
  80. boost::bind(&CClientSession::handle_read, this,
  81. boost::asio::placeholders::error,
  82. boost::asio::placeholders::bytes_transferred));
  83. /*
  84. socket_.async_read_some(boost::asio::buffer(&m_pos_message, sizeof(PosMessage)),
  85. boost::bind(&CClientSession::handle_read, this,
  86. boost::asio::placeholders::error,
  87. boost::asio::placeholders::bytes_transferred));*/
  88. return;
  89. }
  90. std::string username = document["username"].GetString();
  91. std::string timestamp = document["timestamp"].GetString();
  92. std::string is_login = document["is_login"].GetString();
  93. //这个用于判断是否成功绑定用户名了
  94. m_username = username;
  95. if (is_login == "1")
  96. {
  97. //LOG_INFO("get client login message:" << msg.c_str());
  98. //获取到用户名了,进行绑定
  99. m_server->BindUsername(username, this);
  100. //绑定完了检查这个用户有没有离线消息
  101. m_server->CheckOfflineMessage(username);
  102. }
  103. //处理完了,重新读取
  104. m_pos_message.clear();
  105. boost::asio::async_read(socket_, boost::asio::buffer(&m_pos_message, sizeof(PosMessage)),
  106. boost::bind(&CClientSession::handle_read, this,
  107. boost::asio::placeholders::error,
  108. boost::asio::placeholders::bytes_transferred));
  109. /*
  110. socket_.async_read_some(boost::asio::buffer(&m_pos_message, sizeof(PosMessage)),
  111. boost::bind(&CClientSession::handle_read, this,
  112. boost::asio::placeholders::error,
  113. boost::asio::placeholders::bytes_transferred));*/
  114. }
  115. else
  116. {
  117. std::string err = error.message();
  118. //LOG_ERROR("handle_read error, err message:" << err.c_str());
  119. if (socket_.is_open())
  120. {
  121. //这个是客户端断开了链接
  122. if (m_username == "")
  123. {
  124. //说明这个socket还没获取过用户名
  125. delete this;
  126. }
  127. else
  128. {
  129. m_server->DeleteClient(m_username);
  130. delete this;
  131. }
  132. }
  133. else
  134. {
  135. //这个是服务端断开了链接
  136. //LOG_ERROR("服务端断开了链接");
  137. delete this;
  138. }
  139. }
  140. }