NetManager.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595
  1. #include "NetManager.h"
  2. #include "DebugManager.h"
  3. #include "Compiler/BfSystem.h"
  4. #include "BeefySysLib/util/AllocDebug.h"
  5. USING_NS_BF;
  6. #ifdef BF_CURL
  7. #define CURL_STATICLIB
  8. #include "curl/curl.h"
  9. #include "curl/multi.h"
  10. static int TransferInfoCallback(void* userp,
  11. curl_off_t dltotal, curl_off_t dlnow,
  12. curl_off_t ultotal, curl_off_t ulnow)
  13. {
  14. NetRequest* netRequest = (NetRequest*)userp;
  15. if (netRequest->mCancelling)
  16. return 1;
  17. return 0;
  18. }
  19. static size_t WriteMemoryCallback(void* contents, size_t size, size_t nmemb, void* userp)
  20. {
  21. NetRequest* netRequest = (NetRequest*)userp;
  22. long response_code = 0;
  23. curl_easy_getinfo(netRequest->mCURL, CURLINFO_RESPONSE_CODE, &response_code);
  24. if (netRequest->mCancelling)
  25. {
  26. netRequest->mFailed = true;
  27. return 0;
  28. }
  29. if (response_code != 200)
  30. return 0;
  31. if (netRequest->mFailed)
  32. return 0;
  33. if (!netRequest->mOutFile.IsOpen())
  34. {
  35. RecursiveCreateDirectory(GetFileDir(netRequest->mOutPath));
  36. if (!netRequest->mOutFile.Open(netRequest->mOutTempPath, "wb"))
  37. {
  38. netRequest->Fail(StrFormat("Failed to create file '%s'", netRequest->mOutTempPath.c_str()));
  39. return 0;
  40. }
  41. }
  42. uint32 tickNow = BFTickCount();
  43. if (tickNow - netRequest->mLastUpdateTick >= 500)
  44. {
  45. curl_off_t downloadSize = 0;
  46. curl_easy_getinfo(netRequest->mCURL, CURLINFO_SIZE_DOWNLOAD_T, &downloadSize);
  47. curl_off_t length = 0;
  48. curl_easy_getinfo(netRequest->mCURL, CURLINFO_CONTENT_LENGTH_DOWNLOAD_T, &length);
  49. if (netRequest->mShowTracking)
  50. {
  51. String msg = StrFormat("symsrv Getting '%s' - %dk", netRequest->mURL.c_str(), (int)(downloadSize / 1024));
  52. if (length > 0)
  53. {
  54. msg += StrFormat(" (%d%%)", (int)(downloadSize * 100 / length));
  55. }
  56. netRequest->mNetManager->mDebugManager->OutputRawMessage(msg);
  57. }
  58. netRequest->mLastUpdateTick = tickNow;
  59. }
  60. size_t realsize = size * nmemb;
  61. netRequest->mOutFile.Write(contents, (int)realsize);
  62. return realsize;
  63. }
  64. void NetRequest::Cleanup()
  65. {
  66. if (mCURLMulti != NULL)
  67. {
  68. curl_multi_remove_handle(mCURLMulti, mCURL);
  69. }
  70. if (mCURL != NULL)
  71. curl_easy_cleanup(mCURL);
  72. if (mCURLMulti != NULL)
  73. {
  74. curl_multi_cleanup(mCURLMulti);
  75. }
  76. }
  77. void NetRequest::Perform()
  78. {
  79. if (mCancelling)
  80. return;
  81. // {
  82. // mFailed = true;
  83. // return;
  84. // }
  85. BfLogDbg("NetManager starting get on %s\n", mURL.c_str());
  86. mNetManager->mDebugManager->OutputRawMessage(StrFormat("msgLo Getting '%s'\n", mURL.c_str()));
  87. mOutTempPath = mOutPath + "__partial";
  88. mCURLMulti = curl_multi_init();
  89. mCURL = curl_easy_init();
  90. if (mShowTracking)
  91. {
  92. mNetManager->mDebugManager->OutputRawMessage(StrFormat("symsrv Getting '%s'", mURL.c_str()));
  93. }
  94. //OutputDebugStrF("Getting '%s'\n", mURL.c_str());
  95. curl_easy_setopt(mCURL, CURLOPT_URL, mURL.c_str());
  96. curl_easy_setopt(mCURL, CURLOPT_WRITEDATA, (void*)this);
  97. curl_easy_setopt(mCURL, CURLOPT_WRITEFUNCTION, WriteMemoryCallback);
  98. curl_easy_setopt(mCURL, CURLOPT_XFERINFODATA, (void*)this);
  99. curl_easy_setopt(mCURL, CURLOPT_XFERINFOFUNCTION, TransferInfoCallback);
  100. curl_easy_setopt(mCURL, CURLOPT_FOLLOWLOCATION, 1L);
  101. curl_easy_setopt(mCURL, CURLOPT_NOPROGRESS, 0L);
  102. //auto result = curl_easy_perform(mCURL);
  103. CURLMcode mcode = curl_multi_add_handle(mCURLMulti, mCURL);
  104. if (mcode != CURLM_OK)
  105. {
  106. mFailed = true;
  107. return;
  108. }
  109. while (true)
  110. {
  111. int activeCount = 0;
  112. curl_multi_perform(mCURLMulti, &activeCount);
  113. if (activeCount == 0)
  114. break;
  115. int waitRet = 0;
  116. curl_multi_wait(mCURLMulti, NULL, 0, 20, &waitRet);
  117. if (mCancelling)
  118. {
  119. mFailed = true;
  120. return;
  121. }
  122. }
  123. // if (result != CURLE_OK)
  124. // {
  125. // mFailed = true;
  126. // return;
  127. // }
  128. long response_code = 0;
  129. curl_easy_getinfo(mCURL, CURLINFO_RESPONSE_CODE, &response_code);
  130. mNetManager->mDebugManager->OutputRawMessage(StrFormat("msgLo Result for '%s': %d\n", mURL.c_str(), response_code));
  131. if (response_code != 200)
  132. {
  133. mOutFile.Close();
  134. // Bad result
  135. mFailed = true;
  136. return;
  137. }
  138. BfLogDbg("NetManager successfully completed %s\n", mURL.c_str());
  139. if (mCancelOnSuccess != NULL)
  140. mNetManager->Cancel(mCancelOnSuccess);
  141. if (!mOutFile.IsOpen())
  142. {
  143. mFailed = true;
  144. return; // No data
  145. }
  146. mOutFile.Close();
  147. BfpFile_Delete(mOutPath.c_str(), NULL);
  148. BfpFileResult renameResult;
  149. BfpFile_Rename(mOutTempPath.c_str(), mOutPath.c_str(), &renameResult);
  150. if (renameResult != BfpFileResult_Ok)
  151. {
  152. mFailed = true;
  153. }
  154. }
  155. #else
  156. #include <windows.h>
  157. #include <wininet.h>
  158. #include <stdio.h>
  159. #pragma comment (lib, "wininet.lib")
  160. void NetRequest::Perform()
  161. {
  162. if (mCancelling)
  163. return;
  164. // {
  165. // mFailed = true;
  166. // return;
  167. // }
  168. BfLogDbg("NetManager starting get on %s\n", mURL.c_str());
  169. String protoName;
  170. String serverName;
  171. String objectName;
  172. int colonPos = (int)mURL.IndexOf("://");
  173. if (colonPos == -1)
  174. {
  175. Fail("Invalid URL");
  176. return;
  177. }
  178. protoName = mURL.Substring(0, colonPos);
  179. serverName = mURL.Substring(colonPos + 3);
  180. int slashPos = (int)serverName.IndexOf('/');
  181. if (slashPos != -1)
  182. {
  183. objectName = serverName.Substring(slashPos);
  184. serverName.RemoveToEnd(slashPos);
  185. }
  186. mOutTempPath = mOutPath + "__partial";
  187. HINTERNET hSession = NULL;
  188. HINTERNET hConnect = NULL;
  189. HINTERNET hHttpFile = NULL;
  190. defer
  191. {
  192. if (hHttpFile != NULL)
  193. InternetCloseHandle(hHttpFile);
  194. if (hConnect != NULL)
  195. InternetCloseHandle(hConnect);
  196. if (hSession != NULL)
  197. InternetCloseHandle(hSession);
  198. if (mOutFile.IsOpen())
  199. mOutFile.Close();
  200. };
  201. bool isHttp = protoName.Equals("http", StringImpl::CompareKind_OrdinalIgnoreCase);
  202. bool isHttps = protoName.Equals("https", StringImpl::CompareKind_OrdinalIgnoreCase);
  203. if ((!isHttp) && (!isHttps))
  204. {
  205. Fail("Invalid URL protocol");
  206. return;
  207. }
  208. hSession = InternetOpenW(
  209. L"Mozilla/5.0",
  210. INTERNET_OPEN_TYPE_PRECONFIG,
  211. NULL,
  212. NULL,
  213. 0);
  214. hConnect = InternetConnectW(
  215. hSession,
  216. UTF8Decode(serverName).c_str(),
  217. isHttps ? INTERNET_DEFAULT_HTTPS_PORT : INTERNET_DEFAULT_HTTP_PORT,
  218. NULL,
  219. NULL,
  220. INTERNET_SERVICE_HTTP,
  221. 0,
  222. 0);
  223. DWORD httpOpenFlags = INTERNET_FLAG_RELOAD;
  224. if (isHttps)
  225. httpOpenFlags |= INTERNET_FLAG_SECURE;
  226. hHttpFile = HttpOpenRequestW(
  227. hConnect,
  228. L"GET",
  229. UTF8Decode(objectName).c_str(),
  230. NULL,
  231. NULL,
  232. NULL,
  233. httpOpenFlags,
  234. 0);
  235. if (!HttpSendRequestW(hHttpFile, NULL, 0, 0, 0))
  236. {
  237. int err = GetLastError();
  238. OutputDebugStrF("Failed: %s %X\n", mURL.c_str(), err);;
  239. Fail("Failed to send request");
  240. return;
  241. }
  242. DWORD statusCode = 0;
  243. DWORD length = sizeof(DWORD);
  244. HttpQueryInfo(hHttpFile, HTTP_QUERY_STATUS_CODE | HTTP_QUERY_FLAG_NUMBER, &statusCode, &length, NULL);
  245. if (statusCode != 200)
  246. {
  247. Fail("Invalid HTTP response");
  248. return;
  249. }
  250. if (mShowTracking)
  251. {
  252. mNetManager->mDebugManager->OutputRawMessage(StrFormat("symsrv Getting '%s'", mURL.c_str()));
  253. }
  254. uint8 buffer[4096];
  255. while (true)
  256. {
  257. DWORD dwBytesRead = 0;
  258. BOOL bRead = InternetReadFile(hHttpFile, buffer, 4096, &dwBytesRead);
  259. if (dwBytesRead == 0)
  260. break;
  261. if (!bRead)
  262. {
  263. //printf("InternetReadFile error : <%lu>\n", GetLastError());
  264. Fail("Failed to receive");
  265. return;
  266. }
  267. if (!mOutFile.IsOpen())
  268. {
  269. RecursiveCreateDirectory(GetFileDir(mOutPath));
  270. if (!mOutFile.Open(mOutTempPath, "wb"))
  271. {
  272. Fail(StrFormat("Failed to create file '%s'", mOutTempPath.c_str()));
  273. return;
  274. }
  275. }
  276. mOutFile.Write(buffer, (int)dwBytesRead);
  277. }
  278. BfLogDbg("NetManager successfully completed %s\n", mURL.c_str());
  279. if (mCancelOnSuccess != NULL)
  280. mNetManager->Cancel(mCancelOnSuccess);
  281. if (!mOutFile.IsOpen())
  282. {
  283. mFailed = true;
  284. return; // No data
  285. }
  286. mOutFile.Close();
  287. BfpFile_Delete(mOutPath.c_str(), NULL);
  288. BfpFileResult renameResult;
  289. BfpFile_Rename(mOutTempPath.c_str(), mOutPath.c_str(), &renameResult);
  290. if (renameResult != BfpFileResult_Ok)
  291. mFailed = true;
  292. }
  293. void NetRequest::Cleanup()
  294. {
  295. }
  296. #endif
  297. NetRequest::~NetRequest()
  298. {
  299. Cleanup();
  300. // This is synchronized
  301. if (mResult != NULL)
  302. {
  303. mResult->mFailed = mFailed;
  304. mResult->mCurRequest = NULL;
  305. if (mResult->mRemoved)
  306. delete mResult;
  307. }
  308. mNetManager->mRequestDoneEvent.Set();
  309. }
  310. void NetRequest::Fail(const StringImpl& error)
  311. {
  312. if (mFailed)
  313. return;
  314. mError = error;
  315. mFailed = true;
  316. }
  317. bool NetRequest::Cancel()
  318. {
  319. mCancelling = true;
  320. return true;
  321. }
  322. void NetRequest::ShowTracking()
  323. {
  324. //mNetManager->mDebugManager->OutputMessage(StrFormat("Getting '%s'\n", mURL.c_str()));
  325. mNetManager->mDebugManager->OutputRawMessage(StrFormat("symsrv Getting '%s'", mURL.c_str()));
  326. mShowTracking = true;
  327. }
  328. void NetManagerThread()
  329. {
  330. }
  331. NetManager::NetManager() : mThreadPool(8, 1*1024*1024)
  332. {
  333. mWaitingResult = NULL;
  334. mWaitingRequest = NULL;
  335. }
  336. NetManager::~NetManager()
  337. {
  338. mThreadPool.Shutdown();
  339. for (auto kv : mCachedResults)
  340. {
  341. delete kv.mValue;
  342. }
  343. }
  344. NetRequest* NetManager::CreateGetRequest(const StringImpl& url, const StringImpl& destPath)
  345. {
  346. AutoCrit autoCrit(mThreadPool.mCritSect);
  347. NetRequest* netRequest = new NetRequest();
  348. netRequest->mNetManager = this;
  349. netRequest->mURL = url;
  350. netRequest->mOutPath = destPath;
  351. NetResult* netResult = new NetResult();
  352. netResult->mURL = url;
  353. netResult->mOutPath = destPath;
  354. netResult->mFailed = false;
  355. netResult->mCurRequest = netRequest;
  356. NetResult** netResultPtr;
  357. if (mCachedResults.TryAdd(url, NULL, &netResultPtr))
  358. {
  359. *netResultPtr = netResult;
  360. }
  361. else
  362. {
  363. mOldResults.Add(*netResultPtr);
  364. *netResultPtr = netResult;
  365. }
  366. netRequest->mResult = netResult;
  367. return netRequest;
  368. }
  369. NetResult* NetManager::QueueGet(const StringImpl& url, const StringImpl& destPath)
  370. {
  371. BfLogDbg("NetManager queueing %s\n", url.c_str());
  372. auto netRequest = CreateGetRequest(url, destPath);
  373. auto netResult = netRequest->mResult;
  374. mThreadPool.AddJob(netRequest);
  375. return netResult;
  376. }
  377. bool NetManager::Get(const StringImpl& url, const StringImpl& destPath)
  378. {
  379. NetRequest* netRequest = NULL;
  380. int waitCount = 0;
  381. while (true)
  382. {
  383. // Check cached
  384. {
  385. AutoCrit autoCrit(mThreadPool.mCritSect);
  386. mWaitingResult = NULL;
  387. NetResult* netResult;
  388. if (mCachedResults.TryGetValue(url, &netResult))
  389. {
  390. if (netResult->mCurRequest == NULL)
  391. {
  392. BfLogDbg("NetManager::Get using cached result for %s: %d. WaitCount: %d \n", url.c_str(), !netResult->mFailed, waitCount);
  393. return (!netResult->mFailed) && (FileExists(netResult->mOutPath));
  394. }
  395. else if (!netResult->mCurRequest->mShowTracking) // Is done?
  396. {
  397. if (!netResult->mCurRequest->mProcessing)
  398. {
  399. BfLogDbg("NetManager::Get pulling queued request into current thread %s\n", url.c_str());
  400. netRequest = netResult->mCurRequest;
  401. bool didRemove = mThreadPool.mJobs.Remove(netRequest);
  402. BF_ASSERT(didRemove);
  403. break;
  404. }
  405. mWaitingResult = netResult;
  406. netResult->mCurRequest->ShowTracking();
  407. }
  408. }
  409. else
  410. {
  411. netRequest = CreateGetRequest(url, destPath);
  412. break;
  413. }
  414. }
  415. waitCount++;
  416. mRequestDoneEvent.WaitFor();
  417. }
  418. // Perform this in the requesting thread
  419. {
  420. AutoCrit autoCrit(mThreadPool.mCritSect);
  421. mWaitingRequest = netRequest;
  422. }
  423. netRequest->mShowTracking = true;
  424. netRequest->Perform();
  425. AutoCrit autoCrit(mThreadPool.mCritSect);
  426. mWaitingRequest = NULL;
  427. auto netResult = netRequest->mResult;
  428. delete netRequest;
  429. BfLogDbg("NetManager::Get requested %s: %d\n", url.c_str(), !netResult->mFailed);
  430. mDebugManager->OutputRawMessage(StrFormat("msgLo Result for '%s': %d\n", url.c_str(), !netResult->mFailed));
  431. return (!netResult->mFailed) && (FileExists(netResult->mOutPath));
  432. }
  433. void NetManager::CancelAll()
  434. {
  435. AutoCrit autoCrit(mThreadPool.mCritSect);
  436. if (mWaitingRequest != NULL)
  437. mWaitingRequest->Cancel();
  438. mThreadPool.CancelAll();
  439. }
  440. void NetManager::Clear()
  441. {
  442. AutoCrit autoCrit(mThreadPool.mCritSect);
  443. BF_ASSERT(mWaitingResult == NULL); // The debugger thread shouldn't be waiting on anything, it should be detached at this point
  444. for (auto job : mThreadPool.mJobs)
  445. {
  446. auto netRequest = (NetRequest*)job;
  447. netRequest->Cancel();
  448. }
  449. for (auto kv : mCachedResults)
  450. {
  451. NetResult* netResult = kv.mValue;
  452. if (netResult->mCurRequest != NULL)
  453. netResult->mRemoved = true;
  454. else
  455. delete netResult;
  456. }
  457. mCachedResults.Clear();
  458. for (auto netResult : mOldResults)
  459. {
  460. delete netResult;
  461. }
  462. mOldResults.Clear();
  463. }
  464. void NetManager::CancelCurrent()
  465. {
  466. AutoCrit autoCrit(mThreadPool.mCritSect);
  467. if (mWaitingRequest != NULL)
  468. mWaitingRequest->Cancel();
  469. else if ((mWaitingResult != NULL) && (mWaitingResult->mCurRequest != NULL))
  470. mWaitingResult->mCurRequest->Cancel();
  471. }
  472. void NetManager::Cancel(NetResult* netResult)
  473. {
  474. AutoCrit autoCrit(mThreadPool.mCritSect);
  475. BfLogDbg("NetManager cancel %s\n", netResult->mURL.c_str());
  476. if (netResult->mCurRequest != NULL)
  477. netResult->mCurRequest->Cancel();
  478. }
  479. void NetManager::SetCancelOnSuccess(NetResult* dependentResult, NetResult* cancelOnSucess)
  480. {
  481. AutoCrit autoCrit(mThreadPool.mCritSect);
  482. if (dependentResult->mCurRequest != NULL)
  483. {
  484. dependentResult->mCurRequest->mCancelOnSuccess = cancelOnSucess;
  485. }
  486. }