# HG changeset patch
# User Bernhard Herzog <bh@intevation.de>
# Date 1173439561 -3600
# Node ID fee641fec94e91f8e5c4b6bff680242e45f2ba43
# Parent  7e9db903ba1662f0b9bcc4ea257c7d6aac9e7b80
Separate the kolab specific parts.

diff -r 7e9db903ba16 -r fee641fec94e enterprise/kdepim.py
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/enterprise/kdepim.py	Fri Mar 09 12:26:01 2007 +0100
@@ -0,0 +1,82 @@
+# Copyright (C) 2007 by Intevation GmbH
+# Authors:
+# Bernhard Herzog <bh@intevation.de>
+#
+# This program is free software under the GPL (>=v2)
+# Read the file COPYING coming with the software for details.
+
+"""Packager that builds KDE-PIM debian packages from the enterprise branch"""
+
+import os
+import time
+import re
+
+import treepkg.util
+import treepkg.packager
+
+
+class SourcePackager(treepkg.packager.SourcePackager):
+
+    pkg_basename = "kdepim"
+
+    def __init__(self, *args, **kw):
+        super(SourcePackager, self).__init__(*args, **kw)
+        self.enterprise_version = (time.strftime("%Y%m%d", time.localtime()) \
+                                   + "." + str(self.revision))
+
+    def kdepim_version(self, directory):
+        """Determine the kdepim version.
+
+        The version is taken from the kdepim.lsm file.
+        """
+        return treepkg.util.extract_lsm_version(os.path.join(directory,
+                                                             "kdepim.lsm"))
+
+    def determine_package_version(self, directory):
+        enterprise_version = self.enterprise_version
+        kdepimversion = self.kdepim_version(directory)
+        version_template = "%(kdepimversion)s.enterprise.0.%(enterprise_version)s"
+        return version_template % locals()
+
+    def update_version_numbers(self, pkgbasedir):
+        """Overrides the inherited method to update version numbers in the code
+        """
+        versionstring = "(enterprise %s)" % self.enterprise_version
+        for versionfile in ["kmail/kmversion.h", "kontact/src/main.cpp",
+                            "korganizer/version.h"]:
+            filename = os.path.join(pkgbasedir, versionfile)
+            patched = re.sub("\(enterprise ([^)]*)\)", versionstring,
+                              open(filename).read())
+            f = open(filename, "w")
+            f.write(patched)
+            f.close()
+
+    def do_package(self):
+        pkgbaseversion, pkgbasedir = self.export_sources()
+        self.update_version_numbers(pkgbasedir)
+
+        pkgbasename = "kdepim_" + pkgbaseversion
+        origtargz = os.path.join(self.work_dir,
+                                 pkgbasename + ".orig.tar.gz")
+        self.create_tarball(origtargz, self.work_dir,
+                            os.path.basename(pkgbasedir))
+
+        changemsg = ("Update to SVN enterprise branch rev. %d"
+                     % (self.revision,))
+        self.copy_debian_directory(pkgbasedir, pkgbaseversion,
+                                   changemsg)
+
+        self.create_source_package(pkgbasedir, origtargz)
+        self.move_source_package(pkgbasename)
+
+
+class RevisionPackager(treepkg.packager.RevisionPackager):
+
+    source_packager_cls = SourcePackager
+
+
+class AssemblyLine(treepkg.packager.AssemblyLine):
+
+    revision_packager_cls = RevisionPackager
+
+    svn_external_subdirs = ["admin"]
diff -r 7e9db903ba16 -r fee641fec94e runtreepkg.py
--- a/runtreepkg.py	Thu Mar 08 20:23:28 2007 +0100
+++ b/runtreepkg.py	Fri Mar 09 12:26:01 2007 +0100
@@ -12,7 +12,7 @@
 import logging
 from optparse import OptionParser
 
-from treepkg.packager import AssemblyLine, Packager
+from treepkg.packager import create_packager, Packager
 from treepkg.readconfig import read_config
 
 def initialize_logging():
@@ -36,8 +36,8 @@
 
     initialize_logging()
 
