Code

Updated divlists.
[gosa.git] / plugins / admin / fai / class_faiPackage.inc
index 5bbfd54f615f7a33aeb2452174f8269c0f9aecb0..9e45b692ed8751eeea978efbdc4761052d28ed98 100644 (file)
@@ -11,7 +11,7 @@ class faiPackage extends plugin
   var $ignore_account   = TRUE;
 
   /* Attributes for this Object */
-  var $attributes       = array("cn","description","FAIpackage","FAIdebianRelease","FAIdebianSection","FAIdebianMirror");
+  var $attributes       = array("cn","description","FAIpackage","FAIdebianRelease","FAIdebianSection", "FAIinstallMethod");
 
   /* ObjectClasses for this Object*/
   var $objectclasses    = array("top","FAIclass","FAIpackageList","FAIrepository");
@@ -27,6 +27,8 @@ class faiPackage extends plugin
   var $subAttributes    = array("cn","description"); 
   var $sub64coded       = array();
 
+  var $ConfiguredPackages = array();
+
   /* Specific attributes */
   var $cn               = "";       // The class name for this object
   var $description      = "";       // The description for this set of partitions
@@ -36,7 +38,7 @@ class faiPackage extends plugin
 
   var $FAIdebianRelease          = ""; // The selected release
   var $FAIdebianSection          = ""; // selected section
-  var $FAIdebianMirror           = ""; // selected mirror
+  var $FAIinstallMethod          = "aptitude"; // hard coded
   var $mirror                    = ""; // selected mirror
 
   var $servers          = array();  // All available servers
@@ -52,6 +54,14 @@ class faiPackage extends plugin
   var $strID            ="";
   var $newDialogShown   =false;
 
