Project
Loading...
Searching...
No Matches
CCDBDownloader.h
Go to the documentation of this file.
1// Copyright 2019-2023 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11#ifndef O2_CCDBDOWNLOADER_H_
12#define O2_CCDBDOWNLOADER_H_
13
14#if !defined(__CINT__) && !defined(__MAKECINT__) && !defined(__ROOTCLING__) && !defined(__CLING__)
16#endif
17
18#include <cstdio>
19#include <cstdlib>
20#include <curl/curl.h>
21#include <string>
22#include <vector>
23#include <iostream>
24#include <thread>
25#include <mutex>
26#include <condition_variable>
27#include <unordered_map>
28#include <map>
29#include <functional>
30
31typedef struct uv_loop_s uv_loop_t;
32typedef struct uv_timer_s uv_timer_t;
33typedef struct uv_poll_s uv_poll_t;
34typedef struct uv_signal_s uv_signal_t;
35typedef struct uv_async_s uv_async_t;
36typedef struct uv_handle_s uv_handle_t;
37
38namespace o2::ccdb
39{
40
41#if !defined(__CINT__) && !defined(__MAKECINT__) && !defined(__ROOTCLING__) && !defined(__CLING__)
43 std::multimap<std::string, std::string> header;
44 o2::pmr::vector<char>* object = nullptr;
45 int counter = 0;
46};
47
48typedef struct DownloaderRequestData {
49 std::vector<std::string> hosts;
50 std::vector<std::string> locations;
51 std::string path;
54 std::map<std::string, std::string>* headers;
55 std::string userAgent;
56 curl_slist* optionsList;
57
58 std::function<bool(std::string)> localContentCallback;
59} DownloaderRequestData;
60#endif
61
62/*
63 Some functions below aren't member functions of CCDBDownloader because both curl and libuv require callback functions which have to be either static or non-member.
64 Because non-static functions are used in the functions below, they must be non-member.
65*/
66
73void closeHandles(uv_handle_t* handle, void* arg);
74
82curl_socket_t opensocketCallback(void* clientp, curlsocktype purpose, struct curl_sockaddr* address);
83
89void onUVClose(uv_handle_t* handle);
90
95
104{
105 public:
110 std::unordered_map<curl_socket_t, uv_timer_t*> mSocketTimerMap;
111
116
120 std::unordered_map<uv_handle_t*, bool> mHandleMap;
121
126
131
135 int mRequestTimeoutMS = 300000;
136
141
146
151
152 CCDBDownloader(uv_loop_t* uv_loop = nullptr);
154
160 CURLcode perform(CURL* handle);
161
166 std::vector<CURLcode> batchBlockingPerform(std::vector<CURL*> const& handleVector);
167
174 void asynchSchedule(CURL* handle, size_t* requestCounter);
175
179 void setMaxParallelConnections(int limit);
180
184 void setKeepaliveTimeoutTime(int timeoutMS);
185
189 void setConnectionTimeoutTime(int timeoutMS);
190
194 void setRequestTimeoutTime(int timeoutMS);
195
199 void setHappyEyeballsHeadstartTime(int headstartMS);
200
205
210
216 void runLoop(bool noWait);
217
221 std::string prepareLogMessage(std::string host_url, std::string userAgent, const std::string& path, long ts, const std::map<std::string, std::string>* headers, long httpCode) const;
222
226 std::string trimHostUrl(std::string full_host_url) const;
227
228 private:
232 std::string prepareRedirectedURL(std::string address, std::string potentialHost) const;
233
241 void updateLocations(std::multimap<std::string, std::string>* headerMap, std::vector<std::string>* locations, int* locIndex) const;
242
243 std::string mUserAgentId = "CCDBDownloader";
247 void setupInternalUVLoop();
248
252 int mHandlesInUse = 0;
253
257 CURLM* mCurlMultiHandle = nullptr;
258
262 uv_timer_t* mTimeoutTimer;
263
267 std::vector<CURL*> mHandlesToBeAdded;
268
272 enum RequestType {
273 BLOCKING,
274 ASYNCHRONOUS
275 };
276
280 typedef struct curl_context_s {
281 uv_poll_t* poll_handle;
282 curl_socket_t sockfd = -1;
283 CCDBDownloader* CD = nullptr;
284 } curl_context_t;
285
289 typedef struct DataForSocket {
290 CCDBDownloader* CD;
291 CURLM* curlm;
292 } DataForSocket;
293
294 DataForSocket mSocketData;
295
296#if !defined(__CINT__) && !defined(__MAKECINT__) && !defined(__ROOTCLING__) && !defined(__CLING__)
300 typedef struct PerformData {
301 CURLcode* codeDestination;
302 size_t* requestsLeft;
303 RequestType type;
304 int hostInd;
305 int locInd;
306 DownloaderRequestData* requestData;
307 curl_slist** options;
308 } PerformData;
309#endif
310
317 static void closesocketCallback(void* clientp, curl_socket_t item);
318
319#if !defined(__CINT__) && !defined(__MAKECINT__) && !defined(__ROOTCLING__) && !defined(__CLING__)
320 // Returns a new location string or an empty string if all locations under current host have been accessedd
321 std::string getNewLocation(PerformData* performData, std::vector<std::string>& locations) const;
322
323 // Reschedules the transfer to be performed with a different host.
324 void tryNewHost(PerformData* performData, CURL* easy_handle);
325
326 // Retrieves content from either alien, cvmfs or local storage using a callback to CCDBApi.
327 void getLocalContent(PerformData* performData, std::string& newLocation, bool& contentRetrieved, std::vector<std::string>& locations);
328
329 // Continues a transfer via a http redirect.
330 void httpRedirect(PerformData* performData, std::string& newLocation, CURL* easy_handle);
331
332 // Continues a transfer via a redirect. The redirect can point to a local file, alien file or a http address.
333 void followRedirect(PerformData* performData, CURL* easy_handle, std::vector<std::string>& locations, bool& rescheduled, bool& contentRetrieved);
334#endif
335
343 static void curlPerform(uv_poll_t* handle, int status, int events);
344
348 static int handleSocket(CURL* easy, curl_socket_t s, int action, void* userp, void* socketp);
349
355 static void closeSocketByTimer(uv_timer_t* handle);
356
362 static void curlTimeout(uv_timer_t* req);
363
369 static void curlCloseCB(uv_handle_t* handle);
370
376 static void destroyCurlContext(curl_context_t* context);
377
385 static int startTimeout(CURLM* multi, long timeout_ms, void* userp);
386
390 void initializeMultiHandle();
391
398 void transferFinished(CURL* handle, CURLcode curlCode);
399
403 void checkMultiInfo();
404
405#if !defined(__CINT__) && !defined(__MAKECINT__) && !defined(__ROOTCLING__) && !defined(__CLING__)
409 void setHandleOptions(CURL* handle, PerformData* data);
410#endif
411
417 curl_context_t* createCurlContext(curl_socket_t sockfd);
418
422 void checkHandleQueue();
423};
424
428typedef struct DataForClosingSocket {
430 curl_socket_t socket;
431} DataForClosingSocket;
432
433} // namespace o2::ccdb
434
435#endif // O2_CCDB_CCDBDOWNLOADER_H
struct uv_timer_s uv_timer_t
struct uv_signal_s uv_signal_t
struct uv_async_s uv_async_t
struct uv_handle_s uv_handle_t
struct uv_poll_s uv_poll_t
struct uv_loop_s uv_loop_t
void setRequestTimeoutTime(int timeoutMS)
CURLcode perform(CURL *handle)
void asynchSchedule(CURL *handle, size_t *requestCounter)
std::string trimHostUrl(std::string full_host_url) const
void setKeepaliveTimeoutTime(int timeoutMS)
std::vector< CURLcode > batchBlockingPerform(std::vector< CURL * > const &handleVector)
void setConnectionTimeoutTime(int timeoutMS)
void setHappyEyeballsHeadstartTime(int headstartMS)
std::string prepareLogMessage(std::string host_url, std::string userAgent, const std::string &path, long ts, const std::map< std::string, std::string > *headers, long httpCode) const
std::unordered_map< uv_handle_t *, bool > mHandleMap
std::unordered_map< curl_socket_t, uv_timer_t * > mSocketTimerMap
void setMaxParallelConnections(int limit)
GLuint const GLint * locations
Definition glcorearb.h:5740
GLuint GLuint64EXT address
Definition glcorearb.h:5846
GLint GLint GLsizei GLint GLenum GLenum type
Definition glcorearb.h:275
GLboolean * data
Definition glcorearb.h:298
GLsizei const GLchar *const * path
Definition glcorearb.h:3591
GLuint counter
Definition glcorearb.h:3987
information complementary to a CCDB object (path, metadata, startTimeValidity, endTimeValidity etc)
void closeHandles(uv_handle_t *handle, void *arg)
curl_socket_t opensocketCallback(void *clientp, curlsocktype purpose, struct curl_sockaddr *address)
void onUVClose(uv_handle_t *handle)
std::vector< T, o2::pmr::polymorphic_allocator< T > > vector
std::function< bool(std::string)> localContentCallback
std::vector< std::string > locations
std::vector< std::string > hosts
std::map< std::string, std::string > * headers
std::multimap< std::string, std::string > header