2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
8 /** @file strgen.cpp Tool to create computer readable (stand-alone) translation files. */
10 #include "../stdafx.h"
11 #include "../core/endian_func.hpp"
12 #include "../core/mem_func.hpp"
13 #include "../error_func.h"
14 #include "../string_func.h"
15 #include "../strings_type.h"
16 #include "../misc/getoptdata.h"
17 #include "../table/control_codes.h"
18 #include "../3rdparty/fmt/std.h"
25 #include "../table/strgen_tables.h"
27 #include "../safeguards.h"
31 # define LINE_NUM_FMT(s) "{} ({}): warning: {} (" s ")\n"
33 # define LINE_NUM_FMT(s) "{}:{}: " s ": {}\n"
36 void StrgenWarningI(const std::string
&msg
)
39 fmt::print(stderr
, LINE_NUM_FMT("warning"), _file
, _cur_line
, msg
);
41 fmt::print(stderr
, LINE_NUM_FMT("info"), _file
, _cur_line
, msg
);
46 void StrgenErrorI(const std::string
&msg
)
48 fmt::print(stderr
, LINE_NUM_FMT("error"), _file
, _cur_line
, msg
);
52 [[noreturn
]] void StrgenFatalI(const std::string
&msg
)
54 fmt::print(stderr
, LINE_NUM_FMT("FATAL"), _file
, _cur_line
, msg
);
56 fmt::print(stderr
, LINE_NUM_FMT("warning"), _file
, _cur_line
, "language is not compiled");
58 throw std::exception();
61 [[noreturn
]] void FatalErrorI(const std::string
&msg
)
63 fmt::print(stderr
, LINE_NUM_FMT("FATAL"), _file
, _cur_line
, msg
);
65 fmt::print(stderr
, LINE_NUM_FMT("warning"), _file
, _cur_line
, "language is not compiled");
70 /** A reader that simply reads using fopen. */
71 struct FileStringReader
: StringReader
{
72 std::ifstream input_stream
;
76 * @param data The data to fill during reading.
77 * @param file The file we are reading.
78 * @param master Are we reading the master file?
79 * @param translation Are we reading a translation?
81 FileStringReader(StringData
&data
, const std::filesystem::path
&file
, bool master
, bool translation
) :
82 StringReader(data
, file
.generic_string(), master
, translation
)
84 this->input_stream
.open(file
, std::ifstream::binary
);
87 std::optional
<std::string
> ReadLine() override
90 if (!std::getline(this->input_stream
, result
)) return std::nullopt
;
94 void HandlePragma(char *str
) override
;
96 void ParseFile() override
98 this->StringReader::ParseFile();
100 if (StrEmpty(_lang
.name
) || StrEmpty(_lang
.own_name
) || StrEmpty(_lang
.isocode
)) {
101 FatalError("Language must include ##name, ##ownname and ##isocode");
106 void FileStringReader::HandlePragma(char *str
)
108 if (!memcmp(str
, "id ", 3)) {
109 this->data
.next_string_id
= std::strtoul(str
+ 3, nullptr, 0);
110 } else if (!memcmp(str
, "name ", 5)) {
111 strecpy(_lang
.name
, str
+ 5);
112 } else if (!memcmp(str
, "ownname ", 8)) {
113 strecpy(_lang
.own_name
, str
+ 8);
114 } else if (!memcmp(str
, "isocode ", 8)) {
115 strecpy(_lang
.isocode
, str
+ 8);
116 } else if (!memcmp(str
, "textdir ", 8)) {
117 if (!memcmp(str
+ 8, "ltr", 3)) {
118 _lang
.text_dir
= TD_LTR
;
119 } else if (!memcmp(str
+ 8, "rtl", 3)) {
120 _lang
.text_dir
= TD_RTL
;
122 FatalError("Invalid textdir {}", str
+ 8);
124 } else if (!memcmp(str
, "digitsep ", 9)) {
126 strecpy(_lang
.digit_group_separator
, strcmp(str
, "{NBSP}") == 0 ? NBSP
: str
);
127 } else if (!memcmp(str
, "digitsepcur ", 12)) {
129 strecpy(_lang
.digit_group_separator_currency
, strcmp(str
, "{NBSP}") == 0 ? NBSP
: str
);
130 } else if (!memcmp(str
, "decimalsep ", 11)) {
132 strecpy(_lang
.digit_decimal_separator
, strcmp(str
, "{NBSP}") == 0 ? NBSP
: str
);
133 } else if (!memcmp(str
, "winlangid ", 10)) {
134 const char *buf
= str
+ 10;
135 long langid
= std::strtol(buf
, nullptr, 16);
136 if (langid
> (long)UINT16_MAX
|| langid
< 0) {
137 FatalError("Invalid winlangid {}", buf
);
139 _lang
.winlangid
= (uint16_t)langid
;
140 } else if (!memcmp(str
, "grflangid ", 10)) {
141 const char *buf
= str
+ 10;
142 long langid
= std::strtol(buf
, nullptr, 16);
143 if (langid
>= 0x7F || langid
< 0) {
144 FatalError("Invalid grflangid {}", buf
);
146 _lang
.newgrflangid
= (uint8_t)langid
;
147 } else if (!memcmp(str
, "gender ", 7)) {
148 if (this->master
) FatalError("Genders are not allowed in the base translation.");
152 const char *s
= ParseWord(&buf
);
154 if (s
== nullptr) break;
155 if (_lang
.num_genders
>= MAX_NUM_GENDERS
) FatalError("Too many genders, max {}", MAX_NUM_GENDERS
);
156 strecpy(_lang
.genders
[_lang
.num_genders
], s
);
159 } else if (!memcmp(str
, "case ", 5)) {
160 if (this->master
) FatalError("Cases are not allowed in the base translation.");
164 const char *s
= ParseWord(&buf
);
166 if (s
== nullptr) break;
167 if (_lang
.num_cases
>= MAX_NUM_CASES
) FatalError("Too many cases, max {}", MAX_NUM_CASES
);
168 strecpy(_lang
.cases
[_lang
.num_cases
], s
);
172 StringReader::HandlePragma(str
);
176 bool CompareFiles(const std::filesystem::path
&path1
, const std::filesystem::path
&path2
)
178 /* Check for equal size, but ignore the error code for cases when a file does not exist. */
179 std::error_code error_code
;
180 if (std::filesystem::file_size(path1
, error_code
) != std::filesystem::file_size(path2
, error_code
)) return false;
182 std::ifstream
stream1(path1
, std::ifstream::binary
);
183 std::ifstream
stream2(path2
, std::ifstream::binary
);
185 return std::equal(std::istreambuf_iterator
<char>(stream1
.rdbuf()),
186 std::istreambuf_iterator
<char>(),
187 std::istreambuf_iterator
<char>(stream2
.rdbuf()));
190 /** Base class for writing data to disk. */
192 std::ofstream output_stream
; ///< The stream to write all the output to.
193 const std::filesystem::path path
; ///< The file name we're writing to.
196 * Open a file to write to.
197 * @param path The path to the file to open.
198 * @param openmode The openmode flags for opening the file.
200 FileWriter(const std::filesystem::path
&path
, std::ios_base::openmode openmode
) : path(path
)
202 this->output_stream
.open(path
, openmode
);
205 /** Finalise the writing. */
208 this->output_stream
.close();
211 /** Make sure the file is closed. */
212 virtual ~FileWriter()
214 /* If we weren't closed an exception was thrown, so remove the temporary file. */
215 if (this->output_stream
.is_open()) {
216 this->output_stream
.close();
217 std::filesystem::remove(this->path
);
222 struct HeaderFileWriter
: HeaderWriter
, FileWriter
{
223 /** The real path we eventually want to write to. */
224 const std::filesystem::path real_path
;
225 /** The previous string ID that was printed. */
230 * Open a file to write to.
231 * @param path The path to the file to open.
233 HeaderFileWriter(const std::filesystem::path
&path
) : FileWriter("tmp.xxx", std::ofstream::out
),
234 real_path(path
), prev(0), total_strings(0)
236 this->output_stream
<< "/* This file is automatically generated. Do not modify */\n\n";
237 this->output_stream
<< "#ifndef TABLE_STRINGS_H\n";
238 this->output_stream
<< "#define TABLE_STRINGS_H\n";
241 void WriteStringID(const std::string
&name
, int stringid
) override
243 if (prev
+ 1 != stringid
) this->output_stream
<< "\n";
244 fmt::print(this->output_stream
, "static const StringID {} = 0x{:X};\n", name
, stringid
);
249 void Finalise(const StringData
&data
) override
251 /* Find the plural form with the most amount of cases. */
252 int max_plural_forms
= 0;
253 for (const auto &pf
: _plural_forms
) {
254 max_plural_forms
= std::max(max_plural_forms
, pf
.plural_count
);
257 fmt::print(this->output_stream
,
259 "static const uint LANGUAGE_PACK_VERSION = 0x{:X};\n"
260 "static const uint LANGUAGE_MAX_PLURAL = {};\n"
261 "static const uint LANGUAGE_MAX_PLURAL_FORMS = {};\n"
262 "static const uint LANGUAGE_TOTAL_STRINGS = {};\n"
264 data
.Version(), std::size(_plural_forms
), max_plural_forms
, total_strings
267 this->output_stream
<< "#endif /* TABLE_STRINGS_H */\n";
269 this->FileWriter::Finalise();
271 std::error_code error_code
;
272 if (CompareFiles(this->path
, this->real_path
)) {
273 /* files are equal. tmp.xxx is not needed */
274 std::filesystem::remove(this->path
, error_code
); // Just ignore the error
276 /* else rename tmp.xxx into filename */
277 std::filesystem::rename(this->path
, this->real_path
, error_code
);
278 if (error_code
) FatalError("rename({}, {}) failed: {}", this->path
, this->real_path
, error_code
.message());
283 /** Class for writing a language to disk. */
284 struct LanguageFileWriter
: LanguageWriter
, FileWriter
{
286 * Open a file to write to.
287 * @param path The path to the file to open.
289 LanguageFileWriter(const std::filesystem::path
&path
) : FileWriter(path
, std::ofstream::binary
| std::ofstream::out
)
293 void WriteHeader(const LanguagePackHeader
*header
) override
295 this->Write((const uint8_t *)header
, sizeof(*header
));
298 void Finalise() override
300 this->output_stream
.put(0);
301 this->FileWriter::Finalise();
304 void Write(const uint8_t *buffer
, size_t length
) override
306 this->output_stream
.write((const char *)buffer
, length
);
310 /** Options of strgen. */
311 static const OptionData _opts
[] = {
312 { .type
= ODF_NO_VALUE
, .id
= 'C', .longname
= "-export-commands" },
313 { .type
= ODF_NO_VALUE
, .id
= 'L', .longname
= "-export-plurals" },
314 { .type
= ODF_NO_VALUE
, .id
= 'P', .longname
= "-export-pragmas" },
315 { .type
= ODF_NO_VALUE
, .id
= 't', .shortname
= 't', .longname
= "--todo" },
316 { .type
= ODF_NO_VALUE
, .id
= 'w', .shortname
= 'w', .longname
= "--warning" },
317 { .type
= ODF_NO_VALUE
, .id
= 'h', .shortname
= 'h', .longname
= "--help" },
318 { .type
= ODF_NO_VALUE
, .id
= 'h', .shortname
= '?' },
319 { .type
= ODF_HAS_VALUE
, .id
= 's', .shortname
= 's', .longname
= "--source_dir" },
320 { .type
= ODF_HAS_VALUE
, .id
= 'd', .shortname
= 'd', .longname
= "--dest_dir" },
323 int CDECL
main(int argc
, char *argv
[])
325 std::filesystem::path
src_dir(".");
326 std::filesystem::path dest_dir
;
328 GetOptData
mgo(std::span(argv
+ 1, argc
- 1), _opts
);
330 int i
= mgo
.GetOpt();
335 fmt::print("args\tflags\tcommand\treplacement\n");
336 for (const auto &cs
: _cmd_structs
) {
338 if (cs
.proc
== EmitGender
) {
339 flags
= 'g'; // Command needs number of parameters defined by number of genders
340 } else if (cs
.proc
== EmitPlural
) {
341 flags
= 'p'; // Command needs number of parameters defined by plural value
342 } else if (cs
.flags
& C_DONTCOUNT
) {
343 flags
= 'i'; // Command may be in the translation when it is not in base
345 flags
= '0'; // Command needs no parameters
347 fmt::print("{}\t{:c}\t\"{}\"\t\"{}\"\n", cs
.consumes
, flags
, cs
.cmd
, strstr(cs
.cmd
, "STRING") ? "STRING" : cs
.cmd
);
352 fmt::print("count\tdescription\tnames\n");
353 for (const auto &pf
: _plural_forms
) {
354 fmt::print("{}\t\"{}\"\t{}\n", pf
.plural_count
, pf
.description
, pf
.names
);
359 fmt::print("name\tflags\tdefault\tdescription\n");
360 for (const auto &pragma
: _pragmas
) {
361 fmt::print("\"{}\"\t{}\t\"{}\"\t\"{}\"\n",
362 pragma
[0], pragma
[1], pragma
[2], pragma
[3]);
377 " -t | --todo replace any untranslated strings with '<TODO>'\n"
378 " -w | --warning print a warning for any untranslated strings\n"
379 " -h | -? | --help print this help message and exit\n"
380 " -s | --source_dir search for english.txt in the specified directory\n"
381 " -d | --dest_dir put output file in the specified directory, create if needed\n"
382 " -export-commands export all commands and exit\n"
383 " -export-plurals export all plural forms and exit\n"
384 " -export-pragmas export all pragmas and exit\n"
385 " Run without parameters and strgen will search for english.txt and parse it,\n"
386 " creating strings.h. Passing an argument, strgen will translate that language\n"
387 " file using english.txt as a reference and output <language>.lng.\n"
400 fmt::print(stderr
, "Invalid arguments\n");
405 if (dest_dir
.empty()) dest_dir
= src_dir
; // if dest_dir is not specified, it equals src_dir
408 /* strgen has two modes of operation. If no (free) arguments are passed
409 * strgen generates strings.h to the destination directory. If it is supplied
410 * with a (free) parameter the program will translate that language to destination
411 * directory. As input english.txt is parsed from the source directory */
412 if (mgo
.arguments
.empty()) {
413 std::filesystem::path input_path
= src_dir
;
414 input_path
/= "english.txt";
416 /* parse master file */
417 StringData
data(TEXT_TAB_END
);
418 FileStringReader
master_reader(data
, input_path
, true, false);
419 master_reader
.ParseFile();
420 if (_errors
!= 0) return 1;
422 /* write strings.h */
423 std::filesystem::path output_path
= dest_dir
;
424 std::filesystem::create_directories(dest_dir
);
425 output_path
/= "strings.h";
427 HeaderFileWriter
writer(output_path
);
428 writer
.WriteHeader(data
);
429 writer
.Finalise(data
);
430 if (_errors
!= 0) return 1;
432 std::filesystem::path input_path
= src_dir
;
433 input_path
/= "english.txt";
435 StringData
data(TEXT_TAB_END
);
436 /* parse master file and check if target file is correct */
437 FileStringReader
master_reader(data
, input_path
, true, false);
438 master_reader
.ParseFile();
440 for (auto &argument
: mgo
.arguments
) {
441 data
.FreeTranslation();
443 std::filesystem::path lang_file
= argument
;
444 FileStringReader
translation_reader(data
, lang_file
, false, lang_file
.filename() != "english.txt");
445 translation_reader
.ParseFile(); // target file
446 if (_errors
!= 0) return 1;
448 /* get the targetfile, strip any directories and append to destination path */
449 std::filesystem::path output_file
= dest_dir
;
450 output_file
/= lang_file
.filename();
451 output_file
.replace_extension("lng");
453 LanguageFileWriter
writer(output_file
);
454 writer
.WriteLang(data
);
457 /* if showing warnings, print a summary of the language */
458 if ((_show_todo
& 2) != 0) {
459 fmt::print("{} warnings and {} errors for {}\n", _warnings
, _errors
, output_file
);