#!/usr/bin/env bash #Common clean script to uninstall provisioning server #author: Tim Rozet (trozet@redhat.com) # #Removes Libvirt, KVM, Vagrant, VirtualBox # #Destroys Vagrant VMs running in $vm_dir/ #Shuts down all nodes found in Khaleesi settings #Removes hypervisor kernel modules (VirtualBox & KVM/Libvirt) ##VARS reset=`tput sgr0` blue=`tput setaf 4` red=`tput setaf 1` green=`tput setaf 2` pxe_bridge='pxebr' vm_dir=/var/opt/opnfv private_interface='enp6s0' management_vid=300 management_interface="${private_interface}.${management_vid}" ##END VARS ##FUNCTIONS display_usage() { echo -e "\n\n${blue}This script is used to uninstall and clean the OPNFV Target System${reset}\n\n" echo -e "\nUsage:\n$0 [arguments] \n" echo -e "\n -no_parse : No variable parsing into config. Flag. \n" echo -e "\n -base_config : Full path of ksgen settings file to parse. Required. Will provide BMC info to shutdown hosts. Example: -base_config /opt/myinventory.yml \n" } remove_interface_with_name_pattern() { if [ -z $1 ]; then echo "${red}Cannot remove interface. No interface name pattern specified!${reset}" exit 1 fi local interface_name_pattern=$1 echo "${blue} Looking for interface with name pattern: ${interface_name_pattern}${reset}" interface=$(ip link show | grep -oP ${interface_name_pattern}) if [ ! -z "${interface}" ]; then echo "${blue}Interface ${interface} detected! Removing...${reset}" ip link del ${interface} if ip link show | grep -oP ${interface_name_pattern}; then echo "${red}Could not remove interface ${interface} ${reset}" exit 1 else echo "${blue}Interface ${interface} successfully removed${reset}" fi else echo "${blue}Interface with name pattern ${interface_name_pattern} does not exist, nothing to remove${reset}" fi } ##END FUNCTIONS if [[ ( $1 == "--help") || $1 == "-h" ]]; then display_usage exit 0 fi echo -e "\n\n${blue}This script is used to uninstall and clean the OPNFV Target System${reset}\n\n" echo "Use -h to display help" sleep 2 while [ "`echo $1 | cut -c1`" = "-" ] do echo $1 case "$1" in -base_config) base_config=$2 shift 2 ;; *) display_usage exit 1 ;; esac done if [ ! -z "$base_config" ]; then # Install ipmitool # Major version is pinned to force some consistency for Arno if ! yum list installed | grep -i ipmitool; then if ! yum -y install ipmitool-1*; then echo "${red}Unable to install ipmitool!${reset}" exit 1 fi else echo "${blue}Skipping ipmitool as it is already installed!${reset}" fi ###find all the bmc IPs and number of nodes node_counter=0 output=`grep bmc_ip $base_config | grep -Eo '[0-9]+.[0-9]+.[0-9]+.[0-9]+'` for line in ${output} ; do bmc_ip[$node_counter]=$line ((node_counter++)) done max_nodes=$((node_counter-1)) ###find bmc_users per node node_counter=0 output=`grep bmc_user $base_config | sed 's/\s*bmc_user:\s*//'` for line in ${output} ; do bmc_user[$node_counter]=$line ((node_counter++)) done ###find bmc_pass per node node_counter=0 output=`grep bmc_pass $base_config | sed 's/\s*bmc_pass:\s*//'` for line in ${output} ; do bmc_pass[$node_counter]=$line ((node_counter++)) done for mynode in `seq 0 $max_nodes`; do echo "${blue}Node: ${bmc_ip[$mynode]} ${bmc_user[$mynode]} ${bmc_pass[$mynode]} ${reset}" if ipmitool -I lanplus -P ${bmc_pass[$mynode]} -U ${bmc_user[$mynode]} -H ${bmc_ip[$mynode]} chassis power off; then echo "${blue}Node: $mynode, ${bmc_ip[$mynode]} powered off!${reset}" else echo "${red}Error: Unable to power off $mynode, ${bmc_ip[$mynode]} ${reset}" exit 1 fi done else echo "${blue}Skipping Baremetal node poweroff as base_config was not provided${reset}" fi ###check to see if vbox is installed vboxpkg=`rpm -qa | grep VirtualBox` if [ $? -eq 0 ]; then skip_vagrant=0 else skip_vagrant=1 fi ###legacy VM location check ###remove me later if [ -d /tmp/bgs_vagrant ]; then cd /tmp/bgs_vagrant vagrant destroy -f rm -rf /tmp/bgs_vagrant fi ###destroy vagrant if [ $skip_vagrant -eq 0 ]; then if [ -d $vm_dir ]; then ##all vm directories for vm in $( ls $vm_dir ); do cd $vm_dir/$vm if vagrant destroy -f; then echo "${blue}Successfully destroyed $vm Vagrant VM ${reset}" else echo "${red}Unable to destroy $vm Vagrant VM! Attempting to killall vagrant if process is hung ${reset}" killall vagrant echo "${blue}Checking if vagrant was already destroyed and no process is active...${reset}" if ps axf | grep vagrant; then echo "${red}Vagrant process still exists after kill...exiting ${reset}" exit 1 else echo "${blue}Vagrant process doesn't exist. Moving on... ${reset}" fi fi ##Vagrant boxes appear as VboxHeadless processes ##try to gracefully destroy the VBox VM if it still exists if vboxmanage list runningvms | grep $vm; then echo "${red} $vm VBoxHeadless process still exists...Removing${reset}" vbox_id=$(vboxmanage list runningvms | grep $vm | awk '{print $1}' | sed 's/"//g') vboxmanage controlvm $vbox_id poweroff if vboxmanage unregistervm --delete $vbox_id; then echo "${blue}$vm VM is successfully deleted! ${reset}" else echo "${red} Unable to delete VM $vm ...Exiting ${reset}" exit 1 fi else echo "${blue}$vm VM is successfully deleted! ${reset}" fi done else echo "${blue}${vm_dir} doesn't exist, no VMs in OPNFV directory to destroy! ${reset}" fi echo "${blue}Checking for any remaining virtual box processes...${reset}" ###kill virtualbox if ps axf | grep virtualbox; then echo "${blue}virtualbox processes are still running. Killing any remaining VirtualBox processes...${reset}" killall virtualbox fi ###kill any leftover VMs (brute force) if ps axf | grep VBoxHeadless; then echo "${blue}VBoxHeadless processes are still running. Killing any remaining VBoxHeadless processes...${reset}" killall VBoxHeadless fi ###remove virtualbox echo "${blue}Removing VirtualBox... ${reset}" yum -y remove $vboxpkg else echo "${blue}Skipping Vagrant destroy + VBox Removal as VirtualBox package is already removed ${reset}" fi ###remove working vm directory echo "${blue}Removing working VM directory: $vm_dir ${reset}" rm -rf $vm_dir ###check to see if libvirt is installed echo "${blue}Checking if libvirt/KVM is installed" if rpm -qa | grep -iE 'libvirt|kvm'; then echo "${blue}Libvirt/KVM is installed${reset}" echo "${blue}Checking for any QEMU/KVM VMs...${reset}" vm_count=0 while read -r line; do ((vm_count++)); done < <(virsh list --all | sed 1,2d | head -n -1) if [ $vm_count -gt 0 ]; then echo "${blue}VMs Found: $vm_count${reset}" vm_runnning=0 while read -r line; do ((vm_running++)); done < <(virsh list --all | sed 1,2d | head -n -1| grep -i running) echo "${blue}Powering off $vm_running VM(s)${reset}" while read -r vm; do if ! virsh destroy $vm; then echo "${red}WARNING: Unable to power off VM ${vm}${reset}" else echo "${blue}VM $vm powered off!${reset}" fi done < <(virsh list --all | sed 1,2d | head -n -1| grep -i running | sed 's/^[ \t]*//' | awk '{print $2}') echo "${blue}Destroying libvirt VMs...${reset}" while read -r vm; do if ! virsh undefine --remove-all-storage $vm; then echo "${red}ERROR: Unable to remove the VM ${vm}${reset}" exit 1 else echo "${blue}VM $vm removed!${reset}" fi done < <(virsh list --all | sed 1,2d | head -n -1| awk '{print $2}') else echo "${blue}No VMs found for removal" fi echo "${blue}Removing libvirt and kvm packages" yum -y remove libvirt-* yum -y remove *qemu* else echo "${blue}libvirt/KVM is not installed${reset}" fi ###remove possible VMs (needed for 'rmmod kvm_intel') if [ -n "$(ps -ef | grep qemu-kvm | grep -v grep)" ]; then echo "${blue}Removing existing VMs ${reset}" killall -9 qemu-kvm fi ###remove kernel modules echo "${blue}Removing kernel modules ${reset}" for kernel_mod in vboxnetadp vboxnetflt vboxpci vboxdrv kvm_intel kvm; do if ! rmmod $kernel_mod; then if rmmod $kernel_mod 2>&1 | grep -i 'not currently loaded'; then echo "${blue} $kernel_mod is not currently loaded! ${reset}" else echo "${red}Error trying to remove Kernel Module: $kernel_mod ${reset}" exit 1 fi else echo "${blue}Removed Kernel Module: $kernel_mod ${reset}" fi done ###remove PXE bridge echo "${blue}Checking whether PXE bridge ${pxe_bridge} exists${reset}" if ! brctl show ${pxe_bridge} 2>&1 | grep -i 'No such device'; then echo "${blue}PXE bridge ${pxe_bridge} detected! Removing...${reset}" link_state=$(ip link show ${pxe_bridge} | grep -oP 'state \K[^ ]+') if [[ ${link_state} != 'DOWN' ]]; then ip link set dev ${pxe_bridge} down sleep 5 link_state=$(ip link show ${pxe_bridge} | grep -oP 'state \K[^ ]+') if [[ ${link_state} != 'DOWN' ]]; then echo "${red}Could not bring DOWN bridge ${pxe_bridge} link state is ${link_state}${reset}" exit 1 fi fi brctl delbr ${pxe_bridge} if ifconfig | grep ${pxe_bridge} || brctl show | grep ${pxe_bridge}; then echo "${red}Error trying to remove ${pxe_bridge}${reset}" exit 1 else echo "${blue}PXE bridge ${pxe_bridge} removed${reset}" fi else echo "${blue}PXE bridge ${pxe_bridge} does not exist${reset}" fi ###remove PXE interface (VLAN 0) echo "${blue}Checking whether PXE interface (VLAN 0) exists and remove it${reset}" remove_interface_with_name_pattern "enp.+s.+\.0" ###moving IP Address from Openstack Management interface back to base interface echo "${blue}Moving IP addresses from VLAN ${management_vid} interface ${management_interface} back to interface ${private_interface}${reset}" management_interface_ip_addr_list=$(ip addr show ${management_interface} | grep -oP 'inet \K[^ ]+') if [[ ! -z ${management_interface_ip_addr_list} ]]; then echo -e "${blue}Found IP addresses on VLAN ${management_vid} interface ${management_interface}:\n${management_interface_ip_addr_list}${reset}" for management_interface_ip_addr in ${management_interface_ip_addr_list} do echo "${blue}Removing IP address ${management_interface_ip_addr} from VLAN ${management_vid} interface ${management_interface}${reset}" ip addr del ${management_interface_ip_addr} dev ${management_interface} if ip addr show ${management_interface} | grep ${management_interface_ip_addr}; then echo "${red}Could not remove IP address ${management_interface_ip_addr} from VLAN ${management_vid} interface ${management_interface}${reset}" exit 1 fi if ! ip addr show ${private_interface} | grep ${management_interface_ip_addr}; then echo "${blue}Adding IP address ${management_interface_ip_addr} to interface ${private_interface}${reset}" ip addr add ${management_interface_ip_addr} dev ${private_interface} if ! ip addr show ${private_interface} | grep ${management_interface_ip_addr}; then echo "${red}Could not set IP address ${management_interface_ip_addr} to interface ${private_interface}${reset}" exit 1 fi else echo "${blue}Interface ${private_interface} already has assigned to itself this IP address ${management_interface_ip_addr}${reset}" fi done else echo "${red}No IP Address is assigned to VLAN ${management_vid} interface ${management_interface}, there isn't any IP address to move to interface ${private_interface}${reset}" fi ###remove Openstack Management interface (VLAN 300) echo "${blue}Checking whether Openstack Management interface (VLAN 300) exists and remove it${reset}" remove_interface_with_name_pattern "enp.+s.+\.${management_vid}"