397 lines
13 KiB
PHP
397 lines
13 KiB
PHP
<?php
|
|
/* Icinga Web 2 | (c) 2015 Icinga Development Team | GPLv2+ */
|
|
|
|
namespace Icinga\Repository;
|
|
|
|
use Exception;
|
|
use Icinga\Application\Config;
|
|
use Icinga\Data\ConfigObject;
|
|
use Icinga\Data\Extensible;
|
|
use Icinga\Data\Filter\Filter;
|
|
use Icinga\Data\Updatable;
|
|
use Icinga\Data\Reducible;
|
|
use Icinga\Exception\ProgrammingError;
|
|
use Icinga\Exception\StatementException;
|
|
|
|
/**
|
|
* Abstract base class for concrete INI repository implementations
|
|
*
|
|
* Additionally provided features:
|
|
* <ul>
|
|
* <li>Insert, update and delete capabilities</li>
|
|
* <li>Triggers for inserts, updates and deletions</li>
|
|
* </ul>
|
|
*/
|
|
abstract class IniRepository extends Repository implements Extensible, Updatable, Reducible
|
|
{
|
|
/**
|
|
* Per-table configs
|
|
*
|
|
* Example:
|
|
* <code>
|
|
* array(
|
|
* 'event-type' => array(
|
|
* 'module' => 'elasticsearch',
|
|
* 'path' => 'event-types',
|
|
* 'keyColumn' => 'name'
|
|
* )
|
|
* )
|
|
* </code>
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $configs = null;
|
|
|
|
/**
|
|
* The tables for which triggers are available when inserting, updating or deleting rows
|
|
*
|
|
* This may be initialized by concrete repository implementations and describes for which table names triggers
|
|
* are available. The repository attempts to find a method depending on the type of event and table for which
|
|
* to run the trigger. The name of such a method is expected to be declared using lowerCamelCase.
|
|
* (e.g. group_membership will be translated to onUpdateGroupMembership and groupmembership will be translated
|
|
* to onUpdateGroupmembership) The available events are onInsert, onUpdate and onDelete.
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $triggers;
|
|
|
|
/**
|
|
* Create a new INI repository object
|
|
*
|
|
* @param Config|null $ds The data source to use
|
|
*
|
|
* @throws ProgrammingError In case the given data source does not provide a valid key column
|
|
*/
|
|
public function __construct(Config $ds = null)
|
|
{
|
|
parent::__construct($ds); // First! Due to init().
|
|
|
|
if (! ($ds === null || $ds->getConfigObject()->getKeyColumn())) {
|
|
throw new ProgrammingError('INI repositories require their data source to provide a valid key column');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
*/
|
|
public function getDataSource($table = null)
|
|
{
|
|
if ($this->ds === null) {
|
|
if ($table === null) {
|
|
$table = $this->getBaseTable();
|
|
}
|
|
|
|
if ($this->configs === null) {
|
|
$this->configs = $this->initializeConfigs();
|
|
if ($this->configs === null) {
|
|
throw new ProgrammingError('Per-table configs missing');
|
|
}
|
|
}
|
|
|
|
if (! isset($this->configs[$table])) {
|
|
throw new ProgrammingError('Config for table "%s" missing', $table);
|
|
}
|
|
|
|
$config = $this->configs[$table];
|
|
if ($config instanceof Config) {
|
|
if (! $config->getConfigObject()->getKeyColumn()) {
|
|
throw new ProgrammingError(
|
|
'INI repositories require their data source to provide a valid key column'
|
|
);
|
|
}
|
|
} elseif (is_array($config)) {
|
|
if (! isset($config['path'])) {
|
|
throw new ProgrammingError('Path to config for table "%s" missing', $table);
|
|
}
|
|
if (! isset($config['keyColumn'])) {
|
|
throw new ProgrammingError(
|
|
'INI repositories require their data source to provide a valid key column'
|
|
);
|
|
}
|
|
|
|
$newConfig = isset($config['module'])
|
|
? Config::module($config['module'], $config['path'])
|
|
: Config::app($config['path']);
|
|
$newConfig->getConfigObject()->setKeyColumn($config['keyColumn']);
|
|
$this->configs[$table] = $config = $newConfig;
|
|
} else {
|
|
throw new ProgrammingError(
|
|
'Config for table "%s" is a %s, expected either Icinga\Application\Config or an associative array',
|
|
$table,
|
|
is_object($config) ? get_class($config) : gettype($config)
|
|
);
|
|
}
|
|
|
|
return $config;
|
|
} else {
|
|
return parent::getDataSource($table);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return the tables for which triggers are available when inserting, updating or deleting rows
|
|
*
|
|
* Calls $this->initializeTriggers() in case $this->triggers is null.
|
|
*
|
|
* @return array
|
|
*/
|
|
public function getTriggers()
|
|
{
|
|
if ($this->triggers === null) {
|
|
$this->triggers = $this->initializeTriggers();
|
|
}
|
|
|
|
return $this->triggers;
|
|
}
|
|
|
|
/**
|
|
* Overwrite this in your repository implementation in case you need to initialize the triggers lazily
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function initializeTriggers()
|
|
{
|
|
return array();
|
|
}
|
|
|
|
/**
|
|
* Overwrite this in your INI repository implementation in case you need to initialize the configs lazily
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function initializeConfigs()
|
|
{
|
|
}
|
|
|
|
/**
|
|
* Run a trigger for the given table and row which is about to be inserted
|
|
*
|
|
* @param string $table
|
|
* @param ConfigObject $new
|
|
*
|
|
* @return ConfigObject
|
|
*/
|
|
public function onInsert($table, ConfigObject $new)
|
|
{
|
|
$trigger = $this->getTrigger($table, 'onInsert');
|
|
if ($trigger !== null) {
|
|
$row = $this->$trigger($new);
|
|
if ($row !== null) {
|
|
$new = $row;
|
|
}
|
|
}
|
|
|
|
return $new;
|
|
}
|
|
|
|
/**
|
|
* Run a trigger for the given table and row which is about to be updated
|
|
*
|
|
* @param string $table
|
|
* @param ConfigObject $old
|
|
* @param ConfigObject $new
|
|
*
|
|
* @return ConfigObject
|
|
*/
|
|
public function onUpdate($table, ConfigObject $old, ConfigObject $new)
|
|
{
|
|
$trigger = $this->getTrigger($table, 'onUpdate');
|
|
if ($trigger !== null) {
|
|
$row = $this->$trigger($old, $new);
|
|
if ($row !== null) {
|
|
$new = $row;
|
|
}
|
|
}
|
|
|
|
return $new;
|
|
}
|
|
|
|
/**
|
|
* Run a trigger for the given table and row which has been deleted
|
|
*
|
|
* @param string $table
|
|
* @param ConfigObject $old
|
|
*
|
|
* @return ConfigObject
|
|
*/
|
|
public function onDelete($table, ConfigObject $old)
|
|
{
|
|
$trigger = $this->getTrigger($table, 'onDelete');
|
|
if ($trigger !== null) {
|
|
$this->$trigger($old);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return the name of the trigger method for the given table and event-type
|
|
*
|
|
* @param string $table The table name for which to return a trigger method
|
|
* @param string $event The name of the event type
|
|
*
|
|
* @return string
|
|
*/
|
|
protected function getTrigger($table, $event)
|
|
{
|
|
if (! in_array($table, $this->getTriggers())) {
|
|
return;
|
|
}
|
|
|
|
$identifier = join('', array_map('ucfirst', explode('_', $table)));
|
|
if (method_exists($this, $event . $identifier)) {
|
|
return $event . $identifier;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Insert the given data for the given target
|
|
*
|
|
* $data must provide a proper value for the data source's key column.
|
|
*
|
|
* @param string $target
|
|
* @param array $data
|
|
*
|
|
* @throws StatementException In case the operation has failed
|
|
*/
|
|
public function insert($target, array $data)
|
|
{
|
|
$newData = $this->requireStatementColumns($target, $data);
|
|
$config = $this->onInsert($target, new ConfigObject($newData));
|
|
$section = $this->extractSectionName($config, $target);
|
|
|
|
if ($this->getDataSource($target)->hasSection($section)) {
|
|
throw new StatementException(t('Cannot insert. Section "%s" does already exist'), $section);
|
|
}
|
|
|
|
$this->getDataSource($target)->setSection($section, $config);
|
|
|
|
try {
|
|
$this->getDataSource($target)->saveIni();
|
|
} catch (Exception $e) {
|
|
throw new StatementException(t('Failed to insert. An error occurred: %s'), $e->getMessage());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update the target with the given data and optionally limit the affected entries by using a filter
|
|
*
|
|
* @param string $target
|
|
* @param array $data
|
|
* @param Filter $filter
|
|
*
|
|
* @throws StatementException In case the operation has failed
|
|
*/
|
|
public function update($target, array $data, Filter $filter = null)
|
|
{
|
|
$newData = $this->requireStatementColumns($target, $data);
|
|
$keyColumn = $this->getDataSource($target)->getConfigObject()->getKeyColumn();
|
|
if ($filter === null && isset($newData[$keyColumn])) {
|
|
throw new StatementException(
|
|
t('Cannot update. Column "%s" holds a section\'s name which must be unique'),
|
|
$keyColumn
|
|
);
|
|
}
|
|
|
|
$query = $this->getDataSource($target)->select();
|
|
if ($filter !== null) {
|
|
$query->addFilter($this->requireFilter($target, $filter));
|
|
}
|
|
|
|
/** @var ConfigObject $config */
|
|
$newSection = null;
|
|
foreach ($query as $section => $config) {
|
|
if ($newSection !== null) {
|
|
throw new StatementException(
|
|
t('Cannot update. Column "%s" holds a section\'s name which must be unique'),
|
|
$keyColumn
|
|
);
|
|
}
|
|
|
|
$newConfig = clone $config;
|
|
foreach ($newData as $column => $value) {
|
|
if ($column === $keyColumn) {
|
|
$newSection = $value;
|
|
} else {
|
|
$newConfig->$column = $value;
|
|
}
|
|
}
|
|
|
|
// This is necessary as the query result set contains the key column.
|
|
unset($newConfig->$keyColumn);
|
|
|
|
if ($newSection) {
|
|
if ($this->getDataSource($target)->hasSection($newSection)) {
|
|
throw new StatementException(t('Cannot update. Section "%s" does already exist'), $newSection);
|
|
}
|
|
|
|
$this->getDataSource($target)->removeSection($section)->setSection(
|
|
$newSection,
|
|
$this->onUpdate($target, $config, $newConfig)
|
|
);
|
|
} else {
|
|
$this->getDataSource($target)->setSection(
|
|
$section,
|
|
$this->onUpdate($target, $config, $newConfig)
|
|
);
|
|
}
|
|
}
|
|
|
|
try {
|
|
$this->getDataSource($target)->saveIni();
|
|
} catch (Exception $e) {
|
|
throw new StatementException(t('Failed to update. An error occurred: %s'), $e->getMessage());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Delete entries in the given target, optionally limiting the affected entries by using a filter
|
|
*
|
|
* @param string $target
|
|
* @param Filter $filter
|
|
*
|
|
* @throws StatementException In case the operation has failed
|
|
*/
|
|
public function delete($target, Filter $filter = null)
|
|
{
|
|
$query = $this->getDataSource($target)->select();
|
|
if ($filter !== null) {
|
|
$query->addFilter($this->requireFilter($target, $filter));
|
|
}
|
|
|
|
/** @var ConfigObject $config */
|
|
foreach ($query as $section => $config) {
|
|
$this->getDataSource($target)->removeSection($section);
|
|
$this->onDelete($target, $config);
|
|
}
|
|
|
|
try {
|
|
$this->getDataSource($target)->saveIni();
|
|
} catch (Exception $e) {
|
|
throw new StatementException(t('Failed to delete. An error occurred: %s'), $e->getMessage());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Extract and return the section name off of the given $config
|
|
*
|
|
* @param array|ConfigObject $config
|
|
* @param string $target The table whose datasource to get the key column from
|
|
*
|
|
* @return string
|
|
*
|
|
* @throws ProgrammingError In case no valid section name is available
|
|
*/
|
|
protected function extractSectionName( & $config, $target)
|
|
{
|
|
$keyColumn = $this->getDataSource($target)->getConfigObject()->getKeyColumn();
|
|
if (! is_array($config) && !$config instanceof ConfigObject) {
|
|
throw new ProgrammingError('$config is neither an array nor a ConfigObject');
|
|
} elseif (! isset($config[$keyColumn])) {
|
|
throw new ProgrammingError('$config does not provide a value for key column "%s"', $keyColumn);
|
|
}
|
|
|
|
$section = $config[$keyColumn];
|
|
unset($config[$keyColumn]);
|
|
return $section;
|
|
}
|
|
}
|