blob: b97f0d86a939969477cba9815835f58da79ccd5f [file] [log] [blame]
Sean Dagued0931212012-10-04 16:06:44 -04001# lib/tempest
Dean Troyer6d04fd72012-12-21 11:03:37 -06002# Install and configure Tempest
Sean Dagued0931212012-10-04 16:06:44 -04003
4# Dependencies:
5# ``functions`` file
Joe Gordon46400262013-06-30 04:32:27 -07006# ``lib/nova`` service is running
Sean Dagued0931212012-10-04 16:06:44 -04007# <list other global vars that are assumed to be defined>
Attila Fazekas3c529222013-01-21 06:50:33 +01008# - ``DEST``, ``FILES``
Attila Fazekas65c08462012-12-07 14:20:51 +01009# - ``ADMIN_PASSWORD``
10# - ``DEFAULT_IMAGE_NAME``
11# - ``S3_SERVICE_PORT``
12# - ``SERVICE_HOST``
13# - ``BASE_SQL_CONN`` ``lib/database`` declares
Maru Newby31c94ab2012-12-19 03:59:20 +000014# - ``PUBLIC_NETWORK_NAME``
15# - ``Q_USE_NAMESPACE``
16# - ``Q_ROUTER_NAME``
Attila Fazekas1d29d8b2013-01-07 15:51:32 +010017# - ``VIRT_DRIVER``
18# - ``LIBVIRT_TYPE``
Attila Fazekas97d3d202013-01-19 19:20:49 +010019# - ``KEYSTONE_SERVICE_PROTOCOL``, ``KEYSTONE_SERVICE_HOST`` from lib/keystone
Attila Fazekas2aa35172012-12-05 20:03:40 +010020# Optional Dependencies:
Attila Fazekas2aa35172012-12-05 20:03:40 +010021# ALT_* (similar vars exists in keystone_data.sh)
Attila Fazekas65c08462012-12-07 14:20:51 +010022# ``LIVE_MIGRATION_AVAILABLE``
23# ``USE_BLOCK_MIGRATION_FOR_LIVE_MIGRATION``
24# ``DEFAULT_INSTANCE_TYPE``
25# ``DEFAULT_INSTANCE_USER``
Giulio Fidente97e1bd02013-06-04 05:33:52 +020026# ``CINDER_MULTI_LVM_BACKEND``
Sean Dagued0931212012-10-04 16:06:44 -040027# ``stack.sh`` calls the entry points in this order:
28#
Attila Fazekas2aa35172012-12-05 20:03:40 +010029# install_tempest
30# configure_tempest
Attila Fazekas1d29d8b2013-01-07 15:51:32 +010031# init_tempest
Sean Dagued0931212012-10-04 16:06:44 -040032
33# Save trace setting
34XTRACE=$(set +o | grep xtrace)
35set +o xtrace
36
Dean Troyer6d04fd72012-12-21 11:03:37 -060037
Sean Dagued0931212012-10-04 16:06:44 -040038# Defaults
39# --------
40
Sean Dagued0931212012-10-04 16:06:44 -040041# Set up default directories
42TEMPEST_DIR=$DEST/tempest
Attila Fazekas2aa35172012-12-05 20:03:40 +010043TEMPEST_CONF_DIR=$TEMPEST_DIR/etc
44TEMPEST_CONF=$TEMPEST_CONF_DIR/tempest.conf
Matthew Treinish14ccba02013-07-29 16:15:53 -040045TEMPEST_STATE_PATH=${TEMPEST_STATE_PATH:=$DATA_DIR/tempest}
Attila Fazekas2aa35172012-12-05 20:03:40 +010046
Dean Troyer6d04fd72012-12-21 11:03:37 -060047NOVA_SOURCE_DIR=$DEST/nova
48
Attila Fazekasbae02332013-05-09 09:24:49 +020049BUILD_INTERVAL=1
Attila Fazekas2aa35172012-12-05 20:03:40 +010050BUILD_TIMEOUT=400
Sean Dagued0931212012-10-04 16:06:44 -040051
Dean Troyer6d04fd72012-12-21 11:03:37 -060052
Scott Mosercde655a2013-02-20 12:33:39 -050053BOTO_MATERIALS_PATH="$FILES/images/s3-materials/cirros-0.3.1"
Attila Fazekas1d29d8b2013-01-07 15:51:32 +010054
Dean Troyercc6b4432013-04-08 15:38:03 -050055
56# Functions
57# ---------
Sean Dagued0931212012-10-04 16:06:44 -040058
Sean Dagued0931212012-10-04 16:06:44 -040059# configure_tempest() - Set config files, create data dirs, etc
60function configure_tempest() {
Matthew Treinish95fb0d42013-01-28 16:24:14 -050061 setup_develop $TEMPEST_DIR
Attila Fazekas65c08462012-12-07 14:20:51 +010062 local image_lines
63 local images
64 local num_images
65 local image_uuid
66 local image_uuid_alt
Attila Fazekas2aa35172012-12-05 20:03:40 +010067 local errexit
Attila Fazekas65c08462012-12-07 14:20:51 +010068 local password
69 local line
70 local flavors
71 local flavors_ref
72 local flavor_lines
Maru Newbya5c774e2012-12-10 10:40:01 +000073 local public_network_id
Maru Newby31c94ab2012-12-19 03:59:20 +000074 local public_router_id
Maru Newbya5c774e2012-12-10 10:40:01 +000075 local tenant_networks_reachable
Attila Fazekas1d29d8b2013-01-07 15:51:32 +010076 local boto_instance_type="m1.tiny"
Attila Fazekas2aa35172012-12-05 20:03:40 +010077
Dean Troyer6d04fd72012-12-21 11:03:37 -060078 # TODO(afazekas):
Sean Dagued0931212012-10-04 16:06:44 -040079 # sudo python setup.py deploy
Attila Fazekas2aa35172012-12-05 20:03:40 +010080
81 # This function exits on an error so that errors don't compound and you see
Joe Gordon46400262013-06-30 04:32:27 -070082 # only the first error that occurred.
Attila Fazekas2aa35172012-12-05 20:03:40 +010083 errexit=$(set +o | grep errexit)
84 set -o errexit
85
Dean Troyer6d04fd72012-12-21 11:03:37 -060086 # Save IFS
Attila Fazekas2aa35172012-12-05 20:03:40 +010087 ifs=$IFS
88
89 # Glance should already contain images to be used in tempest
90 # testing. Here we simply look for images stored in Glance
91 # and set the appropriate variables for use in the tempest config
92 # We ignore ramdisk and kernel images, look for the default image
Attila Fazekas65c08462012-12-07 14:20:51 +010093 # ``DEFAULT_IMAGE_NAME``. If not found, we set the ``image_uuid`` to the
94 # first image returned and set ``image_uuid_alt`` to the second,
Attila Fazekas2aa35172012-12-05 20:03:40 +010095 # if there is more than one returned...
96 # ... Also ensure we only take active images, so we don't get snapshots in process
Cody A.W. Somervillec24e23b2012-12-21 02:10:45 -050097 declare -a images
98
99 while read -r IMAGE_NAME IMAGE_UUID; do
100 if [ "$IMAGE_NAME" = "$DEFAULT_IMAGE_NAME" ]; then
101 image_uuid="$IMAGE_UUID"
102 image_uuid_alt="$IMAGE_UUID"
Attila Fazekas2aa35172012-12-05 20:03:40 +0100103 fi
Cody A.W. Somervillec24e23b2012-12-21 02:10:45 -0500104 images+=($IMAGE_UUID)
105 done < <(glance image-list --status=active | awk -F'|' '!/^(+--)|ID|aki|ari/ { print $3,$2 }')
106
107 case "${#images[*]}" in
108 0)
109 echo "Found no valid images to use!"
110 exit 1
111 ;;
112 1)
113 if [ -z "$image_uuid" ]; then
114 image_uuid=${images[0]}
115 image_uuid_alt=${images[0]}
116 fi
117 ;;
118 *)
119 if [ -z "$image_uuid" ]; then
120 image_uuid=${images[0]}
121 image_uuid_alt=${images[1]}
122 fi
123 ;;
124 esac
Attila Fazekas2aa35172012-12-05 20:03:40 +0100125
126 # Create tempest.conf from tempest.conf.sample
127 # copy every time, because the image UUIDS are going to change
128 cp $TEMPEST_CONF.sample $TEMPEST_CONF
129
Attila Fazekas65c08462012-12-07 14:20:51 +0100130 password=${ADMIN_PASSWORD:-secrete}
Attila Fazekas2aa35172012-12-05 20:03:40 +0100131
132 # See files/keystone_data.sh where alt_demo user
133 # and tenant are set up...
134 ALT_USERNAME=${ALT_USERNAME:-alt_demo}
135 ALT_TENANT_NAME=${ALT_TENANT_NAME:-alt_demo}
136
Attila Fazekas7bf1dd32013-01-12 17:31:26 +0100137 # If the ``DEFAULT_INSTANCE_TYPE`` not declared, use the new behavior
138 # Tempest creates instane types for himself
139 if [[ -z "$DEFAULT_INSTANCE_TYPE" ]]; then
Attila Fazekas02c0bcc2013-01-14 19:10:17 +0100140 nova flavor-create m1.nano 42 64 0 1
Attila Fazekas7bf1dd32013-01-12 17:31:26 +0100141 flavor_ref=42
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100142 boto_instance_type=m1.nano
Attila Fazekas02c0bcc2013-01-14 19:10:17 +0100143 nova flavor-create m1.micro 84 128 0 1
Attila Fazekas7bf1dd32013-01-12 17:31:26 +0100144 flavor_ref_alt=84
145 else
146 # Check Nova for existing flavors and, if set, look for the
147 # ``DEFAULT_INSTANCE_TYPE`` and use that.
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100148 boto_instance_type=$DEFAULT_INSTANCE_TYPE
Attila Fazekas7bf1dd32013-01-12 17:31:26 +0100149 flavor_lines=`nova flavor-list`
150 IFS=$'\r\n'
151 flavors=""
Dean Troyerceaa38b2012-12-12 17:09:57 -0600152 for line in $flavor_lines; do
153 f=$(echo $line | awk "/ $DEFAULT_INSTANCE_TYPE / { print \$2 }")
154 flavors="$flavors $f"
155 done
Attila Fazekas2aa35172012-12-05 20:03:40 +0100156
Attila Fazekas7bf1dd32013-01-12 17:31:26 +0100157 for line in $flavor_lines; do
158 flavors="$flavors `echo $line | grep -v "^\(|\s*ID\|+--\)" | cut -d' ' -f2`"
159 done
160
161 IFS=" "
162 flavors=($flavors)
163 num_flavors=${#flavors[*]}
164 echo "Found $num_flavors flavors"
165 if [[ $num_flavors -eq 0 ]]; then
166 echo "Found no valid flavors to use!"
167 exit 1
168 fi
169 flavor_ref=${flavors[0]}
170 flavor_ref_alt=$flavor_ref
Adalberto Medeiros63a71a22013-06-06 08:46:04 -0400171
172 # ensure flavor_ref and flavor_ref_alt have different values
173 # some resize instance in tempest tests depends on this.
174 for f in ${flavors[@]:1}; do
175 if [[ $f -ne $flavor_ref ]]; then
176 flavor_ref_alt=$f
177 break
178 fi
179 done
Attila Fazekas2aa35172012-12-05 20:03:40 +0100180 fi
181
Maru Newbya5c774e2012-12-10 10:40:01 +0000182 if [ "$Q_USE_NAMESPACE" != "False" ]; then
183 tenant_networks_reachable=false
184 else
185 tenant_networks_reachable=true
186 fi
187
188 if is_service_enabled q-l3; then
Mark McClainb05c8762013-07-06 23:29:39 -0400189 public_network_id=$(neutron net-list | grep $PUBLIC_NETWORK_NAME | \
Maru Newbya5c774e2012-12-10 10:40:01 +0000190 awk '{print $2}')
Maru Newby31c94ab2012-12-19 03:59:20 +0000191 if [ "$Q_USE_NAMESPACE" == "False" ]; then
192 # If namespaces are disabled, devstack will create a single
193 # public router that tempest should be configured to use.
Mark McClainb05c8762013-07-06 23:29:39 -0400194 public_router_id=$(neutron router-list | awk "/ $Q_ROUTER_NAME / \
Maru Newby31c94ab2012-12-19 03:59:20 +0000195 { print \$2 }")
196 fi
Maru Newbya5c774e2012-12-10 10:40:01 +0000197 fi
198
Matthew Treinish859cc682013-07-26 15:22:44 -0400199 # Oslo
Matthew Treinish14ccba02013-07-29 16:15:53 -0400200 iniset $TEMPEST_CONF DEFAULT lock_path $TEMPEST_STATE_PATH
201 mkdir -p $TEMPEST_STATE_PATH
Matthew Treinish4ee4a012013-07-31 16:05:42 -0400202 iniset $TEMPEST_CONF DEFAULT use_stderr False
203 iniset $TEMPEST_CONF DEFAULT log_file tempest.log
Matthew Treinish385152c2013-08-09 11:13:28 -0400204 iniset $TEMPEST_CONF DEFAULT debug True
Matthew Treinish859cc682013-07-26 15:22:44 -0400205
Attila Fazekas2aa35172012-12-05 20:03:40 +0100206 # Timeouts
207 iniset $TEMPEST_CONF compute build_timeout $BUILD_TIMEOUT
208 iniset $TEMPEST_CONF volume build_timeout $BUILD_TIMEOUT
209 iniset $TEMPEST_CONF boto build_timeout $BUILD_TIMEOUT
210 iniset $TEMPEST_CONF compute build_interval $BUILD_INTERVAL
211 iniset $TEMPEST_CONF volume build_interval $BUILD_INTERVAL
212 iniset $TEMPEST_CONF boto build_interval $BUILD_INTERVAL
213 iniset $TEMPEST_CONF boto http_socket_timeout 5
214
Attila Fazekas97d3d202013-01-19 19:20:49 +0100215 # Identity
216 iniset $TEMPEST_CONF identity uri "$KEYSTONE_SERVICE_PROTOCOL://$KEYSTONE_SERVICE_HOST:5000/v2.0/"
217 iniset $TEMPEST_CONF identity password "$password"
218 iniset $TEMPEST_CONF identity alt_username $ALT_USERNAME
219 iniset $TEMPEST_CONF identity alt_password "$password"
220 iniset $TEMPEST_CONF identity alt_tenant_name $ALT_TENANT_NAME
221 iniset $TEMPEST_CONF identity admin_password "$password"
Attila Fazekas2aa35172012-12-05 20:03:40 +0100222
Sean Daguec3771452013-06-13 14:23:37 -0400223 # Image
224 # for the gate we want to be able to override this variable so we aren't
225 # doing an HTTP fetch over the wide internet for this test
226 if [[ ! -z "$TEMPEST_HTTP_IMAGE" ]]; then
227 iniset $TEMPEST_CONF image http_image $TEMPEST_HTTP_IMAGE
228 fi
229
Attila Fazekas97d3d202013-01-19 19:20:49 +0100230 # Compute
Attila Fazekas2aa35172012-12-05 20:03:40 +0100231 iniset $TEMPEST_CONF compute change_password_available False
Nachi Ueno06fac372012-12-26 14:09:43 -0800232 # Note(nati) current tempest don't create network for each tenant
233 # so reuse same tenant for now
Mark McClainb05c8762013-07-06 23:29:39 -0400234 if is_service_enabled neutron; then
Nachi Ueno06fac372012-12-26 14:09:43 -0800235 TEMPEST_ALLOW_TENANT_ISOLATION=${TEMPEST_ALLOW_TENANT_ISOLATION:-False}
236 fi
237 iniset $TEMPEST_CONF compute allow_tenant_isolation ${TEMPEST_ALLOW_TENANT_ISOLATION:-True}
Attila Fazekas97d3d202013-01-19 19:20:49 +0100238 iniset $TEMPEST_CONF compute ssh_user ${DEFAULT_INSTANCE_USER:-cirros} # DEPRECATED
Akihiro MOTOKI66afb472012-12-21 15:34:13 +0900239 iniset $TEMPEST_CONF compute network_for_ssh $PRIVATE_NETWORK_NAME
Attila Fazekas2aa35172012-12-05 20:03:40 +0100240 iniset $TEMPEST_CONF compute ip_version_for_ssh 4
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100241 iniset $TEMPEST_CONF compute ssh_timeout $BUILD_TIMEOUT
Attila Fazekas65c08462012-12-07 14:20:51 +0100242 iniset $TEMPEST_CONF compute image_ref $image_uuid
Attila Fazekas97d3d202013-01-19 19:20:49 +0100243 iniset $TEMPEST_CONF compute image_ssh_user ${DEFAULT_INSTANCE_USER:-cirros}
Attila Fazekas65c08462012-12-07 14:20:51 +0100244 iniset $TEMPEST_CONF compute image_ref_alt $image_uuid_alt
Attila Fazekas97d3d202013-01-19 19:20:49 +0100245 iniset $TEMPEST_CONF compute image_alt_ssh_user ${DEFAULT_INSTANCE_USER:-cirros}
Attila Fazekas65c08462012-12-07 14:20:51 +0100246 iniset $TEMPEST_CONF compute flavor_ref $flavor_ref
247 iniset $TEMPEST_CONF compute flavor_ref_alt $flavor_ref_alt
Attila Fazekas2aa35172012-12-05 20:03:40 +0100248 iniset $TEMPEST_CONF compute live_migration_available ${LIVE_MIGRATION_AVAILABLE:-False}
Akihiro MOTOKI66afb472012-12-21 15:34:13 +0900249 iniset $TEMPEST_CONF compute use_block_migration_for_live_migration ${USE_BLOCK_MIGRATION_FOR_LIVE_MIGRATION:-False}
Attila Fazekas97d3d202013-01-19 19:20:49 +0100250
251 # Whitebox
252 iniset $TEMPEST_CONF whitebox source_dir $NOVA_SOURCE_DIR
253 iniset $TEMPEST_CONF whitebox bin_dir $NOVA_BIN_DIR
Attila Fazekas2aa35172012-12-05 20:03:40 +0100254 # TODO(jaypipes): Create the key file here... right now, no whitebox
255 # tests actually use a key.
Attila Fazekas97d3d202013-01-19 19:20:49 +0100256 iniset $TEMPEST_CONF whitebox path_to_private_key $TEMPEST_DIR/id_rsa
257 iniset $TEMPEST_CONF whitebox db_uri $BASE_SQL_CONN/nova
258
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200259 # Compute admin
Attila Fazekas97d3d202013-01-19 19:20:49 +0100260 iniset $TEMPEST_CONF "compute-admin" password "$password" # DEPRECATED
Attila Fazekas2aa35172012-12-05 20:03:40 +0100261
Attila Fazekas2aa35172012-12-05 20:03:40 +0100262 iniset $TEMPEST_CONF network api_version 2.0
Maru Newbya5c774e2012-12-10 10:40:01 +0000263 iniset $TEMPEST_CONF network tenant_networks_reachable "$tenant_networks_reachable"
264 iniset $TEMPEST_CONF network public_network_id "$public_network_id"
Maru Newby31c94ab2012-12-19 03:59:20 +0000265 iniset $TEMPEST_CONF network public_router_id "$public_router_id"
Attila Fazekas2aa35172012-12-05 20:03:40 +0100266
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200267 # boto
Attila Fazekas2aa35172012-12-05 20:03:40 +0100268 iniset $TEMPEST_CONF boto ec2_url "http://$SERVICE_HOST:8773/services/Cloud"
269 iniset $TEMPEST_CONF boto s3_url "http://$SERVICE_HOST:${S3_SERVICE_PORT:-3333}"
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100270 iniset $TEMPEST_CONF boto s3_materials_path "$BOTO_MATERIALS_PATH"
271 iniset $TEMPEST_CONF boto instance_type "$boto_instance_type"
272 iniset $TEMPEST_CONF boto http_socket_timeout 30
Attila Fazekas97d3d202013-01-19 19:20:49 +0100273 iniset $TEMPEST_CONF boto ssh_user ${DEFAULT_INSTANCE_USER:-cirros}
Attila Fazekas2aa35172012-12-05 20:03:40 +0100274
William Marshall24d866e2013-07-02 12:26:31 -0500275 # Scenario
276 iniset $TEMPEST_CONF scenario img_dir "$FILES/images/cirros-0.3.1-x86_64-uec"
277
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200278 # Volume
279 CINDER_MULTI_LVM_BACKEND=$(trueorfalse False $CINDER_MULTI_LVM_BACKEND)
Ian Wienandc42ed252013-06-17 13:23:44 +1000280 if [ $CINDER_MULTI_LVM_BACKEND == "True" ]; then
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200281 iniset $TEMPEST_CONF volume multi_backend_enabled "True"
282 iniset $TEMPEST_CONF volume backend1_name "LVM_iSCSI"
283 iniset $TEMPEST_CONF volume backend2_name "LVM_iSCSI_2"
284 fi
285
Julie Pichonfea70f82013-07-29 11:22:08 +0100286 # Dashboard
287 iniset $TEMPEST_CONF dashboard dashboard_url "http://$SERVICE_HOST/"
288 iniset $TEMPEST_CONF dashboard login_url "http://$SERVICE_HOST/auth/login/"
289
Ian Wienand7fa19022013-06-25 14:11:48 +1000290 # cli
291 iniset $TEMPEST_CONF cli cli_dir $NOVA_BIN_DIR
292
Matthew Treinishb56d81d2013-07-23 17:25:39 -0400293 # service_available
Julie Pichonfea70f82013-07-29 11:22:08 +0100294 for service in nova cinder glance neutron swift heat horizon ; do
Matthew Treinishb56d81d2013-07-23 17:25:39 -0400295 if is_service_enabled $service ; then
296 iniset $TEMPEST_CONF service_available $service "True"
297 else
298 iniset $TEMPEST_CONF service_available $service "False"
299 fi
300 done
301
Attila Fazekas2aa35172012-12-05 20:03:40 +0100302 echo "Created tempest configuration file:"
303 cat $TEMPEST_CONF
304
305 # Restore IFS
306 IFS=$ifs
307 #Restore errexit
308 $errexit
Sean Dagued0931212012-10-04 16:06:44 -0400309}
310
Sean Dagued0931212012-10-04 16:06:44 -0400311# install_tempest() - Collect source and prepare
312function install_tempest() {
313 git_clone $TEMPEST_REPO $TEMPEST_DIR $TEMPEST_BRANCH
Sean Dagued0931212012-10-04 16:06:44 -0400314}
315
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100316# init_tempest() - Initialize ec2 images
317function init_tempest() {
Scott Mosercde655a2013-02-20 12:33:39 -0500318 local base_image_name=cirros-0.3.1-x86_64
319 # /opt/stack/devstack/files/images/cirros-0.3.1-x86_64-uec
Attila Fazekas3c529222013-01-21 06:50:33 +0100320 local image_dir="$FILES/images/${base_image_name}-uec"
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100321 local kernel="$image_dir/${base_image_name}-vmlinuz"
322 local ramdisk="$image_dir/${base_image_name}-initrd"
323 local disk_image="$image_dir/${base_image_name}-blank.img"
324 # if the cirros uec downloaded and the system is uec capable
325 if [ -f "$kernel" -a -f "$ramdisk" -a -f "$disk_image" -a "$VIRT_DRIVER" != "openvz" \
326 -a \( "$LIBVIRT_TYPE" != "lxc" -o "$VIRT_DRIVER" != "libvirt" \) ]; then
327 echo "Prepare aki/ari/ami Images"
328 ( #new namespace
329 # tenant:demo ; user: demo
Chris Yeohb0b98b72013-01-23 21:14:49 +1030330 source $TOP_DIR/accrc/demo/demo
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100331 euca-bundle-image -i "$kernel" --kernel true -d "$BOTO_MATERIALS_PATH"
332 euca-bundle-image -i "$ramdisk" --ramdisk true -d "$BOTO_MATERIALS_PATH"
333 euca-bundle-image -i "$disk_image" -d "$BOTO_MATERIALS_PATH"
334 ) 2>&1 </dev/null | cat
335 else
336 echo "Boto materials are not prepared"
337 fi
338}
339
Sean Dagued0931212012-10-04 16:06:44 -0400340# Restore xtrace
341$XTRACE
Sean Dague584d90e2013-03-29 14:34:53 -0400342
343# Local variables:
344# mode: shell-script
345# End: