mirror of
https://git.launchpad.net/ubuntu-dev-tools
synced 2025-03-12 23:51:08 +00:00
* New scripts:
- reverse-depends: Replaces reverse-build-depends. Uses an UbuntuWire webservice for determining all reverse(-build)-dependencies for a package. (LP: #696373) - requestbackport: Files a backport request bug report, including a full testing checklist. * Don't allow boilerplate prompts through in submittodebian and requestsync (LP: #887336)
This commit is contained in:
commit
cefbe3ccc2
7
debian/NEWS
vendored
7
debian/NEWS
vendored
@ -1,3 +1,10 @@
|
|||||||
|
ubuntu-dev-tools (0.135) unstable; urgency=low
|
||||||
|
|
||||||
|
reverse-build-depends was removed from ubuntu-dev-tools. reverse-depends -b
|
||||||
|
is equivalent.
|
||||||
|
|
||||||
|
-- Stefano Rivera <stefanor@debian.org> Sat, 12 Nov 2011 13:11:21 +0200
|
||||||
|
|
||||||
ubuntu-dev-tools (0.131) unstable; urgency=low
|
ubuntu-dev-tools (0.131) unstable; urgency=low
|
||||||
|
|
||||||
get-build-deps was removed from ubuntu-dev-tools. The newer mk-build-deps in
|
get-build-deps was removed from ubuntu-dev-tools. The newer mk-build-deps in
|
||||||
|
12
debian/changelog
vendored
12
debian/changelog
vendored
@ -1,4 +1,4 @@
|
|||||||
ubuntu-dev-tools (0.136) UNRELEASED; urgency=low
|
ubuntu-dev-tools (0.135) UNRELEASED; urgency=low
|
||||||
|
|
||||||
* grab-merge: Use wget -nv rather than -q, so that we see error messages
|
* grab-merge: Use wget -nv rather than -q, so that we see error messages
|
||||||
(LP: #881967)
|
(LP: #881967)
|
||||||
@ -10,8 +10,16 @@ ubuntu-dev-tools (0.136) UNRELEASED; urgency=low
|
|||||||
- Add --eatmydata flag (LP: #888440)
|
- Add --eatmydata flag (LP: #888440)
|
||||||
* pbuilder-dist: Support using non-master mirrors. Thanks Mathieu Parent.
|
* pbuilder-dist: Support using non-master mirrors. Thanks Mathieu Parent.
|
||||||
(LP: #824285)
|
(LP: #824285)
|
||||||
|
* New scripts:
|
||||||
|
- reverse-depends: Replaces reverse-build-depends. Uses an UbuntuWire
|
||||||
|
webservice for determining all reverse(-build)-dependencies for a
|
||||||
|
package. (LP: #696373)
|
||||||
|
- requestbackport: Files a backport request bug report, including a full
|
||||||
|
testing checklist.
|
||||||
|
* Don't allow boilerplate prompts through in submittodebian and requestsync
|
||||||
|
(LP: #887336)
|
||||||
|
|
||||||
-- Stefano Rivera <stefanor@debian.org> Sat, 12 Nov 2011 23:28:05 +0200
|
-- Stefano Rivera <stefanor@debian.org> Sat, 12 Nov 2011 13:09:05 +0200
|
||||||
|
|
||||||
ubuntu-dev-tools (0.134) unstable; urgency=low
|
ubuntu-dev-tools (0.134) unstable; urgency=low
|
||||||
|
|
||||||
|
5
debian/control
vendored
5
debian/control
vendored
@ -93,9 +93,10 @@ Description: useful tools for Ubuntu developers
|
|||||||
Debian of a package.
|
Debian of a package.
|
||||||
- pull-lp-source - downloads lastest source package from Launchpad.
|
- pull-lp-source - downloads lastest source package from Launchpad.
|
||||||
- pull-revu-source - downloads the latest source package from REVU
|
- pull-revu-source - downloads the latest source package from REVU
|
||||||
|
- requestbackport - file a backporting request.
|
||||||
- requestsync - files a sync request with Debian changelog and rationale.
|
- requestsync - files a sync request with Debian changelog and rationale.
|
||||||
- reverse-build-depends - find the reverse build dependencies that a package
|
- reverse-depends - find the reverse dependencies (or build dependencies) of
|
||||||
has.
|
a package.
|
||||||
- setup-packaging-environment - assistant to get an Ubuntu installation
|
- setup-packaging-environment - assistant to get an Ubuntu installation
|
||||||
ready for packaging work.
|
ready for packaging work.
|
||||||
- sponsor-patch - Downloads a patch from a Launchpad bug, patches the source
|
- sponsor-patch - Downloads a patch from a Launchpad bug, patches the source
|
||||||
|
8
debian/copyright
vendored
8
debian/copyright
vendored
@ -47,18 +47,15 @@ Files: 404main
|
|||||||
doc/import-bug-from-debian.1
|
doc/import-bug-from-debian.1
|
||||||
doc/pbuilder-dist-simple.1
|
doc/pbuilder-dist-simple.1
|
||||||
doc/pbuilder-dist.1
|
doc/pbuilder-dist.1
|
||||||
doc/reverse-build-depends.1
|
|
||||||
doc/submittodebian.1
|
doc/submittodebian.1
|
||||||
import-bug-from-debian
|
import-bug-from-debian
|
||||||
pbuilder-dist
|
pbuilder-dist
|
||||||
pbuilder-dist-simple
|
pbuilder-dist-simple
|
||||||
reverse-build-depends
|
|
||||||
submittodebian
|
submittodebian
|
||||||
Copyright: 2007-2010, Canonical Ltd.
|
Copyright: 2007-2010, Canonical Ltd.
|
||||||
2009, James Westby <james.westby@ubuntu.com>
|
2009, James Westby <james.westby@ubuntu.com>
|
||||||
2008, Jamin W. Collins <jcollins@asgardsrealm.net>
|
2008, Jamin W. Collins <jcollins@asgardsrealm.net>
|
||||||
2008, Jordan Mantha <mantha@ubuntu.com>
|
2008, Jordan Mantha <mantha@ubuntu.com>
|
||||||
2008-2009, Patrick Schoenfeld <schoenfeld@in-medias-res.com>
|
|
||||||
2006-2007, Pete Savage <petesavage@ubuntu.com>
|
2006-2007, Pete Savage <petesavage@ubuntu.com>
|
||||||
2009, Ryan Kavanagh <ryanakca@kubuntu.org>
|
2009, Ryan Kavanagh <ryanakca@kubuntu.org>
|
||||||
2007, Siegfried-Angel Gevatter Pujals <rainct@ubuntu.com>
|
2007, Siegfried-Angel Gevatter Pujals <rainct@ubuntu.com>
|
||||||
@ -152,17 +149,22 @@ License: GPL-3+
|
|||||||
|
|
||||||
Files: doc/pull-debian-debdiff.1
|
Files: doc/pull-debian-debdiff.1
|
||||||
doc/pull-debian-source.1
|
doc/pull-debian-source.1
|
||||||
|
doc/requestbackport.1
|
||||||
|
doc/reverse-depends.1
|
||||||
doc/sponsor-patch.1
|
doc/sponsor-patch.1
|
||||||
doc/ubuntu-dev-tools.5
|
doc/ubuntu-dev-tools.5
|
||||||
doc/update-maintainer.1
|
doc/update-maintainer.1
|
||||||
pull-debian-debdiff
|
pull-debian-debdiff
|
||||||
pull-debian-source
|
pull-debian-source
|
||||||
|
requestbackport
|
||||||
|
reverse-depends
|
||||||
sponsor-patch
|
sponsor-patch
|
||||||
test-data/*
|
test-data/*
|
||||||
ubuntutools/archive.py
|
ubuntutools/archive.py
|
||||||
ubuntutools/builder.py
|
ubuntutools/builder.py
|
||||||
ubuntutools/config.py
|
ubuntutools/config.py
|
||||||
ubuntutools/question.py
|
ubuntutools/question.py
|
||||||
|
ubuntutools/rdepends.py
|
||||||
ubuntutools/sponsor_patch/*
|
ubuntutools/sponsor_patch/*
|
||||||
ubuntutools/test/*
|
ubuntutools/test/*
|
||||||
ubuntutools/update_maintainer.py
|
ubuntutools/update_maintainer.py
|
||||||
|
56
doc/requestbackport.1
Normal file
56
doc/requestbackport.1
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
.\" Copyright (C) 2011, Stefano Rivera <stefanor@ubuntu.com>
|
||||||
|
.\"
|
||||||
|
.\" Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
.\" purpose with or without fee is hereby granted, provided that the above
|
||||||
|
.\" copyright notice and this permission notice appear in all copies.
|
||||||
|
.\"
|
||||||
|
.\" THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
||||||
|
.\" REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
|
.\" AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
||||||
|
.\" INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
||||||
|
.\" LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
||||||
|
.\" OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
||||||
|
.\" PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
.TH requestbackport 1 "November 2011" ubuntu\-dev\-tools
|
||||||
|
|
||||||
|
.SH NAME
|
||||||
|
requestbackport \- File a backport request bug
|
||||||
|
|
||||||
|
.SH SYNOPSIS
|
||||||
|
.B requestbackport \fR[\fIoptions\fR] \fIpackage\fR
|
||||||
|
|
||||||
|
.SH DESCRIPTION
|
||||||
|
Determine the intermediate releases that \fIpackage\fR needs to be
|
||||||
|
backported to, list all reverse\-dependencies, and file the backporting
|
||||||
|
request.
|
||||||
|
\fBrequestbackport\fR will include a testing checklist in the bug.
|
||||||
|
|
||||||
|
.SH OPTIONS
|
||||||
|
.TP
|
||||||
|
\fB\-d\fR \fIDEST\fR, \fB\-\-destination\fR=\fIDEST\fR
|
||||||
|
Backport to \fIDEST\fR release and necessary intermediate
|
||||||
|
releases. Default: current stable release.
|
||||||
|
.TP
|
||||||
|
\fB\-s\fR \fISOURCE\fR, \fB\-\-source\fR=\fISOURCE\fR
|
||||||
|
Backport from \fISOURCE\fR release.
|
||||||
|
Default: current development release.
|
||||||
|
.TP
|
||||||
|
\fB\-l\fR \fIINSTANCE\fR, \fB\-\-lpinstance\fR=\fIINSTANCE\fR
|
||||||
|
Launchpad instance to connect to.
|
||||||
|
Default: \fBproduction\fR.
|
||||||
|
.TP
|
||||||
|
\fB\-\-no\-conf\fR
|
||||||
|
Don't read config files or environment variables
|
||||||
|
.TP
|
||||||
|
\fB\-h\fR, \fB\-\-help\fR
|
||||||
|
Display a help message and exit.
|
||||||
|
|
||||||
|
.SH SEE ALSO
|
||||||
|
.BR backportpackage (1),
|
||||||
|
.BR reverse\-depends (1).
|
||||||
|
|
||||||
|
.SH AUTHORS
|
||||||
|
\fBreverse\-depends\fR and this manpage were written by Stefano Rivera
|
||||||
|
<stefanor@ubuntu.com>.
|
||||||
|
.PP
|
||||||
|
Both are released under the terms of the ISC License.
|
@ -1,172 +0,0 @@
|
|||||||
.\" Automatically generated by Pod::Man 2.1801 (Pod::Simple 3.05)
|
|
||||||
.\"
|
|
||||||
.\" Standard preamble:
|
|
||||||
.\" ========================================================================
|
|
||||||
.de Sp \" Vertical space (when we can't use .PP)
|
|
||||||
.if t .sp .5v
|
|
||||||
.if n .sp
|
|
||||||
..
|
|
||||||
.de Vb \" Begin verbatim text
|
|
||||||
.ft CW
|
|
||||||
.nf
|
|
||||||
.ne \\$1
|
|
||||||
..
|
|
||||||
.de Ve \" End verbatim text
|
|
||||||
.ft R
|
|
||||||
.fi
|
|
||||||
..
|
|
||||||
.\" Set up some character translations and predefined strings. \*(-- will
|
|
||||||
.\" give an unbreakable dash, \*(PI will give pi, \*(L" will give a left
|
|
||||||
.\" double quote, and \*(R" will give a right double quote. \*(C+ will
|
|
||||||
.\" give a nicer C++. Capital omega is used to do unbreakable dashes and
|
|
||||||
.\" therefore won't be available. \*(C` and \*(C' expand to `' in nroff,
|
|
||||||
.\" nothing in troff, for use with C<>.
|
|
||||||
.tr \(*W-
|
|
||||||
.ds C+ C\v'-.1v'\h'-1p'\s-2+\h'-1p'+\s0\v'.1v'\h'-1p'
|
|
||||||
.ie n \{\
|
|
||||||
. ds -- \(*W-
|
|
||||||
. ds PI pi
|
|
||||||
. if (\n(.H=4u)&(1m=24u) .ds -- \(*W\h'-12u'\(*W\h'-12u'-\" diablo 10 pitch
|
|
||||||
. if (\n(.H=4u)&(1m=20u) .ds -- \(*W\h'-12u'\(*W\h'-8u'-\" diablo 12 pitch
|
|
||||||
. ds L" ""
|
|
||||||
. ds R" ""
|
|
||||||
. ds C` ""
|
|
||||||
. ds C' ""
|
|
||||||
'br\}
|
|
||||||
.el\{\
|
|
||||||
. ds -- \|\(em\|
|
|
||||||
. ds PI \(*p
|
|
||||||
. ds L" ``
|
|
||||||
. ds R" ''
|
|
||||||
'br\}
|
|
||||||
.\"
|
|
||||||
.\" Escape single quotes in literal strings from groff's Unicode transform.
|
|
||||||
.ie \n(.g .ds Aq \(aq
|
|
||||||
.el .ds Aq '
|
|
||||||
.\"
|
|
||||||
.\" If the F register is turned on, we'll generate index entries on stderr for
|
|
||||||
.\" titles (.TH), headers (.SH), subsections (.SS), items (.Ip), and index
|
|
||||||
.\" entries marked with X<> in POD. Of course, you'll have to process the
|
|
||||||
.\" output yourself in some meaningful fashion.
|
|
||||||
.ie \nF \{\
|
|
||||||
. de IX
|
|
||||||
. tm Index:\\$1\t\\n%\t"\\$2"
|
|
||||||
..
|
|
||||||
. nr % 0
|
|
||||||
. rr F
|
|
||||||
.\}
|
|
||||||
.el \{\
|
|
||||||
. de IX
|
|
||||||
..
|
|
||||||
.\}
|
|
||||||
.\"
|
|
||||||
.\" Accent mark definitions (@(#)ms.acc 1.5 88/02/08 SMI; from UCB 4.2).
|
|
||||||
.\" Fear. Run. Save yourself. No user-serviceable parts.
|
|
||||||
. \" fudge factors for nroff and troff
|
|
||||||
.if n \{\
|
|
||||||
. ds #H 0
|
|
||||||
. ds #V .8m
|
|
||||||
. ds #F .3m
|
|
||||||
. ds #[ \f1
|
|
||||||
. ds #] \fP
|
|
||||||
.\}
|
|
||||||
.if t \{\
|
|
||||||
. ds #H ((1u-(\\\\n(.fu%2u))*.13m)
|
|
||||||
. ds #V .6m
|
|
||||||
. ds #F 0
|
|
||||||
. ds #[ \&
|
|
||||||
. ds #] \&
|
|
||||||
.\}
|
|
||||||
. \" simple accents for nroff and troff
|
|
||||||
.if n \{\
|
|
||||||
. ds ' \&
|
|
||||||
. ds ` \&
|
|
||||||
. ds ^ \&
|
|
||||||
. ds , \&
|
|
||||||
. ds ~ ~
|
|
||||||
. ds /
|
|
||||||
.\}
|
|
||||||
.if t \{\
|
|
||||||
. ds ' \\k:\h'-(\\n(.wu*8/10-\*(#H)'\'\h"|\\n:u"
|
|
||||||
. ds ` \\k:\h'-(\\n(.wu*8/10-\*(#H)'\`\h'|\\n:u'
|
|
||||||
. ds ^ \\k:\h'-(\\n(.wu*10/11-\*(#H)'^\h'|\\n:u'
|
|
||||||
. ds , \\k:\h'-(\\n(.wu*8/10)',\h'|\\n:u'
|
|
||||||
. ds ~ \\k:\h'-(\\n(.wu-\*(#H-.1m)'~\h'|\\n:u'
|
|
||||||
. ds / \\k:\h'-(\\n(.wu*8/10-\*(#H)'\z\(sl\h'|\\n:u'
|
|
||||||
.\}
|
|
||||||
. \" troff and (daisy-wheel) nroff accents
|
|
||||||
.ds : \\k:\h'-(\\n(.wu*8/10-\*(#H+.1m+\*(#F)'\v'-\*(#V'\z.\h'.2m+\*(#F'.\h'|\\n:u'\v'\*(#V'
|
|
||||||
.ds 8 \h'\*(#H'\(*b\h'-\*(#H'
|
|
||||||
.ds o \\k:\h'-(\\n(.wu+\w'\(de'u-\*(#H)/2u'\v'-.3n'\*(#[\z\(de\v'.3n'\h'|\\n:u'\*(#]
|
|
||||||
.ds d- \h'\*(#H'\(pd\h'-\w'~'u'\v'-.25m'\f2\(hy\fP\v'.25m'\h'-\*(#H'
|
|
||||||
.ds D- D\\k:\h'-\w'D'u'\v'-.11m'\z\(hy\v'.11m'\h'|\\n:u'
|
|
||||||
.ds th \*(#[\v'.3m'\s+1I\s-1\v'-.3m'\h'-(\w'I'u*2/3)'\s-1o\s+1\*(#]
|
|
||||||
.ds Th \*(#[\s+2I\s-2\h'-\w'I'u*3/5'\v'-.3m'o\v'.3m'\*(#]
|
|
||||||
.ds ae a\h'-(\w'a'u*4/10)'e
|
|
||||||
.ds Ae A\h'-(\w'A'u*4/10)'E
|
|
||||||
. \" corrections for vroff
|
|
||||||
.if v .ds ~ \\k:\h'-(\\n(.wu*9/10-\*(#H)'\s-2\u~\d\s+2\h'|\\n:u'
|
|
||||||
.if v .ds ^ \\k:\h'-(\\n(.wu*10/11-\*(#H)'\v'-.4m'^\v'.4m'\h'|\\n:u'
|
|
||||||
. \" for low resolution devices (crt and lpr)
|
|
||||||
.if \n(.H>23 .if \n(.V>19 \
|
|
||||||
\{\
|
|
||||||
. ds : e
|
|
||||||
. ds 8 ss
|
|
||||||
. ds o a
|
|
||||||
. ds d- d\h'-1'\(ga
|
|
||||||
. ds D- D\h'-1'\(hy
|
|
||||||
. ds th \o'bp'
|
|
||||||
. ds Th \o'LP'
|
|
||||||
. ds ae ae
|
|
||||||
. ds Ae AE
|
|
||||||
.\}
|
|
||||||
.rm #[ #] #H #V #F C
|
|
||||||
.\" ========================================================================
|
|
||||||
.\"
|
|
||||||
.IX Title "BUILD-RDEPS 1"
|
|
||||||
.TH BUILD-RDEPS 1 "2008-08-14" "Debian Utilities" " "
|
|
||||||
.\" For nroff, turn off justification. Always turn off hyphenation; it makes
|
|
||||||
.\" way too many mistakes in technical documents.
|
|
||||||
.if n .ad l
|
|
||||||
.nh
|
|
||||||
.SH "NAME"
|
|
||||||
build\-rdeps \- find packages that depend on a specific package to build (reverse build depends)
|
|
||||||
.SH "SYNOPSIS"
|
|
||||||
.IX Header "SYNOPSIS"
|
|
||||||
\&\fBubuild-rdeps\fR \fIpackage\fR
|
|
||||||
.SH "DESCRIPTION"
|
|
||||||
.IX Header "DESCRIPTION"
|
|
||||||
\&\fBubuild-rdeps\fR searches for all packages that build-depend on the specified package.
|
|
||||||
.SH "OPTIONS"
|
|
||||||
.IX Header "OPTIONS"
|
|
||||||
.IP "\fB\-u\fR \fB\-\-update\fR" 4
|
|
||||||
.IX Item "-u --update"
|
|
||||||
Run apt-get update before searching for build-depends.
|
|
||||||
.IP "\fB\-s\fR \fB\-\-sudo\fR" 4
|
|
||||||
.IX Item "-s --sudo"
|
|
||||||
Use sudo when running apt-get update. Has no effect if \-u is omitted.
|
|
||||||
.IP "\fB\-\-distribution\fR" 4
|
|
||||||
.IX Item "--distribution"
|
|
||||||
Select another distribution, which is searched for build-depends.
|
|
||||||
.IP "\fB\-m\fR \fB\-\-print\-maintainer\fR" 4
|
|
||||||
.IX Item "-m --print-maintainer"
|
|
||||||
Print the value of the maintainer field for each package.
|
|
||||||
.IP "\fB\-d\fR \fB\-\-debug\fR" 4
|
|
||||||
.IX Item "-d --debug"
|
|
||||||
Run the debug mode
|
|
||||||
.IP "\fB\-\-help\fR" 4
|
|
||||||
.IX Item "--help"
|
|
||||||
Show the usage information.
|
|
||||||
.IP "\fB\-\-version\fR" 4
|
|
||||||
.IX Item "--version"
|
|
||||||
Show the version information.
|
|
||||||
.SH "LICENSE"
|
|
||||||
.IX Header "LICENSE"
|
|
||||||
This code is copyright by Patrick Schoenfeld
|
|
||||||
<schoenfeld@in\-medias\-res.com>, all rights reserved.
|
|
||||||
This program comes with \s-1ABSOLUTELEY\s0 \s-1NO\s0 \s-1WARRANTY\s0.
|
|
||||||
You are free to redistribute this code under the terms of the
|
|
||||||
\&\s-1GNU\s0 General Public License, version 2 or later.
|
|
||||||
.SH "AUTHOR"
|
|
||||||
.IX Header "AUTHOR"
|
|
||||||
Patrick Schoenfeld <schoenfeld@in\-medias\-res.com>
|
|
81
doc/reverse-depends.1
Normal file
81
doc/reverse-depends.1
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
.\" Copyright (C) 2011, Stefano Rivera <stefanor@ubuntu.com>
|
||||||
|
.\"
|
||||||
|
.\" Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
.\" purpose with or without fee is hereby granted, provided that the above
|
||||||
|
.\" copyright notice and this permission notice appear in all copies.
|
||||||
|
.\"
|
||||||
|
.\" THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
||||||
|
.\" REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
|
.\" AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
||||||
|
.\" INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
||||||
|
.\" LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
||||||
|
.\" OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
||||||
|
.\" PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
.TH reverse\-depends 1 "November 2011" ubuntu\-dev\-tools
|
||||||
|
|
||||||
|
.SH NAME
|
||||||
|
reverse\-depends \- List the reverse\-dependencies (or
|
||||||
|
build\-dependencies) of a package
|
||||||
|
|
||||||
|
.SH SYNOPSIS
|
||||||
|
.B reverse\-depends \fR[\fIoptions\fR] \fIpackage
|
||||||
|
|
||||||
|
.SH DESCRIPTION
|
||||||
|
List reverse\-dependencies (or build\-dependencies) of \fIpackage\fR.
|
||||||
|
If the package name is prefixed with \fBsrc:\fR then the
|
||||||
|
reverse\-dependencies of all the binary packages that the specified
|
||||||
|
source package builds will be listed.
|
||||||
|
|
||||||
|
.SH OPTIONS
|
||||||
|
.TP
|
||||||
|
\fB\-r\fR \fIRELEASE\fR, \fB\-\-release\fR=\fIRELEASE\fR
|
||||||
|
Query dependencies in \fIRELEASE\fR.
|
||||||
|
Default: current development release.
|
||||||
|
.TP
|
||||||
|
\fB\-R\fR, \fB\-\-without\-recommends\fR
|
||||||
|
Only consider Depends relationships, not Recommends.
|
||||||
|
.TP
|
||||||
|
\fB\-s\fR, \fB\-\-with\-suggests\fR
|
||||||
|
Also consider Suggests relationships.
|
||||||
|
.TP
|
||||||
|
\fB\-b\fR, \fB\-\-build\-depends\fR
|
||||||
|
Query build dependencies.
|
||||||
|
Synonym for \fB\-\-arch\fR=\fIsource\fR.
|
||||||
|
.TP
|
||||||
|
\fB\-a\fR \fIARCH\fR, \fB\-\-arch\fR=\fIARCH\fR
|
||||||
|
Query dependencies in \fIARCH\fR.
|
||||||
|
Besides valid architecture names, the special values \fBany\fR and
|
||||||
|
\fBsource\fR may be used.
|
||||||
|
\fBany\fR displays all reverse dependencies, the union across all
|
||||||
|
architecture.
|
||||||
|
\fBsource\fR displays build dependencies.
|
||||||
|
Default: \fBany\fR.
|
||||||
|
.TP
|
||||||
|
\fB\-c\fR \fICOMPONENT\fR, \fB\-\-component\fR=\fICOMPONENT\fR
|
||||||
|
Only consider reverse\-dependencies in \fICOMPONENT\fR. Can
|
||||||
|
be specified multiple times.
|
||||||
|
Default: all components.
|
||||||
|
.TP
|
||||||
|
\fB\-l\fR, \fB\-\-list\fR
|
||||||
|
Display a simple, machine\-readable list.
|
||||||
|
.TP
|
||||||
|
\fB\-u\fR \fIURL\fR, \fB\-\-service\-url\fR=\fIURL\fR
|
||||||
|
Reverse Dependencies web\-service \fIURL\fR.
|
||||||
|
Default: UbuntuWire's service at
|
||||||
|
\fBhttp://qa.ubuntuwire.org/rdepends/\fR.
|
||||||
|
.TP
|
||||||
|
\fB\-h\fR, \fB\-\-help\fR
|
||||||
|
Display a help message and exit
|
||||||
|
|
||||||
|
.SH EXAMPLES
|
||||||
|
All reverse dependencies of source package bash:
|
||||||
|
.IP
|
||||||
|
.nf
|
||||||
|
.B reverse\-depends src:bash
|
||||||
|
.fi
|
||||||
|
|
||||||
|
.SH AUTHORS
|
||||||
|
\fBreverse\-depends\fR and this manpage were written by Stefano Rivera
|
||||||
|
<stefanor@ubuntu.com>.
|
||||||
|
.PP
|
||||||
|
Both are released under the terms of the ISC License.
|
@ -38,6 +38,8 @@ from distro_info import DebianDistroInfo
|
|||||||
import ubuntutools.misc
|
import ubuntutools.misc
|
||||||
from ubuntutools.config import UDTConfig
|
from ubuntutools.config import UDTConfig
|
||||||
from ubuntutools import subprocess
|
from ubuntutools import subprocess
|
||||||
|
from ubuntutools.question import YesNoQuestion
|
||||||
|
|
||||||
|
|
||||||
class PbuilderDist:
|
class PbuilderDist:
|
||||||
def __init__(self, builder):
|
def __init__(self, builder):
|
||||||
@ -136,9 +138,10 @@ class PbuilderDist:
|
|||||||
# Debian experimental doesn't have a debootstrap file but
|
# Debian experimental doesn't have a debootstrap file but
|
||||||
# should work nevertheless.
|
# should work nevertheless.
|
||||||
if distro not in self._debian_distros:
|
if distro not in self._debian_distros:
|
||||||
answer = ask(('Warning: Unknown distribution "%s". Do you '
|
answer = YesNoQuestion().ask(
|
||||||
'want to continue [y/N]? ') % distro)
|
'Warning: Unknown distribution "%s". '
|
||||||
if answer not in ('y', 'Y'):
|
'Do you want to continue' % distro, 'no')
|
||||||
|
if answer == 'yes':
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
else:
|
else:
|
||||||
Logger.error('Please install package "debootstrap".')
|
Logger.error('Please install package "debootstrap".')
|
||||||
@ -307,22 +310,6 @@ class PbuilderDist:
|
|||||||
self.builder,
|
self.builder,
|
||||||
] + arguments
|
] + arguments
|
||||||
|
|
||||||
|
|
||||||
def ask(question):
|
|
||||||
""" ask(question) -> string
|
|
||||||
|
|
||||||
Ask the given question and return the answer. Also catch
|
|
||||||
KeyboardInterrupt (Ctrl+C) and EOFError (Ctrl+D) exceptions and
|
|
||||||
immediately return None if one of those is found.
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
answer = raw_input(question)
|
|
||||||
except (KeyboardInterrupt, EOFError):
|
|
||||||
print
|
|
||||||
answer = None
|
|
||||||
|
|
||||||
return answer
|
|
||||||
|
|
||||||
def show_help(exit_code = 0):
|
def show_help(exit_code = 0):
|
||||||
""" help() -> None
|
""" help() -> None
|
||||||
|
|
||||||
|
258
requestbackport
Executable file
258
requestbackport
Executable file
@ -0,0 +1,258 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
#
|
||||||
|
# Copyright (C) 2011, Stefano Rivera <stefanor@ubuntu.com>
|
||||||
|
#
|
||||||
|
# Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
# purpose with or without fee is hereby granted, provided that the above
|
||||||
|
# copyright notice and this permission notice appear in all copies.
|
||||||
|
#
|
||||||
|
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||||
|
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||||
|
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||||
|
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
|
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
|
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
|
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
|
||||||
|
from collections import defaultdict
|
||||||
|
import optparse
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import apt
|
||||||
|
from devscripts.logger import Logger
|
||||||
|
from distro_info import UbuntuDistroInfo
|
||||||
|
|
||||||
|
from ubuntutools.lp.lpapicache import Launchpad, Distribution
|
||||||
|
from ubuntutools.lp.udtexceptions import PackageNotFoundException
|
||||||
|
from ubuntutools.config import UDTConfig
|
||||||
|
from ubuntutools.rdepends import query_rdepends, RDependsException
|
||||||
|
from ubuntutools.question import YesNoQuestion, EditBugReport
|
||||||
|
|
||||||
|
|
||||||
|
class DestinationException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def determine_destinations(source, destination):
|
||||||
|
ubuntu_info = UbuntuDistroInfo()
|
||||||
|
if destination is None:
|
||||||
|
destination = ubuntu_info.stable()
|
||||||
|
|
||||||
|
if source not in ubuntu_info.all:
|
||||||
|
raise DestinationException("Source release %s does not exist" % source)
|
||||||
|
if destination not in ubuntu_info.all:
|
||||||
|
raise DestinationException("Destination release %s does not exist"
|
||||||
|
% destination)
|
||||||
|
if destination not in ubuntu_info.supported():
|
||||||
|
raise DestinationException("Destination release %s is not supported"
|
||||||
|
% destination)
|
||||||
|
|
||||||
|
found = False
|
||||||
|
destinations = []
|
||||||
|
support_gap = False
|
||||||
|
for release in ubuntu_info.all:
|
||||||
|
if release == destination:
|
||||||
|
found = True
|
||||||
|
if release == source:
|
||||||
|
break
|
||||||
|
if found:
|
||||||
|
if support_gap:
|
||||||
|
if ubuntu_info.is_lts(release):
|
||||||
|
support_gap = False
|
||||||
|
else:
|
||||||
|
continue
|
||||||
|
if release not in ubuntu_info.supported():
|
||||||
|
support_gap = True
|
||||||
|
continue
|
||||||
|
destinations.append(release)
|
||||||
|
|
||||||
|
assert found
|
||||||
|
assert len(destinations) > 0
|
||||||
|
|
||||||
|
return destinations
|
||||||
|
|
||||||
|
|
||||||
|
def find_rdepends(package, releases, published_binaries):
|
||||||
|
intermediate = defaultdict(lambda: defaultdict(list))
|
||||||
|
|
||||||
|
# We want to display every pubilshed binary, even if it has no rdepends
|
||||||
|
for binpkg in published_binaries:
|
||||||
|
intermediate[binpkg]
|
||||||
|
|
||||||
|
for arch in ('any', 'source'):
|
||||||
|
for release in releases:
|
||||||
|
for binpkg in published_binaries:
|
||||||
|
try:
|
||||||
|
raw_rdeps = query_rdepends(binpkg, release, arch)
|
||||||
|
except RDependsException:
|
||||||
|
# Not published? TODO: Check
|
||||||
|
continue
|
||||||
|
for relationship, rdeps in raw_rdeps.iteritems():
|
||||||
|
for rdep in rdeps:
|
||||||
|
if rdep['Package'] in published_binaries:
|
||||||
|
continue
|
||||||
|
intermediate[binpkg][rdep['Package']] \
|
||||||
|
.append((release, relationship))
|
||||||
|
|
||||||
|
output = []
|
||||||
|
for binpkg, rdeps in intermediate.iteritems():
|
||||||
|
output += ['', binpkg, '-' * len(binpkg)]
|
||||||
|
for pkg, appearences in rdeps.iteritems():
|
||||||
|
output += ['* %s' % pkg]
|
||||||
|
for release, relationship in appearences:
|
||||||
|
output += [' [ ] %s (%s)' % (release, relationship)]
|
||||||
|
|
||||||
|
found_any = sum(len(rdeps) for rdeps in intermediate.itervalues())
|
||||||
|
if found_any:
|
||||||
|
output = [
|
||||||
|
"Reverse dependencies:",
|
||||||
|
"=====================",
|
||||||
|
"The following reverse-dependencies need to be tested against the "
|
||||||
|
"new version of %(package)s. "
|
||||||
|
"For reverse-build-dependencies (-Indep), please test that the "
|
||||||
|
"package still builds against the new %(package)s. "
|
||||||
|
"For reverse-dependencies, please test that the version of the "
|
||||||
|
"package currently in the release still works with the new "
|
||||||
|
"%(package)s installed. "
|
||||||
|
"Reverse- Recommends, Suggests, and Enhances don't need to be "
|
||||||
|
"tested, and are listed for completeness-sake."
|
||||||
|
] + output
|
||||||
|
else:
|
||||||
|
output = ["No reverse dependencies"]
|
||||||
|
|
||||||
|
return output
|
||||||
|
|
||||||
|
|
||||||
|
def locate_package(package, distribution):
|
||||||
|
archive = Distribution('ubuntu').getArchive()
|
||||||
|
for pass_ in ('source', 'binary'):
|
||||||
|
try:
|
||||||
|
package_spph = archive.getSourcePackage(package, distribution)
|
||||||
|
return package_spph
|
||||||
|
except PackageNotFoundException, e:
|
||||||
|
if pass_ == 'binary':
|
||||||
|
Logger.error(str(e))
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
try:
|
||||||
|
apt_pkg = apt.Cache()[package]
|
||||||
|
except KeyError:
|
||||||
|
continue
|
||||||
|
package = apt_pkg.candidate.source_name
|
||||||
|
Logger.normal("Binary package specified, considering its source "
|
||||||
|
"package instead: %s", package)
|
||||||
|
|
||||||
|
|
||||||
|
def request_backport(package_spph, source, destinations):
|
||||||
|
|
||||||
|
published_binaries = set()
|
||||||
|
for bpph in package_spph._lpobject.getPublishedBinaries():
|
||||||
|
published_binaries.add(bpph.binary_package_name)
|
||||||
|
|
||||||
|
testing = []
|
||||||
|
testing += ["You can test-build the backport in your PPA with "
|
||||||
|
"backportpackage:"]
|
||||||
|
testing += ["$ backportpackage -u ppa:<lp username>/<ppa name> "
|
||||||
|
"-s %s -d %s %s"
|
||||||
|
% (source, dest, package_spph.getPackageName())
|
||||||
|
for dest in destinations]
|
||||||
|
testing += [""]
|
||||||
|
for dest in destinations:
|
||||||
|
testing += ['* %s:' % dest]
|
||||||
|
testing += ["[ ] Package builds without modification"]
|
||||||
|
testing += ["[ ] %s installs cleanly and runs" % binary
|
||||||
|
for binary in published_binaries]
|
||||||
|
|
||||||
|
subst = {
|
||||||
|
'package': package_spph.getPackageName(),
|
||||||
|
'version': package_spph.getVersion(),
|
||||||
|
'component': package_spph.getComponent(),
|
||||||
|
'source': source,
|
||||||
|
'destinations': ', '.join(destinations),
|
||||||
|
}
|
||||||
|
subject = ("Please backport %(package)s %(version)s (%(component)s) "
|
||||||
|
"from %(source)s" % subst)
|
||||||
|
body = ('\n'.join(
|
||||||
|
[
|
||||||
|
"Please backport %(package)s %(version)s (%(component)s) "
|
||||||
|
"from %(source)s to %(destinations)s.",
|
||||||
|
"",
|
||||||
|
"Reason for the backport:",
|
||||||
|
"========================",
|
||||||
|
"<<< Enter your reasoning here >>>",
|
||||||
|
"",
|
||||||
|
"Testing:",
|
||||||
|
"========",
|
||||||
|
"Mark off items in the checklist [X] as you test them, "
|
||||||
|
"but please leave the checklist so that backporters can quickly "
|
||||||
|
"evaluate the state of testing.",
|
||||||
|
""
|
||||||
|
]
|
||||||
|
+ testing
|
||||||
|
+ [""]
|
||||||
|
+ find_rdepends(package_spph, destinations, published_binaries)
|
||||||
|
+ [""]
|
||||||
|
) % subst)
|
||||||
|
|
||||||
|
editor = EditBugReport(subject, body)
|
||||||
|
editor.edit()
|
||||||
|
subject, body = editor.get_report()
|
||||||
|
|
||||||
|
Logger.normal('The final report is:\nSummary: %s\nDescription:\n%s\n',
|
||||||
|
subject, body)
|
||||||
|
if YesNoQuestion().ask("Request this backport", "yes") == "no":
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
targets = [Launchpad.projects['%s-backports' % destination]
|
||||||
|
for destination in destinations]
|
||||||
|
bug = Launchpad.bugs.createBug(title=subject, description=body,
|
||||||
|
target=targets[0])
|
||||||
|
for target in targets[1:]:
|
||||||
|
bug.addTask(target=target)
|
||||||
|
|
||||||
|
Logger.normal("Backport request filed as %s", bug.web_link)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
parser = optparse.OptionParser('%progname [options] package')
|
||||||
|
parser.add_option('-d', '--destination', metavar='DEST',
|
||||||
|
help='Backport to DEST release and necessary '
|
||||||
|
'intermediate releases '
|
||||||
|
'(default: current stable release)')
|
||||||
|
parser.add_option('-s', '--source', metavar='SOURCE',
|
||||||
|
help='Backport from SOURCE release '
|
||||||
|
'(default: current devel release)')
|
||||||
|
parser.add_option('-l', '--lpinstance', metavar='INSTANCE', default=None,
|
||||||
|
help='Launchpad instance to connect to '
|
||||||
|
'(default: production).')
|
||||||
|
parser.add_option('--no-conf', action='store_true',
|
||||||
|
dest='no_conf', default=False,
|
||||||
|
help="Don't read config files or environment variables")
|
||||||
|
options, args = parser.parse_args()
|
||||||
|
|
||||||
|
if len(args) != 1:
|
||||||
|
parser.error("One (and only one) package must be specified")
|
||||||
|
package = args[0]
|
||||||
|
|
||||||
|
config = UDTConfig(options.no_conf)
|
||||||
|
|
||||||
|
if options.lpinstance is None:
|
||||||
|
options.lpinstance = config.get_value('LPINSTANCE')
|
||||||
|
Launchpad.login(options.lpinstance)
|
||||||
|
|
||||||
|
if options.source is None:
|
||||||
|
options.source = Distribution('ubuntu').getDevelopmentSeries().name
|
||||||
|
|
||||||
|
try:
|
||||||
|
destinations = determine_destinations(options.source,
|
||||||
|
options.destination)
|
||||||
|
except DestinationException, e:
|
||||||
|
Logger.error(str(e))
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
package_spph = locate_package(package, options.source)
|
||||||
|
request_backport(package_spph, options.source, destinations)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
16
requestsync
16
requestsync
@ -36,8 +36,8 @@ from distro_info import UbuntuDistroInfo
|
|||||||
from ubuntutools.config import UDTConfig, ubu_email
|
from ubuntutools.config import UDTConfig, ubu_email
|
||||||
from ubuntutools.lp import udtexceptions
|
from ubuntutools.lp import udtexceptions
|
||||||
from ubuntutools.misc import require_utf8
|
from ubuntutools.misc import require_utf8
|
||||||
from ubuntutools.requestsync.common import (edit_report, get_debian_changelog,
|
from ubuntutools.question import confirmation_prompt, EditBugReport
|
||||||
raw_input_exit_on_ctrlc)
|
from ubuntutools.requestsync.common import get_debian_changelog
|
||||||
|
|
||||||
#
|
#
|
||||||
# entry point
|
# entry point
|
||||||
@ -212,8 +212,7 @@ def main():
|
|||||||
print ("'%s' doesn't exist in 'Ubuntu %s'.\n"
|
print ("'%s' doesn't exist in 'Ubuntu %s'.\n"
|
||||||
"Do you want to sync a new package?"
|
"Do you want to sync a new package?"
|
||||||
% (srcpkg, release))
|
% (srcpkg, release))
|
||||||
raw_input_exit_on_ctrlc('Press [Enter] to continue '
|
confirmation_prompt()
|
||||||
'or [Ctrl-C] to abort. ')
|
|
||||||
newsource = True
|
newsource = True
|
||||||
|
|
||||||
# Get the requested Debian source package
|
# Get the requested Debian source package
|
||||||
@ -290,8 +289,7 @@ def main():
|
|||||||
'>>> ENTER_EXPLANATION_HERE <<<\n\n')
|
'>>> ENTER_EXPLANATION_HERE <<<\n\n')
|
||||||
|
|
||||||
if need_interaction:
|
if need_interaction:
|
||||||
raw_input_exit_on_ctrlc('Press [Enter] to continue.'
|
confirmation_prompt()
|
||||||
'Press [Ctrl-C] to abort now. ')
|
|
||||||
|
|
||||||
base_version = force_base_version or ubuntu_version
|
base_version = force_base_version or ubuntu_version
|
||||||
|
|
||||||
@ -313,8 +311,10 @@ def main():
|
|||||||
changelog = "XXX FIXME: add changelog here XXX"
|
changelog = "XXX FIXME: add changelog here XXX"
|
||||||
report += changelog
|
report += changelog
|
||||||
|
|
||||||
(title, report) = edit_report(title, report,
|
editor = EditBugReport(title, report)
|
||||||
changes_required=need_interaction)
|
editor.edit(optional=not need_interaction)
|
||||||
|
title, report = editor.get_report()
|
||||||
|
|
||||||
if 'XXX FIXME' in report:
|
if 'XXX FIXME' in report:
|
||||||
print >> sys.stderr, ("E: changelog boilerplate found in report, "
|
print >> sys.stderr, ("E: changelog boilerplate found in report, "
|
||||||
"please manually add changelog when using '-C'")
|
"please manually add changelog when using '-C'")
|
||||||
|
@ -1,271 +0,0 @@
|
|||||||
#!/usr/bin/perl
|
|
||||||
# Copyright (C) Patrick Schoenfeld
|
|
||||||
# Copyright (C) 2009 Ryan Kavanagh
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation; either version 2 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License along
|
|
||||||
# with this program; if not, write to the Free Software Foundation, Inc.,
|
|
||||||
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
||||||
|
|
||||||
=head1 NAME
|
|
||||||
|
|
||||||
build-rdeps - find packages that depend on a specific package to build (reverse build depends)
|
|
||||||
|
|
||||||
=head1 SYNOPSIS
|
|
||||||
|
|
||||||
B<build-rdeps> I<package>
|
|
||||||
|
|
||||||
=head1 DESCRIPTION
|
|
||||||
|
|
||||||
B<build-rdeps> searches for all packages that build-depend on the specified package.
|
|
||||||
|
|
||||||
=head1 OPTIONS
|
|
||||||
|
|
||||||
=over 4
|
|
||||||
|
|
||||||
=item B<-u> B<--update>
|
|
||||||
|
|
||||||
Run apt-get update before searching for build-depends.
|
|
||||||
|
|
||||||
=item B<-s> B<--sudo>
|
|
||||||
|
|
||||||
Use sudo when running apt-get update. Has no effect if -u is omitted.
|
|
||||||
|
|
||||||
=item B<--distribution>
|
|
||||||
|
|
||||||
Select another distribution, which is searched for build-depends.
|
|
||||||
|
|
||||||
=item B<-m> B<--print-maintainer>
|
|
||||||
|
|
||||||
Print the value of the maintainer field for each package.
|
|
||||||
|
|
||||||
=item B<-d> B<--debug>
|
|
||||||
|
|
||||||
Run the debug mode
|
|
||||||
|
|
||||||
=item B<--help>
|
|
||||||
|
|
||||||
Show the usage information.
|
|
||||||
|
|
||||||
=item B<--version>
|
|
||||||
|
|
||||||
Show the version information.
|
|
||||||
|
|
||||||
=back
|
|
||||||
|
|
||||||
=cut
|
|
||||||
|
|
||||||
use warnings;
|
|
||||||
use strict;
|
|
||||||
use File::Basename;
|
|
||||||
use File::Find;
|
|
||||||
use Getopt::Long;
|
|
||||||
use Pod::Usage;
|
|
||||||
use Data::Dumper;
|
|
||||||
my $progname = basename($0);
|
|
||||||
my $version = '1.0';
|
|
||||||
my $default_distribution = `ubuntu-distro-info --devel`;
|
|
||||||
chomp($default_distribution);
|
|
||||||
my $dctrl = "/usr/bin/grep-dctrl";
|
|
||||||
my $sources_path = "/var/lib/apt/lists/";
|
|
||||||
my $source_pattern = ".*_dists_${default_distribution}_.*Sources\$";
|
|
||||||
my @source_files;
|
|
||||||
my $sources_count=0;
|
|
||||||
my $opt_debug;
|
|
||||||
my $opt_update;
|
|
||||||
my $opt_sudo;
|
|
||||||
my $opt_maintainer;
|
|
||||||
my $opt_mainonly;
|
|
||||||
my $opt_distribution;
|
|
||||||
|
|
||||||
if (!(-x $dctrl)) {
|
|
||||||
die "$progname: Fatal error. grep-dctrl is not available.\nPlease install the 'dctrl-tools' package.\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
sub version {
|
|
||||||
print <<"EOT";
|
|
||||||
This is $progname $version, from the Debian devscripts package, v. ###VERSION###
|
|
||||||
This code is copyright by Patrick Schoenfeld, all rights reserved.
|
|
||||||
It comes with ABSOLUTELY NO WARRANTY. You are free to redistribute this code
|
|
||||||
under the terms of the GNU General Public License, version 2 or later.
|
|
||||||
EOT
|
|
||||||
exit (0);
|
|
||||||
}
|
|
||||||
|
|
||||||
sub usage {
|
|
||||||
print <<"EOT";
|
|
||||||
usage: $progname packagename
|
|
||||||
$progname --help
|
|
||||||
$progname --version
|
|
||||||
|
|
||||||
Searches for all packages that build-depend on the specified package.
|
|
||||||
|
|
||||||
Options:
|
|
||||||
-u, --update Run apt-get update before searching for build-depends.
|
|
||||||
(needs root privileges)
|
|
||||||
-s, --sudo Use sudo when running apt-get update
|
|
||||||
(has no effect when -u is omitted)
|
|
||||||
-d, --debug Enable the debug mode
|
|
||||||
-m, --print-maintainer Print the maintainer information (experimental)
|
|
||||||
--distribution distribution Select a distribution to search for build-depends
|
|
||||||
(Default: $default_distribution)
|
|
||||||
--only-main Ignore universe and multiverse
|
|
||||||
|
|
||||||
EOT
|
|
||||||
version;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub findsources {
|
|
||||||
if (/$source_pattern/ and $sources_count <= 3) {
|
|
||||||
unless ($opt_mainonly and /(universe|multiverse)/) {
|
|
||||||
push(@source_files, $_);
|
|
||||||
$sources_count+=1;
|
|
||||||
print STDERR "DEBUG: Added source file: $_ (#$sources_count)\n" if ($opt_debug);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sub findreversebuilddeps {
|
|
||||||
my ($package, $source_file) = @_;
|
|
||||||
my %packages;
|
|
||||||
my $depending_package;
|
|
||||||
my $count=0;
|
|
||||||
my $maintainer_info='';
|
|
||||||
|
|
||||||
open(PACKAGES, "$dctrl -F Build-Depends,Build-Depends-Indep $package -s Package,Build-Depends,Build-Depends-Indep,Maintainer $source_file|");
|
|
||||||
|
|
||||||
while(<PACKAGES>) {
|
|
||||||
chomp;
|
|
||||||
print STDERR "$_\n" if ($opt_debug);
|
|
||||||
if (/Package: (.*)$/) {
|
|
||||||
$depending_package = $1;
|
|
||||||
$packages{$depending_package}->{'Build-Depends'} = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (/Maintainer: (.*)$/) {
|
|
||||||
if ($depending_package) {
|
|
||||||
$packages{$depending_package}->{'Maintainer'} = $1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (/Build-Depends: (.*)$/ or /Build-Depends-Indep: (.*)$/) {
|
|
||||||
if ($depending_package) {
|
|
||||||
print STDERR "$1\n" if ($opt_debug);
|
|
||||||
if ($1 =~ /^(.*\s)?$package([\s,]|$)/) {
|
|
||||||
$packages{$depending_package}->{'Build-Depends'} = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while($depending_package = each(%packages)) {
|
|
||||||
if ($packages{$depending_package}->{'Build-Depends'} != 1) {
|
|
||||||
print STDERR "Ignoring package $depending_package because its not really build depending on $package.\n" if ($opt_debug);
|
|
||||||
next;
|
|
||||||
}
|
|
||||||
if ($opt_maintainer) {
|
|
||||||
$maintainer_info = "($packages{$depending_package}->{'Maintainer'})";
|
|
||||||
}
|
|
||||||
|
|
||||||
$count+=1;
|
|
||||||
print "$depending_package $maintainer_info \n";
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($count == 0) {
|
|
||||||
print "No reverse build-depends found for $package.\n\n"
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
print "\nFound a total of $count reverse build-depend(s) for $package.\n\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($#ARGV < 0) { usage; exit(0); }
|
|
||||||
|
|
||||||
|
|
||||||
Getopt::Long::Configure('bundling');
|
|
||||||
GetOptions(
|
|
||||||
"u|update" => \$opt_update,
|
|
||||||
"s|sudo" => \$opt_sudo,
|
|
||||||
"m|print-maintainer" => \$opt_maintainer,
|
|
||||||
"distribution=s" => \$opt_distribution,
|
|
||||||
"only-main" => \$opt_mainonly,
|
|
||||||
"d|debug" => \$opt_debug,
|
|
||||||
"h|help" => sub { usage; },
|
|
||||||
"v|version" => sub { version; }
|
|
||||||
);
|
|
||||||
|
|
||||||
my $package = shift;
|
|
||||||
|
|
||||||
if (!$package) {
|
|
||||||
die "$progname: missing argument. expecting packagename\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
print STDERR "DEBUG: Package => $package\n" if ($opt_debug);
|
|
||||||
|
|
||||||
if ($opt_update) {
|
|
||||||
print STDERR "DEBUG: Updating apt-cache before search\n" if ($opt_debug);
|
|
||||||
my @cmd;
|
|
||||||
if ($opt_sudo) {
|
|
||||||
print STDERR "DEBUG: Using sudo to become root\n" if ($opt_debug);
|
|
||||||
push(@cmd, 'sudo');
|
|
||||||
}
|
|
||||||
push(@cmd, 'apt-get', 'update');
|
|
||||||
system @cmd;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($opt_distribution) {
|
|
||||||
print STDERR "DEBUG: Setting distribution to $opt_distribution" if ($opt_debug);
|
|
||||||
$source_pattern = ".*_dists_" . $opt_distribution . "_.*Sources\$";
|
|
||||||
}
|
|
||||||
|
|
||||||
# Find sources files
|
|
||||||
find(\&findsources, $sources_path);
|
|
||||||
|
|
||||||
if (($#source_files+1) <= 0) {
|
|
||||||
die "$progname: unable to find sources files.\nDid you forget to run apt-get update (or add --update to this command)?";
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach my $source_file (@source_files) {
|
|
||||||
if ($source_file =~ /main/) {
|
|
||||||
print "Reverse Build-depends in main:\n";
|
|
||||||
print "------------------------------\n\n";
|
|
||||||
findreversebuilddeps($package, "$sources_path/$source_file");
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($source_file =~ /universe/) {
|
|
||||||
print "Reverse Build-depends in universe:\n";
|
|
||||||
print "---------------------------------\n\n";
|
|
||||||
findreversebuilddeps($package, "$sources_path/$source_file");
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($source_file =~ /multiverse/) {
|
|
||||||
print "Reverse Build-depends in multiverse:\n";
|
|
||||||
print "----------------------------------\n\n";
|
|
||||||
findreversebuilddeps($package, "$sources_path/$source_file");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
=head1 LICENSE
|
|
||||||
|
|
||||||
This code is copyright by Patrick Schoenfeld
|
|
||||||
<schoenfeld@in-medias-res.com>, all rights reserved.
|
|
||||||
This program comes with ABSOLUTELEY NO WARRANTY.
|
|
||||||
You are free to redistribute this code under the terms of the
|
|
||||||
GNU General Public License, version 2 or later.
|
|
||||||
|
|
||||||
=head1 AUTHOR
|
|
||||||
|
|
||||||
Patrick Schoenfeld <schoenfeld@in-medias-res.com>
|
|
||||||
|
|
||||||
=cut
|
|
148
reverse-depends
Executable file
148
reverse-depends
Executable file
@ -0,0 +1,148 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
#
|
||||||
|
# Copyright (C) 2011, Stefano Rivera <stefanor@ubuntu.com>
|
||||||
|
#
|
||||||
|
# Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
# purpose with or without fee is hereby granted, provided that the above
|
||||||
|
# copyright notice and this permission notice appear in all copies.
|
||||||
|
#
|
||||||
|
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||||
|
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||||
|
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||||
|
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
|
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
|
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
|
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
|
||||||
|
import optparse
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from devscripts.logger import Logger
|
||||||
|
from distro_info import UbuntuDistroInfo
|
||||||
|
|
||||||
|
from ubuntutools.rdepends import query_rdepends, RDependsException
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
parser = optparse.OptionParser('%progname [options] package',
|
||||||
|
description="List reverse-dependencies of package. "
|
||||||
|
"If the package name is prefixed with src: then the "
|
||||||
|
"reverse-dependencies of all the binary packages that "
|
||||||
|
"the specified source package builds will be listed.")
|
||||||
|
parser.add_option('-r', '--release', metavar='RELEASE',
|
||||||
|
default=UbuntuDistroInfo().devel(),
|
||||||
|
help='Query dependencies in RELEASE. Default: devel')
|
||||||
|
parser.add_option('-R', '--without-recommends',
|
||||||
|
action='store_false', dest='recommends', default=True,
|
||||||
|
help='Only consider Depends relationships, '
|
||||||
|
'not Recommends')
|
||||||
|
parser.add_option('-s', '--with-suggests',
|
||||||
|
action='store_true', dest='suggests', default=False,
|
||||||
|
help='Also consider Suggests relationships')
|
||||||
|
parser.add_option('-b', '--build-depends',
|
||||||
|
action='store_const', dest='arch', const='source',
|
||||||
|
help='Query build dependencies (synonym for '
|
||||||
|
'--arch=source)')
|
||||||
|
parser.add_option('-a', '--arch', metavar='ARCH', default='any',
|
||||||
|
help='Query dependencies in ARCH. '
|
||||||
|
'Default: any')
|
||||||
|
parser.add_option('-c', '--component', metavar='COMPONENT',
|
||||||
|
action='append',
|
||||||
|
help='Only consider reverse-dependencies in COMPONENT. '
|
||||||
|
'Can be specified multiple times. Default: all')
|
||||||
|
parser.add_option('-l', '--list',
|
||||||
|
action='store_true', default=False,
|
||||||
|
help='Display a simple, machine-readable list')
|
||||||
|
parser.add_option('-u', '--service-url', metavar='URL',
|
||||||
|
dest='server', default=None,
|
||||||
|
help='Reverse Dependencies webservice URL. '
|
||||||
|
'Default: UbuntuWire')
|
||||||
|
|
||||||
|
options, args = parser.parse_args()
|
||||||
|
|
||||||
|
if len(args) != 1:
|
||||||
|
parser.error("One (and only one) package must be specified")
|
||||||
|
package = args[0]
|
||||||
|
|
||||||
|
opts = {}
|
||||||
|
if options.server is not None:
|
||||||
|
opts['server'] = options.server
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = query_rdepends(package, options.release, options.arch, **opts)
|
||||||
|
except RDependsException, e:
|
||||||
|
Logger.error(str(e))
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
if options.arch == 'source':
|
||||||
|
fields = ['Reverse-Build-Depends', 'Reverse-Build-Depends-Indep']
|
||||||
|
else:
|
||||||
|
fields = ['Reverse-Depends']
|
||||||
|
if options.recommends:
|
||||||
|
fields.append('Reverse-Recommends')
|
||||||
|
if options.suggests:
|
||||||
|
fields.append('Reverse-Suggests')
|
||||||
|
|
||||||
|
for field in data.keys():
|
||||||
|
if field not in fields:
|
||||||
|
del data[field]
|
||||||
|
|
||||||
|
if options.component:
|
||||||
|
for field, rdeps in data.items():
|
||||||
|
filtered = [rdep for rdep in rdeps
|
||||||
|
if rdep['Component'] in options.component]
|
||||||
|
if not filtered:
|
||||||
|
del data[field]
|
||||||
|
else:
|
||||||
|
data[field] = filtered
|
||||||
|
|
||||||
|
if options.list:
|
||||||
|
display_consise(data)
|
||||||
|
else:
|
||||||
|
display_verbose(data)
|
||||||
|
|
||||||
|
|
||||||
|
def display_verbose(data):
|
||||||
|
if not data:
|
||||||
|
print "No reverse dependencies found"
|
||||||
|
return
|
||||||
|
|
||||||
|
all_archs = set()
|
||||||
|
# This isn't accurate, but we make up for it by displaying what we found
|
||||||
|
for rdeps in data.itervalues():
|
||||||
|
for rdep in rdeps:
|
||||||
|
if 'Architectures' in rdep:
|
||||||
|
all_archs.update(rdep['Architectures'])
|
||||||
|
|
||||||
|
for field, rdeps in data.iteritems():
|
||||||
|
print field
|
||||||
|
print '=' * len(field)
|
||||||
|
rdeps.sort(key=lambda x: x['Package'])
|
||||||
|
for rdep in rdeps:
|
||||||
|
line = '* %s' % rdep['Package']
|
||||||
|
if all_archs and set(rdep['Architectures']) != all_archs:
|
||||||
|
line += ' [%s]' % ' '.join(sorted(rdep['Architectures']))
|
||||||
|
if 'Dependency' in rdep:
|
||||||
|
if len(line) < 30:
|
||||||
|
line += ' ' * (30 - len(line))
|
||||||
|
line += ' (for %s)' % rdep['Dependency']
|
||||||
|
print line
|
||||||
|
print
|
||||||
|
|
||||||
|
if all_archs:
|
||||||
|
print ("Packages without architectures listed are "
|
||||||
|
"reverse-dependencies in: %s"
|
||||||
|
% ', '.join(sorted(list(all_archs))))
|
||||||
|
|
||||||
|
|
||||||
|
def display_consise(data):
|
||||||
|
result = set()
|
||||||
|
for rdeps in data.itervalues():
|
||||||
|
for rdep in rdeps:
|
||||||
|
result.add(rdep['Package'])
|
||||||
|
|
||||||
|
print u'\n'.join(sorted(list(result)))
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
3
setup.py
3
setup.py
@ -33,8 +33,9 @@ scripts = ['404main',
|
|||||||
'pull-debian-source',
|
'pull-debian-source',
|
||||||
'pull-lp-source',
|
'pull-lp-source',
|
||||||
'pull-revu-source',
|
'pull-revu-source',
|
||||||
|
'requestbackport',
|
||||||
'requestsync',
|
'requestsync',
|
||||||
'reverse-build-depends',
|
'reverse-depends',
|
||||||
'setup-packaging-environment',
|
'setup-packaging-environment',
|
||||||
'sponsor-patch',
|
'sponsor-patch',
|
||||||
'submittodebian',
|
'submittodebian',
|
||||||
|
@ -30,7 +30,8 @@ from tempfile import mkdtemp
|
|||||||
from distro_info import UbuntuDistroInfo
|
from distro_info import UbuntuDistroInfo
|
||||||
|
|
||||||
from ubuntutools.config import ubu_email
|
from ubuntutools.config import ubu_email
|
||||||
from ubuntutools.question import YesNoQuestion
|
from ubuntutools.question import YesNoQuestion, EditFile
|
||||||
|
from ubuntutools.subprocess import call, check_call, Popen, PIPE
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from debian.changelog import Changelog
|
from debian.changelog import Changelog
|
||||||
@ -76,11 +77,10 @@ def gen_debdiff(tmpdir, changelog):
|
|||||||
|
|
||||||
debdiff = os.path.join(tmpdir, '%s_%s.debdiff' % (pkg, newver))
|
debdiff = os.path.join(tmpdir, '%s_%s.debdiff' % (pkg, newver))
|
||||||
|
|
||||||
if os.system('bzr diff -r tag:%s > /dev/null 2>&1' % oldver) == 256:
|
devnull = open('/dev/null', 'w')
|
||||||
|
diff_cmd = ['bzr', 'diff', '-r', 'tag:' + str(oldver)]
|
||||||
|
if call(diff_cmd, stdout=devnull, stderr=devnull) == 1:
|
||||||
print "Extracting bzr diff between %s and %s" % (oldver, newver)
|
print "Extracting bzr diff between %s and %s" % (oldver, newver)
|
||||||
cmd = 'bzr diff -r tag:%s | filterdiff -x "*changelog*" > %s' % \
|
|
||||||
(oldver, debdiff)
|
|
||||||
run_cmd(cmd)
|
|
||||||
else:
|
else:
|
||||||
if oldver.epoch is not None:
|
if oldver.epoch is not None:
|
||||||
oldver = str(oldver)[str(oldver).index(":")+1:]
|
oldver = str(oldver)[str(oldver).index(":")+1:]
|
||||||
@ -94,9 +94,16 @@ def gen_debdiff(tmpdir, changelog):
|
|||||||
check_file(newdsc)
|
check_file(newdsc)
|
||||||
|
|
||||||
print "Generating debdiff between %s and %s" % (oldver, newver)
|
print "Generating debdiff between %s and %s" % (oldver, newver)
|
||||||
cmd = 'debdiff %s %s | filterdiff -x "*changelog*" > %s' % \
|
diff_cmd = ['debdiff', olddsc, newdsc]
|
||||||
(olddsc, newdsc, debdiff)
|
|
||||||
run_cmd(cmd)
|
diff = Popen(diff_cmd, stdout=PIPE)
|
||||||
|
debdiff_f = open(debdiff, 'w')
|
||||||
|
filterdiff = Popen(['filterdiff', '-x', '*changelog*'],
|
||||||
|
stdin=diff.stdout, stdout=debdiff_f)
|
||||||
|
diff.stdout.close()
|
||||||
|
filterdiff.wait()
|
||||||
|
debdiff_f.close()
|
||||||
|
devnull.close()
|
||||||
|
|
||||||
return debdiff
|
return debdiff
|
||||||
|
|
||||||
@ -109,22 +116,13 @@ def check_file(fname, critical = True):
|
|||||||
print u"Couldn't find «%s».\n" % fname
|
print u"Couldn't find «%s».\n" % fname
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
def edit_debdiff(debdiff):
|
|
||||||
cmd = 'sensible-editor %s' % (debdiff)
|
|
||||||
run_cmd(cmd)
|
|
||||||
|
|
||||||
def submit_bugreport(body, debdiff, deb_version, changelog):
|
def submit_bugreport(body, debdiff, deb_version, changelog):
|
||||||
cmd = ('reportbug -P "User: ubuntu-devel@lists.ubuntu.com" '
|
devel = UbuntuDistroInfo().devel()
|
||||||
'-P "Usertags: origin-ubuntu %s ubuntu-patch" -T patch -A %s '
|
cmd = ('reportbug', '-P', 'User: ubuntu-devel@lists.ubuntu.com',
|
||||||
'-B debian -i %s -V %s %s') % \
|
'-P', 'Usertags: origin-ubuntu %s ubuntu-patch' % devel,
|
||||||
(UbuntuDistroInfo().devel(), debdiff, body, deb_version,
|
'-T', 'patch', '-A', debdiff, '-B', 'debian', '-i', body,
|
||||||
changelog.package)
|
'-V', deb_version, changelog.package)
|
||||||
run_cmd(cmd)
|
check_call(cmd)
|
||||||
|
|
||||||
def run_cmd(cmd):
|
|
||||||
if os.getenv('DEBUG'):
|
|
||||||
print "%s\n" % cmd
|
|
||||||
os.system(cmd)
|
|
||||||
|
|
||||||
def check_reportbug_config():
|
def check_reportbug_config():
|
||||||
fn = os.path.expanduser('~/.reportbugrc')
|
fn = os.path.expanduser('~/.reportbugrc')
|
||||||
@ -182,7 +180,12 @@ def main():
|
|||||||
fp.close()
|
fp.close()
|
||||||
|
|
||||||
debdiff = gen_debdiff(tmpdir, changelog)
|
debdiff = gen_debdiff(tmpdir, changelog)
|
||||||
edit_debdiff(debdiff)
|
|
||||||
|
EditFile(debdiff, 'debdiff').edit(optional=True)
|
||||||
|
EditFile(body, 'bug report', [
|
||||||
|
re.compile('.*REPLACE THIS WITH ACTUAL INFORMATION.*')
|
||||||
|
]).edit()
|
||||||
|
|
||||||
submit_bugreport(body, debdiff, deb_version, changelog)
|
submit_bugreport(body, debdiff, deb_version, changelog)
|
||||||
os.unlink(body)
|
os.unlink(body)
|
||||||
os.unlink(debdiff)
|
os.unlink(debdiff)
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
#
|
#
|
||||||
# question.py - Helper class for asking questions
|
# question.py - Helper class for asking questions
|
||||||
#
|
#
|
||||||
# Copyright (C) 2010, Benjamin Drung <bdrung@ubuntu.com>
|
# Copyright (C) 2010, Benjamin Drung <bdrung@ubuntu.com>,
|
||||||
|
# 2011, Stefano Rivera <stefanor@ubuntu.com>
|
||||||
#
|
#
|
||||||
# Permission to use, copy, modify, and/or distribute this software for any
|
# Permission to use, copy, modify, and/or distribute this software for any
|
||||||
# purpose with or without fee is hereby granted, provided that the above
|
# purpose with or without fee is hereby granted, provided that the above
|
||||||
@ -15,6 +16,14 @@
|
|||||||
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
|
||||||
|
import tempfile
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import ubuntutools.subprocess
|
||||||
|
|
||||||
|
|
||||||
class Question(object):
|
class Question(object):
|
||||||
def __init__(self, options, show_help=True):
|
def __init__(self, options, show_help=True):
|
||||||
assert len(options) >= 2
|
assert len(options) >= 2
|
||||||
@ -85,3 +94,104 @@ def input_number(question, min_number, max_number, default=None):
|
|||||||
print "Please input a number."
|
print "Please input a number."
|
||||||
assert type(selected) == int
|
assert type(selected) == int
|
||||||
return selected
|
return selected
|
||||||
|
|
||||||
|
|
||||||
|
def confirmation_prompt(message=None, action=None):
|
||||||
|
'''Display message, or a stock message including action, and wait for the
|
||||||
|
user to press Enter
|
||||||
|
'''
|
||||||
|
if message is None:
|
||||||
|
if action is None:
|
||||||
|
action = 'continue'
|
||||||
|
message = 'Press [Enter] to %s. Press [Ctrl-C] to abort now.' % action
|
||||||
|
try:
|
||||||
|
raw_input(message)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
print '\nAborting as requested.'
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
class EditFile(object):
|
||||||
|
def __init__(self, filename, description, placeholders=None):
|
||||||
|
self.filename = filename
|
||||||
|
self.description = description
|
||||||
|
if placeholders is None:
|
||||||
|
placeholders = (re.compile(r'^<<<.*>>>$', re.UNICODE),)
|
||||||
|
self.placeholders = placeholders
|
||||||
|
|
||||||
|
def edit(self, optional=False):
|
||||||
|
if optional:
|
||||||
|
print "Currently the %s looks like:" % self.description
|
||||||
|
with open(self.filename, 'r') as f:
|
||||||
|
print f.read()
|
||||||
|
if YesNoQuestion().ask("Edit", "no") == "no":
|
||||||
|
return
|
||||||
|
|
||||||
|
done = False
|
||||||
|
while not done:
|
||||||
|
old_mtime = os.stat(self.filename).st_mtime
|
||||||
|
ubuntutools.subprocess.check_call(['sensible-editor',
|
||||||
|
self.filename])
|
||||||
|
modified = old_mtime != os.stat(self.filename).st_mtime
|
||||||
|
placeholders_present = False
|
||||||
|
if self.placeholders:
|
||||||
|
with open(self.filename, 'r') as f:
|
||||||
|
for line in f:
|
||||||
|
for placeholder in self.placeholders:
|
||||||
|
if placeholder.search(line.strip()):
|
||||||
|
placeholders_present = True
|
||||||
|
|
||||||
|
if placeholders_present:
|
||||||
|
print ("Placeholders still present in the %s. "
|
||||||
|
"Please replace them with useful information."
|
||||||
|
% self.description)
|
||||||
|
confirmation_prompt(action='edit again')
|
||||||
|
elif not modified:
|
||||||
|
print "The %s was not modified" % self.description
|
||||||
|
if YesNoQuestion().ask("Edit again", "yes") == "no":
|
||||||
|
done = True
|
||||||
|
elif self.check_edit():
|
||||||
|
done = True
|
||||||
|
|
||||||
|
def check_edit(self):
|
||||||
|
'''Override this to implement extra checks on the edited report.
|
||||||
|
Should return False if another round of editing is needed,
|
||||||
|
and should prompt the user to confirm that, if necessary.
|
||||||
|
'''
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class EditBugReport(EditFile):
|
||||||
|
split_re = re.compile(r'^Summary.*?:\s+(.*)\s+'
|
||||||
|
r'Description:\s+(.*)$',
|
||||||
|
re.DOTALL | re.UNICODE)
|
||||||
|
|
||||||
|
def __init__(self, subject, body, placeholders=None):
|
||||||
|
tmpfile = tempfile.NamedTemporaryFile(prefix=sys.argv[0] + '_',
|
||||||
|
suffix='.txt',
|
||||||
|
delete=False)
|
||||||
|
tmpfile.write((u'Summary (one line):\n%s\n\nDescription:\n%s'
|
||||||
|
% (subject, body)).encode('utf-8'))
|
||||||
|
tmpfile.close()
|
||||||
|
super(EditBugReport, self).__init__(tmpfile.name, 'bug report',
|
||||||
|
placeholders)
|
||||||
|
|
||||||
|
def check_edit(self):
|
||||||
|
with open(self.filename, 'r') as f:
|
||||||
|
report = f.read().decode('utf-8')
|
||||||
|
|
||||||
|
if self.split_re.match(report) is None:
|
||||||
|
print ("The %s doesn't start with 'Summary:' and 'Description:' "
|
||||||
|
"blocks" % self.description)
|
||||||
|
confirmation_prompt('edit again')
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def get_report(self):
|
||||||
|
with open(self.filename, 'r') as f:
|
||||||
|
report = f.read().decode('utf-8')
|
||||||
|
|
||||||
|
m = self.split_re.match(report)
|
||||||
|
report = (m.group(1), m.group(2))
|
||||||
|
os.unlink(self.filename)
|
||||||
|
return report
|
||||||
|
35
ubuntutools/rdepends.py
Normal file
35
ubuntutools/rdepends.py
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
# Copyright (C) 2011, Stefano Rivera <stefanor@ubuntu.com>
|
||||||
|
#
|
||||||
|
# Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
# purpose with or without fee is hereby granted, provided that the above
|
||||||
|
# copyright notice and this permission notice appear in all copies.
|
||||||
|
#
|
||||||
|
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||||
|
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||||
|
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||||
|
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
|
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
|
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
|
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import urllib2
|
||||||
|
|
||||||
|
|
||||||
|
class RDependsException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def query_rdepends(package, release, arch,
|
||||||
|
server='http://qa.ubuntuwire.org/rdepends'):
|
||||||
|
"""Look up a packages reverse-dependencies on the Ubuntuwire
|
||||||
|
Reverse- webservice
|
||||||
|
"""
|
||||||
|
|
||||||
|
url = os.path.join(server, 'v1', release, arch, package)
|
||||||
|
|
||||||
|
try:
|
||||||
|
return json.load(urllib2.urlopen(url))
|
||||||
|
except urllib2.HTTPError, e:
|
||||||
|
raise RDependsException(e.read().strip())
|
@ -22,22 +22,10 @@
|
|||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import urllib2
|
import urllib2
|
||||||
import re
|
|
||||||
import tempfile
|
|
||||||
from debian.changelog import Changelog
|
from debian.changelog import Changelog
|
||||||
|
|
||||||
from ubuntutools import subprocess
|
from ubuntutools import subprocess
|
||||||
|
|
||||||
def raw_input_exit_on_ctrlc(*args, **kwargs):
|
|
||||||
'''
|
|
||||||
A wrapper around raw_input() to exit with a normalized message on Control-C
|
|
||||||
'''
|
|
||||||
try:
|
|
||||||
return raw_input(*args, **kwargs)
|
|
||||||
except KeyboardInterrupt:
|
|
||||||
print '\nAbort requested. No sync request filed.'
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
def get_changelog(srcpkg, distro):
|
def get_changelog(srcpkg, distro):
|
||||||
'''
|
'''
|
||||||
Download and return a parsed changelog for srcpackage, from
|
Download and return a parsed changelog for srcpackage, from
|
||||||
@ -83,68 +71,3 @@ def get_debian_changelog(srcpkg, version):
|
|||||||
break
|
break
|
||||||
new_entries.append(unicode(block))
|
new_entries.append(unicode(block))
|
||||||
return u''.join(new_entries)
|
return u''.join(new_entries)
|
||||||
|
|
||||||
def edit_report(subject, body, changes_required = False):
|
|
||||||
'''
|
|
||||||
Ask if the user wants to edit a report (consisting of subject and body)
|
|
||||||
in sensible-editor.
|
|
||||||
|
|
||||||
If changes_required is True then the file has to be edited before we
|
|
||||||
can proceed.
|
|
||||||
|
|
||||||
Returns (new_subject, new_body).
|
|
||||||
'''
|
|
||||||
|
|
||||||
editing_finished = False
|
|
||||||
while not editing_finished:
|
|
||||||
report = 'Summary (one line):\n%s\n\nDescription:\n%s' % (subject, body)
|
|
||||||
|
|
||||||
if not changes_required:
|
|
||||||
print 'Currently the report looks as follows:\n%s' % report
|
|
||||||
while True:
|
|
||||||
val = raw_input_exit_on_ctrlc('Do you want to edit the report '
|
|
||||||
'[y/N]? ')
|
|
||||||
if val.lower() in ('y', 'yes'):
|
|
||||||
break
|
|
||||||
elif val.lower() in ('n', 'no', ''):
|
|
||||||
editing_finished = True
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
print 'Invalid answer.'
|
|
||||||
|
|
||||||
if not editing_finished:
|
|
||||||
# Create tempfile and remember mtime
|
|
||||||
report_file = tempfile.NamedTemporaryFile(prefix='requestsync_')
|
|
||||||
report_file.write(report.encode('utf-8'))
|
|
||||||
report_file.flush()
|
|
||||||
mtime_before = os.stat(report_file.name).st_mtime
|
|
||||||
|
|
||||||
# Launch editor
|
|
||||||
try:
|
|
||||||
subprocess.check_call(['sensible-editor', report_file.name])
|
|
||||||
except subprocess.CalledProcessError, e:
|
|
||||||
print >> sys.stderr, ('Error calling sensible-editor: %s\n'
|
|
||||||
'Aborting.' % e)
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
# Check if the tempfile has been changed
|
|
||||||
if changes_required:
|
|
||||||
if mtime_before == os.stat(report_file.name).st_mtime:
|
|
||||||
print ('The report has not been changed, but you have to '
|
|
||||||
'explain why the Ubuntu changes can be dropped.')
|
|
||||||
raw_input_exit_on_ctrlc('Press [Enter] to retry or '
|
|
||||||
'[Control-C] to abort. ')
|
|
||||||
else:
|
|
||||||
changes_required = False
|
|
||||||
|
|
||||||
report_file.seek(0)
|
|
||||||
report = report_file.read().decode('utf-8')
|
|
||||||
report_file.close()
|
|
||||||
|
|
||||||
# Undecorate report again
|
|
||||||
(subject, body) = report.split("\nDescription:\n", 1)
|
|
||||||
# Remove prefix and whitespace from subject
|
|
||||||
subject = re.sub('^Summary \(one line\):\s*', '', subject,
|
|
||||||
1).strip()
|
|
||||||
|
|
||||||
return (subject, body)
|
|
||||||
|
@ -26,9 +26,9 @@ import urllib2
|
|||||||
from debian.deb822 import Changes
|
from debian.deb822 import Changes
|
||||||
from distro_info import DebianDistroInfo
|
from distro_info import DebianDistroInfo
|
||||||
|
|
||||||
from ubuntutools.requestsync.common import raw_input_exit_on_ctrlc
|
|
||||||
from ubuntutools.lp.lpapicache import (Launchpad, Distribution, PersonTeam,
|
from ubuntutools.lp.lpapicache import (Launchpad, Distribution, PersonTeam,
|
||||||
DistributionSourcePackage)
|
DistributionSourcePackage)
|
||||||
|
from ubuntutools.question import confirmation_prompt
|
||||||
|
|
||||||
def get_debian_srcpkg(name, release):
|
def get_debian_srcpkg(name, release):
|
||||||
debian = Distribution('debian')
|
debian = Distribution('debian')
|
||||||
@ -60,7 +60,7 @@ Your sync request shall require an approval by a member of the appropriate
|
|||||||
sponsorship team, who shall be subscribed to this bug report.
|
sponsorship team, who shall be subscribed to this bug report.
|
||||||
This must be done before it can be processed by a member of the Ubuntu Archive
|
This must be done before it can be processed by a member of the Ubuntu Archive
|
||||||
team.'''
|
team.'''
|
||||||
raw_input_exit_on_ctrlc('If the above is correct please press [Enter] ')
|
confirmation_prompt()
|
||||||
|
|
||||||
return need_sponsor
|
return need_sponsor
|
||||||
|
|
||||||
@ -88,8 +88,7 @@ def check_existing_reports(srcpkg):
|
|||||||
'Please check the above URL to verify this before '
|
'Please check the above URL to verify this before '
|
||||||
'continuing.'
|
'continuing.'
|
||||||
% (bug.title, bug.web_link))
|
% (bug.title, bug.web_link))
|
||||||
raw_input_exit_on_ctrlc('Press [Enter] to continue or [Ctrl-C] '
|
confirmation_prompt()
|
||||||
'to abort. ')
|
|
||||||
|
|
||||||
def get_ubuntu_delta_changelog(srcpkg):
|
def get_ubuntu_delta_changelog(srcpkg):
|
||||||
'''
|
'''
|
||||||
@ -133,7 +132,7 @@ def post_bug(srcpkg, subscribe, status, bugtitle, bugtext):
|
|||||||
|
|
||||||
print ('The final report is:\nSummary: %s\nDescription:\n%s\n'
|
print ('The final report is:\nSummary: %s\nDescription:\n%s\n'
|
||||||
% (bugtitle, bugtext))
|
% (bugtitle, bugtext))
|
||||||
raw_input_exit_on_ctrlc('Press [Enter] to continue or [Ctrl-C] to abort. ')
|
confirmation_prompt()
|
||||||
|
|
||||||
if srcpkg:
|
if srcpkg:
|
||||||
bug_target = DistributionSourcePackage(
|
bug_target = DistributionSourcePackage(
|
||||||
|
@ -30,8 +30,8 @@ from devscripts.logger import Logger
|
|||||||
from distro_info import DebianDistroInfo
|
from distro_info import DebianDistroInfo
|
||||||
|
|
||||||
from ubuntutools.archive import rmadison, FakeSPPH
|
from ubuntutools.archive import rmadison, FakeSPPH
|
||||||
from ubuntutools.requestsync.common import (get_changelog,
|
from ubuntutools.requestsync.common import get_changelog
|
||||||
raw_input_exit_on_ctrlc)
|
from ubuntutools.question import confirmation_prompt, YesNoQuestion
|
||||||
from ubuntutools import subprocess
|
from ubuntutools import subprocess
|
||||||
from ubuntutools.lp.udtexceptions import PackageNotFoundException
|
from ubuntutools.lp.udtexceptions import PackageNotFoundException
|
||||||
|
|
||||||
@ -71,17 +71,12 @@ def need_sponsorship(name, component, release):
|
|||||||
component.
|
component.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
while True:
|
val = YesNoQuestion().ask("Do you have upload permissions for the "
|
||||||
print ("Do you have upload permissions for the '%s' component "
|
"'%s' component or the package '%s' in "
|
||||||
"or the package '%s' in Ubuntu %s?"
|
"Ubuntu %s?\n"
|
||||||
% (component, name, release))
|
"If in doubt answer 'n'."
|
||||||
val = raw_input_exit_on_ctrlc("If in doubt answer 'n'. [y/N]? ")
|
% (component, name, release), 'no')
|
||||||
if val.lower() in ('y', 'yes'):
|
return val == 'no'
|
||||||
return False
|
|
||||||
elif val.lower() in ('n', 'no', ''):
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
print 'Invalid answer'
|
|
||||||
|
|
||||||
def check_existing_reports(srcpkg):
|
def check_existing_reports(srcpkg):
|
||||||
'''
|
'''
|
||||||
@ -90,7 +85,7 @@ def check_existing_reports(srcpkg):
|
|||||||
print ('Please check on '
|
print ('Please check on '
|
||||||
'https://bugs.launchpad.net/ubuntu/+source/%s/+bugs\n'
|
'https://bugs.launchpad.net/ubuntu/+source/%s/+bugs\n'
|
||||||
'for duplicate sync requests before continuing.' % srcpkg)
|
'for duplicate sync requests before continuing.' % srcpkg)
|
||||||
raw_input_exit_on_ctrlc('Press [Enter] to continue or [Ctrl-C] to abort. ')
|
confirmation_prompt()
|
||||||
|
|
||||||
def get_ubuntu_delta_changelog(srcpkg):
|
def get_ubuntu_delta_changelog(srcpkg):
|
||||||
'''
|
'''
|
||||||
@ -166,7 +161,7 @@ Content-Type: text/plain; charset=UTF-8
|
|||||||
%s''' % (myemailaddr, to, bugtitle, signed_report)
|
%s''' % (myemailaddr, to, bugtitle, signed_report)
|
||||||
|
|
||||||
print 'The final report is:\n%s' % mail
|
print 'The final report is:\n%s' % mail
|
||||||
raw_input_exit_on_ctrlc('Press [Enter] to continue or [Ctrl-C] to abort. ')
|
confirmation_prompt()
|
||||||
|
|
||||||
# connect to the server
|
# connect to the server
|
||||||
try:
|
try:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user