| # vim: tabstop=4 shiftwidth=4 softtabstop=4 |
| |
| # Copyright (c) 2012 Hewlett-Packard Development Company, L.P. |
| # All Rights Reserved. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| # not use this file except in compliance with the License. You may obtain |
| # a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| # License for the specific language governing permissions and limitations |
| # under the License. |
| |
| |
| # This file provides devstack with the environment and utilities to |
| # control nova-compute's baremetal driver. |
| # It sets reasonable defaults to run within a single host, |
| # using virtual machines in place of physical hardware. |
| # However, by changing just a few options, devstack+baremetal can in fact |
| # control physical hardware resources on the same network, if you know |
| # the MAC address(es) and IPMI credentials. |
| # |
| # At a minimum, to enable the baremetal driver, you must set these in loclarc: |
| # VIRT_DRIVER=baremetal |
| # ENABLED_SERVICES="$ENABLED_SERVICES,baremetal" |
| # |
| # |
| # We utilize diskimage-builder to create a ramdisk, and then |
| # baremetal driver uses that to push a disk image onto the node(s). |
| # |
| # Below we define various defaults which control the behavior of the |
| # baremetal compute service, and inform it of the hardware it will contorl. |
| # |
| # Below that, various functions are defined, which are called by devstack |
| # in the following order: |
| # |
| # before nova-cpu starts: |
| # - prepare_baremetal_toolchain |
| # - configure_baremetal_nova_dirs |
| # |
| # after nova and glance have started: |
| # - build_and_upload_baremetal_deploy_k_and_r $token |
| # - create_baremetal_flavor $BM_DEPLOY_KERNEL_ID $BM_DEPLOY_RAMDISK_ID |
| # - upload_baremetal_image $url $token |
| # - add_baremetal_node <first_mac> <second_mac> |
| |
| |
| # Save trace setting |
| XTRACE=$(set +o | grep xtrace) |
| set +o xtrace |
| |
| # Sub-driver settings |
| # ------------------- |
| |
| # sub-driver to use for kernel deployment |
| # - nova.virt.baremetal.pxe.PXE |
| # - nova.virt.baremetal.tilera.TILERA |
| BM_DRIVER=${BM_DRIVER:-nova.virt.baremetal.pxe.PXE} |
| |
| # sub-driver to use for remote power management |
| # - nova.virt.baremetal.fake.FakePowerManager, for manual power control |
| # - nova.virt.baremetal.ipmi.Ipmi, for remote IPMI |
| # - nova.virt.baremetal.tilera_pdu.Pdu, for TilePro hardware |
| BM_POWER_MANAGER=${BM_POWER_MANAGER:-nova.virt.baremetal.fake.FakePowerManager} |
| |
| |
| # These should be customized to your environment and hardware |
| # ----------------------------------------------------------- |
| |
| # whether to create a fake environment, eg. for devstack-gate |
| BM_USE_FAKE_ENV=`trueorfalse False $BM_USE_FAKE_ENV` |
| |
| # Extra options to pass to bm_poseur |
| # change the bridge name or IP: --bridge br99 --bridge-ip 192.0.2.1 |
| # change the virtualization type: --engine qemu |
| BM_POSEUR_EXTRA_OPTS=${BM_POSEUR_EXTRA_OPTS:-} |
| |
| # BM_DNSMASQ_IFACE should match FLAT_NETWORK_BRIDGE |
| if [ "$BM_USE_FAKE_ENV" ]; then |
| BM_DNSMASQ_IFACE=${BM_DNSMASQ_IFACE:-br99} |
| BM_DNSMASQ_RANGE=${BM_DNSMASQ_RANGE:-192.0.2.32,192.0.2.48} |
| else |
| BM_DNSMASQ_IFACE=${BM_DNSMASQ_IFACE:-eth0} |
| # if testing on a physical network, |
| # BM_DNSMASQ_RANGE must be changed to suit your network |
| BM_DNSMASQ_RANGE=${BM_DNSMASQ_RANGE:-} |
| fi |
| |
| # BM_DNSMASQ_DNS provide dns server to bootstrap clients |
| BM_DNSMASQ_DNS=${BM_DNSMASQ_DNS:-} |
| |
| # BM_FIRST_MAC *must* be set to the MAC address of the node you will boot. |
| # This is passed to dnsmasq along with the kernel/ramdisk to |
| # deploy via PXE. |
| BM_FIRST_MAC=${BM_FIRST_MAC:-} |
| |
| # BM_SECOND_MAC is only important if the host has >1 NIC. |
| BM_SECOND_MAC=${BM_SECOND_MAC:-} |
| |
| # Hostname for the baremetal nova-compute node, if not run on this host |
| BM_HOSTNAME=${BM_HOSTNAME:-$(hostname -f)} |
| |
| # BM_PM_* options are only necessary if BM_POWER_MANAGER=...IPMI |
| BM_PM_ADDR=${BM_PM_ADDR:-0.0.0.0} |
| BM_PM_USER=${BM_PM_USER:-user} |
| BM_PM_PASS=${BM_PM_PASS:-pass} |
| |
| # BM_FLAVOR_* options are arbitrary and not necessarily related to physical |
| # hardware capacity. These can be changed if you are testing |
| # BaremetalHostManager with multiple nodes and different flavors. |
| BM_CPU_ARCH=${BM_CPU_ARCH:-x86_64} |
| BM_FLAVOR_CPU=${BM_FLAVOR_CPU:-1} |
| BM_FLAVOR_RAM=${BM_FLAVOR_RAM:-1024} |
| BM_FLAVOR_ROOT_DISK=${BM_FLAVOR_ROOT_DISK:-10} |
| BM_FLAVOR_EPHEMERAL_DISK=${BM_FLAVOR_EPHEMERAL_DISK:-0} |
| BM_FLAVOR_SWAP=${BM_FLAVOR_SWAP:-1} |
| BM_FLAVOR_NAME=${BM_FLAVOR_NAME:-bm.small} |
| BM_FLAVOR_ID=${BM_FLAVOR_ID:-11} |
| BM_FLAVOR_ARCH=${BM_FLAVOR_ARCH:-$BM_CPU_ARCH} |
| |
| |
| # Below this, we set some path and filenames. |
| # Defaults are probably sufficient. |
| BM_IMAGE_BUILD_DIR=${BM_IMAGE_BUILD_DIR:-$DEST/diskimage-builder} |
| BM_POSEUR_DIR=${BM_POSEUR_DIR:-$DEST/bm_poseur} |
| |
| BM_HOST_CURRENT_KERNEL=$(uname -r) |
| BM_DEPLOY_RAMDISK=${BM_DEPLOY_RAMDISK:-bm-deploy-$BM_HOST_CURRENT_KERNEL-initrd} |
| BM_DEPLOY_KERNEL=${BM_DEPLOY_KERNEL:-bm-deploy-$BM_HOST_CURRENT_KERNEL-vmlinuz} |
| |
| # If you need to add any extra flavors to the deploy ramdisk image |
| # eg, specific network drivers, specify them here |
| BM_DEPLOY_FLAVOR=${BM_DEPLOY_FLAVOR:-} |
| |
| # set URL and version for google shell-in-a-box |
| BM_SHELL_IN_A_BOX=${BM_SHELL_IN_A_BOX:-http://shellinabox.googlecode.com/files/shellinabox-2.14.tar.gz} |
| |
| |
| # Functions |
| # --------- |
| |
| # Check if baremetal is properly enabled |
| # Returns false if VIRT_DRIVER is not baremetal, or if ENABLED_SERVICES |
| # does not contain "baremetal" |
| function is_baremetal() { |
| if [[ "$ENABLED_SERVICES" =~ 'baremetal' && "$VIRT_DRIVER" = 'baremetal' ]]; then |
| return 0 |
| fi |
| return 1 |
| } |
| |
| # Install diskimage-builder and shell-in-a-box |
| # so that we can build the deployment kernel & ramdisk |
| function prepare_baremetal_toolchain() { |
| git_clone $BM_IMAGE_BUILD_REPO $BM_IMAGE_BUILD_DIR $BM_IMAGE_BUILD_BRANCH |
| git_clone $BM_POSEUR_REPO $BM_POSEUR_DIR $BM_POSEUR_BRANCH |
| |
| local shellinabox_basename=$(basename $BM_SHELL_IN_A_BOX) |
| if [[ ! -e $DEST/$shellinabox_basename ]]; then |
| cd $DEST |
| wget $BM_SHELL_IN_A_BOX |
| fi |
| if [[ ! -d $DEST/${shellinabox_basename%%.tar.gz} ]]; then |
| cd $DEST |
| tar xzf $shellinabox_basename |
| fi |
| if [[ ! $(which shellinaboxd) ]]; then |
| cd $DEST/${shellinabox_basename%%.tar.gz} |
| ./configure |
| make |
| sudo make install |
| fi |
| } |
| |
| # set up virtualized environment for devstack-gate testing |
| function create_fake_baremetal_env() { |
| local bm_poseur="$BM_POSEUR_DIR/bm_poseur" |
| # TODO(deva): add support for >1 VM |
| sudo $bm_poseur $BM_POSEUR_EXTRA_OPTS create-bridge |
| sudo $bm_poseur $BM_POSEUR_EXTRA_OPTS create-vm |
| BM_FIRST_MAC=$(sudo $bm_poseur get-macs) |
| |
| # NOTE: there is currently a limitation in baremetal driver |
| # that requires second MAC even if it is not used. |
| # Passing a fake value allows this to work. |
| # TODO(deva): remove this after driver issue is fixed. |
| BM_SECOND_MAC='12:34:56:78:90:12' |
| } |
| |
| function cleanup_fake_baremetal_env() { |
| local bm_poseur="$BM_POSEUR_DIR/bm_poseur" |
| sudo $bm_poseur $BM_POSEUR_EXTRA_OPTS destroy-vm |
| sudo $bm_poseur $BM_POSEUR_EXTRA_OPTS destroy-bridge |
| } |
| |
| # prepare various directories needed by baremetal hypervisor |
| function configure_baremetal_nova_dirs() { |
| # ensure /tftpboot is prepared |
| sudo mkdir -p /tftpboot |
| sudo mkdir -p /tftpboot/pxelinux.cfg |
| sudo cp /usr/lib/syslinux/pxelinux.0 /tftpboot/ |
| sudo chown -R $STACK_USER:libvirtd /tftpboot |
| |
| # ensure $NOVA_STATE_PATH/baremetal is prepared |
| sudo mkdir -p $NOVA_STATE_PATH/baremetal |
| sudo mkdir -p $NOVA_STATE_PATH/baremetal/console |
| sudo mkdir -p $NOVA_STATE_PATH/baremetal/dnsmasq |
| sudo touch $NOVA_STATE_PATH/baremetal/dnsmasq/dnsmasq-dhcp.host |
| sudo chown -R $STACK_USER $NOVA_STATE_PATH/baremetal |
| |
| # ensure dnsmasq is installed but not running |
| # because baremetal driver will reconfigure and restart this as needed |
| is_package_installed dnsmasq || install_package dnsmasq |
| stop_service dnsmasq |
| } |
| |
| # build deploy kernel+ramdisk, then upload them to glance |
| # this function sets BM_DEPLOY_KERNEL_ID and BM_DEPLOY_RAMDISK_ID |
| function upload_baremetal_deploy() { |
| token=$1 |
| |
| if [ ! -e $TOP_DIR/files/$BM_DEPLOY_KERNEL -a -e /boot/vmlinuz-$BM_HOST_CURRENT_KERNEL ]; then |
| sudo cp /boot/vmlinuz-$BM_HOST_CURRENT_KERNEL $TOP_DIR/files/$BM_DEPLOY_KERNEL |
| sudo chmod a+r $TOP_DIR/files/$BM_DEPLOY_KERNEL |
| fi |
| if [ ! -e $TOP_DIR/files/$BM_DEPLOY_RAMDISK ]; then |
| $BM_IMAGE_BUILD_DIR/bin/ramdisk-image-create $BM_DEPLOY_FLAVOR deploy \ |
| -o $TOP_DIR/files/$BM_DEPLOY_RAMDISK -k $BM_HOST_CURRENT_KERNEL |
| fi |
| |
| # load them into glance |
| BM_DEPLOY_KERNEL_ID=$(glance \ |
| --os-auth-token $token \ |
| --os-image-url http://$GLANCE_HOSTPORT \ |
| image-create \ |
| --name $BM_DEPLOY_KERNEL \ |
| --public --disk-format=aki \ |
| < $TOP_DIR/files/$BM_DEPLOY_KERNEL | grep ' id ' | get_field 2) |
| BM_DEPLOY_RAMDISK_ID=$(glance \ |
| --os-auth-token $token \ |
| --os-image-url http://$GLANCE_HOSTPORT \ |
| image-create \ |
| --name $BM_DEPLOY_RAMDISK \ |
| --public --disk-format=ari \ |
| < $TOP_DIR/files/$BM_DEPLOY_RAMDISK | grep ' id ' | get_field 2) |
| } |
| |
| # create a basic baremetal flavor, associated with deploy kernel & ramdisk |
| # |
| # Usage: create_baremetal_flavor <aki_uuid> <ari_uuid> |
| function create_baremetal_flavor() { |
| aki=$1 |
| ari=$2 |
| nova flavor-create $BM_FLAVOR_NAME $BM_FLAVOR_ID \ |
| $BM_FLAVOR_RAM $BM_FLAVOR_ROOT_DISK $BM_FLAVOR_CPU |
| nova flavor-key $BM_FLAVOR_NAME set \ |
| cpu_arch=$BM_FLAVOR_ARCH \ |
| deploy_kernel_id=$aki \ |
| deploy_ramdisk_id=$ari |
| } |
| |
| # pull run-time kernel/ramdisk out of disk image and load into glance |
| # note that $file is currently expected to be in qcow2 format |
| # Sets KERNEL_ID and RAMDISK_ID |
| # |
| # Usage: extract_and_upload_k_and_r_from_image $token $file |
| function extract_and_upload_k_and_r_from_image() { |
| token=$1 |
| file=$2 |
| image_name=$(basename "$file" ".qcow2") |
| |
| # this call returns the file names as "$kernel,$ramdisk" |
| out=$($BM_IMAGE_BUILD_DIR/bin/disk-image-get-kernel \ |
| -x -d $TOP_DIR/files -o bm-deploy -i $file) |
| if [ $? -ne 0 ]; then |
| die "Failed to get kernel and ramdisk from $file" |
| fi |
| XTRACE=$(set +o | grep xtrace) |
| set +o xtrace |
| out=$(echo "$out" | tail -1) |
| $XTRACE |
| OUT_KERNEL=${out%%,*} |
| OUT_RAMDISK=${out##*,} |
| |
| # load them into glance |
| KERNEL_ID=$(glance \ |
| --os-auth-token $token \ |
| --os-image-url http://$GLANCE_HOSTPORT \ |
| image-create \ |
| --name $image_name-kernel \ |
| --public --disk-format=aki \ |
| < $TOP_DIR/files/$OUT_KERNEL | grep ' id ' | get_field 2) |
| RAMDISK_ID=$(glance \ |
| --os-auth-token $token \ |
| --os-image-url http://$GLANCE_HOSTPORT \ |
| image-create \ |
| --name $image_name-initrd \ |
| --public --disk-format=ari \ |
| < $TOP_DIR/files/$OUT_RAMDISK | grep ' id ' | get_field 2) |
| } |
| |
| |
| # Re-implementation of devstack's "upload_image" function |
| # |
| # Takes the same parameters, but has some peculiarities which made it |
| # easier to create a separate method, rather than complicate the logic |
| # of the existing function. |
| function upload_baremetal_image() { |
| local image_url=$1 |
| local token=$2 |
| |
| # Create a directory for the downloaded image tarballs. |
| mkdir -p $FILES/images |
| |
| # Downloads the image (uec ami+aki style), then extracts it. |
| IMAGE_FNAME=`basename "$image_url"` |
| if [[ ! -f $FILES/$IMAGE_FNAME || \ |
| "$(stat -c "%s" $FILES/$IMAGE_FNAME)" = "0" ]]; then |
| wget -c $image_url -O $FILES/$IMAGE_FNAME |
| if [[ $? -ne 0 ]]; then |
| echo "Not found: $image_url" |
| return |
| fi |
| fi |
| |
| local KERNEL="" |
| local RAMDISK="" |
| local DISK_FORMAT="" |
| local CONTAINER_FORMAT="" |
| case "$IMAGE_FNAME" in |
| *.tar.gz|*.tgz) |
| # Extract ami and aki files |
| [ "${IMAGE_FNAME%.tar.gz}" != "$IMAGE_FNAME" ] && |
| IMAGE_NAME="${IMAGE_FNAME%.tar.gz}" || |
| IMAGE_NAME="${IMAGE_FNAME%.tgz}" |
| xdir="$FILES/images/$IMAGE_NAME" |
| rm -Rf "$xdir"; |
| mkdir "$xdir" |
| tar -zxf $FILES/$IMAGE_FNAME -C "$xdir" |
| KERNEL=$(for f in "$xdir/"*-vmlinuz* "$xdir/"aki-*/image; do |
| [ -f "$f" ] && echo "$f" && break; done; true) |
| RAMDISK=$(for f in "$xdir/"*-initrd* "$xdir/"ari-*/image; do |
| [ -f "$f" ] && echo "$f" && break; done; true) |
| IMAGE=$(for f in "$xdir/"*.img "$xdir/"ami-*/image; do |
| [ -f "$f" ] && echo "$f" && break; done; true) |
| if [[ -z "$IMAGE_NAME" ]]; then |
| IMAGE_NAME=$(basename "$IMAGE" ".img") |
| fi |
| DISK_FORMAT=ami |
| CONTAINER_FORMAT=ami |
| ;; |
| *.qcow2) |
| IMAGE="$FILES/${IMAGE_FNAME}" |
| IMAGE_NAME=$(basename "$IMAGE" ".qcow2") |
| DISK_FORMAT=qcow2 |
| CONTAINER_FORMAT=bare |
| ;; |
| *) echo "Do not know what to do with $IMAGE_FNAME"; false;; |
| esac |
| |
| if [ "$CONTAINER_FORMAT" = "bare" ]; then |
| extract_and_upload_k_and_r_from_image $token $IMAGE |
| elif [ "$CONTAINER_FORMAT" = "ami" ]; then |
| KERNEL_ID=$(glance \ |
| --os-auth-token $token \ |
| --os-image-url http://$GLANCE_HOSTPORT \ |
| image-create \ |
| --name "$IMAGE_NAME-kernel" --public \ |
| --container-format aki \ |
| --disk-format aki < "$KERNEL" | grep ' id ' | get_field 2) |
| RAMDISK_ID=$(glance \ |
| --os-auth-token $token \ |
| --os-image-url http://$GLANCE_HOSTPORT \ |
| image-create \ |
| --name "$IMAGE_NAME-ramdisk" --public \ |
| --container-format ari \ |
| --disk-format ari < "$RAMDISK" | grep ' id ' | get_field 2) |
| else |
| # TODO(deva): add support for other image types |
| return |
| fi |
| |
| glance \ |
| --os-auth-token $token \ |
| --os-image-url http://$GLANCE_HOSTPORT \ |
| image-create \ |
| --name "${IMAGE_NAME%.img}" --public \ |
| --container-format $CONTAINER_FORMAT \ |
| --disk-format $DISK_FORMAT \ |
| ${KERNEL_ID:+--property kernel_id=$KERNEL_ID} \ |
| ${RAMDISK_ID:+--property ramdisk_id=$RAMDISK_ID} < "${IMAGE}" |
| |
| # override DEFAULT_IMAGE_NAME so that tempest can find the image |
| # that we just uploaded in glance |
| DEFAULT_IMAGE_NAME="${IMAGE_NAME%.img}" |
| } |
| |
| function clear_baremetal_of_all_nodes() { |
| list=$(nova baremetal-node-list | awk -F '| ' 'NR>3 {print $2}' ) |
| for node in $list |
| do |
| nova baremetal-node-delete $node |
| done |
| } |
| |
| # inform nova-baremetal about nodes, MACs, etc |
| # Defaults to using BM_FIRST_MAC and BM_SECOND_MAC if parameters not specified |
| # |
| # Usage: add_baremetal_node <first_mac> <second_mac> |
| function add_baremetal_node() { |
| mac_1=${1:-$BM_FIRST_MAC} |
| mac_2=${2:-$BM_SECOND_MAC} |
| |
| id=$(nova baremetal-node-create \ |
| --pm_address="$BM_PM_ADDR" \ |
| --pm_user="$BM_PM_USER" \ |
| --pm_password="$BM_PM_PASS" \ |
| "$BM_HOSTNAME" \ |
| "$BM_FLAVOR_CPU" \ |
| "$BM_FLAVOR_RAM" \ |
| "$BM_FLAVOR_ROOT_DISK" \ |
| "$mac_1" \ |
| | grep ' id ' | get_field 2 ) |
| [ $? -eq 0 ] || [ "$id" ] || die "Error adding baremetal node" |
| id2=$(nova baremetal-add-interface "$id" "$mac_2" ) |
| [ $? -eq 0 ] || [ "$id2" ] || die "Error adding interface to barmetal node $id" |
| } |
| |
| |
| # Restore xtrace |
| $XTRACE |