3 ### Build script for Debian packages
5 ### (c) 2008 Mark Wooding
8 ###----- Licensing notice ---------------------------------------------------
10 ### This program is free software; you can redistribute it and/or modify
11 ### it under the terms of the GNU General Public License as published by
12 ### the Free Software Foundation; either version 2 of the License, or
13 ### (at your option) any later version.
15 ### This program is distributed in the hope that it will be useful,
16 ### but WITHOUT ANY WARRANTY; without even the implied warranty of
17 ### MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 ### GNU General Public License for more details.
20 ### You should have received a copy of the GNU General Public License
21 ### along with this program; if not, write to the Free Software Foundation,
22 ### Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24 ###--------------------------------------------------------------------------
25 ### Conventions for build systems.
27 ### This script is designed to work with a variety of `make'-based build
28 ### systems, but there are a number of conventions which must be followed if
29 ### this is going to work properly.
31 ### * There must be a `configure.ac', `configure.in', or `.links' file, or
32 ### a `.git' directory in the project top-level, so that we can find it.
34 ### * The following `make' variables must be assigned in the top-level
35 ### Makefile, after `mdw-build' has constructed it.
37 ### distdir The name of the top-level project directory in the
38 ### source distribution, and the base name for
39 ### distribution archives; should be of the form
40 ### `PROJECT-VERSION'.
42 ### The following `make' targets must be available in the top-level
45 ### dist Write to $(distdir).tar.gz a source distribution of
48 ### distcheck As for `dist', but also build and test the project.
50 ### * The source distribution constructed by `make dist' must contain a file
51 ### $(distdir)/RELEASE containing the release name. This isn't currently
52 ### tested, but it might be later.
56 ###--------------------------------------------------------------------------
59 unset checkout checkoutrev
62 unset sbuild sbuildsrv
63 unset upload uploadpath
65 unset build distcheck debian clean vpath native
67 "/etc/mdw-build.conf" \
68 "${XDG_CONFIG_HOME-$HOME/.config}/mdw-build.conf" \
71 if [ -f "$i" ]; then . "$i"; fi
75 eval "p=\${$var+t} q=\${$want+t}"
76 case $p,$q in t,*) ;; *,t) eval "$var=yes" ;; *) eval "$var=no" ;; esac
78 : ${checkout=yes} ${checkoutrev=HEAD}
80 : ${setup=yes} ${setupcmd=!guess}
88 default_depends sbuild sbuildsrv
89 default_depends sign signkey
90 default_depends upload uploadpath
91 default_depends dput dputtarget
92 : ${DEB_BUILD_OPTIONS=parallel=4}; export DEB_BUILD_OPTIONS
94 ###--------------------------------------------------------------------------
101 Usage: $prog [-v] BUILDOPT
110 [no]upload[=SERVER:PATH]
122 ## Initialize qualifiers.
125 ## Parse simple options.
127 while getopts "hv" opt; do
134 shift $((OPTIND - 1))
136 ## Parse the build options.
139 eval "p=\${$want+t}\${$want-nil}"
142 nil) echo >&2 "$prog: $want not set"; exit 1 ;;
147 checkout) checkout=yes checkoutrev=HEAD ;;
148 checkout=*) checkout=yes checkoutrev=${opt#*=} ;;
149 release) build=release ;;
150 norelease) build=test ;;
151 setup) setup=yes setupcmd=!guess ;;
152 setup=*) setup=yes setupcmd=${opt#*=} ;;
153 upload) maybe_set upload uploadpath ;;
154 upload=*) upload=yes uploadpath=${opt#*=} ;;
155 sign) maybe_set sign signkey ;;
156 sign=*) sign=yes signkey=${opt#*=} ;;
157 sbuild) maybe_set sbuild sbuildsrv ;;
158 sbuild=*) sbuild=yes sbuildsrv=${opt#*=} ;;
159 dput) maybe_set dput dputtarget ;;
160 dput=*) dput=yes dputtarget=${opt#*=} ;;
161 make=*) make=${opt#*=} ;;
163 distcheck | debian | clean | vpath | native | test)
166 nocheckout | nosetup | nodistcheck | nodebian | \
167 noupload | nodput | noclean | novpath | nonative | notest | \
178 ## Parse DEB_BUILD_OPTIONS.
180 set -- $DEB_BUILD_OPTIONS
183 parallel=*) jobs=${opt#*=} ;;
189 case $jobs in 1) ;; *) makeopts="$makeopts -j$jobs" ;; esac
191 ###--------------------------------------------------------------------------
192 ### Utility functions.
194 ## File descriptor assignments:
196 ## 0 -- original stdin (never touched)
197 ## 1, 2 -- stdout, stderr, redirected to 3 while running comamnds
198 ## log -- logfile and original stderr (verbose), or logfile only (quiet);
199 ## captures command output
200 ## diag -- logfile; primary diagnostic output
201 ## term -- orginal stderr; secondary diagnostic output (with colours)
205 echo $message >&$diag
206 echo "$(tput bold; tput setaf $colour)$message$(tput sgr0; tput op)" >&$term
216 release) fail "$*" ;;
217 *) notify 5 "??? $*" ;;
232 nice "$@" 2>&$log {log}>&- {diag}>&- {term}>&- || fail "$1: exit $?"
235 run () { runx "$@" >&$log; }
238 echo -n "(test $*)" >&$diag
239 if "$@" >&$diag 2>&$diag {log}>&- {diag}>&- {term}>&-; then
248 ###--------------------------------------------------------------------------
251 ## Some preflight checks.
253 no,release) fail "refusing to make release build without testing" ;;
255 case $test,$distcheck in
257 info "forcing \`distcheck' off because tsting disabled"
262 ## Find the top-level package directory.
263 while [ ! -f configure.ac -a ! -f configure.in -a \
264 ! -f .links -a ! -d .git ]; do
267 fail "couldn't find top-level directory"
274 ## Add a qualifier for Scratchbox builds.
275 case ${SBOX_SESSION_DIR+t},${DEB_BUILD_ARCH+t} in
276 t,t) qual=$qual$qualsep$DEB_BUILD_ARCH; qualsep=- ;;
277 t,*) fail "unknown build arch" ;;
280 ## Construct the output directory.
281 releasepath=$toppath/dist-$build$qual
282 chmod -R +w $releasepath 2>/dev/null || :
283 rm -rf $releasepath 2>/dev/null || :
285 logfile=$releasepath/mdw-build.log
287 exec {diag}>>$logfile || fail "Failed to create log."
289 no) exec {log}>&$diag ;;
290 yes) exec {log}> >(tee -a $logfile >&$term {term}>&- {diag}>&-) ;;
293 ## Repeat the earlier assignments for tbe benefit of the logfile.
294 assign toppath $toppath
295 assign releasepath $releasepath
296 assign logfile $logfile
298 ## Do we have a Git repository?
299 case "$checkout,$setup,$(yesno [ -d $toppath/.git ])" in
301 fail "Inconsistent options: can't check out without setup."
304 info "No Git repository found."
305 checkout=no gitver=none
309 [ "$(git ls-files -m)" = "" ] ||
310 warn "working tree has uncommitted changes"
314 ## Is there Debian build equipment?
315 case "$debian,$(yesno [ -d $toppath/debian ])" in
317 info "No debian directory found."
318 debian=no debver=none
324 debver=$(dpkg-parsechangelog | sed -n 's/^Version: //p')
325 debsrc=$(dpkg-parsechangelog | sed -n 's/^Source: //p')
326 debname=$(git config user.name) debemail=$(git config user.email)
330 ## Maybe check out a copy of the source.
334 run git clone -sn $toppath/.git _source
335 assign srcpath $releasepath/_source
337 run git update-ref refs/heads/mdw-build $checkoutrev ""
338 run git symbolic-ref HEAD refs/heads/mdw-build
339 run git read-tree --reset refs/heads/mdw-build
340 run git checkout-index -afu
341 assign gitversion "$(git describe --abbrev=4)"
344 assign srcpath $toppath
348 ## Check the version number.
350 case "$gitversion,$debver" in
354 dvref=$(echo "$debver" | tr '~' '_')
355 if [ "$gitversion" = "$dvref" ]; then
356 assign debversion "$debver"
358 warn "Git version $gitversion doesn't match Debian version $debver"
360 dver=$(echo $gitversion | sed 's/-/+/; s/-/./g')
361 case $debver in *~) dver=$debver$dver ;; esac
362 assign debversion "$dver"
368 ## Maybe refresh the build machinery.
373 if [ -f .links ]; then setupcmd=mdw-setup
374 elif [ -x autogen.sh ]; then setupcmd=./autogen.sh
375 elif [ -x setup ]; then setupcmd=./setup
376 elif [ -f configure.ac ]; then setupcmd="autoreconf -is"
377 else setupcmd=mdw-setup
385 ## Initialize the build directory.
386 case "$vpath,$(yesno [ -e $srcpath/configure ])" in
388 assign buildpath $releasepath/_build
391 run $srcpath/configure
394 info "VPATH build disabled"
395 assign buildpath $srcpath
401 info "no configure script"
402 assign buildpath $srcpath
407 ## Discover the release name.
408 cat >find-distdir.mk <<'EOF'
411 @bash -c 'echo >&$(fd) $(distdir)'
414 $({ $make -f find-distdir.mk print-distdir fd=$t >/dev/null 2>&1; } {t}>&1)
416 ## Get a tarball distribution.
419 run $make $makeopts distcheck
422 run $make $makeopts dist
430 if ! tar tf $buildpath/$distdir.tar.gz 2>/dev/null | grep -q RELEASE
432 fail "missing RELEASE file in distribution"
437 run mv $buildpath/$distdir.tar.gz .
440 run gpg -u$signkey -ab -o$distdir.tar.gz.gpg $distdir.tar.gz
444 ## Maybe build the Debian packages.
447 run tar xvfz $distdir.tar.gz
451 cat - debian/changelog >debian/changelog.new <<EOF
452 $debsrc ($debversion) experimental; urgency=low
454 * Hacking in process, not intended for release.
456 -- $debname <$debemail> $now
459 mv debian/changelog.new debian/changelog
462 sbuildargs=$sbuildsrv
463 case $sbuild,$build in
465 case $sign in yes) sbuildargs="-k$signkey $sbuildargs" ;; esac
468 if [ -d $toppath/dist-$build.pkgs ]; then
469 sbuildargs="-p$toppath/dist-$build.pkgs $sbuildargs"
473 case "$sbuild,$test, $DEB_BUILD_OPTIONS " in
474 yes,no,*) sbuildargs="-T $sbuildargs" ;;
477 DEB_BUILD_OPTIONS=${DEB_BUILD_OPTIONS+"$DEB_BUILD_OPTIONS nocheck"}
480 case $sbuild,$build,$sign in
481 yes,*) run mdw-sbuild $sbuildargs ;;
482 no,release,yes) run dpkg-buildpackage -k$signkey ;;
483 no,*) run dpkg-buildpackage -us -uc ;;
488 ## Maybe upload Debian packages.
490 case "$upload,$build" in
491 yes,test) info "Test build: not uploading." ;;
492 yes,release) run rsync $distdir.tar.gz $distdir.tar.gz.gpg $uploadpath ;;
494 case "$debian,$upload,$dput,$build" in
495 yes,yes,yes,release) run dput -f $dputtarget *.changes ;;
501 rm -rf $releasepath/$distdir
502 rm -rf $releasepath/_source
503 rm -rf $releasepath/_build
510 ###----- That's all, folks --------------------------------------------------