Statistics
| Branch: | Tag: | Revision:

root / snf-image-host / common.sh.in @ 3707aa73

History | View | Annotate | Download (11.6 kB)

1
# Copyright (C) 2011 GRNET S.A. 
2
# Copyright (C) 2007, 2008, 2009 Google Inc.
3
#
4
# This program is free software; you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation; either version 2 of the License, or
7
# (at your option) any later version.
8
#
9
# This program is distributed in the hope that it will be useful, but
10
# WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12
# General Public License for more details.
13
#
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
17
# 02110-1301, USA.
18

    
19
AWK="awk"
20
KPARTX="kpartx"
21
LOSETUP="losetup"
22
SFDISK="sfdisk"
23
QEMU_IMG="qemu-img"
24
INSTALL_MBR="install-mbr"
25
TIMEOUT="timeout"
26
CURL="curl"
27

    
28
# Temporary use stderr as monitoring file descriptor.
29
# `create' will overwrite this
30
MONITOR_FD="2"
31

    
32
network_backend_support="@network_backend_support@"
33

    
34
MSG_TYPE_ERROR="image-error"
35
MSG_TYPE_INFO="image-info"
36

    
37
CLEANUP=( )
38
ERROR_MSGS=( )
39

    
40
add_cleanup() {
41
    local cmd=""
42
    for arg; do cmd+=$(printf "%q " "$arg"); done
43
    CLEANUP+=("$cmd")
44
}
45

    
46
log_error() {
47
    ERROR_MSGS+=("$@")
48
    echo "$*" >&2
49
}
50

    
51

    
52
log_info() {
53
    echo "$*" >&2
54

    
55
    local report="$(./host-monitor.py info <<< "$*")"
56

    
57
    eval "echo $(printf "%q" "$report") >&${MONITOR_FD}"
58
}
59

    
60

    
61
close_fd() {
62
    local fd="$1"
63
    exec {fd}>&-
64
}
65

    
66
report_error() {
67
    local error_file=$1
68
    local report=""
69
    if [ ${#ERROR_MSGS[@]} -gt 0 ]; then
70
        local msg=""
71
        for err in "${ERROR_MSGS[@]}"; do
72
            msg+="$(echo "$err")"
73
        done
74
        report="$(./host-monitor.py error <<< "$msg")"
75
    else
76
        report=$(tail -10 "$error_file" | ./host-monitor.py stderr)
77
    fi
78

    
79
    eval "echo $(printf "%q" "$report") >&${MONITOR_FD}"
80
}
81

    
82
get_api5_arguments() {
83
    GETOPT_RESULT=$*
84
    # Note the quotes around `$TEMP': they are essential!
85
    eval set -- "$GETOPT_RESULT"
86
    while true; do
87
        case "$1" in
88
            -i|-n) instance=$2; shift 2;;
89

    
90
            -o) old_name=$2; shift 2;;
91

    
92
            -b) blockdev=$2; shift 2;;
93

    
94
            -s) swapdev=$2; shift 2;;
95

    
96
            --) shift; break;;
97

    
98
            *)  log_error "Internal error!" >&2; exit 1;;
99
        esac
100
    done
101
    if [ -z "$instance" -o -z "$blockdev" ]; then
102
        log_error "Missing OS API Argument (-i, -n, or -b)"
103
        exit 1
104
    fi
105
    if [ "$SCRIPT_NAME" != "export" -a -z "$swapdev"  ]; then
106
        log_error "Missing OS API Argument -s (swapdev)"
107
        exit 1
108
    fi
109
    if [ "$SCRIPT_NAME" = "rename" -a -z "$old_name"  ]; then
110
        log_error "Missing OS API Argument -o (old_name)"
111
        exit 1
112
    fi
