The Battle for Wesnoth  1.19.10+dev
filesystem_common.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2017 - 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 
15 #include "filesystem.hpp"
16 #include "wesconfig.h"
17 
18 #include "config.hpp"
19 #include "log.hpp"
20 #include "serialization/chrono.hpp"
23 #include "utils/general.hpp"
24 
25 #include <boost/algorithm/string.hpp>
26 
27 static lg::log_domain log_filesystem("filesystem");
28 #define LOG_FS LOG_STREAM(info, log_filesystem)
29 #define ERR_FS LOG_STREAM(err, log_filesystem)
30 
31 namespace filesystem
32 {
33 
34 bool is_legal_user_file_name(const std::string& name, bool allow_whitespace)
35 {
36  //
37  // IMPORTANT NOTE:
38  //
39  // If you modify this function you must be aware that it is used by the
40  // add-on server validation routines both on the client and server sides.
41  // The addition or removal of any criteria here should be carefully
42  // evaluated with this in mind.
43  //
44 
45  if(name.empty() || name.back() == '.' || name.find("..") != std::string::npos || name.size() > 255) {
46  return false;
47  }
48 
49  // Reserved DOS device names on Windows.
50  static const std::set<std::string> dos_device_names = {
51  // Hardware devices
52  "NUL", "CON", "AUX", "PRN",
53  "COM1", "COM2", "COM3", "COM4", "COM5", "COM6", "COM7", "COM8", "COM9",
54  "LPT1", "LPT2", "LPT3", "LPT4", "LPT5", "LPT6", "LPT7", "LPT8", "LPT9",
55  // Console API pseudo-devices
56  "CONIN$", "CONOUT$",
57  };
58 
59  // We can't use filesystem::base_name() here, because it returns the
60  // filename up to the *last* dot. "CON.foo.bar" is still redirected to
61  // "CON" on Windows, although "foo.CON.bar" and "foo.bar.CON" are not.
62  //
63  // Do also note that we're relying on the char-by-char check further below
64  // to flag the name as illegal if it contains a colon ':', the reason
65  // being that is valid to refer to DOS device names with a trailing colon
66  // (e.g. "CON:" is synonymous with "CON").
67 
68  const auto& first_name =
69  boost::algorithm::to_upper_copy(name.substr(0, name.find('.')), std::locale::classic());
70 
71  if(dos_device_names.count(first_name)) {
72  return false;
73  }
74 
75  const auto& name_ucs4 = unicode_cast<std::u32string>(name);
76  if(name != unicode_cast<std::string>(name_ucs4)){
77  return false; // name is an invalid UTF-8 sequence
78  }
79 
80  return name_ucs4.end() == std::find_if(name_ucs4.begin(), name_ucs4.end(), [=](char32_t c)
81  {
82  switch(c) {
83  case ' ':
84  return !allow_whitespace;
85  case '"':
86  case '*':
87  case '/':
88  case ':':
89  case '<':
90  case '>':
91  case '?':
92  case '\\':
93  case '|':
94  case '~':
95  case 0x7F: // DEL
96  return true;
97  default:
98  return c < 0x20 || // C0 control characters
99  (c >= 0x80 && c < 0xA0) || // C1 control characters
100  (c >= 0xD800 && c < 0xE000); // surrogate pairs
101  }
102  });
103 }
104 
105 void blacklist_pattern_list::remove_blacklisted_files_and_dirs(std::vector<std::string>& files, std::vector<std::string>& directories) const
106 {
107  utils::erase_if(files, [this](const std::string& name) { return match_file(name); });
108  utils::erase_if(directories, [this](const std::string& name) { return match_dir(name); });
109 }
110 
111 bool blacklist_pattern_list::match_file(const std::string& name) const
112 {
113  return std::any_of(file_patterns_.begin(), file_patterns_.end(),
114  std::bind(&utils::wildcard_string_match, std::ref(name), std::placeholders::_1));
115 }
116 
117 bool blacklist_pattern_list::match_dir(const std::string& name) const
118 {
119  return std::any_of(directory_patterns_.begin(), directory_patterns_.end(),
120  std::bind(&utils::wildcard_string_match, std::ref(name), std::placeholders::_1));
121 }
122 
123 std::string autodetect_game_data_dir(std::string exe_dir)
124 {
125  std::string auto_dir;
126 
127  // scons leaves the resulting binaries at the root of the source
128  // tree by default.
129  if(filesystem::file_exists(exe_dir + "/data/_main.cfg")) {
130  auto_dir = std::move(exe_dir);
131  }
132  // cmake encourages creating a subdir at the root of the source
133  // tree for the build, and the resulting binaries are found in it.
134  else if(filesystem::file_exists(exe_dir + "/../data/_main.cfg")) {
135  auto_dir = filesystem::normalize_path(exe_dir + "/..");
136  }
137  // Allow using the current working directory as the game data dir
138  else if(filesystem::file_exists(filesystem::get_cwd() + "/data/_main.cfg")) {
139  auto_dir = filesystem::get_cwd();
140  }
141 #ifdef _WIN32
142  // In Windows builds made using Visual Studio and its CMake
143  // integration, the EXE is placed a few levels below the game data
144  // dir (e.g. .\out\build\x64-Debug).
145  else if(filesystem::file_exists(exe_dir + "/../../build") && filesystem::file_exists(exe_dir + "/../../../out")
146  && filesystem::file_exists(exe_dir + "/../../../data/_main.cfg")) {
147  auto_dir = filesystem::normalize_path(exe_dir + "/../../..");
148  }
149 #endif
150 
151  return auto_dir;
152 }
153 
155 {
156  return get_sync_dir() + "/preferences";
157 }
158 
160 {
161  return get_user_data_dir() + "/preferences";
162 }
163 
164 std::string get_credentials_file()
165 {
166  return get_user_data_dir() + "/credentials-aes";
167 }
168 
170 {
171 #ifdef HAS_RELATIVE_DEFPREF
173 #else
175 #endif
176 }
177 
178 std::string get_save_index_file()
179 {
180  return get_user_data_dir() + "/save_index";
181 }
182 
183 std::string get_lua_history_file()
184 {
185  return get_sync_dir() + "/lua_command_history";
186 }
187 
188 std::string get_sync_dir()
189 {
190  return get_user_data_dir() + "/sync";
191 }
192 
193 std::string get_saves_dir()
194 {
195  const std::string dir_path = get_sync_dir() + "/saves";
196  return get_dir(dir_path);
197 }
198 
199 std::string get_addons_data_dir()
200 {
201  const std::string dir_path = get_user_data_dir() + "/data";
202  return get_dir(dir_path);
203 }
204 
205 std::string get_addons_dir()
206 {
207  const std::string dir_path = get_addons_data_dir() + "/add-ons";
208  return get_dir(dir_path);
209 }
210 
211 std::string get_wml_persist_dir()
212 {
213  const std::string dir_path = get_sync_dir() + "/persist";
214  return get_dir(dir_path);
215 }
216 
218 {
219  const std::string dir_path = get_sync_dir() + "/editor";
220  return get_dir(dir_path);
221 }
222 
223 std::string get_current_editor_dir(const std::string& addon_id)
224 {
225  if(addon_id == "mainline") {
226  return get_dir(game_config::path) + "/data/multiplayer";
227  } else {
228  return get_addons_dir() + "/" + addon_id;
229  }
230 }
231 
232 std::string get_core_images_dir()
233 {
234  return get_dir(game_config::path + "/data/core/images");
235 }
236 
237 std::string get_intl_dir()
238 {
239 #ifdef _WIN32
240  return game_config::path + "/" LOCALEDIR;
241 #else
242 
243 #ifdef USE_INTERNAL_DATA
244  return get_cwd() + "/" LOCALEDIR;
245 #endif
246 
247 #if HAS_RELATIVE_LOCALEDIR
248  std::string res = game_config::path + "/" LOCALEDIR;
249 #else
250  std::string res = LOCALEDIR;
251 #endif
252 
253  return res;
254 #endif
255 }
256 
257 std::string get_screenshot_dir()
258 {
259  const std::string dir_path = get_user_data_dir() + "/screenshots";
260  return get_dir(dir_path);
261 }
262 
263 bool looks_like_pbl(const std::string& file)
264 {
265  return utils::wildcard_string_match(utf8::lowercase(file), "*.pbl");
266 }
267 
268 file_tree_checksum::file_tree_checksum(const config& cfg)
269  : nfiles(cfg["nfiles"].to_size_t())
270  , sum_size(cfg["size"].to_size_t())
271  , modified(chrono::parse_timestamp(cfg["modified"]))
272 {
273 }
274 
276 {
277  cfg["nfiles"] = nfiles;
278  cfg["size"] = sum_size;
279  cfg["modified"] = chrono::serialize_timestamp(modified);
280 }
281 
283 {
284  return nfiles == rhs.nfiles && sum_size == rhs.sum_size &&
285  modified == rhs.modified;
286 }
287 
288 std::string read_map(const std::string& name)
289 {
290  std::string res;
291  auto map_location = get_wml_location(name);
292  if(!map_location) {
293  // Consult [binary_path] for maps as well.
294  map_location = get_binary_file_location("maps", name);
295  }
296  if(map_location) {
297  res = read_file(map_location.value());
298  }
299 
300  if(res.empty()) {
301  res = read_file(get_legacy_editor_dir() + "/maps/" + name);
302  }
303 
304  return res;
305 }
306 
307 std::string read_scenario(const std::string& name)
308 {
309  std::string res;
310  auto file_location = get_wml_location(name);
311  if(!file_location) {
312  // Consult [binary_path] for scenarios as well.
313  file_location = get_binary_file_location("scenarios", name);
314  }
315  if(file_location) {
316  res = read_file(file_location.value());
317  }
318 
319  if(res.empty()) {
320  res = read_file(get_legacy_editor_dir() + "/scenarios/" + name);
321  }
322 
323  return res;
324 }
325 
326 static void get_file_tree_checksum_internal(const std::string& path, file_tree_checksum& res)
327 {
328 
329  std::vector<std::string> dirs;
331 
332  for(std::vector<std::string>::const_iterator j = dirs.begin(); j != dirs.end(); ++j) {
334  }
335 }
336 
338 {
339  static file_tree_checksum checksum;
340  if (reset)
341  checksum = file_tree_checksum{};
342  if(checksum.nfiles == 0) {
343  get_file_tree_checksum_internal("data/",checksum);
344  get_file_tree_checksum_internal(get_user_data_dir() + "/data/",checksum);
345  LOG_FS << "calculated data tree checksum: "
346  << checksum.nfiles << " files; "
347  << checksum.sum_size << " bytes";
348  }
349 
350  return checksum;
351 }
352 
353 }
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
Definitions for the interface to Wesnoth Markup Language (WML).
Declarations for File-IO.
static lg::log_domain log_filesystem("filesystem")
#define LOG_FS
Standard logging facilities (interface).
auto serialize_timestamp(const std::chrono::system_clock::time_point &time)
Definition: chrono.hpp:57
auto parse_timestamp(long long val)
Definition: chrono.hpp:47
std::string get_legacy_editor_dir()
void get_files_in_dir(const std::string &dir, std::vector< std::string > *files, std::vector< std::string > *dirs, name_mode mode, filter_mode filter, reorder_mode reorder, file_tree_checksum *checksum)
Get a list of all files and/or directories in a given directory.
Definition: filesystem.cpp:450
static bfs::path get_dir(const bfs::path &dirpath)
Definition: filesystem.cpp:339
std::string get_user_data_dir()
Definition: filesystem.cpp:856
std::string get_wml_persist_dir()
static bool file_exists(const bfs::path &fpath)
Definition: filesystem.cpp:328
bool is_legal_user_file_name(const std::string &name, bool allow_whitespace=true)
Returns whether the given filename is a legal name for a user-created file.
std::string get_synced_prefs_file()
location of preferences file containing preferences that are synced between computers note that wesno...
utils::optional< std::string > get_wml_location(const std::string &path, const utils::optional< std::string > &current_dir)
Returns a translated path to the actual file or directory, if it exists.
std::string get_saves_dir()
const file_tree_checksum & data_tree_checksum(bool reset=false)
Get the time at which the data/ tree was last modified at.
std::string read_file(const std::string &fname)
Basic disk I/O - read file.
std::string get_unsynced_prefs_file()
location of preferences file containing preferences that aren't synced between computers
std::string get_save_index_file()
utils::optional< std::string > get_binary_file_location(const std::string &type, const std::string &filename)
Returns a complete path to the actual file of a given type, if it exists.
std::string get_lua_history_file()
std::string autodetect_game_data_dir(std::string exe_dir)
Try to autodetect the location of the game data dir.
std::string read_scenario(const std::string &name)
std::string get_screenshot_dir()
std::string get_credentials_file()
std::string get_sync_dir()
parent directory for everything that should be synced between systems.
bool looks_like_pbl(const std::string &file)
std::string get_addons_data_dir()
std::string get_default_prefs_file()
std::string get_addons_dir()
std::string get_intl_dir()
std::string get_core_images_dir()
static void get_file_tree_checksum_internal(const std::string &path, file_tree_checksum &res)
std::string get_current_editor_dir(const std::string &addon_id)
std::string read_map(const std::string &name)
std::string get_cwd()
Definition: filesystem.cpp:948
std::string normalize_path(const std::string &fpath, bool normalize_separators, bool resolve_dot_entries)
Returns the absolute path of a file.
std::string path
Definition: filesystem.cpp:93
std::string default_preferences_path
Definition: filesystem.cpp:99
std::string lowercase(std::string_view s)
Returns a lowercased version of the string.
Definition: unicode.cpp:50
bool wildcard_string_match(const std::string &str, const std::string &match)
Match using '*' as any number of characters (including none), '+' as one or more characters,...
void erase_if(Container &container, const Predicate &predicate)
Convenience wrapper for using std::remove_if on a container.
Definition: general.hpp:106
bool operator==(const file_tree_checksum &rhs) const
std::chrono::system_clock::time_point modified
Definition: filesystem.hpp:317
Encapsulates the map of the game.
Definition: location.hpp:45
mock_char c
Some defines: VERSION, PACKAGE, MIN_SAVEGAME_VERSION.
#define LOCALEDIR
Definition: wesconfig.h:19