1 /******************************************************************************
3 * Copyright (C) 2012-2017 Icinga Development Team (https://www.icinga.com/) *
5 * This program is free software; you can redistribute it and/or *
6 * modify it under the terms of the GNU General Public License *
7 * as published by the Free Software Foundation; either version 2 *
8 * of the License, or (at your option) any later version. *
10 * This program is distributed in the hope that it will be useful, *
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
13 * GNU General Public License for more details. *
15 * You should have received a copy of the GNU General Public License *
16 * along with this program; if not, write to the Free Software Foundation *
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
18 ******************************************************************************/
23 #include "config/i2-config.hpp"
24 #include "config/expression.hpp"
25 #include "config/configitembuilder.hpp"
26 #include "config/applyrule.hpp"
27 #include "config/objectrule.hpp"
28 #include "base/debuginfo.hpp"
29 #include "base/array.hpp"
30 #include "base/dictionary.hpp"
31 #include "base/function.hpp"
32 #include "base/scriptglobal.hpp"
33 #include "base/exception.hpp"
34 #include "base/convert.hpp"
35 #include "base/objectlock.hpp"
36 #include <boost/smart_ptr/make_shared.hpp>
46 static inline bool FindVarImportRef(ScriptFrame& frame, const String& name, Value *result, const DebugInfo& debugInfo = DebugInfo())
48 Array::Ptr imports = ScriptFrame::GetImports();
50 ObjectLock olock(imports);
51 for (const Value& import : imports) {
52 Object::Ptr obj = import;
53 if (obj->HasOwnField(name)) {
62 static inline bool FindVarImport(ScriptFrame& frame, const String& name, Value *result, const DebugInfo& debugInfo = DebugInfo())
66 if (FindVarImportRef(frame, name, &parent, debugInfo)) {
67 *result = GetField(parent, name, frame.Sandboxed, debugInfo);
74 static inline Value ConstructorCall(const Type::Ptr& type, const std::vector<Value>& args, const DebugInfo& debugInfo = DebugInfo())
76 if (type->GetName() == "String") {
79 else if (args.size() == 1)
80 return Convert::ToString(args[0]);
82 BOOST_THROW_EXCEPTION(ScriptError("Too many arguments for constructor."));
83 } else if (type->GetName() == "Number") {
86 else if (args.size() == 1)
87 return Convert::ToDouble(args[0]);
89 BOOST_THROW_EXCEPTION(ScriptError("Too many arguments for constructor."));
90 } else if (type->GetName() == "Boolean") {
93 else if (args.size() == 1)
94 return Convert::ToBool(args[0]);
96 BOOST_THROW_EXCEPTION(ScriptError("Too many arguments for constructor."));
97 } else if (args.size() == 1 && type->IsAssignableFrom(args[0].GetReflectionType()))
100 return type->Instantiate(args);
103 static inline Value FunctionCall(ScriptFrame& frame, const Value& self, const Function::Ptr& func, const std::vector<Value>& arguments)
105 if (!self.IsEmpty() || self.IsString())
106 return func->Invoke(self, arguments);
108 return func->Invoke(arguments);
112 static inline Value NewFunction(ScriptFrame& frame, const String& name, const std::vector<String>& args,
113 std::map<String, Expression *> *closedVars, const boost::shared_ptr<Expression>& expression)
115 return new Function(name, boost::bind(&FunctionWrapper, _1, args,
116 EvaluateClosedVars(frame, closedVars), expression), args);
119 static inline Value NewApply(ScriptFrame& frame, const String& type, const String& target, const String& name, const boost::shared_ptr<Expression>& filter,
120 const String& package, const String& fkvar, const String& fvvar, const boost::shared_ptr<Expression>& fterm, std::map<String, Expression *> *closedVars,
121 bool ignoreOnError, const boost::shared_ptr<Expression>& expression, const DebugInfo& debugInfo = DebugInfo())
123 ApplyRule::AddRule(type, target, name, expression, filter, package, fkvar,
124 fvvar, fterm, ignoreOnError, debugInfo, EvaluateClosedVars(frame, closedVars));
129 static inline Value NewObject(ScriptFrame& frame, bool abstract, const Type::Ptr& type, const String& name, const boost::shared_ptr<Expression>& filter,
130 const String& zone, const String& package, bool defaultTmpl, bool ignoreOnError, std::map<String, Expression *> *closedVars, const boost::shared_ptr<Expression>& expression, const DebugInfo& debugInfo = DebugInfo())
132 ConfigItemBuilder::Ptr item = new ConfigItemBuilder(debugInfo);
134 String checkName = name;
137 NameComposer *nc = dynamic_cast<NameComposer *>(type.get());
140 checkName = nc->MakeName(name, Dictionary::Ptr());
143 if (!checkName.IsEmpty()) {
144 ConfigItem::Ptr oldItem = ConfigItem::GetByTypeAndName(type, checkName);
147 std::ostringstream msgbuf;
148 msgbuf << "Object '" << name << "' of type '" << type->GetName() << "' re-defined: " << debugInfo << "; previous definition: " << oldItem->GetDebugInfo();
149 BOOST_THROW_EXCEPTION(ScriptError(msgbuf.str(), debugInfo));
153 if (filter && !ObjectRule::IsValidSourceType(type->GetName())) {
154 std::ostringstream msgbuf;
155 msgbuf << "Object '" << name << "' of type '" << type->GetName() << "' must not have 'assign where' and 'ignore where' rules: " << debugInfo;
156 BOOST_THROW_EXCEPTION(ScriptError(msgbuf.str(), debugInfo));
163 item->AddExpression(new ImportDefaultTemplatesExpression());
165 item->AddExpression(new OwnedExpression(expression));
166 item->SetAbstract(abstract);
167 item->SetScope(EvaluateClosedVars(frame, closedVars));
169 item->SetPackage(package);
170 item->SetFilter(filter);
171 item->SetDefaultTemplate(defaultTmpl);
172 item->SetIgnoreOnError(ignoreOnError);
173 item->Compile()->Register();
178 static inline ExpressionResult For(ScriptFrame& frame, const String& fkvar, const String& fvvar, const Value& value, Expression *expression, const DebugInfo& debugInfo = DebugInfo())
180 if (value.IsObjectType<Array>()) {
181 if (!fvvar.IsEmpty())
182 BOOST_THROW_EXCEPTION(ScriptError("Cannot use dictionary iterator for array.", debugInfo));
184 Array::Ptr arr = value;
186 for (Array::SizeType i = 0; i < arr->GetLength(); i++) {
187 frame.Locals->Set(fkvar, arr->Get(i));
188 ExpressionResult res = expression->Evaluate(frame);
189 CHECK_RESULT_LOOP(res);
191 } else if (value.IsObjectType<Dictionary>()) {
193 BOOST_THROW_EXCEPTION(ScriptError("Cannot use array iterator for dictionary.", debugInfo));
195 Dictionary::Ptr dict = value;
196 std::vector<String> keys;
199 ObjectLock olock(dict);
200 for (const Dictionary::Pair& kv : dict) {
201 keys.push_back(kv.first);
205 for (const String& key : keys) {
206 frame.Locals->Set(fkvar, key);
207 frame.Locals->Set(fvvar, dict->Get(key));
208 ExpressionResult res = expression->Evaluate(frame);
209 CHECK_RESULT_LOOP(res);
212 BOOST_THROW_EXCEPTION(ScriptError("Invalid type in for expression: " + value.GetTypeName(), debugInfo));
217 static inline Value GetField(const Value& context, const String& field, bool sandboxed = false, const DebugInfo& debugInfo = DebugInfo())
219 if (unlikely(context.IsEmpty() && !context.IsString()))
222 if (unlikely(!context.IsObject()))
223 return GetPrototypeField(context, field, true, debugInfo);
225 Object::Ptr object = context;
227 return object->GetFieldByName(field, sandboxed, debugInfo);
230 static inline void SetField(const Object::Ptr& context, const String& field, const Value& value, const DebugInfo& debugInfo = DebugInfo())
233 BOOST_THROW_EXCEPTION(ScriptError("Cannot set field '" + field + "' on a value that is not an object.", debugInfo));
235 return context->SetFieldByName(field, value, debugInfo);
239 static inline Value FunctionWrapper(const std::vector<Value>& arguments,
240 const std::vector<String>& funcargs, const Dictionary::Ptr& closedVars, const boost::shared_ptr<Expression>& expr)
242 if (arguments.size() < funcargs.size())
243 BOOST_THROW_EXCEPTION(std::invalid_argument("Too few arguments for function"));
245 ScriptFrame *frame = ScriptFrame::GetCurrentFrame();
248 closedVars->CopyTo(frame->Locals);
250 for (std::vector<Value>::size_type i = 0; i < std::min(arguments.size(), funcargs.size()); i++)
251 frame->Locals->Set(funcargs[i], arguments[i]);
253 return expr->Evaluate(*frame);
256 static inline Dictionary::Ptr EvaluateClosedVars(ScriptFrame& frame, std::map<String, Expression *> *closedVars)
258 Dictionary::Ptr locals;
261 locals = new Dictionary();
263 typedef std::pair<String, Expression *> ClosedVar;
264 for (const ClosedVar& cvar : *closedVars) {
265 locals->Set(cvar.first, cvar.second->Evaluate(frame));