The Battle for Wesnoth  1.19.10+dev
lua_wml.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2014 - 2025
3  by Chris Beck <render787@gmail.com>
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 #include "scripting/lua_wml.hpp"
18 #include "scripting/lua_common.hpp"
19 
21 #include "serialization/parser.hpp"
23 #include "variable.hpp" // for config_variable_set
24 
25 #include <fstream>
26 
27 #include "lua/wrapper_lauxlib.h"
28 
29 namespace lua_wml {
30 
31 /**
32 * Dumps a wml table or userdata wml object into a pretty string.
33 * - Arg 1: wml table or vconfig userdata
34 * - Ret 1: string
35 */
36 static int intf_wml_tostring(lua_State* L) {
37  const config& arg = luaW_checkconfig(L, 1);
38  std::ostringstream stream;
39  io::write(stream, arg);
40  lua_pushstring(L, stream.str().c_str());
41  return 1;
42 }
43 
44 /**
45  * Loads a WML file into a config
46  * - Arg 1: WML file path
47  * - Arg 2: (optional) Array of preprocessor defines, or false to skip preprocessing (true is also valid)
48  * - Arg 3: (optional) Path to a schema file for validation (omit for no validation)
49  * - Ret: config
50  */
51 static int intf_load_wml(lua_State* L)
52 {
53  std::string file = luaL_checkstring(L, 1);
54  bool preprocess = true;
55  preproc_map defines_map;
56  if(lua_type(L, 2) == LUA_TBOOLEAN) {
57  preprocess = luaW_toboolean(L, 2);
58  } else if(lua_type(L, 2) == LUA_TTABLE || lua_type(L, 2) == LUA_TUSERDATA) {
59  lua_len(L, 2);
60  int n = lua_tointeger(L, -1);
61  lua_pop(L, 1);
62  for(int i = 0; i < n; i++) {
63  lua_geti(L, 2, i);
64  if(!lua_isstring(L, -1)) {
65  return luaL_argerror(L, 2, "expected bool or array of strings");
66  }
67  std::string define = lua_tostring(L, -1);
68  lua_pop(L, 1);
69  if(!define.empty()) {
70  defines_map.emplace(define, preproc_define(define));
71  }
72  }
73  } else if(!lua_isnoneornil(L, 2)) {
74  return luaL_argerror(L, 2, "expected bool or array of strings");
75  }
76  std::string schema_path = luaL_optstring(L, 3, "");
77  std::shared_ptr<schema_validation::schema_validator> validator;
78  if(!schema_path.empty()) {
80  validator->set_create_exceptions(false); // Don't crash if there's an error, just go ahead anyway
81  }
82  std::string wml_file = filesystem::get_wml_location(file).value();
84  if(preprocess) {
85  stream = preprocess_file(wml_file, &defines_map);
86  } else {
87  stream.reset(new std::ifstream(wml_file));
88  }
89  config result = io::read(*stream, validator.get());
90  luaW_pushconfig(L, result);
91  return 1;
92 }
93 
94 /**
95  * Parses a WML string into a config; does not preprocess or validate
96  * - Arg 1: WML string
97  * - Ret: config
98  */
99 static int intf_parse_wml(lua_State* L)
100 {
101  std::string wml = luaL_checkstring(L, 1);
102  std::string schema_path = luaL_optstring(L, 2, "");
103  std::shared_ptr<schema_validation::schema_validator> validator;
104  if(!schema_path.empty()) {
106  validator->set_create_exceptions(false); // Don't crash if there's an error, just go ahead anyway
107  }
108  config result = io::read(wml, validator.get());
109  luaW_pushconfig(L, result);
110  return 1;
111 }
112 
113 /**
114  * Returns a clone (deep copy) of the passed config, which can be either a normal config or a vconfig
115  * If it is a vconfig, the underlying config is also cloned.
116  * - Arg 1: a config
117  * - Ret: the cloned config
118  */
119 static int intf_clone_wml(lua_State* L)
120 {
121  const vconfig* vcfg = nullptr;
122  const config& cfg = luaW_checkconfig(L, 1, vcfg);
123  if(vcfg) {
124  config clone_underlying = vcfg->get_config();
125  vconfig clone(clone_underlying);
126  luaW_pushvconfig(L, clone);
127  } else {
128  luaW_pushconfig(L, cfg);
129  }
130  return 1;
131 }
132 
133 /**
134 * Interpolates variables into a WML table, including [insert_tag]
135 * Arg 1: WML table to interpolate into
136 * Arg 2: WML table of variables
137 */
138 static int intf_wml_interpolate(lua_State* L)
139 {
140  config cfg = luaW_checkconfig(L, 1), vars_cfg = luaW_checkconfig(L, 2);
141  config_variable_set vars(vars_cfg);
142  vconfig vcfg(cfg, vars);
144  return 1;
145 }
146 
147 /**
148 * Tests if a WML table matches a filter
149 * Arg 1: table to test
150 * Arg 2: filter
151 */
152 static int intf_wml_matches_filter(lua_State* L)
153 {
154  config cfg = luaW_checkconfig(L, 1);
156  lua_pushboolean(L, cfg.matches(filter));
157  return 1;
158 }
159 
160 /**
161 * Merges two WML tables
162 * Arg 1: base table
163 * Arg 2: table to merge in
164 */
165 static int intf_wml_merge(lua_State* L)
166 {
167  config base = luaW_checkconfig(L, 1);
168  config merge = luaW_checkconfig(L, 2);
169  const std::string mode = lua_isstring(L, 3) ? luaL_checkstring(L, 3) : "merge";
170  if(mode == "append") {
171  base.merge_attributes(merge);
172  base.append_children(merge);
173  } else {
174  if(mode == "replace") {
175  for(const auto [key, _] : merge.all_children_view()) {
176  base.clear_children(key);
177  }
178  } else if(mode != "merge") {
179  return luaL_argerror(L, 3, "invalid merge mode - must be merge, append, or replace");
180  }
181  base.merge_with(merge);
182  }
183  luaW_pushconfig(L, base);
184  return 1;
185 }
186 
187 /**
188 * Computes a diff of two WML tables
189 * Arg 1: left table
190 * Arg 2: right table
191 */
192 static int intf_wml_diff(lua_State* L)
193 {
194  config lhs = luaW_checkconfig(L, 1);
195  config rhs = luaW_checkconfig(L, 2);
196  luaW_pushconfig(L, lhs.get_diff(rhs));
197  return 1;
198 }
199 
200 /**
201 * Applies a diff to a WML table
202 * Arg 1: base table
203 * Arg 2: WML diff
204 */
205 static int intf_wml_patch(lua_State* L)
206 {
207  config base = luaW_checkconfig(L, 1);
208  config patch = luaW_checkconfig(L, 2);
209  base.apply_diff(patch);
210  luaW_pushconfig(L, base);
211  return 1;
212 }
213 
214 /**
215 * Tests if two WML tables are equal (have the same keys and values, same tags, recursively)
216 * Arg 1: left table
217 * Arg 2: right table
218 */
219 static int intf_wml_equal(lua_State* L)
220 {
221  config left = luaW_checkconfig(L, 1);
222  config right = luaW_checkconfig(L, 2);
223  lua_pushboolean(L, left == right);
224  return 1;
225 }
226 
227 /**
228 * Tests if a table represents a valid WML table
229 * Arg 1: table
230 */
231 static int intf_wml_valid(lua_State* L)
232 {
233  config test;
234  if(luaW_toconfig(L, 1, test)) {
235  // The validate_wml call is PROBABLY redundant, but included just in case validation changes and toconfig isn't updated to match
236  lua_pushboolean(L, test.validate_wml());
237  } else lua_pushboolean(L, false);
238  return 1;
239 }
240 
241 int luaW_open(lua_State* L) {
242  auto& lk = lua_kernel_base::get_lua_kernel<lua_kernel_base>(L);
243  lk.add_log("Adding wml module...\n");
244  static luaL_Reg const wml_callbacks[]= {
245  { "load", &intf_load_wml},
246  { "parse", &intf_parse_wml},
247  { "clone", &intf_clone_wml},
248  { "merge", &intf_wml_merge},
249  { "diff", &intf_wml_diff},
250  { "patch", &intf_wml_patch},
251  { "equal", &intf_wml_equal},
252  { "valid", &intf_wml_valid},
253  { "matches_filter", &intf_wml_matches_filter},
254  { "tostring", &intf_wml_tostring},
255  { "interpolate", &intf_wml_interpolate},
256  { nullptr, nullptr },
257  };
258  lua_newtable(L);
259  luaL_setfuncs(L, wml_callbacks, 0);
260  return 1;
261 }
262 
263 }
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
bool matches(const config &filter) const
Definition: config.cpp:1190
bool validate_wml() const
Returns true if this object represents valid WML, i.e.
Definition: config.cpp:1344
auto all_children_view() const
In-order iteration over all children.
Definition: config.hpp:796
void merge_attributes(const config &)
Definition: config.cpp:738
void clear_children(T... keys)
Definition: config.hpp:602
void merge_with(const config &c)
Merge config 'c' into this config, overwriting this config's values.
Definition: config.cpp:1119
void apply_diff(const config &diff, bool track=false)
A function to apply a diff config onto this config object.
Definition: config.cpp:1022
config get_diff(const config &c) const
A function to get the differences between this object, and 'c', as another config object.
Definition: config.cpp:906
void append_children(const config &cfg)
Adds children from cfg.
Definition: config.cpp:167
Realization of serialization/validator.hpp abstract validator.
A variable-expanding proxy for the config class.
Definition: variable.hpp:45
const config & get_config() const
Definition: variable.hpp:75
config get_parsed_config() const
Definition: variable.cpp:176
std::size_t i
Definition: function.cpp:1022
static std::string _(const char *str)
Definition: gettext.hpp:103
void luaW_pushconfig(lua_State *L, const config &cfg)
Converts a config object to a Lua table pushed at the top of the stack.
Definition: lua_common.cpp:905
config luaW_checkconfig(lua_State *L, int index)
Converts an optional table or vconfig to a config object.
void luaW_pushvconfig(lua_State *L, const vconfig &cfg)
Pushes a vconfig on the top of the stack.
Definition: lua_common.cpp:539
bool luaW_toboolean(lua_State *L, int n)
bool luaW_toconfig(lua_State *L, int index, config &cfg)
Converts an optional table or vconfig to a config object.
Definition: lua_common.cpp:934
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::unique_ptr< std::istream > scoped_istream
Definition: filesystem.hpp:53
config read(std::istream &in, abstract_validator *validator)
Definition: parser.cpp:627
void write(std::ostream &out, const configr_of &cfg, unsigned int level)
Definition: parser.cpp:764
static int intf_wml_patch(lua_State *L)
Applies a diff to a WML table Arg 1: base table Arg 2: WML diff.
Definition: lua_wml.cpp:205
static int intf_wml_matches_filter(lua_State *L)
Tests if a WML table matches a filter Arg 1: table to test Arg 2: filter.
Definition: lua_wml.cpp:152
static int intf_wml_tostring(lua_State *L)
Dumps a wml table or userdata wml object into a pretty string.
Definition: lua_wml.cpp:36
static int intf_wml_equal(lua_State *L)
Tests if two WML tables are equal (have the same keys and values, same tags, recursively) Arg 1: left...
Definition: lua_wml.cpp:219
int luaW_open(lua_State *L)
Definition: lua_wml.cpp:241
static int intf_wml_valid(lua_State *L)
Tests if a table represents a valid WML table Arg 1: table.
Definition: lua_wml.cpp:231
static int intf_load_wml(lua_State *L)
Loads a WML file into a config.
Definition: lua_wml.cpp:51
static int intf_wml_interpolate(lua_State *L)
Interpolates variables into a WML table, including [insert_tag] Arg 1: WML table to interpolate into ...
Definition: lua_wml.cpp:138
static int intf_parse_wml(lua_State *L)
Parses a WML string into a config; does not preprocess or validate.
Definition: lua_wml.cpp:99
static int intf_wml_diff(lua_State *L)
Computes a diff of two WML tables Arg 1: left table Arg 2: right table.
Definition: lua_wml.cpp:192
static int intf_clone_wml(lua_State *L)
Returns a clone (deep copy) of the passed config, which can be either a normal config or a vconfig If...
Definition: lua_wml.cpp:119
static int intf_wml_merge(lua_State *L)
Merges two WML tables Arg 1: base table Arg 2: table to merge in.
Definition: lua_wml.cpp:165
constexpr auto filter
Definition: ranges.hpp:38
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
One of the realizations of serialization/validator.hpp abstract validator.
static map_location::direction n