3 #   FILE:  PrivilegeSet.php 
    5 #   Part of the Collection Workflow Integration System (CWIS) 
    6 #   Copyright 2013 Edward Almasy and Internet Scout Research Group 
    7 #   http://scout.wisc.edu/cwis/ 
   18     # used as a field ID in conditions to test whether a resources is 
   19     #       available as part of the privilege check 
   31         # if privilege data supplied 
   34             # if data is in legacy form (an array of privileges) 
   37                 # set internal privilege set from array 
   38                 $this->Privileges = $Data;
 
   42                 # set internal values from data 
   43                 $this->LoadFromData($Data);
 
   57     function Data($NewValue = NULL)
 
   59         # if new data supplied 
   60         if ($NewValue !== NULL)
 
   62             # unpack privilege data and load 
   63             $this->LoadFromData($NewValue);
 
   66         # serialize current data and return to caller 
   68         if (count($this->Privileges))
 
   70             foreach ($this->Privileges as $Priv)
 
   72                 $Data[
"Privileges"][] = is_object($Priv)
 
   73                         ? array(
"SUBSET" => $Priv->Data())
 
   77         $Data[
"Logic"] = $this->Logic;
 
   78         return serialize($Data);
 
   93         # when there are no requirements, then every user meets them 
   96         # for each privilege requirement 
   97         foreach ($this->Privileges as $Priv)
 
   99             # if privilege is actually a privilege subgroup 
  100             if (is_object($Priv))
 
  102                 # check if the subgroup is satisfied 
  103                 $Satisfied = $Priv->MeetsRequirements($User, $Resource);
 
  105             # else if privilege is actually a condition 
  106             elseif (is_array($Priv))
 
  108                 # check if condition is satisfied for the given resource 
  109                 $Satisfied = $this->MeetsCondition($Priv, $Resource, $User);
 
  111             # else privilege is actually a privilege 
  114                 # check if user has the spcified privilege 
  115                 $Satisfied = $User->
HasPriv( $Priv );
 
  118             # for AND logic, we can bail as soon as the first 
  119             # condition is not met 
  120             if ($this->Logic == 
"AND")
 
  127             # conversely, for OR logic, we can bail as soon as any 
  138         # report result of the test back to caller 
  150         # add privilege if not currently in set 
  153             if (is_object($Privilege)) {  $Privilege = $Privilege->Id();  }
 
  154             $this->Privileges[] = $Privilege;
 
  166         # remove privilege if currently in set 
  169             if (is_object($Privilege)) {  $Privilege = $Privilege->Id();  }
 
  170             $Index = array_search($Privilege, $this->Privileges);
 
  171             unset($this->Privileges[$Index]);
 
  182         # check whether privilege is in our list and report to caller 
  183         if (is_object($Privilege)) {  $Privilege = $Privilege->Id();  }
 
  184         return $this->IsInPrivilegeData($Privilege) ? TRUE : FALSE;
 
  197         # grab privilege information and add logic 
  198         $Info = $this->Privileges;
 
  199         $Info[
"Logic"] = $this->Logic;
 
  201         # return privilege info array to caller 
  213         # create list of privileges with conditions stripped out 
  215         foreach ($this->Privileges as $Priv)
 
  217             if (!is_array($Priv)) {  $List[] = $Priv;  }
 
  220         # return list of privileges to caller 
  241         $FieldId = is_object($Field) ? $Field->Id() : $Field;
 
  243         # set up condition array 
  245                 "FieldId" => intval($FieldId),
 
  246                 "Operator" => trim($Operator),
 
  249         # if condition is not already in set 
  250         if (!$this->IsInPrivilegeData($Condition))
 
  252             # add condition to privilege set 
  253             $this->Privileges[] = $Condition;
 
  271         $FieldId = is_object($Field) ? $Field->Id() : $Field;
 
  273         # set up condition array 
  275                 "FieldId" => intval($FieldId),
 
  276                 "Operator" => trim($Operator),
 
  279         # if condition is in set 
  280         if ($this->IsInPrivilegeData($Condition))
 
  282             # remove condition from privilege set 
  283             $Index = array_search($Condition, $this->Privileges);
 
  284             unset($this->Privileges[$Index]);
 
  294         # if subgroup is not already in set 
  295         if (!$this->IsInPrivilegeData($Set))
 
  297             # add subgroup to privilege set 
  298             $this->Privileges[] = $Set;
 
  313         if ($NewValue !== NULL)
 
  315             $this->Logic = $NewValue ? 
