blob: 8d39703a58f810555d38e7dcf808efb8caa5aad2 [file] [log] [blame]
Anthony Young440be4b2012-02-10 21:42:39 -08001#!/usr/bin/env bash
2
3# **boot_from_volume.sh**
4
5# This script demonstrates how to boot from a volume. It does the following:
6# * Create a 'builder' instance
7# * Attach a volume to the instance
8# * Format and install an os onto the volume
9# * Detach volume from builder, and then boot volume-backed instance
10
11echo "**************************************************"
12echo "Begin DevStack Exercise: $0"
13echo "**************************************************"
14
15# This script exits on an error so that errors don't compound and you see
16# only the first error that occured.
17set -o errexit
18
19# Print the commands being run so that we can see the command that triggers
20# an error. It is also useful for following allowing as the install occurs.
21set -o xtrace
22
23
24# Settings
25# ========
26
27# Keep track of the current directory
28EXERCISE_DIR=$(cd $(dirname "$0") && pwd)
29TOP_DIR=$(cd $EXERCISE_DIR/..; pwd)
30
31# Import common functions
32source $TOP_DIR/functions
33
34# Import configuration
35source $TOP_DIR/openrc
36
37# Import exercise configuration
38source $TOP_DIR/exerciserc
39
40# Boot this image, use first AMI image if unset
41DEFAULT_IMAGE_NAME=${DEFAULT_IMAGE_NAME:-ami}
42
43# Instance type
44DEFAULT_INSTANCE_TYPE=${DEFAULT_INSTANCE_TYPE:-m1.tiny}
45
46# Default floating IP pool name
47DEFAULT_FLOATING_POOL=${DEFAULT_FLOATING_POOL:-nova}
48
49# Grab the id of the image to launch
50IMAGE=`glance -f index | egrep $DEFAULT_IMAGE_NAME | head -1 | cut -d" " -f1`
51
52die_if_not_set IMAGE "Failure getting image"
53
54# Instance and volume names
55INSTANCE_NAME=${INSTANCE_NAME:-test_instance}
56VOL_INSTANCE_NAME=${VOL_INSTANCE_NAME:-test_vol_instance}
57VOL_NAME=${VOL_NAME:-test_volume}
58
59# Clean-up from previous runs
60nova delete $VOL_INSTANCE_NAME || true
61nova delete $INSTANCE_NAME || true
62
63# Wait till server is gone
64if ! timeout $ACTIVE_TIMEOUT sh -c "while nova show $INSTANCE_NAME; do sleep 1; done"; then
65 echo "server didn't terminate!"
66 exit 1
67fi
68
69# Configure Security Groups
70SECGROUP=${SECGROUP:-test_secgroup}
71nova secgroup-delete $SECGROUP || true
72nova secgroup-create $SECGROUP "$SECGROUP description"
73nova secgroup-add-rule $SECGROUP icmp -1 -1 0.0.0.0/0
74nova secgroup-add-rule $SECGROUP tcp 22 22 0.0.0.0/0
75
76# Determinine instance type
77INSTANCE_TYPE=`nova flavor-list | grep $DEFAULT_INSTANCE_TYPE | cut -d"|" -f2`
78if [[ -z "$INSTANCE_TYPE" ]]; then
79 # grab the first flavor in the list to launch if default doesn't exist
80 INSTANCE_TYPE=`nova flavor-list | head -n 4 | tail -n 1 | cut -d"|" -f2`
81fi
82
83# Setup Keypair
84KEY_NAME=test_key
85KEY_FILE=key.pem
86nova keypair-delete $KEY_NAME || true
87nova keypair-add $KEY_NAME > $KEY_FILE
88chmod 600 $KEY_FILE
89
90# Boot our instance
91VM_UUID=`nova boot --flavor $INSTANCE_TYPE --image $IMAGE --security_groups=$SECGROUP --key_name $KEY_NAME $INSTANCE_NAME | grep ' id ' | cut -d"|" -f3 | sed 's/ //g'`
92
93# check that the status is active within ACTIVE_TIMEOUT seconds
94if ! timeout $ACTIVE_TIMEOUT sh -c "while ! nova show $VM_UUID | grep status | grep -q ACTIVE; do sleep 1; done"; then
95 echo "server didn't become active!"
96 exit 1
97fi
98
99# Delete the old volume
100nova volume-delete $VOL_NAME || true
101
102# Free every floating ips - setting FREE_ALL_FLOATING_IPS=True in localrc will make life easier for testers
103if [ "$FREE_ALL_FLOATING_IPS" = "True" ]; then
104 nova floating-ip-list | grep nova | cut -d "|" -f2 | tr -d " " | xargs -n1 nova floating-ip-delete || true
105fi
106
107# Allocate floating ip
108FLOATING_IP=`nova floating-ip-create | grep $DEFAULT_FLOATING_POOL | cut -d '|' -f2 | tr -d ' '`
109
110# Make sure the ip gets allocated
111if ! timeout $ASSOCIATE_TIMEOUT sh -c "while ! nova floating-ip-list | grep -q $FLOATING_IP; do sleep 1; done"; then
112 echo "Floating IP not allocated"
113 exit 1
114fi
115
116# Add floating ip to our server
117nova add-floating-ip $VM_UUID $FLOATING_IP
118
119# Test we can ping our floating ip within ASSOCIATE_TIMEOUT seconds
120if ! timeout $ASSOCIATE_TIMEOUT sh -c "while ! ping -c1 -w1 $FLOATING_IP; do sleep 1; done"; then
121 echo "Couldn't ping server with floating ip"
122 exit 1
123fi
124
125# Create our volume
126nova volume-create --display_name=$VOL_NAME 1
127
128# Wait for volume to activate
129if ! timeout $ACTIVE_TIMEOUT sh -c "while ! nova volume-list | grep $VOL_NAME | grep available; do sleep 1; done"; then
130 echo "Volume $VOL_NAME not created"
131 exit 1
132fi
133
134# FIXME (anthony) - python-novaclient should accept a volume_name for the attachment param?
135DEVICE=/dev/vdb
136VOLUME_ID=`nova volume-list | grep $VOL_NAME | cut -d '|' -f 2 | tr -d ' '`
137nova volume-attach $INSTANCE_NAME $VOLUME_ID $DEVICE
138
139# Wait till volume is attached
140if ! timeout $ACTIVE_TIMEOUT sh -c "while ! nova volume-list | grep $VOL_NAME | grep in-use; do sleep 1; done"; then
141 echo "Volume $VOL_NAME not created"
142 exit 1
143fi
144
145# The following script builds our bootable volume.
146# To do this, ssh to the builder instance, mount volume, and build a volume-backed image.
147STAGING_DIR=/tmp/stage
148CIRROS_DIR=/tmp/cirros
149ssh -o StrictHostKeyChecking=no -i $KEY_FILE cirros@$FLOATING_IP << EOF
150set -o errexit
151set -o xtrace
152sudo mkdir -p $STAGING_DIR
153sudo mkfs.ext3 -b 1024 $DEVICE 1048576
154sudo mount $DEVICE $STAGING_DIR
155# The following lines create a writable empty file so that we can scp
156# the actual file
157sudo touch $STAGING_DIR/cirros-0.3.0-x86_64-rootfs.img.gz
158sudo chown cirros $STAGING_DIR/cirros-0.3.0-x86_64-rootfs.img.gz
159EOF
160
161# Download cirros
162if [ ! -e cirros-0.3.0-x86_64-rootfs.img.gz ]; then
163 wget http://images.ansolabs.com/cirros-0.3.0-x86_64-rootfs.img.gz
164fi
165
166# Copy cirros onto the volume
167scp -o StrictHostKeyChecking=no -i $KEY_FILE cirros-0.3.0-x86_64-rootfs.img.gz cirros@$FLOATING_IP:$STAGING_DIR
168
169# Unpack cirros into volume
170ssh -o StrictHostKeyChecking=no -i $KEY_FILE cirros@$FLOATING_IP << EOF
171set -o errexit
172set -o xtrace
173cd $STAGING_DIR
174sudo mkdir -p $CIRROS_DIR
175sudo gunzip cirros-0.3.0-x86_64-rootfs.img.gz
176sudo mount cirros-0.3.0-x86_64-rootfs.img $CIRROS_DIR
177
178# Copy cirros into our volume
179sudo cp -pr $CIRROS_DIR/* $STAGING_DIR/
180
181cd
182sync
183sudo umount $CIRROS_DIR
184# The following typically fails. Don't know why.
185sudo umount $STAGING_DIR || true
186EOF
187
188# Detach the volume from the builder instance
189nova volume-detach $INSTANCE_NAME $VOLUME_ID
190
191# Boot instance from volume! This is done with the --block_device_mapping param.
192# The format of mapping is:
193# <dev_name>=<id>:<type>:<size(GB)>:<delete_on_terminate>
194# Leaving the middle two fields blank appears to do-the-right-thing
195VOL_VM_UUID=`nova boot --flavor $INSTANCE_TYPE --image $IMAGE --block_device_mapping vda=$VOLUME_ID:::0 --security_groups=$SECGROUP --key_name $KEY_NAME $VOL_INSTANCE_NAME | grep ' id ' | cut -d"|" -f3 | sed 's/ //g'`
196
197# Check that the status is active within ACTIVE_TIMEOUT seconds
198if ! timeout $ACTIVE_TIMEOUT sh -c "while ! nova show $VOL_VM_UUID | grep status | grep -q ACTIVE; do sleep 1; done"; then
199 echo "server didn't become active!"
200 exit 1
201fi
202
203# Add floating ip to our server
204nova remove-floating-ip $VM_UUID $FLOATING_IP
205
206# Gratuitous sleep, probably hiding a race condition :/
207sleep 1
208
209# Add floating ip to our server
210nova add-floating-ip $VOL_VM_UUID $FLOATING_IP
211
212# Test we can ping our floating ip within ASSOCIATE_TIMEOUT seconds
213if ! timeout $ASSOCIATE_TIMEOUT sh -c "while ! ping -c1 -w1 $FLOATING_IP; do sleep 1; done"; then
214 echo "Couldn't ping volume-backed server with floating ip"
215 exit 1
216fi
217
218# Make sure our volume-backed instance launched
219ssh -o StrictHostKeyChecking=no -i $KEY_FILE cirros@$FLOATING_IP << EOF
220echo "success!"
221EOF
222
223# Delete volume backed instance
224nova delete $VOL_INSTANCE_NAME
225
226# Wait till our volume is no longer in-use
227if ! timeout $ACTIVE_TIMEOUT sh -c "while ! nova volume-list | grep $VOL_NAME | grep available; do sleep 1; done"; then
228 echo "Volume $VOL_NAME not created"
229 exit 1
230fi
231
232# Delete the volume
233nova volume-delete $VOL_NAME
234
235# Delete instance
236nova delete $INSTANCE_NAME
237
238# Wait for termination
239if ! timeout $ACTIVE_TIMEOUT sh -c "while nova show $INSTANCE_NAME; do sleep 1; done"; then
240 echo "server didn't terminate!"
241 exit 1
242fi
243
244# De-allocate the floating ip
245nova floating-ip-delete $FLOATING_IP
246
247# Delete secgroup
248nova secgroup-delete $SECGROUP
249
250set +o xtrace
251echo "**************************************************"
252echo "End DevStack Exercise: $0"
253echo "**************************************************"