\version 2.00 \date 24.07.2003 This is the base class for all plugins. It can be used standalone or can be included by the tabs class. All management should be done within this class. Extend your plugins from this class. */ class plugin { /*! \brief Reference to parent object This variable is used when the plugin is included in tabs and keeps reference to the tab class. Communication to other tabs is possible by 'name'. So the 'fax' plugin can ask the 'userinfo' plugin for the fax number. \sa tab */ var $parent= NULL; /*! \brief Configuration container Access to global configuration */ var $config= NULL; /*! \brief Mark plugin as account Defines whether this plugin is defined as an account or not. This has consequences for the plugin to be saved from tab mode. If it is set to 'FALSE' the tab will call the delete function, else the save function. Should be set to 'TRUE' if the construtor detects a valid LDAP object. \sa plugin::plugin() */ var $is_account= FALSE; var $initially_was_account= FALSE; /*! \brief Mark plugin as template Defines whether we are creating a template or a normal object. Has conseqences on the way execute() shows the formular and how save() puts the data to LDAP. \sa plugin::save() plugin::execute() */ var $is_template= FALSE; var $ignore_account= FALSE; var $is_modified= FALSE; /*! \brief Represent temporary LDAP data This is only used internally. */ var $attrs= array(); /* Keep set of conflicting plugins */ var $conflicts= array(); /* Save unit tags */ var $gosaUnitTag= ""; var $skipTagging= FALSE; /*! \brief Used standard values dn */ var $dn= ""; var $uid= ""; var $sn= ""; var $givenName= ""; var $acl= "*none*"; var $dialog= FALSE; var $snapDialog = NULL; /* attribute list for save action */ var $attributes= array(); var $objectclasses= array(); var $is_new= TRUE; var $saved_attributes= array(); var $acl_base= ""; var $acl_category= ""; /* This can be set to render the tabulators in another stylesheet */ var $pl_notify= FALSE; /* Object entry CSN */ var $entryCSN = ""; var $CSN_check_active = FALSE; /* This variable indicates that this class can handle multiple dns at once. */ var $multiple_support = FALSE; var $multi_attrs = array(); var $multi_attrs_all = array(); /* This aviable indicates, that we are currently in multiple edit handle */ var $multiple_support_active = FALSE; var $selected_edit_values = array(); var $multi_boxes = array(); /*! \brief plugin constructor If 'dn' is set, the node loads the given 'dn' from LDAP \param dn Distinguished name to initialize plugin from \sa plugin() */ function plugin (&$config, $dn= NULL, $parent= NULL) { /* Configuration is fine, allways */ $this->config= &$config; $this->dn= $dn; /* Handle new accounts, don't read information from LDAP */ if ($dn == "new"){ return; } /* Save current dn as acl_base */ $this->acl_base= $dn; /* Get LDAP descriptor */ if ($dn !== NULL){ /* Load data to 'attrs' and save 'dn' */ if ($parent !== NULL){ $this->attrs= $parent->attrs; } else { $ldap= $this->config->get_ldap_link(); $ldap->cat ($dn); $this->attrs= $ldap->fetch(); } /* Copy needed attributes */ foreach ($this->attributes as $val){ $found= array_key_ics($val, $this->attrs); if ($found != ""){ $this->$val= $this->attrs["$found"][0]; } } /* gosaUnitTag loading... */ if (isset($this->attrs['gosaUnitTag'][0])){ $this->gosaUnitTag= $this->attrs['gosaUnitTag'][0]; } /* Set the template flag according to the existence of objectClass gosaUserTemplate */ if (isset($this->attrs['objectClass'])){ if (in_array_ics ("gosaUserTemplate", $this->attrs['objectClass'])){ $this->is_template= TRUE; @DEBUG (DEBUG_TRACE, __LINE__, __FUNCTION__, __FILE__, "found", "Template check"); } } /* Is Account? */ $found= TRUE; foreach ($this->objectclasses as $obj){ if (preg_match('/top/i', $obj)){ continue; } if (!isset($this->attrs['objectClass']) || !in_array_ics ($obj, $this->attrs['objectClass'])){ $found= FALSE; break; } } if ($found){ $this->is_account= TRUE; @DEBUG (DEBUG_TRACE, __LINE__, __FUNCTION__, __FILE__, "found", "Object check"); } /* Prepare saved attributes */ $this->saved_attributes= $this->attrs; foreach ($this->saved_attributes as $index => $value){ if (preg_match('/^[0-9]+$/', $index)){ unset($this->saved_attributes[$index]); continue; } if (!in_array($index, $this->attributes) && $index != "objectClass"){ unset($this->saved_attributes[$index]); continue; } if (isset($this->saved_attributes[$index][0])){ if(!isset($this->saved_attributes[$index]["count"])){ $this->saved_attributes[$index]["count"] = count($this->saved_attributes[$index]); } if($this->saved_attributes[$index]["count"] == 1){ $tmp= $this->saved_attributes[$index][0]; unset($this->saved_attributes[$index]); $this->saved_attributes[$index]= $tmp; continue; } } unset($this->saved_attributes["$index"]["count"]); } if(isset($this->attrs['gosaUnitTag'])){ $this->saved_attributes['gosaUnitTag'] = $this->attrs['gosaUnitTag'][0]; } } /* Save initial account state */ $this->initially_was_account= $this->is_account; } /*! \brief execute plugin Generates the html output for this node */ function execute() { /* This one is empty currently. Fabian - please fill in the docu code */ session::set('current_class_for_help',get_class($this)); /* Reset Lock message POST/GET check array, to prevent perg_match errors*/ session::set('LOCK_VARS_TO_USE',array()); session::set('LOCK_VARS_USED',array()); } /*! \brief execute plugin Removes object from parent */ function remove_from_parent() { /* include global link_info */ $ldap= $this->config->get_ldap_link(); /* Get current objectClasses in order to add the required ones */ $ldap->cat($this->dn); $tmp= $ldap->fetch (); $oc= array(); if (isset($tmp['objectClass'])){ $oc= $tmp['objectClass']; unset($oc['count']); } /* Remove objectClasses from entry */ $ldap->cd($this->dn); $this->attrs= array(); $this->attrs['objectClass']= array_remove_entries_ics($this->objectclasses,$oc); /* Unset attributes from entry */ foreach ($this->attributes as $val){ $this->attrs["$val"]= array(); } /* Unset account info */ $this->is_account= FALSE; /* Do not write in plugin base class, this must be done by children, since there are normally additional attribs, lists, etc. */ /* $ldap->modify($this->attrs); */ } /*! \brief Save HTML posted data to object */ function save_object() { /* Update entry CSN if it is empty. */ if(empty($this->entryCSN) && $this->CSN_check_active){ $this->entryCSN = getEntryCSN($this->dn); } /* Save values to object */ foreach ($this->attributes as $val){ if ($this->acl_is_writeable($val) && isset ($_POST["$val"])){ /* Check for modifications */ if (get_magic_quotes_gpc()) { $data= stripcslashes($_POST["$val"]); } else { $data= $this->$val = $_POST["$val"]; } if ($this->$val != $data){ $this->is_modified= TRUE; } /* Okay, how can I explain this fix ... * In firefox, disabled option fields aren't selectable ... but in IE you can select these fileds. * So IE posts these 'unselectable' option, with value = chr(194) * chr(194) seems to be the   in between the ...option> $val= $data; } } } /* Save data to LDAP, depending on is_account we save or delete */ function save() { /* include global link_info */ $ldap= $this->config->get_ldap_link(); /* Save all plugins */ $this->entryCSN = ""; /* Start with empty array */ $this->attrs= array(); /* Get current objectClasses in order to add the required ones */ $ldap->cat($this->dn); $tmp= $ldap->fetch (); $oc= array(); if (isset($tmp['objectClass'])){ $oc= $tmp["objectClass"]; $this->is_new= FALSE; unset($oc['count']); } else { $this->is_new= TRUE; } /* Load (minimum) attributes, add missing ones */ $this->attrs['objectClass']= gosa_array_merge($oc,$this->objectclasses); /* Copy standard attributes */ foreach ($this->attributes as $val){ if ($this->$val != ""){ $this->attrs["$val"]= $this->$val; } elseif (!$this->is_new) { $this->attrs["$val"]= array(); } } /* Handle tagging */ $this->tag_attrs($this->attrs); } function cleanup() { foreach ($this->attrs as $index => $value){ /* Convert arrays with one element to non arrays, if the saved attributes are no array, too */ if (is_array($this->attrs[$index]) && count ($this->attrs[$index]) == 1 && isset($this->saved_attributes[$index]) && !is_array($this->saved_attributes[$index])){ $tmp= $this->attrs[$index][0]; $this->attrs[$index]= $tmp; } /* Remove emtpy arrays if they do not differ */ if (is_array($this->attrs[$index]) && count($this->attrs[$index]) == 0 && !isset($this->saved_attributes[$index])){ unset ($this->attrs[$index]); continue; } /* Remove single attributes that do not differ */ if (!is_array($this->attrs[$index]) && isset($this->saved_attributes[$index]) && !is_array($this->saved_attributes[$index]) && $this->attrs[$index] == $this->saved_attributes[$index]){ unset ($this->attrs[$index]); continue; } /* Remove arrays that do not differ */ if (is_array($this->attrs[$index]) && isset($this->saved_attributes[$index]) && is_array($this->saved_attributes[$index])){ if (!array_differs($this->attrs[$index],$this->saved_attributes[$index])){ unset ($this->attrs[$index]); continue; } } } /* Update saved attributes and ensure that next cleanups will be successful too */ foreach($this->attrs as $name => $value){ $this->saved_attributes[$name] = $value; } } /* Check formular input */ function check() { $message= array(); /* Skip if we've no config object */ if (!isset($this->config) || !is_object($this->config)){ return $message; } /* Find hooks entries for this class */ $command= $this->config->search(get_class($this), "CHECK", array('menu', 'tabs')); if ($command != ""){ if (!check_command($command)){ $message[]= msgPool::cmdnotfound("CHECK", get_class($this)); } else { /* Generate "ldif" for check hook */ $ldif= "dn: $this->dn\n"; /* ... objectClasses */ foreach ($this->objectclasses as $oc){ $ldif.= "objectClass: $oc\n"; } /* ... attributes */ foreach ($this->attributes as $attr){ if ($this->$attr == ""){ continue; } if (is_array($this->$attr)){ foreach ($this->$attr as $val){ $ldif.= "$attr: $val\n"; } } else { $ldif.= "$attr: ".$this->$attr."\n"; } } /* Append empty line */ $ldif.= "\n"; /* Feed "ldif" into hook and retrieve result*/ $descriptorspec = array( 0 => array("pipe", "r"), 1 => array("pipe", "w"), 2 => array("pipe", "w")); $fh= proc_open($command, $descriptorspec, $pipes); if (is_resource($fh)) { fwrite ($pipes[0], $ldif); fclose($pipes[0]); $result= stream_get_contents($pipes[1]); if ($result != ""){ $message[]= $result; } fclose($pipes[1]); fclose($pipes[2]); proc_close($fh); } } } /* Check entryCSN */ if($this->CSN_check_active){ $current_csn = getEntryCSN($this->dn); if($current_csn != $this->entryCSN && !empty($this->entryCSN) && !empty($current_csn)){ $this->entryCSN = $current_csn; $message[] = _("The object has changed since opened in GOsa. All changes that may be done by others get lost if you save this entry!"); } } return ($message); } /* Adapt from template, using 'dn' */ function adapt_from_template($dn, $skip= array()) { /* Include global link_info */ $ldap= $this->config->get_ldap_link(); /* Load requested 'dn' to 'attrs' */ $ldap->cat ($dn); $this->attrs= $ldap->fetch(); /* Walk through attributes */ foreach ($this->attributes as $val){ /* Skip the ones in skip list */ if (in_array($val, $skip)){ continue; } if (isset($this->attrs["$val"][0])){ /* If attribute is set, replace dynamic parts: %sn, %givenName and %uid. Fill these in our local variables. */ $value= $this->attrs["$val"][0]; foreach (array("sn", "givenName", "uid") as $repl){ if (preg_match("/%$repl/i", $value)){ $value= preg_replace ("/%$repl/i", $this->parent->$repl, $value); } } $this->$val= $value; } } /* Is Account? */ $found= TRUE; foreach ($this->objectclasses as $obj){ if (preg_match('/top/i', $obj)){ continue; } if (!in_array_ics ($obj, $this->attrs['objectClass'])){ $found= FALSE; break; } } if ($found){ $this->is_account= TRUE; } } /* Indicate whether a password change is needed or not */ function password_change_needed() { return FALSE; } /* Show header message for tab dialogs */ function show_enable_header($button_text, $text, $disabled= FALSE) { if (($disabled == TRUE) || (!$this->acl_is_createable())){ $state= "disabled"; } else { $state= ""; } $display= "\n

$text

\n"; $display.= "

 

"; return($display); } /* Show header message for tab dialogs */ function show_disable_header($button_text, $text, $disabled= FALSE) { if (($disabled == TRUE) || !$this->acl_is_removeable()){ $state= "disabled"; } else { $state= ""; } $display= "\n

$text

\n"; $display.= "

 

"; return($display); } /* Show header message for tab dialogs */ function show_header($button_text, $text, $disabled= FALSE) { echo "FIXME: show_header should be replaced by show_disable_header and show_enable_header
"; if ($disabled == TRUE){ $state= "disabled"; } else { $state= ""; } $display= "\n

$text

\n"; $display.= "acl_is_createable()?'':'disabled')." ".$state. ">

 

"; return($display); } function postcreate($add_attrs= array()) { /* Find postcreate entries for this class */ $command= $this->config->search(get_class($this), "POSTCREATE",array('menu', 'tabs')); if ($command != ""){ /* Walk through attribute list */ foreach ($this->attributes as $attr){ if (!is_array($this->$attr)){ $add_attrs[$attr] = $this->$attr; } } $add_attrs['dn']=$this->dn; $tmp = array(); foreach($add_attrs as $name => $value){ $tmp[$name] = strlen($name); } arsort($tmp); /* Additional attributes */ foreach ($tmp as $name => $len){ $value = $add_attrs[$name]; $command= preg_replace("/%$name/", "$value", $command); } if (check_command($command)){ @DEBUG (DEBUG_SHELL, __LINE__, __FUNCTION__, __FILE__, $command, "Execute"); exec($command,$arr); foreach($arr as $str){ @DEBUG (DEBUG_SHELL, __LINE__, __FUNCTION__, __FILE__, $command, "Result: ".$str); } } else { $message= msgPool::cmdnotfound("POSTCREATE", get_class($this)); msg_dialog::display(_("Error"), $message, ERROR_DIALOG); } } } function postmodify($add_attrs= array()) { /* Find postcreate entries for this class */ $command= $this->config->search(get_class($this), "POSTMODIFY",array('menu','tabs')); if ($command != ""){ /* Walk through attribute list */ foreach ($this->attributes as $attr){ if (!is_array($this->$attr)){ $add_attrs[$attr] = $this->$attr; } } $add_attrs['dn']=$this->dn; $tmp = array(); foreach($add_attrs as $name => $value){ $tmp[$name] = strlen($name); } arsort($tmp); /* Additional attributes */ foreach ($tmp as $name => $len){ $value = $add_attrs[$name]; $command= preg_replace("/%$name/", "$value", $command); } if (check_command($command)){ @DEBUG (DEBUG_SHELL, __LINE__, __FUNCTION__, __FILE__,$command, "Execute"); exec($command,$arr); foreach($arr as $str){ @DEBUG (DEBUG_SHELL, __LINE__, __FUNCTION__, __FILE__, $command, "Result: ".$str); } } else { $message= msgPool::cmdnotfound("POSTMODIFY", get_class($this)); msg_dialog::display(_("Error"), $message, ERROR_DIALOG); } } } function postremove($add_attrs= array()) { /* Find postremove entries for this class */ $command= $this->config->search(get_class($this), "POSTREMOVE",array('menu','tabs')); if ($command != ""){ /* Walk through attribute list */ foreach ($this->attributes as $attr){ if (!is_array($this->$attr)){ $add_attrs[$attr] = $this->$attr; } } $add_attrs['dn']=$this->dn; $tmp = array(); foreach($add_attrs as $name => $value){ $tmp[$name] = strlen($name); } arsort($tmp); /* Additional attributes */ foreach ($tmp as $name => $len){ $value = $add_attrs[$name]; $command= preg_replace("/%$name/", "$value", $command); } if (check_command($command)){ @DEBUG (DEBUG_SHELL, __LINE__, __FUNCTION__, __FILE__, $command, "Execute"); exec($command,$arr); foreach($arr as $str){ @DEBUG (DEBUG_SHELL, __LINE__, __FUNCTION__, __FILE__, $command, "Result: ".$str); } } else { $message= msgPool::cmdnotfound("POSTREMOVE", get_class($this)); msg_dialog::display(_("Error"), $message, ERROR_DIALOG); } } } /* Create unique DN */ function create_unique_dn($attribute, $base) { $ldap= $this->config->get_ldap_link(); $base= preg_replace("/^,*/", "", $base); /* Try to use plain entry first */ $dn= "$attribute=".$this->$attribute.",$base"; $ldap->cat ($dn, array('dn')); if (!$ldap->fetch()){ return ($dn); } /* Look for additional attributes */ foreach ($this->attributes as $attr){ if ($attr == $attribute || $this->$attr == ""){ continue; } $dn= "$attribute=".$this->$attribute."+$attr=".$this->$attr.",$base"; $ldap->cat ($dn, array('dn')); if (!$ldap->fetch()){ return ($dn); } } /* None found */ return ("none"); } function rebind($ldap, $referral) { $credentials= LDAP::get_credentials($referral, $this->config->current['REFERRAL']); if (ldap_bind($ldap, $credentials['ADMIN'], $this->config->get_credentials($credentials['PASSWORD']))) { $this->error = "Success"; $this->hascon=true; $this->reconnect= true; return (0); } else { $this->error = "Could not bind to " . $credentials['ADMIN']; return NULL; } } /* Recursively copy ldap object */ function _copy($src_dn,$dst_dn) { $ldap=$this->config->get_ldap_link(); $ldap->cat($src_dn); $attrs= $ldap->fetch(); /* Grummble. This really sucks. PHP ldap doesn't support rdn stuff. */ $ds= ldap_connect($this->config->current['SERVER']); ldap_set_option($ds, LDAP_OPT_PROTOCOL_VERSION, 3); if (function_exists("ldap_set_rebind_proc") && isset($this->config->current['REFERRAL'])) { ldap_set_rebind_proc($ds, array(&$this, "rebind")); } $r=ldap_bind($ds,$this->config->current['ADMIN'], $this->config->current['PASSWORD']); $sr=ldap_read($ds, @LDAP::fix($src_dn), "objectClass=*"); /* Fill data from LDAP */ $new= array(); if ($sr) { $ei=ldap_first_entry($ds, $sr); if ($ei) { foreach($attrs as $attr => $val){ if ($info = @ldap_get_values_len($ds, $ei, $attr)){ for ($i= 0; $i<$info['count']; $i++){ if ($info['count'] == 1){ $new[$attr]= $info[$i]; } else { $new[$attr][]= $info[$i]; } } } } } } /* close conncetion */ ldap_unbind($ds); /* Adapt naming attribute */ $dst_name= preg_replace("/^([^=]+)=.*$/", "\\1", $dst_dn); $dst_val = preg_replace("/^[^=]+=([^,+]+).*,.*$/", "\\1", $dst_dn); $new[$dst_name]= @LDAP::fix($dst_val); /* Check if this is a department. * If it is a dep. && there is a , override in his ou * change \2C to , again, else this entry can't be saved ... */ if((isset($new['ou'])) &&( preg_match("/\\,/",$new['ou']))){ $new['ou'] = preg_replace("/\\\\,/",",",$new['ou']); } /* Save copy */ $ldap->connect(); $ldap->cd($this->config->current['BASE']); $ldap->create_missing_trees(preg_replace('/^[^,]+,/', '', $dst_dn)); /* FAIvariable=.../..., cn=.. could not be saved, because the attribute FAIvariable was different to the dn FAIvariable=..., cn=... */ if(in_array_ics("FAIdebconfInfo",$new['objectClass'])){ $new['FAIvariable'] = $ldap->fix($new['FAIvariable']); } $ldap->cd($dst_dn); $ldap->add($new); if (!$ldap->success()){ trigger_error("Trying to save $dst_dn failed.", E_USER_WARNING); return(FALSE); } return(TRUE); } /* This is a workaround function. */ function copy($src_dn, $dst_dn) { /* Rename dn in possible object groups */ $ldap= $this->config->get_ldap_link(); $ldap->search('(&(objectClass=gosaGroupOfNames)(member='.@LDAP::prepare4filter($src_dn).'))', array('cn')); while ($attrs= $ldap->fetch()){ $og= new ogroup($this->config, $ldap->getDN()); unset($og->member[$src_dn]); $og->member[$dst_dn]= $dst_dn; $og->save (); } $ldap->cat($dst_dn); $attrs= $ldap->fetch(); if (count($attrs)){ trigger_error("Trying to overwrite ".@LDAP::fix($dst_dn).", which already exists.", E_USER_WARNING); return (FALSE); } $ldap->cat($src_dn); $attrs= $ldap->fetch(); if (!count($attrs)){ trigger_error("Trying to move ".@LDAP::fix($src_dn).", which does not seem to exist.", E_USER_WARNING); return (FALSE); } $ldap->cd($src_dn); $ldap->search("objectClass=*",array("dn")); while($attrs = $ldap->fetch()){ $src = $attrs['dn']; $dst = preg_replace("/".normalizePreg($src_dn)."$/",$dst_dn,$attrs['dn']); $this->_copy($src,$dst); } return (TRUE); } /*! \brief Move a given ldap object indentified by $src_dn \ to the given destination $dst_dn \ * Ensure that all references are updated (ogroups) \ * Update ACLs \ * Update accessTo \ @param String The source dn. @param String The destination dn. @return Boolean TRUE on success else FALSE. */ function rename($src_dn, $dst_dn) { $start = microtime(1); /* Try to move the source entry to the destination position */ $ldap = $this->config->get_ldap_link(); $ldap->cd($this->config->current['BASE']); $ldap->create_missing_trees(preg_replace("/^[^,]+,/","",$dst_dn)); if (!$ldap->rename_dn($src_dn,$dst_dn)){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap->get_error(), $src_dn, "", get_class())); return(FALSE); } /* Get list of groups within this tree, maybe we have to update ACL references. */ $leaf_groups = get_list("(objectClass=posixGroup)",array("all"),$dst_dn, array("dn","objectClass"),GL_SUBSEARCH | GL_NO_ACL_CHECK); /* Get list of users within this tree, maybe we have to update ACL references. */ $leaf_users= get_list("(objectClass=gosaAccount)",array("all"),$dst_dn, array("dn","objectClass"),GL_SUBSEARCH | GL_NO_ACL_CHECK); /* Updated acls set for this groups */ foreach($leaf_groups as $group){ $new_dn = $group['dn']; $old_dn = preg_replace("/".normalizePreg($dst_dn)."$/i",$src_dn,$new_dn); $this->update_acls($old_dn,$new_dn); } /* Updated acls set for this users */ foreach($leaf_users as $user){ $new_dn = $user['dn']; $old_dn = preg_replace("/".normalizePreg($dst_dn)."$/i",$src_dn,$new_dn); $this->update_acls($old_dn,$new_dn); } /* Get all objectGroups defined in this database. and check if there is an entry matching the source dn, if this is the case, then update this objectgroup to use the new dn. */ $ogroups = get_sub_list("(&(objectClass=gosaGroupOfNames)(member=*))","ogroups", array(get_ou("ogroupou")),$this->config->current['BASE'],array("member"), GL_SUBSEARCH | GL_NO_ACL_CHECK) ; /* Walk through all objectGroups and check if there are members matching the source dn */ foreach($ogroups as $ogroup){ if(isset($ogroup['member'])){ /* Reset class object, this will be initialized with class_ogroup on demand */ $o_ogroup = NULL; for($i = 0 ; $i < $ogroup['member']['count'] ; $i ++){ $c_mem = $ogroup['member'][$i]; if(preg_match("/".normalizePreg($src_dn)."$/i",$c_mem)){ $d_mem = preg_replace("/".normalizePreg($src_dn)."$/i",$dst_dn,$ogroup['member'][$i]); if($o_ogroup == NULL){ $o_ogroup = new ogroup($this->config,$ogroup['dn']); } unset($o_ogroup->member[$c_mem]); $o_ogroup->member[$d_mem]= $d_mem; } } /* Save object group if there were changes made on the membership */ if($o_ogroup != NULL){ $o_ogroup->save(); } } } /* Check if there are gosa departments moved. If there were deps moved, the force reload of config->deps. */ $leaf_deps= get_list("(objectClass=gosaDepartment)",array("all"),$dst_dn, array("dn","objectClass"),GL_SUBSEARCH | GL_NO_ACL_CHECK); if(count($leaf_deps)){ $this->config->get_departments(); $this->config->make_idepartments(); session::set("config",$this->config); $ui =get_userinfo(); $ui->reset_acl_cache(); } return(TRUE); } function move($src_dn, $dst_dn) { /* Do not copy if only upper- lowercase has changed */ if(strtolower($src_dn) == strtolower($dst_dn)){ return(TRUE); } /* Try to move the entry instead of copy & delete */ if(TRUE){ /* Try to move with ldap routines, if this was not successfull fall back to the old style copy & remove method */ if($this->rename($src_dn, $dst_dn)){ return(TRUE); }else{ // See code below. } } /* Copy source to destination */ if (!$this->copy($src_dn, $dst_dn)){ return (FALSE); } /* Delete source */ $ldap= $this->config->get_ldap_link(); $ldap->rmdir_recursive($src_dn); if (!$ldap->success()){ trigger_error("Trying to delete $src_dn failed.", E_USER_WARNING); return (FALSE); } return (TRUE); } /* Move/Rename complete trees */ function recursive_move($src_dn, $dst_dn) { /* Check if the destination entry exists */ $ldap= $this->config->get_ldap_link(); /* Check if destination exists - abort */ $ldap->cat($dst_dn, array('dn')); if ($ldap->fetch()){ trigger_error("recursive_move $dst_dn already exists.", E_USER_WARNING); return (FALSE); } $this->copy($src_dn, $dst_dn); /* Remove src_dn */ $ldap->cd($src_dn); $ldap->recursive_remove($src_dn); return (TRUE); } function handle_post_events($mode, $add_attrs= array()) { switch ($mode){ case "add": $this->postcreate($add_attrs); break; case "modify": $this->postmodify($add_attrs); break; case "remove": $this->postremove($add_attrs); break; } } function saveCopyDialog(){ } function getCopyDialog(){ return(array("string"=>"","status"=>"")); } function PrepareForCopyPaste($source) { $todo = $this->attributes; if(isset($this->CopyPasteVars)){ $todo = array_merge($todo,$this->CopyPasteVars); } if(count($this->objectclasses)){ $this->is_account = TRUE; foreach($this->objectclasses as $class){ if(!in_array($class,$source['objectClass'])){ $this->is_account = FALSE; } } } foreach($todo as $var){ if (isset($source[$var])){ if(isset($source[$var]['count'])){ if($source[$var]['count'] > 1){ $this->$var = array(); $tmp = array(); for($i = 0 ; $i < $source[$var]['count']; $i++){ $tmp = $source[$var][$i]; } $this->$var = $tmp; }else{ $this->$var = $source[$var][0]; } }else{ $this->$var= $source[$var]; } } } } function tag_attrs(&$at, $dn= "", $tag= "", $show= false) { /* Skip tagging? If this is called from departmentGeneric, we have to skip this tagging procedure. */ if($this->skipTagging){ return; } /* No dn? Self-operation... */ if ($dn == ""){ $dn= $this->dn; /* No tag? Find it yourself... */ if ($tag == ""){ $len= strlen($dn); @DEBUG (DEBUG_TRACE, __LINE__, __FUNCTION__, __FILE__, "No tag for $dn - looking for one...", "Tagging"); $relevant= array(); foreach ($this->config->adepartments as $key => $ntag){ /* This one is bigger than our dn, its not relevant... */ if ($len < strlen($key)){ continue; } /* This one matches with the latter part. Break and don't fix this entry */ if (preg_match('/(^|,)'.normalizePreg($key).'$/', $dn)){ @DEBUG (DEBUG_TRACE, __LINE__, __FUNCTION__, __FILE__, "DEBUG: Possibly relevant: $key", "Tagging"); $relevant[strlen($key)]= $ntag; continue; } } /* If we've some relevant tags to set, just get the longest one */ if (count($relevant)){ ksort($relevant); $tmp= array_keys($relevant); $idx= end($tmp); $tag= $relevant[$idx]; $this->gosaUnitTag= $tag; } } } /* Remove tags that may already be here... */ remove_objectClass("gosaAdministrativeUnitTag", $at); if (isset($at['gosaUnitTag'])){ unset($at['gosaUnitTag']); } /* Set tag? */ if ($tag != ""){ add_objectClass("gosaAdministrativeUnitTag", $at); $at['gosaUnitTag']= $tag; } /* Initially this object was tagged. - But now, it is no longer inside a tagged department. So force the remove of the tag. (objectClass was already removed obove) */ if($tag == "" && $this->gosaUnitTag){ $at['gosaUnitTag'] = array(); } } /* Add possibility to stop remove process */ function allow_remove() { $reason= ""; return $reason; } /* Create a snapshot of the current object */ function create_snapshot($type= "snapshot", $description= array()) { /* Check if snapshot functionality is enabled */ if(!$this->snapshotEnabled()){ return; } /* Get configuration from gosa.conf */ $config = $this->config; /* Create lokal ldap connection */ $ldap= $this->config->get_ldap_link(); $ldap->cd($this->config->current['BASE']); /* check if there are special server configurations for snapshots */ if($config->get_cfg_value("snapshot_server") == ""){ /* Source and destination server are both the same, just copy source to dest obj */ $ldap_to = $ldap; $snapldapbase = $this->config->current['BASE']; }else{ $server = $config->get_cfg_value("snapshot_server"); $user = $config->get_cfg_value("snapshot_user"); $password = $config->get_cfg_value("snapshot_password"); $snapldapbase = $config->get_cfg_value("snapshot_base"); $ldap_to = new ldapMultiplexer(new LDAP($user,$password, $server)); $ldap_to -> cd($snapldapbase); if (!$ldap_to->success()){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap_to->get_error(), $snapldapbase, "", get_class())); } } /* check if the dn exists */ if ($ldap->dn_exists($this->dn)){ /* Extract seconds & mysecs, they are used as entry index */ list($usec, $sec)= explode(" ", microtime()); /* Collect some infos */ $base = $this->config->current['BASE']; $snap_base = $config->get_cfg_value("snapshot_base"); $base_of_object = preg_replace ('/^[^,]+,/i', '', $this->dn); $new_base = preg_replace("/".normalizePreg($base)."$/","",$base_of_object).$snap_base; /* Create object */ #$data = preg_replace('/^dn:.*\n/', '', $ldap->gen_ldif($this->dn,"(!(objectClass=gosaDepartment))")); $data = $ldap->gen_ldif($this->dn,"(&(!(objectClass=gosaDepartment))(!(objectClass=FAIclass)))"); $newName = preg_replace("/\./", "", $sec."-".$usec); $target= array(); $target['objectClass'] = array("top", "gosaSnapshotObject"); $target['gosaSnapshotData'] = gzcompress($data, 6); $target['gosaSnapshotType'] = $type; $target['gosaSnapshotDN'] = $this->dn; $target['description'] = $description; $target['gosaSnapshotTimestamp'] = $newName; /* Insert the new snapshot But we have to check first, if the given gosaSnapshotTimestamp is already used, in this case we should increment this value till there is an unused value. */ $new_dn = "gosaSnapshotTimestamp=".$newName.",".$new_base; $ldap_to->cat($new_dn); while($ldap_to->count()){ $ldap_to->cat($new_dn); $newName = preg_replace("/\./", "", $sec."-".($usec++)); $new_dn = "gosaSnapshotTimestamp=".$newName.",".$new_base; $target['gosaSnapshotTimestamp'] = $newName; } /* Inset this new snapshot */ $ldap_to->cd($snapldapbase); $ldap_to->create_missing_trees($snapldapbase); $ldap_to->create_missing_trees($new_base); $ldap_to->cd($new_dn); $ldap_to->add($target); if (!$ldap_to->success()){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap_to->get_error(), $new_dn, LDAP_ADD, get_class())); } if (!$ldap->success()){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap->get_error(), $new_base, "", get_class())); } } } function remove_snapshot($dn) { $ui = get_userinfo(); $old_dn = $this->dn; $this->dn = $dn; $ldap = $this->config->get_ldap_link(); $ldap->cd($this->config->current['BASE']); $ldap->rmdir_recursive($dn); $this->dn = $old_dn; } /* returns true if snapshots are enabled, and false if it is disalbed There will also be some errors psoted, if the configuration failed */ function snapshotEnabled() { $config = $this->config; if($config->get_cfg_value("enable_snapshot") == "true"){ /* Check if the snapshot_base is defined */ if ($config->get_cfg_value("snapshot_base") == ""){ msg_dialog::display(_("Configuration error"), sprintf(_("The snapshot functionality is enabled, but the required variable '%s' is not set."),"SNAPSHOT_BASE"), ERROR_DIALOG); return(FALSE); } /* check if there are special server configurations for snapshots */ if ($config->get_cfg_value("snapshot_server") != ""){ /* check if all required vars are available to create a new ldap connection */ $missing = ""; foreach(array("snapshot_server","snapshot_user","snapshot_password","snapshot_base") as $var){ if($config->get_cfg_value($var) == ""){ $missing .= $var." "; msg_dialog::display(_("Configuration error"), sprintf(_("The snapshot functionality is enabled, but the required variable '%s' is not set."), $missing), ERROR_DIALOG); return(FALSE); } } } return(TRUE); } return(FALSE); } /* Return available snapshots for the given base */ function Available_SnapsShots($dn,$raw = false) { if(!$this->snapshotEnabled()) return(array()); /* Create an additional ldap object which points to our ldap snapshot server */ $ldap= $this->config->get_ldap_link(); $ldap->cd($this->config->current['BASE']); $cfg= &$this->config->current; /* check if there are special server configurations for snapshots */ if(isset($cfg['SERVER']) && isset($cfg['SNAPSHOT_SERVER']) && $cfg['SERVER'] == $cfg['SNAPSHOT_SERVER']){ $ldap_to = $ldap; }elseif(isset($cfg['SNAPSHOT_SERVER'])){ $server = $cfg['SNAPSHOT_SERVER']; $user = $cfg['SNAPSHOT_USER']; $password = $cfg['SNAPSHOT_PASSWORD']; $snapldapbase = $cfg['SNAPSHOT_BASE']; $ldap_to = new ldapMultiplexer(new LDAP($user,$password, $server)); $ldap_to -> cd ($snapldapbase); if (!$ldap_to->success()){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap_to->get_error(), $snapldapbase, "", get_class())); } }else{ $ldap_to = $ldap; } /* Prepare bases and some other infos */ $base = $this->config->current['BASE']; $snap_base = $cfg['SNAPSHOT_BASE']; $base_of_object = preg_replace ('/^[^,]+,/i', '', $dn); $new_base = preg_replace("/".normalizePreg($base)."$/","",$base_of_object).$snap_base; $tmp = array(); /* Fetch all objects with gosaSnapshotDN=$dn */ $ldap_to->cd($new_base); $ldap_to->ls("(&(objectClass=gosaSnapshotObject)(gosaSnapshotDN=".$dn."))",$new_base, array("gosaSnapshotType","gosaSnapshotTimestamp","gosaSnapshotDN","description")); /* Put results into a list and add description if missing */ while($entry = $ldap_to->fetch()){ if(!isset($entry['description'][0])){ $entry['description'][0] = ""; } $tmp[] = $entry; } /* Return the raw array, or format the result */ if($raw){ return($tmp); }else{ $tmp2 = array(); foreach($tmp as $entry){ $tmp2[base64_encode($entry['dn'])] = $entry['description'][0]; } } return($tmp2); } function getAllDeletedSnapshots($base_of_object,$raw = false) { if(!$this->snapshotEnabled()) return(array()); /* Create an additional ldap object which points to our ldap snapshot server */ $ldap= $this->config->get_ldap_link(); $ldap->cd($this->config->current['BASE']); $cfg= &$this->config->current; /* check if there are special server configurations for snapshots */ if(isset($cfg['SNAPSHOT_SERVER'])){ $server = $cfg['SNAPSHOT_SERVER']; $user = $cfg['SNAPSHOT_USER']; $password = $cfg['SNAPSHOT_PASSWORD']; $snapldapbase = $cfg['SNAPSHOT_BASE']; $ldap_to = new ldapMultiplexer(new LDAP($user,$password, $server)); $ldap_to->cd ($snapldapbase); if (!$ldap_to->success()){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap_to->get_error(), $snapldapbase, "", get_class())); } }else{ $ldap_to = $ldap; } /* Prepare bases */ $base = $this->config->current['BASE']; $snap_base = $cfg['SNAPSHOT_BASE']; $new_base = preg_replace("/".normalizePreg($base)."$/","",$base_of_object).$snap_base; /* Fetch all objects and check if they do not exist anymore */ $ui = get_userinfo(); $tmp = array(); $ldap_to->cd($new_base); $ldap_to->ls("(objectClass=gosaSnapshotObject)",$new_base,array("gosaSnapshotType","gosaSnapshotTimestamp","gosaSnapshotDN","description")); while($entry = $ldap_to->fetch()){ $chk = str_replace($new_base,"",$entry['dn']); if(preg_match("/,ou=/",$chk)) continue; if(!isset($entry['description'][0])){ $entry['description'][0] = ""; } $tmp[] = $entry; } /* Check if entry still exists */ foreach($tmp as $key => $entry){ $ldap->cat($entry['gosaSnapshotDN'][0]); if($ldap->count()){ unset($tmp[$key]); } } /* Format result as requested */ if($raw) { return($tmp); }else{ $tmp2 = array(); foreach($tmp as $key => $entry){ $tmp2[base64_encode($entry['dn'])] = $entry['description'][0]; } } return($tmp2); } /* Restore selected snapshot */ function restore_snapshot($dn) { if(!$this->snapshotEnabled()) return(array()); $ldap= $this->config->get_ldap_link(); $ldap->cd($this->config->current['BASE']); $cfg= &$this->config->current; /* check if there are special server configurations for snapshots */ if(isset($cfg['SNAPSHOT_SERVER'])){ $server = $cfg['SNAPSHOT_SERVER']; $user = $cfg['SNAPSHOT_USER']; $password = $cfg['SNAPSHOT_PASSWORD']; $snapldapbase = $cfg['SNAPSHOT_BASE']; $ldap_to = new ldapMultiplexer(new LDAP($user,$password, $server)); $ldap_to->cd ($snapldapbase); if (!$ldap_to->success()){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap_to->get_error(), $snapldapbase, "", get_class())); } }else{ $ldap_to = $ldap; } /* Get the snapshot */ $ldap_to->cat($dn); $restoreObject = $ldap_to->fetch(); /* Prepare import string */ $data = gzuncompress($ldap_to->get_attribute($dn,'gosaSnapshotData')); /* Import the given data */ $err = ""; $ldap->import_complete_ldif($data,$err,false,false); if (!$ldap->success()){ msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap->get_error(), $dn, "", get_class())); } } function showSnapshotDialog($base,$baseSuffixe,&$parent) { $once = true; $ui = get_userinfo(); $this->parent = $parent; foreach($_POST as $name => $value){ /* Create a new snapshot, display a dialog */ if(preg_match("/^CreateSnapShotDialog_/",$name) && $once){ $once = false; $entry = preg_replace("/^CreateSnapShotDialog_/","",$name); $entry = base64_decode(preg_replace("/_[xy]$/","",$entry)); if(!empty($entry) && $ui->allow_snapshot_create($entry,$this->parent->acl_module)){ $this->snapDialog = new SnapShotDialog($this->config,$entry,$this); }else{ msg_dialog::display(_("Permission"),sprintf(_("You are not allowed to create a snapshot for %s."),$entry),ERROR_DIALOG); } } /* Restore a snapshot, display a dialog with all snapshots of the current object */ if(preg_match("/^RestoreSnapShotDialog_/",$name) && $once){ $once = false; $entry = preg_replace("/^RestoreSnapShotDialog_/","",$name); $entry = base64_decode(preg_replace("/_[xy]$/","",$entry)); if(!empty($entry) && $ui->allow_snapshot_restore($entry,$this->parent->acl_module)){ $this->snapDialog = new SnapShotDialog($this->config,$entry,$this); $this->snapDialog->display_restore_dialog = true; }else{ msg_dialog::display(_("Permission"),sprintf(_("You are not allowed to restore a snapshot for %s."),$entry),ERROR_DIALOG); } } /* Restore one of the already deleted objects */ if(((isset($_POST['menu_action']) && $_POST['menu_action'] == "RestoreDeletedSnapShot") || preg_match("/^RestoreDeletedSnapShot_/",$name)) && $once){ $once = false; if($ui->allow_snapshot_restore($base,$this->parent->acl_module)){ $this->snapDialog = new SnapShotDialog($this->config,"",$this); $this->snapDialog->set_snapshot_bases($baseSuffixe); $this->snapDialog->display_restore_dialog = true; $this->snapDialog->display_all_removed_objects = true; }else{ msg_dialog::display(_("Permission"),sprintf(_("You are not allowed to restore a snapshot for %s."),$base),ERROR_DIALOG); } } /* Restore selected snapshot */ if(preg_match("/^RestoreSnapShot_/",$name) && $once){ $once = false; $entry = preg_replace("/^RestoreSnapShot_/","",$name); $entry = base64_decode(trim(preg_replace("/_[xy]$/","",$entry))); if(!empty($entry) && $ui->allow_snapshot_restore($entry,$this->parent->acl_module)){ $this->restore_snapshot($entry); $this->snapDialog = NULL; }else{ msg_dialog::display(_("Permission"),sprintf(_("You are not allowed to restore a snapshot for %s."),$entry),ERROR_DIALOG); } } } /* Create a new snapshot requested, check the given attributes and create the snapshot*/ if(isset($_POST['CreateSnapshot']) && is_object($this->snapDialog)){ $this->snapDialog->save_object(); $msgs = $this->snapDialog->check(); if(count($msgs)){ foreach($msgs as $msg){ msg_dialog::display(_("Error"), $msg, ERROR_DIALOG); } }else{ $this->dn = $this->snapDialog->dn; $this->create_snapshot("snapshot",$this->snapDialog->CurrentDescription); $this->snapDialog = NULL; } } /* Restore is requested, restore the object with the posted dn .*/ if((isset($_POST['RestoreSnapshot'])) && (isset($_POST['SnapShot']))){ } if(isset($_POST['CancelSnapshot'])){ $this->snapDialog = NULL; } if(is_object($this->snapDialog )){ $this->snapDialog->save_object(); return($this->snapDialog->execute()); } } static function plInfo() { return array(); } function set_acl_base($base) { $this->acl_base= $base; } function set_acl_category($category) { $this->acl_category= "$category/"; } function acl_is_writeable($attribute,$skip_write = FALSE) { $ui= get_userinfo(); return preg_match('/w/', $ui->get_permissions($this->acl_base, $this->acl_category.get_class($this), $attribute, $skip_write)); } function acl_is_readable($attribute) { $ui= get_userinfo(); return preg_match('/r/', $ui->get_permissions($this->acl_base, $this->acl_category.get_class($this), $attribute)); } function acl_is_createable($base ="") { $ui= get_userinfo(); if($base == "") $base = $this->acl_base; return preg_match('/c/', $ui->get_permissions($base, $this->acl_category.get_class($this), '0')); } function acl_is_removeable($base ="") { $ui= get_userinfo(); if($base == "") $base = $this->acl_base; return preg_match('/d/', $ui->get_permissions($base, $this->acl_category.get_class($this), '0')); } function acl_is_moveable($base = "") { $ui= get_userinfo(); if($base == "") $base = $this->acl_base; return preg_match('/m/', $ui->get_permissions($base, $this->acl_category.get_class($this), '0')); } function acl_have_any_permissions() { } function getacl($attribute,$skip_write= FALSE) { $ui= get_userinfo(); return $ui->get_permissions($this->acl_base, $this->acl_category.get_class($this), $attribute,$skip_write); } /*! \brief Returns a list of all available departments for this object. If this object is new, all departments we are allowed to create a new user in are returned. If this is an existing object, return all deps. we are allowed to move tis object too. @return Array [dn] => "..name" // All deps. we are allowed to act on. */ function get_allowed_bases() { $ui = get_userinfo(); $deps = array(); /* Is this a new object ? Or just an edited existing object */ if(!$this->initially_was_account && $this->is_account){ $new = true; }else{ $new = false; } foreach($this->config->idepartments as $dn => $name){ if($new && $this->acl_is_createable($dn)){ $deps[$dn] = $name; }elseif(!$new && $this->acl_is_moveable($dn)){ $deps[$dn] = $name; } } /* Add current base */ if(isset($this->base) && isset($this->config->idepartments[$this->base])){ $deps[$this->base] = $this->config->idepartments[$this->base]; }elseif(strtolower($this->dn) == strtolower($this->config->current['BASE'])){ }else{ trigger_error("Cannot return list of departments, no default base found in class ".get_class($this).". ".$this->base); } return($deps); } /* This function modifies object acls too, if an object is moved. * $old_dn specifies the actually used dn * $new_dn specifies the destiantion dn */ function update_acls($old_dn,$new_dn,$output_changes = FALSE) { /* Check if old_dn is empty. This should never happen */ if(empty($old_dn) || empty($new_dn)){ trigger_error("Failed to check acl dependencies, wrong dn given."); return; } /* Update userinfo if necessary */ $ui = session::get('ui'); if($ui->dn == $old_dn){ $ui->dn = $new_dn; session::set('ui',$ui); new log("view","acl/".get_class($this),$this->dn,array(),"Updated current user dn from '".$old_dn."' to '".$new_dn."'"); } /* Object was moved, ensure that all acls will be moved too */ if($new_dn != $old_dn && $old_dn != "new"){ /* get_ldap configuration */ $update = array(); $ldap = $this->config->get_ldap_link(); $ldap->cd ($this->config->current['BASE']); $ldap->search("(&(objectClass=gosaAcl)(gosaAclEntry=*))",array("cn","gosaAclEntry")); while($attrs = $ldap->fetch()){ $acls = array(); /* Reset vars */ $found = false; /* Walk through acls */ for($i = 0 ; $i < $attrs['gosaAclEntry']['count'] ; $i ++ ){ /* Get Acl parts */ $acl_parts = split(":",$attrs['gosaAclEntry'][$i]); /* Get every single member for this acl */ $members = array(); if(preg_match("/,/",$acl_parts[2])){ $members = split(",",$acl_parts[2]); }else{ $members = array($acl_parts[2]); } /* Check if member match current dn */ foreach($members as $key => $member){ $member = base64_decode($member); if($member == $old_dn){ $found = true; $members[$key] = base64_encode($new_dn); } } /* Create new member string */ $new_members = ""; foreach($members as $member){ $new_members .= $member.","; } $new_members = preg_replace("/,$/","",$new_members); $acl_parts[2] = $new_members; /* Reconstruckt acl entry */ $acl_str =""; foreach($acl_parts as $t){ $acl_str .= $t.":"; } $acl_str = preg_replace("/:$/","",$acl_str); $acls[] = $acl_str; } /* Acls for this object must be adjusted */ if($found){ $debug_info= _("Changing ACL dn")." : 
 -"._("from")."  ". $old_dn."
 -"._("to")." ".$new_dn."
"; @DEBUG (DEBUG_ACL, __LINE__, __FUNCTION__, __FILE__,$debug_info,"ACL"); $update[$attrs['dn']] =array(); foreach($acls as $acl){ $update[$attrs['dn']]['gosaAclEntry'][] = $acl; } } } /* Write updated acls */ foreach($update as $dn => $attrs){ $ldap->cd($dn); $ldap->modify($attrs); } } } /* This function enables the entry Serial ID check. * If an entry was edited while we have edited the entry too, * an error message will be shown. * To configure this check correctly read the FAQ. */ function enable_CSN_check() { $this->CSN_check_active =TRUE; $this->entryCSN = getEntryCSN($this->dn); } /*! \brief Prepares the plugin to be used for multiple edit * Update plugin attributes with given array of attribtues. * @param array Array with attributes that must be updated. */ function init_multiple_support($attrs,$all) { $ldap= $this->config->get_ldap_link(); $this->multi_attrs = $attrs; $this->multi_attrs_all= $all; /* Copy needed attributes */ foreach ($this->attributes as $val){ $found= array_key_ics($val, $this->multi_attrs); if ($found != ""){ if(isset($this->multi_attrs["$found"][0])){ $this->$val= $this->multi_attrs["$found"][0]; } } } } /*! \brief Enables multiple support for this plugin */ function enable_multiple_support() { $this->ignore_account = TRUE; $this->multiple_support_active = TRUE; } /*! \brief Returns all values that have been modfied in multiple edit mode. @return array Cotaining all mdofied values. */ function get_multi_edit_values() { $ret = array(); foreach($this->attributes as $attr){ if(in_array($attr,$this->multi_boxes)){ $ret[$attr] = $this->$attr; } } return($ret); } /*! \brief Update class variables with values collected by multiple edit. */ function set_multi_edit_values($attrs) { foreach($attrs as $name => $value){ $this->$name = $value; } } /*! \brief execute plugin Generates the html output for this node */ function multiple_execute() { /* This one is empty currently. Fabian - please fill in the docu code */ session::set('current_class_for_help',get_class($this)); /* Reset Lock message POST/GET check array, to prevent perg_match errors*/ session::set('LOCK_VARS_TO_USE',array()); session::set('LOCK_VARS_USED',array()); return("Multiple edit is currently not implemented for this plugin."); } /*! \brief Save HTML posted data to object for multiple edit */ function multiple_save_object() { if(empty($this->entryCSN) && $this->CSN_check_active){ $this->entryCSN = getEntryCSN($this->dn); } /* Save values to object */ $this->multi_boxes = array(); foreach ($this->attributes as $val){ /* Get selected checkboxes from multiple edit */ if(isset($_POST["use_".$val])){ $this->multi_boxes[] = $val; } if ($this->acl_is_writeable($val) && isset ($_POST["$val"])){ /* Check for modifications */ if (get_magic_quotes_gpc()) { $data= stripcslashes($_POST["$val"]); } else { $data= $this->$val = $_POST["$val"]; } if ($this->$val != $data){ $this->is_modified= TRUE; } /* IE post fix */ if(isset($data[0]) && $data[0] == chr(194)) { $data = ""; } $this->$val= $data; } } } /*! \brief Returns all attributes of this plugin, to be able to detect multiple used attributes in multi_plugg::detect_multiple_used_attributes(). @return array Attributes required for intialization of multi_plug */ public function get_multi_init_values() { $attrs = $this->attrs; return($attrs); } /*! \brief Check given values in multiple edit @return array Error messages */ function multiple_check() { $message = plugin::check(); return($message); } /*! \brief Returns the snapshot header part for "Actions" menu in management dialogs @param $layer_menu */ function get_snapshot_header($base,$category) { $str = ""; $ui = get_userinfo(); if($this->snapshotEnabled() && $ui->allow_snapshot_restore($base,$category)){ $ok = false; foreach($this->get_used_snapshot_bases() as $base){ $ok |= count($this->getAllDeletedSnapshots($base)) >= 1 ; } if($ok){ $str = "..| "._("Restore"). "|RestoreDeletedSnapShot|\n"; }else{ $str = "..| "._("Restore")."||\n"; } } return($str); } function get_snapshot_action($base,$category) { $str= ""; $ui = get_userinfo(); if($this->snapshotEnabled()){ if ($ui->allow_snapshot_restore($base,$category)){ if(count($this->Available_SnapsShots($base))){ $str.= " "; } else { $str = " "; } } if($ui->allow_snapshot_create($base,$category)){ $str.= " "; }else{ $str = "  "; } } return($str); } function get_copypaste_action($base,$category,$class,$copy = TRUE, $cut = TRUE) { $ui = get_userinfo(); $action = ""; if($this->CopyPasteHandler){ if($cut){ if($ui->is_cutable($base,$category,$class)){ $action .= " "; }else{ $action.="  "; } } if($copy){ if($ui->is_copyable($base,$category,$class)){ $action.= " "; }else{ $action.="  "; } } } return($action); } function get_copypaste_header($base,$category,$copy = TRUE, $cut = TRUE) { $s = ""; $ui =get_userinfo(); if(!is_array($category)){ $category = array($category); } /* Check permissions for each category, if there is at least one category which support read or paste permissions for the given base, then display the specific actions. */ $readable = $pasteable = TRUE; foreach($category as $cat){ $readable |= $ui->get_category_permissions($base,$cat); $pasteable|= $ui->is_pasteable($base,$cat); } if(($cut || $copy) && isset($this->CopyPasteHandler) && is_object($this->CopyPasteHandler)){ if($readable){ $s.= "..|---|\n"; if($copy){ $s.= "..|". " "._("Copy")."|"."multiple_copy_systems|\n"; } if($cut){ $s.= "..|". " "._("Cut")."|"."multiple_cut_systems|\n"; } } if($pasteable){ if($this->CopyPasteHandler->entries_queued()){ $img = ""; $s.="..|".$img." "._("Paste")."|editPaste|\n"; }else{ $img = ""; $s.="..|".$img." "._("Paste")."\n"; } } } return($s); } function get_used_snapshot_bases() { return(array()); } } // vim:tabstop=2:expandtab:shiftwidth=2:filetype=php:syntax:ruler: ?>