DDNet documentation
Loading...
Searching...
No Matches
http.h
Go to the documentation of this file.
1#ifndef ENGINE_SHARED_HTTP_H
2#define ENGINE_SHARED_HTTP_H
3
4#include <base/dbg.h>
5#include <base/hash_ctxt.h>
6#include <base/mem.h>
7#include <base/str.h>
8#include <base/types.h>
9
10#include <engine/http.h>
11#include <engine/shared/jobs.h>
12
13#include <algorithm>
14#include <atomic>
15#include <condition_variable>
16#include <cstdlib>
17#include <deque>
18#include <mutex>
19#include <optional>
20#include <unordered_map>
21
22typedef struct _json_value json_value;
23class IStorage;
24
33
34enum class HTTPLOG
35{
39};
40
41enum class IPRESOLVE
42{
46};
47
49{
50public:
55};
56
58{
59 friend class CHttp;
60
61 enum class REQUEST
62 {
63 GET = 0,
67 };
68
69 static constexpr const char *GetRequestType(REQUEST Type)
70 {
71 switch(Type)
72 {
73 case REQUEST::GET:
74 return "GET";
75 case REQUEST::HEAD:
76 return "HEAD";
77 case REQUEST::POST:
79 return "POST";
80 }
81
82 dbg_assert_failed("unreachable");
83 }
84
85 char m_aUrl[256] = {0};
86
87 void *m_pHeaders = nullptr;
88 unsigned char *m_pBody = nullptr;
89 size_t m_BodyLength = 0;
90
92 bool m_SkipByFileTime = true;
93
95 int64_t m_MaxResponseSize = -1;
96 int64_t m_IfModifiedSince = -1;
98
99 std::optional<SHA256_DIGEST> m_ActualSha256;
101 std::optional<SHA256_DIGEST> m_ExpectedSha256;
102
103 bool m_WriteToMemory = true;
104 bool m_WriteToFile = false;
105
106 uint64_t m_ResponseLength = 0;
107
108 // If `m_WriteToMemory` is true.
109 size_t m_BufferSize = 0;
110 unsigned char *m_pBuffer = nullptr;
111
112 // If `m_WriteToFile` is true.
113 IOHANDLE m_File = nullptr;
114 int m_StorageType = 0xdeadbeef;
118
119 std::atomic<double> m_Size{0.0};
120 std::atomic<double> m_Current{0.0};
121 std::atomic<int> m_Progress{0};
124
126
127 char m_aErr[256]; // 256 == CURL_ERROR_SIZE
128 std::atomic<EHttpState> m_State{EHttpState::QUEUED};
129 std::atomic<bool> m_Abort{false};
130 std::mutex m_WaitMutex;
131 std::condition_variable m_WaitCondition;
132
134 bool m_HeadersEnded = false;
135 std::optional<int64_t> m_ResultDate = std::nullopt;
136 std::optional<int64_t> m_ResultLastModified = std::nullopt;
137
138 bool ShouldSkipRequest();
139 // Abort the request with an error if `BeforeInit()` returns false.
140 bool BeforeInit();
141 bool ConfigureHandle(void *pHandle); // void * == CURL *
142 // `pHandle` can be nullptr if no handle was ever created for this request.
143 void OnCompletionInternal(void *pHandle, unsigned int Result); // void * == CURL *, unsigned int == CURLcode
144
145 // Abort the request if `OnHeader()` returns something other than
146 // `DataSize`. `pHeader` is NOT null-terminated.
147 size_t OnHeader(char *pHeader, size_t HeaderSize);
148 // Abort the request if `OnData()` returns something other than
149 // `DataSize`.
150 size_t OnData(char *pData, size_t DataSize);
151
152 static int ProgressCallback(void *pUser, double DlTotal, double DlCurr, double UlTotal, double UlCurr);
153 static size_t HeaderCallback(char *pData, size_t Size, size_t Number, void *pUser);
154 static size_t WriteCallback(char *pData, size_t Size, size_t Number, void *pUser);
155
156protected:
157 // These run on the curl thread now, DO NOT STALL THE THREAD
158 virtual void OnProgress() {}
160
161public:
162 CHttpRequest(const char *pUrl);
163 virtual ~CHttpRequest();
164
166 // Skip the download if the local file is newer or as new as the remote file.
172 // Download to memory only. Get the result via `Result*`.
174 {
175 m_WriteToMemory = true;
176 m_WriteToFile = false;
177 }
178 // Download to filesystem and memory.
179 void WriteToFileAndMemory(IStorage *pStorage, const char *pDest, int StorageType);
180 // Download to the filesystem only.
181 void WriteToFile(IStorage *pStorage, const char *pDest, int StorageType);
182 // Don't place the file in the specified location until
183 // `OnValidation(true)` has been called.
185 void ExpectSha256(const SHA256_DIGEST &Sha256) { m_ExpectedSha256 = Sha256; }
187 void Post(const unsigned char *pData, size_t DataLength)
188 {
190 m_BodyLength = DataLength;
191 m_pBody = (unsigned char *)malloc(std::max((size_t)1, DataLength));
192 mem_copy(m_pBody, pData, DataLength);
193 }
194 void PostJson(const char *pJson)
195 {
197 m_BodyLength = str_length(pJson);
198 m_pBody = (unsigned char *)malloc(m_BodyLength);
200 }
201 void Header(const char *pNameColonValue);
202 void HeaderString(const char *pName, const char *pValue)
203 {
204 char aHeader[256];
205 str_format(aHeader, sizeof(aHeader), "%s: %s", pName, pValue);
206 Header(aHeader);
207 }
208 void HeaderInt(const char *pName, int Value)
209 {
210 char aHeader[256];
211 str_format(aHeader, sizeof(aHeader), "%s: %d", pName, Value);
212 Header(aHeader);
213 }
214
215 const char *Dest()
216 {
217 if(m_WriteToFile)
218 {
219 return m_aDest;
220 }
221 else
222 {
223 return nullptr;
224 }
225 }
226
227 double Current() const { return m_Current.load(std::memory_order_relaxed); }
228 double Size() const { return m_Size.load(std::memory_order_relaxed); }
229 int Progress() const { return m_Progress.load(std::memory_order_relaxed); }
230 EHttpState State() const { return m_State; }
231 bool Done() const
232 {
235 }
236 void Abort() { m_Abort = true; }
237 // If `ValidateBeforeOverwrite` is set, this needs to be called after
238 // validating that the downloaded file has the correct format.
239 //
240 // If called with `true`, it'll place the downloaded file at the final
241 // destination, if called with `false`, it'll instead delete the
242 // temporary downloaded file.
243 void OnValidation(bool Success);
244
245 void Wait();
246
247 void Result(unsigned char **ppResult, size_t *pResultLength) const;
248 json_value *ResultJson() const;
249 const SHA256_DIGEST &ResultSha256() const;
250
251 int StatusCode() const;
252 std::optional<int64_t> ResultAgeSeconds() const;
253 std::optional<int64_t> ResultLastModified() const;
254};
255
256inline std::unique_ptr<CHttpRequest> HttpHead(const char *pUrl)
257{
258 auto pResult = std::make_unique<CHttpRequest>(pUrl);
259 pResult->Head();
260 return pResult;
261}
262
263inline std::unique_ptr<CHttpRequest> HttpGet(const char *pUrl)
264{
265 return std::make_unique<CHttpRequest>(pUrl);
266}
267
268inline std::unique_ptr<CHttpRequest> HttpGetFile(const char *pUrl, IStorage *pStorage, const char *pOutputFile, int StorageType)
269{
270 std::unique_ptr<CHttpRequest> pResult = HttpGet(pUrl);
271 pResult->WriteToFile(pStorage, pOutputFile, StorageType);
272 pResult->Timeout(CTimeout{4000, 0, 500, 5});
273 return pResult;
274}
275
276inline std::unique_ptr<CHttpRequest> HttpGetBoth(const char *pUrl, IStorage *pStorage, const char *pOutputFile, int StorageType)
277{
278 std::unique_ptr<CHttpRequest> pResult = HttpGet(pUrl);
279 pResult->WriteToFileAndMemory(pStorage, pOutputFile, StorageType);
280 pResult->Timeout(CTimeout{4000, 0, 500, 5});
281 return pResult;
282}
283
284inline std::unique_ptr<CHttpRequest> HttpPost(const char *pUrl, const unsigned char *pData, size_t DataLength)
285{
286 auto pResult = std::make_unique<CHttpRequest>(pUrl);
287 pResult->Post(pData, DataLength);
288 pResult->Timeout(CTimeout{4000, 15000, 500, 5});
289 return pResult;
290}
291
292inline std::unique_ptr<CHttpRequest> HttpPostJson(const char *pUrl, const char *pJson)
293{
294 auto pResult = std::make_unique<CHttpRequest>(pUrl);
295 pResult->PostJson(pJson);
296 pResult->Timeout(CTimeout{4000, 15000, 500, 5});
297 return pResult;
298}
299
300void EscapeUrl(char *pBuf, int Size, const char *pStr);
301
302template<int N>
303void EscapeUrl(char (&aBuf)[N], const char *pStr)
304{
305 EscapeUrl(aBuf, N, pStr);
306}
307
309
310// In an ideal world this would be a kernel interface
311class CHttp : public IHttp
312{
319
320 void *m_pThread = nullptr;
321
322 std::mutex m_Lock;
323 std::condition_variable m_Cv;
324 std::atomic<EState> m_State = UNINITIALIZED;
325 std::deque<std::shared_ptr<CHttpRequest>> m_PendingRequests;
326 std::unordered_map<void *, std::shared_ptr<CHttpRequest>> m_RunningRequests; // void * == CURL *
327 std::chrono::milliseconds m_ShutdownDelay{};
328 std::optional<std::chrono::time_point<std::chrono::steady_clock>> m_ShutdownTime;
329 std::atomic<bool> m_Shutdown = false;
330
331 // Only to be used with curl_multi_wakeup
332 void *m_pMultiH = nullptr; // void * == CURLM *
333
334 static void ThreadMain(void *pUser);
335 void RunLoop();
336
337public:
338 // Startup
339 bool Init(std::chrono::milliseconds ShutdownDelay);
340
341 // User
342 void Run(std::shared_ptr<IHttpRequest> pRequest) override;
343 void Shutdown() override;
344 ~CHttp() override;
345};
346
347#endif // ENGINE_SHARED_HTTP_H
void FailOnErrorStatus(bool FailOnErrorStatus)
Definition http.h:171
void WriteToFileAndMemory(IStorage *pStorage, const char *pDest, int StorageType)
Definition http.cpp:520
bool m_SkipByFileTime
Definition http.h:92
static size_t HeaderCallback(char *pData, size_t Size, size_t Number, void *pUser)
Definition http.cpp:348
const SHA256_DIGEST & ResultSha256() const
Definition http.cpp:556
int64_t m_MaxResponseSize
Definition http.h:95
static int ProgressCallback(void *pUser, double DlTotal, double DlCurr, double UlTotal, double UlCurr)
Definition http.cpp:359
REQUEST
Definition http.h:62
@ POST_JSON
Definition http.h:66
@ GET
Definition http.h:63
@ POST
Definition http.h:65
@ HEAD
Definition http.h:64
bool ConfigureHandle(void *pHandle)
Definition http.cpp:146
HTTPLOG m_LogProgress
Definition http.h:122
void * m_pHeaders
Definition http.h:87
bool m_HeadersEnded
Definition http.h:134
const char * Dest()
Definition http.h:215
char m_aUrl[256]
Definition http.h:85
std::mutex m_WaitMutex
Definition http.h:130
void Result(unsigned char **ppResult, size_t *pResultLength) const
Definition http.cpp:540
virtual ~CHttpRequest()
Definition http.cpp:69
char m_aDestAbsolute[IO_MAX_PATH_LENGTH]
Definition http.h:116
std::atomic< int > m_Progress
Definition http.h:121
void IpResolve(IPRESOLVE IpResolve)
Definition http.h:170
void Abort()
Definition http.h:236
std::optional< int64_t > m_ResultLastModified
Definition http.h:136
REQUEST m_Type
Definition http.h:97
void Header(const char *pNameColonValue)
Definition http.cpp:526
static constexpr const char * GetRequestType(REQUEST Type)
Definition http.h:69
void Wait()
Definition http.cpp:531
json_value * ResultJson() const
Definition http.cpp:548
void MaxResponseSize(int64_t MaxResponseSize)
Definition http.h:168
size_t m_BodyLength
Definition http.h:89
double Size() const
Definition http.h:228
IPRESOLVE m_IpResolve
Definition http.h:123
void WriteToMemory()
Definition http.h:173
std::optional< int64_t > ResultAgeSeconds() const
Definition http.cpp:569
unsigned char * m_pBuffer
Definition http.h:110
bool m_WriteToMemory
Definition http.h:103
void OnCompletionInternal(void *pHandle, unsigned int Result)
Definition http.cpp:369
int m_StorageType
Definition http.h:114
std::optional< SHA256_DIGEST > m_ActualSha256
Definition http.h:99
void WriteToFile(IStorage *pStorage, const char *pDest, int StorageType)
Definition http.cpp:503
void Timeout(CTimeout Timeout)
Definition http.h:165
std::atomic< double > m_Size
Definition http.h:119
int StatusCode() const
Definition http.cpp:563
EHttpState State() const
Definition http.h:230
bool ShouldSkipRequest()
Definition http.cpp:103
virtual void OnCompletion(EHttpState State)
Definition http.h:159
void Head()
Definition http.h:186
void ExpectSha256(const SHA256_DIGEST &Sha256)
Definition http.h:185
std::atomic< double > m_Current
Definition http.h:120
std::optional< int64_t > ResultLastModified() const
Definition http.cpp:579
int Progress() const
Definition http.h:229
size_t OnData(char *pData, size_t DataSize)
Definition http.cpp:308
static size_t WriteCallback(char *pData, size_t Size, size_t Number, void *pUser)
Definition http.cpp:354
std::optional< int64_t > m_ResultDate
Definition http.h:135
size_t m_BufferSize
Definition http.h:109
void HeaderInt(const char *pName, int Value)
Definition http.h:208
virtual void OnProgress()
Definition http.h:158
void Post(const unsigned char *pData, size_t DataLength)
Definition http.h:187
CTimeout m_Timeout
Definition http.h:94
bool m_WriteToFile
Definition http.h:104
uint64_t m_ResponseLength
Definition http.h:106
char m_aDestAbsoluteTmp[IO_MAX_PATH_LENGTH]
Definition http.h:115
void ValidateBeforeOverwrite(bool ValidateBeforeOverwrite)
Definition http.h:184
friend class CHttp
Definition http.h:59
int m_StatusCode
Definition http.h:133
IOHANDLE m_File
Definition http.h:113
unsigned char * m_pBody
Definition http.h:88
void OnValidation(bool Success)
Definition http.cpp:478
std::atomic< EHttpState > m_State
Definition http.h:128
std::atomic< bool > m_Abort
Definition http.h:129
std::condition_variable m_WaitCondition
Definition http.h:131
char m_aDest[IO_MAX_PATH_LENGTH]
Definition http.h:117
int64_t m_IfModifiedSince
Definition http.h:96
SHA256_CTX m_ActualSha256Ctx
Definition http.h:100
bool BeforeInit()
Definition http.cpp:117
CHttpRequest(const char *pUrl)
Definition http.cpp:63
void HeaderString(const char *pName, const char *pValue)
Definition http.h:202
double Current() const
Definition http.h:227
bool m_ValidateBeforeOverwrite
Definition http.h:91
char m_aErr[256]
Definition http.h:127
std::optional< SHA256_DIGEST > m_ExpectedSha256
Definition http.h:101
bool Done() const
Definition http.h:231
void PostJson(const char *pJson)
Definition http.h:194
void SkipByFileTime(bool SkipByFileTime)
Definition http.h:167
void LogProgress(HTTPLOG LogProgress)
Definition http.h:169
bool m_FailOnErrorStatus
Definition http.h:125
size_t OnHeader(char *pHeader, size_t HeaderSize)
Definition http.cpp:262
Definition http.h:312
~CHttp() override
Definition http.cpp:824
std::atomic< bool > m_Shutdown
Definition http.h:329
void Run(std::shared_ptr< IHttpRequest > pRequest) override
Definition http.cpp:799
std::chrono::milliseconds m_ShutdownDelay
Definition http.h:327
std::atomic< EState > m_State
Definition http.h:324
std::unordered_map< void *, std::shared_ptr< CHttpRequest > > m_RunningRequests
Definition http.h:326
EState
Definition http.h:314
@ ERROR
Definition http.h:317
@ UNINITIALIZED
Definition http.h:315
@ RUNNING
Definition http.h:316
void Shutdown() override
Definition http.cpp:814
std::optional< std::chrono::time_point< std::chrono::steady_clock > > m_ShutdownTime
Definition http.h:328
std::condition_variable m_Cv
Definition http.h:323
std::mutex m_Lock
Definition http.h:322
std::deque< std::shared_ptr< CHttpRequest > > m_PendingRequests
Definition http.h:325
static void ThreadMain(void *pUser)
Definition http.cpp:606
void * m_pMultiH
Definition http.h:332
void RunLoop()
Definition http.cpp:612
bool Init(std::chrono::milliseconds ShutdownDelay)
Definition http.cpp:585
void * m_pThread
Definition http.h:320
Definition http.h:49
long m_TimeoutMs
Definition http.h:52
long m_ConnectTimeoutMs
Definition http.h:51
long m_LowSpeedLimit
Definition http.h:53
long m_LowSpeedTime
Definition http.h:54
Definition http.h:9
Definition http.h:13
Definition storage.h:21
struct _json_value json_value
Definition serverbrowser.h:18
#define dbg_assert_failed(fmt,...)
Definition dbg.h:47
void mem_copy(void *dest, const void *source, size_t size)
Definition mem.cpp:6
int str_length(const char *str)
Definition str.cpp:52
void * IOHANDLE
Definition logger.h:12
std::unique_ptr< CHttpRequest > HttpGet(const char *pUrl)
Definition http.h:263
std::unique_ptr< CHttpRequest > HttpHead(const char *pUrl)
Definition http.h:256
std::unique_ptr< CHttpRequest > HttpPost(const char *pUrl, const unsigned char *pData, size_t DataLength)
Definition http.h:284
IPRESOLVE
Definition http.h:42
@ WHATEVER
Definition http.h:43
@ V4
Definition http.h:44
@ V6
Definition http.h:45
std::unique_ptr< CHttpRequest > HttpGetFile(const char *pUrl, IStorage *pStorage, const char *pOutputFile, int StorageType)
Definition http.h:268
void EscapeUrl(char *pBuf, int Size, const char *pStr)
Definition http.cpp:49
std::unique_ptr< CHttpRequest > HttpGetBoth(const char *pUrl, IStorage *pStorage, const char *pOutputFile, int StorageType)
Definition http.h:276
std::unique_ptr< CHttpRequest > HttpPostJson(const char *pUrl, const char *pJson)
Definition http.h:292
EHttpState
Definition http.h:26
@ DONE
Definition http.h:30
@ RUNNING
Definition http.h:29
@ ABORTED
Definition http.h:31
@ QUEUED
Definition http.h:28
@ ERROR
Definition http.h:27
bool HttpHasIpresolveBug()
Definition http.cpp:56
HTTPLOG
Definition http.h:35
@ FAILURE
Definition http.h:37
@ ALL
Definition http.h:38
@ NONE
Definition http.h:36
#define str_format
Definition str.cpp:89
Definition hash.h:15
constexpr auto IO_MAX_PATH_LENGTH
Definition types.h:51