1 /******************************************************************************
3 * Copyright (C) 2012-2016 Icinga Development Team (https://www.icinga.org/) *
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/foreach.hpp>
37 #include <boost/smart_ptr/make_shared.hpp>
47 static inline Value Variable(ScriptFrame& frame, const String& name, const DebugInfo& debugInfo = DebugInfo())
50 if (frame.Locals && frame.Locals->Get(name, &value))
52 else if (frame.Self.IsObject() && frame.Locals != static_cast<Object::Ptr>(frame.Self) && HasField(frame.Self, name))
53 return GetField(frame.Self, name, frame.Sandboxed, debugInfo);
55 return ScriptGlobal::Get(name);
58 static inline Value ConstructorCall(const Type::Ptr& type, const std::vector<Value>& args, const DebugInfo& debugInfo = DebugInfo())
60 if (type->GetName() == "String") {
63 else if (args.size() == 1)
64 return Convert::ToString(args[0]);
66 BOOST_THROW_EXCEPTION(ScriptError("Too many arguments for constructor."));
67 } else if (type->GetName() == "Number") {
70 else if (args.size() == 1)
71 return Convert::ToDouble(args[0]);
73 BOOST_THROW_EXCEPTION(ScriptError("Too many arguments for constructor."));
74 } else if (type->GetName() == "Boolean") {
77 else if (args.size() == 1)
78 return Convert::ToBool(args[0]);
80 BOOST_THROW_EXCEPTION(ScriptError("Too many arguments for constructor."));
81 } else if (args.size() == 1 && type->IsAssignableFrom(args[0].GetReflectionType()))
84 return type->Instantiate(args);
87 static inline Value FunctionCall(ScriptFrame& frame, const Value& self, const Function::Ptr& func, const std::vector<Value>& arguments)
91 if (!self.IsEmpty() || self.IsString())
94 return func->Invoke(arguments);
97 static inline Value NewFunction(ScriptFrame& frame, const std::vector<String>& args,
98 std::map<String, Expression *> *closedVars, const boost::shared_ptr<Expression>& expression)
100 return new Function(boost::bind(&FunctionWrapper, _1, args,
101 EvaluateClosedVars(frame, closedVars), expression));
104 static inline Value NewApply(ScriptFrame& frame, const String& type, const String& target, const String& name, const boost::shared_ptr<Expression>& filter,
105 const String& package, const String& fkvar, const String& fvvar, const boost::shared_ptr<Expression>& fterm, std::map<String, Expression *> *closedVars,
106 bool ignoreOnError, const boost::shared_ptr<Expression>& expression, const DebugInfo& debugInfo = DebugInfo())
108 ApplyRule::AddRule(type, target, name, expression, filter, package, fkvar,
109 fvvar, fterm, ignoreOnError, debugInfo, EvaluateClosedVars(frame, closedVars));
114 static inline Value NewObject(ScriptFrame& frame, bool abstract, const String& type, const String& name, const boost::shared_ptr<Expression>& filter,
115 const String& zone, const String& package, bool ignoreOnError, std::map<String, Expression *> *closedVars, const boost::shared_ptr<Expression>& expression, const DebugInfo& debugInfo = DebugInfo())
117 ConfigItemBuilder::Ptr item = new ConfigItemBuilder(debugInfo);
119 String checkName = name;
122 Type::Ptr ptype = Type::GetByName(type);
124 NameComposer *nc = dynamic_cast<NameComposer *>(ptype.get());
127 checkName = nc->MakeName(name, Dictionary::Ptr());
130 if (!checkName.IsEmpty()) {
131 ConfigItem::Ptr oldItem = ConfigItem::GetByTypeAndName(type, checkName);
134 std::ostringstream msgbuf;
135 msgbuf << "Object '" << name << "' of type '" << type << "' re-defined: " << debugInfo << "; previous definition: " << oldItem->GetDebugInfo();
136 BOOST_THROW_EXCEPTION(ScriptError(msgbuf.str(), debugInfo));
143 item->AddExpression(new OwnedExpression(expression));
144 item->SetAbstract(abstract);
145 item->SetScope(EvaluateClosedVars(frame, closedVars));
147 item->SetPackage(package);
148 item->SetFilter(filter);
149 item->SetIgnoreOnError(ignoreOnError);
150 item->Compile()->Register();
155 static inline ExpressionResult For(ScriptFrame& frame, const String& fkvar, const String& fvvar, const Value& value, Expression *expression, const DebugInfo& debugInfo = DebugInfo())
157 if (value.IsObjectType<Array>()) {
158 if (!fvvar.IsEmpty())
159 BOOST_THROW_EXCEPTION(ScriptError("Cannot use dictionary iterator for array.", debugInfo));
161 Array::Ptr arr = value;
163 for (Array::SizeType i = 0; i < arr->GetLength(); i++) {
164 frame.Locals->Set(fkvar, arr->Get(i));
165 ExpressionResult res = expression->Evaluate(frame);
166 CHECK_RESULT_LOOP(res);
168 } else if (value.IsObjectType<Dictionary>()) {
170 BOOST_THROW_EXCEPTION(ScriptError("Cannot use array iterator for dictionary.", debugInfo));
172 Dictionary::Ptr dict = value;
173 std::vector<String> keys;
176 ObjectLock olock(dict);
177 BOOST_FOREACH(const Dictionary::Pair& kv, dict) {
178 keys.push_back(kv.first);
182 BOOST_FOREACH(const String& key, keys) {
183 frame.Locals->Set(fkvar, key);
184 frame.Locals->Set(fvvar, dict->Get(key));
185 ExpressionResult res = expression->Evaluate(frame);
186 CHECK_RESULT_LOOP(res);
189 BOOST_THROW_EXCEPTION(ScriptError("Invalid type in for expression: " + value.GetTypeName(), debugInfo));
194 static inline bool HasField(const Object::Ptr& context, const String& field)
196 Dictionary::Ptr dict = dynamic_pointer_cast<Dictionary>(context);
199 return dict->Contains(field);
201 Type::Ptr type = context->GetReflectionType();
206 return type->GetFieldId(field) != -1;
210 static inline Value GetPrototypeField(const Value& context, const String& field, bool not_found_error = true, const DebugInfo& debugInfo = DebugInfo())
212 Type::Ptr ctype = context.GetReflectionType();
213 Type::Ptr type = ctype;
216 Object::Ptr object = type->GetPrototype();
218 if (object && HasField(object, field))
219 return GetField(object, field, false, debugInfo);
221 type = type->GetBaseType();
225 BOOST_THROW_EXCEPTION(ScriptError("Invalid field access (for value of type '" + ctype->GetName() + "'): '" + field + "'", debugInfo));
230 static inline Value GetField(const Value& context, const String& field, bool sandboxed = false, const DebugInfo& debugInfo = DebugInfo())
232 if (context.IsEmpty() && !context.IsString())
235 if (!context.IsObject())
236 return GetPrototypeField(context, field, true, debugInfo);
238 Object::Ptr object = context;
240 Dictionary::Ptr dict = dynamic_pointer_cast<Dictionary>(object);
244 if (dict->Get(field, &value))
247 return GetPrototypeField(context, field, false, debugInfo);
250 Array::Ptr arr = dynamic_pointer_cast<Array>(object);
256 index = Convert::ToLong(field);
258 return GetPrototypeField(context, field, true, debugInfo);
261 if (index < 0 || index >= arr->GetLength())
262 BOOST_THROW_EXCEPTION(ScriptError("Array index '" + Convert::ToString(index) + "' is out of bounds.", debugInfo));
264 return arr->Get(index);
267 Type::Ptr type = object->GetReflectionType();
272 int fid = type->GetFieldId(field);
275 return GetPrototypeField(context, field, true, debugInfo);
278 Field fieldInfo = type->GetFieldInfo(fid);
280 if (fieldInfo.Attributes & FANoUserView)
281 BOOST_THROW_EXCEPTION(ScriptError("Accessing the field '" + field + "' for type '" + type->GetName() + "' is not allowed in sandbox mode."));
284 return object->GetField(fid);
287 static inline void SetField(const Object::Ptr& context, const String& field, const Value& value, const DebugInfo& debugInfo = DebugInfo())
290 BOOST_THROW_EXCEPTION(ScriptError("Cannot set field '" + field + "' on a value that is not an object.", debugInfo));
292 Dictionary::Ptr dict = dynamic_pointer_cast<Dictionary>(context);
295 dict->Set(field, value);
299 Array::Ptr arr = dynamic_pointer_cast<Array>(context);
302 int index = Convert::ToLong(field);
303 if (index >= arr->GetLength())
304 arr->Resize(index + 1);
305 arr->Set(index, value);
309 Type::Ptr type = context->GetReflectionType();
312 BOOST_THROW_EXCEPTION(ScriptError("Cannot set field on object.", debugInfo));
314 int fid = type->GetFieldId(field);
317 BOOST_THROW_EXCEPTION(ScriptError("Attribute '" + field + "' does not exist.", debugInfo));
320 context->SetField(fid, value);
321 } catch (const boost::bad_lexical_cast&) {
322 Field fieldInfo = type->GetFieldInfo(fid);
323 Type::Ptr ftype = Type::GetByName(fieldInfo.TypeName);
324 BOOST_THROW_EXCEPTION(ScriptError("Attribute '" + field + "' cannot be set to value of type '" + value.GetTypeName() + "', expected '" + ftype->GetName() + "'", debugInfo));
325 } catch (const std::bad_cast&) {
326 Field fieldInfo = type->GetFieldInfo(fid);
327 Type::Ptr ftype = Type::GetByName(fieldInfo.TypeName);
328 BOOST_THROW_EXCEPTION(ScriptError("Attribute '" + field + "' cannot be set to value of type '" + value.GetTypeName() + "', expected '" + ftype->GetName() + "'", debugInfo));
333 static inline Value FunctionWrapper(const std::vector<Value>& arguments,
334 const std::vector<String>& funcargs, const Dictionary::Ptr& closedVars, const boost::shared_ptr<Expression>& expr)
336 if (arguments.size() < funcargs.size())
337 BOOST_THROW_EXCEPTION(std::invalid_argument("Too few arguments for function"));
339 ScriptFrame *frame = ScriptFrame::GetCurrentFrame();
342 closedVars->CopyTo(frame->Locals);
344 for (std::vector<Value>::size_type i = 0; i < std::min(arguments.size(), funcargs.size()); i++)
345 frame->Locals->Set(funcargs[i], arguments[i]);
347 return expr->Evaluate(*frame);
350 static inline Dictionary::Ptr EvaluateClosedVars(ScriptFrame& frame, std::map<String, Expression *> *closedVars)
352 Dictionary::Ptr locals;
355 locals = new Dictionary();
357 typedef std::pair<String, Expression *> ClosedVar;
358 BOOST_FOREACH(const ClosedVar& cvar, *closedVars) {
359 locals->Set(cvar.first, cvar.second->Evaluate(frame));