* apt-pkg/deb/dpkgpm.cc:
[ntk/apt.git] / test / integration / framework
1 #!/bin/sh -- # no runable script, just for vi
2
3 # we all like colorful messages
4 CERROR="\e[1;31m" # red
5 CWARNING="\e[1;33m" # yellow
6 CMSG="\e[1;32m" # green
7 CINFO="\e[1;96m" # light blue
8 CDEBUG="\e[1;94m" # blue
9 CNORMAL="\e[0;39m" # default system console color
10 CDONE="\e[1;32m" # green
11 CPASS="\e[1;32m" # green
12 CFAIL="\e[1;31m" # red
13 CCMD="\e[1;35m" # pink
14
15 msgdie() { echo "${CERROR}E: $1${CNORMAL}" >&2; exit 1; }
16 msgwarn() { echo "${CWARNING}W: $1${CNORMAL}" >&2; }
17 msgmsg() { echo "${CMSG}$1${CNORMAL}" >&2; }
18 msginfo() { echo "${CINFO}I: $1${CNORMAL}" >&2; }
19 msgdebug() { echo "${CDEBUG}D: $1${CNORMAL}" >&2; }
20 msgdone() { echo "${CDONE}DONE${CNORMAL}" >&2; }
21 msgnwarn() { echo -n "${CWARNING}W: $1${CNORMAL}" >&2; }
22 msgnmsg() { echo -n "${CMSG}$1${CNORMAL}" >&2; }
23 msgninfo() { echo -n "${CINFO}I: $1${CNORMAL}" >&2; }
24 msgndebug() { echo -n "${CDEBUG}D: $1${CNORMAL}" >&2; }
25 msgtest() { echo -n "${CINFO}$1 ${CCMD}$(echo "$2" | sed -e 's/^aptc/apt-c/' -e 's/^aptg/apt-g/' -e 's/^aptf/apt-f/')${CINFO} …${CNORMAL} " >&2; }
26 msgpass() { echo "${CPASS}PASS${CNORMAL}" >&2; }
27 msgskip() { echo "${CWARNING}SKIP${CNORMAL}" >&2; }
28 msgfail() { echo "${CFAIL}FAIL${CNORMAL}" >&2; }
29
30 # enable / disable Debugging
31 MSGLEVEL=${MSGLEVEL:-3}
32 if [ $MSGLEVEL -le 0 ]; then
33 msgdie() { true; }
34 fi
35 if [ $MSGLEVEL -le 1 ]; then
36 msgwarn() { true; }
37 msgnwarn() { true; }
38 fi
39 if [ $MSGLEVEL -le 2 ]; then
40 msgmsg() { true; }
41 msgnmsg() { true; }
42 fi
43 if [ $MSGLEVEL -le 3 ]; then
44 msginfo() { true; }
45 msgninfo() { true; }
46 fi
47 if [ $MSGLEVEL -le 4 ]; then
48 msgdebug() { true; }
49 msgndebug() { true; }
50 fi
51 msgdone() {
52 if [ "$1" = "debug" -a $MSGLEVEL -le 4 ] ||
53 [ "$1" = "info" -a $MSGLEVEL -le 3 ] ||
54 [ "$1" = "msg" -a $MSGLEVEL -le 2 ] ||
55 [ "$1" = "warn" -a $MSGLEVEL -le 1 ] ||
56 [ "$1" = "die" -a $MSGLEVEL -le 0 ]; then
57 true;
58 else
59 echo "${CDONE}DONE${CNORMAL}" >&2;
60 fi
61 }
62
63 runapt() {
64 msgdebug "Executing: ${CCMD}$*${CDEBUG} "
65 if [ -f ./aptconfig.conf ]; then
66 APT_CONFIG=aptconfig.conf LD_LIBRARY_PATH=${BUILDDIRECTORY} ${BUILDDIRECTORY}/$*
67 else
68 LD_LIBRARY_PATH=${BUILDDIRECTORY} ${BUILDDIRECTORY}/$*
69 fi
70 }
71 aptconfig() { runapt apt-config $*; }
72 aptcache() { runapt apt-cache $*; }
73 aptget() { runapt apt-get $*; }
74 aptftparchive() { runapt apt-ftparchive $*; }
75
76 setupenvironment() {
77 TMPWORKINGDIRECTORY=$(mktemp -d)
78 local TESTDIR=$(readlink -f $(dirname $0))
79 msgninfo "Preparing environment for ${CCMD}$0${CINFO} in ${TMPWORKINGDIRECTORY}… "
80 BUILDDIRECTORY="${TESTDIR}/../../build/bin"
81 test -x "${BUILDDIRECTORY}/apt-get" || msgdie "You need to build tree first"
82 local OLDWORKINGDIRECTORY=$(pwd)
83 # trap "cd /; rm -rf $TMPWORKINGDIRECTORY; cd $OLDWORKINGDIRECTORY" 0 HUP INT QUIT ILL ABRT FPE SEGV PIPE TERM
84 cd $TMPWORKINGDIRECTORY
85 mkdir rootdir aptarchive keys
86 cd rootdir
87 mkdir -p etc/apt/apt.conf.d etc/apt/sources.list.d etc/apt/trusted.gpg.d etc/apt/preferences.d var/cache
88 mkdir -p var/log/apt var/lib/apt
89 mkdir -p var/lib/dpkg/info var/lib/dpkg/updates var/lib/dpkg/triggers
90 local STATUSFILE=$(echo "$(basename $0)" | sed 's/^test-/status-/')
91 if [ -f "${TESTDIR}/${STATUSFILE}" ]; then
92 cp "${TESTDIR}/${STATUSFILE}" var/lib/dpkg/status
93 else
94 touch var/lib/dpkg/status
95 fi
96 touch var/lib/dpkg/available
97 mkdir -p usr/lib/apt
98 ln -s ${BUILDDIRECTORY}/methods usr/lib/apt/methods
99 cd ..
100 local PACKAGESFILE=$(echo "$(basename $0)" | sed 's/^test-/Packages-/')
101 if [ -f "${TESTDIR}/${PACKAGESFILE}" ]; then
102 cp "${TESTDIR}/${PACKAGESFILE}" aptarchive/Packages
103 else
104 touch aptarchive/Packages
105 fi
106 cp $(find $TESTDIR -name '*.pub' -o -name '*.sec') keys/
107 ln -s ${TMPWORKINGDIRECTORY}/keys/joesixpack.pub rootdir/etc/apt/trusted.gpg.d/joesixpack.gpg
108 echo "Dir \"${TMPWORKINGDIRECTORY}/rootdir\";" > aptconfig.conf
109 echo "Debug::NoLocking \"true\";" >> aptconfig.conf
110 echo "APT::Get::Show-User-Simulation-Note \"false\";" >> aptconfig.conf
111 echo "Dir::Bin::dpkg \"fakeroot\";" >> aptconfig.conf
112 echo "DPKG::options:: \"dpkg\";" >> aptconfig.conf
113 echo "DPKG::options:: \"--root=${TMPWORKINGDIRECTORY}/rootdir\";" >> aptconfig.conf
114 echo "DPKG::options:: \"--force-not-root\";" >> aptconfig.conf
115 echo "DPKG::options:: \"--force-bad-path\";" >> aptconfig.conf
116 echo "DPKG::options:: \"--log=${TMPWORKINGDIRECTORY}/rootdir/var/log/dpkg.log\";" >> aptconfig.conf
117 export LC_ALL=C
118 msgdone "info"
119 }
120
121 configarchitecture() {
122 local CONFFILE=rootdir/etc/apt/apt.conf.d/01multiarch.conf
123 echo "APT::Architecture \"$1\";" > $CONFFILE
124 shift
125 while [ -n "$1" ]; do
126 echo "APT::Architectures:: \"$1\";" >> $CONFFILE
127 shift
128 done
129 }
130
131 setupsimplenativepackage() {
132 local NAME="$1"
133 local ARCH="$2"
134 local VERSION="$3"
135 local RELEASE="${4:-unstable}"
136 local DEPENDENCIES="$5"
137 local DESCRIPTION="$6"
138 local SECTION="${7:-others}"
139 local DISTSECTION
140 if [ "$SECTION" = "$(echo "$SECTION" | cut -d'/' -f 2)" ]; then
141 DISTSECTION="main"
142 else
143 DISTSECTION="$(echo "$SECTION" | cut -d'/' -f 1)"
144 fi
145 local BUILDDIR=incoming/${NAME}-${VERSION}
146 mkdir -p ${BUILDDIR}/debian/source
147 cd ${BUILDDIR}
148 echo "* most suckless software product ever" > FEATURES
149 test -e debian/copyright || echo "Copyleft by Joe Sixpack $(date +%Y)" > debian/copyright
150 test -e debian/changelog || echo "$NAME ($VERSION) $RELEASE; urgency=low
151
152 * Initial release
153
154 -- Joe Sixpack <joe@example.org> $(date -R)" > debian/changelog
155 test -e debian/control || echo "Source: $NAME
156 Section: $SECTION
157 Priority: optional
158 Maintainer: Joe Sixpack <joe@example.org>
159 Build-Depends: debhelper (>= 7)
160 Standards-Version: 3.9.1
161
162 Package: $NAME
163 Architecture: $ARCH" > debian/control
164 test -z "$DEPENDENCIES" || echo "$DEPENDENCIES" >> debian/control
165 if [ -z "$DESCRIPTION" ]; then
166 echo "Description: an autogenerated dummy ${NAME}=${VERSION}/${RELEASE}
167 If you find such a package installed on your system,
168 YOU did something horribly wrong! They are autogenerated
169 und used only by testcases for APT and surf no other propose…" >> debian/control
170 else
171 echo "Description: $DESCRIPTION" >> debian/control
172 fi
173 test -e debian/compat || echo "7" > debian/compat
174 test -e debian/source/format || echo "3.0 (native)" > debian/source/format
175 test -e debian/rules || cp /usr/share/doc/debhelper/examples/rules.tiny debian/rules
176 cd - > /dev/null
177 }
178
179 buildsimplenativepackage() {
180 local NAME="$1"
181 local ARCH="$2"
182 local VERSION="$3"
183 local RELEASE="${4:-unstable}"
184 local DEPENDENCIES="$5"
185 local DESCRIPTION="$6"
186 local SECTION="${7:-others}"
187 local DISTSECTION
188 if [ "$SECTION" = "$(echo "$SECTION" | cut -d'/' -f 2)" ]; then
189 DISTSECTION="main"
190 else
191 DISTSECTION="$(echo "$SECTION" | cut -d'/' -f 1)"
192 fi
193 setupsimplenativepackage "$NAME" "$ARCH" "$VERSION" "$RELEASE" "$DEPENDENCIES" "$DESCRIPTION" "$SECTION"
194 buildpackage "incoming/${NAME}-${VERSION}" "$RELEASE" "$DISTSECTION"
195 rm -rf "incoming/${NAME}-${VERSION}"
196 }
197
198 buildpackage() {
199 local BUILDDIR=$1
200 local RELEASE=$2
201 local SECTION=$3
202 msgninfo "Build package $(echo "$BUILDDIR" | grep -o '[^/]*$') for ${RELEASE} in ${SECTION}… "
203 cd $BUILDDIR
204 if [ "$ARCH" = "all" ]; then
205 ARCH="$(dpkg-architecture -qDEB_HOST_ARCH 2> /dev/null)"
206 fi
207 local BUILT="$(dpkg-buildpackage -uc -us -a$ARCH 2> /dev/null)"
208 local PKGS="$( echo "$BUILT" | grep '^dpkg-deb: building package' | cut -d'/' -f 2 | sed -e "s#'\.##")"
209 local SRCS="$( echo "$BUILT" | grep '^dpkg-source: info: building' | grep -o '[a-z0-9._-]*$')"
210 cd - > /dev/null
211 for PKG in $PKGS; do
212 echo "pool/${PKG}" >> ${TMPWORKINGDIRECTORY}/incoming/${RELEASE}.${SECTION}.pkglist
213 done
214 for SRC in $SRCS; do
215 echo "pool/${SRC}" >> ${TMPWORKINGDIRECTORY}/incoming/${RELEASE}.${SECTION}.srclist
216 done
217 msgdone "info"
218 }
219
220 buildaptarchive() {
221 if [ -d incoming ]; then
222 buildaptarchivefromincoming $*
223 else
224 buildaptarchivefromfiles $*
225 fi
226 }
227
228 createaptftparchiveconfig() {
229 local ARCHS="$(find pool/ -name '*.deb' | grep -oE '_[a-z0-9-]+\.deb$' | sort | uniq | sed -e '/^_all.deb$/ d' -e 's#^_\([a-z0-9-]*\)\.deb$#\1#' | tr '\n' ' ')"
230 echo -n 'Dir {
231 ArchiveDir "' >> ftparchive.conf
232 echo -n $(readlink -f .) >> ftparchive.conf
233 echo -n '";
234 CacheDir "' >> ftparchive.conf
235 echo -n $(readlink -f ..) >> ftparchive.conf
236 echo -n '";
237 FileListDir "' >> ftparchive.conf
238 echo -n $(readlink -f pool/) >> ftparchive.conf
239 echo -n '";
240 };
241 Default {
242 Packages::Compress ". gzip bzip2 lzma";
243 Sources::Compress ". gzip bzip2 lzma";
244 Contents::Compress ". gzip bzip2 lzma";
245 };
246 TreeDefault {
247 Directory "pool/";
248 SrcDirectory "pool/";
249 };
250 APT {
251 FTPArchive {
252 Release {
253 Origin "joesixpack";
254 Label "apttestcases";
255 Suite "unstable";
256 Description "repository with dummy packages";
257 Architectures "' >> ftparchive.conf
258 echo -n "$ARCHS" >> ftparchive.conf
259 echo 'source";
260 };
261 };
262 };' >> ftparchive.conf
263 for DIST in $(find ./pool/ -maxdepth 1 -name '*.pkglist' -type f | cut -d'/' -f 3 | cut -d'.' -f 1 | sort | uniq); do
264 echo -n 'tree "dists/' >> ftparchive.conf
265 echo -n "$DIST" >> ftparchive.conf
266 echo -n '" {
267 Architectures "' >> ftparchive.conf
268 echo -n "$ARCHS" >> ftparchive.conf
269 echo -n 'source";
270 FileList "' >> ftparchive.conf
271 echo -n "${DIST}.\$(SECTION).pkglist" >> ftparchive.conf
272 echo -n '";
273 SourceFileList "' >> ftparchive.conf
274 echo -n "${DIST}.\$(SECTION).srclist" >> ftparchive.conf
275 echo -n '";
276 Sections "' >> ftparchive.conf
277 echo -n "$(find ./pool/ -maxdepth 1 -name "${DIST}.*.pkglist" -type f | cut -d'/' -f 3 | cut -d'.' -f 2 | sort | uniq | tr '\n' ' ')" >> ftparchive.conf
278 echo '";
279 };' >> ftparchive.conf
280 done
281 }
282
283 buildaptftparchivedirectorystructure() {
284 local DISTS="$(grep -i '^tree ' ftparchive.conf | cut -d'/' -f 2 | sed -e 's#".*##')"
285 for DIST in $DISTS; do
286 local SECTIONS="$(grep -i -A 5 "dists/$DIST" ftparchive.conf | grep -i 'Sections' | cut -d'"' -f 2)"
287 for SECTION in $SECTIONS; do
288 local ARCHS="$(grep -A 5 "dists/$DIST" ftparchive.conf | grep Architectures | cut -d'"' -f 2 | sed -e 's#source##')"
289 for ARCH in $ARCHS; do
290 mkdir -p dists/${DIST}/${SECTION}/binary-${ARCH}
291 done
292 mkdir -p dists/${DIST}/${SECTION}/source
293 mkdir -p dists/${DIST}/${SECTION}/i18n
294 done
295 done
296 }
297
298 buildaptarchivefromincoming() {
299 msginfo "Build APT archive for ${CCMD}$0${CINFO} based on incoming packages…"
300 cd aptarchive
301 [ -e pool ] || ln -s ../incoming pool
302 [ -e ftparchive.conf ] || createaptftparchiveconfig
303 [ -e dists ] || buildaptftparchivedirectorystructure
304 msgninfo "\tGenerate Packages, Sources and Contents files… "
305 aptftparchive -qq generate ftparchive.conf
306 msgdone "info"
307 msgninfo "\tGenerate Release files… "
308 for dir in $(find ./dists -mindepth 1 -maxdepth 1 -type d); do
309 aptftparchive -qq release $dir -o APT::FTPArchive::Release::Codename="$(echo "$dir" | cut -d'/' -f 3)" | sed -e '/0 Release$/ d' > $dir/Release # remove the self reference
310 done
311 cd - > /dev/null
312 msgdone "info"
313 }
314
315 buildaptarchivefromfiles() {
316 msginfo "Build APT archive for ${CCMD}$0${CINFO} based on prebuild files…"
317 cd aptarchive
318 if [ -f Packages ]; then
319 msgninfo "\tPackages file… "
320 cat Packages | gzip > Packages.gz
321 cat Packages | bzip2 > Packages.bz2
322 cat Packages | lzma > Packages.lzma
323 msgdone "info"
324 fi
325 if [ -f Sources ]; then
326 msgninfo "\tSources file… "
327 cat Sources | gzip > Sources.gz
328 cat Sources | bzip2 > Sources.bz2
329 cat Sources | lzma > Sources.lzma
330 msgdone "info"
331 fi
332 msgninfo "\tRelease file… "
333 aptftparchive -qq release . | sed -e '/0 Release$/ d' > Release # remove the self reference
334 msgdone "info"
335 cd ..
336 }
337
338 setupdistsaptarchive() {
339 local APTARCHIVE=$(readlink -f ./aptarchive)
340 rm -f root/etc/apt/sources.list.d/apt-test-*-deb.list
341 rm -f root/etc/apt/sources.list.d/apt-test-*-deb-src.list
342 for DISTS in $(find ./aptarchive/dists/ -mindepth 1 -maxdepth 1 -type d | cut -d'/' -f 4); do
343 SECTIONS=$(find ./aptarchive/dists/${DISTS}/ -mindepth 1 -maxdepth 1 -type d | cut -d'/' -f 5 | tr '\n' ' ')
344 msgninfo "\tadd deb and deb-src sources.list lines for ${CCMD}${DISTS} ${SECTIONS}${CINFO}… "
345 echo "deb file://$APTARCHIVE $DISTS $SECTIONS" > rootdir/etc/apt/sources.list.d/apt-test-${DISTS}-deb.list
346 echo "deb-src file://$APTARCHIVE $DISTS $SECTIONS" > rootdir/etc/apt/sources.list.d/apt-test-${DISTS}-deb-src.list
347 msgdone "info"
348 done
349 }
350
351 setupflataptarchive() {
352 local APTARCHIVE=$(readlink -f ./aptarchive)
353 if [ -f ${APTARCHIVE}/Packages ]; then
354 msgninfo "\tadd deb sources.list line… "
355 echo "deb file://$APTARCHIVE /" > rootdir/etc/apt/sources.list.d/apt-test-archive-deb.list
356 msgdone "info"
357 else
358 rm -f rootdir/etc/apt/sources.list.d/apt-test-archive-deb.list
359 fi
360 if [ -f ${APTARCHIVE}/Sources ]; then
361 msgninfo "\tadd deb-src sources.list line… "
362 echo "deb-src file://$APTARCHIVE /" > rootdir/etc/apt/sources.list.d/apt-test-archive-deb-src.list
363 msgdone "info"
364 else
365 rm -f rootdir/etc/apt/sources.list.d/apt-test-archive-deb-src.list
366 fi
367 }
368
369 setupaptarchive() {
370 buildaptarchive
371 if [ -e aptarchive/dists ]; then
372 setupdistsaptarchive
373 else
374 setupflataptarchive
375 fi
376 msgninfo "\tSign archive with Joe Sixpack key… "
377 for RELEASE in $(find aptarchive/ -name Release); do
378 gpg --no-default-keyring --secret-keyring ./keys/joesixpack.sec --keyring ./keys/joesixpack.pub --default-key Joe -abs -o ${RELEASE}.gpg ${RELEASE}
379 done
380 msgdone "info"
381 msgninfo "\tSync APT's cache with the archive… "
382 aptget update -qq
383 msgdone "info"
384 }
385
386 diff() {
387 local DIFFTEXT="$($(which diff) -u $* | sed -e '/^---/ d' -e '/^+++/ d' -e '/^@@/ d')"
388 if [ -n "$DIFFTEXT" ]; then
389 echo
390 echo "$DIFFTEXT"
391 return 1
392 else
393 return 0
394 fi
395 }
396
397 testfileequal() {
398 local FILE="$1"
399 shift
400 msgtest "Test for correctness of file" "$FILE"
401 if [ -z "$*" ]; then
402 echo -n "" | diff $FILE - && msgpass || msgfail
403 else
404 echo "$*" | diff $FILE - && msgpass || msgfail
405 fi
406 }
407
408 testequal() {
409 local COMPAREFILE=$(mktemp)
410 echo "$1" > $COMPAREFILE
411 shift
412 msgtest "Test for equality of" "$*"
413 $* 2>&1 | diff $COMPAREFILE - && msgpass || msgfail
414 rm $COMPAREFILE
415 }
416
417 testequalor2() {
418 local COMPAREFILE1=$(mktemp)
419 local COMPAREFILE2=$(mktemp)
420 local COMPAREAGAINST=$(mktemp)
421 echo "$1" > $COMPAREFILE1
422 echo "$2" > $COMPAREFILE2
423 shift 2
424 msgtest "Test for equality OR of" "$*"
425 $* 2>&1 1> $COMPAREAGAINST
426 (diff $COMPAREFILE1 $COMPAREAGAINST 1> /dev/null ||
427 diff $COMPAREFILE2 $COMPAREAGAINST 1> /dev/null) && msgpass ||
428 ( echo "\n${CINFO}Diff against OR 1${CNORMAL}" "$(diff $COMPAREFILE1 $COMPAREAGAINST)" \
429 "\n${CINFO}Diff against OR 2${CNORMAL}" "$(diff $COMPAREFILE2 $COMPAREAGAINST)" &&
430 msgfail )
431 rm $COMPAREFILE1 $COMPAREFILE2 $COMPAREAGAINST
432 }
433
434 testshowvirtual() {
435 local VIRTUAL="N: Can't select versions from package '$1' as it purely virtual"
436 local PACKAGE="$1"
437 shift
438 while [ -n "$1" ]; do
439 VIRTUAL="${VIRTUAL}
440 N: Can't select versions from package '$1' as it purely virtual"
441 PACKAGE="${PACKAGE} $1"
442 shift
443 done
444 msgtest "Test for virtual packages" "apt-cache show $PACKAGE"
445 VIRTUAL="${VIRTUAL}
446 N: No packages found"
447 local COMPAREFILE=$(mktemp)
448 local ARCH=$(dpkg-architecture -qDEB_HOST_ARCH_CPU)
449 eval `apt-config shell ARCH APT::Architecture`
450 echo "$VIRTUAL" | sed -e "s/:$ARCH//" -e 's/:all//' > $COMPAREFILE
451 aptcache show $PACKAGE 2>&1 | diff $COMPAREFILE - && msgpass || msgfail
452 rm $COMPAREFILE
453 }
454
455 testnopackage() {
456 msgtest "Test for non-existent packages" "apt-cache show $*"
457 local SHOWPKG="$(aptcache show $* 2>&1 | grep '^Package: ')"
458 if [ -n "$SHOWPKG" ]; then
459 echo
460 echo "$SHOWPKG"
461 msgfail
462 return 1
463 fi
464 msgpass
465 }