2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2014-12-11 21:12:34 +01:00
|
|
|
|
2014-12-12 15:33:02 +01:00
|
|
|
#include "base/scriptframe.hpp"
|
2015-03-29 00:03:47 +01:00
|
|
|
#include "base/scriptglobal.hpp"
|
2018-08-07 13:55:41 +02:00
|
|
|
#include "base/namespace.hpp"
|
2015-09-23 09:21:45 +02:00
|
|
|
#include "base/exception.hpp"
|
2018-08-09 15:37:23 +02:00
|
|
|
#include "base/configuration.hpp"
|
2023-01-09 17:09:46 +01:00
|
|
|
#include "base/utility.hpp"
|
2014-12-11 21:12:34 +01:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2015-03-29 22:26:07 +02:00
|
|
|
boost::thread_specific_ptr<std::stack<ScriptFrame *> > ScriptFrame::m_ScriptFrames;
|
2018-08-07 13:55:41 +02:00
|
|
|
|
2023-01-09 17:09:46 +01:00
|
|
|
static Namespace::Ptr l_SystemNS, l_TypesNS, l_StatsNS, l_InternalNS;
|
2014-12-22 14:14:16 +01:00
|
|
|
|
2018-09-04 15:17:34 +02:00
|
|
|
/* Ensure that this gets called with highest priority
|
|
|
|
* and wins against other static initializers in lib/icinga, etc.
|
|
|
|
* LTO-enabled builds will cause trouble otherwise, see GH #6575.
|
|
|
|
*/
|
2016-08-27 09:35:08 +02:00
|
|
|
INITIALIZE_ONCE_WITH_PRIORITY([]() {
|
2018-08-07 13:55:41 +02:00
|
|
|
Namespace::Ptr globalNS = ScriptGlobal::GetGlobals();
|
|
|
|
|
2023-01-09 17:09:46 +01:00
|
|
|
l_SystemNS = new Namespace(true);
|
|
|
|
l_SystemNS->Set("PlatformKernel", Utility::GetPlatformKernel());
|
|
|
|
l_SystemNS->Set("PlatformKernelVersion", Utility::GetPlatformKernelVersion());
|
|
|
|
l_SystemNS->Set("PlatformName", Utility::GetPlatformName());
|
|
|
|
l_SystemNS->Set("PlatformVersion", Utility::GetPlatformVersion());
|
|
|
|
l_SystemNS->Set("PlatformArchitecture", Utility::GetPlatformArchitecture());
|
|
|
|
l_SystemNS->Set("BuildHostName", ICINGA_BUILD_HOST_NAME);
|
|
|
|
l_SystemNS->Set("BuildCompilerName", ICINGA_BUILD_COMPILER_NAME);
|
|
|
|
l_SystemNS->Set("BuildCompilerVersion", ICINGA_BUILD_COMPILER_VERSION);
|
2023-01-09 16:12:59 +01:00
|
|
|
globalNS->Set("System", l_SystemNS, true);
|
2016-08-12 15:36:47 +02:00
|
|
|
|
2023-01-09 16:12:59 +01:00
|
|
|
l_SystemNS->Set("Configuration", new Configuration());
|
2018-08-09 15:37:23 +02:00
|
|
|
|
2023-01-09 17:09:46 +01:00
|
|
|
l_TypesNS = new Namespace(true);
|
2023-01-09 16:12:59 +01:00
|
|
|
globalNS->Set("Types", l_TypesNS, true);
|
2016-08-12 16:53:44 +02:00
|
|
|
|
2023-01-09 17:09:46 +01:00
|
|
|
l_StatsNS = new Namespace(true);
|
2023-01-09 16:12:59 +01:00
|
|
|
globalNS->Set("StatsFunctions", l_StatsNS, true);
|
2018-08-07 13:55:41 +02:00
|
|
|
|
2022-12-07 11:18:28 +01:00
|
|
|
l_InternalNS = new Namespace(true);
|
2023-01-09 16:12:59 +01:00
|
|
|
globalNS->Set("Internal", l_InternalNS, true);
|
2022-12-12 12:32:59 +01:00
|
|
|
}, InitializePriority::CreateNamespaces);
|
2016-08-12 15:36:47 +02:00
|
|
|
|
2023-01-09 17:09:46 +01:00
|
|
|
INITIALIZE_ONCE_WITH_PRIORITY([]() {
|
|
|
|
l_SystemNS->Freeze();
|
|
|
|
l_TypesNS->Freeze();
|
|
|
|
l_StatsNS->Freeze();
|
2022-12-07 11:18:28 +01:00
|
|
|
l_InternalNS->Freeze();
|
2023-01-09 17:09:46 +01:00
|
|
|
}, InitializePriority::FreezeNamespaces);
|
2018-08-07 13:55:41 +02:00
|
|
|
|
2017-12-18 10:30:20 +01:00
|
|
|
ScriptFrame::ScriptFrame(bool allocLocals)
|
|
|
|
: Locals(allocLocals ? new Dictionary() : nullptr), Self(ScriptGlobal::GetGlobals()), Sandboxed(false), Depth(0)
|
2016-08-12 11:42:59 +02:00
|
|
|
{
|
|
|
|
InitializeFrame();
|
|
|
|
}
|
|
|
|
|
2018-01-04 08:54:18 +01:00
|
|
|
ScriptFrame::ScriptFrame(bool allocLocals, Value self)
|
|
|
|
: Locals(allocLocals ? new Dictionary() : nullptr), Self(std::move(self)), Sandboxed(false), Depth(0)
|
2016-08-12 11:42:59 +02:00
|
|
|
{
|
|
|
|
InitializeFrame();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void ScriptFrame::InitializeFrame()
|
2014-12-22 14:14:16 +01:00
|
|
|
{
|
2016-07-29 14:11:52 +02:00
|
|
|
std::stack<ScriptFrame *> *frames = m_ScriptFrames.get();
|
|
|
|
|
2016-08-12 11:25:36 +02:00
|
|
|
if (frames && !frames->empty()) {
|
|
|
|
ScriptFrame *frame = frames->top();
|
|
|
|
|
|
|
|
Sandboxed = frame->Sandboxed;
|
|
|
|
}
|
2016-07-29 14:11:52 +02:00
|
|
|
|
2015-03-29 22:26:07 +02:00
|
|
|
PushFrame(this);
|
2014-12-22 14:14:16 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
ScriptFrame::~ScriptFrame()
|
2014-12-22 14:14:16 +01:00
|
|
|
{
|
2015-03-29 22:26:07 +02:00
|
|
|
ScriptFrame *frame = PopFrame();
|
|
|
|
ASSERT(frame == this);
|
2019-03-08 14:07:29 +01:00
|
|
|
|
|
|
|
#ifndef I2_DEBUG
|
|
|
|
(void)frame;
|
|
|
|
#endif /* I2_DEBUG */
|
2014-12-22 14:14:16 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void ScriptFrame::IncreaseStackDepth()
|
2016-03-23 08:40:32 +01:00
|
|
|
{
|
|
|
|
if (Depth + 1 > 300)
|
|
|
|
BOOST_THROW_EXCEPTION(ScriptError("Stack overflow while evaluating expression: Recursion level too deep."));
|
|
|
|
|
|
|
|
Depth++;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void ScriptFrame::DecreaseStackDepth()
|
2016-03-23 08:40:32 +01:00
|
|
|
{
|
|
|
|
Depth--;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
ScriptFrame *ScriptFrame::GetCurrentFrame()
|
2014-12-22 14:14:16 +01:00
|
|
|
{
|
2015-03-29 22:26:07 +02:00
|
|
|
std::stack<ScriptFrame *> *frames = m_ScriptFrames.get();
|
2014-12-22 14:14:16 +01:00
|
|
|
|
2015-03-29 22:26:07 +02:00
|
|
|
ASSERT(!frames->empty());
|
|
|
|
return frames->top();
|
2014-12-22 14:14:16 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
ScriptFrame *ScriptFrame::PopFrame()
|
2014-12-22 14:14:16 +01:00
|
|
|
{
|
2015-03-29 22:26:07 +02:00
|
|
|
std::stack<ScriptFrame *> *frames = m_ScriptFrames.get();
|
|
|
|
|
|
|
|
ASSERT(!frames->empty());
|
|
|
|
|
|
|
|
ScriptFrame *frame = frames->top();
|
|
|
|
frames->pop();
|
|
|
|
|
|
|
|
return frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptFrame::PushFrame(ScriptFrame *frame)
|
|
|
|
{
|
|
|
|
std::stack<ScriptFrame *> *frames = m_ScriptFrames.get();
|
|
|
|
|
|
|
|
if (!frames) {
|
|
|
|
frames = new std::stack<ScriptFrame *>();
|
|
|
|
m_ScriptFrames.reset(frames);
|
|
|
|
}
|
|
|
|
|
2016-03-23 08:40:32 +01:00
|
|
|
if (!frames->empty()) {
|
|
|
|
ScriptFrame *parent = frames->top();
|
|
|
|
frame->Depth += parent->Depth;
|
|
|
|
}
|
2015-09-23 09:21:45 +02:00
|
|
|
|
2015-03-29 22:26:07 +02:00
|
|
|
frames->push(frame);
|
2015-03-29 00:03:47 +01:00
|
|
|
}
|