The Battle for Wesnoth  1.19.12+dev
network_download_file.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2003 - 2025
3  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
4 
5  This program is free software; you can redistribute it and/or modify
6  it under the terms of the GNU General Public License as published by
7  the Free Software Foundation; either version 2 of the License, or
8  (at your option) any later version.
9  This program is distributed in the hope that it will be useful,
10  but WITHOUT ANY WARRANTY.
11 
12  See the COPYING file for more details.
13 */
14 
16 
17 #include "filesystem.hpp"
18 #include "gettext.hpp"
19 #include "gui/dialogs/message.hpp"
20 #include "log.hpp"
21 
22 #include <curl/curl.h>
23 
24 static lg::log_domain log_network("network");
25 #define ERR_NW LOG_STREAM(err, log_network)
26 #define DBG_NW LOG_STREAM(debug, log_network)
27 
28 namespace network
29 {
30  static size_t write_callback(char* contents, size_t size, size_t nmemb, void* buffer)
31  {
32  size_t amount = size * nmemb;
33  static_cast<std::string*>(buffer)->append(contents, amount);
34  DBG_NW << "Downloaded " << amount << " bytes.";
35  return amount;
36  }
37 
38  void gui_download(const std::string& url, const std::string& local_path) {
39  if(filesystem::file_exists(local_path)) {
40  const int res = gui2::show_message(_("Confirm overwrite"), _("Overwrite existing file?"), gui2::dialogs::message::yes_no_buttons);
41  if(res != gui2::retval::OK) {
42  return;
43  }
44  }
45  if (download(url, local_path)) {
46  gui2::show_message(_("Download complete"), _("The file has been downloaded."), gui2::dialogs::message::button_style::auto_close);
47  } else {
48  gui2::show_message(_("Download error"), _("An error occurred when downloading the file. Check the game logs for more information."), gui2::dialogs::message::button_style::auto_close);
49  }
50  }
51 
52  bool download(const std::string& url, const std::string& local_path)
53  {
54  std::unique_ptr<CURL, decltype(&curl_easy_cleanup)> curl(curl_easy_init(), curl_easy_cleanup);
55  std::string buffer;
56  // curl doesn't initialize the error buffer until version 7.60.0, which isn't currently available on all supported macOS versions
57  char error[CURL_ERROR_SIZE];
58  std::fill_n(error, CURL_ERROR_SIZE-1, ' ');
59  error[CURL_ERROR_SIZE-1] = '\0';
60 
61  if(!curl) {
62  ERR_NW << "curl_easy_init failed initialization, unable to download file.";
63  return false;
64  }
65 
66  CURLcode res;
67  if((res = curl_easy_setopt(curl.get(), CURLOPT_URL, url.c_str())) != CURLE_OK ||
68  (res = curl_easy_setopt(curl.get(), CURLOPT_WRITEFUNCTION, write_callback)) != CURLE_OK ||
69  (res = curl_easy_setopt(curl.get(), CURLOPT_WRITEDATA, &buffer)) != CURLE_OK ||
70  (res = curl_easy_setopt(curl.get(), CURLOPT_ERRORBUFFER, error)) != CURLE_OK ||
71  (res = curl_easy_setopt(curl.get(), CURLOPT_FORBID_REUSE, 1L)) != CURLE_OK ||
72  (res = curl_easy_setopt(curl.get(), CURLOPT_FRESH_CONNECT, 1L)) != CURLE_OK ||
73  (res = curl_easy_setopt(curl.get(), CURLOPT_FAILONERROR, 1L)) != CURLE_OK ||
74  (res = curl_easy_setopt(curl.get(), CURLOPT_TIMEOUT_MS, 5000L)) != CURLE_OK ||
75 #if LIBCURL_VERSION_NUM >= 0x075500
76  (res = curl_easy_setopt(curl.get(), CURLOPT_PROTOCOLS_STR, "https")) != CURLE_OK
77 #else
78  (res = curl_easy_setopt(curl.get(), CURLOPT_PROTOCOLS, CURLPROTO_HTTPS)) != CURLE_OK
79 #endif
80  ) {
81  ERR_NW << "Error setting curl option: " << curl_easy_strerror(res);
82  return false;
83  }
84 
85  res = curl_easy_perform(curl.get());
86  if(res != CURLE_OK) {
87  ERR_NW << "Error downloading file from url `" << url << "`.\n"
88  << "Short error: " << curl_easy_strerror(res) << "\n"
89  << "Long error: " << std::string(error);
90  return false;
91  }
92 
93  try {
94  filesystem::write_file(local_path, buffer);
95  DBG_NW << "Wrote downloaded file to: " << local_path;
96  } catch(const filesystem::io_exception& e) {
97  ERR_NW << "io_exception writing downloaded data to file at: " << local_path << "\n" << e.what() << " : " << e.message;
98  return false;
99  }
100  return true;
101  }
102 }
@ yes_no_buttons
Shows a yes and no button.
Definition: message.hpp:81
Declarations for File-IO.
static std::string _(const char *str)
Definition: gettext.hpp:97
Standard logging facilities (interface).
static bool file_exists(const bfs::path &fpath)
Definition: filesystem.cpp:337
void write_file(const std::string &fname, const std::string &data, std::ios_base::openmode mode)
Throws io_exception if an error occurs.
void show_message(const std::string &title, const std::string &msg, const std::string &button_caption, const bool auto_close, const bool message_use_markup, const bool title_use_markup)
Shows a message to the user.
Definition: message.cpp:148
@ OK
Dialog was closed with the OK button.
Definition: retval.hpp:35
High level network layer for config object transport.
void gui_download(const std::string &url, const std::string &local_path)
Initiates a standalone download of a single file from an HTTPS URL.
bool download(const std::string &url, const std::string &local_path)
static size_t write_callback(char *contents, size_t size, size_t nmemb, void *buffer)
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
Definition: unicode.cpp:85
#define ERR_NW
static lg::log_domain log_network("network")
#define DBG_NW
An exception object used when an IO error occurs.
Definition: filesystem.hpp:67
#define e