alb_curl_helper.cpp
Go to the documentation of this file.
1 // Standard headers
2 #include <map>
3 #include <stdio.h>
4 
5 // ROS headers
6 #include "ros/ros.h"
7 
8 // Local headers
9 #include "alb_curl_helper.h"
10 
11 // Constant definition.
12 constexpr const char *k_http = "http://";
13 constexpr const char *k_api = "/api/v1/";
14 const std::map<AlbCurlHelper::processing_service_e, std::string> k_processing_commands = {
15  { AlbCurlHelper::processing_service_e::Restart, "processing/restart" },
16  { AlbCurlHelper::processing_service_e::Stop, "processing/stop" },
17  { AlbCurlHelper::processing_service_e::Status, "processing/status" }
18 };
19 constexpr const char *k_processing_config = "processing/config";
20 constexpr const char *k_processing_zones = "processing/zones";
21 constexpr const char *k_storage = "storage/";
22 
23 constexpr const char *k_header_accept = "accept: ";
24 constexpr const char *k_header_content = "Content-Type: ";
25 constexpr const char *k_header_app_json = "application/json";
26 constexpr const char *k_header_app_octet = "application/octet-stream";
27 
28 constexpr const char *k_display_config = "Display current ALB config";
29 
30 // Server response error code.
31 constexpr const int k_server_success = 200;
32 constexpr const int k_server_upload_success = 201;
33 
34 namespace
35 {
36 // Function to read the curl output.
37 static size_t write_callback(void *contents, size_t size, size_t nmemb, void *userp)
38 {
39  ((std::string *)userp)->append((char *)contents, size * nmemb);
40  return size * nmemb;
41 }
42 
43 // Function to write the curl output to a file.
44 static size_t write_to_file_callback(void *ptr, size_t size, size_t nmemb, FILE *stream)
45 {
46  size_t written = fwrite(ptr, size, nmemb, stream);
47  return written;
48 }
49 }; // namespace
50 
51 AlbCurlHelper::AlbCurlHelper(const std::string &ip_address) : curl(nullptr), chunk(NULL), file(nullptr)
52 {
53  // Define curl address.
54  if (!ip_address.empty()) {
55  curl_address = k_http + ip_address + k_api;
56  }
57 }
58 
60 {
61  // Clean up curl.
62  curl_easy_cleanup(curl);
63  curl_slist_free_all(chunk);
64 
65  if (file) {
66  fclose(file);
67  }
68 }
69 
70 const std::string AlbCurlHelper::getErrorMessage(void) const
71 {
72  return error_message;
73 }
74 
76 {
77  if (!initCurl()) {
78  error_message = std::string("[AlbCurlHelper] Invalid init");
79  return false;
80  }
81 
83 
84  // Define headers.
85  chunk = curl_slist_append(chunk, std::string(k_header_accept).append(k_header_app_json).c_str());
86  curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
87 
88  curl_easy_setopt(curl, CURLOPT_POST, 1L);
89  curl_easy_setopt(curl, CURLOPT_POSTFIELDS, "");
90 
91  return executeCurlCommand(k_processing_commands.at(service));
92 }
93 
95 {
96  if (!initCurl()) {
97  error_message = std::string("[AlbCurlHelper] Invalid init.");
98  return false;
99  }
100 
101  defineBaseCommand(k_processing_commands.at(AlbCurlHelper::processing_service_e::Status));
102 
103  // Define headers.
104  chunk = curl_slist_append(chunk, std::string(k_header_accept).append(k_header_app_json).c_str());
105  curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
106 
107  if (!executeCurlCommand(k_processing_commands.at(AlbCurlHelper::processing_service_e::Status))) {
108  error_message = std::string("[AlbCurlHelper] Unable to get processing status.");
109  return false;
110  }
111 
112  running = (callback_data.find("true") != std::string::npos);
113 
114  return true;
115 }
116 
118  outsight_alb_driver::AlbConfig::Request &request)
119 {
120  if (!initCurl()) {
121  error_message = std::string("[AlbCurlHelper] Invalid init");
122  return false;
123  }
124 
126 
127  if (!defineProcessingConfig(service, request)) {
128  return false;
129  }
130 
132  return false;
133  }
134 
135  if (service == processing_config_e::Get) {
136  if (!exportData(callback_data, request.filepath, k_display_config)) {
137  error_message = std::string("[AlbCurlHelper] Error exporting the config data.");
138  return false;
139  }
140  }
141 
142  return true;
143 }
144 
146 {
147  if (!initCurl()) {
148  error_message = std::string("[AlbCurlHelper] Invalid init");
149  return false;
150  }
151 
153 
154  // Define Processing Zones for curl request
155  chunk = curl_slist_append(chunk, std::string(k_header_accept).append(k_header_app_json).c_str());
156  curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
157 
159  return false;
160  }
161 
162  zones = callback_data;
163  return true;
164 }
165 
166 bool AlbCurlHelper::executeStorage(const storage_service_e service, outsight_alb_driver::AlbFile::Request &request)
167 {
168  if (!initCurl()) {
169  error_message = std::string("[AlbCurlHelper] Invalid init");
170  return false;
171  }
172 
173  bool define_command = false;
174  if (service == storage_service_e::List) {
175  define_command = defineStorageListCommand(request);
176  } else {
177  define_command = defineStorageCommand(service, request);
178  }
179 
180  if (!define_command) {
181  return false;
182  }
183 
185  return false;
186  }
187 
188  if (service == storage_service_e::List) {
189  exportData(callback_data, request.filepath,
190  std::string("List ") + request.category + std::string(" files on ALB:"));
191  }
192 
193  return true;
194 }
195 
197 {
198  if (curl_address.empty()) {
199  return false;
200  }
201 
202  curl = curl_easy_init();
203  return curl;
204 }
205 
206 void AlbCurlHelper::defineBaseCommand(const std::string &alb_command)
207 {
208  std::string full_address;
209  full_address = curl_address + alb_command;
210  callback_data.clear();
211 
212  // Define the URL for the command.
213  curl_easy_setopt(curl, CURLOPT_URL, full_address.c_str());
214 
215  // Define the callback function.
216  curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback);
217  curl_easy_setopt(curl, CURLOPT_WRITEDATA, &callback_data);
218 }
219 
221  outsight_alb_driver::AlbConfig::Request &request)
222 {
223  if (service == processing_config_e::Get) {
224  chunk = curl_slist_append(chunk, std::string(k_header_accept).append(k_header_app_json).c_str());
225  curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
226  } else if (service == processing_config_e::Put) {
227  chunk = curl_slist_append(chunk, std::string(k_header_accept).append(k_header_app_json).c_str());
228  chunk = curl_slist_append(chunk, std::string(k_header_content).append(k_header_app_json).c_str());
229  curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
230 
231  std::string post_data = readData(request.filepath);
232 
233  if (post_data.empty()) {
234  error_message = std::string("No data for configuration.");
235  return false;
236  }
237 
238  // Manually set the PUT command because CURLOPT_POSTFIELDS defines the POST one.
239  curl_easy_setopt(curl, CURLOPT_COPYPOSTFIELDS, post_data.c_str());
240  curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, "PUT");
241  }
242 
243  return true;
244 }
245 
247  outsight_alb_driver::AlbFile::Request &request)
248 {
249  std::string storage_command;
250  storage_command.append(k_storage);
251  storage_command.append(request.category);
252  storage_command.append("/");
253  storage_command.append(request.alb_filename);
254 
255  std::string full_address;
256  full_address = curl_address + storage_command;
257  callback_data.clear();
258 
259  // Define the URL for the command.
260  curl_easy_setopt(curl, CURLOPT_URL, full_address.c_str());
261 
262  // Define the callback function.
263  if (service == storage_service_e::Download) {
264  file = fopen(request.filepath.c_str(), "wb");
265 
266  if (!file) {
267  error_message = std::string("Unable to open the output filepath.");
268  return false;
269  }
270 
271  curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_to_file_callback);
272  curl_easy_setopt(curl, CURLOPT_WRITEDATA, file);
273  } else {
274  curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback);
275  curl_easy_setopt(curl, CURLOPT_WRITEDATA, &callback_data);
276 
277  chunk = curl_slist_append(chunk, std::string(k_header_accept).append(k_header_app_json).c_str());
278  chunk = curl_slist_append(chunk, std::string(k_header_content).append(k_header_app_octet).c_str());
279  curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
280 
281  file = fopen(request.filepath.c_str(), "rb");
282 
283  if (!file) {
284  error_message = std::string("Unable to open the output filepath.");
285  return false;
286  }
287 
288  curl_easy_setopt(curl, CURLOPT_READDATA, file);
289  curl_easy_setopt(curl, CURLOPT_UPLOAD, 1L);
290  }
291 
292  return true;
293 }
294 
295 bool AlbCurlHelper::executeCurlCommand(const std::string &command)
296 {
297  CURLcode curl_code;
298  curl_code = curl_easy_perform(curl);
299 
300  if (curl_code != CURLE_OK) {
301  error_message = "curl command failed: ";
302  error_message.append(curl_easy_strerror(curl_code));
303  return false;
304  }
305 
306  curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &server_response);
307 
309  error_message = "Server error ";
310  error_message.append(std::to_string(server_response));
311  error_message.append("\nResponse body: ");
313  return false;
314  }
315 
316  return true;
317 }
318 
319 bool AlbCurlHelper::exportData(const std::string &data, const std::string &output_path, const std::string &print_text)
320 {
321  if (output_path.empty()) {
322  ROS_INFO("%s\n%s", print_text.c_str(), data.c_str());
323  return true;
324  }
325 
326  FILE *fd;
327  fd = fopen(output_path.c_str(), "w");
328 
329  if (!fd) {
330  return false;
331  }
332 
333  fwrite(data.c_str(), sizeof(char), strlen(data.c_str()), fd);
334  fclose(fd);
335 
336  return true;
337 }
338 
339 const std::string AlbCurlHelper::readData(const std::string &filepath)
340 {
341  std::string data;
342  FILE *fd;
343  fd = fopen(filepath.c_str(), "r");
344 
345  if (!fd) {
346  error_message = std::string("Unable to open file ") + filepath;
347  return data;
348  }
349 
350  fseek(fd, 0, SEEK_END);
351  long fsize = ftell(fd);
352  fseek(fd, 0, SEEK_SET);
353 
354  char *buffer;
355  buffer = (char *)malloc(sizeof(char) * (fsize + 1));
356 
357  fread(buffer, 1, fsize, fd);
358  buffer[fsize] = 0;
359  fclose(fd);
360  data = std::string(buffer);
361  free(buffer);
362 
363  return data;
364 }
365 
366 bool AlbCurlHelper::defineStorageListCommand(const outsight_alb_driver::AlbFile::Request &request)
367 {
368  std::string storage_command;
369  storage_command.append(k_storage);
370  storage_command.append(request.category);
371 
372  std::string full_address;
373  full_address = curl_address + storage_command;
374  callback_data.clear();
375 
376  // Define the URL for the command.
377  curl_easy_setopt(curl, CURLOPT_URL, full_address.c_str());
378 
379  // Define the callback function.
380  curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback);
381  curl_easy_setopt(curl, CURLOPT_WRITEDATA, &callback_data);
382 
383  chunk = curl_slist_append(chunk, std::string(k_header_accept).append(k_header_app_json).c_str());
384  curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
385 
386  return true;
387 }
AlbCurlHelper::server_response
long server_response
Definition: alb_curl_helper.h:118
k_header_app_json
constexpr const char * k_header_app_json
Definition: alb_curl_helper.cpp:25
k_processing_zones
constexpr const char * k_processing_zones
Definition: alb_curl_helper.cpp:20
AlbCurlHelper::chunk
struct curl_slist * chunk
Definition: alb_curl_helper.h:117
AlbCurlHelper::AlbCurlHelper
AlbCurlHelper(const std::string &ip_address)
Definition: alb_curl_helper.cpp:51
AlbCurlHelper::storage_service_e
storage_service_e
Storage services.
Definition: alb_curl_helper.h:28
AlbCurlHelper::curl
CURL * curl
Definition: alb_curl_helper.h:116
ros.h
AlbCurlHelper::error_message
std::string error_message
Definition: alb_curl_helper.h:122
k_processing_config
constexpr const char * k_processing_config
Definition: alb_curl_helper.cpp:19
AlbCurlHelper::curl_address
std::string curl_address
Definition: alb_curl_helper.h:120
AlbCurlHelper::initCurl
bool initCurl(void)
Initialize the curl command.
Definition: alb_curl_helper.cpp:196
k_http
constexpr const char * k_http
Definition: alb_curl_helper.cpp:12
AlbCurlHelper::executeCurlCommand
bool executeCurlCommand(const std::string &command)
Execute the curl command.
Definition: alb_curl_helper.cpp:295
k_api
constexpr const char * k_api
Definition: alb_curl_helper.cpp:13
AlbCurlHelper::file
FILE * file
Definition: alb_curl_helper.h:124
AlbCurlHelper::getErrorMessage
const std::string getErrorMessage(void) const
Get the error message.
Definition: alb_curl_helper.cpp:70
AlbCurlHelper::executeProcessingConfig
bool executeProcessingConfig(const processing_config_e &service, outsight_alb_driver::AlbConfig::Request &request)
Execute the processing command for the configuration.
Definition: alb_curl_helper.cpp:117
AlbCurlHelper::processing_config_e
processing_config_e
Processing configuration services.
Definition: alb_curl_helper.h:25
k_header_accept
constexpr const char * k_header_accept
Definition: alb_curl_helper.cpp:23
AlbCurlHelper::defineProcessingConfig
bool defineProcessingConfig(const processing_config_e &service, outsight_alb_driver::AlbConfig::Request &request)
Define the processing configuration command.
Definition: alb_curl_helper.cpp:220
AlbCurlHelper::defineStorageListCommand
bool defineStorageListCommand(const outsight_alb_driver::AlbFile::Request &request)
Define the command to list storage files.
Definition: alb_curl_helper.cpp:366
k_server_success
constexpr const int k_server_success
Definition: alb_curl_helper.cpp:31
AlbCurlHelper::executeStorage
bool executeStorage(const storage_service_e service, outsight_alb_driver::AlbFile::Request &request)
Execute the storage command.
Definition: alb_curl_helper.cpp:166
AlbCurlHelper::defineStorageCommand
bool defineStorageCommand(const storage_service_e service, outsight_alb_driver::AlbFile::Request &request)
Define the storage command.
Definition: alb_curl_helper.cpp:246
AlbCurlHelper::executeProcessing
bool executeProcessing(const processing_service_e service)
Execute the processing command.
Definition: alb_curl_helper.cpp:75
AlbCurlHelper::processing_service_e
processing_service_e
Processing services of the ALB.
Definition: alb_curl_helper.h:18
AlbCurlHelper::exportData
bool exportData(const std::string &data, const std::string &output_path, const std::string &print_text=std::string("Print data:"))
Export the data to the output filepath.
Definition: alb_curl_helper.cpp:319
AlbCurlHelper::defineBaseCommand
void defineBaseCommand(const std::string &alb_command)
Define base curl command for the ALB.
Definition: alb_curl_helper.cpp:206
k_processing_commands
const std::map< AlbCurlHelper::processing_service_e, std::string > k_processing_commands
Definition: alb_curl_helper.cpp:14
AlbCurlHelper::callback_data
std::string callback_data
Definition: alb_curl_helper.h:121
AlbCurlHelper::executeProcessingZones
bool executeProcessingZones(std::string &zones)
Execute the processing command for the zones.
Definition: alb_curl_helper.cpp:145
k_storage
constexpr const char * k_storage
Definition: alb_curl_helper.cpp:21
append
ROSCPP_DECL std::string append(const std::string &left, const std::string &right)
k_header_content
constexpr const char * k_header_content
Definition: alb_curl_helper.cpp:24
k_display_config
constexpr const char * k_display_config
Definition: alb_curl_helper.cpp:28
k_server_upload_success
constexpr const int k_server_upload_success
Definition: alb_curl_helper.cpp:32
AlbCurlHelper::~AlbCurlHelper
~AlbCurlHelper(void)
Definition: alb_curl_helper.cpp:59
ROS_INFO
#define ROS_INFO(...)
AlbCurlHelper::readData
const std::string readData(const std::string &filepath)
Read data from the input filepath.
Definition: alb_curl_helper.cpp:339
k_header_app_octet
constexpr const char * k_header_app_octet
Definition: alb_curl_helper.cpp:26
alb_curl_helper.h
AlbCurlHelper::isProcessingRunning
bool isProcessingRunning(bool &running)
Check if processing is running.
Definition: alb_curl_helper.cpp:94


outsight_alb_driver
Author(s): Outsight
autogenerated on Thu Oct 13 2022 02:21:45