Script: Allow text param mismatches for scripts with 13 or below compatibility
This commit is contained in:
@@ -13,6 +13,8 @@
|
|||||||
#include "../../game/game_text.hpp"
|
#include "../../game/game_text.hpp"
|
||||||
#include "script_text.hpp"
|
#include "script_text.hpp"
|
||||||
#include "../script_fatalerror.hpp"
|
#include "../script_fatalerror.hpp"
|
||||||
|
#include "../script_instance.hpp"
|
||||||
|
#include "script_log.hpp"
|
||||||
#include "../../table/control_codes.h"
|
#include "../../table/control_codes.h"
|
||||||
#include "../../3rdparty/fmt/format.h"
|
#include "../../3rdparty/fmt/format.h"
|
||||||
|
|
||||||
@@ -168,6 +170,15 @@ const std::string ScriptText::GetEncodedText()
|
|||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ScriptText::_TextParamError(const std::string &msg)
|
||||||
|
{
|
||||||
|
if (this->GetActiveInstance()->IsTextParamMismatchAllowed()) {
|
||||||
|
ScriptLog::Error(msg.c_str());
|
||||||
|
} else {
|
||||||
|
throw Script_FatalError(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
char *ScriptText::_GetEncodedText(char *p, char *lastofp, int ¶m_count, StringIDList &seen_ids)
|
char *ScriptText::_GetEncodedText(char *p, char *lastofp, int ¶m_count, StringIDList &seen_ids)
|
||||||
{
|
{
|
||||||
const std::string &name = GetGameStringName(this->string);
|
const std::string &name = GetGameStringName(this->string);
|
||||||
@@ -182,27 +193,45 @@ char *ScriptText::_GetEncodedText(char *p, char *lastofp, int ¶m_count, Stri
|
|||||||
int cur_idx = 0;
|
int cur_idx = 0;
|
||||||
|
|
||||||
for (const StringParam &cur_param : params) {
|
for (const StringParam &cur_param : params) {
|
||||||
if (cur_idx >= this->paramc) throw Script_FatalError(fmt::format("{}: Not enough parameters", name));
|
if (cur_idx >= this->paramc) {
|
||||||
|
this->_TextParamError(fmt::format("{}: Not enough parameters", name));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
switch (cur_param.type) {
|
switch (cur_param.type) {
|
||||||
case StringParam::RAW_STRING:
|
case StringParam::RAW_STRING:
|
||||||
if (!std::holds_alternative<std::string>(this->param[cur_idx])) throw Script_FatalError(fmt::format("{}: Parameter {} expects a raw string", name, cur_idx));
|
if (!std::holds_alternative<std::string>(this->param[cur_idx])) {
|
||||||
|
this->_TextParamError(fmt::format("{}: Parameter {} expects a raw string", name, cur_idx));
|
||||||
|
p += seprintf(p, lastofp, ":\"\"");
|
||||||
|
break;
|
||||||
|
}
|
||||||
p += seprintf(p, lastofp, ":\"%s\"", std::get<std::string>(this->param[cur_idx++]).c_str());
|
p += seprintf(p, lastofp, ":\"%s\"", std::get<std::string>(this->param[cur_idx++]).c_str());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case StringParam::STRING: {
|
case StringParam::STRING: {
|
||||||
if (!std::holds_alternative<ScriptTextRef>(this->param[cur_idx])) throw Script_FatalError(fmt::format("{}: Parameter {} expects a substring", name, cur_idx));
|
if (!std::holds_alternative<ScriptTextRef>(this->param[cur_idx])) {
|
||||||
|
this->_TextParamError(fmt::format("{}: Parameter {} expects a substring", name, cur_idx));
|
||||||
|
p += seprintf(p, lastofp, ":\"\"");
|
||||||
|
break;
|
||||||
|
}
|
||||||
int count = 1; // 1 because the string id is included in consumed parameters
|
int count = 1; // 1 because the string id is included in consumed parameters
|
||||||
p += seprintf(p, lastofp, ":");
|
p += seprintf(p, lastofp, ":");
|
||||||
p = std::get<ScriptTextRef>(this->param[cur_idx++])->_GetEncodedText(p, lastofp, count, seen_ids);
|
p = std::get<ScriptTextRef>(this->param[cur_idx++])->_GetEncodedText(p, lastofp, count, seen_ids);
|
||||||
if (count != cur_param.consumes) throw Script_FatalError(fmt::format("{}: Parameter {} substring consumes {}, but expected {} to be consumed", name, cur_idx, count - 1, cur_param.consumes - 1));
|
if (count != cur_param.consumes) this->_TextParamError(fmt::format("{}: Parameter {} substring consumes {}, but expected {} to be consumed", name, cur_idx, count - 1, cur_param.consumes - 1));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
if (cur_idx + cur_param.consumes > this->paramc) throw Script_FatalError(fmt::format("{}: Not enough parameters", name));
|
if (cur_idx + cur_param.consumes > this->paramc) {
|
||||||
|
this->_TextParamError(fmt::format("{}: Not enough parameters", name));
|
||||||
|
break;
|
||||||
|
}
|
||||||
for (int i = 0; i < cur_param.consumes; i++) {
|
for (int i = 0; i < cur_param.consumes; i++) {
|
||||||
if (!std::holds_alternative<SQInteger>(this->param[cur_idx])) throw Script_FatalError(fmt::format("{}: Parameter {} expects an integer", name, cur_idx));
|
if (!std::holds_alternative<SQInteger>(this->param[cur_idx])) {
|
||||||
|
this->_TextParamError(fmt::format("{}: Parameter {} expects an integer", name, cur_idx));
|
||||||
|
p += seprintf(p, lastofp, ":0");
|
||||||
|
break;
|
||||||
|
}
|
||||||
p += seprintf(p, lastofp,":" OTTD_PRINTFHEX64, std::get<SQInteger>(this->param[cur_idx++]));
|
p += seprintf(p, lastofp,":" OTTD_PRINTFHEX64, std::get<SQInteger>(this->param[cur_idx++]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -135,6 +135,8 @@ private:
|
|||||||
std::variant<SQInteger, std::string, ScriptTextRef> param[SCRIPT_TEXT_MAX_PARAMETERS];
|
std::variant<SQInteger, std::string, ScriptTextRef> param[SCRIPT_TEXT_MAX_PARAMETERS];
|
||||||
int paramc;
|
int paramc;
|
||||||
|
|
||||||
|
void _TextParamError(const std::string &msg);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Internal function for recursive calling this function over multiple
|
* Internal function for recursive calling this function over multiple
|
||||||
* instances, while writing in the same buffer.
|
* instances, while writing in the same buffer.
|
||||||
|
@@ -62,7 +62,8 @@ ScriptInstance::ScriptInstance(const char *APIName) :
|
|||||||
is_paused(false),
|
is_paused(false),
|
||||||
in_shutdown(false),
|
in_shutdown(false),
|
||||||
callback(nullptr),
|
callback(nullptr),
|
||||||
APIName(APIName)
|
APIName(APIName),
|
||||||
|
allow_text_param_mismatch(false)
|
||||||
{
|
{
|
||||||
this->storage = new ScriptStorage();
|
this->storage = new ScriptStorage();
|
||||||
this->engine = new Squirrel(APIName);
|
this->engine = new Squirrel(APIName);
|
||||||
@@ -122,6 +123,16 @@ void ScriptInstance::RegisterAPI()
|
|||||||
|
|
||||||
bool ScriptInstance::LoadCompatibilityScripts(const char *api_version, Subdirectory dir)
|
bool ScriptInstance::LoadCompatibilityScripts(const char *api_version, Subdirectory dir)
|
||||||
{
|
{
|
||||||
|
const char *api_vers[] = { "1.2", "1.3", "1.4", "1.5", "1.6", "1.7", "1.8", "1.9", "1.10", "1.11", "12", "13", "14" };
|
||||||
|
uint api_idx = 0;
|
||||||
|
for (; api_idx < lengthof(api_vers) ; api_idx++) {
|
||||||
|
if (strcmp(api_version, api_vers[api_idx]) == 0) break;
|
||||||
|
}
|
||||||
|
if (api_idx < 12) {
|
||||||
|
/* 13 and below */
|
||||||
|
this->allow_text_param_mismatch = true;
|
||||||
|
}
|
||||||
|
|
||||||
char script_name[32];
|
char script_name[32];
|
||||||
seprintf(script_name, lastof(script_name), "compat_%s.nut", api_version);
|
seprintf(script_name, lastof(script_name), "compat_%s.nut", api_version);
|
||||||
for (Searchpath sp : _valid_searchpaths) {
|
for (Searchpath sp : _valid_searchpaths) {
|
||||||
|
@@ -296,6 +296,7 @@ private:
|
|||||||
Script_SuspendCallbackProc *callback; ///< Callback that should be called in the next tick the script runs.
|
Script_SuspendCallbackProc *callback; ///< Callback that should be called in the next tick the script runs.
|
||||||
size_t last_allocated_memory; ///< Last known allocated memory value (for display for crashed scripts)
|
size_t last_allocated_memory; ///< Last known allocated memory value (for display for crashed scripts)
|
||||||
const char *APIName; ///< Name of the API used for this squirrel.
|
const char *APIName; ///< Name of the API used for this squirrel.
|
||||||
|
bool allow_text_param_mismatch; ///< Whether ScriptText parameter mismatches are allowed
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Call the script Load function if it exists and data was loaded
|
* Call the script Load function if it exists and data was loaded
|
||||||
@@ -322,6 +323,9 @@ private:
|
|||||||
static bool LoadObjects(ScriptData *data);
|
static bool LoadObjects(ScriptData *data);
|
||||||
|
|
||||||
static bool LoadObjects(HSQUIRRELVM vm, ScriptData *data);
|
static bool LoadObjects(HSQUIRRELVM vm, ScriptData *data);
|
||||||
|
|
||||||
|
public:
|
||||||
|
inline bool IsTextParamMismatchAllowed() const { return this->allow_text_param_mismatch; }
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* SCRIPT_INSTANCE_HPP */
|
#endif /* SCRIPT_INSTANCE_HPP */
|
||||||
|
Reference in New Issue
Block a user