blob: aaa7281a98c919123d291c4fc27df793854ce03f [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 Treinish859cc682013-07-26 15:22:44 -0400204
Attila Fazekas2aa35172012-12-05 20:03:40 +0100205 # Timeouts
206 iniset $TEMPEST_CONF compute build_timeout $BUILD_TIMEOUT
207 iniset $TEMPEST_CONF volume build_timeout $BUILD_TIMEOUT
208 iniset $TEMPEST_CONF boto build_timeout $BUILD_TIMEOUT
209 iniset $TEMPEST_CONF compute build_interval $BUILD_INTERVAL
210 iniset $TEMPEST_CONF volume build_interval $BUILD_INTERVAL
211 iniset $TEMPEST_CONF boto build_interval $BUILD_INTERVAL
212 iniset $TEMPEST_CONF boto http_socket_timeout 5
213
Attila Fazekas97d3d202013-01-19 19:20:49 +0100214 # Identity
215 iniset $TEMPEST_CONF identity uri "$KEYSTONE_SERVICE_PROTOCOL://$KEYSTONE_SERVICE_HOST:5000/v2.0/"
216 iniset $TEMPEST_CONF identity password "$password"
217 iniset $TEMPEST_CONF identity alt_username $ALT_USERNAME
218 iniset $TEMPEST_CONF identity alt_password "$password"
219 iniset $TEMPEST_CONF identity alt_tenant_name $ALT_TENANT_NAME
220 iniset $TEMPEST_CONF identity admin_password "$password"
Attila Fazekas2aa35172012-12-05 20:03:40 +0100221
Sean Daguec3771452013-06-13 14:23:37 -0400222 # Image
223 # for the gate we want to be able to override this variable so we aren't
224 # doing an HTTP fetch over the wide internet for this test
225 if [[ ! -z "$TEMPEST_HTTP_IMAGE" ]]; then
226 iniset $TEMPEST_CONF image http_image $TEMPEST_HTTP_IMAGE
227 fi
228
Attila Fazekas97d3d202013-01-19 19:20:49 +0100229 # Compute
Attila Fazekas2aa35172012-12-05 20:03:40 +0100230 iniset $TEMPEST_CONF compute change_password_available False
Nachi Ueno06fac372012-12-26 14:09:43 -0800231 # Note(nati) current tempest don't create network for each tenant
232 # so reuse same tenant for now
Mark McClainb05c8762013-07-06 23:29:39 -0400233 if is_service_enabled neutron; then
Nachi Ueno06fac372012-12-26 14:09:43 -0800234 TEMPEST_ALLOW_TENANT_ISOLATION=${TEMPEST_ALLOW_TENANT_ISOLATION:-False}
235 fi
236 iniset $TEMPEST_CONF compute allow_tenant_isolation ${TEMPEST_ALLOW_TENANT_ISOLATION:-True}
Attila Fazekas97d3d202013-01-19 19:20:49 +0100237 iniset $TEMPEST_CONF compute ssh_user ${DEFAULT_INSTANCE_USER:-cirros} # DEPRECATED
Akihiro MOTOKI66afb472012-12-21 15:34:13 +0900238 iniset $TEMPEST_CONF compute network_for_ssh $PRIVATE_NETWORK_NAME
Attila Fazekas2aa35172012-12-05 20:03:40 +0100239 iniset $TEMPEST_CONF compute ip_version_for_ssh 4
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100240 iniset $TEMPEST_CONF compute ssh_timeout $BUILD_TIMEOUT
Attila Fazekas65c08462012-12-07 14:20:51 +0100241 iniset $TEMPEST_CONF compute image_ref $image_uuid
Attila Fazekas97d3d202013-01-19 19:20:49 +0100242 iniset $TEMPEST_CONF compute image_ssh_user ${DEFAULT_INSTANCE_USER:-cirros}
Attila Fazekas65c08462012-12-07 14:20:51 +0100243 iniset $TEMPEST_CONF compute image_ref_alt $image_uuid_alt
Attila Fazekas97d3d202013-01-19 19:20:49 +0100244 iniset $TEMPEST_CONF compute image_alt_ssh_user ${DEFAULT_INSTANCE_USER:-cirros}
Attila Fazekas65c08462012-12-07 14:20:51 +0100245 iniset $TEMPEST_CONF compute flavor_ref $flavor_ref
246 iniset $TEMPEST_CONF compute flavor_ref_alt $flavor_ref_alt
Attila Fazekas2aa35172012-12-05 20:03:40 +0100247 iniset $TEMPEST_CONF compute live_migration_available ${LIVE_MIGRATION_AVAILABLE:-False}
Akihiro MOTOKI66afb472012-12-21 15:34:13 +0900248 iniset $TEMPEST_CONF compute use_block_migration_for_live_migration ${USE_BLOCK_MIGRATION_FOR_LIVE_MIGRATION:-False}
Attila Fazekas97d3d202013-01-19 19:20:49 +0100249
250 # Whitebox
251 iniset $TEMPEST_CONF whitebox source_dir $NOVA_SOURCE_DIR
252 iniset $TEMPEST_CONF whitebox bin_dir $NOVA_BIN_DIR
Attila Fazekas2aa35172012-12-05 20:03:40 +0100253 # TODO(jaypipes): Create the key file here... right now, no whitebox
254 # tests actually use a key.
Attila Fazekas97d3d202013-01-19 19:20:49 +0100255 iniset $TEMPEST_CONF whitebox path_to_private_key $TEMPEST_DIR/id_rsa
256 iniset $TEMPEST_CONF whitebox db_uri $BASE_SQL_CONN/nova
257
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200258 # Compute admin
Attila Fazekas97d3d202013-01-19 19:20:49 +0100259 iniset $TEMPEST_CONF "compute-admin" password "$password" # DEPRECATED
Attila Fazekas2aa35172012-12-05 20:03:40 +0100260
Attila Fazekas2aa35172012-12-05 20:03:40 +0100261 iniset $TEMPEST_CONF network api_version 2.0
Maru Newbya5c774e2012-12-10 10:40:01 +0000262 iniset $TEMPEST_CONF network tenant_networks_reachable "$tenant_networks_reachable"
263 iniset $TEMPEST_CONF network public_network_id "$public_network_id"
Maru Newby31c94ab2012-12-19 03:59:20 +0000264 iniset $TEMPEST_CONF network public_router_id "$public_router_id"
Attila Fazekas2aa35172012-12-05 20:03:40 +0100265
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200266 # boto
Attila Fazekas2aa35172012-12-05 20:03:40 +0100267 iniset $TEMPEST_CONF boto ec2_url "http://$SERVICE_HOST:8773/services/Cloud"
268 iniset $TEMPEST_CONF boto s3_url "http://$SERVICE_HOST:${S3_SERVICE_PORT:-3333}"
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100269 iniset $TEMPEST_CONF boto s3_materials_path "$BOTO_MATERIALS_PATH"
270 iniset $TEMPEST_CONF boto instance_type "$boto_instance_type"
271 iniset $TEMPEST_CONF boto http_socket_timeout 30
Attila Fazekas97d3d202013-01-19 19:20:49 +0100272 iniset $TEMPEST_CONF boto ssh_user ${DEFAULT_INSTANCE_USER:-cirros}
Attila Fazekas2aa35172012-12-05 20:03:40 +0100273
William Marshall24d866e2013-07-02 12:26:31 -0500274 # Scenario
275 iniset $TEMPEST_CONF scenario img_dir "$FILES/images/cirros-0.3.1-x86_64-uec"
276
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200277 # Volume
278 CINDER_MULTI_LVM_BACKEND=$(trueorfalse False $CINDER_MULTI_LVM_BACKEND)
Ian Wienandc42ed252013-06-17 13:23:44 +1000279 if [ $CINDER_MULTI_LVM_BACKEND == "True" ]; then
Giulio Fidente97e1bd02013-06-04 05:33:52 +0200280 iniset $TEMPEST_CONF volume multi_backend_enabled "True"
281 iniset $TEMPEST_CONF volume backend1_name "LVM_iSCSI"
282 iniset $TEMPEST_CONF volume backend2_name "LVM_iSCSI_2"
283 fi
284
Julie Pichonfea70f82013-07-29 11:22:08 +0100285 # Dashboard
286 iniset $TEMPEST_CONF dashboard dashboard_url "http://$SERVICE_HOST/"
287 iniset $TEMPEST_CONF dashboard login_url "http://$SERVICE_HOST/auth/login/"
288
Ian Wienand7fa19022013-06-25 14:11:48 +1000289 # cli
290 iniset $TEMPEST_CONF cli cli_dir $NOVA_BIN_DIR
291
Matthew Treinishb56d81d2013-07-23 17:25:39 -0400292 # service_available
Julie Pichonfea70f82013-07-29 11:22:08 +0100293 for service in nova cinder glance neutron swift heat horizon ; do
Matthew Treinishb56d81d2013-07-23 17:25:39 -0400294 if is_service_enabled $service ; then
295 iniset $TEMPEST_CONF service_available $service "True"
296 else
297 iniset $TEMPEST_CONF service_available $service "False"
298 fi
299 done
300
Attila Fazekas2aa35172012-12-05 20:03:40 +0100301 echo "Created tempest configuration file:"
302 cat $TEMPEST_CONF
303
304 # Restore IFS
305 IFS=$ifs
306 #Restore errexit
307 $errexit
Sean Dagued0931212012-10-04 16:06:44 -0400308}
309
Sean Dagued0931212012-10-04 16:06:44 -0400310# install_tempest() - Collect source and prepare
311function install_tempest() {
312 git_clone $TEMPEST_REPO $TEMPEST_DIR $TEMPEST_BRANCH
Sean Dagued0931212012-10-04 16:06:44 -0400313}
314
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100315# init_tempest() - Initialize ec2 images
316function init_tempest() {
Scott Mosercde655a2013-02-20 12:33:39 -0500317 local base_image_name=cirros-0.3.1-x86_64
318 # /opt/stack/devstack/files/images/cirros-0.3.1-x86_64-uec
Attila Fazekas3c529222013-01-21 06:50:33 +0100319 local image_dir="$FILES/images/${base_image_name}-uec"
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100320 local kernel="$image_dir/${base_image_name}-vmlinuz"
321 local ramdisk="$image_dir/${base_image_name}-initrd"
322 local disk_image="$image_dir/${base_image_name}-blank.img"
323 # if the cirros uec downloaded and the system is uec capable
324 if [ -f "$kernel" -a -f "$ramdisk" -a -f "$disk_image" -a "$VIRT_DRIVER" != "openvz" \
325 -a \( "$LIBVIRT_TYPE" != "lxc" -o "$VIRT_DRIVER" != "libvirt" \) ]; then
326 echo "Prepare aki/ari/ami Images"
327 ( #new namespace
328 # tenant:demo ; user: demo
Chris Yeohb0b98b72013-01-23 21:14:49 +1030329 source $TOP_DIR/accrc/demo/demo
Attila Fazekas1d29d8b2013-01-07 15:51:32 +0100330 euca-bundle-image -i "$kernel" --kernel true -d "$BOTO_MATERIALS_PATH"
331 euca-bundle-image -i "$ramdisk" --ramdisk true -d "$BOTO_MATERIALS_PATH"
332 euca-bundle-image -i "$disk_image" -d "$BOTO_MATERIALS_PATH"
333 ) 2>&1 </dev/null | cat
334 else
335 echo "Boto materials are not prepared"
336 fi
337}
338
Sean Dagued0931212012-10-04 16:06:44 -0400339# Restore xtrace
340$XTRACE
Sean Dague584d90e2013-03-29 14:34:53 -0400341
342# Local variables:
343# mode: shell-script
344# End: