booleans); } public function supportsCustomVars() { return $this->supportsCustomVars; } public function supportsGroups() { return $this->supportsGroups; } public function supportsRanges() { return $this->supportsRanges; } public function supportsArguments() { return $this->supportsArguments; } public function supportsImports() { return $this->supportsImports; } public function supportsFields() { return $this->supportsFields; } public function hasBeenModified() { if ($this->supportsCustomVars() && $this->vars !== null && $this->vars()->hasBeenModified()) { return true; } if ($this->supportsGroups() && $this->groups !== null && $this->groups()->hasBeenModified()) { return true; } if ($this->supportsImports() && $this->imports !== null && $this->imports()->hasBeenModified()) { return true; } if ($this->supportsRanges() && $this->ranges !== null && $this->ranges()->hasBeenModified()) { return true; } return parent::hasBeenModified(); } public function set($key, $value) { if ($key === 'groups') { $this->groups()->set($value); return $this; } elseif ($key === 'imports') { $this->imports()->set($value); return $this; } elseif ($key === 'vars') { $value = (array) $value; $unset = array(); foreach ($this->vars() as $k => $f) { if (! array_key_exists($k, $value)) { $unset[] = $k; } } foreach ($unset as $k) { unset($this->vars()->$k); } foreach ($value as $k => $v) { $this->vars()->set($k, $v); } return $this; } if ($this->propertyIsBoolean($key) && $value !== null) { if ($value === 'y' || $value === '1' || $value === true || $value === 1) { return parent::set($key, 'y'); } elseif ($value === 'n' || $value === '0' || $value === false || $value === 0) { return parent::set($key, 'n'); } else { throw new ProgrammingError( 'Got invalid boolean: %s', var_export($value, 1) ); } } return parent::set($key, $value); } public function groups() { $this->assertGroupsSupport(); if ($this->groups === null) { if ($this->hasBeenLoadedFromDb()) { $this->groups = IcingaObjectGroups::loadForStoredObject($this); } else { $this->groups = new IcingaObjectGroups($this); } } return $this->groups; } public function ranges() { $this->assertRangesSupport(); if ($this->ranges === null) { if ($this->hasBeenLoadedFromDb()) { $this->ranges = IcingaTimePeriodRanges::loadForStoredObject($this); } else { $this->ranges = new IcingaTimePeriodRanges($this); } } return $this->ranges; } public function arguments() { $this->assertArgumentsSupport(); if ($this->arguments === null) { if ($this->hasBeenLoadedFromDb()) { $this->arguments = IcingaArguments::loadForStoredObject($this); } else { $this->arguments = new IcingaArguments($this); } } return $this->arguments; } public function imports() { $this->assertImportsSupport(); if ($this->imports === null) { if ($this->hasBeenLoadedFromDb()) { $this->imports = IcingaObjectImports::loadForStoredObject($this); } else { $this->imports = new IcingaObjectImports($this); } } return $this->imports; } // TODO: what happens if imports change at runtime? public function importedObjects() { $this->assertImportsSupport(); if ($this->importedObjects === null) { $this->importedObjects = array(); foreach ($this->imports()->listImportNames() as $import) { $this->importedObjects[$import] = self::load($import, $this->connection); } } return $this->importedObjects; } public function clearImportedObjects() { $this->importedObjects = null; return $this; } public function getResolvedProperty($key) { $properties = $this->getResolvedProperties(); if (property_exists($properties, $key)) { return $properties->$key; } return null; } public function getResolvedProperties() { return $this->getResolved('Properties'); } public function getInheritedProperties() { return $this->getInherited('Properties'); } public function getOriginsProperties() { return $this->getOrigins('Properties'); } public function resolveProperties() { return $this->resolve('Properties'); } public function getResolvedFields() { return $this->getResolved('Fields'); } public function getInheritedFields() { return $this->getInherited('Fields'); } public function getOriginsFields() { return $this->getOrigins('Fields'); } public function resolveFields() { return $this->resolve('Fields'); } public function getResolvedVars() { return $this->getResolved('Vars'); } public function getInheritedVars() { return $this->getInherited('Vars'); } public function resolveVars() { return $this->resolve('Vars'); } public function getOriginsVars() { return $this->getOrigins('Vars'); } public function getVars() { $vars = (object) array(); foreach ($this->vars() as $key => $var) { $vars->$key = $var->getValue(); } return $vars; } protected function getResolved($what) { $func = 'resolve' . $what; $res = $this->$func(); return $res['_MERGED_']; } protected function getInherited($what) { $func = 'resolve' . $what; $res = $this->$func(); return $res['_INHERITED_']; } protected function getOrigins($what) { $func = 'resolve' . $what; $res = $this->$func(); return $res['_ORIGINS_']; } protected function resolve($what) { $vals = array(); $vals['_MERGED_'] = (object) array(); $vals['_INHERITED_'] = (object) array(); $vals['_ORIGINS_'] = (object) array(); $objects = $this->importedObjects(); $get = 'get' . $what; $getInherited = 'getInherited' . $what; $getOrigins = 'getOrigins' . $what; foreach ($objects as $name => $object) { $origins = $object->$getOrigins(); foreach ($object->$getInherited() as $key => $value) { // $vals[$name]->$key = $value; $vals['_MERGED_']->$key = $value; $vals['_INHERITED_']->$key = $value; $vals['_ORIGINS_']->$key = $origins->$key; } foreach ($object->$get() as $key => $value) { if ($value === null) continue; $vals['_MERGED_']->$key = $value; $vals['_INHERITED_']->$key = $value; $vals['_ORIGINS_']->$key = $name; } } $blacklist = array('id', 'object_type', 'object_name'); foreach ($this->$get() as $key => $value) { if ($value === null) continue; if (in_array($key, $blacklist)) continue; // $vals[$this->object_name]->$key = $value; $vals['_MERGED_']->$key = $value; } return $vals; } protected function assertCustomVarsSupport() { if (! $this->supportsCustomVars()) { throw new ProgrammingError( 'Objects of type "%s" have no custom vars', $this->getType() ); } return $this; } protected function assertGroupsSupport() { if (! $this->supportsGroups()) { throw new ProgrammingError( 'Objects of type "%s" have no groups', $this->getType() ); } return $this; } protected function assertRangesSupport() { if (! $this->supportsRanges()) { throw new ProgrammingError( 'Objects of type "%s" have no ranges', $this->getType() ); } return $this; } protected function assertArgumentsSupport() { if (! $this->supportsArguments()) { throw new ProgrammingError( 'Objects of type "%s" have no arguments', $this->getType() ); } return $this; } protected function assertImportsSupport() { if (! $this->supportsImports()) { throw new ProgrammingError( 'Objects of type "%s" have no imports', $this->getType() ); } return $this; } public function vars() { $this->assertCustomVarsSupport(); if ($this->vars === null) { if ($this->hasBeenLoadedFromDb()) { $this->vars = CustomVariables::loadForStoredObject($this); } else { $this->vars = new CustomVariables(); } } return $this->vars; } public function getVarsTableName() { return $this->getTableName() . '_var'; } public function getShortTableName() { // strlen('icinga_') = 7 return substr($this->getTableName(), 7); } public function getVarsIdColumn() { return $this->getShortTableName() . '_id'; } public function getFields() { $fields = (object) array(); if (! $this->supportsFields()) { return $fields; } $db = $this->getDb(); $query = $db->select()->from( array('df' => 'director_datafield'), array( 'datafield_id' => 'f.datafield_id', 'is_required' => 'f.is_required', 'varname' => 'df.varname', 'description' => 'df.description', 'datatype' => 'df.datatype', 'format' => 'df.format', ) )->join( array('f' => $this->getTableName() . '_field'), 'df.id = f.datafield_id', array() )->where('f.' . $this->getShortTableName() . '_id = ?', (int) $this->id) ->order('df.caption ASC'); $res = $db->fetchAll($query); foreach ($res as $r) { $fields->{$r->varname} = $r; } return $fields; } public function isTemplate() { return $this->hasProperty('object_type') && $this->object_type === 'template'; } public function isApplyRule() { return $this->hasProperty('object_type') && $this->object_type === 'apply'; } protected function storeRelatedObjects() { $this ->storeCustomVars() ->storeGroups() ->storeImports() ->storeRanges() ->storeArguments(); } public function onInsert() { DirectorActivityLog::logCreation($this, $this->connection); $this->storeRelatedObjects(); } public function onUpdate() { DirectorActivityLog::logModification($this, $this->connection); $this->storeRelatedObjects(); } protected function storeCustomVars() { if ($this->supportsCustomVars()) { $this->vars !== null && $this->vars()->storeToDb($this); } return $this; } protected function storeGroups() { if ($this->supportsGroups()) { $this->groups !== null && $this->groups()->store(); } return $this; } protected function storeRanges() { if ($this->supportsRanges()) { $this->ranges !== null && $this->ranges()->store(); } return $this; } protected function storeArguments() { if ($this->supportsArguments()) { $this->arguments !== null && $this->arguments()->store(); } return $this; } protected function storeImports() { if ($this->supportsImports()) { $this->imports !== null && $this->imports()->store(); } return $this; } public function onDelete() { DirectorActivityLog::logRemoval($this, $this->connection); } protected function renderImports() { // TODO: parent_host ORDERed by weigth... if ($this->supportsImports()) { return $this->imports()->toConfigString(); } else { return ''; } } protected function renderProperties() { $out = ''; $blacklist = array( 'id', 'object_name', 'object_type', ); foreach ($this->properties as $key => $value) { if ($value === null) continue; if (in_array($key, $blacklist)) continue; $method = 'render' . ucfirst($key); if (method_exists($this, $method)) { $out .= $this->$method($value); } else { if ($this->propertyIsBoolean($key)) { if ($value !== $this->defaultProperties[$key]) { $out .= c::renderKeyValue( $this->booleans[$key], c::renderBoolean($value) ); } } else { $out .= c::renderKeyValue($key, c::renderString($value)); } } } return $out; } protected function renderBooleanProperty($key) { return c::renderKeyValue($key, c::renderBoolean($this->$key)); } protected function renderSuffix() { return "}\n\n"; } /** * @return string */ protected function renderCustomVars() { if ($this->supportsCustomVars()) { return $this->vars()->toConfigString(); } else { return ''; } } /** * @return string */ protected function renderGroups() { if ($this->supportsGroups()) { return $this->groups()->toConfigString(); } else { return ''; } } /** * @return string */ protected function renderRanges() { if ($this->supportsRanges()) { return $this->ranges()->toConfigString(); } else { return ''; } } /** * @return string */ protected function renderArguments() { if ($this->supportsArguments()) { return $this->arguments()->toConfigString(); } else { return ''; } } protected function renderCommandProperty($commandId, $propertyName = 'check_command') { return c::renderKeyValue( $propertyName, c::renderString($this->connection->getCommandName($commandId)) ); } protected function renderZoneProperty($zoneId, $propertyName = 'zone') { // TODO: set file info? return ''; return c::renderKeyValue( $propertyName, c::renderString($this->connection->getZoneName($zoneId)) ); } protected function renderZone_id() { return $this->renderZoneProperty($this->zone_id); } protected function renderObjectHeader() { return sprintf( "%s %s %s {\n", $this->getObjectTypeName(), $this->getType(), c::renderString($this->getObjectName()) ); } public function toConfigString() { return implode(array( $this->renderObjectHeader(), $this->renderImports(), $this->renderProperties(), $this->renderRanges(), $this->renderArguments(), $this->renderGroups(), $this->renderCustomVars(), $this->renderSuffix() )); } public function isGroup() { return substr($this->getType(), -5) === 'Group'; } public function hasCheckCommand() { return false; } protected function getType() { if ($this->type === null) { $parts = explode('\\', get_class($this)); // 6 = strlen('Icinga'); $this->type = substr(end($parts), 6); } return $this->type; } protected function getObjectTypeName() { if ($this->isTemplate()) { return 'template'; } elseif ($this->isApplyRule()) { return 'apply'; } else { return 'object'; } } protected function getObjectName() { if ($this->hasProperty('object_name')) { return $this->object_name; } else { // TODO: replace with an exception once finished return 'ERROR: NO NAME'; } } protected static function classByType($type) { // allow for icinga_host and host $type = preg_replace('/^icinga_/', '', $type); if (strpos($type, 'data') === false) { $prefix = 'Icinga'; } else { $prefix = 'Director'; } return 'Icinga\\Module\\Director\\Objects\\' . $prefix . ucfirst($type); } public static function createByType($type, $properties = array(), Db $db = null) { $class = self::classByType($type); return $class::create($properties, $db); } public static function loadByType($type, $id, Db $db) { $class = self::classByType($type); return $class::load($id, $db); } public static function loadAllByType($type, Db $db, $query = null, $keyColumn = 'object_name') { if ($type === 'datalistEntry') $keyColumn = 'entry_name'; $class = self::classByType($type); return $class::loadAll($db, $query, $keyColumn); } public function __toString() { try { return $this->toConfigString(); } catch (Exception $e) { trigger_error($e); $previousHandler = set_exception_handler(function () {}); restore_error_handler(); if ($previousHandler !== null) { call_user_func($previousHandler, $e); die(); } else { die($e->getMessage()); } } } }