2012-06-06 14:38:28 +02:00
|
|
|
/******************************************************************************
|
|
|
|
* Icinga 2 *
|
2014-03-19 01:02:29 +01:00
|
|
|
* Copyright (C) 2012-2014 Icinga Development Team (http://www.icinga.org) *
|
2012-06-06 14:38:28 +02:00
|
|
|
* *
|
|
|
|
* This program is free software; you can redistribute it and/or *
|
|
|
|
* modify it under the terms of the GNU General Public License *
|
|
|
|
* as published by the Free Software Foundation; either version 2 *
|
|
|
|
* of the License, or (at your option) any later version. *
|
|
|
|
* *
|
|
|
|
* This program is distributed in the hope that it will be useful, *
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
|
|
* GNU General Public License for more details. *
|
|
|
|
* *
|
|
|
|
* You should have received a copy of the GNU General Public License *
|
|
|
|
* along with this program; if not, write to the Free Software Foundation *
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
|
|
******************************************************************************/
|
|
|
|
|
2013-03-17 20:19:29 +01:00
|
|
|
#include "config/configitem.h"
|
|
|
|
#include "config/configcompilercontext.h"
|
2014-03-18 11:44:09 +01:00
|
|
|
#include "config/applyrule.h"
|
2013-08-29 10:40:43 +02:00
|
|
|
#include "base/application.h"
|
2013-03-16 21:18:53 +01:00
|
|
|
#include "base/dynamictype.h"
|
|
|
|
#include "base/objectlock.h"
|
2013-12-13 15:24:24 +01:00
|
|
|
#include "base/convert.h"
|
2013-03-16 21:18:53 +01:00
|
|
|
#include "base/logger_fwd.h"
|
2013-08-28 08:18:58 +02:00
|
|
|
#include "base/debug.h"
|
2013-12-10 21:44:38 +01:00
|
|
|
#include "base/workqueue.h"
|
2014-01-28 14:32:56 +01:00
|
|
|
#include "base/exception.h"
|
2013-03-16 21:18:53 +01:00
|
|
|
#include <sstream>
|
|
|
|
#include <boost/foreach.hpp>
|
2012-06-01 16:49:33 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2013-03-15 18:21:29 +01:00
|
|
|
boost::mutex ConfigItem::m_Mutex;
|
2012-07-27 16:05:02 +02:00
|
|
|
ConfigItem::ItemMap ConfigItem::m_Items;
|
|
|
|
|
2012-09-19 12:32:39 +02:00
|
|
|
/**
|
|
|
|
* Constructor for the ConfigItem class.
|
|
|
|
*
|
|
|
|
* @param type The object type.
|
|
|
|
* @param name The name of the item.
|
2013-03-11 12:04:10 +01:00
|
|
|
* @param unit The unit of the item.
|
|
|
|
* @param abstract Whether the item is a template.
|
2012-09-19 12:32:39 +02:00
|
|
|
* @param exprl Expression list for the item.
|
|
|
|
* @param debuginfo Debug information.
|
|
|
|
*/
|
2012-08-02 09:38:08 +02:00
|
|
|
ConfigItem::ConfigItem(const String& type, const String& name,
|
2014-03-23 11:27:40 +01:00
|
|
|
bool abstract, const AExpression::Ptr& exprl,
|
2014-03-27 12:30:24 +01:00
|
|
|
const DebugInfo& debuginfo, const Dictionary::Ptr& scope)
|
2013-10-16 13:37:54 +02:00
|
|
|
: m_Type(type), m_Name(name), m_Abstract(abstract), m_Validated(false),
|
2014-03-27 12:30:24 +01:00
|
|
|
m_ExpressionList(exprl), m_DebugInfo(debuginfo),
|
2014-03-24 11:23:47 +01:00
|
|
|
m_Scope(scope)
|
2012-06-05 15:05:15 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:32:39 +02:00
|
|
|
/**
|
|
|
|
* Retrieves the type of the configuration item.
|
|
|
|
*
|
|
|
|
* @returns The type.
|
|
|
|
*/
|
2012-08-02 09:38:08 +02:00
|
|
|
String ConfigItem::GetType(void) const
|
2012-06-05 15:05:15 +02:00
|
|
|
{
|
|
|
|
return m_Type;
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:32:39 +02:00
|
|
|
/**
|
|
|
|
* Retrieves the name of the configuration item.
|
|
|
|
*
|
|
|
|
* @returns The name.
|
|
|
|
*/
|
2012-08-02 09:38:08 +02:00
|
|
|
String ConfigItem::GetName(void) const
|
2012-06-05 15:05:15 +02:00
|
|
|
{
|
|
|
|
return m_Name;
|
|
|
|
}
|
|
|
|
|
2013-03-11 12:04:10 +01:00
|
|
|
/**
|
|
|
|
* Checks whether the item is abstract.
|
|
|
|
*
|
|
|
|
* @returns true if the item is abstract, false otherwise.
|
|
|
|
*/
|
|
|
|
bool ConfigItem::IsAbstract(void) const
|
|
|
|
{
|
|
|
|
return m_Abstract;
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:32:39 +02:00
|
|
|
/**
|
|
|
|
* Retrieves the debug information for the configuration item.
|
|
|
|
*
|
|
|
|
* @returns The debug information.
|
|
|
|
*/
|
2012-07-06 11:22:38 +02:00
|
|
|
DebugInfo ConfigItem::GetDebugInfo(void) const
|
|
|
|
{
|
|
|
|
return m_DebugInfo;
|
|
|
|
}
|
|
|
|
|
2014-03-24 11:23:47 +01:00
|
|
|
Dictionary::Ptr ConfigItem::GetScope(void) const
|
|
|
|
{
|
|
|
|
return m_Scope;
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:32:39 +02:00
|
|
|
/**
|
|
|
|
* Retrieves the expression list for the configuration item.
|
|
|
|
*
|
|
|
|
* @returns The expression list.
|
|
|
|
*/
|
2014-03-23 11:27:40 +01:00
|
|
|
AExpression::Ptr ConfigItem::GetExpressionList(void) const
|
2012-06-01 16:49:33 +02:00
|
|
|
{
|
|
|
|
return m_ExpressionList;
|
|
|
|
}
|
|
|
|
|
2013-12-03 09:59:21 +01:00
|
|
|
Dictionary::Ptr ConfigItem::GetProperties(void)
|
|
|
|
{
|
2013-12-14 23:10:37 +01:00
|
|
|
ASSERT(OwnsLock());
|
2013-12-12 06:30:11 +01:00
|
|
|
|
2014-03-24 11:23:47 +01:00
|
|
|
if (!m_Properties) {
|
|
|
|
m_Properties = make_shared<Dictionary>();
|
2014-03-28 23:00:20 +01:00
|
|
|
m_Properties->Set("type", m_Type);
|
|
|
|
m_Properties->Set("name", m_Name);
|
2014-03-24 11:23:47 +01:00
|
|
|
m_Properties->Set("__parent", m_Scope);
|
2014-03-27 12:30:24 +01:00
|
|
|
GetExpressionList()->Evaluate(m_Properties);
|
2014-03-24 11:23:47 +01:00
|
|
|
m_Properties->Remove("__parent");
|
2014-03-24 11:34:41 +01:00
|
|
|
|
2014-03-28 13:52:29 +01:00
|
|
|
VERIFY(m_Properties->Get("type") == GetType() && m_Properties->Get("name") == GetName());
|
2014-03-24 11:23:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return m_Properties;
|
2013-12-03 09:59:21 +01:00
|
|
|
}
|
|
|
|
|
2012-09-19 12:32:39 +02:00
|
|
|
/**
|
2013-12-03 09:59:21 +01:00
|
|
|
* Commits the configuration item by creating a DynamicObject
|
2012-09-19 12:32:39 +02:00
|
|
|
* object.
|
|
|
|
*
|
|
|
|
* @returns The DynamicObject that was created/updated.
|
|
|
|
*/
|
2013-03-02 09:07:47 +01:00
|
|
|
DynamicObject::Ptr ConfigItem::Commit(void)
|
2012-06-05 15:05:15 +02:00
|
|
|
{
|
2013-03-07 16:00:10 +01:00
|
|
|
ASSERT(!OwnsLock());
|
2013-02-20 19:52:25 +01:00
|
|
|
|
2013-11-13 14:56:31 +01:00
|
|
|
#ifdef _DEBUG
|
2013-03-16 21:18:53 +01:00
|
|
|
Log(LogDebug, "base", "Commit called for ConfigItem Type=" + GetType() + ", Name=" + GetName());
|
2013-11-13 14:56:31 +01:00
|
|
|
#endif /* _DEBUG */
|
2013-01-31 15:26:54 +01:00
|
|
|
|
2013-02-08 21:05:08 +01:00
|
|
|
/* Make sure the type is valid. */
|
2013-03-02 09:07:47 +01:00
|
|
|
DynamicType::Ptr dtype = DynamicType::GetByName(GetType());
|
2013-02-08 21:05:08 +01:00
|
|
|
|
|
|
|
if (!dtype)
|
2013-03-16 21:18:53 +01:00
|
|
|
BOOST_THROW_EXCEPTION(std::runtime_error("Type '" + GetType() + "' does not exist."));
|
2013-03-02 09:07:47 +01:00
|
|
|
|
2013-11-27 12:23:27 +01:00
|
|
|
if (dtype->GetObject(GetName()))
|
2013-08-20 11:06:04 +02:00
|
|
|
BOOST_THROW_EXCEPTION(std::runtime_error("An object with type '" + GetType() + "' and name '" + GetName() + "' already exists."));
|
2013-02-20 19:52:25 +01:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
if (IsAbstract())
|
|
|
|
return DynamicObject::Ptr();
|
2013-02-08 21:05:08 +01:00
|
|
|
|
2013-12-14 23:10:37 +01:00
|
|
|
Dictionary::Ptr properties;
|
|
|
|
|
|
|
|
{
|
|
|
|
ObjectLock olock(this);
|
|
|
|
|
|
|
|
properties = GetProperties();
|
|
|
|
}
|
2013-03-02 09:07:47 +01:00
|
|
|
|
2013-12-03 09:59:21 +01:00
|
|
|
DynamicObject::Ptr dobj = dtype->CreateObject(properties);
|
2013-08-20 11:06:04 +02:00
|
|
|
dobj->Register();
|
2012-07-06 11:22:38 +02:00
|
|
|
|
2013-12-06 21:46:50 +01:00
|
|
|
m_Object = dobj;
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2012-07-06 11:22:38 +02:00
|
|
|
return dobj;
|
2012-06-05 15:05:15 +02:00
|
|
|
}
|
|
|
|
|
2013-03-19 07:09:06 +01:00
|
|
|
/**
|
|
|
|
* Registers the configuration item.
|
|
|
|
*/
|
|
|
|
void ConfigItem::Register(void)
|
|
|
|
{
|
2013-12-12 06:30:11 +01:00
|
|
|
std::pair<String, String> key = std::make_pair(m_Type, m_Name);
|
|
|
|
ConfigItem::Ptr self = GetSelf();
|
|
|
|
|
2013-12-06 21:46:50 +01:00
|
|
|
boost::mutex::scoped_lock lock(m_Mutex);
|
2013-03-19 07:09:06 +01:00
|
|
|
|
2013-12-12 06:30:11 +01:00
|
|
|
m_Items[key] = self;
|
2013-03-19 07:09:06 +01:00
|
|
|
}
|
|
|
|
|
2012-09-19 12:32:39 +02:00
|
|
|
/**
|
|
|
|
* Retrieves a configuration item by type and name.
|
|
|
|
*
|
|
|
|
* @param type The type of the ConfigItem that is to be looked up.
|
|
|
|
* @param name The name of the ConfigItem that is to be looked up.
|
|
|
|
* @returns The configuration item.
|
|
|
|
*/
|
2012-08-02 09:38:08 +02:00
|
|
|
ConfigItem::Ptr ConfigItem::GetObject(const String& type, const String& name)
|
2012-06-05 15:05:15 +02:00
|
|
|
{
|
2013-12-12 06:30:11 +01:00
|
|
|
std::pair<String, String> key = std::make_pair(type, name);
|
2013-02-22 08:12:43 +01:00
|
|
|
ConfigItem::ItemMap::iterator it;
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2013-12-12 06:30:11 +01:00
|
|
|
{
|
|
|
|
boost::mutex::scoped_lock lock(m_Mutex);
|
|
|
|
|
|
|
|
it = m_Items.find(key);
|
|
|
|
}
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2013-02-22 08:12:43 +01:00
|
|
|
if (it != m_Items.end())
|
|
|
|
return it->second;
|
2012-07-27 16:05:02 +02:00
|
|
|
|
2013-02-05 13:06:42 +01:00
|
|
|
return ConfigItem::Ptr();
|
2012-06-05 15:05:15 +02:00
|
|
|
}
|
2013-01-30 23:02:46 +01:00
|
|
|
|
2013-12-10 20:01:38 +01:00
|
|
|
bool ConfigItem::HasObject(const String& type, const String& name)
|
|
|
|
{
|
2013-12-12 06:30:11 +01:00
|
|
|
std::pair<String, String> key = std::make_pair(type, name);
|
2013-12-10 20:01:38 +01:00
|
|
|
ConfigItem::ItemMap::iterator it;
|
|
|
|
|
2013-12-12 06:30:11 +01:00
|
|
|
{
|
|
|
|
boost::mutex::scoped_lock lock(m_Mutex);
|
|
|
|
|
|
|
|
it = m_Items.find(key);
|
|
|
|
}
|
2013-12-10 20:01:38 +01:00
|
|
|
|
|
|
|
return (it != m_Items.end());
|
|
|
|
}
|
|
|
|
|
2013-09-24 13:13:14 +02:00
|
|
|
void ConfigItem::ValidateItem(void)
|
2013-01-30 23:02:46 +01:00
|
|
|
{
|
2013-10-16 13:37:54 +02:00
|
|
|
if (m_Validated)
|
|
|
|
return;
|
|
|
|
|
2013-09-24 13:13:14 +02:00
|
|
|
ConfigType::Ptr ctype = ConfigType::GetByName(GetType());
|
2013-03-02 09:07:47 +01:00
|
|
|
|
2013-09-24 13:13:14 +02:00
|
|
|
if (!ctype) {
|
|
|
|
ConfigCompilerContext::GetInstance()->AddMessage(false, "No validation type found for object '" + GetName() + "' of type '" + GetType() + "'");
|
|
|
|
|
|
|
|
return;
|
2013-08-20 11:06:04 +02:00
|
|
|
}
|
2013-09-24 13:13:14 +02:00
|
|
|
|
|
|
|
ctype->ValidateItem(GetSelf());
|
2013-10-16 13:37:54 +02:00
|
|
|
|
|
|
|
m_Validated = true;
|
2013-08-20 11:06:04 +02:00
|
|
|
}
|
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
bool ConfigItem::ActivateItems(ValidationType validate)
|
2013-08-20 11:06:04 +02:00
|
|
|
{
|
2013-09-24 13:13:14 +02:00
|
|
|
if (ConfigCompilerContext::GetInstance()->HasErrors())
|
|
|
|
return false;
|
|
|
|
|
2013-10-16 13:37:54 +02:00
|
|
|
if (ConfigCompilerContext::GetInstance()->HasErrors())
|
|
|
|
return false;
|
|
|
|
|
2013-12-10 21:44:38 +01:00
|
|
|
ParallelWorkQueue upq;
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
if (validate != ValidateNone) {
|
|
|
|
Log(LogInformation, "config", "Validating config items (step 1)...");
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
BOOST_FOREACH(const ItemMap::value_type& kv, m_Items) {
|
|
|
|
upq.Enqueue(boost::bind(&ConfigItem::ValidateItem, kv.second));
|
|
|
|
}
|
2013-10-16 13:37:54 +02:00
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
upq.Join();
|
|
|
|
|
|
|
|
if (ConfigCompilerContext::GetInstance()->HasErrors())
|
|
|
|
return false;
|
|
|
|
} else
|
|
|
|
Log(LogInformation, "config", "Skipping validating config items (step 1)...");
|
2013-09-24 13:13:14 +02:00
|
|
|
|
2013-12-15 18:47:11 +01:00
|
|
|
Log(LogInformation, "config", "Committing config items");
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2013-12-06 21:46:50 +01:00
|
|
|
BOOST_FOREACH(const ItemMap::value_type& kv, m_Items) {
|
2013-12-10 21:44:38 +01:00
|
|
|
upq.Enqueue(boost::bind(&ConfigItem::Commit, kv.second));
|
2013-12-06 21:46:50 +01:00
|
|
|
}
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-12-10 21:44:38 +01:00
|
|
|
upq.Join();
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
std::vector<DynamicObject::Ptr> objects;
|
2013-11-30 17:42:50 +01:00
|
|
|
BOOST_FOREACH(const ItemMap::value_type& kv, m_Items) {
|
2013-12-12 06:30:11 +01:00
|
|
|
DynamicObject::Ptr object = kv.second->m_Object;
|
2013-02-06 00:32:05 +01:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
if (object)
|
|
|
|
objects.push_back(object);
|
|
|
|
}
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-12-06 21:46:50 +01:00
|
|
|
Log(LogInformation, "config", "Triggering OnConfigLoaded signal for config items");
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-08-29 19:05:06 +02:00
|
|
|
BOOST_FOREACH(const DynamicObject::Ptr& object, objects) {
|
2013-12-10 21:44:38 +01:00
|
|
|
upq.Enqueue(boost::bind(&DynamicObject::OnConfigLoaded, object));
|
2013-08-29 19:05:06 +02:00
|
|
|
}
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-12-10 21:44:38 +01:00
|
|
|
upq.Join();
|
2013-08-29 19:05:06 +02:00
|
|
|
|
2014-03-18 11:44:09 +01:00
|
|
|
Log(LogInformation, "config", "Evaluating 'apply' rules...");
|
|
|
|
ApplyRule::EvaluateRules();
|
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
if (validate != ValidateNone) {
|
|
|
|
Log(LogInformation, "config", "Validating config items (step 2)...");
|
2013-10-16 13:37:54 +02:00
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
BOOST_FOREACH(const ItemMap::value_type& kv, m_Items) {
|
|
|
|
upq.Enqueue(boost::bind(&ConfigItem::ValidateItem, kv.second));
|
|
|
|
}
|
2013-09-24 13:13:14 +02:00
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
upq.Join();
|
|
|
|
} else
|
|
|
|
Log(LogInformation, "config", "Skipping validating config items (step 2)...");
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-12-20 13:15:05 +01:00
|
|
|
ConfigItem::DiscardItems();
|
|
|
|
ConfigType::DiscardTypes();
|
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
if (validate != ValidateNone) {
|
|
|
|
/* log stats for external parsers */
|
|
|
|
BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) {
|
|
|
|
if (type->GetObjects().size() > 0)
|
|
|
|
Log(LogInformation, "config", "Checked " + Convert::ToString(type->GetObjects().size()) + " " + type->GetName() + "(s).");
|
|
|
|
}
|
2013-12-13 15:24:24 +01:00
|
|
|
}
|
|
|
|
|
2013-09-24 13:13:14 +02:00
|
|
|
if (ConfigCompilerContext::GetInstance()->HasErrors())
|
|
|
|
return false;
|
|
|
|
|
2014-03-19 20:59:18 +01:00
|
|
|
if (validate == ValidateOnly)
|
2013-09-24 13:13:14 +02:00
|
|
|
return true;
|
|
|
|
|
2013-08-29 10:40:43 +02:00
|
|
|
/* restore the previous program state */
|
2014-01-28 14:32:56 +01:00
|
|
|
try {
|
|
|
|
DynamicObject::RestoreObjects(Application::GetStatePath());
|
|
|
|
} catch (const std::exception& ex) {
|
|
|
|
Log(LogCritical, "config", "Failed to restore state file: " + DiagnosticInformation(ex));
|
|
|
|
}
|
2013-08-29 10:40:43 +02:00
|
|
|
|
2013-12-06 21:46:50 +01:00
|
|
|
Log(LogInformation, "config", "Triggering Start signal for config items");
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-08-29 19:25:34 +02:00
|
|
|
BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) {
|
|
|
|
BOOST_FOREACH(const DynamicObject::Ptr& object, type->GetObjects()) {
|
|
|
|
if (object->IsActive())
|
|
|
|
continue;
|
2013-08-29 19:05:06 +02:00
|
|
|
|
2013-11-13 14:56:31 +01:00
|
|
|
#ifdef _DEBUG
|
2013-08-29 19:25:34 +02:00
|
|
|
Log(LogDebug, "config", "Activating object '" + object->GetName() + "' of type '" + object->GetType()->GetName() + "'");
|
2013-11-13 14:56:31 +01:00
|
|
|
#endif /* _DEBUG */
|
2013-12-14 07:36:49 +01:00
|
|
|
upq.Enqueue(boost::bind(&DynamicObject::Activate, object));
|
2013-12-06 21:46:50 +01:00
|
|
|
}
|
|
|
|
}
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-12-10 21:44:38 +01:00
|
|
|
upq.Join();
|
2013-12-13 15:24:24 +01:00
|
|
|
|
2013-12-06 21:46:50 +01:00
|
|
|
#ifdef _DEBUG
|
|
|
|
BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) {
|
|
|
|
BOOST_FOREACH(const DynamicObject::Ptr& object, type->GetObjects()) {
|
2013-08-29 19:25:34 +02:00
|
|
|
ASSERT(object->IsActive());
|
|
|
|
}
|
2013-02-06 00:32:05 +01:00
|
|
|
}
|
2013-12-06 21:46:50 +01:00
|
|
|
#endif /* _DEBUG */
|
|
|
|
|
|
|
|
Log(LogInformation, "config", "Activated all objects.");
|
2013-09-24 13:13:14 +02:00
|
|
|
|
|
|
|
return true;
|
2013-08-20 11:06:04 +02:00
|
|
|
}
|
2013-02-06 00:32:05 +01:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
void ConfigItem::DiscardItems(void)
|
|
|
|
{
|
2013-12-12 06:30:11 +01:00
|
|
|
boost::mutex::scoped_lock lock(m_Mutex);
|
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
m_Items.clear();
|
2013-02-06 00:32:05 +01:00
|
|
|
}
|