The Battle for Wesnoth  1.19.10+dev
config_cache.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2008 - 2025
3  by Pauli Nieminen <paniemin@cc.hut.fi>
4  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
5 
6  This program is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10  This program is distributed in the hope that it will be useful,
11  but WITHOUT ANY WARRANTY.
12 
13  See the COPYING file for more details.
14 */
15 
16 #define GETTEXT_DOMAIN "wesnoth-lib"
17 
18 #include "config_cache.hpp"
19 #include "filesystem.hpp"
20 #include "gettext.hpp"
21 #include "log.hpp"
22 #include "hash.hpp"
24 #include "serialization/parser.hpp"
26 #include "game_version.hpp"
27 
28 #include <boost/algorithm/string/replace.hpp>
29 #include <boost/iostreams/filter/gzip.hpp>
30 
31 static lg::log_domain log_cache("cache");
32 #define ERR_CACHE LOG_STREAM(err, log_cache)
33 #define LOG_CACHE LOG_STREAM(info, log_cache)
34 #define DBG_CACHE LOG_STREAM(debug, log_cache)
35 
36 namespace game_config
37 {
38 
39 namespace
40 {
41 
42 void add_builtin_defines(preproc_map& target)
43 {
44 #ifdef __APPLE__
45  target["APPLE"] = preproc_define();
46 #endif
47 
48 #if defined(MOUSE_TOUCH_EMULATION) || defined(TARGET_OS_IPHONE)
49  target["IPHONEOS"] = preproc_define();
50 #endif
51 
52  target["WESNOTH_VERSION"] = preproc_define(game_config::wesnoth_version.str());
53 }
54 
55 }
56 
58 {
59  static config_cache cache;
60  return cache;
61 }
62 
64  : force_valid_cache_(false)
65  , use_cache_(true)
66  , fake_invalid_cache_(false)
67  , defines_map_()
68  , cache_file_prefix_("cache-v" + boost::algorithm::replace_all_copy(game_config::revision, ":", "_") + "-")
69 {
70  // To set-up initial defines map correctly
71  clear_defines();
72 }
73 
75 {
76  return defines_map_;
77 }
78 
80 {
81  LOG_CACHE << "Clearing defines map!";
82 
83  defines_map_.clear();
84 
85  //
86  // Set-up default defines map.
87  //
88 
89  add_builtin_defines(defines_map_);
90 }
91 
92 void config_cache::get_config(const std::string& file_path, config& cfg, abstract_validator* validator)
93 {
94  load_configs(file_path, cfg, validator);
95 }
96 
97 void config_cache::write_file(const std::string& file_path, const config& cfg)
98 {
101  writer.write(cfg);
102 }
103 
104 void config_cache::write_file(const std::string& file_path, const preproc_map& defines)
105 {
106  if(defines.empty()) {
107  if(filesystem::file_exists(file_path)) {
108  filesystem::delete_directory(file_path);
109  }
110  return;
111  }
112 
115 
116  // Write all defines to stream.
117  for(const preproc_map::value_type& define : defines) {
118  define.second.write(writer, define.first);
119  }
120 }
121 
122 void config_cache::read_file(const std::string& file_path, config& cfg)
123 {
124  cfg = io::read_gz(*filesystem::istream_file(file_path));
125 }
126 
128 {
130  // HACK: copy_map doesn't have built-in defines in some cases (issue #1924)
131  // or they may be out of date, so brute-force them in.
132  add_builtin_defines(res);
133  return res;
134 }
135 
137 {
139 }
140 
141 void config_cache::read_configs(const std::string& file_path, config& cfg, preproc_map& defines_map, abstract_validator* validator)
142 {
143  //read the file and then write to the cache
144  cfg = io::read(*preprocess_file(file_path, &defines_map), validator);
145 }
146 
147 void config_cache::read_cache(const std::string& file_path, config& cfg, abstract_validator* validator)
148 {
149  static const std::string extension = ".gz";
150 
151  std::stringstream defines_string;
152  defines_string << file_path;
153 
154  bool is_valid = true;
155 
156  for(const preproc_map::value_type& d : defines_map_) {
157  //
158  // Only WESNOTH_VERSION is allowed to be non-empty.
159  //
160  if((!d.second.value.empty() || !d.second.arguments.empty()) &&
161  d.first != "WESNOTH_VERSION")
162  {
163  is_valid = false;
164  ERR_CACHE << "Invalid preprocessor define: " << d.first;
165  break;
166  }
167 
168  defines_string << " " << d.first;
169  }
170 
171  // Do cache check only if define map is valid and
172  // caching is allowed.
173  const std::string& cache_path = filesystem::get_cache_dir();
174 
175  if(is_valid && !cache_path.empty()) {
176  // Use a hash for a shorter display of the defines.
177  const std::string fname = cache_path + "/" +
179  utils::md5(defines_string.str()).hex_digest();
180  const std::string fname_checksum = fname + ".checksum" + extension;
181 
182  filesystem::file_tree_checksum dir_checksum;
183 
185  try {
186  if(filesystem::file_exists(fname_checksum)) {
187  config checksum_cfg;
188 
189  DBG_CACHE << "Reading checksum: " << fname_checksum;
190  read_file(fname_checksum, checksum_cfg);
191 
192  dir_checksum = filesystem::file_tree_checksum(checksum_cfg);
193  }
194  } catch(const config::error&) {
195  ERR_CACHE << "cache checksum is corrupt";
196  } catch(const filesystem::io_exception&) {
197  ERR_CACHE << "error reading cache checksum";
198  } catch(const std::ios_base::failure&) {
199  ERR_CACHE << "error reading cache checksum";
200  }
201  }
202 
203  if(force_valid_cache_) {
204  LOG_CACHE << "skipping cache validation (forced)";
205  }
206 
207  if(filesystem::file_exists(fname + extension) && (force_valid_cache_ || (dir_checksum == filesystem::data_tree_checksum()))) {
208  LOG_CACHE << "found valid cache at '" << fname << extension << "' with defines_map " << defines_string.str();
209  log_scope("read cache");
210 
211  try {
212  read_file(fname + extension,cfg);
213  const std::string define_file = fname + ".define" + extension;
214 
215  if(filesystem::file_exists(define_file)) {
217  }
218 
219  return;
220  } catch(const config::error& e) {
221  ERR_CACHE << "cache " << fname << extension << " is corrupt. Loading from files: "<< e.message;
222  } catch(const filesystem::io_exception&) {
223  ERR_CACHE << "error reading cache " << fname << extension << ". Loading from files";
224  } catch (const boost::iostreams::gzip_error& e) {
225  //read_file -> ... -> read_gz can throw this exception.
226  ERR_CACHE << "cache " << fname << extension << " is corrupt. Error code: " << e.error();
227  }
228  }
229 
230  LOG_CACHE << "no valid cache found. Writing cache to '" << fname << extension << " with defines_map "<< defines_string.str() << "'";
231 
232  // Now we need queued defines so read them to memory
234 
235  preproc_map copy_map(make_copy_map());
236 
237  read_configs(file_path, cfg, copy_map, validator);
238  add_defines_map_diff(copy_map);
239 
240  try {
241  write_file(fname + extension, cfg);
242  write_file(fname + ".define" + extension, copy_map);
243 
244  config checksum_cfg;
245 
246  filesystem::data_tree_checksum().write(checksum_cfg);
247  write_file(fname_checksum, checksum_cfg);
248  } catch(const filesystem::io_exception&) {
249  ERR_CACHE << "could not write to cache '" << fname << "'";
250  }
251 
252  return;
253  }
254 
255  LOG_CACHE << "Loading plain config instead of cache";
256 
257  preproc_map copy_map(make_copy_map());
258  read_configs(file_path, cfg, copy_map, validator);
259  add_defines_map_diff(copy_map);
260 }
261 
262 void config_cache::read_defines_file(const std::string& file_path)
263 {
264  config cfg;
265  read_file(file_path, cfg);
266 
267  DBG_CACHE << "Reading cached defines from: " << file_path;
268 
269  // use static preproc_define::read_pair(config) to make a object
270  // and pass that object config_cache_transaction::insert_to_active method
271  for(const auto [key, cfg] : cfg.all_children_view()) {
273  }
274 }
275 
277 {
278  const std::vector<std::string>& files = config_cache_transaction::instance().get_define_files();
279 
280  for(const std::string &p : files) {
282  }
283 }
284 
285 void config_cache::load_configs(const std::string& config_path, config& cfg, abstract_validator* validator)
286 {
287  // Make sure that we have fake transaction if no real one is going on
289 
290  if (use_cache_) {
291  read_cache(config_path, cfg, validator);
292  } else {
293  preproc_map copy_map(make_copy_map());
294  read_configs(config_path, cfg, copy_map, validator);
295  add_defines_map_diff(copy_map);
296  }
297 }
298 
300 {
301  fake_invalid_cache_ = force;
302 }
303 
305 {
306  use_cache_ = use;
307 }
308 
310 {
311  force_valid_cache_ = force;
312 }
313 
315 {
317 }
318 
319 void config_cache::add_define(const std::string& define)
320 {
321  DBG_CACHE << "adding define: " << define;
322  defines_map_[define] = preproc_define();
323 
325  // we have to add this to active map too
327  }
328 
329 }
330 
331 void config_cache::remove_define(const std::string& define)
332 {
333  DBG_CACHE << "removing define: " << define;
334  defines_map_.erase(define);
335 
337  // we have to remove this from active map too
339  }
340 }
341 
343 {
344  std::vector<std::string> files, dirs;
346 
347  LOG_CACHE << "clean_cache(): " << files.size() << " files, "
348  << dirs.size() << " dirs to check";
349 
350  const std::string& exclude_current = cache_file_prefix_ + "*";
351 
352  bool status = true;
353 
354  status &= delete_cache_files(files, exclude_current);
355  status &= delete_cache_files(dirs, exclude_current);
356 
357  LOG_CACHE << "clean_cache(): done";
358 
359  return status;
360 }
361 
363 {
364  std::vector<std::string> files, dirs;
366 
367  LOG_CACHE << "purge_cache(): deleting " << files.size() << " files, "
368  << dirs.size() << " dirs";
369 
370  bool status = true;
371 
372  status &= delete_cache_files(files);
373  status &= delete_cache_files(dirs);
374 
375  LOG_CACHE << "purge_cache(): done";
376  return status;
377 }
378 
379 bool config_cache::delete_cache_files(const std::vector<std::string>& paths,
380  const std::string& exclude_pattern)
381 {
382  const bool delete_everything = exclude_pattern.empty();
383  bool status = true;
384 
385  for(const std::string& file_path : paths)
386  {
387  if(!delete_everything) {
388  const std::string& fn = filesystem::base_name(file_path);
389 
390  if(utils::wildcard_string_match(fn, exclude_pattern)) {
391  LOG_CACHE << "delete_cache_files(): skipping " << file_path
392  << " excluded by '" << exclude_pattern << "'";
393  continue;
394  }
395  }
396 
397  LOG_CACHE << "delete_cache_files(): deleting " << file_path;
398  if(!filesystem::delete_directory(file_path)) {
399  ERR_CACHE << "delete_cache_files(): could not delete "
400  << file_path;
401  status = false;
402  }
403  }
404 
405  return status;
406 }
407 
410 
412  : define_filenames_()
413  , active_map_()
414 {
415  assert(state_ == FREE);
416  state_ = NEW;
417  active_ = this;
418 }
419 
421 {
422  state_ = FREE;
423  active_ = nullptr;
424 }
425 
427 {
428  state_ = LOCKED;
429 }
430 
431 const std::vector<std::string>& config_cache_transaction::get_define_files() const
432 {
433  return define_filenames_;
434 }
435 
436 void config_cache_transaction::add_define_file(const std::string& file)
437 {
438  define_filenames_.push_back(file);
439 }
440 
442 {
443  if(active_map_.empty()) {
444  active_map_.insert(defines_map.begin(), defines_map.end());
445  if(get_state() == NEW) {
446  state_ = ACTIVE;
447  }
448  }
449 
450  return active_map_;
451 }
452 
453 namespace
454 {
455 
456 bool compare_define(const preproc_map::value_type& a, const preproc_map::value_type& b)
457 {
458  if(a.first < b.first) {
459  return true;
460  }
461 
462  if(b.first < a.first) {
463  return false;
464  }
465 
466  if(a.second < b.second) {
467  return true;
468  }
469 
470  return false;
471 }
472 
473 } // end anonymous namespace
474 
476 {
477  if(get_state() == ACTIVE) {
478  preproc_map temp;
479  std::set_difference(new_map.begin(),
480  new_map.end(),
481  active_map_.begin(),
482  active_map_.end(),
483  std::insert_iterator<preproc_map>(temp,temp.begin()),
484  &compare_define);
485 
486  for(const preproc_map::value_type &def : temp) {
487  insert_to_active(def);
488  }
489 
490  temp.swap(new_map);
491  } else if (get_state() == LOCKED) {
492  new_map.clear();
493  }
494 }
495 
496 void config_cache_transaction::insert_to_active(const preproc_map::value_type& def)
497 {
498  active_map_[def.first] = def.second;
499 }
500 
501 }
Used in parsing config file.
Definition: validator.hpp:38
Class for writing a config out to a file in pieces.
void write(const config &cfg)
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
auto all_children_view() const
In-order iteration over all children.
Definition: config.hpp:796
Used to share macros between cache objects You have to create transaction object to load all macros t...
void add_defines_map_diff(preproc_map &defines_map)
std::vector< std::string > define_filenames_
static config_cache_transaction & instance()
const std::vector< std::string > & get_define_files() const
void lock()
Lock the transaction so no more macros are added.
preproc_map & get_active_map(const preproc_map &defines_map)
void insert_to_active(const preproc_map::value_type &def)
Used to let std::for_each insert new defines to active_map map to active.
static config_cache_transaction * active_
void add_define_file(const std::string &file)
Singleton class to manage game config file caching.
void load_configs(const std::string &path, config &cfg, abstract_validator *validator=nullptr)
void add_define(const std::string &define)
Add a entry to preproc defines map.
void recheck_filetree_checksum()
Force cache checksum validation.
void read_cache(const std::string &path, config &cfg, abstract_validator *validator=nullptr)
void write_file(const std::string &file, const config &cfg)
bool purge_cache()
Deletes all cache files.
void set_use_cache(bool use)
Enable/disable caching.
static config_cache & instance()
Get reference to the singleton object.
preproc_map & make_copy_map()
const preproc_map & get_preproc_map() const
void remove_define(const std::string &define)
Remove a entry to preproc defines map.
void clear_defines()
Clear stored defines map to default values.
void read_configs(const std::string &path, config &cfg, preproc_map &defines, abstract_validator *validator=nullptr)
bool clean_cache()
Deletes stale cache files not in use by the game.
void read_file(const std::string &file, config &cfg)
bool delete_cache_files(const std::vector< std::string > &paths, const std::string &exclude_pattern="")
void read_defines_file(const std::string &path)
void add_defines_map_diff(preproc_map &)
void set_force_valid_cache(bool force)
Enable/disable cache validation.
void get_config(const std::string &path, config &cfg, abstract_validator *validator=nullptr)
Gets a config object from given path.
Holds a fake cache transaction if no real one is used.
virtual std::string hex_digest() const override
Definition: hash.cpp:120
#define LOG_CACHE
static lg::log_domain log_cache("cache")
#define DBG_CACHE
#define ERR_CACHE
Declarations for File-IO.
Interfaces for manipulating version numbers of engine, add-ons, etc.
Standard logging facilities (interface).
#define log_scope(description)
Definition: log.hpp:275
std::string get_cache_dir()
Definition: filesystem.cpp:866
filesystem::scoped_istream istream_file(const std::string &fname, bool treat_failure_as_error)
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
std::string base_name(const std::string &file, const bool remove_extension)
Returns the base filename of a file, with directory name stripped.
static bool file_exists(const bfs::path &fpath)
Definition: filesystem.cpp:328
bool delete_directory(const std::string &dirname, const bool keep_pbl)
const file_tree_checksum & data_tree_checksum(bool reset=false)
Get the time at which the data/ tree was last modified at.
filesystem::scoped_ostream ostream_file(const std::string &fname, std::ios_base::openmode mode, bool create_directory)
std::unique_ptr< std::ostream > scoped_ostream
Definition: filesystem.hpp:54
Game configuration data as global variables.
Definition: build_info.cpp:61
const version_info wesnoth_version(VERSION)
int cache_compression_level
Definition: filesystem.cpp:105
const std::string revision
config read(std::istream &in, abstract_validator *validator)
Definition: parser.cpp:627
config read_gz(std::istream &file, abstract_validator *validator)
Might throw a std::ios_base::failure especially a gzip_error.
Definition: parser.cpp:683
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,...
fake
For describing the type of faked display, if any.
Definition: video.hpp:44
filesystem::scoped_istream preprocess_file(const std::string &fname, preproc_map *defines)
Function to use the WML preprocessor on a file.
std::map< std::string, struct preproc_define > preproc_map
An exception object used when an IO error occurs.
Definition: filesystem.hpp:67
static preproc_map::value_type read_pair(const config &)
mock_party p
#define d
#define e
#define b