6fee386d084b333dc738ccd17c354e17203f52f3
[metze/ctdb/wip.git] / tests / simple / 16_ctdb_config_add_ip.sh
1 #!/bin/bash
2
3 test_info()
4 {
5     cat <<EOF
6 Verify that an IP address can be added to a node using 'ctdb addip'.
7
8 This test goes to some trouble to figure out which IP address to add
9 but assumes a 24-bit subnet mask.  It does not handle IPv6.  It does
10 not do any network level checks that the new IP address is reachable
11 but simply trusts 'ctdb ip' that the address has been added.  There is
12 also an extra prerequisite that the node being added to already has
13 public addresses - this is difficult to avoid if the extra address is
14 to be sensibly chosen.
15
16 Prerequisites:
17
18 * An active CTDB cluster with at least 2 active nodes.
19
20 Steps:
21
22 1. Verify that the status on all of the ctdb nodes is 'OK'.
23 2. Use 'ctdb ip' on one of the nodes to list the IP addresses being
24    served.
25 3. Add an additional public address to be served by the node, using
26    'ctdb addip'.
27 4. Verify that this IP address has been added to the list of IP
28    addresses being served by the node, using the 'ctdb ip' command.
29
30 Expected results:
31
32 * 'ctdb ip' adds an IP address to the list of public IP addresses
33   being served by a node.
34 EOF
35 }
36
37 . ctdb_test_functions.bash
38
39 ctdb_test_init "$@"
40
41 set -e
42
43 cluster_is_healthy
44
45 # Reset configuration
46 ctdb_restart_when_done
47
48 echo "Getting list of public IPs..."
49 all_ips_on_node 0
50
51 # When selecting test_node we just want a node that has public IPs.
52 # This will work and is economically semi-randomly.  :-)
53 read x test_node <<<"$out"
54
55 test_node_ips=""
56 all_ips=""
57 while read ip pnn ; do
58     all_ips="${all_ips}${all_ips:+ }${ip}"
59     [ "$pnn" = "$test_node" ] && \
60         test_node_ips="${test_node_ips}${test_node_ips:+ }${ip}"
61 done <<<"$out"
62
63 echo "Selected node ${test_node} with IPs: $test_node_ips"
64
65 # Try to find a free IP adddress.  This is inefficient but should
66 # succeed quickly.
67 try_command_on_node $test_node "ip addr show"
68 all_test_node_ips=$(echo "$out" | sed -rn -e 's@^[[:space:]]+inet[[:space:]]+([[:digit:]]+\.[[:digit:]]+\.[[:digit:]]+\.[[:digit:]]+/[[:digit:]]+).*[[:space:]]([^[:space:]]+)+$@\1:\2@p')
69
70 add_ip=""
71
72 # Use an IP already on one of the nodes, remove the last octet and
73 # loop through the possible IP addreses.
74 for i in $test_node_ips ; do
75     prefix="${i%.*}"
76     for j in $(seq 101 199) ; do
77         try="${prefix}.${j}"
78         # Try to make sure it isn't used anywhere!
79
80         # First, make sure it isn't an existing public address on the
81         # cluster.
82         for k in $all_ips ; do
83             [ "$try" = "$k" ] && continue 2
84         done
85
86         # Also make sure it isn't some other address in use on the
87         # node.
88         for k in $all_test_node_ips ; do
89             [ "$try" = "${k%/*}" ] && continue 2
90         done
91
92         # Get the interface details for $i, which our address is a
93         # close relative of.  This should never fail but it can't hurt
94         # to be careful...
95         for k in $all_test_node_ips ; do
96             if [ "$i" = "${k%/*}" ] ; then
97                         # Found one!
98                 add_ip="${try}/${k#*/}"
99                 break 3
100             fi
101         done
102     done
103 done
104
105 if [ -z "$add_ip" ] ; then
106     echo "BAD: Unable to find IP address to add."
107     exit 1
108 fi
109
110 echo "Adding IP: ${add_ip/:/ on interface }"
111 try_command_on_node $test_node $CTDB addip ${add_ip/:/ }
112
113 echo "Waiting for IP to be added..."
114 if wait_until 60 ips_are_on_nodeglob $test_node ${add_ip%/*} ; then
115     echo "That worked!"
116 else
117     echo "BAD: IP didn't get added."
118     try_command_on_node $test_node ctdb ip -n all
119     exit 1
120 fi