Code

Cutted objects will be displayed in light grey in the management lists now.
[gosa.git] / plugins / admin / systems / class_servDNS.inc
index be1cea9ee72b9f922b907073485dd75a185ff7e4..f12ad0f8bcc77f0de704d79532a6409e11fbbd93 100644 (file)
 
 class servdns extends plugin
 {
-  /* CLI vars */
-  var $cli_summary= "Manage server basic objects";
-  var $cli_description= "Some longer text\nfor help";
-  var $cli_parameters= array("eins" => "Eins ist toll", "zwei" => "Zwei ist noch besser");
-
   /* attribute list for save action */
-  var $ignore_account= TRUE;
-  var $attributes= array("zoneName","relativeDomainName","dNSTTL","dNSClass","sOARecord"); 
-  var $objectclasses= array("whatever");
+  var $ignore_account   = FALSE;
+  var $attributes       = array(); 
+  var $objectclasses    = array("whatever");
 
-  var $RecordTypes = array();
-  var $usedRecords = array();
+  var $RecordTypes      = array();
+  var $Zones            = array();
+  var $dialog           = NULL;
 
-  var $Zones = array();
-  var $editMode = false;
+  var $orig_dn          = "";
 
-  function servdns ($config, $dn= NULL)
-  {
-    plugin::plugin ($config, $dn);
-    
-    $RecordTypes['ARecord']     ['name']  ="A Record";
-    $RecordTypes['ARecord']     ['type']  ="string";
-    $RecordTypes['MDRecord']    ['name']  ="MD Record";
-    $RecordTypes['MDRecord']    ['type']  ="string";
-    $RecordTypes['MXRecord']    ['name']  ="MX Record";
-    $RecordTypes['MXRecord']    ['type']  ="string";
-    $RecordTypes['NSRecord']    ['name']  ="NS Record";
-    $RecordTypes['NSRecord']    ['type']  ="string";
-    $RecordTypes['CNAMERecord'] ['name']  ="CNAME Record";
-    $RecordTypes['CNAMERecord'] ['type']  ="string";
-    $RecordTypes['PTRRecord']   ['name']  ="PTR Record";
-    $RecordTypes['PTRRecord']   ['type']  ="string";
-    $RecordTypes['HINFORecord'] ['name']  ="HINFO Record";
-    $RecordTypes['HINFORecord'] ['type']  ="string";
-    $RecordTypes['MINFORecord'] ['name']  ="MINFO Record";
-    $RecordTypes['MINFORecord'] ['type']  ="string";
-    $RecordTypes['TXTRecord']   ['name']  ="TXT Record";
-    $RecordTypes['TXTRecord']   ['type']  ="string";
-    $RecordTypes['AFSDBRecord'] ['name']  ="AFSDB Record";
-    $RecordTypes['AFSDBRecord'] ['type']  ="string";
-    $RecordTypes['SIGRecord']   ['name']  ="SIG Record";
-    $RecordTypes['SIGRecord']   ['type']  ="string";
-    $RecordTypes['KEYRecord']   ['name']  ="KEY Record";
-    $RecordTypes['KEYRecord']   ['type']  ="string";
-    $RecordTypes['AAAARecord']  ['name']  ="AAAA Record";
-    $RecordTypes['AAAARecord']  ['type']  ="string";
-    $RecordTypes['LOCRecord']   ['name']  ="LOC Record";
-    $RecordTypes['LOCRecord']   ['type']  ="string";
-    $RecordTypes['NXTRecord']   ['name']  ="NXT Record";
-    $RecordTypes['NXTRecord']   ['type']  ="string";
-    $RecordTypes['SRVRecord']   ['name']  ="SRV Record";
-    $RecordTypes['SRVRecord']   ['type']  ="string";
-    $RecordTypes['NAPTRRecord'] ['name']  ="NAPTR Record";
-    $RecordTypes['NAPTRRecord'] ['type']  ="string";
-    $RecordTypes['KXRecord']    ['name']  ="KX Record";
-    $RecordTypes['KXRecord']    ['type']  ="string";
-    $RecordTypes['CERTRecord']  ['name']  ="CERT Record";
-    $RecordTypes['CERTRecord']  ['type']  ="string";
-    $RecordTypes['A6Record']    ['name']  ="A6 Record";
-    $RecordTypes['A6Record']    ['type']  ="string";
-    $RecordTypes['DNAMERecord'] ['name']  ="DBNAME Record";
-    $RecordTypes['DNAMERecord'] ['type']  ="string";
-    $RecordTypes['DSRecord']    ['name']  ="DS Record";
-    $RecordTypes['DSRecord']    ['type']  ="string";
-    $RecordTypes['SSHFPRecord'] ['name']  ="SSHFP Record";
-    $RecordTypes['SSHFPRecord'] ['type']  ="string";
-    $RecordTypes['RRSIGRecord'] ['name']  ="RRSIG Record";
-    $RecordTypes['RRSIGRecord'] ['type']  ="string";
-    $RecordTypes['NSECRecord']  ['name']  ="NSEC Record";
-    $RecordTypes['NSECRecord']  ['type']  ="string";
-    $RecordTypes['mXRecord']  ['name']  ="MX Record";
-    $RecordTypes['mXRecord']  ['type']  ="string";
-    $RecordTypes['nSRecord']  ['name']  ="nS Record";
-    $RecordTypes['nSRecord']  ['type']  ="string";
-
-    $this->RecordTypes = $RecordTypes;
-
-    $attrs  = $this->attributes;
-    foreach($RecordTypes as $name => $att){
-      $attrs2[] = $name;
-    } 
-    $attrs3 = array_merge($attrs,$attrs2);
-
-    $ldap = $this->config->get_ldap_link();
-    $ldap->cd($this->dn);
-    $ldap->search("(&(objectClass=dNSZone)(sOARecord=*))",$attrs3);
-    while($zone = $ldap->fetch()){
-      $tmp = array();
-      foreach($attrs as $att){
-        if(isset($zone[$att]['count'])&&(is_array($zone[$att])))
-        unset($zone[$att]['count']);
-        
-        if(isset($zone[$att])){ 
-          $tmp[$att] = $zone[$att][0];
-        }else{
-          $tmp[$att] = "";
-        }
-      }
+  var $DNSinitially_was_account;
 
-      if(isset($tmp['sOARecord'])){
-        $tmp2 = split("\ ",$tmp['sOARecord']);
-        unset($tmp['sOARecord']);
-        $tmp['sOARecord'] = array();
-        $tmp['sOARecord']['sOAprimary'] = $tmp2[0];
-        $tmp['sOARecord']['sOAmail']    = $tmp2[1];
-        $tmp['sOARecord']['sOAserial']  = $tmp2[2];
-        $tmp['sOARecord']['sOArefresh'] = $tmp2[3];
-        $tmp['sOARecord']['sOAretry']   = $tmp2[4];
-        $tmp['sOARecord']['sOAexpire']  = $tmp2[5];
-        $tmp['sOARecord']['sOAttl']     = $tmp2[6];
-      }
 
-      $tmp['usedRecords'] = array();
-      foreach($attrs2 as $name){
-        if(isset($zone[$name])){
-          for($i = 0 ; $i < $zone[$name]['count'] ; $i++ ){
-            $tmp['usedRecords'][] =  array("type"=> $name,"value"=>$zone[$name][$i]);
-          }
-        }
-      }
-      $this->Zones[$zone['zoneName'][0]] = $tmp;
-    }
-    if(count($this->Zones > 0)){
+  function servdns ($config, $dn= NULL, $parent= NULL)
+  {
+    plugin::plugin ($config, $dn, $parent);
+
+    $this->orig_dn = $dn;
+
+    /* Get record types for zones
+     */
+    $this->RecordTypes = getDnsRecordTypes(true);
+
+    /* Get all zone Informations
+     */
+    $this->Zones = getDNSZoneEntries($config,$dn);
+  
+    /* If there is at least one entry in this -> types, we have DNS enabled 
+     */
+    if(count($this->Zones) == 0){
+      $this->is_account = false;
+    }else{
       $this->is_account = true;
     }
+    $this->DNSinitially_was_account = $this->is_account;
   }
 
+
   function execute()
   {
+    /* Call parent execute 
+     */
+    plugin::execute();
 
-  print_a($this->Zones);
-       /* Call parent execute */
-       plugin::execute();
-
-    /* Fill templating stuff */
+    /* Fill templating stuff 
+     */
     $smarty= get_smarty();
     $display= "";
 
-    /* Do we need to flip is_account state? */
+    /* Do we need to flip is_account state? 
+     */
     if (isset($_POST['modify_state'])){
-      $this->is_account= !$this->is_account;
+
+      /* Only change account state if allowed */
+      if($this->is_account && $this->acl == "#all#"){
+        $this->is_account= !$this->is_account;
+        $this->is_modified = true;
+      }elseif(!$this->is_account && chkacl($this->acl,"create") == ""){
+        $this->is_account= !$this->is_account;
+        $this->is_modified = true;
+      }
     }
 
-    /* Show tab dialog headers */
     if ($this->is_account){
       $display= $this->show_header(_("Remove DNS service"),
           _("This server has DNS features enabled. You can disable them by clicking below."));
@@ -155,94 +75,258 @@ class servdns extends plugin
       return ($display);
     }
 
-    $once = false;
-    foreach( $_POST as $name => $value){
-      if(preg_match("/^editZone_/",$name)&&!$once){
-        $once =true;
-        $tmp = preg_replace("/^editZone_/","",$name);
-        $tmp = base64_decode(preg_replace("/_.*$/","...",$tmp));
-        $this->editMode = $this->Zones[$tmp];
+
+    /* Edited or Added zone 
+     */
+    if((isset($_POST['SaveZoneChanges'])) && is_object($this->dialog)){
+      $this->dialog->save_object();
+
+      /* Check for errors  
+       */
+      if(count($this->dialog->check())){
+        foreach($this->dialog->check() as $msgs){
+          print_red($msgs); 
+        }
+      }else{
+        /* add new/edited zone 
+         */
+        $ret = $this->dialog->save();
+        if(!$this->dialog->isNew){
+          unset($this->Zones[$this->dialog->OldZoneName]);
+        }
+        $this->Zones[$ret['zoneName']] = $ret;
+        $this->dialog = NULL;
       }
     }
 
+    /* Cancel zone edit / new 
+     */
+    if(isset($_POST['CancelZoneChanges'])){
+      $this->dialog = NULL;
+    }
 
-    $smarty->assign("editMode",$this->editMode);
-    if($this->editMode){
+    /* Add empty new zone 
+     */
+    if(isset($_POST['AddZone']) && chkacl($this->acl,"servdns") == ""){
+      $this->dialog = new servdnseditZone($this->config,$this->dn);
+    }
 
-      foreach($this->attributes as $attrs){
-        $smarty->assign($attrs,$this->editMode[$attrs]);
-      }
-      
-      foreach($this->editMode['sOARecord'] as $name => $value){
-        $smarty->assign($name,$value);
+    /* Check for edit zone request 
+     */
+    $once = false;
+    foreach( $_POST as $name => $value){
+  
+      /* check all post for edit request 
+       */
+      if(preg_match("/^editZone_/",$name)&&!$once && chkacl($this->acl,"servdns") == ""){
+        $once =true;
+        $tmp = preg_replace("/^editZone_/","",$name);
+        $tmp = base64_decode(preg_replace("/_.*$/","",$tmp));
+        $this->dialog= new servdnseditZone($this->config,$this->dn,$this->Zones[$tmp]);
       }
-      $smarty->assign("records","");
 
-    }else{
-      $ZoneList = new divSelectBox("dNSZones");
-      $ZoneList -> SetHeight(254);
-
-      $editImg = "<input type='image' src='images/edit.png' name='editZone_%s'>";
-
-      foreach($this->Zones as $zone => $values ){
-        $ZoneList->AddEntry(array(array("string" => $zone.str_replace("%s",base64_encode($zone),$editImg))));
-      }    
+      /* check posts for delete zone 
+       */
+      if(preg_match("/^delZone_/",$name)&&!$once && chkacl($this->acl,"servdns") == ""){
 
+        $once =true;
+        $tmp = preg_replace("/^delZone_/","",$name);
+        $tmp = base64_decode(preg_replace("/_.*$/","",$tmp));
+     
+        /* Initiate deletion
+         */ 
+        $this->RemoveZone($tmp); 
+      }
+    }
 
-      $smarty->assign("ZoneList",$ZoneList->DrawList());
+    /* Show dialog 
+     */
+    if($this->dialog!= NULL){
+      $this->dialog->save_object();
+      $this->dialog->parent = $this;
+      return($this->dialog->execute());
     }
 
+    /* Create Listbox with existing Zones 
+     */
+    $ZoneList = new divSelectBox("dNSZones");
+    $ZoneList -> SetHeight(300);
+
+    /* Add entries to divlist
+     */
+    $editImg = "<input type='image' src='images/edit.png' name='editZone_%s'>
+      <input type='image' src='images/edittrash.png' name='delZone_%s'>";
+    foreach($this->Zones as $zone => $values ){
+      $ZoneList->AddEntry(array(
+            array("string" => getNameFromMix($zone)),
+            array("string" => _("Reverse zone")." : ".getNameFromMix($values['ReverseZone'])),
+            array("string" => _("TTL")." : ".$values['sOAttl']),
+            array("string" => _("Class")." : ".$values['dNSClass']),
+            array("string" => str_replace("%s",base64_encode($zone),$editImg))
+            ));
+    }    
+  
+    $smarty->assign("servdnsACL",chkacl($this->acl,"servdns"));
+  
+    /* Display tempalte 
+     */
+    $smarty->assign("ZoneList",$ZoneList->DrawList());
     $display.= $smarty->fetch(get_template_path('servdns.tpl', TRUE));
     return($display);
   }
 
-  function remove_from_parent()
+
+  /* Delete specified zone
+   */
+  function RemoveZone($id)
   {
-    /* This cannot be removed... */
-  }
+    $zones =  $this->getUsedZoneNames();
+    $rev  ="";
+  
+    if(isset($this->Zones[$id]['InitialReverseZone'])){
+      $rev = FlipIp(getNameFromMix($this->Zones[$id]['InitialReverseZone']));
+    }elseif(isset($this->Zones[$id]['ReverseZone'])){
+      $rev = FlipIp(getNameFromMix($this->Zones[$id]['ReverseZone']));
+    }
 
+    $zonename = "";
+    if(isset($this->Zones[$id]['InitialzoneName'])){
+      $zonename= getNameFromMix($this->Zones[$id]['InitialzoneName']);
+    }
 
-  /* Save data to object */
-  function save_object()
-  {
-    //plugin::save_object();
+    $used = array();
+
+    /* Add Records which use this zoneName
+     */
+    if(isset($zones[$zonename])){
+      $used = array_merge($used,$zones[$zonename]);
+    }
 
-    if(!$this->editMode){return;}
+    /* Add Records which uses this reverse zone
+     */
+    if(isset($zones[$rev.".in-addr.arpa"])){
+      $used = array_merge($used,$zones[$rev.".in-addr.arpa"]);
+    } 
 
-    foreach($this->attributes as $attr){
-      if(isset($_POST[$attr])){
-        $this->editMode[$attr] = $_POST[$attr];
+    /* There are still entries using this configuration
+     *  Abort deletion
+     */ 
+    if(count($used)){
+      $i = 2;
+      $str ="";
+      foreach($used as $dn){
+        if($i > 0 ){
+          $i --;
+          $str.=$dn." ";
+        }
       }
-    }
 
-    foreach($this->editMode['sOARecord'] as $name => $value){
-      if(isset($_POST[$name])){
-        $this->editMode['sOARecord'][$name] = $_POST[$name];
+      /*  Only show 2 dns in the error message 
+       */
+      if(count($used)> 2) {
+        $str .=" ... ";
       }
+      print_red(sprintf(_("Can't delete the selected zone, because it is still in use by these entry/entries '%s'"),trim($str)));
+
+    }else{
+      unset($this->Zones[$id]);
+      return(true);
     }
+    return(false);
+  } 
 
+
+  /* This funtion returns all used Zonenames 
+   */
+  function getUsedZoneNames()
+  {
+    $ret = array();
+    $ldap = $this->config->get_ldap_link();
+    $ldap->cd($this->config->current['BASE']);
+    $ldap->search("(&(objectClass=dNSZone)(!(relativeDomainName=@))(zoneName=*))",array("zoneName","relativeDomainName"));
+    while($attr = $ldap->fetch()){
+      $ret[$attr['zoneName'][0]][] = $attr['dn'];
+    }
+    return($ret);
   }
 
 
-  /* Check supplied data */
-  function check()
+
+    /* Remove dns service
+   */
+  function remove_from_parent()
   {
-    $message= array();
+    if($this->DNSinitially_was_account){
+      $bool = true;
+      foreach($this->Zones as $key => $zone){
+        $bool= $bool & $this->RemoveZone($key);
+      }
 
-    return ($message);
+      if($bool){
+        $this->save();
+      }
+      return($bool);
+    }
   }
 
 
+
   /* Save to LDAP */
   function save()
   {
-    plugin::save();
+    $ldap = $this->config->get_ldap_link();
+    $ldap->cd($this->config->current['BASE']);  
+  
+    /* Get differences 
+     */
+
+    $old_dn =  $this->orig_dn;
+    if($old_dn == "new"){
+      $old_dn = $this->dn;
+    }
+  
+    $tmp = getDNSZoneEntriesDiff($this->config,$this->Zones,$old_dn);
+
+    /* Updated zone entries if reverser or forward name has changed  
+     * Must be done before moving entries, else the given dn is invalid
+     */
+    if(isset($tmp['zoneUpdates'])){
+      foreach($tmp['zoneUpdates'] as $dn => $attrs){
+        $ldap->cd($dn);
+        $ldap->modify($attrs);
+        show_ldap_error("Zone:".$ldap->get_error(), _("Updating DNS service failed"));
+      }
+    }
 
-    /* Optionally execute a command after we're done */
-    #$this->handle_post_events($mode);
-  }
+    /* Delete dns 
+     */
+    foreach($tmp['del'] as $dn => $del){
+      $ldap->cd($dn);
+      $ldap->rmdir_recursive($dn);
+      show_ldap_error($ldap->get_error(), _("Removing DNS entries failed"));
+    }
 
-}
+    /* move follwoing entries
+     */
+    foreach($tmp['move'] as $src => $dst){
+      $this->recursive_move($src,$dst);
+    }
 
+    /* Add || Update new DNS entries
+     */
+    foreach($tmp['add'] as $dn => $attrs){
+      $ldap->cd($dn);
+      $ldap->cat($dn, array('dn'));
+      if(count($ldap->fetch())){
+        $ldap->cd($dn);
+        $ldap->modify ($attrs);
+      }else{
+        $ldap->cd($dn);
+        $ldap->add($attrs);
+      }
+      show_ldap_error($ldap->get_error(), _("Saving DNS entries failed"));
+    }
+  }
+}
 // vim:tabstop=2:expandtab:shiftwidth=2:filetype=php:syntax:ruler:
 ?>