113
}
114

    
115
get_api10_arguments() {
116
    if [ -z "$INSTANCE_NAME" -o -z "$HYPERVISOR" -o -z "$DISK_COUNT" ]; then
117
        log_error "Missing OS API Variable:"
118
        log_error "(INSTANCE_NAME HYPERVISOR or DISK_COUNT)"
119
        exit 1
120
    fi
121
    instance=$INSTANCE_NAME
122
    if [ $DISK_COUNT -lt 1 -o -z "$DISK_0_PATH" ]; then
123
        log_error "At least one disk is needed"
124
        exit 1
125
    fi
126
    if [ "$SCRIPT_NAME" = "export" ]; then
127
        if [ -z "$EXPORT_DEVICE" ]; then
128
            log_error "Missing OS API Variable EXPORT_DEVICE"
129
        fi
130
        blockdev=$EXPORT_DEVICE
131
    elif [ "$SCRIPT_NAME" = "import" ]; then
132
        if [ -z "$IMPORT_DEVICE" ]; then
133
        log_error "Missing OS API Variable IMPORT_DEVICE"
134
        fi
135
        blockdev=$IMPORT_DEVICE
136
    else
137
        blockdev=$DISK_0_PATH
138
    fi
139
    if [ "$SCRIPT_NAME" = "rename" -a -z "$OLD_INSTANCE_NAME" ]; then
140
        log_error "Missing OS API Variable OLD_INSTANCE_NAME"
141
    fi
142
    old_name=$OLD_INSTANCE_NAME
143
}
144

    
145
get_api20_arguments() {
146
    get_api10_arguments
147

    
148
    local required_osparams="IMG_ID IMG_FORMAT IMG_PASSWD"
149
    local osparams="$required_osparams IMG_PROPERTIES IMG_PERSONALITY CONFIG_URL"
150

    
151
    # Store OSP_VAR in VAR
152
    for param in $osparams; do
153
        eval $param=\"\$OSP_$param\"
154
    done
155

    
156
    if [ -n "$CONFIG_URL" ]; then
157
	source <($CURL -f "$CONFIG_URL" | ./decode-config.py $osparams)
158
    fi
159

    
160
    for var in $required_osparams; do
161
        if [ -z "${!var}" ]; then
162
             log_error "Missing OS API Parameter: ${var}"
163
             exit 1
164
        fi
165
    done
166
}
167

    
168
map_disk0() {
169
    blockdev="$1"
170
    filesystem_dev_base=$($KPARTX -l -p- $blockdev | \
171
                            grep -m 1 -- "-1.*$blockdev" | \
172
                            $AWK '{print $1}')
173
    if [ -z "$filesystem_dev_base" ]; then
174
        log_error "Cannot interpret kpartx output and get partition mapping"
175
        exit 1
176
    fi
177
    $KPARTX -a -p- "$blockdev" > /dev/null
178
    filesystem_dev="/dev/mapper/${filesystem_dev_base/%-1/}"
179
    if [ ! -b "/dev/mapper/$filesystem_dev_base" ]; then
180
        log_error "Can't find kpartx mapped partition:" \
181
                                            "/dev/mapper/$filesystem_dev_base"
182
        exit 1
183
    fi
184
    echo "$filesystem_dev"
185
}
186

    
187
unmap_disk0() {
188
    $KPARTX -d -p- "$1"
189
}
190

    
191
format_disk0() {
192
    local device="$1"
193
    local image_type="$2"
194

    
195
    declare -A part_id=( ['extdump']="83" ["ntfsdump"]="7" )
196

    
197
    # The -f is needed, because we use an optimal alignment and sfdisk complains
198
    # about partitions not ending on clylinder boundary.
199
    local sfdisk_cmd="$SFDISK -uS -H 255 -S 63 -f --quiet --Linux --DOS $device"
200

    
201
    $sfdisk_cmd > /dev/null <<EOF
202
2048,,${part_id["$image_type"]},*
203
EOF
204
}
205

    
206
create_floppy() {
207
    local img=$1
208

    
209
    local target=$(mktemp -d)
210
    add_cleanup rmdir "$target"
211

    
212
    dd bs=512 count=2880 if=/dev/zero of="$img"
213
    mkfs.ext2 -F "$img" > /dev/null
214
    mount "$img" "$target" -o loop
215
    set | egrep ^snf_export_\\w+= | sed -e 's/^snf_export_/export SNF_IMAGE_/' \
216
        > "$target/rules"
217
    if [ -n "$UNATTEND" ]; then
218
        if [ -f "$UNATTEND" ]; then
219
            cat "$UNATTEND" > "$target/unattend.xml"
220
        else
221
            log_error "Unattend file: \`"$UNATTEND"' does not exist"
222
        fi
223
    fi
224
    umount "$target"
225
}
226

    
227
get_backend_type() {
228
    local id=$1
229

    
230
    if [[ "$id" =~ ^pithos: ]]; then
231
        echo "pithos"
232
    elif [[ "$id" =~ ^(http|ftp)s?: ]]; then
233
        if [ "$network_backend_support" = "yes" ]; then
234
            echo "network";
235
        else
236
            log_error "Retrieving images from the network is not supported."
237
            exit 1
238
        fi
239
    elif [ "$id" = "null" ]; then
240
        echo "null"
241
    else
242
        echo "local"
243
    fi
244
}
245

    
246
# this one is only to be called by create
247
ganeti_os_main() {
248
    if [ -z "$OS_API_VERSION" -o "$OS_API_VERSION" = "5" ]; then
249
        OS_API_VERSION=5
250
        GETOPT_RESULT=`getopt -o o:n:i:b:s: -n '$0' -- "$@"`
251
        if [ $? != 0 ] ; then log_error "Terminating..."; exit 1 ; fi
252
        get_api5_arguments $GETOPT_RESULT
253
    elif [ "$OS_API_VERSION" = "10" -o "$OS_API_VERSION" = "15" ]; then
254
        get_api10_arguments
255
    elif [ "$OS_API_VERSION" = "20" ]; then
256
        get_api20_arguments
257
        IMAGE_NAME="$IMG_ID"
258
        IMAGE_TYPE="$IMG_FORMAT"
259
        BACKEND_TYPE=$(get_backend_type $IMG_ID)
260
    else
261
        log_error "Unknown OS API VERSION $OS_API_VERSION"
262
        exit 1
263
    fi
264
    
265
    if [ -n "$OS_VARIANT" ]; then
266
        if [ ! -d "$VARIANTS_DIR" ]; then
267
            log_error "OS Variants directory $VARIANTS_DIR doesn't exist"
268
            exit 1
269
        fi
270
        VARIANT_CONFIG="$VARIANTS_DIR/$OS_VARIANT.conf"
271
        if [ -f "$VARIANT_CONFIG" ]; then
272
            . "$VARIANT_CONFIG"
273
        else
274
            if grep -qxF "$OS_VARIANT" variants.list; then
275
                log_error "ERROR: instance-image configuration error"
276
                log_error "  Published variant $OS_VARIANT is missing its" \
277
                    "config file"
278
                log_error "  Please create $VARIANT_CONFIG or unpublish the" \
279
                    "variant"
280
                log_error "  (by removing $OS_VARIANT from variants.list)"
281
            else
282
                log_error "Unofficial variant $OS_VARIANT is unsupported"
283
                log_error "Most probably this is a user error, forcing a" \
284
                    "wrong name"
285
                log_error "To support this variant please create file" \
286
                    "$VARIANT_CONFIG"
287
            fi
288
            exit 1
289
        fi
290
    fi
291

    
292
}
293

    
294
do_debootstrap() {
295
    local target="$1"
296

    
297
    echo "Debootstraping to create a new root filesystem:"
298

    
299
    # Create a policy-rc.d file to deny init script execution
300
    mkdir -p "$target/usr/sbin"
301
    cat > "$target/usr/sbin/policy-rc.d" <<EOF
302
#!/bin/sh
303
exit 101
304
EOF
305
    chmod +x "$target/usr/sbin/policy-rc.d"
306

    
307
    debootstrap --arch $(dpkg --print-architecture) \
308
        --include "$HELPER_EXTRA_PKGS" --variant=minbase stable "$target" \
309
        "$HELPER_MIRROR" 2>&1 | sed -e 's/^/DEBOOTSTRAP: /g'
310

    
311
    # Save the package list
312
    chroot "$target" dpkg-query -W -f "\${Package}\n" > "$HELPER_CACHE_PKGS"
313

    
314
    rm "$target/usr/sbin/policy-rc.d"
315

    
316
    # remove the downloaded debs, as they are no longer needed
317
    find "$target/var/cache/apt/archives" -type f -name '*.deb' -print0 | \
318
        xargs -r0 rm -f
319

    
320
    local tmp_cache=$(mktemp "$CACHE_FILE.XXXXXX")
321
    tar cf "$tmp_cache" --one-file-system -C "$target" . || \
322
        { rm "$tmp_cache"; false; }
323
    # Overwrite the default cache file. Not the user specified if present.
324
    mv -f "$tmp_cache" "$HELPER_CACHE_FILE"
325
}
326

    
327
report_and_cleanup(){
328

    
329
    local err_file="$1"
330

    
331
    report_error "$err_file"
332
    cleanup
333
}
334

    
335
cleanup() {
336
    # if something fails here, it souldn't call cleanup again...
337
    trap - EXIT
338

    
339
    if [ ${#CLEANUP[*]} -gt 0 ]; then
340
        LAST_ELEMENT=$((${#CLEANUP[*]}-1))
341
        REVERSE_INDEXES=$(seq ${LAST_ELEMENT} -1 0)
342
        for i in $REVERSE_INDEXES; do
343
            # If something fails here, it's better to retry it for a few times
344
            # before we give up with an error. This is needed for kpartx when
345
            # dealing with ntfs partitions mounted through fuse. umount is not
346
            # synchronous and may return while the partition is still busy. A
347
            # premature attempt to delete partition mappings through kpartx on
348
            # a device that hosts previously mounted ntfs partition may fail
349
            # with errors like this one:
350
            # `device-mapper: remove ioctl failed: Device or resource busy'
351
            # A sensible workaround for this is to wait for a while and then
352
            # retry it.
353
            local cmd=${CLEANUP[$i]}
354
            $cmd || for interval in 0.25 0.5 1 2 4; do
355
            echo "Command $cmd failed!"
356
            echo "I'll wait for $interval secs and will retry..."
357
            sleep $interval
358
            $cmd && break
359
        done
360
        if [ "$?" != "0" ]; then
361
            echo "Giving Up..."
362
            exit 1;
363
        fi
364
    done
365
  fi
366
}
367

    
368
trap cleanup EXIT
369

    
370
DEFAULT_FILE="@sysconfdir@/default/snf-image"
371
if [ -f "$DEFAULT_FILE" ]; then
372
    . "$DEFAULT_FILE"
373
fi
374

    
375
: ${VARIANTS_DIR:="@sysconfdir@/ganeti/snf-image/variants"}
376
: ${IMAGE_DIR:="@localstatedir@/lib/snf-image"}
377
: ${HELPER_DIR:="@HELPER_DIR@"}
378
: ${HELPER_IMG:="@HELPER_IMG@"}
379
: ${HELPER_KERNEL:="@HELPER_KERNEL@"}
380
: ${HELPER_INITRD:="@HELPER_INITRD@"}
381
: ${HELPER_PKG:="@HELPER_DIR@/snf-image-helper.deb"}
382
: ${HELPER_SOFT_TIMEOUT:=20}
383
: ${HELPER_HARD_TIMEOUT:=5}
384
: ${HELPER_USER:="nobody"}
385
: ${HELPER_CACHE_FILE:="@HELPER_DIR@/cache.tar"}
386
: ${HELPER_CACHE_PKGS:="@HELPER_DIR@/packages"}
387
: ${HELPER_EXTRA_PKGS:="linux-image-amd64,e2fsprogs,ntfs-3g,ntfsprogs,xmlstarlet,python,parted,reglookup,chntpw,util-linux"}
388
: ${HELPER_MIRROR:=""}
389
: ${PITHOS_DB:="sqlite:////@localstatedir@/lib/pithos/backend.db"}
390
: ${PITHOS_DATA:="@localstatedir@/lib/pithos/data/"}
391
: ${PROGRESS_MONITOR:="@PROGRESS_MONITOR@"}
392
: ${UNATTEND:="@UNATTEND@"}
393

    
394
SCRIPT_NAME=$(basename $0)
395

    
396
# vim: set sta sts=4 shiftwidth=4 sw=4 et ai :