-    treepkg_opts, assembly_line_opts = read_config(options.config_file)
-    packager = Packager([AssemblyLine(**opts) for opts in assembly_line_opts],
+    treepkg_opts, packager_opts = read_config(options.config_file)
+    packager = Packager([create_packager(**opts) for opts in packager_opts],
                         **treepkg_opts)
     packager.run()
 
diff -r 7e9db903ba16 -r fee641fec94e treepkg/packager.py
--- a/treepkg/packager.py	Thu Mar 08 20:23:28 2007 +0100
+++ b/treepkg/packager.py	Fri Mar 09 12:26:01 2007 +0100
@@ -27,56 +27,87 @@
 
 class SourcePackager(object):
 
+    # Derived classes must supply the package basename
+    pkg_basename = None
+
     def __init__(self, plant, status, work_dir, src_dir, revision):
         self.plant = plant
         self.status = status
         self.work_dir = work_dir
         self.src_dir = src_dir
         self.revision = revision
-        self.enterprise_version = (time.strftime("%Y%m%d", time.localtime()) \
-                                   + "." + str(self.revision))
-
-    def kdepim_version(self, directory):
-        """Determine the kdepim version.
-
-        The version is taken from the kdepim.lsm file in the plants
-        checkout dir.
-        """
-        return util.extract_lsm_version(os.path.join(directory, "kdepim.lsm"))
+        assert(self.pkg_basename)
 
     def determine_package_version(self, directory):
-        enterprise_version = self.enterprise_version
-        kdepimversion = self.kdepim_version(directory)
-        version_template = "%(kdepimversion)s.enterprise.0.%(enterprise_version)s"
-        return version_template % locals()
+        """Returns the version number of the new package as a string
+
+        The directory parameter is the name of the directory containing
+        the newly exported sources.  The sources were exported with the
+        export_sources method.
+
+        The default implementation simply returns the revision converted
+        to a string.
+        """
+        return str(self.revision)
 
     def export_sources(self):
+        """Export the sources from the subversion working directory
+
+        This method first exports the sources to a temporary directory
+        and then renames the directory.  The new name is of the form
+
+          <pkg_basename>-<version>
+
+        Where pkg_basename is the value of self.pkg_basename and version
+        is the return value of the determine_package_version() method.
+        """
         temp_dir = os.path.join(self.work_dir, "temp")
         self.plant.export_sources(temp_dir)
 
         pkgbaseversion = self.determine_package_version(temp_dir)
-        pkgbasedir = os.path.join(self.work_dir, "kdepim-" + pkgbaseversion)
+        pkgbasedir = os.path.join(self.work_dir,
+                                  self.pkg_basename + "-" + pkgbaseversion)
 
         os.rename(temp_dir, pkgbasedir)
         return pkgbaseversion, pkgbasedir
 
 
     def update_version_numbers(self, pkgbasedir):
-        versionstring = "(enterprise %s)" % self.enterprise_version
-        for versionfile in ["kmail/kmversion.h", "kontact/src/main.cpp",
-                            "korganizer/version.h"]:
-            filename = os.path.join(pkgbasedir, versionfile)
-            patched = re.sub("\(enterprise ([^)]*)\)", versionstring,
-                              open(filename).read())
-            f = open(filename, "w")
-            f.write(patched)
-            f.close()
+        """Updates the version numbers in the code in pkgbasedir.
+
+        The default implementation does nothing.  Derived classes should
+        override this method if necessary.
+        """
 
     def create_tarball(self, tarballname, workdir, basedir):
+        """Creates a new tarball.
+
+        Parameters:
+
+          tarballname -- the filename of the new tarball
+          workdir -- The directory into which to change before running tar.
+                     (actually this is done with GNUI tar's -C option)
+          basedir -- The basedirectory of the files that are packaged
+                     into the tarfile.  This should be a relative
+                     filename directly in workdir.
+        """
         logging.info("Creating tarball %r", tarballname)
         run.call(["tar", "czf", tarballname, "-C", workdir, basedir])
 
     def copy_debian_directory(self, pkgbasedir, pkgbaseversion, changemsg):
+        """Copies the debian directory and updates the copy's changelog
+
+        Parameter:
+          pkgbasedir -- The directory holding the unpacked source package
+          pkgbaseversion -- The version to update the changelog to
+          changemsg -- The message for the changelog
+
+        When determining the actual version for the new package, this
+        function looks at the previous version in the changelog.  If it
+        has a prefix separated from the version number by a colon this
+        prefix is prepended to the pkgbaseversion parameter.  Debian
+        uses such prefixes for the kde packages.
+        """
         debian_dir = os.path.join(pkgbasedir, "debian")
         changelog = os.path.join(debian_dir, "changelog")
 
@@ -95,6 +126,7 @@
 
 
     def create_source_package(self, pkgbasedir, origtargz):
+        """Creates a new source package from pkgbasedir and origtargz"""
         logging.info("Creating new source package")
         run.call(["dpkg-source", "-b", os.path.basename(pkgbasedir),
                   os.path.basename(origtargz)],
@@ -102,37 +134,41 @@
                  suppress_output=True,
                  env=self.plant.debian_environment())
 
+    def move_source_package(self, pkgbasename):
+        """Moves the new source package from the work_dir to the src_dir"""
+        logging.info("Moving source package to %r", self.src_dir)
+        util.ensure_directory(self.src_dir)
+        for filename in [filename for filename in os.listdir(self.work_dir)
+                                  if filename.startswith(pkgbasename)]:
+            os.rename(os.path.join(self.work_dir, filename),
+                      os.path.join(self.src_dir, filename))
+
     def package(self):
+        """Creates a source package from a subversion checkout.
+
+        After setting up the working directory, this method calls the
+        do_package method to do the actual packaging.  Afterwards the
+        work directory is removed.
+        """
         util.ensure_directory(self.work_dir)
         try:
             self.status.set("creating_source_package")
-            pkgbaseversion, pkgbasedir = self.export_sources()
-            self.update_version_numbers(pkgbasedir)
-
-            pkgbasename = "kdepim_" + pkgbaseversion
-            origtargz = os.path.join(self.work_dir,
-                                     pkgbasename + ".orig.tar.gz")
-            self.create_tarball(origtargz, self.work_dir,
-                                os.path.basename(pkgbasedir))
-
-            changemsg = ("Update to SVN enterprise branch rev. %d"
-                         % (self.revision,))
-            self.copy_debian_directory(pkgbasedir, pkgbaseversion,
-                                       changemsg)
-
-            self.create_source_package(pkgbasedir, origtargz)
-
-            logging.info("Moving source package to %r", self.src_dir)
-            util.ensure_directory(self.src_dir)
-            for filename in [filename for filename in os.listdir(self.work_dir)
-                                      if filename.startswith(pkgbasename)]:
-                os.rename(os.path.join(self.work_dir, filename),
-                          os.path.join(self.src_dir, filename))
+            self.do_package()
             self.status.set("source_package_created")
         finally:
             logging.info("Removing workdir %r", self.work_dir)
             shutil.rmtree(self.work_dir)
 
+    def do_package(self):
+        """Does the work of creating a source package
+        This method must be overriden by derived classes.
+
+        The method should do the work in self.work_dir.  When the
+        package is done, the source package files should be in
+        self.src_dir.
+        """
+        raise NotImplementedError
+
 
 class BinaryPackager(object):
 
@@ -155,6 +191,9 @@
 
 class RevisionPackager(object):
 
+    source_packager_cls = SourcePackager
+    binary_packager_cls = BinaryPackager
+
     def __init__(self, plant, revision):
         self.plant = plant
         self.revision = revision
@@ -173,19 +212,19 @@
 
     def package(self):
         try:
-            src_packager = SourcePackager(self.plant, self.status,
-                                          self.work_dir, self.src_dir,
-                                          self.revision)
+            src_packager = self.source_packager_cls(self.plant, self.status,
+                                                    self.work_dir, self.src_dir,
+                                                    self.revision)
             src_packager.package()
 
             dsc_file = self.find_dsc_file()
             if dsc_file is None:
                 raise RuntimeError("Cannot find dsc File in %r" % self.src_dir)
 
-            bin_packager = BinaryPackager(self.plant, self.status,
-                                          self.binary_dir, dsc_file,
-                                          os.path.join(self.base_dir,
-                                                       "build.log"))
+            bin_packager = self.binary_packager_cls(self.plant, self.status,
+                                                    self.binary_dir, dsc_file,
+                                                    os.path.join(self.base_dir,
+                                                                 "build.log"))
             bin_packager.package()
         except:
             self.status.set("failed", traceback.format_exc())
@@ -198,8 +237,14 @@
 
 class AssemblyLine(object):
 
+    revision_packager_cls = RevisionPackager
+
+    svn_external_subdirs = []
+
+    extra_config_desc = []
+
     def __init__(self, name, base_dir, svn_url, root_cmd, deb_email,
-                 deb_fullname):
+                 deb_fullname, packager_class="treepkg.packager"):
         self.name = name
         self.base_dir = base_dir
         self.svn_url = svn_url
