2015-06-11 22:48:41 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Icinga\Module\Director\IcingaConfig;
|
|
|
|
|
2016-02-17 11:49:56 +01:00
|
|
|
use Icinga\Application\Benchmark;
|
|
|
|
use Icinga\Application\Hook;
|
2016-03-20 01:36:09 +01:00
|
|
|
use Icinga\Application\Icinga;
|
2016-07-22 18:15:22 +02:00
|
|
|
use Icinga\Exception\ConfigurationError;
|
2016-03-02 17:12:09 +01:00
|
|
|
use Icinga\Exception\IcingaException;
|
2016-11-01 18:28:36 +01:00
|
|
|
use Icinga\Exception\NotFoundError;
|
2015-10-15 20:35:21 +02:00
|
|
|
use Icinga\Exception\ProgrammingError;
|
2016-06-30 10:57:35 +02:00
|
|
|
use Icinga\Module\Director\Db\Cache\PrefetchCache;
|
2015-12-15 16:46:19 +01:00
|
|
|
use Icinga\Module\Director\Db;
|
2016-11-01 18:28:36 +01:00
|
|
|
use Icinga\Module\Director\Hook\ShipConfigFilesHook;
|
|
|
|
use Icinga\Module\Director\Objects\IcingaObject;
|
2015-06-23 14:37:23 +02:00
|
|
|
use Icinga\Module\Director\Util;
|
2015-06-11 22:48:41 +02:00
|
|
|
use Icinga\Module\Director\Objects\IcingaHost;
|
2015-12-18 10:53:27 +01:00
|
|
|
use Icinga\Module\Director\Objects\IcingaZone;
|
2015-06-11 22:48:41 +02:00
|
|
|
|
|
|
|
class IcingaConfig
|
|
|
|
{
|
|
|
|
protected $files = array();
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
protected $checksum;
|
|
|
|
|
2015-07-28 15:22:48 +02:00
|
|
|
protected $zoneMap = array();
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
protected $lastActivityChecksum;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \Zend_Db_Adapter_Abstract
|
|
|
|
*/
|
|
|
|
protected $db;
|
|
|
|
|
|
|
|
protected $connection;
|
|
|
|
|
|
|
|
protected $generationTime;
|
|
|
|
|
2016-08-01 08:59:31 +02:00
|
|
|
protected $configFormat;
|
2016-07-22 18:15:22 +02:00
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
public static $table = 'director_generated_config';
|
|
|
|
|
2016-03-19 22:16:24 +01:00
|
|
|
public function __construct(Db $connection)
|
2015-06-17 19:03:23 +02:00
|
|
|
{
|
2016-03-20 01:36:09 +01:00
|
|
|
// Make sure module hooks are loaded:
|
|
|
|
Icinga::app()->getModuleManager()->loadEnabledModules();
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
$this->connection = $connection;
|
|
|
|
$this->db = $connection->getDbAdapter();
|
2016-08-30 09:52:48 +02:00
|
|
|
$this->configFormat = $this->connection->settings()->config_format;
|
2015-06-17 19:03:23 +02:00
|
|
|
}
|
|
|
|
|
2015-12-16 16:19:58 +01:00
|
|
|
public function getSize()
|
|
|
|
{
|
|
|
|
$size = 0;
|
|
|
|
foreach ($this->getFiles() as $file) {
|
|
|
|
$size += $file->getSize();
|
|
|
|
}
|
|
|
|
return $size;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getDuration()
|
|
|
|
{
|
2016-11-01 18:28:36 +01:00
|
|
|
return $this->generationTime;
|
2015-12-16 16:19:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public function getFileCount()
|
|
|
|
{
|
|
|
|
return count($this->files);
|
|
|
|
}
|
|
|
|
|
2016-07-22 18:15:22 +02:00
|
|
|
public function getConfigFormat()
|
|
|
|
{
|
|
|
|
return $this->configFormat;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setConfigFormat($format)
|
|
|
|
{
|
|
|
|
if (! in_array($format, array('v1', 'v2'))) {
|
|
|
|
throw new ConfigurationError(
|
|
|
|
'Only Icinga v1 and v2 config format is supported, got "%s"',
|
|
|
|
$format
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->configFormat = $format;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function isLegacy()
|
|
|
|
{
|
|
|
|
return $this->configFormat === 'v1';
|
|
|
|
}
|
|
|
|
|
2015-12-16 16:19:58 +01:00
|
|
|
public function getObjectCount()
|
|
|
|
{
|
|
|
|
$cnt = 0;
|
|
|
|
foreach ($this->getFiles() as $file) {
|
|
|
|
$cnt += $file->getObjectCount();
|
|
|
|
}
|
|
|
|
return $cnt;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getTemplateCount()
|
|
|
|
{
|
|
|
|
$cnt = 0;
|
|
|
|
foreach ($this->getFiles() as $file) {
|
|
|
|
$cnt += $file->getTemplateCount();
|
|
|
|
}
|
|
|
|
return $cnt;
|
|
|
|
}
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
public function getChecksum()
|
|
|
|
{
|
|
|
|
return $this->checksum;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getHexChecksum()
|
2015-06-11 22:48:41 +02:00
|
|
|
{
|
2015-06-23 14:37:23 +02:00
|
|
|
return Util::binary2hex($this->checksum);
|
2015-06-11 22:48:41 +02:00
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @return IcingaConfigFile[]
|
|
|
|
*/
|
2015-06-11 22:48:41 +02:00
|
|
|
public function getFiles()
|
|
|
|
{
|
|
|
|
return $this->files;
|
|
|
|
}
|
|
|
|
|
2015-07-23 15:36:08 +02:00
|
|
|
public function getFileContents()
|
|
|
|
{
|
|
|
|
$result = array();
|
|
|
|
foreach ($this->files as $name => $file) {
|
|
|
|
$result[$name] = $file->getContent();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
2015-06-17 19:03:23 +02:00
|
|
|
public function getFileNames()
|
|
|
|
{
|
|
|
|
return array_keys($this->files);
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @param string $name
|
|
|
|
*
|
|
|
|
* @return IcingaConfigFile
|
|
|
|
*/
|
2015-12-17 10:54:38 +01:00
|
|
|
public function getFile($name)
|
|
|
|
{
|
|
|
|
return $this->files[$name];
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @param string $checksum
|
|
|
|
* @param Db $connection
|
|
|
|
*
|
|
|
|
* @return static
|
|
|
|
*/
|
2015-12-15 16:46:19 +01:00
|
|
|
public static function load($checksum, Db $connection)
|
2015-06-18 10:54:44 +02:00
|
|
|
{
|
|
|
|
$config = new static($connection);
|
|
|
|
$config->loadFromDb($checksum);
|
|
|
|
return $config;
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @param string $checksum
|
|
|
|
* @param Db $connection
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2016-06-16 15:48:54 +02:00
|
|
|
public static function exists($checksum, Db $connection)
|
|
|
|
{
|
2016-06-16 16:04:05 +02:00
|
|
|
$db = $connection->getDbAdapter();
|
2016-06-16 15:48:54 +02:00
|
|
|
$query = $db->select()->from(
|
|
|
|
array('c' => self::$table),
|
|
|
|
array('checksum' => $connection->dbHexFunc('c.checksum'))
|
|
|
|
)->where(
|
|
|
|
'checksum = ?',
|
|
|
|
$connection->quoteBinary(Util::hex2binary($checksum))
|
|
|
|
);
|
|
|
|
|
|
|
|
return $db->fetchOne($query) === $checksum;
|
|
|
|
}
|
|
|
|
|
2016-05-25 11:53:01 +02:00
|
|
|
public static function loadByActivityChecksum($checksum, Db $connection)
|
|
|
|
{
|
|
|
|
$db = $connection->getDbAdapter();
|
|
|
|
$query = $db->select()->from(
|
|
|
|
array('c' => self::$table),
|
|
|
|
array('checksum' => 'c.checksum')
|
|
|
|
)->join(
|
|
|
|
array('l' => 'director_activity_log'),
|
|
|
|
'l.checksum = c.last_activity_checksum',
|
|
|
|
array()
|
|
|
|
)->where(
|
|
|
|
'last_activity_checksum = ?',
|
|
|
|
$connection->quoteBinary(Util::hex2binary($checksum))
|
|
|
|
)->order('l.id DESC')->limit(1);
|
|
|
|
|
|
|
|
return self::load($db->fetchOne($query), $connection);
|
|
|
|
}
|
|
|
|
|
2016-06-30 10:57:35 +02:00
|
|
|
public static function existsForActivityChecksum($checksum, Db $connection)
|
|
|
|
{
|
|
|
|
$db = $connection->getDbAdapter();
|
|
|
|
$query = $db->select()->from(
|
|
|
|
array('c' => self::$table),
|
|
|
|
array('checksum' => $connection->dbHexFunc('c.checksum'))
|
|
|
|
)->join(
|
|
|
|
array('l' => 'director_activity_log'),
|
|
|
|
'l.checksum = c.last_activity_checksum',
|
|
|
|
array()
|
|
|
|
)->where(
|
|
|
|
'last_activity_checksum = ?',
|
|
|
|
$connection->quoteBinary(Util::hex2binary($checksum))
|
|
|
|
)->order('l.id DESC')->limit(1);
|
|
|
|
|
|
|
|
return $db->fetchOne($query) === $checksum;
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @param Db $connection
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2015-12-15 16:46:19 +01:00
|
|
|
public static function generate(Db $connection)
|
2015-06-17 19:03:23 +02:00
|
|
|
{
|
|
|
|
$config = new static($connection);
|
|
|
|
return $config->storeIfModified();
|
|
|
|
}
|
|
|
|
|
2015-12-15 16:46:19 +01:00
|
|
|
public static function wouldChange(Db $connection)
|
2015-11-06 09:47:36 +01:00
|
|
|
{
|
|
|
|
$config = new static($connection);
|
|
|
|
return $config->hasBeenModified();
|
|
|
|
}
|
|
|
|
|
2016-03-19 22:16:24 +01:00
|
|
|
public function hasBeenModified()
|
2015-06-17 19:03:23 +02:00
|
|
|
{
|
|
|
|
$this->generateFromDb();
|
2015-10-15 20:35:21 +02:00
|
|
|
$this->collectExtraFiles();
|
2015-06-17 19:03:23 +02:00
|
|
|
$checksum = $this->calculateChecksum();
|
2016-02-27 21:06:22 +01:00
|
|
|
$activity = $this->getLastActivityChecksum();
|
|
|
|
|
|
|
|
$lastActivity = $this->binFromDb(
|
|
|
|
$this->db->fetchOne(
|
|
|
|
$this->db->select()->from(
|
|
|
|
self::$table,
|
|
|
|
'last_activity_checksum'
|
|
|
|
)->where(
|
|
|
|
'checksum = ?',
|
|
|
|
$this->dbBin($checksum)
|
|
|
|
)
|
2015-11-03 09:10:34 +01:00
|
|
|
)
|
2015-06-17 19:03:23 +02:00
|
|
|
);
|
2015-11-03 09:10:34 +01:00
|
|
|
|
2016-02-27 21:06:22 +01:00
|
|
|
if ($lastActivity === false || $lastActivity === null) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($lastActivity !== $activity) {
|
|
|
|
$this->db->update(
|
|
|
|
self::$table,
|
|
|
|
array(
|
|
|
|
'last_activity_checksum' => $this->dbBin($activity)
|
|
|
|
),
|
|
|
|
$this->db->quoteInto('checksum = ?', $this->dbBin($checksum))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2015-11-03 09:10:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function storeIfModified()
|
|
|
|
{
|
|
|
|
if ($this->hasBeenModified()) {
|
2015-06-17 19:03:23 +02:00
|
|
|
$this->store();
|
|
|
|
}
|
2015-11-03 09:10:34 +01:00
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2015-06-23 15:13:34 +02:00
|
|
|
protected function dbBin($binary)
|
|
|
|
{
|
2016-03-02 13:42:37 +01:00
|
|
|
if ($this->connection->isPgsql()) {
|
2015-06-23 15:13:34 +02:00
|
|
|
return Util::pgBinEscape($binary);
|
|
|
|
} else {
|
|
|
|
return $binary;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-27 18:14:21 +01:00
|
|
|
protected function binFromDb($value)
|
|
|
|
{
|
|
|
|
if (is_resource($value)) {
|
|
|
|
return stream_get_contents($value);
|
|
|
|
}
|
|
|
|
|
2016-02-27 22:14:47 +01:00
|
|
|
return $value;
|
2016-02-27 18:14:21 +01:00
|
|
|
}
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
protected function calculateChecksum()
|
|
|
|
{
|
2016-02-27 22:14:47 +01:00
|
|
|
$files = array();
|
2015-06-17 19:03:23 +02:00
|
|
|
$sortedFiles = $this->files;
|
|
|
|
ksort($sortedFiles);
|
|
|
|
/** @var IcingaConfigFile $file */
|
|
|
|
foreach ($sortedFiles as $name => $file) {
|
|
|
|
$files[] = $name . '=' . $file->getHexChecksum();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->checksum = sha1(implode(';', $files), true);
|
|
|
|
return $this->checksum;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getFilesChecksums()
|
2015-06-11 22:48:41 +02:00
|
|
|
{
|
2015-06-17 19:03:23 +02:00
|
|
|
$checksums = array();
|
|
|
|
|
|
|
|
/** @var IcingaConfigFile $file */
|
|
|
|
foreach ($this->files as $name => $file) {
|
|
|
|
$checksums[] = $file->getChecksum();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $checksums;
|
|
|
|
}
|
2015-07-28 15:22:48 +02:00
|
|
|
|
2016-04-22 18:15:33 +02:00
|
|
|
// TODO: prepare lookup cache if empty?
|
|
|
|
public function getZoneName($id)
|
2015-07-28 15:22:48 +02:00
|
|
|
{
|
2016-04-22 18:15:33 +02:00
|
|
|
if (! array_key_exists($id, $this->zoneMap)) {
|
|
|
|
$zone = IcingaZone::loadWithAutoIncId($id, $this->connection);
|
2016-11-01 18:28:36 +01:00
|
|
|
$this->zoneMap[$id] = $zone->get('object_name');
|
2016-04-22 18:15:33 +02:00
|
|
|
}
|
|
|
|
|
2015-07-28 15:22:48 +02:00
|
|
|
return $this->zoneMap[$id];
|
|
|
|
}
|
2015-06-17 19:03:23 +02:00
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @return self
|
|
|
|
*/
|
2016-03-19 22:16:24 +01:00
|
|
|
public function store()
|
2015-06-17 19:03:23 +02:00
|
|
|
{
|
|
|
|
$fileTable = IcingaConfigFile::$table;
|
|
|
|
$fileKey = IcingaConfigFile::$keyName;
|
|
|
|
|
2016-06-28 14:01:43 +02:00
|
|
|
$existingQuery = $this->db->select()
|
|
|
|
->from($fileTable, 'checksum')
|
|
|
|
->where('checksum IN (?)', array_map(array($this, 'dbBin'), $this->getFilesChecksums()));
|
|
|
|
|
|
|
|
$existing = $this->db->fetchCol($existingQuery);
|
|
|
|
|
|
|
|
foreach ($existing as $key => $val) {
|
|
|
|
if (is_resource($val)) {
|
|
|
|
$existing[$key] = stream_get_contents($val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$missing = array_diff($this->getFilesChecksums(), $existing);
|
|
|
|
$stored = array();
|
|
|
|
|
|
|
|
/** @var IcingaConfigFile $file */
|
|
|
|
foreach ($this->files as $name => $file) {
|
|
|
|
$checksum = $file->getChecksum();
|
|
|
|
if (! in_array($checksum, $missing)) {
|
|
|
|
continue;
|
2015-06-23 16:45:25 +02:00
|
|
|
}
|
2015-06-17 19:03:23 +02:00
|
|
|
|
2016-06-28 14:01:43 +02:00
|
|
|
if (array_key_exists($checksum, $stored)) {
|
|
|
|
continue;
|
2015-06-17 19:03:23 +02:00
|
|
|
}
|
|
|
|
|
2016-06-28 14:01:43 +02:00
|
|
|
$stored[$checksum] = true;
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
$this->db->insert(
|
2016-06-28 14:01:43 +02:00
|
|
|
$fileTable,
|
2015-06-17 19:03:23 +02:00
|
|
|
array(
|
2016-06-28 14:01:43 +02:00
|
|
|
$fileKey => $this->dbBin($checksum),
|
|
|
|
'content' => $file->getContent(),
|
|
|
|
'cnt_object' => $file->getObjectCount(),
|
|
|
|
'cnt_template' => $file->getTemplateCount()
|
2015-06-17 19:03:23 +02:00
|
|
|
)
|
|
|
|
);
|
2016-06-28 14:01:43 +02:00
|
|
|
}
|
2015-06-17 19:03:23 +02:00
|
|
|
|
2016-06-28 14:01:43 +02:00
|
|
|
$activity = $this->dbBin($this->getLastActivityChecksum());
|
|
|
|
$this->db->insert(
|
|
|
|
self::$table,
|
|
|
|
array(
|
|
|
|
'duration' => $this->generationTime,
|
|
|
|
'first_activity_checksum' => $activity,
|
|
|
|
'last_activity_checksum' => $activity,
|
|
|
|
'checksum' => $this->dbBin($this->getChecksum()),
|
|
|
|
)
|
|
|
|
);
|
|
|
|
/** @var IcingaConfigFile $file */
|
|
|
|
foreach ($this->files as $name => $file) {
|
|
|
|
$this->db->insert(
|
|
|
|
'director_generated_config_file',
|
|
|
|
array(
|
|
|
|
'config_checksum' => $this->dbBin($this->getChecksum()),
|
|
|
|
'file_checksum' => $this->dbBin($file->getChecksum()),
|
|
|
|
'file_path' => $name,
|
|
|
|
)
|
|
|
|
);
|
2015-06-17 19:03:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
2015-06-11 22:48:41 +02:00
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @throws IcingaException
|
|
|
|
*
|
|
|
|
* @return self
|
|
|
|
*/
|
2015-06-17 19:03:23 +02:00
|
|
|
protected function generateFromDb()
|
2015-06-11 22:48:41 +02:00
|
|
|
{
|
2016-06-30 10:57:35 +02:00
|
|
|
PrefetchCache::initialize($this->connection);
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
$start = microtime(true);
|
|
|
|
|
2016-02-19 11:34:31 +01:00
|
|
|
// Raise limits. TODO: do this in a failsafe way, and only if necessary
|
2016-06-30 12:17:32 +02:00
|
|
|
if ((string) ini_get('memory_limit') !== '-1') {
|
|
|
|
ini_set('memory_limit', '1024M');
|
|
|
|
}
|
|
|
|
|
2016-02-19 11:34:31 +01:00
|
|
|
ini_set('max_execution_time', 0);
|
2016-06-30 12:17:32 +02:00
|
|
|
// Workaround for https://bugs.php.net/bug.php?id=68606 or similar
|
|
|
|
ini_set('zend.enable_gc', 0);
|
2016-02-19 11:34:31 +01:00
|
|
|
|
2016-03-02 21:40:35 +01:00
|
|
|
if (! $this->connection->isPgsql() && $this->db->quote("1\0") !== '\'1\\0\'') {
|
|
|
|
|
2016-03-02 17:12:09 +01:00
|
|
|
throw new IcingaException(
|
|
|
|
'Refusing to render the configuration, your DB layer corrupts binary data.'
|
2016-03-02 21:40:35 +01:00
|
|
|
. ' You might be affected by Zend Framework bug #655'
|
2016-03-02 17:12:09 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-06-11 22:48:41 +02:00
|
|
|
$this
|
2016-06-15 16:55:19 +02:00
|
|
|
->prepareGlobalBasics()
|
2015-06-11 22:48:41 +02:00
|
|
|
->createFileFromDb('zone')
|
2015-06-17 10:53:41 +02:00
|
|
|
->createFileFromDb('endpoint')
|
2015-06-11 22:48:41 +02:00
|
|
|
->createFileFromDb('command')
|
2016-03-19 14:57:22 +01:00
|
|
|
->createFileFromDb('timePeriod')
|
2015-06-17 10:53:41 +02:00
|
|
|
->createFileFromDb('hostGroup')
|
2015-06-11 22:48:41 +02:00
|
|
|
->createFileFromDb('host')
|
2015-06-17 10:53:41 +02:00
|
|
|
->createFileFromDb('serviceGroup')
|
2015-06-11 22:48:41 +02:00
|
|
|
->createFileFromDb('service')
|
2016-10-12 11:04:08 +02:00
|
|
|
->createFileFromDb('serviceSet')
|
2015-06-17 10:53:41 +02:00
|
|
|
->createFileFromDb('userGroup')
|
2015-06-12 13:41:38 +02:00
|
|
|
->createFileFromDb('user')
|
2016-06-03 15:24:17 +02:00
|
|
|
->createFileFromDb('notification')
|
2015-06-11 22:48:41 +02:00
|
|
|
;
|
2015-06-12 13:38:49 +02:00
|
|
|
|
2016-09-05 21:34:14 +02:00
|
|
|
if (! $this->isLegacy()) {
|
|
|
|
$this->configFile('zones.d/director-global/commands')
|
|
|
|
->prepend("library \"methods\"\n\n");
|
|
|
|
}
|
2016-05-13 13:58:12 +02:00
|
|
|
|
2016-06-30 10:57:35 +02:00
|
|
|
PrefetchCache::forget();
|
2016-06-30 12:17:32 +02:00
|
|
|
IcingaHost::clearAllPrefetchCaches();
|
|
|
|
|
2015-06-23 16:07:34 +02:00
|
|
|
$this->generationTime = (int) ((microtime(true) - $start) * 1000);
|
2015-06-17 19:03:23 +02:00
|
|
|
|
2015-06-11 22:48:41 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @return self
|
|
|
|
*/
|
2016-06-15 16:55:19 +02:00
|
|
|
protected function prepareGlobalBasics()
|
|
|
|
{
|
2016-09-05 21:34:14 +02:00
|
|
|
if ($this->isLegacy()) {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2016-06-15 16:55:19 +02:00
|
|
|
$this->configFile(
|
|
|
|
sprintf(
|
|
|
|
'zones.d/%s/001-director-basics',
|
|
|
|
$this->connection->getDefaultGlobalZoneName()
|
|
|
|
)
|
|
|
|
)->prepend(
|
|
|
|
"\nconst DirectorStageDir = dirname(dirname(current_filename))\n"
|
2016-09-08 12:45:02 +02:00
|
|
|
. $this->renderHostOverridableVars()
|
2016-06-15 16:55:19 +02:00
|
|
|
. $this->renderMagicApplyFor()
|
|
|
|
);
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2016-09-08 12:45:02 +02:00
|
|
|
protected function renderHostOverridableVars()
|
|
|
|
{
|
|
|
|
$settings = $this->connection->settings();
|
|
|
|
|
|
|
|
return sprintf(
|
|
|
|
'
|
2016-09-08 22:24:54 +02:00
|
|
|
const DirectorOverrideVars = "%s"
|
|
|
|
const DirectorOverrideTemplate = "%s"
|
2016-09-08 12:45:02 +02:00
|
|
|
|
2016-09-08 22:24:54 +02:00
|
|
|
template Service DirectorOverrideTemplate {
|
2016-09-08 22:13:45 +02:00
|
|
|
/**
|
|
|
|
* Seems that host is missing when used in a service object, works fine for
|
|
|
|
* apply rules
|
|
|
|
*/
|
|
|
|
if (! host) {
|
|
|
|
var host = get_host(host_name)
|
|
|
|
}
|
|
|
|
|
2016-09-08 12:45:02 +02:00
|
|
|
if (vars) {
|
2016-09-08 22:24:54 +02:00
|
|
|
vars += host.vars[DirectorOverrideVars][name]
|
2016-09-08 12:45:02 +02:00
|
|
|
} else {
|
2016-09-08 22:24:54 +02:00
|
|
|
vars = host.vars[DirectorOverrideVars][name]
|
2016-09-08 12:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
',
|
|
|
|
$settings->override_services_varname,
|
|
|
|
$settings->override_services_templatename
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-06-15 16:55:19 +02:00
|
|
|
protected function renderMagicApplyFor()
|
|
|
|
{
|
|
|
|
if (! $this->usesMagicApplyFor()) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2016-06-17 20:06:58 +02:00
|
|
|
$varname = $this->getMagicApplyVarName();
|
|
|
|
|
2016-06-15 16:55:19 +02:00
|
|
|
return sprintf(
|
|
|
|
'
|
2016-10-12 11:04:08 +02:00
|
|
|
/* Warning: this is experimental and might be removed */
|
2016-06-15 16:55:19 +02:00
|
|
|
apply Service for (title => params in host.vars["%s"]) {
|
2016-06-17 20:06:58 +02:00
|
|
|
|
2016-06-26 16:45:25 +02:00
|
|
|
var override = host.vars["%s_vars"][title]
|
2016-06-17 20:06:58 +02:00
|
|
|
|
|
|
|
if (typeof(params["templates"]) in [Array, String]) {
|
2016-09-08 15:07:04 +02:00
|
|
|
for (tpl in params["templates"]) {
|
|
|
|
import tpl
|
|
|
|
}
|
2016-06-15 16:55:19 +02:00
|
|
|
} else {
|
|
|
|
import title
|
|
|
|
}
|
|
|
|
|
2016-06-17 20:06:58 +02:00
|
|
|
if (typeof(params.vars) == Dictionary) {
|
2016-09-08 15:07:04 +02:00
|
|
|
vars += params.vars
|
2016-06-17 20:06:58 +02:00
|
|
|
}
|
|
|
|
|
2016-06-15 16:55:19 +02:00
|
|
|
if (typeof(params["host_name"]) == String) {
|
|
|
|
host_name = params["host_name"]
|
|
|
|
}
|
2016-09-08 15:07:04 +02:00
|
|
|
|
2016-09-08 22:24:54 +02:00
|
|
|
import DirectorOverrideTemplate
|
2016-06-15 16:55:19 +02:00
|
|
|
}
|
|
|
|
',
|
2016-06-17 20:06:58 +02:00
|
|
|
$varname,
|
2016-09-08 22:24:54 +02:00
|
|
|
$varname
|
2016-06-15 16:55:19 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getMagicApplyVarName()
|
|
|
|
{
|
2016-08-30 09:52:48 +02:00
|
|
|
return $this->connection->settings()->magic_apply_for;
|
2016-06-15 16:55:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function usesMagicApplyFor()
|
|
|
|
{
|
|
|
|
$db = $this->db;
|
|
|
|
$query = $db->select()->from(
|
|
|
|
array('hv' => 'icinga_host_var'),
|
|
|
|
array('c' => 'COUNT(*)')
|
|
|
|
)->where(
|
|
|
|
'hv.varname = ?',
|
|
|
|
$this->getMagicApplyVarName()
|
|
|
|
);
|
|
|
|
|
|
|
|
return $db->fetchOne($query);
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @param string $checksum
|
|
|
|
*
|
|
|
|
* @throws NotFoundError
|
|
|
|
*
|
|
|
|
* @return self
|
|
|
|
*/
|
2015-06-18 10:54:44 +02:00
|
|
|
protected function loadFromDb($checksum)
|
|
|
|
{
|
|
|
|
$query = $this->db->select()->from(
|
|
|
|
self::$table,
|
2015-12-16 16:19:58 +01:00
|
|
|
array('checksum', 'last_activity_checksum', 'duration')
|
2015-06-23 15:13:34 +02:00
|
|
|
)->where('checksum = ?', $this->dbBin($checksum));
|
2015-06-18 10:54:44 +02:00
|
|
|
$result = $this->db->fetchRow($query);
|
2015-07-24 10:58:02 +02:00
|
|
|
|
|
|
|
if (empty($result)) {
|
2016-11-01 18:28:36 +01:00
|
|
|
throw new NotFoundError('Got no config for %s', Util::binary2hex($checksum));
|
2015-07-24 10:58:02 +02:00
|
|
|
}
|
|
|
|
|
2016-02-27 21:06:22 +01:00
|
|
|
$this->checksum = $this->binFromDb($result->checksum);
|
2016-11-01 18:28:36 +01:00
|
|
|
$this->generationTime = $result->duration;
|
2016-02-27 21:06:22 +01:00
|
|
|
$this->lastActivityChecksum = $this->binFromDb($result->last_activity_checksum);
|
2015-06-23 16:10:57 +02:00
|
|
|
|
2015-06-18 10:54:44 +02:00
|
|
|
$query = $this->db->select()->from(
|
|
|
|
array('cf' => 'director_generated_config_file'),
|
|
|
|
array(
|
2015-12-15 17:17:32 +01:00
|
|
|
'file_path' => 'cf.file_path',
|
|
|
|
'checksum' => 'f.checksum',
|
|
|
|
'content' => 'f.content',
|
|
|
|
'cnt_object' => 'f.cnt_object',
|
|
|
|
'cnt_template' => 'f.cnt_template',
|
2015-06-18 10:54:44 +02:00
|
|
|
)
|
|
|
|
)->join(
|
|
|
|
array('f' => 'director_generated_file'),
|
|
|
|
'cf.file_checksum = f.checksum',
|
|
|
|
array()
|
2015-06-23 15:13:34 +02:00
|
|
|
)->where('cf.config_checksum = ?', $this->dbBin($checksum));
|
2015-06-18 10:54:44 +02:00
|
|
|
|
|
|
|
foreach ($this->db->fetchAll($query) as $row) {
|
|
|
|
$file = new IcingaConfigFile();
|
2015-12-15 17:17:32 +01:00
|
|
|
$this->files[$row->file_path] = $file
|
|
|
|
->setContent($row->content)
|
|
|
|
->setObjectCount($row->cnt_object)
|
|
|
|
->setTemplateCount($row->cnt_template);
|
2015-06-18 10:54:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2015-06-11 22:48:41 +02:00
|
|
|
protected function createFileFromDb($type)
|
|
|
|
{
|
2016-11-01 18:28:36 +01:00
|
|
|
/** @var IcingaObject $class */
|
2015-06-11 22:48:41 +02:00
|
|
|
$class = 'Icinga\\Module\\Director\\Objects\\Icinga' . ucfirst($type);
|
2016-01-19 16:45:20 +01:00
|
|
|
Benchmark::measure(sprintf('Prefetching %s', $type));
|
2015-12-18 10:53:27 +01:00
|
|
|
$objects = $class::prefetchAll($this->connection);
|
|
|
|
return $this->createFileForObjects($type, $objects);
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @param string $type Short object type, like 'service' or 'zone'
|
|
|
|
* @param IcingaObject[] $objects
|
|
|
|
*
|
|
|
|
* @return self
|
|
|
|
*/
|
2015-12-18 10:53:27 +01:00
|
|
|
protected function createFileForObjects($type, $objects)
|
|
|
|
{
|
2016-05-13 14:20:31 +02:00
|
|
|
if (empty($objects)) {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2016-03-20 01:36:44 +01:00
|
|
|
Benchmark::measure(sprintf('Generating %ss: %s', $type, count($objects)));
|
2015-07-28 15:24:05 +02:00
|
|
|
foreach ($objects as $object) {
|
2015-12-01 14:47:10 +01:00
|
|
|
if ($object->isExternal()) {
|
2015-12-02 16:10:20 +01:00
|
|
|
if ($type === 'zone') {
|
2016-11-01 18:28:36 +01:00
|
|
|
$this->zoneMap[$object->get('id')] = $object->getObjectName();
|
2015-12-02 16:10:20 +01:00
|
|
|
}
|
2015-07-28 15:24:05 +02:00
|
|
|
}
|
|
|
|
|
2016-05-13 13:58:12 +02:00
|
|
|
$object->renderToConfig($this);
|
2015-08-28 17:52:19 +02:00
|
|
|
}
|
2016-05-13 13:58:12 +02:00
|
|
|
|
2016-03-20 01:36:44 +01:00
|
|
|
Benchmark::measure(sprintf('%ss done', $type, count($objects)));
|
2015-06-11 22:48:41 +02:00
|
|
|
return $this;
|
|
|
|
}
|
2015-06-17 19:03:23 +02:00
|
|
|
|
2015-12-17 20:33:16 +01:00
|
|
|
protected function typeWantsGlobalZone($type)
|
|
|
|
{
|
|
|
|
$types = array(
|
|
|
|
'command',
|
|
|
|
);
|
|
|
|
|
|
|
|
return in_array($type, $types);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function typeWantsMasterZone($type)
|
|
|
|
{
|
|
|
|
$types = array(
|
|
|
|
'host',
|
|
|
|
'hostGroup',
|
|
|
|
'service',
|
|
|
|
'serviceGroup',
|
|
|
|
'endpoint',
|
|
|
|
'user',
|
|
|
|
'userGroup',
|
2016-03-19 14:57:22 +01:00
|
|
|
'timePeriod',
|
2015-12-17 20:33:16 +01:00
|
|
|
'notification'
|
|
|
|
);
|
|
|
|
|
|
|
|
return in_array($type, $types);
|
|
|
|
}
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/**
|
|
|
|
* @param string $name Relative config file name
|
|
|
|
* @param string $suffix Config file suffix, defaults to '.conf'
|
|
|
|
*
|
|
|
|
* @return IcingaConfigFile
|
|
|
|
*/
|
2016-04-22 18:15:33 +02:00
|
|
|
public function configFile($name, $suffix = '.conf')
|
2015-07-28 15:22:20 +02:00
|
|
|
{
|
2015-10-15 20:02:44 +02:00
|
|
|
$filename = $name . $suffix;
|
2015-07-28 15:22:20 +02:00
|
|
|
if (! array_key_exists($filename, $this->files)) {
|
|
|
|
$this->files[$filename] = new IcingaConfigFile();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->files[$filename];
|
|
|
|
}
|
|
|
|
|
2015-10-15 20:35:21 +02:00
|
|
|
protected function collectExtraFiles()
|
|
|
|
{
|
2016-11-01 18:28:36 +01:00
|
|
|
/** @var ShipConfigFilesHook $hook */
|
2015-10-15 20:35:21 +02:00
|
|
|
foreach (Hook::all('Director\\ShipConfigFiles') as $hook) {
|
|
|
|
foreach ($hook->fetchFiles() as $filename => $file) {
|
|
|
|
if (array_key_exists($filename, $this->files)) {
|
|
|
|
throw new ProgrammingError(
|
|
|
|
'Cannot ship one file twice: %s',
|
|
|
|
$filename
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if ($file instanceof IcingaConfigFile) {
|
|
|
|
$this->files[$filename] = $file;
|
|
|
|
} else {
|
|
|
|
$this->configFile($filename, '')->setContent((string) $file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2015-06-17 19:03:23 +02:00
|
|
|
public function getLastActivityHexChecksum()
|
|
|
|
{
|
2015-06-23 14:37:23 +02:00
|
|
|
return Util::binary2hex($this->getLastActivityChecksum());
|
2015-06-17 19:03:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function getLastActivityChecksum()
|
|
|
|
{
|
|
|
|
if ($this->lastActivityChecksum === null) {
|
|
|
|
$query = $this->db->select()
|
|
|
|
->from('director_activity_log', 'checksum')
|
2016-03-02 23:53:19 +01:00
|
|
|
->order('id DESC')
|
2015-06-17 19:03:23 +02:00
|
|
|
->limit(1);
|
|
|
|
|
|
|
|
$this->lastActivityChecksum = $this->db->fetchOne($query);
|
2015-06-23 14:12:39 +02:00
|
|
|
|
|
|
|
// PgSQL workaround:
|
|
|
|
if (is_resource($this->lastActivityChecksum)) {
|
|
|
|
$this->lastActivityChecksum = stream_get_contents($this->lastActivityChecksum);
|
|
|
|
}
|
2015-06-17 19:03:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $this->lastActivityChecksum;
|
|
|
|
}
|
2015-06-11 22:48:41 +02:00
|
|
|
}
|