Anthony Young | b62b4ca | 2011-10-26 22:29:08 -0700 | [diff] [blame] | 1 | #!/bin/bash |
| 2 | # |
| 3 | # Copyright (c) 2011 Citrix Systems, Inc. |
| 4 | # Copyright 2011 OpenStack LLC. |
Anthony Young | b62b4ca | 2011-10-26 22:29:08 -0700 | [diff] [blame] | 5 | # All Rights Reserved. |
| 6 | # |
| 7 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 8 | # not use this file except in compliance with the License. You may obtain |
| 9 | # a copy of the License at |
| 10 | # |
| 11 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 12 | # |
| 13 | # Unless required by applicable law or agreed to in writing, software |
| 14 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 15 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 16 | # License for the specific language governing permissions and limitations |
| 17 | # under the License. |
| 18 | # |
| 19 | |
| 20 | set -eu |
| 21 | |
| 22 | set -o xtrace |
| 23 | |
| 24 | VBOX_IMG=/output/packages/vbox-img |
| 25 | |
| 26 | usage() { |
| 27 | cat >&2 <<EOF |
| 28 | $0 -o <output filenames> -t <types> -x <xml files> <fs-staging-dir> <fs-size-MiB> <tmpdir> |
| 29 | -o: Colon-separated list of output filenames (one for each type). |
| 30 | -p: Create a disk label and partition within the output image |
| 31 | -t: Colon-separated list of types of output file. xva and ovf supported. |
| 32 | -x: XML filenames (one for each type) |
| 33 | |
| 34 | EOF |
| 35 | exit 1 |
| 36 | } |
| 37 | |
| 38 | # parse cmdline |
| 39 | |
| 40 | OPT_USE_PARTITION= |
| 41 | OPT_TYPES= |
| 42 | OPT_OUTPUT_FILES= |
| 43 | OPT_XML_FILES= |
| 44 | |
| 45 | while getopts o:pt:x: o |
| 46 | do case "$o" in |
| 47 | o) OPT_OUTPUT_FILES=$(echo "$OPTARG" | sed -e 's/\s*:\s*/ /g') |
| 48 | ;; |
| 49 | p) OPT_USE_PARTITION=1 |
| 50 | ;; |
| 51 | t) OPT_TYPES=$(echo "$OPTARG" | sed -e 's/\s*:\s*/ /g') |
| 52 | ;; |
| 53 | x) OPT_XML_FILES=$(echo "$OPTARG" | sed -e 's/\s*:\s*/ /g') |
| 54 | ;; |
| 55 | [?]) usage |
| 56 | ;; |
| 57 | esac |
| 58 | done |
| 59 | shift $((OPTIND-1)) |
| 60 | |
| 61 | [ $# -ne 3 ] && usage |
| 62 | FS_STAGING="$1" |
| 63 | FS_SIZE_MIB="$2" |
| 64 | TMPDIR="$3" |
| 65 | |
| 66 | if [ "$UID" = "0" ] |
| 67 | then |
| 68 | SUDO= |
| 69 | else |
| 70 | SUDO=sudo |
| 71 | fi |
| 72 | |
| 73 | if [ "$FS_SIZE_MIB" = "0" ] |
| 74 | then |
| 75 | # Just create a dummy file. This allows developers to bypass bits of |
| 76 | # the build by setting the size to 0. |
| 77 | touch $OPT_OUTPUT_FILES |
| 78 | exit 0 |
| 79 | fi |
| 80 | |
| 81 | # create temporary files and dirs |
| 82 | FS_TMPFILE=$(mktemp "$TMPDIR/mkxva-fsimg-XXXXX") |
| 83 | XVA_TARBALL_STAGING=$(mktemp -d "$TMPDIR/mkxva-tarball-staging-XXXXX") |
| 84 | OVF_STAGING=$(mktemp -d "$TMPDIR/mkxva-ovf-staging-XXXXX") |
| 85 | |
| 86 | # Find udevsettle and udevtrigger on this installation |
| 87 | if [ -x "/sbin/udevsettle" ] ; then |
| 88 | UDEVSETTLE="/sbin/udevsettle --timeout=30" |
| 89 | elif [ -x "/sbin/udevadm" ] ; then |
| 90 | UDEVSETTLE='/sbin/udevadm settle' |
| 91 | else |
| 92 | UDEVSETTLE='/bin/true' |
| 93 | fi |
| 94 | |
| 95 | if [ -x "/sbin/udevtrigger" ] ; then |
| 96 | UDEVTRIGGER=/sbin/udevtrigger |
| 97 | elif [ -x "/sbin/udevadm" ] ; then |
| 98 | UDEVTRIGGER='/sbin/udevadm trigger' |
| 99 | else |
| 100 | UDEVTRIGGER= |
| 101 | fi |
| 102 | |
| 103 | # CLEAN_ variables track devices and mounts that must be taken down |
| 104 | # no matter how the script exits. Loop devices are vulnerable to |
| 105 | # exhaustion so we make every effort to remove them |
| 106 | |
| 107 | CLEAN_KPARTX= |
| 108 | CLEAN_LOSETUP= |
| 109 | CLEAN_MOUNTPOINT= |
| 110 | |
| 111 | cleanup_devices () { |
| 112 | if [ -n "$CLEAN_MOUNTPOINT" ] ; then |
| 113 | echo "Mountpoint $CLEAN_MOUNTPOINT removed on abnormal exit" |
| 114 | $SUDO umount "$CLEAN_MOUNTPOINT" || echo "umount failed" |
| 115 | rmdir "$CLEAN_MOUNTPOINT" || echo "rmdir failed" |
| 116 | fi |
| 117 | if [ -n "$CLEAN_KPARTX" ] ; then |
| 118 | echo "kpartx devices for $CLEAN_KPARTX removed on abnormal exit" |
| 119 | $SUDO kpartx -d "$CLEAN_KPARTX" || echo "kpartx -d failed" |
| 120 | fi |
| 121 | if [ -n "$CLEAN_LOSETUP" ] ; then |
| 122 | echo "Loop device $CLEAN_LOSETUP removed on abnormal exit" |
| 123 | $SUDO losetup -d "$CLEAN_LOSETUP" # Allow losetup errors to propagate |
| 124 | fi |
| 125 | } |
| 126 | |
| 127 | trap "cleanup_devices" EXIT |
| 128 | |
| 129 | make_fs_inner () { |
| 130 | local staging="$1" |
| 131 | local output="$2" |
| 132 | local options="$3" |
| 133 | CLEAN_MOUNTPOINT=$(mktemp -d "$TMPDIR/mkfs-XXXXXX") |
| 134 | |
| 135 | # copy staging dir contents to fs image |
| 136 | $SUDO mount $options "$output" "$CLEAN_MOUNTPOINT" |
| 137 | $SUDO tar -C "$staging" -c . | tar -C "$CLEAN_MOUNTPOINT" -x |
| 138 | $SUDO umount "$CLEAN_MOUNTPOINT" |
| 139 | rmdir "$CLEAN_MOUNTPOINT" |
| 140 | CLEAN_MOUNTPOINT= |
| 141 | } |
| 142 | |
| 143 | # Turn a staging dir into an ext3 filesystem within a partition |
| 144 | make_fs_in_partition () { |
| 145 | local staging="$1" |
| 146 | local output="$2" |
| 147 | |
| 148 | # create new empty disk |
| 149 | dd if=/dev/zero of="$output" bs=1M count=$FS_SIZE_MIB |
| 150 | # Set up a loop device on the empty disk image |
| 151 | local loopdevice=$($SUDO losetup -f) |
| 152 | $SUDO losetup "$loopdevice" "$output" |
| 153 | CLEAN_LOSETUP="$loopdevice" |
| 154 | # Create a partition table and single partition. |
| 155 | # Start partition at sector 63 to allow space for grub |
| 156 | cat <<EOF |
| 157 | Errors from sfdisk below are expected because the new disk is uninitialised |
| 158 | Expecting: sfdisk: ERROR: sector 0 does not have an msdos signature |
| 159 | Expecting: /dev/loop0: unrecognized partition table type |
| 160 | EOF |
| 161 | $SUDO sfdisk -uS "$CLEAN_LOSETUP" <<EOF |
| 162 | 63 - - * |
| 163 | EOF |
| 164 | |
| 165 | # kpartx creates a device for the new partition |
| 166 | # in the form /dev/mapper/loop1p1 |
| 167 | $SUDO kpartx -av "$CLEAN_LOSETUP" |
| 168 | CLEAN_KPARTX="$CLEAN_LOSETUP" |
| 169 | # Wait for the device to appear |
| 170 | $UDEVTRIGGER |
| 171 | $UDEVSETTLE || echo "udev settle command return code non-zero" |
| 172 | # Infer the name of the partition device |
| 173 | local partition="${CLEAN_LOSETUP/dev/dev/mapper}p1" |
| 174 | # Set permissive privileges on the device |
| 175 | $SUDO chmod 0777 "$partition" |
| 176 | # Make an ext3 filesystem on the partition |
| 177 | /sbin/mkfs.ext3 -I 128 -m0 -F "$partition" |
| 178 | /sbin/e2label "$partition" vpxroot |
| 179 | make_fs_inner "$staging" "$partition" "" |
| 180 | |
| 181 | # Now run grub on the image we've created |
| 182 | CLEAN_MOUNTPOINT=$(mktemp -d "$TMPDIR/mkfs-XXXXXX") |
| 183 | |
| 184 | # copy Set up[ grub files prior to installing grub within the image |
| 185 | $SUDO mount "$partition" "$CLEAN_MOUNTPOINT" |
| 186 | $SUDO cp $CLEAN_MOUNTPOINT/usr/share/grub/i386-redhat/* "$CLEAN_MOUNTPOINT/boot/grub" |
| 187 | kernel_version=$($SUDO chroot "$CLEAN_MOUNTPOINT" rpm -qv kernel | sed -e 's/kernel-//') |
| 188 | kernel_version_xen=$($SUDO chroot "$CLEAN_MOUNTPOINT" rpm -qv kernel-xen | sed -e 's/kernel-xen-//') |
| 189 | $SUDO cat > "$CLEAN_MOUNTPOINT/boot/grub/grub.conf" <<EOF |
| 190 | default 0 |
| 191 | timeout 2 |
| 192 | |
| 193 | title vmlinuz-$kernel_version (HVM) |
| 194 | root (hd0,0) |
| 195 | kernel /boot/vmlinuz-$kernel_version ro root=LABEL=vpxroot |
| 196 | initrd /boot/initrd-$kernel_version.img |
| 197 | |
| 198 | title vmlinuz-${kernel_version_xen}xen (PV) |
| 199 | root (hd0,0) |
| 200 | kernel /boot/vmlinuz-${kernel_version_xen}xen ro root=LABEL=vpxroot console=xvc0 |
| 201 | initrd /boot/initrd-${kernel_version_xen}xen.img |
| 202 | EOF |
| 203 | |
| 204 | $SUDO umount "$CLEAN_MOUNTPOINT" |
| 205 | CLEAN_MOUNTPOINT= |
| 206 | |
| 207 | # Grub expects a disk with name /dev/xxxx with a first partition |
| 208 | # named /dev/xxxx1, so we give it what it wants using symlinks |
| 209 | # Note: /dev is linked to the real /dev of the build machine, so |
| 210 | # must be cleaned up |
| 211 | local disk_name="/dev/osxva$$bld" |
| 212 | local disk_part1_name="${disk_name}1" |
| 213 | rm -f "$disk_name" |
| 214 | rm -f "$disk_part1_name" |
| 215 | ln -s "$CLEAN_LOSETUP" "$disk_name" |
| 216 | ln -s "$partition" "$disk_part1_name" |
| 217 | |
| 218 | # Feed commands into the grub shell to setup the disk |
| 219 | grub --no-curses --device-map=/dev/null <<EOF |
| 220 | device (hd0) $disk_name |
| 221 | setup (hd0) (hd0,0) |
| 222 | quit |
| 223 | EOF |
| 224 | |
| 225 | # Cleanup |
| 226 | rm -f "$disk_name" |
| 227 | rm -f "$disk_part1_name" |
| 228 | $SUDO kpartx -dv "$CLEAN_KPARTX" |
| 229 | CLEAN_KPARTX= |
| 230 | $SUDO losetup -d "$CLEAN_LOSETUP" |
| 231 | CLEAN_LOSETUP= |
| 232 | } |
| 233 | |
| 234 | # turn a staging dir into an ext3 filesystem image |
| 235 | make_fs () { |
| 236 | local staging="$1" |
| 237 | local output="$2" |
| 238 | |
| 239 | # create new empty fs |
| 240 | dd if=/dev/zero of="$output" bs=1M count=0 seek=$FS_SIZE_MIB |
| 241 | /sbin/mkfs.ext3 -m0 -F "$output" |
| 242 | /sbin/e2label "$output" vpxroot |
| 243 | make_fs_inner "$staging" "$output" "-oloop" |
| 244 | } |
| 245 | |
| 246 | |
| 247 | # split a virtual disk image into the format expected inside an xva file |
| 248 | splitvdi () { |
| 249 | local diskimg="$1" |
| 250 | local outputdir="$2" |
| 251 | local rio="$3" |
| 252 | |
| 253 | local n_bytes=$(stat --printf=%s "$diskimg") |
| 254 | local n_meg=$((($n_bytes+$((1024*1024 -1)))/$((1024*1024)))) |
| 255 | local i=0 |
| 256 | while [ $i -lt $n_meg ] ; do |
| 257 | if [ $rio -eq 0 ] ; then |
| 258 | local file="$outputdir"/chunk-$(printf "%08d" $i) |
| 259 | dd if="$diskimg" of="$file" skip=$i bs=1M count=1 2>/dev/null |
| 260 | gzip "$file" |
| 261 | else |
| 262 | local file="$outputdir"/$(printf "%08d" $i) |
| 263 | dd if="$diskimg" of="$file" skip=$i bs=1M count=1 2>/dev/null |
| 264 | local chksum=$(sha1sum -b "$file") |
| 265 | echo -n "${chksum/ */}" > "$file.checksum" |
| 266 | fi |
| 267 | i=$(($i + 1)) |
| 268 | done |
| 269 | } |
| 270 | |
| 271 | if [ -n "$OPT_USE_PARTITION" ] ; then |
| 272 | make_fs_in_partition "$FS_STAGING" "$FS_TMPFILE" |
| 273 | else |
| 274 | make_fs "$FS_STAGING" "$FS_TMPFILE" |
| 275 | fi |
| 276 | |
| 277 | VDI_SIZE=$(stat --format=%s "$FS_TMPFILE") |
| 278 | |
| 279 | make_xva () { |
| 280 | local output_file="$1" |
| 281 | local xml_file="$2" |
| 282 | local subdir |
| 283 | local rio |
| 284 | |
| 285 | if [[ `cat $xml_file` =~ "<member>\s*<name>class</name>\s*<value>VDI</value>\s*</member>\s*<member>\s*<name>id</name>\s*<value>(Ref:[0-9]+)</value>" ]] |
| 286 | then |
| 287 | # it's a rio style xva |
| 288 | subdir="${BASH_REMATCH[1]}"; |
| 289 | rio=1 |
| 290 | else |
| 291 | # it's a geneva style xva |
| 292 | subdir="xvda" |
| 293 | rio=0 |
| 294 | fi |
| 295 | |
| 296 | cp "$xml_file" "$XVA_TARBALL_STAGING"/ova.xml |
| 297 | sed -i -e "s/@VDI_SIZE@/$VDI_SIZE/" "$XVA_TARBALL_STAGING"/ova.xml |
| 298 | mkdir "$XVA_TARBALL_STAGING/$subdir" |
| 299 | splitvdi "$FS_TMPFILE" "$XVA_TARBALL_STAGING/$subdir" "$rio" |
| 300 | TARFILE_MEMBERS=$(cd "$XVA_TARBALL_STAGING" && echo ova.xml $subdir/*) |
| 301 | tar -C "$XVA_TARBALL_STAGING" --format=v7 -c $TARFILE_MEMBERS -f "$output_file.tmp" |
| 302 | mv "$output_file.tmp" "$output_file" |
| 303 | } |
| 304 | |
| 305 | make_ovf () { |
| 306 | local output_dir="$1" |
| 307 | local xml_file="$2" |
| 308 | local output_base=$(basename "$output_dir") |
| 309 | local disk="$output_dir/${output_base}.vmdk" |
| 310 | local manifest="$output_dir/${output_base}.mf" |
| 311 | local ovf="$output_dir/${output_base}.ovf" |
| 312 | |
| 313 | mkdir -p "$output_dir" |
| 314 | rm -f "$disk" |
| 315 | $VBOX_IMG convert --srcfilename="$FS_TMPFILE" --dstfilename="$disk" \ |
| 316 | --srcformat RAW --dstformat VMDK --variant Stream |
| 317 | chmod 0644 "$disk" |
| 318 | |
| 319 | local n_bytes=$(stat --printf=%s "$disk") |
| 320 | cp "$xml_file" "$ovf" |
| 321 | sed -i -e "s/@MKXVA_DISK_FULLSIZE@/$VDI_SIZE/" "$ovf" |
| 322 | sed -i -e "s/@MKXVA_DISK_SIZE@/$n_bytes/" "$ovf" |
| 323 | sed -i -e "s/@MKXVA_DISK_MIB_SIZE@/$FS_SIZE_MIB/" "$ovf" |
| 324 | sed -i -e "s/@MKXVA_DISK_FILENAME@/${output_base}.vmdk/" "$ovf" |
| 325 | |
| 326 | for to_sign in "$ovf" "$disk" |
| 327 | do |
| 328 | local sha1_sum=$(sha1sum "$to_sign" | cut -d' ' -f1) |
| 329 | echo "SHA1($(basename "$to_sign"))= $sha1_sum" >> $manifest |
| 330 | done |
| 331 | } |
| 332 | |
| 333 | output_files="$OPT_OUTPUT_FILES" |
| 334 | xml_files="$OPT_XML_FILES" |
| 335 | # Iterate through the type list creating the relevant VMs |
| 336 | for create_type in $OPT_TYPES |
| 337 | do |
| 338 | # Shift one parameter from the front of the lists |
| 339 | create_output_file="${output_files%% *}" |
| 340 | output_files="${output_files#* }" |
| 341 | create_xml_file="${xml_files%% *}" |
| 342 | xml_files="${xml_files#* }" |
| 343 | echo "Creating $create_type appliance $create_output_file using metadata file $create_xml_file" |
| 344 | |
| 345 | case "$create_type" in |
| 346 | xva) |
| 347 | make_xva "$create_output_file" "$create_xml_file" |
| 348 | ;; |
| 349 | ovf) |
| 350 | make_ovf "$create_output_file" "$create_xml_file" |
| 351 | ;; |
| 352 | *) |
| 353 | echo "Unknown VM type '$create_type'" |
| 354 | exit 1 |
| 355 | ;; |
| 356 | esac |
| 357 | |
| 358 | done |
| 359 | |
| 360 | |
| 361 | # cleanup |
| 362 | if [ -z "${DO_NOT_CLEANUP:-}" ] ; then |
| 363 | rm -rf "$XVA_TARBALL_STAGING" |
| 364 | rm -f "$FS_TMPFILE" |
| 365 | fi |