@@ -219,10 +264,11 @@
                             self.pkg_dir_template % locals())
 
     def last_changed_revision(self):
-        rev1 = subversion.last_changed_revision(self.checkout_dir)
-        rev2 = subversion.last_changed_revision(os.path.join(self.checkout_dir,
-                                                             "admin"))
-        return max(rev1, rev2)
+        revisions = []
+        for directory in [self.checkout_dir] + self.svn_external_subdirs:
+            directory = os.path.join(self.checkout_dir, directory)
+            revisions.append(subversion.last_changed_revision(directory))
+        return max(revisions)
 
     def last_packaged_revision(self):
         """Returns the revision number of the highest packaged revision.
@@ -265,10 +311,11 @@
         # some versions of svn (notably version 1.4.2 shipped with etch)
         # do export externals such as the admin subdirectory.  We may
         # have to do that in an extra step.
-        admindir = os.path.join(to_dir, "admin")
-        if not os.path.isdir(admindir):
-            subversion.export(os.path.join(self.checkout_dir, "admin"),
-                              admindir)
+        for subdir in self.svn_external_subdirs:
+            absdir = os.path.join(to_dir, subdir)
+            if not os.path.isdir(absdir):
+                subversion.export(os.path.join(self.checkout_dir, subdir),
+                                  absdir)
 
     def copy_debian_directory(self, to_dir):
         logging.info("Copying debian directory to %r", to_dir)
@@ -290,11 +337,15 @@
         logging.info("Previously packaged revision was %d", previous_revision)
         if current_revision > previous_revision:
             logging.info("New revision is not packaged yet")
-            return RevisionPackager(self, current_revision)
+            return self.revision_packager_cls(self, current_revision)
         else:
             logging.info("New revision already packaged.")
 
 
+def create_packager(packager_class, **kw):
+    module = util.import_dotted_name(packager_class)
+    return module.AssemblyLine(**kw)
+
 
 class Packager(object):
 
diff -r 7e9db903ba16 -r fee641fec94e treepkg/readconfig.py
--- a/treepkg/readconfig.py	Thu Mar 08 20:23:28 2007 +0100
+++ b/treepkg/readconfig.py	Fri Mar 09 12:26:01 2007 +0100
@@ -11,10 +11,12 @@
 import shlex
 from ConfigParser import SafeConfigParser, NoOptionError
 
+import util
+
 defaults = dict(root_cmd="sudo")
 
 packager_desc = [
-    "name", "base_dir", "svn_url",
+    "name", "base_dir", "svn_url", "packager_class",
     ("root_cmd", shlex.split),
     "deb_email", "deb_fullname",
     ]
@@ -59,7 +61,10 @@
     packagers = []
     for section in parser.sections():
         if section.startswith("pkg_"):
-            packagers.append(read_config_section(parser, section, packager_desc,
+            packager_class = parser.get(section, "packager_class")
+            module = util.import_dotted_name(packager_class)
+            desc = packager_desc + module.AssemblyLine.extra_config_desc
+            packagers.append(read_config_section(parser, section, desc,
                                                  dict(name=section[4:])))
 
     # main config
diff -r 7e9db903ba16 -r fee641fec94e treepkg/run.py
--- a/treepkg/run.py	Thu Mar 08 20:23:28 2007 +0100
+++ b/treepkg/run.py	Fri Mar 09 12:26:01 2007 +0100
@@ -29,6 +29,7 @@
     """
     if suppress_output:
         kw["stdout"] = open(os.devnull, "w")
