HTTPRequest.cpp 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. /* HTTPRequest.cpp/Open GoPro, Version 2.0 (C) Copyright 2021 GoPro, Inc. (http://gopro.com/OpenGoPro). */
  2. /* This copyright was auto-generated on Sat Mar 5 01:05:53 UTC 2022 */
  3. #include "HTTPRequest.h"
  4. HTTPRequest::HTTPRequest(const std::string& host, const short port)
  5. : Host(host), Port(port)
  6. {
  7. InitWinsock();
  8. if ((Sock = socket(AF_INET, SOCK_STREAM, 0)) == INVALID_SOCKET)
  9. {
  10. int err = WSAGetLastError();
  11. throw std::exception("Couldn't create socket " + err);
  12. }
  13. }
  14. HTTPRequest::~HTTPRequest()
  15. {
  16. WSACleanup();
  17. closesocket(Sock);
  18. }
  19. std::string HTTPRequest::get_response()
  20. {
  21. return Response;
  22. }
  23. bool HTTPRequest::InitWinsock()
  24. {
  25. WSADATA wsaData;
  26. if (WSAStartup(MAKEWORD(1, 1), &wsaData) != 0)
  27. {
  28. return false;
  29. }
  30. return true;
  31. }
  32. bool HTTPRequest::loop_recieve()
  33. {
  34. while (true)
  35. {
  36. char recvBuf[256];
  37. auto nret = recv(Sock, recvBuf, sizeof(recvBuf), 0);
  38. if (nret == -1)
  39. {
  40. return false;
  41. }
  42. else if (nret == 0)
  43. {
  44. break;
  45. }
  46. Response.append(recvBuf, nret);
  47. }
  48. return true;
  49. }
  50. bool HTTPRequest::resolve_and_connect()
  51. {
  52. bool retbool = true;
  53. if (connected)
  54. return retbool;
  55. sockaddr_in add;
  56. add.sin_family = AF_INET;
  57. InetPtonA(AF_INET, Host.c_str(), &add.sin_addr.s_addr);
  58. add.sin_port = htons(Port);
  59. int ret = -1;
  60. // Set non-blocking
  61. unsigned long modeB = 1;
  62. ret = ioctlsocket(Sock, FIONBIO, &modeB);
  63. if (ret != NO_ERROR)
  64. OutputDebugStringA("ioctlsocket failed\n");
  65. try
  66. {
  67. ret = connect(Sock, reinterpret_cast<SOCKADDR*>(&add), sizeof(add));
  68. if (ret < 0)
  69. {
  70. int boo = WSAGetLastError();
  71. if (boo == WSAEWOULDBLOCK)
  72. {
  73. OutputDebugStringA("connect in progress - selecting\n");
  74. fd_set setW, setE;
  75. FD_ZERO(&setW);
  76. FD_SET(Sock, &setW);
  77. FD_ZERO(&setE);
  78. FD_SET(Sock, &setE);
  79. timeval time_out = { 0 };
  80. time_out.tv_sec = 2;
  81. time_out.tv_usec = 0;
  82. ret = select(0, NULL, &setW, &setE, &time_out);
  83. if (ret <= 0)
  84. {
  85. // select() failed or connection timed out
  86. closesocket(Sock);
  87. if (ret == 0)
  88. {
  89. TimedOut = true;
  90. WSASetLastError(WSAETIMEDOUT);
  91. }
  92. ret = -1;
  93. }
  94. if (FD_ISSET(Sock, &setE))
  95. {
  96. // connection failed
  97. closesocket(Sock);
  98. ret = -1;
  99. }
  100. // put socked in blocking mode...
  101. modeB = 0;
  102. if (ioctlsocket(Sock, FIONBIO, &modeB) == SOCKET_ERROR)
  103. {
  104. closesocket(Sock);
  105. ret = -1;
  106. }
  107. }
  108. else
  109. {
  110. ret = -1;
  111. }
  112. }
  113. }
  114. catch(...)
  115. {
  116. }
  117. if (ret < 0)
  118. {
  119. int err = WSAGetLastError();
  120. char buf[128] = { 0 };
  121. sprintf(buf, "error opening socket %d", err);
  122. OutputDebugStringA(buf);
  123. }
  124. else
  125. {
  126. connected = true;
  127. }
  128. return retbool;
  129. }
  130. bool HTTPRequest::get_request(const std::string& path)
  131. {
  132. if (!resolve_and_connect())
  133. return false;
  134. std::string request = "GET " + path + " HTTP/1.1" + "\r\n";
  135. request += "Host: " + Host + "\r\n";
  136. request += "Connection: close\r\n";
  137. request += "\r\n";
  138. if (send(Sock, request.c_str(), request.length(), 0) == SOCKET_ERROR)
  139. {
  140. return false;
  141. }
  142. return loop_recieve();
  143. }