socketio_win32.cpp 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. #include "socketio.hpp"
  2. #include <stdio.h>
  3. #include <winsock2.h>
  4. #include <ws2tcpip.h>
  5. #pragma comment(lib, "ws2_32.lib")
  6. #include <string>
  7. #include <cstring>
  8. #include <vector>
  9. #include <iostream>
  10. using std::size_t;
  11. struct socket_exception : public std::exception{
  12. std::string msg;
  13. socket_exception(std::string&& _msg){
  14. msg = std::move(_msg);
  15. }
  16. socket_exception(const std::string& _msg){
  17. msg = _msg;
  18. }
  19. socket_exception(const char* _msg){
  20. msg = std::string(_msg);
  21. }
  22. const char* what() const throw (){
  23. //return "asdasdsa";
  24. return msg.length() == 0 ? "Connection creation failure" : msg.c_str();
  25. }
  26. };
  27. std::string GetLastErrorAsString(){
  28. //Get the error message, if any.
  29. DWORD errorMessageID = ::GetLastError();
  30. if(errorMessageID == 0)
  31. return std::string();
  32. LPSTR messageBuffer = nullptr;
  33. size_t size = FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
  34. NULL, errorMessageID, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPSTR)&messageBuffer, 0, NULL);
  35. std::string message(messageBuffer, size);
  36. //Free the buffer.
  37. LocalFree(messageBuffer);
  38. return message;
  39. }
  40. cppsocket::cppsocket(){
  41. buffer = std::vector<char>(buffersize + 1, 0);
  42. }
  43. cppsocket::cppsocket(cppsocket&& o){
  44. s = o.s;
  45. o.s = INVALID_SOCKET;
  46. buffer = std::move(o.buffer);
  47. }
  48. cppsocket::cppsocket(WSADATA d, SOCKET _s){
  49. s = _s;
  50. wsa = d;
  51. buffer = std::vector<char>(buffersize + 1, 0);
  52. }
  53. cppsocket::cppsocket(const std::string& addr, unsigned int PORT){
  54. struct sockaddr_in server;
  55. buffer = std::vector<char>(buffersize + 1, 0);
  56. int recv_size;
  57. if (WSAStartup(MAKEWORD(2,2),&wsa) != 0){
  58. throw socket_exception(std::string("Socket creation error: ") + GetLastErrorAsString());
  59. }
  60. if((s = socket(AF_INET , SOCK_STREAM , 0 )) == INVALID_SOCKET){
  61. throw socket_exception(std::string("Socket creation error: ") + GetLastErrorAsString());
  62. }
  63. server.sin_addr.s_addr = inet_addr(addr.c_str());
  64. server.sin_family = AF_INET;
  65. server.sin_port = htons(PORT);
  66. if (connect(s , (struct sockaddr *)&server , sizeof(server)) < 0){
  67. throw socket_exception(std::string("Could not connect to host: ") + GetLastErrorAsString());
  68. }
  69. }
  70. void cppsocket::write(const std::string& message){
  71. std::vector<char> msg(message.c_str(), message.c_str() + message.size());
  72. write(msg);
  73. }
  74. void cppsocket::write(const std::vector<char>& message){
  75. for(size_t i = 0;i < message.size();i += buffersize){
  76. char cs[buffersize + 1] = {0};
  77. std::memcpy(cs, message.data() + i,buffersize);
  78. if((i + buffersize) < message.size()){
  79. cs[buffersize] = 'c';
  80. if(send(s, cs, buffersize + 1, 0) < 0){
  81. throw socket_exception(std::string("Couldn't write to peer: ") + GetLastErrorAsString());
  82. }
  83. }
  84. else{
  85. cs[message.size() - i] = (char)0;
  86. if(send(s, cs, message.size() - i, 0) < 0){
  87. throw socket_exception(std::string("Couldn't write to peer: ") + GetLastErrorAsString());
  88. }
  89. }
  90. }
  91. }
  92. std::vector<char> cppsocket::receive(){
  93. std::vector<char> stor;
  94. while(true){
  95. std::fill(buffer.begin(), buffer.end(), (char)0);
  96. ssize_t val = recv(s, buffer.data(), buffersize + 1,0);
  97. if(val == 0)throw socket_exception("Connection closed by peer");
  98. if(val < 0){
  99. throw socket_exception(GetLastErrorAsString());
  100. }
  101. stor.insert(stor.end(), buffer.begin(), buffer.begin() + std::min(val, (ssize_t)buffersize));
  102. if(buffer.data()[buffersize] == (char)0){break;}
  103. }
  104. std::cout << std::endl;
  105. return stor;
  106. }
  107. void cppsocket::close(){
  108. closesocket(s);
  109. }
  110. server_socket::server_socket(int port){
  111. struct addrinfo *result = NULL;
  112. struct addrinfo hints;
  113. iResult = WSAStartup(MAKEWORD(2,2), &wsaData);
  114. if (iResult != 0) {
  115. printf("WSAStartup failed with error: %d\n", iResult);
  116. }
  117. ZeroMemory(&hints, sizeof(hints));
  118. hints.ai_family = AF_INET;
  119. hints.ai_socktype = SOCK_STREAM;
  120. hints.ai_protocol = IPPROTO_TCP;
  121. hints.ai_flags = AI_PASSIVE;
  122. // Resolve the server address and port
  123. iResult = getaddrinfo(NULL, std::to_string(port).c_str(), &hints, &result);
  124. if ( iResult != 0 ) {
  125. printf("getaddrinfo failed with error: %d\n", iResult);
  126. WSACleanup();
  127. }
  128. // Create a SOCKET for connecting to server
  129. ListenSocket = socket(result->ai_family, result->ai_socktype, result->ai_protocol);
  130. if (ListenSocket == INVALID_SOCKET) {
  131. printf("socket failed with error: %ld\n", WSAGetLastError());
  132. freeaddrinfo(result);
  133. WSACleanup();
  134. }
  135. // Setup the TCP listening socket
  136. iResult = bind( ListenSocket, result->ai_addr, (int)result->ai_addrlen);
  137. if (iResult == SOCKET_ERROR) {
  138. printf("bind failed with error: %d\n", WSAGetLastError());
  139. freeaddrinfo(result);
  140. closesocket(ListenSocket);
  141. WSACleanup();
  142. }
  143. freeaddrinfo(result);
  144. iResult = listen(ListenSocket, SOMAXCONN);
  145. if (iResult == SOCKET_ERROR) {
  146. printf("listen failed with error: %d\n", WSAGetLastError());
  147. closesocket(ListenSocket);
  148. WSACleanup();
  149. }
  150. }
  151. cppsocket server_socket::accept_connection(){
  152. SOCKET ClientSocket = accept(ListenSocket, NULL, NULL);
  153. if (ClientSocket == INVALID_SOCKET) {
  154. printf("accept failed with error: %d\n", WSAGetLastError());
  155. closesocket(ListenSocket);
  156. WSACleanup();
  157. }
  158. return cppsocket(wsaData, ClientSocket);
  159. }
  160. void server_socket::close(){
  161. closesocket(ListenSocket);
  162. }
  163. cppsocket& cppsocket::operator=(cppsocket&& o){
  164. s = o.s;
  165. o.sock = INVALID_SOCKET;
  166. buffer = std::move(o.buffer);
  167. }
  168. /*int main(int argc, char** argv){
  169. server_socket ssock(80);
  170. while(true){
  171. cppsocket sock = ssock.accept_connection();
  172. sock.write("ab");
  173. std::vector<char> rec = sock.receive();
  174. std::cout << rec << ", ";
  175. std::cout << rec << std::endl;
  176. sock.close();
  177. }
  178. ssock.close();
  179. //Send some data
  180. std::string _message = std::string("Hallo");
  181. memcpy(message, _message.c_str(), _message.length() + 1);
  182. if( send(s , message , strlen(message) , 0) < 0){
  183. puts("Send failed");
  184. return 1;
  185. }
  186. puts("Data Send\n");
  187. //Receive a reply from the server
  188. if((recv_size = recv(s , server_reply , 2000 , 0)) == SOCKET_ERROR){
  189. puts("recv failed");
  190. }
  191. puts("Reply received\n");
  192. //Add a NULL terminating character to make it a proper string before printing
  193. server_reply[recv_size] = '\0';
  194. puts(server_reply);
  195. return 0;
  196. }*/