"AND" : 
"OR";
 
  317         return ($this->Logic == 
"AND") ? TRUE : FALSE;
 
  328         unset ($Info[
"Logic"]);
 
  331         foreach ($Info as $Item)
 
  333             if (is_object($Item))
 
  335                 $Result = array_merge($Result, $Item->PrivilegeFlagsChecked() );
 
  337             elseif (!is_array($Item))
 
  342         return array_unique($Result);
 
  354         unset ($Info[
"Logic"]);
 
  357         foreach ($Info as $Item)
 
  359             if (is_object($Item))
 
  361                 $Result = array_merge(
 
  363                     $Item->FieldsWithUserComparisons( $ComparisonType ) );
 
  365             elseif (is_array($Item))
 
  367                 if ($Item[
"Operator"] == $ComparisonType)
 
  373                         $Result []= $Item[
"FieldId"];
 
  379         return array_unique($Result);
 
  382     # ---- PRIVATE INTERFACE ------------------------------------------------- 
  384     private $Privileges = array();
 
  385     private $Logic = 
"OR";
 
  393     private function LoadFromData($Serialized)
 
  395         # save calling context in case load causes out-of-memory crash 
  396         $GLOBALS[
"AF"]->RecordContextInCaseOfCrash();
 
  399         $Data = unserialize($Serialized);
 
  401         # unpack privilege data (if available) and load 
  402         if (array_key_exists(
"Privileges", $Data))
 
  404             $this->Privileges = array();
 
  405             foreach ($Data[
"Privileges"] as $Priv)
 
  407                 if (is_array($Priv) && array_key_exists(
"SUBSET", $Priv))
 
  410                     $Subset->LoadFromData($Priv[
"SUBSET"]);
 
  411                     $this->Privileges[] = $Subset;
 
  415                     $this->Privileges[] = $Priv;
 
  420         # load logic if available 
  421         if (array_key_exists(
"Logic", $Data))
 
  423             $this->Logic = $Data[
"Logic"];
 
  434     private function MeetsCondition($Condition, $Resource, $User)
 
  436         # if condition is a check for whether a resource is available 
  437         if ($Condition[
"FieldId"] == self::HAVE_RESOURCE)
 
  439             # return a result based on whether a resource is available 
  440             return ((
bool)($Resource == self::NO_RESOURCE)
 
  441                     != (
bool)$Condition[
"Value"]) ? TRUE : FALSE;
 
  443         # else if no resource is available 
  444         elseif ($Resource == self::NO_RESOURCE)
 
  446             # return a result that in effect ignores the condition 
  447             return ($this->Logic == 
"AND") ? TRUE : FALSE;
 
  449         # else if resource is valid 
  450         elseif ($Resource instanceof 
Resource)
 
  452             # pre-process condition parameters based on type of field 
  459                 # if the field in a condition was invalid, the condition fails 
  463             $Operator = $Condition[
"Operator"];
 
  464             $Value = $Condition[
"Value"];
 
  465             $FieldValue  = $Resource->Get($Field, TRUE);
 
  466             switch ($Field->Type())
 
  469                     # if supplied value is NULL 
  472                         $Value = $User->Id();
 
  475                     # convert field value to user ID 
  476                     $FieldValue = $FieldValue->Id();
 
  481                     # date field values are Date objects, so handle those 
  482                     if ($FieldValue instanceof 
Date)
 
  484                         $FieldValue = strtotime($FieldValue->Formatted());
 
  487                     # timestamp field values are just the date/time string 
  490                         $FieldValue = strtotime($FieldValue);
 
  493                     # use the current time for the value if it's NULL 
  499                     # otherwise, parse the value to get a numeric timestamp 
  502                         $Value = strtotime($Value);
 
  511                     # for options, construct a list of the CNIDs in this field 
  513                     foreach ($FieldValue as $CName)
 
  515                         $NewValue []= $CName->Id();
 
  517                     $FieldValue = $NewValue;
 
  521                     throw new Exception(
"Unsupported metadata field type (" 
  522                             .print_r($Field->Type(), TRUE)
 
  523                             .
") for condition in privilege set.");
 
  527             # compare field value and supplied value using specified operator 
  531                     if (is_array($FieldValue))
 
  533                         # equality against an option field is a 'contains' condition, 
  534                         # true if the specified value is one of those set 
  536                         foreach ($FieldValue as $FieldValue_i)
 
  538                             $Result |= ($FieldValue_i == $Value);
 
  543                         $Result = ($FieldValue == $Value);
 
  548                     if (is_array($FieldValue))
 
  550                         # not equal against an option field is 'does not contains', 
  551                         # true as long as the spcified value is not one of those set 
  553                         foreach ($FieldValue as $FieldValue_i)
 
  555                             $Result &= ($FieldValue_i != $Value);
 
  560                         $Result = ($FieldValue != $Value);
 
  565                     $Result = ($FieldValue < $Value);
 
  569                     $Result = ($FieldValue > $Value);
 
  573                     $Result = ($FieldValue <= $Value);
 
  577                     $Result = ($FieldValue >= $Value);
 
  581                     throw new Exception(
"Unsupported condition operator (" 
  582                             .print_r($Operator, TRUE).
") in privilege set.");
 
  586             # report to caller whether condition was met 
  587             return $Result ? TRUE : FALSE;
 
  591             # error out because resource was illegal 
  592             throw new Exception(
"Invalid Resource passed in for privilege" 
  593                     .
" set comparison.");
 
  605     private function IsInPrivilegeData($Item)
 
  607         # step through privilege data 
  608         foreach ($this->Privileges as $Priv)
 
  610             # report to caller if item is found 
  611             if (is_object($Item))
 
  613                 if (is_object($Priv) && ($Item == $Priv)) {  
return TRUE;  }
 
  615             elseif (is_array($Item))
 
  617                 if (is_array($Priv) && ($Item == $Priv)) {  
return TRUE;  }
 
  619             elseif ($Item == $Priv) {  
return TRUE;  }
 
  622         # report to caller that item is not in privilege data 
