2014-02-14 12:38:30 +00:00
|
|
|
#include "scriptcheck.hpp"
|
|
|
|
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <exception>
|
2022-06-27 19:32:46 +00:00
|
|
|
#include <sstream>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include <apps/opencs/model/doc/messages.hpp>
|
|
|
|
#include <apps/opencs/model/prefs/category.hpp>
|
|
|
|
#include <apps/opencs/model/prefs/setting.hpp>
|
|
|
|
#include <apps/opencs/model/world/idcollection.hpp>
|
|
|
|
#include <apps/opencs/model/world/record.hpp>
|
|
|
|
#include <apps/opencs/model/world/scriptcontext.hpp>
|
|
|
|
#include <apps/opencs/model/world/universalid.hpp>
|
2022-06-27 19:32:46 +00:00
|
|
|
|
2014-02-14 12:38:30 +00:00
|
|
|
#include <components/compiler/exception.hpp>
|
|
|
|
#include <components/compiler/extensions0.hpp>
|
|
|
|
#include <components/compiler/fileparser.hpp>
|
|
|
|
#include <components/compiler/scanner.hpp>
|
|
|
|
#include <components/compiler/tokenloc.hpp>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <components/esm3/loadscpt.hpp>
|
2014-02-14 12:38:30 +00:00
|
|
|
|
2014-07-21 10:15:21 +00:00
|
|
|
#include "../doc/document.hpp"
|
|
|
|
|
2014-02-14 12:38:30 +00:00
|
|
|
#include "../world/data.hpp"
|
|
|
|
|
2015-12-15 11:44:04 +00:00
|
|
|
#include "../prefs/state.hpp"
|
|
|
|
|
2015-06-20 17:04:19 +00:00
|
|
|
CSMDoc::Message::Severity CSMTools::ScriptCheckStage::getSeverity(Type type)
|
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case WarningMessage:
|
|
|
|
return CSMDoc::Message::Severity_Warning;
|
|
|
|
case ErrorMessage:
|
|
|
|
return CSMDoc::Message::Severity_Error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CSMDoc::Message::Severity_SeriousError;
|
|
|
|
}
|
|
|
|
|
2014-02-14 12:38:30 +00:00
|
|
|
void CSMTools::ScriptCheckStage::report(const std::string& message, const Compiler::TokenLoc& loc, Type type)
|
|
|
|
{
|
|
|
|
std::ostringstream stream;
|
|
|
|
|
|
|
|
CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Script, mId);
|
|
|
|
|
2019-03-05 17:47:19 +00:00
|
|
|
stream << message << " (" << loc.mLiteral << ")"
|
|
|
|
<< " @ line " << loc.mLine + 1 << ", column " << loc.mColumn;
|
2014-02-14 12:38:30 +00:00
|
|
|
|
2014-12-08 11:29:23 +00:00
|
|
|
std::ostringstream hintStream;
|
|
|
|
|
2019-03-05 17:47:19 +00:00
|
|
|
hintStream << "l:" << loc.mLine + 1 << " " << loc.mColumn;
|
2014-12-08 11:29:23 +00:00
|
|
|
|
2015-06-20 17:04:19 +00:00
|
|
|
mMessages->add(id, stream.str(), hintStream.str(), getSeverity(type));
|
2014-02-14 12:38:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSMTools::ScriptCheckStage::report(const std::string& message, Type type)
|
|
|
|
{
|
|
|
|
CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Script, mId);
|
|
|
|
|
2015-06-20 17:04:19 +00:00
|
|
|
std::ostringstream stream;
|
2018-08-24 18:02:42 +00:00
|
|
|
stream << message;
|
2015-12-15 11:44:04 +00:00
|
|
|
|
2015-06-20 17:04:19 +00:00
|
|
|
mMessages->add(id, stream.str(), "", getSeverity(type));
|
2014-02-14 12:38:30 +00:00
|
|
|
}
|
|
|
|
|
2014-07-21 10:15:21 +00:00
|
|
|
CSMTools::ScriptCheckStage::ScriptCheckStage(const CSMDoc::Document& document)
|
2020-11-13 07:39:47 +00:00
|
|
|
: mDocument(document)
|
|
|
|
, mContext(document.getData())
|
|
|
|
, mMessages(nullptr)
|
|
|
|
, mWarningMode(Mode_Ignore)
|
2014-02-14 12:38:30 +00:00
|
|
|
{
|
2014-02-15 11:58:34 +00:00
|
|
|
/// \todo add an option to configure warning mode
|
|
|
|
setWarningsMode(0);
|
|
|
|
|
2014-02-14 12:38:30 +00:00
|
|
|
Compiler::registerExtensions(mExtensions);
|
|
|
|
mContext.setExtensions(&mExtensions);
|
2018-06-19 22:20:03 +00:00
|
|
|
|
2018-06-20 09:29:38 +00:00
|
|
|
mIgnoreBaseRecords = false;
|
2014-02-14 12:38:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSMTools::ScriptCheckStage::setup()
|
|
|
|
{
|
2015-12-15 11:44:04 +00:00
|
|
|
std::string warnings = CSMPrefs::get()["Scripts"]["warnings"].toString();
|
|
|
|
|
|
|
|
if (warnings == "Ignore")
|
|
|
|
mWarningMode = Mode_Ignore;
|
|
|
|
else if (warnings == "Normal")
|
|
|
|
mWarningMode = Mode_Normal;
|
|
|
|
else if (warnings == "Strict")
|
|
|
|
mWarningMode = Mode_Strict;
|
|
|
|
|
2014-02-14 12:38:30 +00:00
|
|
|
mContext.clear();
|
2020-11-13 07:39:47 +00:00
|
|
|
mMessages = nullptr;
|
2023-02-17 18:20:29 +00:00
|
|
|
mId = ESM::RefId();
|
2015-06-20 14:21:04 +00:00
|
|
|
Compiler::ErrorHandler::reset();
|
2014-02-14 12:38:30 +00:00
|
|
|
|
2018-06-19 22:20:03 +00:00
|
|
|
mIgnoreBaseRecords = CSMPrefs::get()["Reports"]["ignore-base-records"].isTrue();
|
|
|
|
|
2014-07-21 10:15:21 +00:00
|
|
|
return mDocument.getData().getScripts().getSize();
|
2014-02-14 12:38:30 +00:00
|
|
|
}
|
|
|
|
|
2014-12-07 17:57:47 +00:00
|
|
|
void CSMTools::ScriptCheckStage::perform(int stage, CSMDoc::Messages& messages)
|
2014-02-14 12:38:30 +00:00
|
|
|
{
|
2018-06-19 22:20:03 +00:00
|
|
|
const CSMWorld::Record<ESM::Script>& record = mDocument.getData().getScripts().getRecord(stage);
|
|
|
|
|
2014-07-21 10:15:21 +00:00
|
|
|
mId = mDocument.getData().getScripts().getId(stage);
|
|
|
|
|
|
|
|
if (mDocument.isBlacklisted(CSMWorld::UniversalId(CSMWorld::UniversalId::Type_Script, mId)))
|
|
|
|
return;
|
|
|
|
|
2018-06-19 22:20:03 +00:00
|
|
|
// Skip "Base" records (setting!) and "Deleted" records
|
2018-06-20 09:29:38 +00:00
|
|
|
if ((mIgnoreBaseRecords && record.mState == CSMWorld::RecordBase::State_BaseOnly) || record.isDeleted())
|
2018-06-19 22:20:03 +00:00
|
|
|
return;
|
|
|
|
|
2014-02-14 12:38:30 +00:00
|
|
|
mMessages = &messages;
|
|
|
|
|
2015-06-20 14:21:04 +00:00
|
|
|
switch (mWarningMode)
|
|
|
|
{
|
|
|
|
case Mode_Ignore:
|
|
|
|
setWarningsMode(0);
|
|
|
|
break;
|
2015-06-20 17:16:15 +00:00
|
|
|
case Mode_Normal:
|
|
|
|
setWarningsMode(1);
|
|
|
|
break;
|
|
|
|
case Mode_Strict:
|
|
|
|
setWarningsMode(2);
|
|
|
|
break;
|
2015-06-20 14:21:04 +00:00
|
|
|
}
|
|
|
|
|
2014-02-14 12:38:30 +00:00
|
|
|
try
|
|
|
|
{
|
2022-10-06 17:39:46 +00:00
|
|
|
mFile = record.get().mId.getRefIdString();
|
2018-06-19 22:20:03 +00:00
|
|
|
std::istringstream input(record.get().mScriptText);
|
2014-02-14 12:38:30 +00:00
|
|
|
|
|
|
|
Compiler::Scanner scanner(*this, input, mContext.getExtensions());
|
|
|
|
|
|
|
|
Compiler::FileParser parser(*this, mContext);
|
|
|
|
|
|
|
|
scanner.scan(parser);
|
|
|
|
}
|
|
|
|
catch (const Compiler::SourceException&)
|
|
|
|
{
|
|
|
|
// error has already been reported via error handler
|
|
|
|
}
|
|
|
|
catch (const std::exception& error)
|
|
|
|
{
|
|
|
|
CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Script, mId);
|
|
|
|
|
2015-06-20 17:04:19 +00:00
|
|
|
std::ostringstream stream;
|
2018-08-24 18:02:42 +00:00
|
|
|
stream << error.what();
|
2015-12-15 11:44:04 +00:00
|
|
|
|
2015-06-20 17:04:19 +00:00
|
|
|
messages.add(id, stream.str(), "", CSMDoc::Message::Severity_SeriousError);
|
2014-02-14 12:38:30 +00:00
|
|
|
}
|
|
|
|
|
2020-11-13 07:39:47 +00:00
|
|
|
mMessages = nullptr;
|
2015-03-11 14:54:45 +00:00
|
|
|
}
|