-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathrun.sh
executable file
·385 lines (325 loc) · 11.5 KB
/
run.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
#!/bin/bash
# shellcheck disable=SC2086,SC2016
set -e
# Common Definitions
TOP_DIR=$(dirname "$(readlink -f "$0")")
SCRIPTS_DIR="${TOP_DIR}/scripts"
TARGET_FILES_DIR="$(mktemp -d /tmp/cvm_target_files.XXXXXX)"
INPUT_IMG=""
OUTPUT_IMG="output.qcow2"
TIMEOUT=6
CONNECTION_SOCK=""
CONSOLE_OPT=""
# Scan all subdirectories from pre-stage and post-stage
plugin_dirs=("$TOP_DIR/plugins"/*/)
# Include common definitions and utilities
# shellcheck disable=SC1091
source ${SCRIPTS_DIR}/common.sh
#
# Display Usage information
#
usage() {
cat <<EOM
Usage: $(basename "$0") [OPTION]...
Required
-i <guest image> Specify initial guest image file
Optional
-t <number of minutes> Specify the timeout of rewriting, 6 minutes default,
If enabling ima, recommend timeout >8 minutes
-o <output file name> Specify the output file name, default is output.qcow2
-s <connection socket> Default connection URI is qemu:///system,
if install libvirt, you can specify to "/var/run/libvirt/libvirt-sock"
then the corresponding URI is "qemu+unix:///system?socket=/var/run/libvirt/libvirt-sock"
-n Silence running for virt-install, no output
EOM
}
#
# Update plugin directory list
#
# To skip the installation of a specific plugin place a file called "NOT_RUN"
# in each plugin directory to be skipped. For example:
# ``touch plugins/01-resize-image/NOT_RUN``
#
update_plugin_dirs() {
echo "Update plugin dirs ..."
tmp_dirs=()
for path_item in "${plugin_dirs[@]}"
do
if [[ -e "$path_item/NOT_RUN" ]]; then
info "Skip to run $path_item ... "
else
tmp_dirs+=("$path_item")
fi
done
plugin_dirs=("${tmp_dirs[@]}")
}
#
# Prepare the files copying to target guest image.
#
# 1. Scan the following content from all subdirectories under plugins
# a) files ==> the root of target guest image
# b) guest_run.sh ==> /opt/guest_scripts at target guest image
# 2. Copy all files to staging directory at $TAGET_FILES_DIR
# 3. Create rootfs_override.tar.gz from $TAGET_FILES_DIR
# 4. Copy rootfs_overide.tar.gz to target system and extract
#
prepare_target_files() {
echo "Prepare target files ..."
# Scan all files directory and copy the content to temporary directory
for path_item in "${plugin_dirs[@]}"
do
# Copy the content from files directory to target guest images
if [[ -d "$path_item/files" ]]; then
info "Copy $path_item/files/ => $TARGET_FILES_DIR"
cp $path_item/files/* $TARGET_FILES_DIR/ -fr
fi
# Copy all guest_run.sh from pre-stage dirs to target guest images
if [[ -f $path_item/pre-stage/guest_run.sh ]]; then
info "Copy $path_item/pre-stage/guest_run.sh ==> $TARGET_FILES_DIR/opt/guest-scripts/$(basename $path_item/pre-stage/)_guest_run.sh"
chmod +x $path_item/pre-stage/guest_run.sh
mkdir -p $TARGET_FILES_DIR/opt/guest-scripts/
cp "$path_item"/pre-stage/guest_run.sh "$TARGET_FILES_DIR"/opt/guest-scripts/"$(basename "$path_item"/pre-stage/)"_guest_run.sh
fi
done
info "List all files to be copied to target image at $TARGET_FILES_DIR/..."
ls $TARGET_FILES_DIR
info "Copy all files to target guest image ..."
pushd $TARGET_FILES_DIR/
tar cpzf /tmp/rootfs_overide.tar.gz .
popd
virt-customize -a ${OUTPUT_IMG} \
--copy-in /tmp/rootfs_overide.tar.gz:/root/ \
--run-command 'mkdir -p /root/rootfs/ && tar zxvf /root/rootfs_overide.tar.gz -C /root/rootfs' \
--run-command 'if [ $(find /root/rootfs/ -type f | wc -l) -ne 0 ]; then cp -r /root/rootfs/* /; fi' \
--run-command 'rm -rf /root/rootfs*' || error "Fail to customize the image..."
rm /tmp/rootfs_overide.tar.gz
}
#
# Run the script from each subdirectories at plugins
#
run_stage() {
sub_dir=$1
file_name=$2
for path_item in "${plugin_dirs[@]}"
do
if [[ -f "$path_item/$sub_dir/$file_name.sh" ]]; then
info "Execute the $file_name.sh at $path_item/$sub_dir/"
chmod +x $path_item/$sub_dir/$file_name.sh
$path_item/$sub_dir/$file_name.sh
fi
done
}
do_pre_stage() {
stage_type="pre-stage"
stage_opreation="host_run"
info "Run pre-stage..."
update_plugin_dirs
prepare_target_files
run_stage $stage_type $stage_opreation
}
do_post_stage() {
stage_type="post-stage"
info "Run post-stage..."
update_plugin_dirs
# post-stage host-run
stage_opreation="host_run"
run_stage $stage_type $stage_opreation
# post-stage clean-up
stage_opreation="clean_up"
run_stage $stage_type $stage_opreation
}
_generate_cloud_init_meta_data() {
info "Generate cloud init meta-data"
pushd ${TOP_DIR}/cloud-init
awk -v inst_name="tdx-inst-$(date '+%Y-%m-%d-%H-%M-%S')" \
'{if ($1 ~ /instance-id/) $2=inst_name; print}' meta-data.template \
> meta-data
popd
ok "Complete cloud init meta-data"
}
_cloud_init_user_data_fromat_check() {
input_file=$1
order_check=$2
if [[ $order_check == "true" ]] && ! [[ $(basename $input_file) =~ ^[0-9]{2} ]]; then
error "User data format file name should start with two digitals: $input_file\n\
e.g. 03-example or 55-another"s
fi
file_type=$(head -1 $input_file)
case "$file_type" in
"#cloud-config")
merge_opt=$(grep "merge_how" "$input_file" || true)
example_path="tools/cvm-image-rewriter/cloud-init/user-data.basic"
if [[ "$merge_opt" == "" ]] || [[ "$merge_opt" == "#"* ]]; then
error "Cloud config need entry 'merge_how',\n\
add one or uncomment it in $input_file,\n\
refer default in $example_path,\n\
or https://cloudinit.readthedocs.io/en/latest/reference/merging.html"
fi
info "user data: cloud-config ---> " $input_file
;;
\#!*)
info "user data: x-shellscript ---> " $input_file
;;
*)
info "Supported user data: cloud-config & x-shellscript"
error "Unsupported user data: $input_file, starting with $file_type"
;;
esac
}
_generate_cloud_init_user_data() {
CLD_INIT_CONFIG_SUFFIX=":cloud-config"
CLD_INIT_SCRIPT_SUFFIX=":x-shellscript"
USER_DATA_BASIC="./cloud-init/user-data.basic"
USER_DATA_TARGET="./cloud-init/user-data"
info "Generate cloud init user-data"
# basic user data
info "Start user data format check"
_cloud_init_user_data_fromat_check $USER_DATA_BASIC false
ARGS=" -a "$(realpath $USER_DATA_BASIC)$CLD_INIT_CONFIG_SUFFIX
# find all cloud init user data in pre-stage
for path_item in "${plugin_dirs[@]}"
do
sub_dirs=("$path_item"/*/)
for dir in "${sub_dirs[@]}"
do
if [[ ${dir} == *"cloud-init"* ]]; then
CLD_CONFIG_DIR=${dir}cloud-config
CLD_SCRIPT_DIR=${dir}x-shellscript
# find cloud init cloud-config
for f in "$CLD_CONFIG_DIR"/*
do
if [[ -f "$f" ]]; then
_cloud_init_user_data_fromat_check $f true
ARGS+=" -a $f$CLD_INIT_CONFIG_SUFFIX"
fi
done
# find cloud init x-shellscript
for f in "$CLD_SCRIPT_DIR"/*
do
if [[ -f "$f" ]]; then
_cloud_init_user_data_fromat_check $f true
ARGS+=" -a $f$CLD_INIT_SCRIPT_SUFFIX"
fi
done
break
fi
done
done
ok "Complete user data format check"
cloud-init devel make-mime $ARGS > $USER_DATA_TARGET
ok "Complete cloud init user-data"
}
do_cloud_init() {
_generate_cloud_init_meta_data
_generate_cloud_init_user_data
info "Run cloud-init..."
pushd ${TOP_DIR}/cloud-init
info "Prepare cloud-init ISO image..."
if [[ -f /tmp/ciiso.iso ]]; then
rm /tmp/ciiso.iso -fr || sudo rm /tmp/ciiso.iso -fr
fi
#cloud-init devel make-mime -a ./cloud-config.yaml:cloud-config > ./user-data
genisoimage -output /tmp/ciiso.iso -volid cidata -joliet -rock user-data meta-data
ok "Generate the cloud-init ISO image..."
popd
CONNECT_URI="qemu:///system"
if [[ -n ${CONNECTION_SOCK} ]]; then
CONNECT_URI="qemu+unix:///system?socket=${CONNECTION_SOCK}"
fi
# Clean up before running virt-install
virsh destroy tdx-config-cloud-init &> /dev/null || true
virsh undefine tdx-config-cloud-init &> /dev/null || true
virt-install --memory 4096 --vcpus 4 --name tdx-config-cloud-init \
--disk ${OUTPUT_IMG} \
--connect ${CONNECT_URI} \
--disk /tmp/ciiso.iso,device=cdrom \
--os-type Linux \
--os-variant ubuntu24.04 \
--graphics none \
--import \
--wait=$TIMEOUT \
${CONSOLE_OPT}
# TODO: check return status
ok "Complete cloud-init..."
sleep 1
virsh destroy tdx-config-cloud-init &> /dev/null || true
virsh undefine tdx-config-cloud-init &> /dev/null || true
}
cleanup() {
exit_code=$?
info "cleanup"
if [[ -d $TARGET_FILES_DIR ]]; then
info "Delete temporary directory $TARGET_FILES_DIR..."
rm -fr $TARGET_FILES_DIR
fi
virsh destroy tdx-config-cloud-init &> /dev/null || true
virsh undefine tdx-config-cloud-init &> /dev/null || true
exit $exit_code
}
#
# when run virt-customize, it need read access on the /boot/vmlinuz-${uname -r}
# of the host, so please make sure read permission or use root user to run
# this script.
#
check_kernel_image_access() {
curr_kernel_image="/boot/vmlinuz-$(uname -r)"
test -r "$curr_kernel_image" || {
error "Need read permission on $curr_kernel_image, please run following"\
"command: \n\t sudo chmod +r $curr_kernel_image \n" \
"Or run this script under root user"
}
}
process_args() {
while getopts ":i:t:s:o:nh" option; do
case "$option" in
i) INPUT_IMG=$OPTARG ;;
t) TIMEOUT=$OPTARG ;;
s) CONNECTION_SOCK=$OPTARG;;
o) OUTPUT_IMG=$OPTARG;;
n) CONSOLE_OPT="--noautoconsole";;
h)
usage
exit 0
;;
*)
echo "Invalid option '-$OPTARG'"
usage
exit 1
;;
esac
done
if [[ -z "$INPUT_IMG" ]]; then
error "Please specify the input guest image file via -i"
else
INPUT_IMG=$(readlink -f "$INPUT_IMG")
if [[ ! -f "${INPUT_IMG}" ]]; then
error "File not exist ${INPUT_IMG}"
fi
fi
ok "================================="
ok "Input image: ${INPUT_IMG}"
ok "Output image: ${OUTPUT_IMG}"
ok "Connection Sock: ${CONNECTION_SOCK}"
ok "Console Opt: ${CONSOLE_OPT}"
ok "================================="
# Create output image
cp "${INPUT_IMG}" "${OUTPUT_IMG}"
if [[ -n "${CONNECTION_SOCK}" ]]; then
[[ ! -f "${CONNECTION_SOCK}" ]] || {
error "File ${CONNECTION_SOCK} does not exist." ;
}
test -r "${CONNECTION_SOCK}" || {
error "File ${CONNECTION_SOCK} could not be accessed, please make"\
" sure current user is belong to libvirtd group."
}
fi
}
check_tools qemu-img virt-customize virt-install genisoimage cloud-init git awk
check_kernel_image_access
trap cleanup EXIT
process_args "$@"
export GUEST_IMG=${TOP_DIR}/${OUTPUT_IMG}
do_pre_stage
do_cloud_init
do_post_stage
ok "Complete."