2 # -*- coding: utf-8 -*-
4 ## Copyright (C) 2009 manatlan manatlan[at]gmail(dot)com
6 ## This program is free software; you can redistribute it and/or modify
7 ## it under the terms of the GNU General Public License as published
8 ## by the Free Software Foundation; version 2 only.
10 ## This program is distributed in the hope that it will be useful,
11 ## but WITHOUT ANY WARRANTY; without even the implied warranty of
12 ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 ## GNU General Public License for more details.
17 - don't sign package (-us -uc)
18 - no distinctions between author and maintainer(packager)
21 - dpkg-dev (dpkg-buildpackage)
27 - ??? 9/14/2009 (By epage)
30 - fixed bug where it couldn't handle the contents of the pre/post scripts being specified
31 - Added customization based on the targeted policy for sections (Maemo support)
32 - Added maemo specific tarball, dsc, changes file generation support (including icon support)
34 - pre/post install/remove scripts enabled
35 - deb package install py2deb in dist-packages for py2.6
37 - use os.environ USERNAME or USER (debian way)
38 - install on py 2.(4,5,6) (*FIX* do better here)
52 from tarfile import TarFile
54 from datetime import datetime
55 import socket # gethostname()
56 from subprocess import Popen, PIPE
58 #~ __version__ = "0.4"
60 __author__ = "manatlan"
61 __mail__ = "manatlan@gmail.com"
64 PERMS_URW_GRW_OR = stat.S_IRUSR | stat.S_IWUSR | \
65 stat.S_IRGRP | stat.S_IWGRP | \
73 p = Popen(cmds, shell=False, stdout=PIPE, stderr=PIPE)
74 time.sleep(0.01) # to avoid "IOError: [Errno 4] Interrupted system call"
75 out = string.join(p.stdout.readlines()).strip()
76 outerr = string.join(p.stderr.readlines()).strip()
81 txt=run(['alien', '-r', file])
82 return txt.split(" generated")[0]
85 def py2src(TEMP, name):
86 l=glob("%(TEMP)s/%(name)s*.tar.gz"%locals())
88 raise Py2debException("don't find source package tar.gz")
90 tar = os.path.basename(l[0])
91 shutil.move(l[0], tar)
97 f = open(filename, "r")
99 return hashlib.md5(f.read()).hexdigest()
104 class Py2changes(object):
106 def __init__(self, ChangedBy, description, changes, files, category, repository, **kwargs):
107 self.options = kwargs # TODO: Is order important?
108 self.description = description
111 self.category=category
112 self.repository=repository
113 self.ChangedBy=ChangedBy
115 def getContent(self):
116 content = ["%s: %s" % (k, v)
117 for k,v in self.options.iteritems()]
120 description=self.description.replace("\n","\n ")
121 content.append('Description: ')
122 content.append(' %s' % description)
124 changes=self.changes.replace("\n","\n ")
125 content.append('Changes: ')
126 content.append(' %s' % changes)
128 content.append("Changed-By: %s" % self.ChangedBy)
130 content.append('Files:')
132 for onefile in self.files:
133 md5 = md5sum(onefile)
134 size = os.stat(onefile).st_size.__str__()
135 content.append(' ' + md5 + ' ' + size + ' ' + self.category +' '+self.repository+' '+os.path.basename(onefile))
137 return "\n".join(content) + "\n\n"
140 def py2changes(params):
141 changescontent = Py2changes(
142 "%(author)s <%(mail)s>" % params,
143 "%(description)s" % params,
144 "%(changelog)s" % params,
146 "%(TEMP)s/%(name)s_%(version)s.tar.gz" % params,
147 "%(TEMP)s/%(name)s_%(version)s.dsc" % params,
149 "%(section)s" % params,
150 "", #"%(repository)s" % params,
152 Date=time.strftime("%a, %d %b %Y %H:%M:%S +0000", time.gmtime()),
153 Source="%(name)s" % params,
154 Architecture="%(arch)s" % params,
155 Version="%(version)s" % params,
156 Distribution="", #"%(distribution)s" % params,
157 Urgency="", #"%(urgency)s" % params,
158 Maintainer="%(author)s <%(mail)s>" % params
160 f = open("%(TEMP)s/%(name)s_%(version)s.changes" % params,"wb")
161 f.write(changescontent.getContent())
164 fileHandle = open('/tmp/py2deb2.tmp', 'w')
165 fileHandle.write('#!/bin/sh\n')
166 fileHandle.write("cd " +os.getcwd()+ "\n")
167 fileHandle.write("gpg --local-user %(mail)s --clearsign %(TEMP)s/%(name)s_%(version)s.changes\n" % params)
168 fileHandle.write("mv %(TEMP)s/%(name)s_%(version)s.changes.asc %(TEMP)s/%(name)s_%(version)s.changes\n" % params)
169 fileHandle.write('\nexit')
171 commands.getoutput("chmod 777 /tmp/py2deb2.tmp")
172 commands.getoutput("/tmp/py2deb2.tmp")
176 l=glob("%(TEMP)s/%(name)s*.tar.gz" % params)
178 raise Py2debException("don't find source package tar.gz")
179 tar = os.path.basename(l[0])
180 shutil.move(l[0],tar)
183 l=glob("%(TEMP)s/%(name)s*.dsc" % params)
185 raise Py2debException("don't find source package dsc")
186 tar = os.path.basename(l[0])
187 shutil.move(l[0],tar)
190 l=glob("%(TEMP)s/%(name)s*.changes" % params)
192 raise Py2debException("don't find source package changes")
193 tar = os.path.basename(l[0])
194 shutil.move(l[0],tar)
200 class Py2dsc(object):
202 def __init__(self, StandardsVersion, BuildDepends, files, **kwargs):
203 self.options = kwargs # TODO: Is order important?
204 self.StandardsVersion = StandardsVersion
205 self.BuildDepends=BuildDepends
210 content = ["%s: %s" % (k, v)
211 for k,v in self.options.iteritems()]
213 if self.BuildDepends:
214 content.append("Build-Depends: %s" % self.BuildDepends)
215 if self.StandardsVersion:
216 content.append("Standards-Version: %s" % self.StandardsVersion)
218 content.append('Files:')
220 for onefile in self.files:
222 md5 = md5sum(onefile)
223 size = os.stat(onefile).st_size.__str__()
224 content.append(' '+md5 + ' ' + size +' '+os.path.basename(onefile))
226 return "\n".join(content)+"\n\n"
229 def py2dsc(TEMP, name, version, depends, author, mail, arch):
230 dsccontent = Py2dsc("%(version)s" % locals(),
231 "%(depends)s" % locals(),
232 ("%(TEMP)s/%(name)s_%(version)s.tar.gz" % locals(),),
234 Source="%(name)s" % locals(),
235 Version="%(version)s" % locals(),
236 Maintainer="%(author)s <%(mail)s>" % locals(),
237 Architecture="%(arch)s" % locals(),
240 filename = "%(TEMP)s/%(name)s_%(version)s.dsc" % locals()
242 f = open(filename, "wb")
244 f.write(dsccontent.content)
248 fileHandle = open('/tmp/py2deb.tmp', 'w')
250 fileHandle.write('#!/bin/sh\n')
251 fileHandle.write("cd " + os.getcwd() + "\n")
252 fileHandle.write("gpg --local-user %(mail)s --clearsign %(TEMP)s/%(name)s_%(version)s.dsc\n" % locals())
253 fileHandle.write("mv %(TEMP)s/%(name)s_%(version)s.dsc.asc %(filename)s\n" % locals())
254 fileHandle.write('\nexit')
259 commands.getoutput("chmod 777 /tmp/py2deb.tmp")
260 commands.getoutput("/tmp/py2deb.tmp")
265 class Py2tar(object):
267 def __init__(self, dataDirectoryPath):
268 self._dataDirectoryPath = dataDirectoryPath
271 return self._getSourcesFiles()
273 def _getSourcesFiles(self):
274 directoryPath = self._dataDirectoryPath
276 outputFileObj = StringIO.StringIO() # TODO: Do more transparently?
278 tarOutput = TarFile.open('sources',
280 fileobj = outputFileObj)
282 # Note: We can't use this because we need to fiddle permissions:
283 # tarOutput.add(directoryPath, arcname = "")
285 for root, dirs, files in os.walk(directoryPath):
286 archiveRoot = root[len(directoryPath):]
288 tarinfo = tarOutput.gettarinfo(root, archiveRoot)
289 # TODO: Make configurable?
290 tarinfo.uid = UID_ROOT
291 tarinfo.gid = GID_ROOT
294 tarOutput.addfile(tarinfo)
297 tarinfo = tarOutput.gettarinfo(os.path.join(root, f),
298 os.path.join(archiveRoot, f))
299 tarinfo.uid = UID_ROOT
300 tarinfo.gid = GID_ROOT
303 tarOutput.addfile(tarinfo, file(os.path.join(root, f)))
307 data_tar_gz = outputFileObj.getvalue()
312 def py2tar(DEST, TEMP, name, version):
313 tarcontent = Py2tar("%(DEST)s" % locals())
314 filename = "%(TEMP)s/%(name)s_%(version)s.tar.gz" % locals()
315 f = open(filename, "wb")
317 f.write(tarcontent.packed())
323 class Py2debException(Exception):
327 SECTIONS_BY_POLICY = {
328 # http://www.debian.org/doc/debian-policy/ch-archive.html#s-subsections
329 "debian": "admin, base, comm, contrib, devel, doc, editors, electronics, embedded, games, gnome, graphics, hamradio, interpreters, kde, libs, libdevel, mail, math, misc, net, news, non-free, oldlibs, otherosfs, perl, python, science, shells, sound, tex, text, utils, web, x11",
330 # http://maemo.org/forrest-images/pdf/maemo-policy.pdf
331 "chinook": "accessories, communication, games, multimedia, office, other, programming, support, themes, tools",
332 # http://wiki.maemo.org/Task:Package_categories
333 "diablo": "user/desktop, user/development, user/education, user/games, user/graphics, user/multimedia, user/navigation, user/network, user/office, user/science, user/system, user/utilities",
334 # http://wiki.maemo.org/Task:Fremantle_application_categories
335 "mer": "user/desktop, user/development, user/education, user/games, user/graphics, user/multimedia, user/navigation, user/network, user/office, user/science, user/system, user/utilities",
336 # http://wiki.maemo.org/Task:Fremantle_application_categories
337 "fremantle": "user/desktop, user/development, user/education, user/games, user/graphics, user/multimedia, user/navigation, user/network, user/office, user/science, user/system, user/utilities",
341 class Py2deb(object):
343 heavily based on technic described here :
344 http://wiki.showmedo.com/index.php?title=LinuxJensMakingDeb
347 clear=False # clear build folder after py2debianization
349 SECTIONS = SECTIONS_BY_POLICY["debian"]
351 #http://www.debian.org/doc/debian-policy/footnotes.html#f69
352 ARCHS="all i386 ia64 alpha amd64 armeb arm hppa m32r m68k mips mipsel powerpc ppc64 s390 s390x sh3 sh3eb sh4 sh4eb sparc darwin-i386 darwin-ia64 darwin-alpha darwin-amd64 darwin-armeb darwin-arm darwin-hppa darwin-m32r darwin-m68k darwin-mips darwin-mipsel darwin-powerpc darwin-ppc64 darwin-s390 darwin-s390x darwin-sh3 darwin-sh3eb darwin-sh4 darwin-sh4eb darwin-sparc freebsd-i386 freebsd-ia64 freebsd-alpha freebsd-amd64 freebsd-armeb freebsd-arm freebsd-hppa freebsd-m32r freebsd-m68k freebsd-mips freebsd-mipsel freebsd-powerpc freebsd-ppc64 freebsd-s390 freebsd-s390x freebsd-sh3 freebsd-sh3eb freebsd-sh4 freebsd-sh4eb freebsd-sparc kfreebsd-i386 kfreebsd-ia64 kfreebsd-alpha kfreebsd-amd64 kfreebsd-armeb kfreebsd-arm kfreebsd-hppa kfreebsd-m32r kfreebsd-m68k kfreebsd-mips kfreebsd-mipsel kfreebsd-powerpc kfreebsd-ppc64 kfreebsd-s390 kfreebsd-s390x kfreebsd-sh3 kfreebsd-sh3eb kfreebsd-sh4 kfreebsd-sh4eb kfreebsd-sparc knetbsd-i386 knetbsd-ia64 knetbsd-alpha knetbsd-amd64 knetbsd-armeb knetbsd-arm knetbsd-hppa knetbsd-m32r knetbsd-m68k knetbsd-mips knetbsd-mipsel knetbsd-powerpc knetbsd-ppc64 knetbsd-s390 knetbsd-s390x knetbsd-sh3 knetbsd-sh3eb knetbsd-sh4 knetbsd-sh4eb knetbsd-sparc netbsd-i386 netbsd-ia64 netbsd-alpha netbsd-amd64 netbsd-armeb netbsd-arm netbsd-hppa netbsd-m32r netbsd-m68k netbsd-mips netbsd-mipsel netbsd-powerpc netbsd-ppc64 netbsd-s390 netbsd-s390x netbsd-sh3 netbsd-sh3eb netbsd-sh4 netbsd-sh4eb netbsd-sparc openbsd-i386 openbsd-ia64 openbsd-alpha openbsd-amd64 openbsd-armeb openbsd-arm openbsd-hppa openbsd-m32r openbsd-m68k openbsd-mips openbsd-mipsel openbsd-powerpc openbsd-ppc64 openbsd-s390 openbsd-s390x openbsd-sh3 openbsd-sh3eb openbsd-sh4 openbsd-sh4eb openbsd-sparc hurd-i386 hurd-ia64 hurd-alpha hurd-amd64 hurd-armeb hurd-arm hurd-hppa hurd-m32r hurd-m68k hurd-mips hurd-mipsel hurd-powerpc hurd-ppc64 hurd-s390 hurd-s390x hurd-sh3 hurd-sh3eb hurd-sh4 hurd-sh4eb hurd-sparc".split(" ")
354 # license terms taken from dh_make
355 LICENSES=["gpl", "lgpl", "bsd", "artistic"]
357 def __setitem__(self, path, files):
359 if not type(files)==list:
360 raise Py2debException("value of key path '%s' is not a list"%path)
362 raise Py2debException("value of key path '%s' should'nt be empty"%path)
363 if not path.startswith("/"):
364 raise Py2debException("key path '%s' malformed (don't start with '/')"%path)
365 if path.endswith("/"):
366 raise Py2debException("key path '%s' malformed (shouldn't ends with '/')"%path)
372 raise Py2debException("file '%s' contains '..', please avoid that!"%file)
376 if file.count("|")!=1:
377 raise Py2debException("file '%s' is incorrect (more than one pipe)"%file)
379 file, nfile = file.split("|")
381 nfile=file # same localisation
383 if os.path.isdir(file):
384 raise Py2debException("file '%s' is a folder, and py2deb refuse folders !"%file)
386 if not os.path.isfile(file):
387 raise Py2debException("file '%s' doesn't exist"%file)
389 if file.startswith("/"): # if an absolute file is defined
390 if file==nfile: # and not renamed (pipe trick)
391 nfile=os.path.basename(file) # it's simply copied to 'path'
393 nfiles.append((file, nfile))
395 nfiles.sort(lambda a, b: cmp(a[1], b[1])) #sort according new name (nfile)
397 self.__files[path]=nfiles
399 def __delitem__(self, k):
404 description="no description",
421 author = ("USERNAME" in os.environ) and os.environ["USERNAME"] or None
423 author = ("USER" in os.environ) and os.environ["USER"] or "unknown"
426 mail = author+"@"+socket.gethostname()
429 self.description = description
430 self.upgradeDescription = ""
431 self.license = license
432 self.depends = depends
434 self.section = section
441 self.preinstall = preinstall
442 self.postinstall = postinstall
443 self.preremove = preremove
444 self.postremove = postremove
450 license = self.license
451 description = self.description
452 depends = self.depends
453 recommends = self.recommends
454 section = self.section
460 preinstall = self.preinstall
461 postinstall = self.postinstall
462 preremove = self.preremove
463 postremove = self.postremove
465 paths=self.__files.keys()
469 for file, nfile in self.__files[path]:
470 #~ rfile=os.path.normpath(os.path.join(path, nfile))
471 rfile=os.path.join(path, nfile)
475 files.append(rfile + " (%s)"%file)
478 files = "\n".join(files)
481 lscripts = [ preinstall and "preinst",
482 postinstall and "postinst",
483 preremove and "prerm",
484 postremove and "postrm",
486 scripts = lscripts and ", ".join([i for i in lscripts if i]) or "None"
488 ----------------------------------------------------------------------
490 ----------------------------------------------------------------------
491 LICENSE : %(license)s
495 ----------------------------------------------------------------------
496 DEPENDS : %(depends)s
497 RECOMMENDS : %(recommends)s
499 SECTION : %(section)s
500 ----------------------------------------------------------------------
503 ----------------------------------------------------------------------
504 SCRIPTS : %(scripts)s
505 ----------------------------------------------------------------------
510 def generate(self, version, changelog="", rpm=False, src=False, build=True, tar=False, changes=False, dsc=False):
511 """ generate a deb of version 'version', with or without 'changelog', with or without a rpm
512 (in the current folder)
513 return a list of generated files
515 if not sum([len(i) for i in self.__files.values()])>0:
516 raise Py2debException("no files are defined")
519 changelog="* no changelog"
522 description = self.description
523 license = self.license
524 depends = self.depends
525 recommends = self.recommends
526 section = self.section
532 preinstall = self.preinstall
533 postinstall = self.postinstall
534 preremove = self.preremove
535 postremove = self.postremove
537 if section not in Py2deb.SECTIONS:
538 raise Py2debException("section '%s' is unknown (%s)" % (section, str(Py2deb.SECTIONS)))
540 if arch not in Py2deb.ARCHS:
541 raise Py2debException("arch '%s' is unknown (%s)"% (arch, str(Py2deb.ARCHS)))
543 if license not in Py2deb.LICENSES:
544 raise Py2debException("License '%s' is unknown (%s)" % (license, str(Py2deb.LICENSES)))
546 # create dates (buildDate, buildDateYear)
548 buildDate=d.strftime("%a, %d %b %Y %H:%M:%S +0000")
549 buildDateYear=str(d.year)
552 #clean description (add a space before each next lines)
553 description=description.replace("\r", "").strip()
554 description = "\n ".join(description.split("\n"))
556 #clean changelog (add 2 spaces before each next lines)
557 changelog=changelog.replace("\r", "").strip()
558 changelog = "\n ".join(changelog.split("\n"))
561 TEMP = ".py2deb_build_folder"
562 DEST = os.path.join(TEMP, name)
563 DEBIAN = os.path.join(DEST, "debian")
565 # let's start the process
576 for ofile, nfile in files[path]:
577 if os.path.isfile(ofile):
580 if ofile.startswith("/"): # if absolute path
581 # we need to change dest
582 dest=os.path.join(DEST, nfile)
584 dest=os.path.join(DEST, ofile)
586 # copy file to be packaged
587 destDir = os.path.dirname(dest)
588 if not os.path.isdir(destDir):
591 shutil.copy2(ofile, dest)
593 ndir = os.path.join(path, os.path.dirname(nfile))
594 nname = os.path.basename(nfile)
596 # make a line RULES to be sure the destination folder is created
597 # and one for copying the file
598 fpath = "/".join(["$(CURDIR)", "debian", name+ndir])
599 rules.append('mkdir -p "%s"' % fpath)
600 rules.append('cp -a "%s" "%s"' % (ofile, os.path.join(fpath, nname)))
606 raise Py2debException("unknown file '' "%ofile) # shouldn't be raised (because controlled before)
609 rules= "\n\t".join(rules) + "\n"
612 dirs= [i[1:] for i in set(dirs)]
615 #==========================================================================
617 #==========================================================================
618 open(os.path.join(DEBIAN, "dirs"), "w").write("\n".join(dirs))
620 #==========================================================================
621 # CREATE debian/changelog
622 #==========================================================================
623 clog="""%(name)s (%(version)s) stable; urgency=low
627 -- %(author)s <%(mail)s> %(buildDate)s
630 open(os.path.join(DEBIAN, "changelog"), "w").write(clog)
632 #==========================================================================
633 #Create pre/post install/remove
634 #==========================================================================
635 def mkscript(name, dest):
636 if name and name.strip()!="":
637 if os.path.isfile(name): # it's a file
638 content = file(name).read()
639 else: # it's a script
641 open(os.path.join(DEBIAN, dest), "w").write(content)
643 mkscript(preinstall, "preinst")
644 mkscript(postinstall, "postinst")
645 mkscript(preremove, "prerm")
646 mkscript(postremove, "postrm")
649 #==========================================================================
650 # CREATE debian/compat
651 #==========================================================================
652 open(os.path.join(DEBIAN, "compat"), "w").write("5\n")
654 #==========================================================================
655 # CREATE debian/control
656 #==========================================================================
657 txt="""Source: %(name)s
660 Maintainer: %(author)s <%(mail)s>
661 Build-Depends: debhelper (>= 5)
662 Standards-Version: 3.7.2
665 Architecture: %(arch)s
667 Recommends: %(recommends)s
668 Description: %(description)s""" % locals()
669 if self.upgradeDescription:
670 upgradeDescription = "XB-Maemo-Upgrade-Description: %s" % self.upgradeDescription.strip()
671 txt = "%s\n%s" % (txt, "\n ".join(upgradeDescription.split("\n")))
673 f = open(self.icon, "rb")
678 uueIcon = base64.b64encode(rawIcon)
680 for i, c in enumerate(uueIcon):
682 uueIconLines.append("")
683 uueIconLines[-1] += c
684 uueIconLines[0:0] = ("XB-Maemo-Icon-26:", )
685 txt = "\n".join((txt, "\n ".join(uueIconLines), ""))
686 open(os.path.join(DEBIAN, "control"), "w").write(txt)
688 #==========================================================================
689 # CREATE debian/copyright
690 #==========================================================================
693 This package is free software; you can redistribute it and/or modify
694 it under the terms of the GNU General Public License as published by
695 the Free Software Foundation; either version 2 of the License, or
696 (at your option) any later version.
698 This package is distributed in the hope that it will be useful,
699 but WITHOUT ANY WARRANTY; without even the implied warranty of
700 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
701 GNU General Public License for more details.
703 You should have received a copy of the GNU General Public License
704 along with this package; if not, write to the Free Software
705 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
707 On Debian systems, the complete text of the GNU General
708 Public License can be found in `/usr/share/common-licenses/GPL'.
711 This package is free software; you can redistribute it and/or
712 modify it under the terms of the GNU Lesser General Public
713 License as published by the Free Software Foundation; either
714 version 2 of the License, or (at your option) any later version.
716 This package is distributed in the hope that it will be useful,
717 but WITHOUT ANY WARRANTY; without even the implied warranty of
718 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
719 Lesser General Public License for more details.
721 You should have received a copy of the GNU Lesser General Public
722 License along with this package; if not, write to the Free Software
723 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
725 On Debian systems, the complete text of the GNU Lesser General
726 Public License can be found in `/usr/share/common-licenses/LGPL'.
729 Redistribution and use in source and binary forms, with or without
730 modification, are permitted under the terms of the BSD License.
732 THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
733 ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
734 IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
735 ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
736 FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
737 DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
738 OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
739 HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
740 LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
741 OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
744 On Debian systems, the complete text of the BSD License can be
745 found in `/usr/share/common-licenses/BSD'.
748 This program is free software; you can redistribute it and/or modify it
749 under the terms of the "Artistic License" which comes with Debian.
751 THIS PACKAGE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR IMPLIED
752 WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES
753 OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
755 On Debian systems, the complete text of the Artistic License
756 can be found in `/usr/share/common-licenses/Artistic'.
759 txtLicense = copy[license]
761 txt="""This package was py2debianized(%(pv)s) by %(author)s <%(mail)s> on
764 It was downloaded from %(url)s
766 Upstream Author: %(author)s <%(mail)s>
768 Copyright: %(buildDateYear)s by %(author)s
774 The Debian packaging is (C) %(buildDateYear)s, %(author)s <%(mail)s> and
775 is licensed under the GPL, see above.
778 # Please also look if there are files or directories which have a
779 # different copyright/license attached and list them here.
781 open(os.path.join(DEBIAN, "copyright"), "w").write(txt)
783 #==========================================================================
784 # CREATE debian/rules
785 #==========================================================================
786 txt="""#!/usr/bin/make -f
788 # Sample debian/rules that uses debhelper.
789 # This file was originally written by Joey Hess and Craig Small.
790 # As a special exception, when this file is copied by dh-make into a
791 # dh-make output file, you may use that output file without restriction.
792 # This special exception was added by Craig Small in version 0.37 of dh-make.
794 # Uncomment this to turn on verbose mode.
802 ifneq (,$(findstring noopt,$(DEB_BUILD_OPTIONS)))
808 configure: configure-stamp
811 # Add here commands to configure the package.
813 touch configure-stamp
818 build-stamp: configure-stamp
825 rm -f build-stamp configure-stamp
834 # ======================================================
835 #$(MAKE) DESTDIR="$(CURDIR)/debian/%(name)s" install
836 mkdir -p "$(CURDIR)/debian/%(name)s"
839 # ======================================================
841 # Build architecture-independent files here.
842 binary-indep: build install
843 # We have nothing to do by default.
845 # Build architecture-dependent files here.
846 binary-arch: build install
849 dh_installchangelogs debian/changelog
855 # dh_installlogrotate
876 binary: binary-indep binary-arch
877 .PHONY: build clean binary-indep binary-arch binary install configure
879 open(os.path.join(DEBIAN, "rules"), "w").write(txt)
880 os.chmod(os.path.join(DEBIAN, "rules"), 0755)
882 ###########################################################################
883 ###########################################################################
884 ###########################################################################
889 #http://www.debian.org/doc/manuals/maint-guide/ch-build.fr.html
890 ret=os.system('cd "%(DEST)s"; dpkg-buildpackage -tc -rfakeroot -us -uc' % locals())
892 raise Py2debException("buildpackage failed (see output)")
894 l=glob("%(TEMP)s/%(name)s*.deb"%locals())
896 raise Py2debException("didn't find builded deb")
899 deb = os.path.basename(tdeb)
900 shutil.move(tdeb, deb)
902 generatedFiles = [deb, ]
905 rpmFilename = deb2rpm(deb)
906 generatedFiles.append(rpmFilename)
909 tarFilename = py2src(TEMP, name)
910 generatedFiles.append(tarFilename)
913 tarFilename = py2tar(DEST, TEMP, name, version)
914 generatedFiles.append(tarFilename)
917 dscFilename = py2dsc(TEMP, name, version, depends, author, mail, arch)
918 generatedFiles.append(dscFilename)
921 changesFilenames = py2changes(locals())
922 generatedFiles.extend(changesFilenames)
924 return generatedFiles
926 #~ except Exception,m:
927 #~ raise Py2debException("build error :"+str(m))
934 if __name__ == "__main__":
936 os.chdir(os.path.dirname(sys.argv[0]))
940 p=Py2deb("python-py2deb")
941 p.description="Generate simple deb(/rpm/tgz) from python (2.4, 2.5 and 2.6)"
942 p.url = "http://www.manatlan.com/page/py2deb"
945 p.depends = "dpkg-dev, fakeroot, alien, python"
947 p["/usr/lib/python2.6/dist-packages"] = ["py2deb.py", ]
948 p["/usr/lib/python2.5/site-packages"] = ["py2deb.py", ]
949 p["/usr/lib/python2.4/site-packages"] = ["py2deb.py", ]
950 #~ p.postinstall = "s.py"
951 #~ p.preinstall = "s.py"
952 #~ p.postremove = "s.py"
953 #~ p.preremove = "s.py"
955 print p.generate(__version__, changelog = __doc__, src=True)