Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 1 | #!/bin/bash |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 2 | silent=false |
| 3 | cleaning=false |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 4 | all_computes=false |
| 5 | fill_mode=false |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 6 | zone=nova |
| 7 | use_fqdn=false |
Ievgeniia Zadorozhna | 66dbe12 | 2024-01-22 23:59:05 +0100 | [diff] [blame^] | 8 | vm_name_prefix=cvp_test_vm_ |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 9 | |
| 10 | tmp_out=$(mktemp) |
| 11 | trap "rm -f ${tmp_out}" EXIT |
| 12 | |
| 13 | declare errors=() |
| 14 | |
| 15 | function show_help { |
| 16 | printf "Compute check/filling script\n\t-h, -?\tShow this help\n" |
| 17 | printf "\t-d\tCleaning of earlier created VMs\n" |
| 18 | printf "\t-q\tSilent mode\n" |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 19 | printf "\t-a\tEnumeratre all computes\n" |
| 20 | printf "\t-f\tFill mode\n" |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 21 | printf "\t-z <zone>\tAvailability zone to use on create\n" |
| 22 | printf "\t-n\tUse compute's FQDN when setting zone hint\n" |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 23 | printf "\nUsage: cmp_check.sh (-a | <compute_hostname>) (-f [<vm_count>|def:1])\n" |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 24 | printf "\t<compute_hostname> is a host shortname\n" |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 25 | printf "\t<vm_count> is optional. Defaults to 1\n" |
| 26 | printf "Examples:\n" |
| 27 | printf "\tFill all computes with 3 VMs:\n\t\t'bash cmp_check.sh -fa 3'\n\n" |
| 28 | printf "\tFill specific compute with 5 VMs:\n\t\t'bash cmp_check.sh -f cmp001 5'\n\n" |
| 29 | printf "\tCheck all computes:\n\t\t'bash cmp_check.sh -a'\n\n" |
| 30 | printf "\tCheck specific compute:\n\t\t'bash cmp_check.sh cmp001'\n\n" |
| 31 | printf "\tClean all computes:\n\t\t'bash cmp_check.sh -da'\n\n" |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 32 | } |
| 33 | |
| 34 | OPTIND=1 # Reset in case getopts has been used previously in the shell. |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 35 | while getopts "h?:qdafz:n" opt; do |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 36 | case "$opt" in |
| 37 | h|\?) |
| 38 | show_help |
| 39 | exit 0 |
| 40 | ;; |
| 41 | q) silent=true |
| 42 | ;; |
| 43 | d) cleaning=true |
| 44 | ;; |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 45 | a) all_computes=true |
| 46 | ;; |
| 47 | f) fill_mode=true |
| 48 | ;; |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 49 | z) zone=${OPTARG} |
| 50 | printf "# Using availability zone of '${zone}'\n" |
| 51 | ;; |
| 52 | n) use_fqdn=true |
| 53 | printf "# Using FQDN as a compute host name\n" |
| 54 | ;; |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 55 | esac |
| 56 | done |
| 57 | |
| 58 | shift $((OPTIND-1)) |
| 59 | [ "${1:-}" = "--" ] && shift |
| 60 | |
| 61 | # Check and create cmp_name var |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 62 | if [[ -z ${1+x} ]] && [[ ! ${all_computes} == "true" ]]; then |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 63 | show_help |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 64 | printf "\nERROR: No compute host specified\n" |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 65 | exit 1 |
| 66 | fi |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 67 | if [[ ${all_computes} == "true" ]]; then |
| 68 | cmp_name=all |
| 69 | # if enumerate mode is set, vmcount source is ${1} |
| 70 | if [[ -z ${1+x} ]] || [[ ! ${fill_mode} == true ]]; then |
| 71 | vmcount=1 |
| 72 | else |
| 73 | vmcount=${1} |
| 74 | fi |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 75 | else |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 76 | cmp_name=${1} |
| 77 | # in single compute mode, vmcount source is ${2} |
| 78 | # in check mode count is always 1 |
| 79 | if [[ -z ${2+x} ]] || [[ ! ${fill_mode} == true ]]; then |
| 80 | vmcount=1 |
| 81 | else |
| 82 | vmcount=${2} |
| 83 | fi |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 84 | fi |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 85 | |
| 86 | function cmp_stats() { |
Ievgeniia Zadorozhna | a33b29a | 2023-08-29 18:19:47 +0300 | [diff] [blame] | 87 | cmpid=$(openstack hypervisor list | grep ${1} | awk '{print $2}') |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 88 | vars=( $(openstack hypervisor show ${cmpid} -f shell -c state -c running_vms -c vcpus -c vcpus_used -c memory_mb -c memory_mb_used) ) |
| 89 | [ ! 0 -eq $? ] && errors+=("${1}: $(cat ${vars[@]})") |
| 90 | if [ ! $state == '"up"' ]; then |
| 91 | echo "# Hypervisor fail, state is '${state}'" |
| 92 | errors |
| 93 | exit 1 |
| 94 | else |
| 95 | declare ${vars[@]} |
| 96 | printf "${1}: vms=%s vcpus=%s/%s ram=%s/%s\n" ${running_vms} ${vcpus_used} ${vcpus} ${memory_mb_used} ${memory_mb} |
| 97 | fi |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 98 | } |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 99 | |
| 100 | function waitfor () { |
| 101 | counter=0 |
| 102 | while [ ${counter} -lt 6 ]; do |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 103 | ids=( $(openstack server list --name ${1} --status ${2} -f value -c ID) ) |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 104 | if [ ${#ids[@]} -eq 0 ]; then |
| 105 | sleep 5 |
| 106 | counter=$((counter + 1)) |
| 107 | else |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 108 | [ ! "$silent" = true ] && printf "# '${1}' reached status ${2}\n" |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 109 | break |
| 110 | fi |
| 111 | done |
| 112 | } |
| 113 | |
| 114 | function getid() { |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 115 | openstack server list -c ID -c Name -f value | grep "${1}" | cut -d' ' -f1 |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 116 | } |
| 117 | |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 118 | function get_all_cmp() { |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 119 | if [ $use_fqdn == true ]; then |
| 120 | openstack hypervisor list -f value -c "Hypervisor Hostname" -c State | grep "up" | sort | cut -d' ' -f1 |
| 121 | else |
| 122 | openstack hypervisor list -f value -c "Hypervisor Hostname" -c State | grep "up" | sort | cut -d'.' -f1 |
| 123 | fi |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 124 | } |
| 125 | |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 126 | function vm_create() { |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 127 | [ ! "$silent" = true ] && set -x |
Alex | db7786b | 2022-02-21 17:58:29 -0600 | [diff] [blame] | 128 | openstack server create --nic net-id=${fixed_net_left_id} --image ${cirros51_id} --flavor ${flavor_tiny_id} --key-name ${keypair_id} --security-group ${secgroup_all_id} --availability-zone ${zone}:${1} ${2} 2>${tmp_out} >/dev/null |
Alex | 83075ac | 2022-02-16 13:39:50 -0600 | [diff] [blame] | 129 | #openstack server create --nic net-id=${fixed_net_left_id} --image ${ubuntu16_id} --flavor ${flavor_high_id} --key-name ${keypair_id} --security-group ${secgroup_all_id} --availability-zone ${zone}:${1} ${2} 2>${tmp_out} >/dev/null |
| 130 | #openstack server create --nic net-id=${fixed_net_right_id} --image ${ubuntu16_id} --flavor ${flavor_high_id} --key-name ${keypair_id} --security-group ${secgroup_all_id} --availability-zone ${zone}:${1} ${2} 2>${tmp_out} >/dev/null |
Alex | db7786b | 2022-02-21 17:58:29 -0600 | [diff] [blame] | 131 | #openstack server create --nic net-id=${fixed_net_left_id} --image ${ubuntu20_id} --flavor ${flavor_high_id} --key-name ${keypair_id} --security-group ${secgroup_all_id} --availability-zone ${zone}:${1} ${2} 2>${tmp_out} >/dev/null |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 132 | [ ! 0 -eq $? ] && errors+=("${1}/${2}: $(cat ${tmp_out})") |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 133 | set +x |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 134 | [ ! "$silent" = true ] && cat ${tmp_out} |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 135 | } |
| 136 | |
| 137 | function vm_action() { |
Alex | 6892c8c | 2021-01-28 16:07:29 -0600 | [diff] [blame] | 138 | openstack server ${1} ${2} 2>${tmp_out} >/dev/null |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 139 | if [ ! 0 -eq $? ]; then |
| 140 | errors+=("${cmp_name}: $(cat ${tmp_out})") |
| 141 | fi |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 142 | } |
| 143 | |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 144 | function errors { |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 145 | echo "==== Errors" |
| 146 | for i in "${!errors[@]}"; do |
| 147 | printf "#%s\n" "${errors[$i]}" |
| 148 | done |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 149 | } |
| 150 | |
| 151 | function join_by { local IFS="$1"; shift; echo "$*"; } |
| 152 | |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 153 | |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 154 | function clean_cmp() { |
| 155 | # #### Cleaning mode |
| 156 | if [ $cleaning = true ]; then |
Ievgeniia Zadorozhna | 66dbe12 | 2024-01-22 23:59:05 +0100 | [diff] [blame^] | 157 | vmname=${vm_name_prefix}${1} |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 158 | vmid=( $(getid ${vmname}) ) |
| 159 | if [ ${#vmid[@]} -ne 0 ]; then |
| 160 | [ ! "$silent" = true ] && echo "# ${1}: cleaning ${#vmid[@]} VMs" |
| 161 | vm_action delete "$(join_by ' ' "${vmid[@]}")" |
| 162 | else |
| 163 | [ ! "$silent" = true ] && echo "# ${1}: ...no VMs found" |
| 164 | fi |
| 165 | fi |
| 166 | } |
| 167 | |
| 168 | function check_cmp_node() { |
| 169 | cmp_stats ${1} |
| 170 | vm_create ${1} ${2} |
| 171 | waitfor ${2} ACTIVE |
| 172 | vmid=$(getid ${2}) |
| 173 | |
| 174 | cmp_stats ${1} |
| 175 | |
| 176 | vm_action pause ${vmid} |
| 177 | waitfor ${2} PAUSED |
| 178 | vm_action unpause ${vmid} |
| 179 | waitfor ${2} ACTIVE |
| 180 | |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 181 | [ ! "$silent" = true ] && echo "# ... deleting created VMs" |
| 182 | clean_cmp ${1} |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 183 | cmp_stats ${1} |
| 184 | } |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 185 | |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 186 | if [ ! -f cvp.manifest ]; then |
| 187 | echo "ERROR: No cvp.manifest file detected. Consider running prepare.sh" |
| 188 | exit 1 |
| 189 | else |
| 190 | source cvp.manifest |
| 191 | fi |
| 192 | |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 193 | [ ! "$silent" = true ] && echo "# Sourcing cvprc" |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 194 | source cvprc |
| 195 | |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 196 | # #### Checking for CMP existence |
| 197 | if [[ ! ${cmp_name} == "all" ]]; then |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 198 | echo "# Inspecting '${zone}:${cmp_name}'" |
| 199 | cmp_fqdn=$(openstack host list --zone ${zone} -f value -c 'Host Name' -c 'Zone' | grep ${cmp_name} | cut -d' ' -f1 2>${tmp_out}) |
| 200 | [ ! 0 -eq $? ] && errors+=("${cmp_name}\@${zone}: $(cat ${tmp_out})") |
| 201 | if [[ -z ${cmp_fqdn} ]]; then |
| 202 | echo "ERROR: ${cmp_name} not found in ${zone}" |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 203 | errors |
| 204 | exit 1 |
| 205 | fi |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 206 | printf "# Found ${cmp_fqdn} in '${zone}' using given name of ${cmp_name}\n" |
| 207 | vars=( $(openstack hypervisor show ${cmp_fqdn} -f shell -c id -c state -c hypervisor_hostname) ) |
| 208 | [ ! 0 -eq $? ] && errors+=("${cmp_name}: $(cat ${tmp_out})") |
| 209 | declare ${vars[@]} |
| 210 | # check that such node exists |
| 211 | if [ -z ${id+x} ]; then |
| 212 | # no id |
| 213 | echo "ERROR: ${cmp_name} not found among hypervisors" |
| 214 | errors |
| 215 | exit 1 |
| 216 | else |
| 217 | echo "# ${id}, ${hypervisor_hostname}, status '${state}'" |
| 218 | if [ ! ${state} == '"up"' ]; then |
| 219 | echo "ERROR: ${hypervisor_hostname} is '${state}'" |
| 220 | exit 1 |
| 221 | else |
| 222 | unset id |
| 223 | unset hypervisor_hostname |
| 224 | unset state |
| 225 | fi |
| 226 | fi |
| 227 | |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 228 | fi |
| 229 | |
| 230 | if [[ ${cmp_name} == all ]]; then |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 231 | echo "# Gathering compute count with state 'up'" |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 232 | cmp_nodes=( $(get_all_cmp) ) |
| 233 | else |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 234 | if [ $use_fqdn == true ]; then |
| 235 | cmp_nodes=( ${cmp_fqdn} ) |
| 236 | else |
| 237 | cmp_nodes=( ${cmp_name} ) |
| 238 | fi |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 239 | fi |
| 240 | |
| 241 | |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 242 | # #### Cleaning mode |
| 243 | if [ $cleaning = true ]; then |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 244 | # get all computes |
| 245 | if [[ ${cmp_name} == all ]]; then |
| 246 | echo "# Cleaning ${#cmp_nodes[@]} computes" |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 247 | else |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 248 | echo "# Cleaning ${cmp_name}" |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 249 | fi |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 250 | |
| 251 | # clean them |
| 252 | for node in ${cmp_nodes[@]}; do |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 253 | cname=$(echo ${node} | cut -d'.' -f1) |
| 254 | clean_cmp ${cname} |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 255 | done |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 256 | echo "# Done cleaning" |
| 257 | errors |
| 258 | exit 0 |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 259 | fi |
| 260 | |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 261 | # ### |
| 262 | if [[ ! ${fill_mode} = true ]]; then |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 263 | # ### CMP Checking mode |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 264 | if [[ ${cmp_name} == all ]]; then |
| 265 | echo "# Checking ${#cmp_nodes[@]} computes" |
| 266 | fi |
| 267 | # check node |
| 268 | for node in ${cmp_nodes[@]}; do |
| 269 | echo "# ${node}: checking" |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 270 | cname=$(echo ${node} | cut -d'.' -f1) |
Ievgeniia Zadorozhna | 66dbe12 | 2024-01-22 23:59:05 +0100 | [diff] [blame^] | 271 | check_cmp_node ${node} ${vm_name_prefix}${cname} |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 272 | echo "# ${node}: done" |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 273 | done |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 274 | errors |
| 275 | exit 0 |
| 276 | else |
| 277 | # ### CMP fillling mode |
| 278 | if [[ ${cmp_name} == all ]]; then |
| 279 | echo "# Filling ${#cmp_nodes[@]} computes" |
| 280 | fi |
| 281 | |
| 282 | for node in ${cmp_nodes[@]}; do |
| 283 | echo "# ${node}: filling" |
| 284 | counter=1 |
| 285 | while [[ $counter -lt ${vmcount}+1 ]]; do |
Alex Savatieiev | 814307f | 2020-04-13 12:47:39 -0500 | [diff] [blame] | 286 | cname=$(echo ${node} | cut -d'.' -f1) |
Ievgeniia Zadorozhna | 66dbe12 | 2024-01-22 23:59:05 +0100 | [diff] [blame^] | 287 | vmname_c=${vm_name_prefix}${cname}_$(printf "%02d" ${counter}) |
Alex Savatieiev | d0ae84f | 2019-10-23 13:36:37 -0500 | [diff] [blame] | 288 | [ ! "$silent" = true ] && echo "# ${node}: creating ${vmname_c}" |
| 289 | vm_create ${node} ${vmname_c} |
| 290 | cmp_stats ${node} |
| 291 | ((counter++)) |
| 292 | done |
| 293 | printf "# ${node}: done\n" |
| 294 | done |
Alex Savatieiev | eaf0a99 | 2019-10-02 17:51:54 -0500 | [diff] [blame] | 295 | fi |
| 296 | |
| 297 | errors |
Alex Savatieiev | badc476 | 2019-09-30 13:46:37 -0500 | [diff] [blame] | 298 | |