httpreqdata.cpp 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. #include "httpreqdata.h"
  2. #include <util/stream/mem.h>
  3. TBaseServerRequestData::TBaseServerRequestData(SOCKET s)
  4. : Addr(nullptr)
  5. , Host()
  6. , Port()
  7. , Path(nullptr)
  8. , Search(nullptr)
  9. , SearchLength(0)
  10. , Socket(s)
  11. , BeginTime(MicroSeconds())
  12. {
  13. }
  14. TBaseServerRequestData::TBaseServerRequestData(const char* qs, SOCKET s)
  15. : Addr(nullptr)
  16. , Host()
  17. , Port()
  18. , Path(nullptr)
  19. , Search((char*)qs)
  20. , SearchLength(qs ? strlen(qs) : 0)
  21. , OrigSearch(Search, SearchLength)
  22. , Socket(s)
  23. , BeginTime(MicroSeconds())
  24. {
  25. }
  26. void TBaseServerRequestData::AppendQueryString(const char* str, size_t length) {
  27. if (Y_UNLIKELY(Search)) {
  28. Y_ASSERT(strlen(Search) == SearchLength);
  29. ModifiedQueryString.Reserve(SearchLength + length + 2);
  30. ModifiedQueryString.Assign(Search, SearchLength);
  31. if (SearchLength > 0 && Search[SearchLength - 1] != '&' &&
  32. length > 0 && str[0] != '&') {
  33. ModifiedQueryString.Append('&');
  34. }
  35. ModifiedQueryString.Append(str, length);
  36. } else {
  37. ModifiedQueryString.Reserve(length + 1);
  38. ModifiedQueryString.Assign(str, length);
  39. }
  40. ModifiedQueryString.Append('\0');
  41. Search = ModifiedQueryString.data();
  42. SearchLength = ModifiedQueryString.size() - 1; // ignore terminator
  43. }
  44. void TBaseServerRequestData::SetRemoteAddr(TStringBuf addr) {
  45. TMemoryOutput out(AddrData, Y_ARRAY_SIZE(AddrData) - 1);
  46. out.Write(addr.substr(0, Y_ARRAY_SIZE(AddrData) - 1));
  47. *out.Buf() = '\0';
  48. Addr = AddrData;
  49. }
  50. const char* TBaseServerRequestData::RemoteAddr() const {
  51. if (!Addr) {
  52. *AddrData = 0;
  53. GetRemoteAddr(Socket, AddrData, sizeof(AddrData));
  54. Addr = AddrData;
  55. }
  56. return Addr;
  57. }
  58. const char* TBaseServerRequestData::HeaderIn(TStringBuf key) const {
  59. auto it = HeadersIn_.find(key);
  60. if (it == HeadersIn_.end()) {
  61. return nullptr;
  62. }
  63. return it->second.data();
  64. }
  65. TString TBaseServerRequestData::HeaderByIndex(size_t n) const noexcept {
  66. if (n >= HeadersCount()) {
  67. return nullptr;
  68. }
  69. THttpHeadersContainer::const_iterator i = HeadersIn_.begin();
  70. while (n) {
  71. ++i;
  72. --n;
  73. }
  74. return TString(i->first) + TStringBuf(": ") + i->second;
  75. }
  76. const char* TBaseServerRequestData::Environment(const char* key) const {
  77. if (stricmp(key, "REMOTE_ADDR") == 0) {
  78. const char* ip = HeaderIn("X-Real-IP");
  79. if (ip)
  80. return ip;
  81. return RemoteAddr();
  82. } else if (stricmp(key, "QUERY_STRING") == 0) {
  83. return QueryString();
  84. } else if (stricmp(key, "SERVER_NAME") == 0) {
  85. return ServerName().data();
  86. } else if (stricmp(key, "SERVER_PORT") == 0) {
  87. return ServerPort().data();
  88. } else if (stricmp(key, "SCRIPT_NAME") == 0) {
  89. return ScriptName();
  90. }
  91. return nullptr;
  92. }
  93. void TBaseServerRequestData::Clear() {
  94. HeadersIn_.clear();
  95. Addr = Path = Search = nullptr;
  96. OrigSearch = {};
  97. SearchLength = 0;
  98. Host.clear();
  99. Port.clear();
  100. CurPage.remove();
  101. ParseBuf.Clear();
  102. BeginTime = MicroSeconds();
  103. }
  104. const char* TBaseServerRequestData::GetCurPage() const {
  105. if (!CurPage && Host) {
  106. CurPage = "http://";
  107. CurPage += Host;
  108. if (Port) {
  109. CurPage += ':';
  110. CurPage += Port;
  111. }
  112. CurPage += Path;
  113. if (Search) {
  114. CurPage += '?';
  115. CurPage += Search;
  116. }
  117. }
  118. return CurPage.data();
  119. }
  120. bool TBaseServerRequestData::Parse(const char* origReq) {
  121. size_t origReqLength = strlen(origReq);
  122. ParseBuf.Assign(origReq, origReqLength + 1);
  123. char* req = ParseBuf.Data();
  124. while (*req == ' ' || *req == '\t')
  125. req++;
  126. if (*req != '/')
  127. return false; // we are not a proxy
  128. while (req[1] == '/') // remove redundant slashes
  129. req++;
  130. // detect url end (can contain some garbage after whitespace, e.g. 'HTTP 1.1')
  131. char* urlEnd = req;
  132. while (*urlEnd && *urlEnd != ' ' && *urlEnd != '\t')
  133. urlEnd++;
  134. if (*urlEnd)
  135. *urlEnd = 0;
  136. // cut fragment if exists
  137. char* fragment = strchr(req, '#');
  138. if (fragment)
  139. *fragment = 0; // ignore fragment
  140. else
  141. fragment = urlEnd;
  142. Path = req;
  143. // calculate Search length without additional strlen-ing
  144. Search = strchr(Path, '?');
  145. if (Search) {
  146. *Search++ = 0;
  147. ptrdiff_t delta = fragment - Search;
  148. // indeed, second case is a parse error
  149. SearchLength = (delta >= 0) ? delta : (urlEnd - Search);
  150. Y_ASSERT(strlen(Search) == SearchLength);
  151. } else {
  152. SearchLength = 0;
  153. }
  154. OrigSearch = {Search, SearchLength};
  155. return true;
  156. }
  157. void TBaseServerRequestData::AddHeader(const TString& name, const TString& value) {
  158. HeadersIn_[name] = value;
  159. if (stricmp(name.data(), "Host") == 0) {
  160. size_t hostLen = strcspn(value.data(), ":");
  161. if (value[hostLen] == ':')
  162. Port = value.substr(hostLen + 1);
  163. Host = value.substr(0, hostLen);
  164. }
  165. }
  166. void TBaseServerRequestData::SetPath(const TString& path) {
  167. PathStorage = TBuffer(path.data(), path.size() + 1);
  168. Path = PathStorage.Data();
  169. }