|  | #!/usr/bin/env bash | 
|  |  | 
|  | # **build_uec.sh** | 
|  |  | 
|  | # Make sure that we have the proper version of ubuntu (only works on oneiric) | 
|  | if ! egrep -q "oneiric" /etc/lsb-release; then | 
|  | echo "This script only works with ubuntu oneiric." | 
|  | exit 1 | 
|  | fi | 
|  |  | 
|  | # Keep track of the current directory | 
|  | TOOLS_DIR=$(cd $(dirname "$0") && pwd) | 
|  | TOP_DIR=$(cd $TOOLS_DIR/..; pwd) | 
|  |  | 
|  | # Import common functions | 
|  | . $TOP_DIR/functions | 
|  |  | 
|  | cd $TOP_DIR | 
|  |  | 
|  | # Source params | 
|  | source ./stackrc | 
|  |  | 
|  | # Ubuntu distro to install | 
|  | DIST_NAME=${DIST_NAME:-oneiric} | 
|  |  | 
|  | # Configure how large the VM should be | 
|  | GUEST_SIZE=${GUEST_SIZE:-10G} | 
|  |  | 
|  | # exit on error to stop unexpected errors | 
|  | set -o errexit | 
|  | set -o xtrace | 
|  |  | 
|  | # Abort if localrc is not set | 
|  | if [ ! -e $TOP_DIR/localrc ]; then | 
|  | echo "You must have a localrc with ALL necessary passwords defined before proceeding." | 
|  | echo "See stack.sh for required passwords." | 
|  | exit 1 | 
|  | fi | 
|  |  | 
|  | # Install deps if needed | 
|  | DEPS="kvm libvirt-bin kpartx cloud-utils curl" | 
|  | apt_get install -y --force-yes $DEPS || true # allow this to fail gracefully for concurrent builds | 
|  |  | 
|  | # Where to store files and instances | 
|  | WORK_DIR=${WORK_DIR:-/opt/uecstack} | 
|  |  | 
|  | # Where to store images | 
|  | image_dir=$WORK_DIR/images/$DIST_NAME | 
|  | mkdir -p $image_dir | 
|  |  | 
|  | # Start over with a clean base image, if desired | 
|  | if [ $CLEAN_BASE ]; then | 
|  | rm -f $image_dir/disk | 
|  | fi | 
|  |  | 
|  | # Get the base image if it does not yet exist | 
|  | if [ ! -e $image_dir/disk ]; then | 
|  | $TOOLS_DIR/get_uec_image.sh -r $GUEST_SIZE $DIST_NAME $image_dir/disk $image_dir/kernel | 
|  | fi | 
|  |  | 
|  | # Copy over dev environment if COPY_ENV is set. | 
|  | # This will also copy over your current devstack. | 
|  | if [ $COPY_ENV ]; then | 
|  | cd $TOOLS_DIR | 
|  | ./copy_dev_environment_to_uec.sh $image_dir/disk | 
|  | fi | 
|  |  | 
|  | # Option to warm the base image with software requirements. | 
|  | if [ $WARM_CACHE ]; then | 
|  | cd $TOOLS_DIR | 
|  | ./warm_apts_and_pips_for_uec.sh $image_dir/disk | 
|  | fi | 
|  |  | 
|  | # Name of our instance, used by libvirt | 
|  | GUEST_NAME=${GUEST_NAME:-devstack} | 
|  |  | 
|  | # Mop up after previous runs | 
|  | virsh destroy $GUEST_NAME || true | 
|  |  | 
|  | # Where this vm is stored | 
|  | vm_dir=$WORK_DIR/instances/$GUEST_NAME | 
|  |  | 
|  | # Create vm dir and remove old disk | 
|  | mkdir -p $vm_dir | 
|  | rm -f $vm_dir/disk | 
|  |  | 
|  | # Create a copy of the base image | 
|  | qemu-img create -f qcow2 -b $image_dir/disk $vm_dir/disk | 
|  |  | 
|  | # Back to devstack | 
|  | cd $TOP_DIR | 
|  |  | 
|  | GUEST_NETWORK=${GUEST_NETWORK:-1} | 
|  | GUEST_RECREATE_NET=${GUEST_RECREATE_NET:-yes} | 
|  | GUEST_IP=${GUEST_IP:-192.168.$GUEST_NETWORK.50} | 
|  | GUEST_CIDR=${GUEST_CIDR:-$GUEST_IP/24} | 
|  | GUEST_NETMASK=${GUEST_NETMASK:-255.255.255.0} | 
|  | GUEST_GATEWAY=${GUEST_GATEWAY:-192.168.$GUEST_NETWORK.1} | 
|  | GUEST_MAC=${GUEST_MAC:-"02:16:3e:07:69:`printf '%02X' $GUEST_NETWORK`"} | 
|  | GUEST_RAM=${GUEST_RAM:-1524288} | 
|  | GUEST_CORES=${GUEST_CORES:-1} | 
|  |  | 
|  | # libvirt.xml configuration | 
|  | NET_XML=$vm_dir/net.xml | 
|  | NET_NAME=${NET_NAME:-devstack-$GUEST_NETWORK} | 
|  | cat > $NET_XML <<EOF | 
|  | <network> | 
|  | <name>$NET_NAME</name> | 
|  | <bridge name="stackbr%d" /> | 
|  | <forward/> | 
|  | <ip address="$GUEST_GATEWAY" netmask="$GUEST_NETMASK"> | 
|  | <dhcp> | 
|  | <range start='192.168.$GUEST_NETWORK.2' end='192.168.$GUEST_NETWORK.127' /> | 
|  | </dhcp> | 
|  | </ip> | 
|  | </network> | 
|  | EOF | 
|  |  | 
|  | if [[ "$GUEST_RECREATE_NET" == "yes" ]]; then | 
|  | virsh net-destroy $NET_NAME || true | 
|  | # destroying the network isn't enough to delete the leases | 
|  | rm -f /var/lib/libvirt/dnsmasq/$NET_NAME.leases | 
|  | virsh net-create $vm_dir/net.xml | 
|  | fi | 
|  |  | 
|  | # libvirt.xml configuration | 
|  | LIBVIRT_XML=$vm_dir/libvirt.xml | 
|  | cat > $LIBVIRT_XML <<EOF | 
|  | <domain type='kvm'> | 
|  | <name>$GUEST_NAME</name> | 
|  | <memory>$GUEST_RAM</memory> | 
|  | <os> | 
|  | <type>hvm</type> | 
|  | <kernel>$image_dir/kernel</kernel> | 
|  | <cmdline>root=/dev/vda ro console=ttyS0 init=/usr/lib/cloud-init/uncloud-init ds=nocloud-net;s=http://192.168.$GUEST_NETWORK.1:4567/ ubuntu-pass=ubuntu</cmdline> | 
|  | </os> | 
|  | <features> | 
|  | <acpi/> | 
|  | </features> | 
|  | <clock offset='utc'/> | 
|  | <vcpu>$GUEST_CORES</vcpu> | 
|  | <devices> | 
|  | <disk type='file'> | 
|  | <driver type='qcow2'/> | 
|  | <source file='$vm_dir/disk'/> | 
|  | <target dev='vda' bus='virtio'/> | 
|  | </disk> | 
|  |  | 
|  | <interface type='network'> | 
|  | <source network='$NET_NAME'/> | 
|  | </interface> | 
|  |  | 
|  | <!-- The order is significant here.  File must be defined first --> | 
|  | <serial type="file"> | 
|  | <source path='$vm_dir/console.log'/> | 
|  | <target port='1'/> | 
|  | </serial> | 
|  |  | 
|  | <console type='pty' tty='/dev/pts/2'> | 
|  | <source path='/dev/pts/2'/> | 
|  | <target port='0'/> | 
|  | </console> | 
|  |  | 
|  | <serial type='pty'> | 
|  | <source path='/dev/pts/2'/> | 
|  | <target port='0'/> | 
|  | </serial> | 
|  |  | 
|  | <graphics type='vnc' port='-1' autoport='yes' keymap='en-us' listen='0.0.0.0'/> | 
|  | </devices> | 
|  | </domain> | 
|  | EOF | 
|  |  | 
|  |  | 
|  | rm -rf $vm_dir/uec | 
|  | cp -r $TOOLS_DIR/uec $vm_dir/uec | 
|  |  | 
|  | # set metadata | 
|  | cat > $vm_dir/uec/meta-data<<EOF | 
|  | hostname: $GUEST_NAME | 
|  | instance-id: i-hop | 
|  | instance-type: m1.ignore | 
|  | local-hostname: $GUEST_NAME.local | 
|  | EOF | 
|  |  | 
|  | # set user-data | 
|  | cat > $vm_dir/uec/user-data<<EOF | 
|  | #!/bin/bash | 
|  | # hostname needs to resolve for rabbit | 
|  | sed -i "s/127.0.0.1/127.0.0.1 \`hostname\`/" /etc/hosts | 
|  | apt-get update | 
|  | apt-get install git sudo -y | 
|  | # Disable byobu | 
|  | sudo apt-get remove -y byobu | 
|  | EOF | 
|  |  | 
|  | # Setup stack user with our key | 
|  | if [[ -e ~/.ssh/id_rsa.pub ]]; then | 
|  | PUB_KEY=`cat  ~/.ssh/id_rsa.pub` | 
|  | cat >> $vm_dir/uec/user-data<<EOF | 
|  | mkdir -p /opt/stack | 
|  | if [ ! -d /opt/stack/devstack ]; then | 
|  | git clone https://github.com/cloudbuilders/devstack.git /opt/stack/devstack | 
|  | cd /opt/stack/devstack | 
|  | cat > localrc <<LOCAL_EOF | 
|  | ROOTSLEEP=0 | 
|  | `cat $TOP_DIR/localrc` | 
|  | LOCAL_EOF | 
|  | fi | 
|  | useradd -U -G sudo -s /bin/bash -d /opt/stack -m stack | 
|  | echo stack:pass | chpasswd | 
|  | mkdir -p /opt/stack/.ssh | 
|  | echo "$PUB_KEY" > /opt/stack/.ssh/authorized_keys | 
|  | chown -R stack /opt/stack | 
|  | chmod 700 /opt/stack/.ssh | 
|  | chmod 600 /opt/stack/.ssh/authorized_keys | 
|  |  | 
|  | grep -q "^#includedir.*/etc/sudoers.d" /etc/sudoers || | 
|  | echo "#includedir /etc/sudoers.d" >> /etc/sudoers | 
|  | ( umask 226 && echo "stack ALL=(ALL) NOPASSWD:ALL" \ | 
|  | > /etc/sudoers.d/50_stack_sh ) | 
|  | EOF | 
|  | fi | 
|  |  | 
|  | # Run stack.sh | 
|  | cat >> $vm_dir/uec/user-data<<EOF | 
|  | su -c "cd /opt/stack/devstack && ./stack.sh" stack | 
|  | EOF | 
|  |  | 
|  | # (re)start a metadata service | 
|  | ( | 
|  | pid=`lsof -iTCP@192.168.$GUEST_NETWORK.1:4567 -n | awk '{print $2}' | tail -1` | 
|  | [ -z "$pid" ] || kill -9 $pid | 
|  | ) | 
|  | cd $vm_dir/uec | 
|  | python meta.py 192.168.$GUEST_NETWORK.1:4567 & | 
|  |  | 
|  | # Create the instance | 
|  | virsh create $vm_dir/libvirt.xml | 
|  |  | 
|  | # Tail the console log till we are done | 
|  | WAIT_TILL_LAUNCH=${WAIT_TILL_LAUNCH:-1} | 
|  | if [ "$WAIT_TILL_LAUNCH" = "1" ]; then | 
|  | set +o xtrace | 
|  | # Done creating the container, let's tail the log | 
|  | echo | 
|  | echo "=============================================================" | 
|  | echo "                          -- YAY! --" | 
|  | echo "=============================================================" | 
|  | echo | 
|  | echo "We're done launching the vm, about to start tailing the" | 
|  | echo "stack.sh log. It will take a second or two to start." | 
|  | echo | 
|  | echo "Just CTRL-C at any time to stop tailing." | 
|  | echo | 
|  |  | 
|  | if ! timeout 60 sh -c "while [ ! -s /var/lib/libvirt/dnsmasq/$NET_NAME.leases ]; do sleep 1; done"; then | 
|  | echo "Your instance failed to acquire an IP address" | 
|  | exit 1 | 
|  | fi | 
|  |  | 
|  | ip=`cat /var/lib/libvirt/dnsmasq/$NET_NAME.leases | cut -d " " -f3` | 
|  | echo "#############################################################" | 
|  | echo "              -- This is your instance's IP: --" | 
|  | echo "                           $ip" | 
|  | echo "#############################################################" | 
|  |  | 
|  | sleep 2 | 
|  |  | 
|  | while [ ! -e "$vm_dir/console.log" ]; do | 
|  | sleep 1 | 
|  | done | 
|  |  | 
|  | tail -F $vm_dir/console.log & | 
|  |  | 
|  | TAIL_PID=$! | 
|  |  | 
|  | function kill_tail() { | 
|  | kill $TAIL_PID | 
|  | exit 1 | 
|  | } | 
|  |  | 
|  | # Let Ctrl-c kill tail and exit | 
|  | trap kill_tail SIGINT | 
|  |  | 
|  | echo "Waiting stack.sh to finish..." | 
|  | while ! egrep -q '^stack.sh (completed|failed)' $vm_dir/console.log ; do | 
|  | sleep 1 | 
|  | done | 
|  |  | 
|  | set -o xtrace | 
|  |  | 
|  | kill $TAIL_PID | 
|  |  | 
|  | if ! grep -q "^stack.sh completed in" $vm_dir/console.log; then | 
|  | exit 1 | 
|  | fi | 
|  |  | 
|  | set +o xtrace | 
|  | echo "" | 
|  | echo "Finished - Zip-a-dee Doo-dah!" | 
|  | fi |