1568 lines
		
	
	
		
			51 KiB
		
	
	
	
		
			PHP
		
	
	
	
			
		
		
	
	
			1568 lines
		
	
	
		
			51 KiB
		
	
	
	
		
			PHP
		
	
	
	
| <?php
 | |
| /**
 | |
|  * Classes and functions for the template engine.
 | |
|  *
 | |
|  * Templates are either:
 | |
|  * + Creating or Editing, (a Container or DN passed to the object)
 | |
|  * + A predefined template, or a default template (template ID passed to the object)
 | |
|  *
 | |
|  * The template object will know which attributes are mandatory (MUST
 | |
|  * attributes) and which attributes are optional (MAY attributes). It will also
 | |
|  * contain a list of optional attributes. These are attributes that the schema
 | |
|  * will allow data for (they are MAY attributes), but the template has not
 | |
|  * included a definition for them.
 | |
|  *
 | |
|  * The template object will be invalidated if it does contain the necessary
 | |
|  * items (objectClass, MUST attributes, etc) to make a successful LDAP update.
 | |
|  *
 | |
|  * @author The phpLDAPadmin development team
 | |
|  * @package phpLDAPadmin
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * Template Class
 | |
|  *
 | |
|  * @package phpLDAPadmin
 | |
|  * @subpackage Templates
 | |
|  * @todo RDN attributes should be treated as MUST attributes even though the schema marks them as MAY
 | |
|  * @todo RDN attributes need to be checked that are included in the schema, otherwise mark it is invalid
 | |
|  * @todo askcontainer is no longer used?
 | |
|  */
 | |