+  var $FAIstate         = "";
+
+  var $FAIinstallMethods  = array( "install", "ninstall", "remove", 
+      "dselect-upgrade", "taskinst", "taskrm",
+      "hold", "clean", "aptitude", "aptitude-r",
+      "pending", "dpkgc" );
+
+
   function faiPackage ($config, $dn= NULL)
   {
     /* Load Attributes */
@@ -63,6 +73,12 @@ class faiPackage extends plugin
      */
     if($dn != "new"){
       $this->dn =$dn;
+
+      /* Get FAIstate
+       */
+      if(isset($this->attrs['FAIstate'][0])){
+        $this->FAIstate = $this->attrs['FAIstate'][0];
+      }
     }
 
     if(isset($this->attrs['FAIpackage'])){
@@ -75,7 +91,54 @@ class faiPackage extends plugin
       $this->usedPackages = array();
     }  
 
-    unset($this->attrs['FAIdebianSection']['count']);
+    if($dn != "new"){
+
+      /* Create one filter with all package names, 
+         instead of calling $ldap->search for every single package 
+       */
+      $PackageFilter = "";
+      foreach($this->usedPackages as $name){
+        $PackageFilter .= "(FAIpackage=".$name.")";
+      }
+      $PackageFilter = "(&(objectClass=FAIdebconfInfo)(|".$PackageFilter."))";
+
+      /* Search for configuration objects */ 
+      $ldap = $this->config->get_ldap_link();
+      $ldap->cd($this->dn);
+      $ldap->search($PackageFilter,array("FAIvariable","FAIvariableType","FAIvariableContent","FAIpackage","FAIdebianSection","FAIstate"));
+
+      /* Walk through configurations and append them to our list of ConfiguredPackages */
+      while($attr = $ldap->fetch()){
+
+        /* Skip objects, that are tagged as removed */
+        if(isset($object['FAIstate'][0])){
+          if(preg_match("/removed$/",$attr['FAIstate'][0])){
+            continue;
+          }
+        }
+
+        $tmp =array(); 
+        $tmp['Name']  = $attr['FAIvariable'][0];
+        $tmp['Type']  = $attr['FAIvariableType'][0];
+
+        if (isset($attr['FAIvariableContent'][0])){
+          if(!in_array($attr['FAIvariableType'],array("multiselect"))){
+            $tmp['Value'] = $attr['FAIvariableContent'][0];
+          }else{
+            $content = array();        
+            unset($attr['FAIvariableContent']['count']);
+            foreach($attr['FAIvariableContent'] as $attr){
+              $tmp['Value'][] = $attr;
+            }
+          }
+          $this->ConfiguredPackages[$attr['FAIpackage'][0]][$attr['FAIvariable'][0]]=$tmp;
+        }
+      }
+    }
+
+    if (isset($this->attrs['FAIdebianSection']['count'])){
+      unset($this->attrs['FAIdebianSection']['count']);
+    }
     if((isset($this->attrs['FAIdebianSection']))&&(is_array($this->attrs['FAIdebianSection']))){
       $this->FAIdebianSection = array();
       foreach($this->attrs['FAIdebianSection'] as $sec){
@@ -89,101 +152,77 @@ class faiPackage extends plugin
     $this->confDir = CONFIG_DIR."/fai/";
     $this->FAIpackage = array();
 
-    /* Extract mirror location */
-    $this->mirror= preg_replace("!^[^/]+//([^/]+).*$!", '$1', $this->FAIdebianMirror);
+
+    $methods = array();
+    foreach($this->FAIinstallMethods as $method){
+      $methods[$method] = $method;
+    }
+    $this->FAIinstallMethods = $methods;
+    /* Check if we exist already - no need to ask for revisions, etc. then */
+    if ($this->dn != "new"){
+      $this->newDialogShown= true;
+    }
+
   }
 
   function execute()
   {
-       /* Call parent execute */
-       plugin::execute();
+    /* Call parent execute */
+
+    plugin::execute();
 
     /* Fill templating stuff */
     $smarty= get_smarty();
     $display= "";
 
-    $this->genMirror();
+    $smarty->assign( "FAIinstallMethods", $this->FAIinstallMethods );
 
-    /* Check if we exist already - no need to ask for revisions, etc. then */
-    if ($this->dn != "new"){
-        $this->newDialogShown= true;
-    }
+    if(!$this->is_account){
 
-    if((!$this->is_account)&&(!$this->newDialogShown)){
-      
-      if($this->dialog==NULL){
-        $this->dialog = new faiPackageNew($this->config, $this->dn,$this->mirrors,$this->servers,$this->sections,$this->releases);
-        $this->is_dialog =true;
+      /* Assemble release name */
+      $tmp= preg_replace('/,ou=fai,ou=configs,ou=systems,.*$/', '', $_SESSION['faifilter']['branch']);
+      $tmp= preg_replace('/ou=/', '', $tmp);
+      $rev= array_reverse(split(',', $tmp));
+      $this->FAIdebianRelease= "";
+      foreach ($rev as $part){
+        $this->FAIdebianRelease.= "/$part";
       }
+      $this->FAIdebianRelease= preg_replace('#^/#', '', $this->FAIdebianRelease);
 
-      /* alert possible missconfigurations */ 
-      if((count($this->releases)==0)||(count($this->servers)==0)||(count($this->sections)==0)){
-        print_red(_("There is no useable package list defined."));
+      /* Assemble sections */
+      $repos= $this->getServerInfos();
+      if(isset($repos[$this->FAIdebianRelease])){
+        $this->FAIdebianSection= $repos[$this->FAIdebianRelease];
+        $this->FAIdebianSection= array_unique($this->FAIdebianSection);
       }
 
-      /* Assign posible changes, for mirror combinations */
-      $this->dialog->save_object();
-      $this->dialog->releases = $this->releases;
-      $this->dialog->servers  = $this->servers;
-      $this->dialog->sections = $this->sections;
-      $this->dialog->mirrors  = $this->mirrors;
-
       /* Assign Repository settings*/ 
-      if(isset($_POST['SaveObjectNew'])){
-        $this->FAIdebianRelease = $this->dialog->FAIdebianRelease;
-        $this->FAIdebianSection = $this->dialog->FAIdebianSection;
-        $this->FAIdebianMirror  = $this->dialog->FAIdebianMirror;
-        unset($this->dialog);
-        $this->dialog = false;
-        $this->is_dialog=false;
-        $this->newDialogShown = true;
-      }
-
-      /* Draw dialog */
-      if($this->dialog){
-        $display=$this->dialog->execute();
-        return($display); 
-      }
+      $this->is_account     = true;
     }
 
     /* Assign variables */
     foreach($this->attributes as $attrs){
       $smarty->assign($attrs,$this->$attrs);
     }
-  
-    /* Set mirror if changed */
-    if((isset($_POST['FAIdebianMirrorS']))&&(!empty($_POST['FAIdebianMirrorS']))){
-      $this->FAIdebianMirror = $_POST['FAIdebianMirrorS'];
-    }
-  
-    /* check servers matching release and section settings */
-    $availableServer = array();
-    foreach($this->mirrors as $mir){
-      if($mir['release'] == $this->FAIdebianRelease){
-        $pass = true;
-        foreach($this->FAIdebianSection as $sec){
-          if($sec != $mir['section']){
-            $pass = false;
-          }
-        if($pass){
-            $availableServer[$mir['mirror']]=$mir['mirror'];
-          }
-        }
-      }
-    }
+
     /* Generate package list */
-    $this->list=$this->genPkgs();
+    $this->list= $this->genPkgs();
 
     /* + was pressed to open the package dialog */
-    if(isset($_POST['Addpkg'])){
+    if(isset($_POST['Addpkg']) && $this->acl_is_writeable("FAIpackage")){
       $this->dialog = new faiPackageEntry($this->config, $this->dn,$this->list,$this->usedPackages);
       $this->is_dialog =true;
     }
 
     /* Delte selected package */ 
-    if(isset($_POST['Delpkg'])){
-      unset($this->usedPackages[$_POST['usedPackages']]);
+    if(isset($_POST['Delpkg']) && $this->acl_is_writeable("FAIpackage")){
+      if($this->FAIstate != "freeze"){
+        foreach($_POST['usedPackages'] as $del){
+          if(isset($this->usedPackages[$del])){
+            unset($this->usedPackages[$del]);
+          }
+        }
+      }
     }
 
     /* Abort package selection dialog */ 
@@ -194,28 +233,61 @@ class faiPackage extends plugin
 
     /* attach new packages */
     if(isset($_POST['SaveSubObject'])) {
-      $this->dialog->save_object();
-      if(count($this->dialog->check())){
-        foreach($this->dialog->check() as $msgs){
-        print_red($msgs);
+      if($this->FAIstate != "freeze"){
+        $this->dialog->save_object();
+        if(count($this->dialog->check())){
+          foreach($this->dialog->check() as $msgs){
+            print_red($msgs);
+          }
+        }else{
+          $use = $this->dialog->save();
+          $this->usedPackages = $use;
+          $this->dialog = false;
+          $this->is_dialog=false;
+          ksort($this->usedPackages);
         }
       }else{
-        $use = $this->dialog->save();
-        $this->usedPackages = $use;
         $this->dialog = false;
         $this->is_dialog=false;
-        ksort($this->usedPackages);
       }
     }
 
     /* Configuration dialog open*/
-    if(isset($_POST['Conpkg'])){
-      $this->dialog = new faiPackageConfiguration($this->config, $this->dn,"none");
+    if((isset($_POST['Conpkg']))&&(isset($_POST['usedPackages']))&&(!empty($_POST['usedPackages'])) && $this->acl_is_writeable("FAIdebconfInfo")){
+      $path = CONFIG_DIR."/fai/".$this->FAIdebianRelease."/debconf.d";
+      $pkg_config = array();
+      $pkg = $_POST['usedPackages'][0];
+
+      if(isset($this->ConfiguredPackages[$pkg])){
+        $pkg_config = $this->ConfiguredPackages[$pkg];
+      }
+
+      $this->dialog = new faiPackageConfiguration($this->config, $this->dn,$pkg, $path, $pkg_config);
       $this->is_dialog =true;
     }
-   
+
+    /* Configuration dialog open*/
+    if($this->FAIstate != "freeze" && $this->acl_is_writeable("FAIpackage")){
+      if((isset($_POST['Markpkg']))&&(isset($_POST['usedPackages']))&&(!empty($_POST['usedPackages']))){
+        foreach($_POST['usedPackages'] as $pkg){
+          if (isset($this->usedPackages[$pkg])){
+            unset($this->usedPackages[$pkg]);
+            if (preg_match('/^-/', $pkg)){
+              $pkg= preg_replace('/^-/', '', $pkg);
+            } else {
+              $pkg= preg_replace('/^/', '-', $pkg);
+            }
+            $this->usedPackages[$pkg]= $pkg;
+          }
+        }
+      }
+    }
+
     /* Save Configuration */
     if(isset($_POST['SaveObjectConfig'])){
+      if($this->FAIstate != "freeze"){
+        $this->ConfiguredPackages= array_merge($this->ConfiguredPackages,$this->dialog->save());
+      }
       $this->dialog = false;
       $this->is_dialog=false;
     }
@@ -234,16 +306,14 @@ class faiPackage extends plugin
     /* Assign section to smarty */
     $strsec = "";
     foreach($this->FAIdebianSection as $sec){
-       $strsec .= $sec." ";
+      $strsec .= $sec." ";
     }
 
-    $smarty->assign("OptionsACL","");
-    if(empty($this->FAIdebianMirror)){
-      $smarty->assign("OptionsACL"," disabled ");
+    $tmp = $this->plInfo();
+    foreach($tmp['plProvidedAcls'] as $name => $translated){
+      $smarty->assign($name."ACL",$this->getacl($name,preg_match("/freeze/",$this->FAIstate)));
     }
 
-    $smarty->assign("mirrors" ,$availableServer);
-    $smarty->assign("mirror"  ,$this->FAIdebianMirror);
     $smarty->assign("releases",$this->releases);
     $smarty->assign("release" ,$this->FAIdebianRelease);
     $smarty->assign("sections",$this->sections);
@@ -253,68 +323,33 @@ class faiPackage extends plugin
     return($display);
   }
 
-  function genMirror(){
-    $confDir = $this->confDir; 
-    if(!is_readable($confDir)){
-      print_red(sprintf(_("Can't read configuration folder '%s'."),$confDir));
-    }else{
-      
-      /* Try to catch all available mirrors 
-         Possibly check if each server is reachable
-       */
-
-      $this->servers= array();
-      $this->releases=array();  
-      $this->sections=array(); 
-      $this->mirrors= array();
-      $fd = opendir($confDir);
-      while($mirror = readdir($fd)){
-        if((is_dir($confDir.$mirror."/"))&&(($mirror != "." )&&($mirror!=".."))){
-
-          $mirrorDir = $confDir.$mirror."/";
-
-          $fe = opendir($mirrorDir);
-          while($release = readdir($fe)){
-            if((is_dir($mirrorDir.$release))&&(($release != "." )&&($release!=".."))){
-
-              $releaseDir = $mirrorDir.$release."/";
-
-              $ff = opendir($releaseDir);
-              while($section = readdir($ff)){
-                if((is_file($releaseDir.$section))&&(($section != "." )&&($section!="..")&&(!preg_match("/.*\.in$/",$section)))){
-              
-                  $this->servers[$mirror]=$mirror;
-                  $this->releases[$release]=$release;
-                  $this->sections[$section]=$section;
-
-                  $arr=array();
-                  $arr['mirror'] = $mirror;
-                  $arr['release']= $release; 
-                  $arr['section']= $section;
-                  $this->mirrors[] = $arr ;
-                }
-              }
-              fclose($ff);
-            }
-          }
-          fclose($fe);
-        }
-      }
-      fclose($fd);
-    }
-
-
-  }
-
   /* Delete me, and all my subtrees
    */
   function remove_from_parent()
   {
     $ldap = $this->config->get_ldap_link();
     $ldap->cd ($this->dn);
-    $ldap->rmdir_recursive($this->dn);
-    $this->handle_post_events("remove");    
+
+#    $use_dn = str_ireplace( get_release_dn($this->dn), $_SESSION['faifilter']['branch'], $this->dn);
+    $use_dn = preg_replace("/".normalizePreg(get_release_dn($this->dn))."/i", $_SESSION['faifilter']['branch'], $this->dn);
+    if($_SESSION['faifilter']['branch'] == "main"){
+      $use_dn = $this->dn;
+    }
+
+    prepare_to_save_FAI_object($use_dn,array(),true);
+
+    foreach($this->ConfiguredPackages as $pkgname => $attrs){
+      foreach($attrs as $name => $attr){
+        $pkgdn =  "FAIvariable=".$name.",".$this->dn;
+#        $use_dn = str_ireplace( get_release_dn($this->dn), $_SESSION['faifilter']['branch'], $pkgdn);
+        $use_dn = preg_replace("/".normalizePreg(get_release_dn($this->dn))."/i", $_SESSION['faifilter']['branch'], $pkgdn);
+        if($_SESSION['faifilter']['branch'] == "main"){
+          $use_dn = $obj['dn'];
+        }
+        prepare_to_save_FAI_object($use_dn,array(),true);
+      }
+    }
+    $this->handle_post_events("remove");
   }
 
 
@@ -322,26 +357,22 @@ class faiPackage extends plugin
    */
   function save_object()
   {
-    
+    if($this->FAIstate == "freeze") return;  
     plugin::save_object();
-    foreach($this->attributes as $attrs){
-      if(isset($_POST[$attrs])){
-        $this->$attrs = $_POST[$attrs];
-      }
-    }
   }
 
 
   /* Check supplied data */
   function check()
   {
-    $message= array();
-  
+    /* Call common method to give check the hook */
+    $message= plugin::check();
+
     if(count($this->usedPackages)==0){
       $message[]=_("Please select a least one Package.");
     }
-  
-    if((empty($this->FAIdebianRelease))||(empty($this->FAIdebianSection))||(empty($this->FAIdebianMirror))){
+
+    if((empty($this->FAIdebianRelease))||(empty($this->FAIdebianSection))){
       $message[]=_("Please choose a valid combination for your repository setup.");
     }
 
@@ -352,17 +383,41 @@ class faiPackage extends plugin
     $a_ret=array(); 
     if(is_array($this->usedPackages)) {
       foreach($this->usedPackages as $usedName){
+
+        $config = 0;
+
+        foreach($this->ConfiguredPackages as $name => $value){
+          if($name == $usedName){
+            $config ++;
+          }
+        }
+
+        $c_str ="";
+        if($config){
+          $c_str = " - "._("package is configured");
+        }
+
+        /* Adapt used name if we're marked for removal */
+        $dsc= "";
+        if (preg_match('/^-/', $usedName)){
+          $dsc= " - "._("Package marked for removal");
+          // Generally a bad idea here, because the toggel triggers on -, not on !
+          //$usedName= preg_replace('/^-/', '! ', $usedName);
+        }else{
+          $usedName2= $usedName;
+        }
+
         if(isset($this->list[$usedName][1])){
-          $a_ret[$usedName] = $usedName." [".$this->list[$usedName][1]."]";
+          $a_ret[$usedName] = $usedName2." [".$this->list[$usedName][1]."]".$c_str.$dsc;
         }else{
-          $a_ret[$usedName] = $usedName;
+          $a_ret[$usedName] = $usedName2.$c_str.$dsc;
         }
       }
     }
     return($a_ret);
   }
 
-   function genPkgs(){
+  function genPkgs(){
     /* Generate a list off available packages for this mirror, section and release
      */
     /* Only read this file if it wasn't read before */
@@ -370,11 +425,10 @@ class faiPackage extends plugin
       $this->buffer=array();
       $a_ret = array();
       foreach($this->FAIdebianSection as $sec){
-        $strID= "/etc/gosa/fai/".preg_replace("#^[^/]+//([^/]+).*$#","\\1",$this->FAIdebianMirror)."/".$this->FAIdebianRelease."/".$sec;
-        
+        $strID= CONFIG_DIR."/fai/".$this->FAIdebianRelease."/".$sec;
+
         if(!is_file($strID)){
           print_red(sprintf(_("Package file '%s' does not exist."),$strID));
-          $this->FAIdebianMirror="";
           unset($this->buffer);
           return(array());
         }
@@ -389,7 +443,7 @@ class faiPackage extends plugin
           }
         }
         fclose($fp);
-          /* Save our Data, to avoid reading it again */
+        /* Save our Data, to avoid reading it again */
       }
       $this->buffer = $a_ret;
       ksort($a_ret);
@@ -404,33 +458,111 @@ class faiPackage extends plugin
   function save()
   {
     plugin::save();
+
     $ldap = $this->config->get_ldap_link();
 
     $this->attrs['FAIpackage'] = array();
     foreach($this->usedPackages as $pkg => $obj){
       $this->attrs['FAIpackage'][] = $pkg;
     } 
-  
+
     $this->attrs['FAIdebianSection'] = array();
     foreach($this->FAIdebianSection as $sec){
       $this->attrs['FAIdebianSection'][] = $sec;
     }
 
-    $ldap->cat($this->dn);
-    if($ldap->count()!=0){
-      /* Write FAIscript to ldap*/
-      $ldap->cd($this->dn);
-      $ldap->modify($this->attrs);
-    }else{
-      /* Write FAIscript to ldap*/
-      $ldap->cd($this->config->current['BASE']);
-      $ldap->create_missing_trees(preg_replace('/^[^,]+,/', '', $this->dn));
-      $ldap->cd($this->dn);
-      $ldap->add($this->attrs);
+    prepare_to_save_FAI_object($this->dn,$this->attrs);
+    show_ldap_error($ldap->get_error(), sprintf(_("Saving of FAI/package list with dn '%s' failed."),$this->dn));
+
+    /* Do object tagging */
+    $this->handle_object_tagging();
+    $ldap->cd($this->dn);
+
+    /* Save Package configurations */
+    foreach($this->ConfiguredPackages as $pkgname => $attrs){
+      foreach($attrs as $name => $attr){
+      
+        $pkgattrs = array();
+
+        foreach($attr as $n=>$v){
+          if(empty($v)) $v = array();
+        }
+
+        /* Set attributes */
+        $pkgattrs['objectClass'][]        = "FAIdebconfInfo";
+
+        $pkgattrs['FAIpackage']           = $pkgname;
+        $pkgattrs['FAIvariable']          = $name;
+        $pkgattrs['FAIvariableType']      = $attr['Type'];
+        $pkgattrs['FAIvariableContent']   = $attr['Value'];
+        $pkgdn =  "FAIvariable=".$name.",".$this->dn;
+
+        /* cehck if object already exists */
+        $ldap->cat($pkgdn,array("objectClass"));
+
+        /* Workaround for missing "gosaAdministrativeUnitTag" */
+        $attrs = $ldap->fetch();
+        if((isset($attrs['objectClass'])) && (in_array_ics("gosaAdministrativeUnitTag",$attrs['objectClass']))){
+          $pkgattrs['objectClass'][] = "gosaAdministrativeUnitTag";
+        }
+
+        if(in_array($pkgattrs['FAIvariableType'],array("boolean","multiselect","password","select","string","text"))){
+          prepare_to_save_FAI_object($pkgdn,$pkgattrs);
+        }
+  
+        /* Handle tagging */
+        $this->handle_object_tagging($pkgdn, $this->gosaUnitTag);
+      }
     }
-    show_ldap_error($ldap->get_error());
   }
+
+  /* Return plugin informations for acl handling */ 
+  function plInfo()
+  {
+    return (array( 
+          "plShortName" => _("Package"),
+          "plDescription" => _("FAI Package list"),
+          "plSelfModify"  => FALSE,
+          "plDepends"     => array(),
+          "plPriority"    => 28,
+          "plSection"     => array("administration"),
+          "plCategory"    => array("fai"),
+          "plProvidedAcls" => array(
+            "cn"                => _("Name"),
+            "description"       => _("Description"),
+            "FAIpackage"        => _("Packages"),
+            "FAIdebianSection"  => _("Section")." ("._("Readonly").")",
+            "FAIinstallMethod"  => _("Install Method"),
+            "FAIdebconfInfo"    => _("Package configuration"),
+            "FAIdebianRelease"  => _("Release")." ("._("Readonly").")")
+          ));
+  }
+
+  function getServerInfos()
+  {
+    $ret = array();
+    $ldap = $this->config->get_ldap_link();
+    $ldap->cd($this->config->current['BASE']);
+    $ldap->search("(objectClass=FAIrepositoryServer)",array("*"));
+    while($attrs = $ldap->fetch()){
+      if(isset($attrs['FAIrepository'])){
+        for($i =0 ; $i < $attrs['FAIrepository']['count']; $i++){
+          $obj = $attrs['FAIrepository'][$i];
+          $tmp = split("\|",$obj);
+          if(count($tmp)==4){
+            foreach(split(",",$tmp[3]) as $sec){
+              if(!empty($sec)){
+                $ret[$tmp[2]][] =  $sec;
+              }
+            }
+          }
+        }
+      }
+    }
+    return($ret);
+  }
+
+
 }
 
 // vim:tabstop=2:expandtab:shiftwidth=2:filetype=php:syntax:ruler: