blob: 53cfb316c3eb5bfe528f4dbf16521deb40e3245e [file] [log] [blame]
Jian Li0b93b002018-07-31 13:41:08 +09001/*
2 * Copyright 2018-present Open Networking Foundation
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16package org.onosproject.openstacktroubleshoot.cli;
17
Jian Lie189c1c2018-08-08 15:55:08 +090018import com.google.common.collect.Sets;
19import org.apache.karaf.shell.commands.Argument;
Jian Li0b93b002018-07-31 13:41:08 +090020import org.apache.karaf.shell.commands.Command;
Jian Lie189c1c2018-08-08 15:55:08 +090021import org.apache.karaf.shell.commands.Option;
22import org.onlab.packet.IpAddress;
Jian Li0b93b002018-07-31 13:41:08 +090023import org.onosproject.cli.AbstractShellCommand;
Jian Li43d04282018-08-22 14:31:21 +090024import org.onosproject.cluster.ClusterService;
25import org.onosproject.cluster.NodeId;
26import org.onosproject.mastership.MastershipService;
Jian Lie189c1c2018-08-08 15:55:08 +090027import org.onosproject.openstacknetworking.api.InstancePort;
28import org.onosproject.openstacknetworking.api.InstancePortService;
Jian Li43d04282018-08-22 14:31:21 +090029import org.onosproject.openstacknode.api.OpenstackNode;
30import org.onosproject.openstacknode.api.OpenstackNodeService;
Jian Li0b93b002018-07-31 13:41:08 +090031import org.onosproject.openstacktroubleshoot.api.OpenstackTroubleshootService;
32import org.onosproject.openstacktroubleshoot.api.Reachability;
33
Jian Lie189c1c2018-08-08 15:55:08 +090034import java.util.Optional;
35import java.util.Set;
Jian Li9171c002018-08-19 22:58:40 +090036import java.util.concurrent.ExecutorService;
Jian Lie189c1c2018-08-08 15:55:08 +090037import java.util.stream.Collectors;
Jian Li0b93b002018-07-31 13:41:08 +090038
Jian Li9171c002018-08-19 22:58:40 +090039import static java.util.concurrent.Executors.newSingleThreadScheduledExecutor;
40import static org.onlab.util.Tools.groupedThreads;
Jian Lic38e9032018-08-09 17:08:38 +090041import static org.onosproject.openstacknetworking.api.InstancePort.State.ACTIVE;
Jian Li43d04282018-08-22 14:31:21 +090042import static org.onosproject.openstacknode.api.OpenstackNode.NodeType.COMPUTE;
Jian Lic38e9032018-08-09 17:08:38 +090043
Jian Li0b93b002018-07-31 13:41:08 +090044/**
45 * Checks the east-west VMs connectivity.
46 */
47@Command(scope = "onos", name = "openstack-check-east-west",
48 description = "Checks the east-west VMs connectivity")
49public class OpenstackEastWestProbeCommand extends AbstractShellCommand {
50
51 private static final String REACHABLE = "Reachable :)";
52 private static final String UNREACHABLE = "Unreachable :(";
53 private static final String ARROW = "->";
54
55 private static final String FORMAT = "%-20s%-5s%-20s%-20s";
56
Jian Lie189c1c2018-08-08 15:55:08 +090057 @Option(name = "-a", aliases = "--all", description = "Apply this command to all VMs",
58 required = false, multiValued = false)
59 private boolean isAll = false;
60
61 @Argument(index = 0, name = "vmIps", description = "VMs' IP addresses",
62 required = false, multiValued = true)
63 private String[] vmIps = null;
64
Jian Li9171c002018-08-19 22:58:40 +090065 private final ExecutorService probeExecutor = newSingleThreadScheduledExecutor(
66 groupedThreads(this.getClass().getSimpleName(), "probe-handler", log));
67
Jian Li0b93b002018-07-31 13:41:08 +090068 @Override
69 protected void execute() {
Jian Li43d04282018-08-22 14:31:21 +090070 OpenstackTroubleshootService tsService = get(OpenstackTroubleshootService.class);
Jian Lic38e9032018-08-09 17:08:38 +090071 InstancePortService instPortService = get(InstancePortService.class);
Jian Li43d04282018-08-22 14:31:21 +090072 MastershipService mastershipService = get(MastershipService.class);
73 ClusterService clusterService = get(ClusterService.class);
74 OpenstackNodeService osNodeService = get(OpenstackNodeService.class);
Jian Lie189c1c2018-08-08 15:55:08 +090075
76 if (tsService == null) {
Jian Li0b93b002018-07-31 13:41:08 +090077 error("Failed to troubleshoot openstack networking.");
78 return;
79 }
80
Jian Lie189c1c2018-08-08 15:55:08 +090081 if ((!isAll && vmIps == null) || (isAll && vmIps != null)) {
82 print("Please specify one of VM IP address or -a option.");
83 return;
84 }
85
Jian Li43d04282018-08-22 14:31:21 +090086 NodeId localNodeId = clusterService.getLocalNode().id();
87
88 for (OpenstackNode node : osNodeService.completeNodes(COMPUTE)) {
89 if (!localNodeId.equals(mastershipService.getMasterFor(node.intgBridge()))) {
90 error("Current node is not the master for all compute nodes. " +
91 "Please enforce mastership first using openstack-reset-mastership -c !");
92 return;
93 }
94 }
95
Jian Lie189c1c2018-08-08 15:55:08 +090096 if (isAll) {
97 printHeader();
Jian Lic38e9032018-08-09 17:08:38 +090098 // send ICMP PACKET_OUT to all connect VMs whose instance port state is ACTIVE
99 Set<InstancePort> activePorts = instPortService.instancePorts().stream()
100 .filter(p -> p.state() == ACTIVE)
101 .collect(Collectors.toSet());
102
103 activePorts.forEach(srcPort ->
104 activePorts.forEach(dstPort ->
105 printReachability(tsService.probeEastWest(srcPort, dstPort))
106 )
107 );
Jian Lie189c1c2018-08-08 15:55:08 +0900108 } else {
109 if (vmIps.length > 2) {
110 print("Too many VM IPs. The number of IP should be limited to 2.");
111 return;
112 }
113
Jian Lic38e9032018-08-09 17:08:38 +0900114 IpAddress srcIp = getIpAddress(vmIps[0]);
115
116 if (srcIp == null) {
117 return;
118 }
119
Jian Lie189c1c2018-08-08 15:55:08 +0900120 InstancePort srcPort = instPort(instPortService, srcIp);
121
122 if (srcPort == null) {
123 print("Specified source IP is not existing.");
124 return;
125 }
126
127 final Set<IpAddress> dstIps = Sets.newConcurrentHashSet();
128
129 if (vmIps.length == 2) {
Jian Lic38e9032018-08-09 17:08:38 +0900130 IpAddress dstIp = getIpAddress(vmIps[1]);
131
132 if (dstIp == null) {
133 return;
134 }
135
136 dstIps.add(dstIp);
Jian Lie189c1c2018-08-08 15:55:08 +0900137 }
138
139 if (vmIps.length == 1) {
140 dstIps.addAll(instPortService.instancePorts().stream()
141 .filter(p -> !p.ipAddress().equals(srcIp))
142 .filter(p -> p.state().equals(InstancePort.State.ACTIVE))
143 .map(InstancePort::ipAddress)
144 .collect(Collectors.toSet()));
145 }
146
147 printHeader();
148 dstIps.stream()
149 .filter(ip -> instPort(instPortService, ip) != null)
150 .map(ip -> instPort(instPortService, ip))
Jian Li9171c002018-08-19 22:58:40 +0900151 .forEach(port -> probeExecutor.execute(() ->
152 printReachability(tsService.probeEastWest(srcPort, port))));
Jian Lie189c1c2018-08-08 15:55:08 +0900153 }
154 }
155
156 private InstancePort instPort(InstancePortService service, IpAddress ip) {
157 Optional<InstancePort> port = service.instancePorts().stream()
158 .filter(p -> p.ipAddress().equals(ip)).findFirst();
159
160 if (port.isPresent()) {
161 return port.get();
162 } else {
163 print("Specified destination IP is not existing.");
164 return null;
165 }
166 }
167
168 private void printHeader() {
Jian Li0b93b002018-07-31 13:41:08 +0900169 print(FORMAT, "Source IP", "", "Destination IP", "Reachability");
Jian Lie189c1c2018-08-08 15:55:08 +0900170 }
Jian Li0b93b002018-07-31 13:41:08 +0900171
Jian Lie189c1c2018-08-08 15:55:08 +0900172 private void printReachability(Reachability r) {
173 String result = r.isReachable() ? REACHABLE : UNREACHABLE;
174 print(FORMAT, r.srcIp().toString(), ARROW, r.dstIp().toString(), result);
Jian Li0b93b002018-07-31 13:41:08 +0900175 }
Jian Lic38e9032018-08-09 17:08:38 +0900176
177 private IpAddress getIpAddress(String ipString) {
178 try {
179 return IpAddress.valueOf(vmIps[0]);
180 } catch (IllegalArgumentException e) {
181 error("Invalid IP address string.");
182 return null;
183 }
184 }
Jian Li0b93b002018-07-31 13:41:08 +0900185}