AddSet(PrivilegeSet $Set)
Add subgroup of privileges/conditions to set. 
HasPriv($Privilege, $Privileges=NULL)
Determine if a user has a given privilege, or satisfies the conditions specified by a given privilege...
Set of privileges used to access resource information or other parts of the system. 
MeetsRequirements(CWUser $User, $Resource=self::NO_RESOURCE)
Determine if a given user meets the requirements specified by this PrivilegeSet. 
__construct($Data=NULL)
Class constructor, used to create a new set or reload an existing set from previously-constructed dat...
IncludesPrivilege($Privilege)
Check whether this privilege set includes the specified privilege. 
GetPrivilegeInfo()
Get privilege information as an array, with numerical indexes except for the logic, which is contained in a element with the index "Logic". 
PrivilegeFlagsChecked()
List which privilege flags (e.g. 
GetPrivilegeList()
Get list of privileges. 
Data($NewValue=NULL)
Get/set privilege set data, in the form of an opaque string. 
Represents a "resource" in CWIS. 
FieldsWithUserComparisons($ComparisonType)
List which fields in this privset are involved in UserIs or UserIsNot comparisons for this privilege ...
AddPrivilege($Privilege)
Add specified privilege to set. 
CWIS-specific user class. 
AddCondition($Field, $Value=NULL, $Operator="==")
Add condition to privilege set. 
RemovePrivilege($Privilege)
Remove specified privilege from set. 
AllRequired($NewValue=NULL)
Get/set whether all privileges/conditions in set are required (i.e. 
RemoveCondition($Field, $Value=NULL, $Operator="==")
Remove condition from privilege set.