*/ private $rootPermissions = [ DefaultPermissions::ROOT_USER => true ]; /** @var PermissionAttachment[] */ private $attachments = []; /** @var PermissionAttachmentInfo[] */ private $permissions = []; public function __construct(?Permissible $permissible, bool $isOp){ $this->parent = $permissible; //TODO: we can't setBasePermission here directly due to bad architecture that causes recalculatePermissions to explode //so, this hack has to be done here to prevent permission recalculations until it's fixed... if($isOp){ $this->rootPermissions[DefaultPermissions::ROOT_OPERATOR] = true; } //TODO: permissions need to be recalculated here, or inherited permissions won't work } private function getRootPermissible() : Permissible{ return $this->parent ?? $this; } public function setBasePermission($name, bool $grant) : void{ if($name instanceof Permission){ $name = $name->getName(); } $this->rootPermissions[$name] = $grant; $this->getRootPermissible()->recalculatePermissions(); } public function unsetBasePermission($name) : void{ unset($this->rootPermissions[$name instanceof Permission ? $name->getName() : $name]); $this->getRootPermissible()->recalculatePermissions(); } /** * @param Permission|string $name */ public function isPermissionSet($name) : bool{ return isset($this->permissions[$name instanceof Permission ? $name->getName() : $name]); } /** * @param Permission|string $name */ public function hasPermission($name) : bool{ if($name instanceof Permission){ $name = $name->getName(); } if($this->isPermissionSet($name)){ return $this->permissions[$name]->getValue(); } return false; } /** * //TODO: tick scheduled attachments */ public function addAttachment(Plugin $plugin, ?string $name = null, ?bool $value = null) : PermissionAttachment{ if(!$plugin->isEnabled()){ throw new PluginException("Plugin " . $plugin->getDescription()->getName() . " is disabled"); } $result = new PermissionAttachment($plugin, $this->getRootPermissible()); $this->attachments[spl_object_id($result)] = $result; if($name !== null and $value !== null){ $result->setPermission($name, $value); } $this->getRootPermissible()->recalculatePermissions(); return $result; } public function removeAttachment(PermissionAttachment $attachment) : void{ if(isset($this->attachments[spl_object_id($attachment)])){ unset($this->attachments[spl_object_id($attachment)]); if(($ex = $attachment->getRemovalCallback()) !== null){ $ex->attachmentRemoved($attachment); } $this->getRootPermissible()->recalculatePermissions(); } } public function recalculatePermissions() : void{ Timings::$permissibleCalculationTimer->startTiming(); $permManager = PermissionManager::getInstance(); $permManager->unsubscribeFromAllPermissions($this->getRootPermissible()); $this->permissions = []; foreach($this->rootPermissions as $name => $isGranted){ $perm = $permManager->getPermission($name); if($perm === null){ throw new \InvalidStateException("Unregistered root permission $name"); } $this->permissions[$name] = new PermissionAttachmentInfo($name, null, $isGranted); $permManager->subscribeToPermission($name, $this->getRootPermissible()); $this->calculateChildPermissions($perm->getChildren(), false, null); } foreach($this->attachments as $attachment){ $this->calculateChildPermissions($attachment->getPermissions(), false, $attachment); } Timings::$permissibleCalculationTimer->stopTiming(); } /** * @param bool[] $children */ private function calculateChildPermissions(array $children, bool $invert, ?PermissionAttachment $attachment) : void{ $permManager = PermissionManager::getInstance(); foreach($children as $name => $v){ $perm = $permManager->getPermission($name); $value = ($v xor $invert); $this->permissions[$name] = new PermissionAttachmentInfo($name, $attachment, $value); $permManager->subscribeToPermission($name, $this->getRootPermissible()); if($perm instanceof Permission){ $this->calculateChildPermissions($perm->getChildren(), !$value, $attachment); } } } /** * @return PermissionAttachmentInfo[] */ public function getEffectivePermissions() : array{ return $this->permissions; } public function destroyCycles() : void{ PermissionManager::getInstance()->unsubscribeFromAllPermissions($this->getRootPermissible()); $this->permissions = []; //PermissionAttachmentInfo doesn't reference Permissible anymore, but it references PermissionAttachment which does $this->attachments = []; //this might still be a problem if the attachments are still referenced, but we can't do anything about that $this->parent = null; } }