thanks daniel
[build.git] / vbuild-init-vserver.sh
1 #!/bin/bash
2 # -*-shell-*-
3
4 COMMAND=$(basename $0)
5
6 # lst parsing utilities
7 PATH=$(dirname $0):$PATH . build.common
8
9 function failure () {
10     echo "$COMMAND : Bailing out"
11     exit 1
12 }
13
14 # overwrite vserver's internal yum config from what is in
15 # .distributions/<distrib>/yum/yum.conf and /yum.repos.d 
16
17 function configure_yum_in_vserver () {
18     set -x 
19     set -e 
20     trap failure ERR INT
21
22     vserver=$1; shift
23     fcdistro=$1; shift
24
25     cd /etc/vservers/.distributions/${fcdistro}
26     if [ -f yum/yum.conf ] ; then
27         echo "Initializing yum.conf in $vserver from $(pwd)/yum"
28         sed -e "s!@YUMETCDIR@!/etc!g;
29                 s!@YUMCACHEDIR@!/var/cache/yum!g;
30                 s!@YUMLOGDIR@!/var/log!g;
31                 s!@YUMLOCKDIR@!/var/lock!g;
32                " yum/yum.conf > /vservers/$vserver/etc/yum.conf
33
34         # post process the various @...@ variables from this yum.conf file.
35     else
36         echo "Using $fcdistro default for yum.conf"
37     fi
38
39     if [ -d yum.repos.d ] ; then
40         echo "Initializing yum.repos.d in $vserver from $(pwd)/yum.repos.d"
41         rm -rf /vservers/$vserver/etc/yum.repos.d
42         tar cf - yum.repos.d | tar -C /vservers/$vserver/etc -xvf -
43     else
44         echo "Cannot initialize yum.repos.d in $vserver"
45     fi
46
47     if [ -n "$MYPLC_MODE" ] ; then
48         if [ ! -d /vservers/$vserver/etc/yum.repos.d ] ; then
49             echo "WARNING : cannot create myplc repo"
50         else
51             # exclude kernel from fedora repos 
52             for i in /vservers/$vserver/etc/yum.repos.d/* ; do
53                 echo "exclude=kernel* ulogd iptables" >> $i
54             done
55             # the build repo is not signed at this stage
56             cat > /vservers/$vserver/etc/yum.repos.d/myplc.repo <<EOF
57 [myplc]
58 name= MyPLC
59 baseurl=$REPO_URL
60 enabled=1
61 gpgcheck=0
62 EOF
63         fi
64     fi
65     cd -
66 }    
67
68 function setup_vserver () {
69
70     set -x
71     set -e
72     trap failure ERR INT
73
74     vserver=$1; shift
75     fcdistro=$1; shift
76     personality=$1; shift
77
78     CLONED=0
79     # create the new vserver
80     if [ ! -d /etc/vservers/$vserver ] ; then
81         # check if we can create the vserver from a reference vserver
82         #if [ -d /vservers/${fcdistro}_reference ] ; then
83         if [ 0 -ne 0 -a -n "$VBUILD_MODE" ] ; then
84             $personality vserver $VERBOSE $vserver build -m clone -- --source /vservers/${fcdistro}_reference
85             CLONED=1
86         else
87             $personality vserver $VERBOSE $vserver build -m yum -- -d $fcdistro
88         fi
89     fi
90
91     if [ ! -z "$personality" ] ; then
92         l32=$(grep $personality /etc/vservers/$vserver/personality | wc -l)
93         [ $l32 -eq 0 ] && echo $personality >> /etc/vservers/$vserver/personality
94     fi
95
96     if [ -n "$VBUILD_MODE" ] ; then 
97     # set up appropriate vserver capabilities to mount, mknod and IPC_LOCK
98         BCAPFILE=/etc/vservers/$vserver/bcapabilities
99         touch $BCAPFILE
100         cap=$(grep ^CAP_SYS_ADMIN /etc/vservers/$vserver/bcapabilities | wc -l)
101         [ $cap -eq 0 ] && echo 'CAP_SYS_ADMIN' >> /etc/vservers/$vserver/bcapabilities
102         cap=$(grep ^CAP_MKNOD /etc/vservers/$vserver/bcapabilities | wc -l)
103         [ $cap -eq 0 ] && echo 'CAP_MKNOD' >> /etc/vservers/$vserver/bcapabilities
104         cap=$(grep ^CAP_IPC_LOCK /etc/vservers/$vserver/bcapabilities | wc -l)
105         [ $cap -eq 0 ] && echo 'CAP_IPC_LOCK' >> /etc/vservers/$vserver/bcapabilities
106     fi
107
108     # start the vserver so we can do the following operations
109     if [ $CLONED -eq 0 ] ; then
110         $personality vyum $vserver -- -y install yum
111         $personality vserver $VERBOSE $vserver pkgmgmt internalize
112     fi
113     $personality vserver $VERBOSE $vserver start
114     $personality vserver $VERBOSE $vserver exec rm -f /var/lib/rpm/__db*
115     $personality vserver $VERBOSE $vserver exec rpm --rebuilddb
116
117     # minimal config in the vserver for yum to work
118     configure_yum_in_vserver $vserver $fcdistro 
119
120     # set up resolv.conf
121     cp /etc/resolv.conf /vservers/$vserver/etc/resolv.conf
122 }
123
124 function devel_tools () {
125
126     set -x 
127     set -e 
128     trap failure ERR INT
129
130     vserver=$1; shift
131     fcdistro=$1; shift
132     pldistro=$1; shift
133     personality=$1; shift
134
135     # check for .lst file based on pldistro
136     if [ -n "$VBUILD_MODE" ] ; then
137         lst=${pldistro}-devel.lst
138     else
139         lst=${pldistro}-shell.lst
140     fi
141     if [ -f $lst ] ; then
142         echo "$COMMAND: Using $lst"
143     else
144         echo "$COMMAND : Cannot locate $lst - exiting"
145         usage
146     fi
147
148     # install individual packages, then groups
149     packages=$(pl_getPackages2 ${fcdistro} $lst)
150     groups=$(pl_getGroups2 ${fcdistro} $lst)
151
152     [ -n "$packages" ] && $personality vserver $vserver exec yum -y install $packages
153     [ -n "$groups" ] && $personality vserver $vserver exec yum -y groupinstall $groups
154     return 0
155 }
156
157 function post_install () {
158     if [ -n "$VBUILD_MODE" ] ; then
159         post_install_vbuild "$@" 
160     else
161         post_install_myplc "$@"
162     fi
163 }
164
165 function post_install_vbuild () {
166
167     set -x 
168     set -e 
169     trap failure ERR INT
170
171     vserver=$1; shift
172     personality=$1; shift
173
174 ### From myplc-devel-native.spec
175 # be careful to backslash $ in this, otherwise it's the root context that's going to do the evaluation
176     cat << EOF | $personality vserver $VERBOSE $vserver exec bash -x
177     # set up /dev/loop* in vserver
178     for i in \$(seq 0 255) ; do
179         mknod -m 640 /dev/loop\$i b 7 \$i
180     done
181     
182     # create symlink for /dev/fd
183     ln -fs /proc/self/fd /dev/fd
184
185     # modify /etc/rpm/macros to not use /sbin/new-kernel-pkg
186     sed -i 's,/sbin/new-kernel-pkg:,,' /etc/rpm/macros
187     if [ -h "/sbin/new-kernel-pkg" ] ; then
188         filename=\$(readlink -f /sbin/new-kernel-pkg)
189         if [ "\$filename" == "/sbin/true" ] ; then
190                 echo "WARNING: /sbin/new-kernel-pkg symlinked to /sbin/true"
191                 echo "\tmost likely /etc/rpm/macros has /sbin/new-kernel-pkg declared in _netsharedpath."
192                 echo "\tPlease remove /sbin/new-kernel-pkg from _netsharedpath and reintall mkinitrd."
193                 exit 1
194         fi
195     fi
196     
197     # customize root's prompt
198     cat << PROFILE > /root/.profile
199 export PS1="[$vserver] \\w # "
200 PROFILE
201
202     uid=2000
203     gid=2000
204     
205     # add a "build" user to the system
206     builduser=\$(grep "^build:" /etc/passwd | wc -l)
207     if [ \$builduser -eq 0 ] ; then
208         groupadd -o -g \$gid build;
209         useradd -o -c 'Automated Build' -u \$uid -g \$gid -n -M -s /bin/bash build;
210     fi
211
212 # Allow build user to build certain RPMs as root
213     if [ -f /etc/sudoers ] ; then
214         buildsudo=\$(grep "^build.*ALL=(ALL).*NOPASSWD:.*ALL"  /etc/sudoers | wc -l)
215         if [ \$buildsudo -eq 0 ] ; then
216             echo "build   ALL=(ALL)       NOPASSWD: ALL" >> /etc/sudoers
217         fi
218         sed -i 's,^Defaults.*requiretty,#Defaults requiretty,' /etc/sudoers
219     fi
220 #
221 EOF
222
223 }
224
225 function post_install_myplc  () {
226     set -x 
227     set -e 
228     trap failure ERR INT
229
230     vserver=$1; shift
231     personality=$1; shift
232
233 # be careful to backslash $ in this, otherwise it's the root context that's going to do the evaluation
234     cat << EOF | $personality vserver $VERBOSE $vserver exec bash -x
235
236     # customize root's prompt
237     cat << PROFILE > /root/.profile
238 export PS1="[$vserver] \\w # "
239 PROFILE
240
241 EOF
242 }
243
244 COMMAND_VBUILD="vbuild-init-vserver.sh"
245 COMMAND_MYPLC="myplc-init-vserver.sh"
246 function usage () {
247     set +x 
248     echo "Usage: $COMMAND_VBUILD [-s|s|p] [-v] vserver-name distribution pldistro [personality]"
249     echo "Usage: $COMMAND_MYPLC [-s|s|p] [-v] vserver-name distribution pldistro repo-url [personality]"
250     echo "Requirements: you need to have a vserver-compliant kernel,"
251     echo "  as well as the util-vserver RPM installed"
252     echo "Description:"
253     echo "  This command creates a fresh vserver instance"
254     echo "  . vserver-name : the vserver's name"
255     echo "  . distribution : for creating the root filesystem, e.g. fc6"
256     echo "  . pldistro: e.g. onelab"
257     echo "  . repo-url: for myplc vserver, used to create a yum repository"
258     echo "  . personality: last, optional, argument defaults to linux32"
259     echo "This is done in three steps"
260     echo " (*) setup phase : vserver creation, yum internalization and config (from /etc/vservers)"
261     echo " (*) tools install the tools required for building are installed"
262     echo "     based on the folowing file for the actual set of packages and groups"
263     echo "     vbuild mode :  <pldistro>-devel.lst"
264     echo "     myplc mode :   <pldistro>-shell.lst"
265     echo " (*) post-install : various tunings required, as well as create a build user (vbuild only)"
266     echo "Options:"
267     echo " -s : skips the setup phase"
268     echo " -t : skips the tools phase"
269     echo " -p : skips the post-install"
270     echo " -v : passes -v to calls to vserver"
271     exit 1
272 }
273
274 ### parse args and 
275 function main () {
276
277     set -e
278     trap failure ERR INT
279
280     case "$COMMAND" in
281         $COMMAND_VBUILD)
282             VBUILD_MODE=true ;;
283         $COMMAND_MYPLC)
284             MYPLC_MODE=true;;
285         *)
286             usage ;;
287     esac
288
289     DO_SETUP=true
290     DO_TOOLS=true
291     DO_POST=true
292     VERBOSE=
293     while getopts "stpvh" opt ; do
294         case $opt in
295             s) DO_SETUP="" ;;
296             t) DO_TOOLS="" ;;
297             p) DO_POST="" ;;
298             v) VERBOSE="-v" ;;
299             h|*) usage ;;
300         esac
301     done
302         
303     shift $(($OPTIND - 1))
304     
305     [[ -z "$@" ]] && usage
306     vserver=$1 ; shift
307     [[ -z "$@" ]] && usage
308     fcdistro=$1 ; shift
309     [[ -z "$@" ]] && usage
310     pldistro=$1 ; shift
311     if [ -n "$MYPLC_MODE" ] ; then
312         [[ -z "$@" ]] && usage
313         REPO_URL=$1 ; shift
314     fi
315     if [[ -z "$@" ]] ; then
316         personality=linux32
317     else
318         personality=$1; shift
319     fi
320     [[ -n "$@" ]] && usage
321
322     [ -n "$DO_SETUP" ] && setup_vserver $vserver $fcdistro $personality 
323     [ -n "$DO_TOOLS" ] && devel_tools $vserver $fcdistro $pldistro $personality
324     [ -n "$DO_POST" ] && post_install $vserver $personality
325
326 }
327
328 main "$@"