+        kw["stderr"] = open(os.devnull, "w")
     ret = subprocess.call(command, **kw)
     if ret != 0:
         raise SubprocessError(command, ret)
diff -r 7e9db903ba16 -r fee641fec94e treepkg/util.py
--- a/treepkg/util.py	Thu Mar 08 20:23:28 2007 +0100
+++ b/treepkg/util.py	Fri Mar 09 12:26:01 2007 +0100
@@ -9,10 +9,17 @@
 
 import os
 import tempfile
+import shutil
 
 import run
 
 
+def import_dotted_name(dotted_name):
+    module = __import__(dotted_name)
+    for name in dotted_name.split(".")[1:]:
+        module = getattr(module, name)
+    return module
+
 def extract_value_for_key(lines, key):
     """Parses a sequence of strings for a key and returns the associated value
 
@@ -42,6 +49,40 @@
         os.makedirs(directory)
 
 
+def copytree(src, dst, symlinks=False):
+    """Recursively copy a directory tree using copy2().
+
+    This version is basically the same as the one in the shutil module
+    in the python standard library, however, it's OK if the destination
+    directory already exists.
+
+    If the optional symlinks flag is true, symbolic links in the
+    source tree result in symbolic links in the destination tree; if
+    it is false, the contents of the files pointed to by symbolic
+    links are copied.
+    """
+    names = os.listdir(src)
+    ensure_directory(dst)
+    errors = []
+    for name in names:
+        srcname = os.path.join(src, name)
+        dstname = os.path.join(dst, name)
+        try:
+            if symlinks and os.path.islink(srcname):
+                linkto = os.readlink(srcname)
+                os.symlink(linkto, dstname)
+            elif os.path.isdir(srcname):
+                copytree(srcname, dstname, symlinks)
+            else:
+                shutil.copy2(srcname, dstname)
+            # XXX What about devices, sockets etc.?
+        except (IOError, os.error), why:
+            errors.append((srcname, dstname, why))
+    if errors:
+        raise Error, errors
+
+
+
 def writefile(filename, contents):
     """Write contents to filename in an atomic way.