mkfedora has gone - now embedded into build.common
[build.git] / build.common
1 # -*-Shell-script-*-
2 #
3 # Common functions for build scripts used by various packages
4 # incorporated (e.g., build, myplc, myplc-devel, vserver-reference)
5 #
6 # Marc E. Fiuczynski <mef@cs.princeton.edu>
7 # Copyright (C) 2007 The Trustees of Princeton University
8 #
9 # $Id$
10 #
11
12 function pl_getDefaultDistro() {
13     # FC4 is currently the default release
14     echo "Fedora"
15     return 0
16 }
17
18 function pl_getDefaultRelease() {
19     # FC4 is currently the default release
20     echo "4"
21     return 0
22 }
23
24 function pl_getDistro() {
25     defaultDistro=$(pl_getDefaultDistro)
26     if [ -f "/etc/redhat-release" ] ; then
27         distro=$(awk ' { print $1 } ' /etc/redhat-release)
28     else
29         distro=$defaultDistro
30     fi
31     echo "$distro"
32     return 0
33 }
34
35 function pl_getRelease() {
36     defaultRelease=$(pl_getDefaultRelease)
37     if [ -f "/etc/redhat-release" ] ; then
38         release=$(awk ' { if ($1=="Fedora" && $2=="Core") print $4 ; if (($1=="Fedora" && $2!="Core")||$1=="CentOS") print $3 } ' /etc/redhat-release)
39         [ $release -lt $defaultRelease ] && release=$defaultRelease
40     else
41         release=$defaultRelease
42     fi
43     echo "$release"
44     return 0
45 }
46
47 # vserver expects something like fc4 or f7
48 # for fedora and centos only for now
49 function pl_getReleaseName () {
50     distro=$1; shift
51     release=$1; shift
52     case $distro in
53         [Ff]edora*)
54             if [ "$release" -le 6 ] ; then
55                 releasename=fc$release
56             else
57                 releasename=f$release
58             fi ;;
59         [Cc]entOS*)
60             if [ "$release" = "4.5" ] ; then
61             # centos 4.5 is just centos4 + enhancements
62                 releasename=centos4
63             else
64                 releasename=centos$release
65             fi ;;
66         *)
67             releasename="unknown-name-for-${pl_DISTRO}-please-edit-build.common"
68             echo 1>&2 "build.common: WARNING - releasename not set for distro=$distro" 
69             return 1
70             ;;
71     esac
72     echo "$releasename"
73     return 0
74 }
75
76 # figure out which redhat distro we are using (fedora, centos, redhat)
77 pl_DISTRO=$(pl_getDistro)
78
79 # select basearch of the host devel environment - protected for macos for local tests
80 pl_DISTRO_ARCH=$(uname -i 2>/dev/null|| echo unknownarch)
81
82 # let mkfedora select one of its mirrors
83 pl_DISTRO_URL=""
84
85 # the release number (plain number)
86 pl_DISTRO_RELEASE=$(pl_getRelease)
87
88 # the release name - for vserver build - like fc4, f8 or centos4
89 pl_DISTRO_NAME=$(pl_getReleaseName $pl_DISTRO $pl_DISTRO_RELEASE)
90
91 # get path to appropriate yumgroups.xml file
92 # Thierry: quick & dirty improvement 
93 # this file is updated by the toplevel build, from groups/<pldistro>.xml
94 pl_DISTRO_YUMGROUPS="../../../RPMS/yumgroups.xml"
95
96 function pl_process_fedora_options () {
97     # Get options
98     shiftcount=0
99     while getopts "l:r:a:h" opt ; do
100         case $opt in
101             l)
102                 pl_DISTRO_URL=$OPTARG
103                 let shiftcount=$shiftcount+2
104                 ;;
105             r)
106                 pl_DISTRO_RELEASE=$OPTARG
107                 let shiftcount=$shiftcount+2
108                 ;;
109             a)
110                 pl_DISTRO_ARCH=$OPTARG
111                 let shiftcount=$shiftcount+2
112                 ;;
113             h|*)
114                 echo "Usage: $0 [OPTION]..."
115                 echo "  -l url          distro mirror location (default: $pl_DISTRO_URL)"
116                 echo "  -r release      distro release number (default: $pl_DISTRO_RELEASE)"
117                 echo "  -a arch         distro architecture (default: $pl_DISTRO_ARCH)"
118                 echo "where distro can be either fedora, centos, or redhat"
119                 echo "  -h              This message"
120                 exit 1
121                 ;;
122         esac
123     done
124     return $shiftcount
125 }
126
127 ######################################## handling a root image
128 function pl_makedevs() {
129     vroot=$1
130     # Clean ${vroot}/dev, but only when ${vroot}!=""
131     [ -n $vroot ] && rm -rf $vroot/dev
132     
133     mkdir -p $vroot/dev
134     mknod -m 666 $vroot/dev/null c 1 3
135     mknod -m 666 $vroot/dev/zero c 1 5
136     mknod -m 666 $vroot/dev/full c 1 7
137     mknod -m 644 $vroot/dev/random c 1 8
138     mknod -m 644 $vroot/dev/urandom c 1 9
139     mknod -m 666 $vroot/dev/tty c 5 0
140     mknod -m 666 $vroot/dev/ptmx c 5 2
141     # For bash command substitution
142     ln -nsf ../proc/self/fd $vroot/dev/fd
143
144     # For df and linuxconf
145     touch $vroot/dev/hdv1
146
147     # For pseudo ttys
148     mkdir -p $vroot/dev/pts
149
150     # (Might have to remove the following for vserver-reference.)
151
152     # for tmpfs mount
153     mkdir -p $vroot/dev/shm
154
155     # For TUN/TAP
156     mkdir -p $vroot/dev/net
157     mknod -m 600 $vroot/dev/net/tun c 10 200
158
159     # For mkinitrd (in case a kernel is being installed)
160     # As well for loop back mounting within a vserver. 
161     for i in $(seq 0 255) ; do
162         mknod -m 640 $vroot/dev/loop$i b 7 $i
163     done
164 }
165
166 function mkfedora_usage()
167 {
168     echo "Usage: mkfedora [OPTION]... [basedir]"
169     echo "      -l url          Fedora mirror location. Defaults to try:"
170     for mirror in "${mirrors[@]}" ; do
171         echo "                  $mirror"
172     done
173     echo "      -f pkgsfile     use this pkgs file for packages, groups, excludes.."
174     echo "      -k              Exclude kernel* packages from all repositories except bootstrap"
175     echo "      -v              Be verbose"
176     echo "      -h              This message"
177     echo " target selection (defaults based on current build vserver)"
178     echo "      -r release      Fedora release number (default: $releasever)"
179     echo "      -a arch         Fedora architecture (default: $basearch)"
180     echo " legacy (use -f instead)"
181     echo "      -g group1 -g group2 ..."
182     echo "                      Yumgroups to install (default: none)"
183     echo "      -p package1 -p package2 ..."
184     echo "                      Additional packages to install (default: none)"
185     echo "      -x package1 -x package2 ..."
186     echo "                      Packages to exclude (default: none)"
187     exit 1
188 }
189
190 function mkfedora () {
191
192 echo "* Entering mkfedora " "$@"
193
194 # Verbosity
195 verbose=0
196
197 # Default yum repositories to try
198 mirrors=(
199 file:///data/fedora
200 http://localhost/fedora
201 http://build.planet-lab.org/fedora
202 http://coblitz.codeen.org/coblitz.planet-lab.org/pub/fedora
203 ftp://mirror.cs.princeton.edu/pub/mirrors/fedora
204 ftp://mirror.stanford.edu/pub/mirrors/fedora
205 ftp://rpmfind.net/linux/fedora
206 )
207
208 # Release and architecture to install
209 releasever=7
210 basearch=i386
211
212 # Yum groups to install
213 groups=()
214
215 # Packages to install
216 packages=()
217
218 # Packages to exclude
219 exclude=()
220
221 # Exclude kernel* (and related) packages from all repositories except bootstrap
222 exclude_kernel=
223
224 # PlanetLab development environment
225 if [ -f /etc/planetlab/plc_config ] ; then
226     . /etc/planetlab/plc_config
227     if [ -n "$PLC_DEVEL_FEDORA_URL" ] ; then
228         echo "* mkfedora : setting mirrors from /etc/planetlab/config"
229         mirrors=($PLC_DEVEL_FEDORA_URL)
230     fi
231 fi
232
233 # Get options
234 while getopts "l:r:a:g:p:x:f:kvh" opt ; do
235     case $opt in
236         l)
237             if echo $OPTARG | grep -q -i '^\(file\|http[s]*\)://' ; then
238                 mirrors=($OPTARG)
239             else
240                 mirrors=(file://$OPTARG)
241             fi
242             ;;
243         r)
244             releasever=$OPTARG
245             ;;
246         a)
247             basearch=$OPTARG
248             ;;
249         g)
250             groups[${#groups[*]}]="$OPTARG"
251             ;;
252         p)
253             packages[${#packages[*]}]="$OPTARG"
254             ;;
255         x)
256             exclude[${#exclude[*]}]="$OPTARG"
257             ;;
258         f)
259             pkgsfile=$OPTARG
260             ;;
261         k)
262             exclude_kernel="exclude=kernel* ulogd iptables"
263             ;;
264         v)
265             verbose=1
266             set -x
267             ;;
268         h|*)
269             mkfedora_usage
270             ;;
271     esac
272 done
273
274 shift $(($OPTIND - 1))
275 if [ ! -d "$1" ] ; then
276     mkfedora_usage
277 fi
278
279 vroot=$(cd $1 && pwd -P)
280
281 if [ $UID -ne 0 ] ; then
282     echo "Error: You must run this script as root."
283     exit 1
284 fi
285
286 function mkfedora_fetch ()
287 {
288     curl --fail --silent --max-time 60 "$1"
289 }
290
291 # hard to find two mirrors with a similar layout
292 # set list of attempted locations according to releasever
293 if [ $releasever -ge 7 ] ; then
294     attempts="
295 linux/releases/$releasever/Everything/$basearch/os
296 $releasever/Everything/$basearch/os
297 core/$releasever/Everything/$basearch/os
298 linux/core/$releasever/$basearch/os
299 "
300 else
301     attempts="
302 linux/core/$releasever/$basearch/os 
303 core/$releasever/$basearch/os 
304 $releasever/$basearch/os
305 "
306 fi
307
308 echo "$0: candidate mirrors"
309 for mirror in "${mirrors[@]}" ; do
310     echo "* candidate mirror $mirror"
311 done
312 baseurl=""
313 for mirror in "${mirrors[@]}" ; do
314     for attempt in $attempts; do 
315         attempturl=$mirror/$attempt
316         if mkfedora_fetch $attempturl/repodata/repomd.xml >/dev/null ; then
317             baseurl=$attempturl
318             break 2
319         fi
320     done
321 done
322
323 if [ -z "$baseurl" ] ; then
324     echo "Error: $releasever/$basearch/os/repodata/repomd.xml"
325     echo "       could not be found in any of the following locations:"
326     echo
327     for mirror in ${mirrors[@]} ; do
328         for attempt in $attempts ; do
329             echo $mirror/$attempt
330         done
331     done
332     echo
333     mkfedora_usage
334 fi
335
336 # this used to be for when mkfedora was called within myplc 
337 # that is, when the bootcd was recomputed from scratch
338 ### exec 3>&1
339 ### exec 4>&2
340 ### if [ $verbose -eq 0 ] ; then
341 ###     exec 1>/dev/null
342 ###     exec 2>/dev/null
343 ### fi
344
345 # Do not tolerate errors
346 set -e
347
348 ## make rpms ignore installing stuff to special fs entries like /proc
349 # Because of https://bugzilla.redhat.com/bugzilla/show_bug.cgi?id=52725
350 # you have to use at least one language beside 'C'
351 # Prevent all locales from being installed in reference image
352 mkdir -p $vroot/etc/rpm
353 cat > $vroot/etc/rpm/macros <<EOF
354 %_install_langs C:en_US:en
355 %_netsharedpath /proc:/dev/pts
356 %_excludedocs 1
357 %__file_context_path /dev/null
358 EOF
359
360 # Trick rpm and yum, who read the real root /etc/rpm/macros file
361 # rather than the one installed in the reference image, despite what
362 # you might expect the --root and --installroot options to mean. Both
363 # programs always read $HOME/.rpmmacros.
364 export HOME=$vroot/tmp
365 mkdir -p $vroot/tmp
366 cp $vroot/etc/rpm/macros $vroot/tmp/.rpmmacros
367
368 ### Checking whether this is really needed
369 ### # copy to the vserver's rpm macros
370 ### # xxx fixme - this must be reviewed once we get the stuff running
371 ### cp $vroot/etc/rpm/macros /etc/rpm/macros 
372
373 # Mount in reference image
374 mount -t devpts none $vroot/dev/pts
375 mount -t tmpfs none $vroot/dev/shm
376 mkdir -p $vroot/proc
377 mount -t proc none $vroot/proc
378
379 function mkfedora_cleanup ()
380 {
381     umount -l $vroot/proc
382     umount -l $vroot/dev/shm
383     umount -l $vroot/dev/pts
384 }
385
386 # Clean up before exiting if anything goes wrong
387 trap "mkfedora_cleanup" ERR INT
388
389 # Create a /var/lib dirs for yum & rpm
390 mkdir -p $vroot/var/lib/yum
391 mkdir -p $vroot/var/lib/rpm
392 mkdir -p $vroot/usr/share/info
393
394 # Create a dummy /etc/fstab in reference image
395 mkdir -p $vroot/etc
396 cat >$vroot/etc/fstab <<EOF
397 # This fake fstab exists only to please df and linuxconf.
398 /dev/hdv1       /       ext2    defaults        1 1
399 EOF
400 cp $vroot/etc/fstab $vroot/etc/mtab
401
402 # Necessary for some scripts
403 mkdir -p $vroot/etc/sysconfig
404 echo "NETWORKING=yes" > $vroot/etc/sysconfig/network
405
406 # Initialize RPM database in reference image
407 mkdir -p $vroot/var/lib/rpm
408 rpm --root $vroot --initdb
409 rpm --root $vroot --import $baseurl/RPM-GPG-KEY-fedora
410
411 # Initialize yum in reference image
412 mkdir -p $vroot/var/cache/yum $vroot/var/log
413 if [ $releasever -lt 7 ] ; then
414         corename="Core "
415 else
416         corename=""
417 fi
418
419 cat >$vroot/etc/yum.conf <<EOF
420 [main]
421 cachedir=/var/cache/yum
422 debuglevel=2
423 logfile=/var/log/yum.log
424 pkgpolicy=newest
425 distroverpkg=redhat-release
426 tolerant=1
427 exactarch=1
428 retries=20
429 obsoletes=1
430 gpgcheck=0
431 # Prevent yum-2.4 from loading additional repository definitions
432 # (e.g., from /etc/yum.repos.d/)
433 reposdir=/dev/null
434
435 [base]
436 name=Fedora ${corename}${releasever} - $basearch - base
437 baseurl=$baseurl/
438 $exclude_kernel
439 EOF
440
441 for optional in updates extras ; do
442     for optionalurl in \
443         $mirror/linux/core/$optional/$releasever/$basearch \
444         $mirror/core/$optional/$releasever/$basearch \
445         $mirror/linux/$optional/$releasever/$basearch \
446         $mirror/$optional/$releasever/$basearch ; do
447         if mkfedora_fetch $optionalurl/repodata/repomd.xml ; then
448             cat >>$vroot/etc/yum.conf <<EOF
449
450 [$(basename $optional)]
451 name=Fedora ${corename}${releasever} - $basearch - $(basename $optional)
452 baseurl=$optionalurl/
453 $exclude_kernel
454 EOF
455             break
456         fi
457     done
458 done
459
460 # If we are being built as part of an automated RPM build, solve the
461 # bootstrap problem by including any just built packages in the yum
462 # configuration. This cooperates with the PlanetLab build system.
463 if [ -n "$RPM_BUILD_DIR" ] ; then
464     RPM_RPMS_DIR=$(cd $(dirname $RPM_BUILD_DIR)/RPMS && pwd -P)
465     # yum-2.0.x
466     if [ -x /usr/bin/yum-arch ] ; then
467         yum-arch -q $RPM_RPMS_DIR
468     fi
469     # yum-2.4.x
470     if [ -x /usr/bin/createrepo ] ; then
471         if [ -f $RPM_RPMS_DIR/yumgroups.xml ] ; then
472             groupfile="-g yumgroups.xml"
473         fi
474         createrepo --quiet $groupfile $RPM_RPMS_DIR
475     fi
476     # If run under sudo, allow user to delete the headers/ and
477     # repodata/ directories.
478     if [ -n "$SUDO_USER" ] ; then
479         chown -R $SUDO_USER $RPM_RPMS_DIR
480     fi
481     cat >>$vroot/etc/yum.conf <<EOF
482
483 [bootstrap]
484 name=Bootstrap - $basearch - $RPM_RPMS_DIR/
485 baseurl=file://$RPM_RPMS_DIR/
486 EOF
487 fi
488
489 # pkgs file
490 if [ -n "$pkgsfile" ] ; then
491     # parse pkgsfile and add to local vars
492     fcdistro=$(pl_getReleaseName "Fedora" $releasever)
493     for i in $(pl_parsePkgs package $fcdistro $pkgsfile)  ; do
494         packages[${#packages[*]}]="$i"
495     done
496     for i in $(pl_parsePkgs group $fcdistro $pkgsfile) ; do
497         groups[${#groups[*]}]="$i"
498     done
499     for i in $(pl_parsePkgs exclude $fcdistro $pkgsfile) ; do
500         exclude[${#exclude[*]}]="$i"
501     done
502     junk=$(pl_parsePkgs junk $fcdistro $pkgsfile)
503     precious=$(pl_parsePkgs precious $fcdistro $pkgsfile)
504 fi
505
506 excludes=
507 for package in "${exclude[@]}" ; do
508     excludes="$excludes --exclude=$package"
509 done
510
511 # glibc must be specified explicitly for the correct arch to be
512 # chosen.
513 echo "* Installing glibc"
514 yum -c $vroot/etc/yum.conf --installroot=$vroot -y $excludes install glibc
515
516 # Go, baby, go
517 if [ ${#packages[*]} -gt 0 ] ; then
518    echo "* Installing optional packages" "${packages[@]}" 
519    yum -c $vroot/etc/yum.conf --installroot=$vroot -y $excludes \
520           install "${packages[@]}"
521    if ! rpm --root $vroot -q "${packages[@]}" >/dev/null ; then
522        echo "* Warning: Missing packages"
523        rpm --root $vroot -q "${packages[@]}" | grep "not installed"
524    fi
525 fi
526
527 if [ ${#groups[*]} -gt 0 ] ; then
528    ## call yum sequentially to get finer-grained info on dependencies
529    for grp in "${groups[@]}" ; do
530       echo "* Installing optional group $grp" 
531       yum -c $vroot/etc/yum.conf --installroot=$vroot -y $excludes \
532         groupinstall "$grp"
533    done
534 fi
535
536 # FC2 dev %preinstall checks /proc/mounts to make sure that /dev is
537 # not currently mounted as devfs. If it thinks it is, it will refuse
538 # to install the package. On a modern system running udev that mounts
539 # /dev as tmpfs, this check fails. Since we are installing into a
540 # chroot, whether /dev is mounted on the host system or not doesn't
541 # matter. If dev was explicitly mentioned in the packages list, force
542 # its installation.
543 if [ "$releasever" = "2" ] ; then
544     for package in "${packages[@]}" ; do
545         if [ "$package" = "dev" ] && ! rpm --root $vroot -q dev >/dev/null 2>&1 ; then
546             rpm --root $vroot -Uvh --noscripts $baseurl/Fedora/RPMS/dev-3.3.13-1.i386.rpm
547             break
548         fi
549     done
550 fi
551
552 # Clean yum cache
553 echo "* Cleaning up"
554 yum -c $vroot/etc/yum.conf --installroot=$vroot -y clean all
555
556 # Clean RPM state
557 rm -f $vroot/var/lib/rpm/__db*
558
559 # Set time zone to UTC
560 if [ -f $vroot/usr/share/zoneinfo/UTC -a -f $vroot/etc/localtime ] ; then
561     rm -f $vroot/etc/localtime
562     ln -s /usr/share/zoneinfo/UTC $vroot/etc/localtime
563 fi
564
565 # formerly in bootcd/prep.sh : to optimize footprint
566 echo "* Removing unnecessary junk"
567
568 pushd $vroot
569
570 # Save precious files
571 [ -n "$precious" ] && tar --ignore-failed-read -cpf precious.tar $precious
572
573 # Remove unnecessary junk
574 [ -n "$junk" ] && rm -rf $junk
575
576 # Restore precious files
577 [ -n "$precious" ] && tar -xpf precious.tar && rm -f precious.tar
578
579 popd
580
581 # remove trap handler, as we are about to call it directly.
582 trap - ERR INT
583
584 # Clean up
585 mkfedora_cleanup
586
587 return 0
588 }
589
590 function pl_mkfedora() {
591     root=$1
592     shift
593     options=$@
594
595     pl_makedevs $root
596
597     [ -n "$pl_DISTRO_URL" ] && options="$options -l $pl_DISTRO_URL"
598     [ -n "$pl_DISTRO_ARCH" ] && options="$options -a $pl_DISTRO_ARCH"
599     [ -n "$pl_DISTRO_RELEASE" ] && options="$options -r $pl_DISTRO_RELEASE"
600     # echo "mkfedora -v $options $root"
601     mkfedora -v $options $root
602 }
603
604 function pl_setup_chroot() {
605     root=$1; shift
606     options="$@"
607
608     pl_mkfedora $root $options
609
610     # Disable all services in reference image
611     chroot $root sh -c "/sbin/chkconfig --list | awk '{ print \$1 }' | xargs -i /sbin/chkconfig {} off"
612
613     # FC2 minilogd starts up during shutdown and makes unmounting
614     # impossible. Just get rid of it.
615     rm -f $root/sbin/minilogd
616     ln -nsf /bin/true $root/sbin/minilogd
617
618     # This tells the Boot Manager that it is okay to update
619     # /etc/resolv.conf and /etc/hosts whenever the network configuration
620     # changes. Users are free to delete this file.
621     touch $vroot/etc/AUTO_UPDATE_NET_FILES
622 }
623
624 # Move specified directories out of a src tree into a dst tree, and
625 # then for each moved directory create a symlink in src to dst.
626 function pl_move_dirs() {
627     root=$1
628     data=$2
629     store=$3
630     shift 3
631
632     mkdir -p $root/data
633     for datadir in "$@" ; do
634         mkdir -p ${data}${datadir}
635         if [ -d ${root}/${datadir} -a ! -h ${root}/${datadir} ] ; then
636             (cd ${root} && find ./${datadir} | cpio -p -d -u ../${data}/)
637         fi
638         rm -rf ${root}/${datadir}
639         mkdir -p $(dirname ${root}/${datadir})
640         ln -nsf /${store}/${datadir} ${root}/${datadir}
641     done
642 }
643
644 # Construct an image file from given some directory
645 # XXX in the future maybe use livecdtools?
646 function pl_make_image() {
647     root=$1
648     image=$2
649     extraspace=$3
650
651     # Leave about 100 MB free space and allow for about 20% inode overhead
652     bytes=$((($(du -sb $root | cut -f1) + $extraspace) * 120 / 100))
653     bs=4096
654     blocks=$(($bytes / $bs))
655     dd bs=$bs count=$blocks if=/dev/zero of=$image
656     mkfs.ext3 -b $bs -j -F $image
657
658     # Temporarily mount it
659     tmp=$(mktemp -d tmp.XXXXXX)
660     mount -o loop $image $tmp
661     trap "umount $tmp; rmdir $tmp" ERR INT
662
663     # Move files to it
664     (cd $root && tar cpf - .) | (cd $tmp && tar xpf -)
665
666     # Unmount it
667     umount -l $tmp
668     rmdir $tmp
669     trap - ERR INT
670 }
671
672 # Fix permissions on tmp directories
673 function pl_fixtmp_permissions() {
674     root=$1
675     chmod 1777 $root/tmp $root/usr/tmp $root/var/tmp
676 }
677
678 function pl_fixdirs() {
679     root=$1
680     datadirs=$2
681     for datadir in datadirs ; do
682         if [ -h ${root}/${datadir} ] ; then
683             rm -f ${root}/${datadir}
684             mkdir -p ${root}/${datadir}
685         fi
686     done
687 }
688
689 ########## .pkgs format
690 # comments start with a # - this is needed only if you use a keyword in a comment
691 # for a given keyword like 'package' :
692 # we support fcdistro-dependant format, for tokens (pkgname) without whitespace
693 # you can e.g. use
694 # package: pkg1 .. pkgn 
695 # package+f8: pkg1 .. pkgn
696 # package-f8: pkg1 .. pkgn
697
698 function pl_parsePkgs () {
699
700     keyword=$1;shift
701     fcdistro=$1; shift
702     # remaining arguments are paths to the pkgs files
703
704     # grab regular descriptions
705     all=$(grep -v '^#' "$@" | grep --regexp="^${keyword}:" | sed -e "s,${keyword}:,,")
706     # grab additions
707     add=$(grep -v '^#' "$@" | grep --regexp="^${keyword}+${fcdistro}:" | sed -e "s,${keyword}\+${fcdistro}:,,")
708     # grab exclusions
709     sub=$(grep -v '^#' "$@" | grep --regexp="^${keyword}\-${fcdistro}:" | sed -e "s,${keyword}\-${fcdistro}:,,")
710
711     for i in $all $add; do
712         for exclude in $sub; do
713             [ "$i" = "$exclude" ] && continue 2
714         done
715         echo "$i "
716     done
717     return 0
718 }
719
720 function pl_getPackages() { fcdistro=$1; shift ; pl_parsePkgs package $fcdistro "$@" ; }
721 function pl_getGroups() { fcdistro=$1; shift ; pl_parsePkgs group $fcdistro "$@" ; }
722
723 # locates a pldistro-dependant file
724 # tries first in build/<pldistro>/, then in build/planetlab/
725 function pl_locateDistroFile () {
726     builddir=$1; shift
727     pldistro=$1; shift
728     pkgsfile=$1; shift
729
730     pkgspath=""
731     # locate it
732     paths="$builddir/config.$pldistro/$pkgsfile $builddir/config.planetlab/$pkgsfile"
733     for path in $paths; do
734         if [ -f $path ] ; then
735             pkgspath=$path
736             break
737         fi
738     done
739     if [ -z "$pkgspath" ] ; then
740         echo 1>&2 "pl_locateDistroFile - in $(pwd) : cannot locate $pkgsfile in $builddir"
741         echo 1>&2 "candidates were $paths"
742         echo "not-found-by-pl_locateDistroFile"
743         return 1
744     else
745         echo 1>&2 "pl_locateDistroFile: using $pkgspath"
746         echo $pkgspath
747         return 0
748     fi
749 }