| class Template extends xmlTemplate {
 | |
| 	# If this template visible on the template choice list
 | |
| 	private $visible = true;
 | |
| 	# Is this template valid after parsing the XML file
 | |
| 	private $invalid = false;
 | |
| 	private $invalid_admin = false;
 | |
| 	private $invalid_reason;
 | |
| 	# The TEMPLATE structural objectclasses
 | |
| 	protected $structural_oclass = array();
 | |
| 	protected $description = '';
 | |
| 	# Is this a read-only template (only valid in modification templates)
 | |
| 	private $readonly = false;
 | |
| 
 | |
| 	# If this is set, it means we are editing an entry.
 | |
| 	private $dn;
 | |
| 	# Where this template will store its data
 | |
| 	protected $container;
 | |
| 	# Does this template prohibit children being created
 | |
| 	private $noleaf = false;
 | |
| 	# A regexp that determines if this template is valid in the container.
 | |
| 	private $regexp;
 | |
| 	# Template Title
 | |
| 	public $title;
 | |
| 	# Icon for the template
 | |
| 	private $icon;
 | |
| 	# Template RDN attributes
 | |
| 	private $rdn;
 | |
| 
 | |
| 	public function __construct($server_id,$name=null,$filename=null,$type=null,$id=null) {
 | |
| 		parent::__construct($server_id,$name,$filename,$type,$id);
 | |
| 
 | |
| 		# If this is the default template, we might disable leafs by default.
 | |
| 		if (is_null($filename))
 | |
| 			$this->noleaf = $_SESSION[APPCONFIG]->getValue('appearance','disable_default_leaf');
 | |
| 	}
 | |
| 
 | |
| 	public function __clone() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		# We need to clone our attributes, when passing back a template with getTemplate
 | |
| 		foreach ($this->attributes as $key => $value)
 | |
| 			$this->attributes[$key] = clone $value;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Main processing to store the template.
 | |
| 	 *
 | |
| 	 * @param xmldata Parsed xmldata from xml2array object
 | |
| 	 */
 | |
| 	protected function storeTemplate($xmldata) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$server = $this->getServer();
 | |
| 		$objectclasses = array();
 | |
| 
 | |
| 		foreach ($xmldata['template'] as $xml_key => $xml_value) {
 | |
| 			if (DEBUG_ENABLED)
 | |
| 				debug_log('Foreach loop Key [%s] Value [%s]',4,0,__FILE__,__LINE__,__METHOD__,$xml_key,is_array($xml_value));
 | |
| 
 | |
| 			switch ($xml_key) {
 | |
| 				# Build our object Classes from the DN and Template.
 | |
| 				case ('objectclasses'):
 | |
| 					if (DEBUG_ENABLED)
 | |
| 						debug_log('Case [%s]',4,0,__FILE__,__LINE__,__METHOD__,$xml_key);
 | |
| 
 | |
| 					if (isset($xmldata['template'][$xml_key]['objectclass']))
 | |
| 						if (is_array($xmldata['template'][$xml_key]['objectclass'])) {
 | |
| 							foreach ($xmldata['template'][$xml_key]['objectclass'] as $index => $details) {
 | |
| 
 | |
| 								# XML files with only 1 objectClass dont have a numeric index.
 | |
| 								$soc = $server->getSchemaObjectClass(strtolower($details));
 | |
| 
 | |
| 								# If we havent recorded this objectclass already, do so now.
 | |
| 								if (is_object($soc) && ! in_array($soc->getName(),$objectclasses))
 | |
| 									array_push($objectclasses,$soc->getName(false));
 | |
| 
 | |
| 								elseif (! is_object($soc) && ! $_SESSION[APPCONFIG]->getValue('appearance','hide_template_warning'))
 | |
| 									system_message(array(
 | |
| 										'title'=>('Automatically removed objectClass from template'),
 | |
| 										'body'=>sprintf('%s: <b>%s</b> %s',$this->getTitle(),$details,('removed from template as it is not defined in the schema')),
 | |
| 										'type'=>'warn'));
 | |
| 							}
 | |
| 
 | |
| 						} else {
 | |
| 							# XML files with only 1 objectClass dont have a numeric index.
 | |
| 							$soc = $server->getSchemaObjectClass(strtolower($xmldata['template'][$xml_key]['objectclass']));
 | |
| 
 | |
| 							# If we havent recorded this objectclass already, do so now.
 | |
| 							if (is_object($soc) && ! in_array($soc->getName(),$objectclasses))
 | |
| 								array_push($objectclasses,$soc->getName(false));
 | |
| 						}
 | |
| 
 | |
| 					break;
 | |
| 
 | |
| 				# Build our attribute list from the DN and Template.
 | |
| 				case ('attributes'):
 | |
| 					if (DEBUG_ENABLED)
 | |
| 						debug_log('Case [%s]',4,0,__FILE__,__LINE__,__METHOD__,$xml_key);
 | |
| 
 | |
| 					if (is_array($xmldata['template'][$xml_key])) {
 | |
| 						foreach ($xmldata['template'][$xml_key] as $tattrs)
 | |
| 							foreach ($tattrs as $index => $details) {
 | |
| 								if (DEBUG_ENABLED)
 | |
| 									debug_log('Foreach tattrs Key [%s] Value [%s]',4,0,__FILE__,__LINE__,__METHOD__,
 | |
| 										$index,$details);
 | |
| 
 | |
| 								# If there is no schema definition for the attribute, it will be ignored.
 | |
| 								if ($sattr = $server->getSchemaAttribute($index))
 | |
| 									if (is_null($this->getAttribute($sattr->getName())))
 | |
| 										$this->addAttribute($sattr->getName(),$details,'XML');
 | |
| 							}
 | |
| 
 | |
| 						masort($this->attributes,'order');
 | |
| 					}
 | |
| 
 | |
| 					break;
 | |
| 
 | |
| 				default:
 | |
| 					if (DEBUG_ENABLED)
 | |
| 						debug_log('Case [%s]',4,0,__FILE__,__LINE__,__METHOD__,$xml_key);
 | |
| 
 | |
| 					# Some key definitions need to be an array, some must not be:
 | |
| 					$allowed_arrays = array('rdn');
 | |
| 					$storelower = array('rdn');
 | |
| 					$storearray = array('rdn');
 | |
| 
 | |
| 					# Items that must be stored lowercase
 | |
| 					if (in_array($xml_key,$storelower))
 | |
| 						if (is_array($xml_value))
 | |
| 							foreach ($xml_value as $index => $value)
 | |
| 								$xml_value[$index] = strtolower($value);
 | |
| 						else
 | |
| 							$xml_value = strtolower($xml_value);
 | |
| 
 | |
| 					# Items that must be stored as arrays
 | |
| 					if (in_array($xml_key,$storearray) && ! is_array($xml_value))
 | |
| 						$xml_value = array($xml_value);
 | |
| 
 | |
| 					# Items that should not be an array
 | |
| 					if (! in_array($xml_key,$allowed_arrays) && is_array($xml_value)) {
 | |
| 						debug_dump(array(__METHOD__,'key'=>$xml_key,'value'=>$xml_value));
 | |
| 						error(sprintf(('In the XML file (%s), [%s] is an array, it must be a string.'),
 | |
| 							$this->filename,$xml_key),'error');
 | |
| 					}
 | |
| 
 | |
| 					$this->$xml_key = $xml_value;
 | |
| 
 | |
| 					if ($xml_key == 'invalid' && $xml_value)
 | |
| 						$this->setInvalid(('Disabled by XML configuration'),true);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (! count($objectclasses)) {
 | |
| 			$this->setInvalid(('ObjectClasses in XML dont exist in LDAP server.'));
 | |
| 			return;
 | |
| 
 | |
| 		} else {
 | |
| 			$attribute = $this->addAttribute('objectClass',array('values'=>$objectclasses),'XML');
 | |
| 			$attribute->justModified();
 | |
| 			$attribute->setRequired();
 | |
| 			$attribute->hide();
 | |
| 		}
 | |
| 
 | |
| 		$this->rebuildTemplateAttrs();
 | |
| 
 | |
| 		# Check we have some manditory items.
 | |
| 		foreach (array('rdn','structural_oclass','visible') as $key) {
 | |
| 			if (! isset($this->$key)
 | |
| 				|| (! is_array($this->$key) && ! trim($this->$key))) {
 | |
| 
 | |
| 				$this->setInvalid(sprintf(('Missing %s in the XML file.'),$key));
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		# Mark our RDN attributes as RDN
 | |
| 		$counter = 1;
 | |
| 		foreach ($this->rdn as $key) {
 | |
| 			if ((is_null($attribute = $this->getAttribute($key))) && (in_array_ignore_case('extensibleobject',$this->getObjectClasses()))) {
 | |
| 				$attribute = $this->addAttribute($key,array('values'=>array()));
 | |
| 				$attribute->show();
 | |
| 			}
 | |
| 
 | |
| 			if (! is_null($attribute))
 | |
| 				$attribute->setRDN($counter++);
 | |
| 			elseif ($this->isType('creation'))
 | |
| 				$this->setInvalid(sprintf(('Missing RDN attribute %s in the XML file.'),$key));
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Is default templates enabled?
 | |
| 	 * This will disable the default template from the engine.
 | |
| 	 *
 | |
| 	 * @return boolean
 | |
| 	 */
 | |
| 	protected function hasDefaultTemplate() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if ($_SESSION[APPCONFIG]->getValue('appearance','disable_default_template'))
 | |
| 			return false;
 | |
| 		else
 | |
| 			return true;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return the templates of type (creation/modification)
 | |
| 	 *
 | |
| 	 * @param $string type - creation/modification
 | |
| 	 * @return array - Array of templates of that type
 | |
| 	 */
 | |
| 	protected function readTemplates($type) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$template_xml = new Templates($this->server_id);
 | |
| 		return $template_xml->getTemplates($type);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * This function will perform the following intialisation steps:
 | |
| 	 * + If a DN is set, query the ldap and load the object
 | |
| 	 * + Read our $_REQUEST variable and set the values
 | |
| 	 * After this action, the template should self describe as to whether it is an update, create
 | |
| 	 * or delete.
 | |
| 	 * (OLD values are IGNORED, we will have got them when we build this object from the LDAP server DN.)
 | |
| 	 */
 | |
| 	public function accept($makeVisible=false) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$server = $this->getServer();
 | |
| 
 | |
| 		# If a DN is set, then query the LDAP server for the details.
 | |
| 		if ($this->dn) {
 | |
| 			if (! $server->dnExists($this->dn))
 | |
| 				system_message(array(
 | |
| 					'title'=>__METHOD__,
 | |
| 					'body'=>sprintf('DN (%s) didnt exist in LDAP?',$this->dn),
 | |
| 					'type'=>'info'));
 | |
| 
 | |
| 			$rdnarray = rdn_explode(strtolower(get_rdn(dn_escape($this->dn))));
 | |
| 
 | |
| 			$counter = 1;
 | |
| 			foreach ($server->getDNAttrValues($this->dn,null,LDAP_DEREF_NEVER,array_merge(array('*'),$server->getValue('server','custom_attrs'))) as $attr => $values) {
 | |
| 				# We ignore DNs.
 | |
| 				if ($attr == 'dn')
 | |
| 					continue;
 | |
| 
 | |
| 				$attribute = $this->getAttribute($attr);
 | |
| 
 | |
| 				if (is_null($attribute))
 | |
| 					$attribute = $this->addAttribute($attr,array('values'=>$values));
 | |
| 				else
 | |
| 					if ($attribute->getValues()) {
 | |
| 						# Override values to those that are defined in the XML file.
 | |
| 						if ($attribute->getSource() != 'XML')
 | |
| 							$attribute->setValue(array_values($values));
 | |
| 						else
 | |
| 							$attribute->setOldValue(array_values($values));
 | |
| 
 | |
| 					} else
 | |
| 						$attribute->initValue(array_values($values));
 | |
| 
 | |
| 				# Work out the RDN attributes
 | |
| 				foreach ($attribute->getValues() as $index => $value)
 | |
| 					if (in_array(sprintf('%s=%s',
 | |
| 						$attribute->getName(),strtolower($attribute->getValue($index))),$rdnarray))
 | |
| 						$attribute->setRDN($counter++);
 | |
| 
 | |
| 				if ($makeVisible)
 | |
| 					$attribute->show();
 | |
| 			}
 | |
| 
 | |
| 			# Get the Internal Attributes
 | |
| 			foreach ($server->getDNAttrValues($this->dn,null,LDAP_DEREF_NEVER,array_merge(array('+'),$server->getValue('server','custom_sys_attrs'))) as $attr => $values) {
 | |
| 				$attribute = $this->getAttribute($attr);
 | |
| 
 | |
| 				if (is_null($attribute))
 | |
| 					$attribute = $this->addAttribute($attr,array('values'=>$values));
 | |
| 				else
 | |
| 					if ($attribute->getValues())
 | |
| 						$attribute->setValue(array_values($values));
 | |
| 					else
 | |
| 						$attribute->initValue(array_values($values));
 | |
| 
 | |
| 				if (! in_array_ignore_case($attribute->getName(),$server->getValue('server','custom_attrs')))
 | |
| 					$attribute->setInternal();
 | |
| 			}
 | |
| 
 | |
| 		# If this is the default template, and our $_REQUEST has defined our objectclass, then query the schema to get the attributes
 | |
| 		} elseif ($this->container) {
 | |
| 			if ($this->isType('default') && ! count($this->getAttributes(true)) && isset($_REQUEST['new_values']['objectclass'])) {
 | |
| 				$attribute = $this->addAttribute('objectclass',array('values'=>$_REQUEST['new_values']['objectclass']));
 | |
| 				$attribute->justModified();
 | |
| 				$this->rebuildTemplateAttrs();
 | |
| 				unset($_REQUEST['new_values']['objectclass']);
 | |
| 			}
 | |
| 
 | |
| 		} elseif (get_request('create_base')) {
 | |
| 			if (get_request('rdn')) {
 | |
| 				$rdn = explode('=',get_request('rdn'));
 | |
| 				$attribute = $this->addAttribute($rdn[0],array('values'=>array($rdn[1])));
 | |
| 				$attribute->setRDN(1);
 | |
| 			}
 | |
| 
 | |
| 		} else {
 | |
| 			debug_dump_backtrace('No DN or CONTAINER?',1);
 | |
| 		}
 | |
| 
 | |
| 		# Read in our new values.
 | |
| 		foreach (array('new_values') as $key) {
 | |
| 			if (isset($_REQUEST[$key]))
 | |
| 				foreach ($_REQUEST[$key] as $attr => $values) {
 | |
| 					# If it isnt an array, silently ignore it.
 | |
| 					if (! is_array($values))
 | |
| 						continue;
 | |
| 
 | |
| 					# If _REQUEST['skip_array'] with this attr set, we'll ignore this new_value
 | |
| 					if (isset($_REQUEST['skip_array'][$attr]) && $_REQUEST['skip_array'][$attr] == 'on')
 | |
| 						continue;
 | |
| 
 | |
| 					# Prune out entries with a blank value.
 | |
| 					foreach ($values as $index => $value)
 | |
| 						if (! strlen(trim($value)))
 | |
| 							unset($values[$index]);
 | |
| 
 | |
| 					$attribute = $this->getAttribute($attr);
 | |
| 					# If the attribute is null, then no attribute exists, silently ignore it (unless this is the default template)
 | |
| 					if (is_null($attribute) && (! $this->isType('default') && ! $this->isType(null)))
 | |
| 						continue;
 | |
| 
 | |
| 					# If it is a binary attribute, the post should have base64 encoded the value, we'll need to reverse that
 | |
| 					if ($server->isAttrBinary($attr))
 | |
| 						foreach ($values as $index => $value)
 | |
| 							$values[$index] = base64_decode($value);
 | |
| 
 | |
| 					if (is_null($attribute)) {
 | |
| 						$attribute = $this->addAttribute($attr,array('values'=>$values));
 | |
| 
 | |
| 						if (count($values))
 | |
| 							$attribute->justModified();
 | |
| 
 | |
| 					} else
 | |
| 						$attribute->setValue(array_values($values));
 | |
| 				}
 | |
| 
 | |
| 			# Read in our new binary values
 | |
| 			if (isset($_FILES[$key]['name']))
 | |
| 				foreach ($_FILES[$key]['name'] as $attr => $values) {
 | |
| 					$new_values = array();
 | |
| 
 | |
| 					foreach ($values as $index => $details) {
 | |
| 						# Ignore empty files
 | |
| 						if (! $_FILES[$key]['size'][$attr][$index])
 | |
| 							continue;
 | |
| 
 | |
| 						if (! is_uploaded_file($_FILES[$key]['tmp_name'][$attr][$index])) {
 | |
| 							if (isset($_FILES[$key]['error'][$attr][$index]))
 | |
| 								switch($_FILES[$key]['error'][$attr][$index]) {
 | |
| 
 | |
| 									# No error; possible file attack!
 | |
| 									case 0:
 | |
| 										$msg = _('Security error: The file being uploaded may be malicious.');
 | |
| 										break;
 | |
| 
 | |
| 									# Uploaded file exceeds the upload_max_filesize directive in php.ini
 | |
| 									case 1:
 | |
| 										$msg = _('The file you uploaded is too large. Please check php.ini, upload_max_size setting');
 | |
| 										break;
 | |
| 
 | |
| 									# Uploaded file exceeds the MAX_FILE_SIZE directive specified in the html form
 | |
| 									case 2:
 | |
| 										$msg = _('The file you uploaded is too large. Please check php.ini, upload_max_size setting');
 | |
| 										break;
 | |
| 
 | |
| 									# Uploaded file was only partially uploaded
 | |
| 									case 3:
 | |
| 										$msg = _('The file you selected was only partially uploaded, likley due to a network error.');
 | |
| 										break;
 | |
| 
 | |
| 									# No file was uploaded
 | |
| 									case 4:
 | |
| 										$msg = _('You left the attribute value blank. Please go back and try again.');
 | |
| 										break;
 | |
| 
 | |
| 									# A default error, just in case! :)
 | |
| 									default:
 | |
| 										$msg = _('Security error: The file being uploaded may be malicious.');
 | |
| 										break;
 | |
| 								}
 | |
| 
 | |
| 							else
 | |
| 								$msg = _('Security error: The file being uploaded may be malicious.');
 | |
| 
 | |
| 							system_message(array(
 | |
| 								'title'=>_('Error'),'body'=>$msg,'type'=>'warn'));
 | |
| 
 | |
| 						} else {
 | |
| 							$binaryfile = array();
 | |
| 							$binaryfile['name'] = $_FILES[$key]['tmp_name'][$attr][$index];
 | |
| 							$binaryfile['handle'] = fopen($binaryfile['name'],'r');
 | |
| 							$binaryfile['data'] = fread($binaryfile['handle'],filesize($binaryfile['name']));
 | |
| 							fclose($binaryfile['handle']);
 | |
| 
 | |
| 							$new_values[$index] = $binaryfile['data'];
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					if (count($new_values)) {
 | |
| 						$attribute = $this->getAttribute($attr);
 | |
| 
 | |
| 						if (is_null($attribute))
 | |
| 							$attribute = $this->addAttribute($attr,array('values'=>$new_values));
 | |
| 						else
 | |
| 							foreach ($new_values as $value)
 | |
| 								$attribute->addValue($value);
 | |
| 
 | |
| 						$attribute->justModified();
 | |
| 					}
 | |
| 				}
 | |
| 		}
 | |
| 
 | |
| 		# If there are any single item additions (from the add_attr form for example)
 | |
| 		if (isset($_REQUEST['single_item_attr'])) {
 | |
| 			if (isset($_REQUEST['single_item_value'])) {
 | |
| 				if (! is_array($_REQUEST['single_item_value']))
 | |
| 					$values = array($_REQUEST['single_item_value']);
 | |
| 				else
 | |
| 					$values = $_REQUEST['single_item_value'];
 | |
| 
 | |
| 			} elseif (isset($_REQUEST['binary'])) {
 | |
| 				/* Special case for binary attributes (like jpegPhoto and userCertificate):
 | |
| 				 * we must go read the data from the file and override $_REQUEST['single_item_value'] with the
 | |
| 				 * binary data. Secondly, we must check if the ";binary" option has to be appended to the name
 | |
| 				 * of the attribute. */
 | |
| 
 | |
| 				if ($_FILES['single_item_value']['size'] === 0)
 | |
| 					system_message(array(
 | |
| 						'title'=>_('Error'),
 | |
| 						'body'=>sprintf('%s %s',_('The file you chose is either empty or does not exist.'),_('Please go back and try again.')),
 | |
| 						'type'=>'warn'));
 | |
| 
 | |
| 				else {
 | |
| 					if (! is_uploaded_file($_FILES['single_item_value']['tmp_name'])) {
 | |
| 						if (isset($_FILES['single_item_value']['error']))
 | |
| 							switch($_FILES['single_item_value']['error']) {
 | |
| 
 | |
| 								# No error; possible file attack!
 | |
| 								case 0:
 | |
| 									$msg = _('Security error: The file being uploaded may be malicious.');
 | |
| 									break;
 | |
| 
 | |
| 								# Uploaded file exceeds the upload_max_filesize directive in php.ini
 | |
| 								case 1:
 | |
| 									$msg = _('The file you uploaded is too large. Please check php.ini, upload_max_size setting');
 | |
| 									break;
 | |
| 
 | |
| 								# Uploaded file exceeds the MAX_FILE_SIZE directive specified in the html form
 | |
| 								case 2:
 | |
| 									$msg = _('The file you uploaded is too large. Please check php.ini, upload_max_size setting');
 | |
| 									break;
 | |
| 
 | |
| 								# Uploaded file was only partially uploaded
 | |
| 								case 3:
 | |
| 									$msg = _('The file you selected was only partially uploaded, likley due to a network error.');
 | |
| 									break;
 | |
| 
 | |
| 								# No file was uploaded
 | |
| 								case 4:
 | |
| 									$msg = _('You left the attribute value blank. Please go back and try again.');
 | |
| 									break;
 | |
| 
 | |
| 								# A default error, just in case! :)
 | |
| 								default:
 | |
| 									$msg = _('Security error: The file being uploaded may be malicious.');
 | |
| 									break;
 | |
| 							}
 | |
| 
 | |
| 						else
 | |
| 							$msg = _('Security error: The file being uploaded may be malicious.');
 | |
| 
 | |
| 						system_message(array(
 | |
| 							'title'=>_('Error'),'body'=>$msg,'type'=>'warn'),'index.php');
 | |
| 					}
 | |
| 
 | |
| 					$binaryfile = array();
 | |
| 					$binaryfile['name'] = $_FILES['single_item_value']['tmp_name'];
 | |
| 					$binaryfile['handle'] = fopen($binaryfile['name'],'r');
 | |
| 					$binaryfile['data'] = fread($binaryfile['handle'],filesize($binaryfile['name']));
 | |
| 					fclose($binaryfile['handle']);
 | |
| 
 | |
| 					$values = array($binaryfile['data']);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (count($values)) {
 | |
| 				$attribute = $this->getAttribute($_REQUEST['single_item_attr']);
 | |
| 
 | |
| 				if (is_null($attribute))
 | |
| 					$attribute = $this->addAttribute($_REQUEST['single_item_attr'],array('values'=>$values));
 | |
| 				else
 | |
| 					$attribute->setValue(array_values($values));
 | |
| 
 | |
| 				$attribute->justModified();
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		# If this is the default creation template, we need to set some additional values
 | |
| 		if ($this->isType('default') && $this->getContext() == 'create') {
 | |
| 			# Load our schema, based on the objectclasses that may have already been defined.
 | |
| 			if (! get_request('create_base'))
 | |
| 				$this->rebuildTemplateAttrs();
 | |
| 
 | |
| 			# Set the RDN attribute
 | |
| 			$counter = 1;
 | |
| 			foreach (get_request('rdn_attribute','REQUEST',false,array()) as $key => $value) {
 | |
| 				$attribute = $this->getAttribute($value);
 | |
| 
 | |
| 				if (! is_null($attribute))
 | |
| 					$attribute->setRDN($counter++);
 | |
| 
 | |
| 				else {
 | |
| 					system_message(array(
 | |
| 						'title'=>_('No RDN attribute was selected.'),
 | |
| 						'body'=>_('No RDN attribute was selected.'),
 | |
| 						'type'=>'warn'),'index.php');
 | |
| 
 | |
| 					die();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the DN for this template, if we are editing entries
 | |
| 	 *
 | |
| 	 * @param dn The DN of the entry
 | |
| 	 */
 | |
| 	public function setDN($dn) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if (isset($this->container))
 | |
| 			system_message(array(
 | |
| 				'title'=>__METHOD__,
 | |
| 				'body'=>'CONTAINER set while setting DN',
 | |
| 				'type'=>'info'));
 | |
| 
 | |
| 		$this->dn = $dn;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the RDN attributes
 | |
| 	 * Given an RDN, mark the attributes as RDN attributes. If there is no defined attribute,
 | |
| 	 * then the remaining RDNs will be returned.
 | |
| 	 *
 | |
| 	 * @param RDN
 | |
| 	 * @return RDN attributes not processed
 | |
| 	 */
 | |
| 	public function setRDNAttributes($rdn) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		# Setup to work out our RDN.
 | |
| 		$rdnarray = rdn_explode($rdn);
 | |
| 
 | |
| 		$counter = 1;
 | |
| 		foreach ($this->getAttributes(true) as $attribute)
 | |
| 			foreach ($rdnarray as $index => $rdnattr) {
 | |
| 				list($attr,$value) = explode('=',$rdnattr);
 | |
| 
 | |
| 				if (strtolower($attr) == $attribute->getName()) {
 | |
| 					$attribute->setRDN($counter++);
 | |
| 					unset($rdnarray[$index]);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 		return $rdnarray;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Display the DN for this template entry. If the DN is not set (creating a new entry), then
 | |
| 	 * a generated DN will be produced, taken from the RDN and the CONTAINER details.
 | |
| 	 *
 | |
| 	 * @return dn
 | |
| 	 */
 | |
| 	public function getDN() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs,$this->dn);
 | |
| 
 | |
| 		if ($this->dn)
 | |
| 			return $this->dn;
 | |
| 
 | |
| 		# If DN is not set, our DN will be made from our RDN and Container.
 | |
| 		elseif ($this->getRDN() && $this->getContainer())
 | |
| 			return sprintf('%s,%s',$this->getRDN(),$this->GetContainer());
 | |
| 
 | |
| 		# If container is not set, we're probably creating the base
 | |
| 		elseif ($this->getRDN() && get_request('create_base'))
 | |
| 			return $this->getRDN();
 | |
| 	}
 | |
| 
 | |
| 	public function getDNEncode($url=true) {
 | |
| 		// @todo Be nice to do all this in 1 location
 | |
| 		if ($url)
 | |
| 			return urlencode(preg_replace('/%([0-9a-fA-F]+)/',"%25\\1",$this->getDN()));
 | |
| 		else
 | |
| 			return preg_replace('/%([0-9a-fA-F]+)/',"%25\\1",$this->getDN());
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the container for this template, if we are creating entries
 | |
| 	 *
 | |
| 	 * @param dn The DN of the container
 | |
| 	 * @todo Trigger a query to the LDAP server and generate an error if the container doesnt exist
 | |
| 	 */
 | |
| 	public function setContainer($container) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if (isset($this->dn))
 | |
| 			system_message(array(
 | |
| 				'title'=>__METHOD__,
 | |
| 				'body'=>'DN set while setting CONTAINER',
 | |
| 				'type'=>'info'));
 | |
| 
 | |
| 		$this->container = $container;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the DN of the container for this entry
 | |
| 	 *
 | |
| 	 * @return dn DN of the container
 | |
| 	 */
 | |
| 	public function getContainer() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,1,__FILE__,__LINE__,__METHOD__,$fargs,$this->container);
 | |
| 
 | |
| 		return $this->container;
 | |
| 	}
 | |
| 
 | |
| 	public function getContainerEncode($url=true) {
 | |
| 		// @todo Be nice to do all this in 1 location
 | |
| 		if ($url)
 | |
| 			return urlencode(preg_replace('/%([0-9a-fA-F]+)/',"%25\\1",$this->container));
 | |
| 		else
 | |
| 			return preg_replace('/%([0-9a-fA-F]+)/',"%25\\1",$this->container);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Copy a DN
 | |
| 	 */
 | |
| 	public function copy($template,$rdn,$asnew=false) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$rdnarray = rdn_explode($rdn);
 | |
| 
 | |
| 		$counter = 1;
 | |
| 		foreach ($template->getAttributes(true) as $sattribute) {
 | |
| 			$attribute = $this->addAttribute($sattribute->getName(false),array('values'=>$sattribute->getValues()));
 | |
| 
 | |
| 			# Set our new RDN, and its values
 | |
| 			if (is_null($attribute)) {
 | |
| 				debug_dump_backtrace('Attribute is null, it probably doesnt exist in the destination server?');
 | |
| 
 | |
| 			} else {
 | |
| 
 | |
| 				# Mark our internal attributes.
 | |
| 				if ($sattribute->isInternal())
 | |
| 					$attribute->setInternal();
 | |
| 
 | |
| 				$modified = false;
 | |
| 				foreach ($rdnarray as $index => $rdnattr) {
 | |
| 					list($attr,$value) = explode('=',$rdnattr);
 | |
| 					if (strtolower($attr) == $attribute->getName()) {
 | |
| 
 | |
| 						# If this is already marked as an RDN, then this multivalue RDN was updated on a previous loop
 | |
| 						if (! $modified) {
 | |
| 							$attribute->setValue(array($value));
 | |
| 							$attribute->setRDN($counter++);
 | |
| 							$modified = true;
 | |
| 
 | |
| 						} else {
 | |
| 							$attribute->addValue($value);
 | |
| 						}
 | |
| 
 | |
| 						# This attribute has been taken care of, we'll drop it from our list.
 | |
| 						unset($rdnarray[$index]);
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// @todo If this is a Jpeg Attribute, we need to mark it read only, since it cant be deleted like text attributes can
 | |
| 			if (strcasecmp(get_class($attribute),'jpegAttribute') == 0)
 | |
| 				$attribute->setReadOnly();
 | |
| 		}
 | |
| 
 | |
| 		# If we have any RDN values left over, there werent in the original entry and need to be added.
 | |
| 		foreach ($rdnarray as $rdnattr) {
 | |
| 			list($attr,$value) = explode('=',$rdnattr);
 | |
| 
 | |
| 			$attribute = $this->addAttribute($attr,array('values'=>array($value)));
 | |
| 
 | |
| 			if (is_null($attribute))
 | |
| 				debug_dump_backtrace('Attribute is null, it probably doesnt exist in the destination server?');
 | |
| 			else
 | |
| 				$attribute->setRDN($counter++);
 | |
| 		}
 | |
| 
 | |
| 		# If we are copying into a new entry, we need to discard all the "old values"
 | |
| 		if ($asnew)
 | |
| 			foreach ($this->getAttributes(true) as $sattribute)
 | |
| 				$sattribute->setOldValue(array());
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get Attributes by LDAP type
 | |
| 	 * This function will return a list of attributes by LDAP type (MUST,MAY).
 | |
| 	 *
 | |
| 	 * @return array Array of attributes.
 | |
| 	 */
 | |
| 	function getAttrbyLdapType($type) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$result = array();
 | |
| 
 | |
| 		foreach ($this->attributes as $index => $attribute) {
 | |
| 			if ($attribute->getLDAPtype() == strtolower($type))
 | |
| 				array_push($result,$attribute->getName());
 | |
| 		}
 | |
| 
 | |
| 		return $result;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return true if this is a MUST,MAY attribute
 | |
| 	 */
 | |
| 	function isAttrType($attr,$type) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if (in_array(strtolower($attr),$this->getAttrbyLdapType($type)))
 | |
| 			return true;
 | |
| 		else
 | |
| 			return false;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return the attributes that comprise the RDN.
 | |
| 	 *
 | |
| 	 * @return array Array of RDN objects
 | |
| 	 */
 | |
| 	private function getRDNObjects() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$return = array();
 | |
| 
 | |
| 		foreach ($this->attributes as $attribute)
 | |
| 			if ($attribute->isRDN())
 | |
| 				array_push($return,$attribute);
 | |
| 
 | |
| 		masort($return,'rdn');
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get all the RDNs for this template, in RDN order.
 | |
| 	 *
 | |
| 	 * @return array RDNs in order.
 | |
| 	 */
 | |
| 	public function getRDNAttrs() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$return = array();
 | |
| 
 | |
| 		foreach ($this->getRDNObjects() as $attribute) {
 | |
| 			# We'll test if two RDN's have the same number (we cant test anywhere else)
 | |
| 			if (isset($return[$attribute->isRDN()]) && $this->getType() == 'creation')
 | |
| 				system_message(array(
 | |
| 					'title'=>('RDN attribute sequence already defined'),
 | |
| 					'body'=>sprintf('%s %s',
 | |
| 						sprintf(('There is a problem in template [%s].'),$this->getName()),
 | |
| 						sprintf(('RDN attribute sequence [%s] is already used by attribute [%s] and cant be used by attribute [%s] also.'),
 | |
| 							$attribute->isRDN(),$return[$attribute->isRDN()],$attribute->getName())),
 | |
| 					'type'=>'error'),'index.php');
 | |
| 
 | |
| 			$return[$attribute->isRDN()] = $attribute->getName();
 | |
| 		}
 | |
| 
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return the RDN for this template. If the DN is already defined, then the RDN will be calculated from it.
 | |
| 	 * If the DN is not set, then the RDN will be calcuated from the template attribute definitions
 | |
| 	 *
 | |
| 	 * @return rdn RDN for this template
 | |
| 	 */
 | |
| 	public function getRDN() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		# If the DN is set, then the RDN will be calculated from it.
 | |
| 		if ($this->dn)
 | |
| 			return get_rdn($this->dn);
 | |
| 
 | |
| 		$rdn = '';
 | |
| 
 | |
| 		foreach ($this->getRDNObjects() as $attribute) {
 | |
| 			$vals = $attribute->getValues();
 | |
| 
 | |
| 			# If an RDN attribute has no values, return with an empty string. The calling script should handle this.
 | |
| 			if (! count($vals))
 | |
| 				return '';
 | |
| 
 | |
| 			foreach ($vals as $val)
 | |
| 				$rdn .= sprintf('%s=%s+',$attribute->getName(false),$val);
 | |
| 		}
 | |
| 
 | |
| 		# Chop the last plus sign off when returning
 | |
| 		return preg_replace('/\+$/','',$rdn);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return the attribute name part of the RDN
 | |
| 	 */
 | |
| 	public function getRDNAttributeName() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$attr = array();
 | |
| 
 | |
| 		if ($this->getDN()) {
 | |
| 			$i = strpos($this->getDN(),',');
 | |
| 			if ($i !== false) {
 | |
| 				$attrs = explode('\+',substr($this->getDN(),0,$i));
 | |
| 				foreach ($attrs as $id => $attr) {
 | |
| 					list ($name,$value) = explode('=',$attr);
 | |
| 					$attrs[$id] = $name;
 | |
| 				}
 | |
| 
 | |
| 				$attr = array_unique($attrs);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $attr;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Determine the type of template this is
 | |
| 	 */
 | |
| 	public function getContext() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if ($this->getContainer() && get_request('cmd','REQUEST') == 'copy')
 | |
| 			return 'copyasnew';
 | |
| 		elseif ($this->getContainer() || get_request('create_base'))
 | |
| 			return 'create';
 | |
| 		elseif ($this->getDN())
 | |
| 			return 'edit';
 | |
| 		else
 | |
| 			return 'unknown';
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Test if the template is visible
 | |
| 	 *
 | |
| 	 * @return boolean
 | |
| 	 */
 | |
| 	public function isVisible() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,1,__FILE__,__LINE__,__METHOD__,$fargs,$this->visible);
 | |
| 
 | |
| 		return $this->visible;
 | |
| 	}
 | |
| 
 | |
| 	public function setVisible() {
 | |
| 		$this->visible = true;
 | |
| 	}
 | |
| 
 | |
| 	public function setInvisible() {
 | |
| 		$this->visible = false;
 | |
| 	}
 | |
| 
 | |
| 	public function getRegExp() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,1,__FILE__,__LINE__,__METHOD__,$fargs,$this->regexp);
 | |
| 
 | |
| 		return $this->regexp;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Test if this template has been marked as a read-only template
 | |
| 	 */
 | |
| 	public function isReadOnly() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if ((($this->getContext() == 'edit') && $this->readonly) || $this->getServer()->isReadOnly())
 | |
| 			return true;
 | |
| 		else
 | |
| 			return false;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the attribute entries
 | |
| 	 *
 | |
| 	 * @param boolean Include the optional attributes
 | |
| 	 * @return array Array of attributes
 | |
| 	 */
 | |
| 	public function getAttributes($optional=false) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if ($optional)
 | |
| 			return $this->attributes;
 | |
| 
 | |
| 		$result = array();
 | |
| 		foreach ($this->attributes as $attribute) {
 | |
| 			if (! $attribute->isRequired())
 | |
| 				continue;
 | |
| 
 | |
| 			array_push($result,$attribute);
 | |
| 		}
 | |
| 
 | |
| 		return $result;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return a list of attributes that should be shown
 | |
| 	 */
 | |
| 	public function getAttributesShown() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$result = array();
 | |
| 
 | |
| 		foreach ($this->attributes as $attribute)
 | |
| 			if ($attribute->isVisible())
 | |
| 				array_push($result,$attribute);
 | |
| 
 | |
| 		return $result;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return a list of the internal attributes
 | |
| 	 */
 | |
| 	public function getAttributesInternal() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$result = array();
 | |
| 
 | |
| 		foreach ($this->attributes as $attribute)
 | |
| 			if ($attribute->isInternal())
 | |
| 				array_push($result,$attribute);
 | |
| 
 | |
| 		return $result;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return the objectclasses defined in this template
 | |
| 	 *
 | |
| 	 * @return array Array of Objects
 | |
| 	 */
 | |
| 	public function getObjectClasses() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$attribute = $this->getAttribute('objectclass');
 | |
| 		if ($attribute)
 | |
| 			return $attribute->getValues();
 | |
| 		else
 | |
| 			return array();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get template icon
 | |
| 	 */
 | |
| 	public function getIcon() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,1,__FILE__,__LINE__,__METHOD__,$fargs,$this->icon);
 | |
| 
 | |
| 		return isset($this->icon) ? sprintf('%s/%s',IMGDIR,$this->icon) : '';
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return the template description
 | |
| 	 *
 | |
| 	 * @return string Description
 | |
| 	 */
 | |
| 	public function getDescription() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,1,__FILE__,__LINE__,__METHOD__,$fargs,$this->description);
 | |
| 
 | |
| 		return $this->description;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set a template as invalid
 | |
| 	 *
 | |
| 	 * @param string Message indicating the reason the template has been invalidated
 | |
| 	 */
 | |
| 	public function setInvalid($msg,$admin=false) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$this->invalid = true;
 | |
| 		$this->invalid_reason = $msg;
 | |
| 		$this->invalid_admin = $admin;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the template validity or the reason it is invalid
 | |
| 	 *
 | |
| 	 * @return string Invalid reason, or false if not invalid
 | |
| 	 */
 | |
| 	public function isInValid() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		if ($this->invalid)
 | |
| 			return $this->invalid_reason;
 | |
| 		else
 | |
| 			return false;
 | |
| 	}
 | |
| 
 | |
| 	public function isAdminDisabled() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,1,__FILE__,__LINE__,__METHOD__,$fargs,$this->invalid_admin);
 | |
| 
 | |
| 		return $this->invalid_admin;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the minimum number of values for an attribute
 | |
| 	 *
 | |
| 	 * @param object Attribute
 | |
| 	 * @param int
 | |
| 	 */
 | |
| 	private function setMinValueCount($attr,$value) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$attribute = $this->getAttribute($attr);
 | |
| 
 | |
| 		if (! is_null($attribute))
 | |
| 			$attribute->setMinValueCount($value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the LDAP type property for an attribute
 | |
| 	 *
 | |
| 	 * @param object Attribute
 | |
| 	 * @param string (MUST,MAY,OPTIONAL)
 | |
| 	 */
 | |
| 	private function setAttrLDAPtype($attr,$value) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$attribute = $this->getAttribute($attr);
 | |
| 
 | |
| 		if (is_null($attribute))
 | |
| 			$attribute = $this->addAttribute($attr,array('values'=>array()));
 | |
| 
 | |
| 		$attribute->setLDAPtype($value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * OnChangeAdd javascript processing
 | |
| 	 */
 | |
| 	public function OnChangeAdd($origin,$value) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$attribute = $this->getAttribute($origin);
 | |
| 
 | |
| 		if (preg_match('/^=(\w+)\((.*)\)$/',$value,$matches)) {
 | |
| 			$command = $matches[1];
 | |
| 			$arg = $matches[2];
 | |
| 		} else
 | |
| 			return;
 | |
| 
 | |
| 		switch ($command) {
 | |
| 			/*
 | |
| 			autoFill:string
 | |
| 			string is a literal string, and may contain many fields like %attr|start-end/flags%
 | |
| 				to substitute values read from other fields.
 | |
| 			|start-end is optional, but must be present if the k flag is used.
 | |
| 			/flags is optional.
 | |
| 
 | |
| 			flags may be:
 | |
| 			T:	Read display text from selection item (drop-down list), otherwise, read the value of the field
 | |
| 				For fields that aren't selection items, /T shouldn't be used, and the field value will always be read.
 | |
| 			k:	Tokenize:
 | |
| 				If the "k" flag is not given:
 | |
| 					A |start-end instruction will perform a sub-string operation upon
 | |
| 					the value of the attr, passing character positions start-end through.
 | |
| 					start can be 0 for first character, or any other integer.
 | |
| 					end can be 0 for last character, or any other integer for a specific position.
 | |
| 				If the "k" flag is given:
 | |
| 					The string read will be split into fields, using : as a delimiter
 | |
| 					"start" indicates which field number to pass through.
 | |
| 			K:	The string read will be split into fields, using ' ' as a delimiter "start" indicates which field number to pass through.
 | |
| 			l:	Make the result lower case.
 | |
| 			U:	Make the result upper case.
 | |
| 			*/
 | |
| 			case 'autoFill':
 | |
| 				if (! preg_match('/;/',$arg)) {
 | |
| 					system_message(array(
 | |
| 						'title'=>('Problem with autoFill() in template'),
 | |
| 						'body'=>sprintf('%s (<b>%s</b>)',('There is only 1 argument, when there should be two'),$attribute->getName(false)),
 | |
| 						'type'=>'warn'));
 | |
| 
 | |
| 					return;
 | |
| 				}
 | |
| 
 | |
| 				list($attr,$string) = preg_split('(([^,]+);(.*))',$arg,-1,PREG_SPLIT_DELIM_CAPTURE | PREG_SPLIT_NO_EMPTY);
 | |
| 				preg_match_all('/%(\w+)(\|[0-9]*-[0-9]*)?(\/[KklTUA]+)?%/U',$string,$matchall);
 | |
| 				//print"<PRE>";print_r($matchall); //0 = highlevel match, 1 = attr, 2 = subst, 3 = mod
 | |
| 
 | |
| 				if (! isset($attribute->js['autoFill']))
 | |
| 					$attribute->js['autoFill'] = '';
 | |
| 
 | |
| 				$formula = $string;
 | |
| 				$formula = preg_replace('/^([^%])/','\'$1',$formula);
 | |
| 				$formula = preg_replace('/([^%])$/','$1\'',$formula);
 | |
| 
 | |
| 				# Check that our attributes match our schema attributes.
 | |
| 				foreach ($matchall[1] as $index => $checkattr) {
 | |
| 					$sattr = $this->getServer()->getSchemaAttribute($checkattr);
 | |
| 
 | |
| 					# If the attribute is the same as in the XML file, then dont need to do anything.
 | |
| 					if (! $sattr || ! strcasecmp($sattr->getName(),$checkattr))
 | |
| 						continue;
 | |
| 
 | |
| 					$formula = preg_replace("/$checkattr/",$sattr->getName(),$formula);
 | |
| 					$matchall[1][$index] = $sattr->getName();
 | |
| 				}
 | |
| 
 | |
| 				$elem_id = 0;
 | |
| 
 | |
| 				foreach ($matchall[0] as $index => $null) {
 | |
| 					$match_attr = strtolower($matchall[1][$index]);
 | |
| 					$match_subst = $matchall[2][$index];
 | |
| 					$match_mod = $matchall[3][$index];
 | |
| 
 | |
| 					$substrarray = array();
 | |
| 
 | |
| 					if (! isset($varcount[$match_attr]))
 | |
| 						$varcount[$match_attr] = 0;
 | |
| 					else
 | |
| 						$varcount[$match_attr]++;
 | |
| 
 | |
| 					$js_match_attr = $match_attr;
 | |
| 					$match_attr = $js_match_attr.'xx'.$varcount[$match_attr];
 | |
| 
 | |
| 					$formula = preg_replace('/%'.$js_match_attr.'([|\/%])/i','%'.$match_attr.'$1',$formula,1);
 | |
| 
 | |
| 					$attribute->js['autoFill'] .= sprintf("  var %s;\n",$match_attr);
 | |
| 					$attribute->js['autoFill'] .= sprintf(
 | |
| 							"  var elem$elem_id = document.getElementById(pre+'%s'+suf);\n".
 | |
| 							"  if (!elem$elem_id) return;\n", $js_match_attr);
 | |
| 
 | |
| 					if (strstr($match_mod,'T')) {
 | |
| 						$attribute->js['autoFill'] .= sprintf("  %s = elem$elem_id.options[elem$elem_id.selectedIndex].text;\n",
 | |
| 							$match_attr);
 | |
| 					} else {
 | |
| 						$attribute->js['autoFill'] .= sprintf("  %s = elem$elem_id.value;\n",$match_attr);
 | |
| 					}
 | |
| 
 | |
| 					$elem_id++;
 | |
| 
 | |
| 					if (strstr($match_mod,'k')) {
 | |
| 						preg_match_all('/([0-9]+)/',trim($match_subst),$substrarray);
 | |
| 						if (isset($substrarray[1][0])) {
 | |
| 							$tok_idx = $substrarray[1][0];
 | |
| 						} else {
 | |
| 							$tok_idx = '0';
 | |
| 						}
 | |
| 						$attribute->js['autoFill'] .= sprintf("   %s = %s.split(':')[%s];\n",$match_attr,$match_attr,$tok_idx);
 | |
| 
 | |
| 					} elseif (strstr($match_mod,'K')) {
 | |
| 						preg_match_all('/([0-9]+)/',trim($match_subst),$substrarray); 
 | |
| 						if (isset($substrarray[1][0])) { 
 | |
| 							$tok_idx = $substrarray[1][0]; 
 | |
| 						} else { 
 | |
| 							$tok_idx = '0'; 
 | |
| 						} 
 | |
| 						$attribute->js['autoFill'] .= sprintf("   %s = %s.split(' ')[%s];\n",$match_attr,$match_attr,$tok_idx); 
 | |
| 
 | |
| 					} else {
 | |
| 						preg_match_all('/([0-9]*)-([0-9]*)/',trim($match_subst),$substrarray);
 | |
| 						if ((isset($substrarray[1][0]) && $substrarray[1][0]) || (isset($substrarray[2][0]) && $substrarray[2][0])) {
 | |
| 							$attribute->js['autoFill'] .= sprintf("   %s = %s.substr(%s,%s);\n",
 | |
| 								$match_attr,$match_attr,
 | |
| 								$substrarray[1][0] ? $substrarray[1][0] : '0',
 | |
| 								$substrarray[2][0] ? $substrarray[2][0] : sprintf('%s.length',$match_attr));
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					if (strstr($match_mod,'l')) {
 | |
| 						$attribute->js['autoFill'] .= sprintf("   %s = %s.toLowerCase();\n",$match_attr,$match_attr);
 | |
| 					}
 | |
| 					if (strstr($match_mod,'U')) {
 | |
| 						$attribute->js['autoFill'] .= sprintf("   %s = %s.toUpperCase();\n",$match_attr,$match_attr);
 | |
| 					}
 | |
| 					if (strstr($match_mod,'A')) {
 | |
| 						$attribute->js['autoFill'] .= sprintf("   %s = toAscii(%s);\n",$match_attr,$match_attr);
 | |
| 					}
 | |
| 
 | |
| 					# Matchfor only entry without modifiers.
 | |
| 					$formula = preg_replace('/^%('.$match_attr.')%$/U','$1 + \'\'',$formula);
 | |
| 					# Matchfor only entry with modifiers.
 | |
| 					$formula = preg_replace('/^%('.$match_attr.')(\|[0-9]*-[0-9]*)?(\/[KklTUA]+)?%$/U','$1 + \'\'',$formula);
 | |
| 					# Matchfor begining entry.
 | |
| 					$formula = preg_replace('/^%('.$match_attr.')(\|[0-9]*-[0-9]*)?(\/[KklTUA]+)?%/U','$1 + \'',$formula);
 | |
| 					# Matchfor ending entry.
 | |
| 					$formula = preg_replace('/%('.$match_attr.')(\|[0-9]*-[0-9]*)?(\/[KklTUA]+)?%$/U','\' + $1 ',$formula);
 | |
| 					# Match for entries not at begin/end.
 | |
| 					$formula = preg_replace('/%('.$match_attr.')(\|[0-9]*-[0-9]*)?(\/[:lTUA]+)?%/U','\' + $1 + \'',$formula);
 | |
| 					$attribute->js['autoFill'] .= "\n";
 | |
| 				}
 | |
| 
 | |
| 				$attribute->js['autoFill'] .= sprintf(" fillRec(pre+'%s'+suf, %s); // %s\n",strtolower($attr),$formula,$string);
 | |
| 				$attribute->js['autoFill'] .= "\n";
 | |
| 				break;
 | |
| 
 | |
| 			default: $return = '';
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * This functions main purpose is to discover our MUST attributes based on objectclass
 | |
| 	 * definitions in the template file and to discover which of the objectclasses are
 | |
| 	 * STRUCTURAL - without one, creating an entry will just product an LDAP error.
 | |
| 	 */
 | |
| 	private function rebuildTemplateAttrs() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$server = $this->getServer();
 | |
| 
 | |
| 		# Collect our structural, MUST & MAY attributes.
 | |
| 		$oclass_processed = array();
 | |
| 		$superclasslist = array();
 | |
| 		$allattrs = array('objectclass');
 | |
| 
 | |
| 		foreach ($this->getObjectClasses() as $oclass) {
 | |
| 			# If we get some superclasses - then we'll need to go through them too.
 | |
| 			$supclass = true;
 | |
| 			$inherited = false;
 | |
| 
 | |
| 			while ($supclass) {
 | |
| 				$soc = $server->getSchemaObjectClass($oclass);
 | |
| 
 | |
| 				if ($soc->getType() == 'structural' && (! $inherited))
 | |
| 					array_push($this->structural_oclass,$oclass);
 | |
| 
 | |
| 				# Make sure our MUST attributes are marked as such for this template.
 | |
| 				if ($soc->getMustAttrs())
 | |
| 					foreach ($soc->getMustAttrs() as $index => $details) {
 | |
| 						$objectclassattr = $details->getName();
 | |
| 
 | |
| 						# We add the 'objectClass' attribute, only if it's explicitly in the template attribute list
 | |
| 						if ((strcasecmp('objectClass',$objectclassattr) != 0) ||
 | |
| 								((strcasecmp('objectClass',$objectclassattr) == 0) && (! is_null($this->getAttribute($objectclassattr))))) {
 | |
| 
 | |
| 							# Go through the aliases, and ignore any that are already defined.
 | |
| 							$ignore = false;
 | |
| 							$sattr = $server->getSchemaAttribute($objectclassattr);
 | |
| 							foreach ($sattr->getAliases() as $alias) {
 | |
| 								if ($this->isAttrType($alias,'must')) {
 | |
| 									$ignore = true;
 | |
| 									break;
 | |
| 								}
 | |
| 							}
 | |
| 
 | |
| 							if ($ignore)
 | |
| 								continue;
 | |
| 
 | |
| 							$this->setAttrLDAPtype($sattr->getName(),'must');
 | |
| 							$this->setMinValueCount($sattr->getName(),1);
 | |
| 
 | |
| 							# We need to mark the attributes as show, except for the objectclass attribute.
 | |
| 							if (strcasecmp('objectClass',$objectclassattr) != 0) {
 | |
| 								$attribute = $this->getAttribute($sattr->getName());
 | |
| 								$attribute->show();
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 						if (! in_array($objectclassattr,$allattrs))
 | |
| 							array_push($allattrs,$objectclassattr);
 | |
| 					}
 | |
| 
 | |
| 				if ($soc->getMayAttrs())
 | |
| 					foreach ($soc->getMayAttrs() as $index => $details) {
 | |
| 						$objectclassattr = $details->getName();
 | |
| 						$sattr = $server->getSchemaAttribute($objectclassattr);
 | |
| 
 | |
| 						# If it is a MUST attribute, skip to the next one.
 | |
| 						if ($this->isAttrType($objectclassattr,'must'))
 | |
| 							continue;
 | |
| 
 | |
| 						if (! $this->isAttrType($objectclassattr,'may'))
 | |
| 							$this->setAttrLDAPtype($sattr->getName(false),'optional');
 | |
| 
 | |
| 						if (! in_array($objectclassattr,$allattrs))
 | |
| 							array_push($allattrs,$objectclassattr);
 | |
| 					}
 | |
| 
 | |
| 				# Keep a list to objectclasses we have processed, so we dont get into a loop.
 | |
| 				array_push($oclass_processed,$oclass);
 | |
| 				$supoclasses = $soc->getSupClasses();
 | |
| 
 | |
| 				if (count($supoclasses) || count($superclasslist)) {
 | |
| 					foreach ($supoclasses as $supoclass) {
 | |
| 						if (! in_array($supoclass,$oclass_processed))
 | |
| 							$superclasslist[] = $supoclass;
 | |
| 					}
 | |
| 
 | |
| 					$oclass = array_shift($superclasslist);
 | |
| 					if ($oclass)
 | |
| 						$inherited = true;
 | |
| 					else
 | |
| 						$supclass = false;
 | |
| 
 | |
| 				} else {
 | |
| 					$supclass = false;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		# Check that attributes are defined by an ObjectClass
 | |
| 		foreach ($this->getAttributes(true) as $index => $attribute)
 | |
| 			if (! in_array($attribute->getName(),$allattrs) && (! array_intersect($attribute->getAliases(),$allattrs))
 | |
| 				&& (! in_array_ignore_case('extensibleobject',$this->getObjectClasses()))
 | |
| 				&& (! in_array_ignore_case($attribute->getName(),$server->getValue('server','custom_attrs')))) {
 | |
| 				unset($this->attributes[$index]);
 | |
| 
 | |
| 				if (! $_SESSION[APPCONFIG]->getValue('appearance','hide_template_warning'))
 | |
| 					system_message(array(
 | |
| 						'title'=>('Automatically removed attribute from template'),
 | |
| 						'body'=>sprintf('%s: <b>%s</b> %s',$this->getTitle(),$attribute->getName(false),('removed from template as it is not defined by an ObjectClass')),
 | |
| 						'type'=>'warn'));
 | |
| 			}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return an array, that can be passed to ldap_add().
 | |
| 	 * Attributes with empty values will be excluded.
 | |
| 	 */
 | |
| 	public function getLDAPadd($attrsOnly=false) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$return = array();
 | |
| 		$returnattrs = array();
 | |
| 
 | |
| 		if ($attrsOnly && count($returnattrs))
 | |
| 			return $returnattrs;
 | |
| 
 | |
| 		foreach ($this->getAttributes(true) as $attribute)
 | |
| 			if (! $attribute->isInternal() && count($attribute->getValues())) {
 | |
| 				$return[$attribute->getName()] = $attribute->getValues();
 | |
| 				$returnattrs[$attribute->getName()] = $attribute;
 | |
| 			}
 | |
| 
 | |
| 		# Ensure that our objectclasses has "top".
 | |
| 		if (isset($return['objectclass']) && ! in_array('top',$return['objectclass']))
 | |
| 			array_push($return['objectclass'],'top');
 | |
| 
 | |
| 		if ($attrsOnly)
 | |
| 			return $returnattrs;
 | |
| 
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return an array, that can be passed to ldap_mod_replace().
 | |
| 	 * Only attributes that have changed their value will be returned.
 | |
| 	 *
 | |
| 	 * This function will cache its results, so that it can be called with count() to see
 | |
| 	 * if there are changes, and if they are, the 2nd call will just return the results
 | |
| 	 *
 | |
| 	 * @param boolean Return the attribute objects (useful for a confirmation process), or the modification array for ldap_modify()
 | |
| 	 */
 | |
| 	public function getLDAPmodify($attrsOnly=false,$index=0) {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		static $return = array();
 | |
| 		static $returnattrs = array();
 | |
| 
 | |
| 		if ($attrsOnly && isset($returnattrs[$index]) && count($returnattrs[$index]))
 | |
| 			return $returnattrs[$index];
 | |
| 
 | |
| 		$returnattrs[$index] = array();
 | |
| 		$return[$index] = array();
 | |
| 
 | |
| 		# If an objectclass is being modified, we need to remove all the orphan attributes that would result.
 | |
| 		if ($this->getAttribute('objectclass')->hasBeenModified()) {
 | |
| 			$attr_to_keep = array();
 | |
| 			$server = $this->getServer();
 | |
| 
 | |
| 			# Make sure that there will be a structural object class remaining.
 | |
| 			$haveStructural = false;
 | |
| 			foreach ($this->getAttribute('objectclass')->getValues() as $value) {
 | |
| 				$soc = $server->getSchemaObjectClass($value);
 | |
| 
 | |
| 				if ($soc) {
 | |
| 					if ($soc->isStructural())
 | |
| 						$haveStructural = true;
 | |
| 
 | |
| 					# While we are looping, workout which attributes these objectclasses define.
 | |
| 					foreach ($soc->getMustAttrs(true) as $value)
 | |
| 						if (! in_array($value->getName(),$attr_to_keep))
 | |
| 							array_push($attr_to_keep,$value->getName());
 | |
| 
 | |
| 					foreach ($soc->getMayAttrs(true) as $value)
 | |
| 						if (! in_array($value->getName(),$attr_to_keep))
 | |
| 							array_push($attr_to_keep,$value->getName());
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (! $haveStructural)
 | |
| 				error(('An entry should have one structural objectClass.'),'error','index.php');
 | |
| 
 | |
| 			# Work out the attributes to delete.
 | |
| 			foreach ($this->getAttribute('objectclass')->getRemovedValues() as $value) {
 | |
| 				$soc = $server->getSchemaObjectClass($value);
 | |
| 
 | |
| 				foreach ($soc->getMustAttrs() as $value) {
 | |
| 					$attribute = $this->getAttribute($value->getName());
 | |
| 
 | |
| 					if ($attribute && (! in_array($value->getName(),$attr_to_keep)) && ($value->getName() != 'objectclass'))
 | |
| 						#array_push($attr_to_delete,$value->getName(false));
 | |
| 						$attribute->setForceDelete();
 | |
| 				}
 | |
| 
 | |
| 				foreach ($soc->getMayAttrs() as $value) {
 | |
| 					$attribute = $this->getAttribute($value->getName());
 | |
| 
 | |
| 					if ($attribute && (! in_array($value->getName(),$attr_to_keep)) && ($value->getName() != 'objectclass'))
 | |
| 						$attribute->setForceDelete();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		foreach ($this->getAttributes(true) as $attribute)
 | |
| 			if ($attribute->hasBeenModified()
 | |
| 				&& (count(array_diff($attribute->getValues(),$attribute->getOldValues())) || ! count($attribute->getValues())
 | |
| 					|| $attribute->isForceDelete() || (count($attribute->getValues()) != count($attribute->getOldValues()))))
 | |
| 				$returnattrs[$index][$attribute->getName()] = $attribute;
 | |
| 
 | |
| 		if ($attrsOnly)
 | |
| 			return $returnattrs[$index];
 | |
| 
 | |
| 		foreach ($returnattrs[$index] as $attribute)
 | |
| 			$return[$index][$attribute->getName()] = $attribute->getValues();
 | |
| 
 | |
| 		return $return[$index];
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the attributes that are marked as force delete
 | |
| 	 * We'll cache this result in the event of multiple calls.
 | |
| 	 */
 | |
| 	public function getForceDeleteAttrs() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		static $result = array();
 | |
| 
 | |
| 		if (count($result))
 | |
| 			return $result;
 | |
| 
 | |
| 		foreach ($this->attributes as $attribute)
 | |
| 			if ($attribute->isForceDelete())
 | |
| 				array_push($result,$attribute);
 | |
| 
 | |
| 		return $result;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get available attributes
 | |
| 	 */
 | |
| 	public function getAvailAttrs() {
 | |
| 		if (DEBUG_ENABLED && (($fargs=func_get_args())||$fargs='NOARGS'))
 | |
| 			debug_log('Entered (%%)',5,0,__FILE__,__LINE__,__METHOD__,$fargs);
 | |
| 
 | |
| 		$attributes = array();
 | |
| 		$server = $this->getServer();
 | |
| 
 | |
| 		# Initialise the Attribute Factory.
 | |
| 		$attribute_factory = new AttributeFactory();
 | |
| 
 | |
| 		if (in_array_ignore_case('extensibleobject',$this->getObjectClasses())) {
 | |
| 			foreach ($server->SchemaAttributes() as $sattr) {
 | |
| 				$attribute = $attribute_factory->newAttribute($sattr->getName(),array('values'=>array()),$server->getIndex(),null);
 | |
| 				array_push($attributes,$attribute);
 | |
| 			}
 | |
| 
 | |
| 		} else {
 | |
| 			$attrs = array();
 | |
| 
 | |
| 			foreach ($this->getObjectClasses() as $oc) {
 | |
| 				$soc = $server->getSchemaObjectClass($oc);
 | |
| 				$attrs = array_merge($attrs,$soc->getMustAttrNames(true),$soc->getMayAttrNames(true));
 | |
| 				$attrs = array_unique($attrs);
 | |
| 			}
 | |
| 
 | |
| 			foreach ($attrs as $attr)
 | |
| 				if (is_null($this->getAttribute($attr))) {
 | |
| 					$attribute = $attribute_factory->newAttribute($attr,array('values'=>array()),$server->getIndex(),null);
 | |
| 					array_push($attributes,$attribute);
 | |
| 				}
 | |
| 		}
 | |
| 
 | |
| 		masort($attributes,'name');
 | |
| 		return $attributes;
 | |
| 	}
 | |
| 
 | |
| 	public function isNoLeaf() {
 | |
| 		return $this->noleaf;
 | |
| 	}
 | |
| }
 | |
| ?>
 |