blob: 41795aa11c36ce0d76b66b83b475fcd843dff663 [file] [log] [blame]
Jian Lib1cd0b02019-02-21 16:41:40 +09001/*
2 * Copyright 2019-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.k8snetworking.cli;
17
18import com.google.common.collect.Lists;
19import io.fabric8.kubernetes.api.model.Endpoints;
Jian Li324d6dc2019-07-10 10:55:15 +090020import io.fabric8.kubernetes.api.model.Namespace;
Jian Lib1cd0b02019-02-21 16:41:40 +090021import io.fabric8.kubernetes.api.model.Pod;
Jian Li1cf51882019-02-26 14:41:20 +090022import io.fabric8.kubernetes.api.model.extensions.Ingress;
Jian Lif4523d82019-07-07 01:06:09 +090023import io.fabric8.kubernetes.api.model.networking.NetworkPolicy;
Jian Lib1cd0b02019-02-21 16:41:40 +090024import io.fabric8.kubernetes.client.KubernetesClient;
25import org.apache.karaf.shell.api.action.Command;
26import org.apache.karaf.shell.api.action.lifecycle.Service;
Jian Li3d1111e2019-02-22 02:02:13 +090027import org.onlab.packet.IpAddress;
28import org.onlab.packet.MacAddress;
Jian Lib1cd0b02019-02-21 16:41:40 +090029import org.onosproject.cli.AbstractShellCommand;
Jian Li3d1111e2019-02-22 02:02:13 +090030import org.onosproject.k8snetworking.api.DefaultK8sPort;
Jian Lib1cd0b02019-02-21 16:41:40 +090031import org.onosproject.k8snetworking.api.K8sEndpointsAdminService;
Jian Li1cf51882019-02-26 14:41:20 +090032import org.onosproject.k8snetworking.api.K8sIngressAdminService;
Jian Li324d6dc2019-07-10 10:55:15 +090033import org.onosproject.k8snetworking.api.K8sNamespaceAdminService;
Jian Li3d1111e2019-02-22 02:02:13 +090034import org.onosproject.k8snetworking.api.K8sNetworkAdminService;
Jian Lif4523d82019-07-07 01:06:09 +090035import org.onosproject.k8snetworking.api.K8sNetworkPolicyAdminService;
Jian Lib1cd0b02019-02-21 16:41:40 +090036import org.onosproject.k8snetworking.api.K8sPodAdminService;
Jian Li3d1111e2019-02-22 02:02:13 +090037import org.onosproject.k8snetworking.api.K8sPort;
Jian Lib1cd0b02019-02-21 16:41:40 +090038import org.onosproject.k8snetworking.api.K8sServiceAdminService;
39import org.onosproject.k8snetworking.util.K8sNetworkingUtil;
40import org.onosproject.k8snode.api.K8sApiConfig;
41import org.onosproject.k8snode.api.K8sApiConfigService;
Jian Li3d1111e2019-02-22 02:02:13 +090042import org.onosproject.net.DeviceId;
43import org.onosproject.net.PortNumber;
Jian Lib1cd0b02019-02-21 16:41:40 +090044
45import java.util.List;
Jian Li3d1111e2019-02-22 02:02:13 +090046import java.util.Map;
47
48import static org.onosproject.k8snetworking.api.K8sPort.State.INACTIVE;
Jian Lib1cd0b02019-02-21 16:41:40 +090049
50/**
51 * Synchronizes kubernetes states.
52 */
53@Service
54@Command(scope = "onos", name = "k8s-sync-states",
55 description = "Synchronizes all kubernetes states")
56public class K8sSyncStateCommand extends AbstractShellCommand {
57
58 private static final String POD_FORMAT = "%-50s%-15s%-15s%-30s";
59 private static final String SERVICE_FORMAT = "%-50s%-30s%-30s";
60 private static final String ENDPOINTS_FORMAT = "%-50s%-50s%-20s";
Jian Li1cf51882019-02-26 14:41:20 +090061 private static final String INGRESS_FORMAT = "%-50s%-15s%-30s";
Jian Lif4523d82019-07-07 01:06:09 +090062 private static final String NETWORK_POLICY_FORMAT = "%-50s%-15s%-30s";
Jian Li324d6dc2019-07-10 10:55:15 +090063 private static final String NAMESPACE_FORMAT = "%-50s%-15s%-30s";
Jian Lib1cd0b02019-02-21 16:41:40 +090064
Jian Li3d1111e2019-02-22 02:02:13 +090065 private static final String PORT_ID = "portId";
66 private static final String DEVICE_ID = "deviceId";
67 private static final String PORT_NUMBER = "portNumber";
68 private static final String IP_ADDRESS = "ipAddress";
69 private static final String MAC_ADDRESS = "macAddress";
70 private static final String NETWORK_ID = "networkId";
71
Jian Lib1cd0b02019-02-21 16:41:40 +090072 @Override
73 protected void doExecute() {
74 K8sApiConfigService configService = get(K8sApiConfigService.class);
75 K8sPodAdminService podAdminService = get(K8sPodAdminService.class);
Jian Li324d6dc2019-07-10 10:55:15 +090076 K8sNamespaceAdminService namespaceAdminService =
77 get(K8sNamespaceAdminService.class);
Jian Lib1cd0b02019-02-21 16:41:40 +090078 K8sServiceAdminService serviceAdminService =
79 get(K8sServiceAdminService.class);
Jian Li1cf51882019-02-26 14:41:20 +090080 K8sIngressAdminService ingressAdminService =
81 get(K8sIngressAdminService.class);
Jian Lib1cd0b02019-02-21 16:41:40 +090082 K8sEndpointsAdminService endpointsAdminService =
83 get(K8sEndpointsAdminService.class);
Jian Li3d1111e2019-02-22 02:02:13 +090084 K8sNetworkAdminService networkAdminService =
85 get(K8sNetworkAdminService.class);
Jian Lif4523d82019-07-07 01:06:09 +090086 K8sNetworkPolicyAdminService networkPolicyAdminService =
87 get(K8sNetworkPolicyAdminService.class);
Jian Lib1cd0b02019-02-21 16:41:40 +090088
89 K8sApiConfig config =
90 configService.apiConfigs().stream().findAny().orElse(null);
91 if (config == null) {
92 log.error("Failed to find valid kubernetes API configuration.");
93 return;
94 }
95
96 KubernetesClient client = K8sNetworkingUtil.k8sClient(config);
97
98 if (client == null) {
99 log.error("Failed to connect to kubernetes API server.");
100 return;
101 }
102
Jian Li324d6dc2019-07-10 10:55:15 +0900103 print("\nSynchronizing kubernetes namespaces");
104 print(NAMESPACE_FORMAT, "Name", "Phase", "Labels");
105 client.namespaces().list().getItems().forEach(ns -> {
106 if (namespaceAdminService.namespace(ns.getMetadata().getUid()) != null) {
107 namespaceAdminService.updateNamespace(ns);
108 } else {
109 namespaceAdminService.createNamespace(ns);
110 }
111 printNamespace(ns);
112 });
113
Jian Lib1cd0b02019-02-21 16:41:40 +0900114 print("Synchronizing kubernetes services");
115 print(SERVICE_FORMAT, "Name", "Cluster IP", "Ports");
Jian Liaf081522019-06-19 20:53:07 +0900116 client.services().inAnyNamespace().list().getItems().forEach(svc -> {
Jian Lib1cd0b02019-02-21 16:41:40 +0900117 if (serviceAdminService.service(svc.getMetadata().getUid()) != null) {
118 serviceAdminService.updateService(svc);
119 } else {
120 serviceAdminService.createService(svc);
121 }
122 printService(svc);
123 });
124
125 print("\nSynchronizing kubernetes endpoints");
126 print(ENDPOINTS_FORMAT, "Name", "IP Addresses", "Ports");
Jian Liaf081522019-06-19 20:53:07 +0900127 client.endpoints().inAnyNamespace().list().getItems().forEach(ep -> {
Jian Lib1cd0b02019-02-21 16:41:40 +0900128 if (endpointsAdminService.endpoints(ep.getMetadata().getUid()) != null) {
129 endpointsAdminService.updateEndpoints(ep);
130 } else {
131 endpointsAdminService.createEndpoints(ep);
132 }
133 printEndpoints(ep);
134 });
135
136 print("\nSynchronizing kubernetes pods");
137 print(POD_FORMAT, "Name", "Namespace", "IP", "Containers");
Jian Liaf081522019-06-19 20:53:07 +0900138 client.pods().inAnyNamespace().list().getItems().forEach(pod -> {
Jian Lib1cd0b02019-02-21 16:41:40 +0900139 if (podAdminService.pod(pod.getMetadata().getUid()) != null) {
140 podAdminService.updatePod(pod);
141 } else {
142 podAdminService.createPod(pod);
143 }
Jian Li3d1111e2019-02-22 02:02:13 +0900144
145 syncPortFromPod(pod, networkAdminService);
146
Jian Lib1cd0b02019-02-21 16:41:40 +0900147 printPod(pod);
148 });
Jian Li1cf51882019-02-26 14:41:20 +0900149
150 print("\nSynchronizing kubernetes ingresses");
151 print(INGRESS_FORMAT, "Name", "Namespace", "LB Addresses");
Jian Liaf081522019-06-19 20:53:07 +0900152 client.extensions().ingresses().inAnyNamespace().list().getItems().forEach(ingress -> {
Jian Li1cf51882019-02-26 14:41:20 +0900153 if (ingressAdminService.ingress(ingress.getMetadata().getUid()) != null) {
154 ingressAdminService.updateIngress(ingress);
155 } else {
156 ingressAdminService.createIngress(ingress);
157 }
158 printIngresses(ingress);
159 });
160
Jian Lif4523d82019-07-07 01:06:09 +0900161 print("\nSynchronizing kubernetes network policies");
162 print(NETWORK_POLICY_FORMAT, "Name", "Namespace", "Types");
163 client.network().networkPolicies().inAnyNamespace().list().getItems().forEach(policy -> {
164 if (networkPolicyAdminService.networkPolicy(policy.getMetadata().getUid()) != null) {
165 networkPolicyAdminService.updateNetworkPolicy(policy);
166 } else {
167 networkPolicyAdminService.createNetworkPolicy(policy);
168 }
169 printNetworkPolicy(policy);
170 });
Jian Li1cf51882019-02-26 14:41:20 +0900171 }
172
173 private void printIngresses(Ingress ingress) {
174
175 List<String> lbIps = Lists.newArrayList();
176
177 ingress.getStatus().getLoadBalancer()
178 .getIngress().forEach(i -> lbIps.add(i.getIp()));
179
180 print(INGRESS_FORMAT,
181 ingress.getMetadata().getName(),
182 ingress.getMetadata().getNamespace(),
183 lbIps.isEmpty() ? "" : lbIps);
Jian Lib1cd0b02019-02-21 16:41:40 +0900184 }
185
186 private void printEndpoints(Endpoints endpoints) {
187 List<String> ips = Lists.newArrayList();
188 List<Integer> ports = Lists.newArrayList();
189
190 endpoints.getSubsets().forEach(e -> {
191 e.getAddresses().forEach(a -> ips.add(a.getIp()));
192 e.getPorts().forEach(p -> ports.add(p.getPort()));
193 });
194
195 print(ENDPOINTS_FORMAT,
196 endpoints.getMetadata().getName(),
197 ips.isEmpty() ? "" : ips,
198 ports.isEmpty() ? "" : ports);
199 }
200
Jian Li324d6dc2019-07-10 10:55:15 +0900201 private void printNamespace(Namespace namespace) {
202 print(NAMESPACE_FORMAT,
203 namespace.getMetadata().getName(),
204 namespace.getStatus().getPhase(),
205 namespace.getMetadata() != null &&
206 namespace.getMetadata().getLabels() != null &&
207 !namespace.getMetadata().getLabels().isEmpty() ?
208 namespace.getMetadata().getLabels() : "");
209 }
210
Jian Lib1cd0b02019-02-21 16:41:40 +0900211 private void printService(io.fabric8.kubernetes.api.model.Service service) {
212
213 List<Integer> ports = Lists.newArrayList();
214
215 service.getSpec().getPorts().forEach(p -> ports.add(p.getPort()));
216
217 print(SERVICE_FORMAT,
218 service.getMetadata().getName(),
219 service.getSpec().getClusterIP(),
220 ports.isEmpty() ? "" : ports);
221 }
222
223 private void printPod(Pod pod) {
224
225 List<String> containers = Lists.newArrayList();
226
227 pod.getSpec().getContainers().forEach(c -> containers.add(c.getName()));
228
229 print(POD_FORMAT,
230 pod.getMetadata().getName(),
231 pod.getMetadata().getNamespace(),
232 pod.getStatus().getPodIP(),
233 containers.isEmpty() ? "" : containers);
234 }
Jian Li3d1111e2019-02-22 02:02:13 +0900235
Jian Lif4523d82019-07-07 01:06:09 +0900236 private void printNetworkPolicy(NetworkPolicy policy) {
237 print(NETWORK_POLICY_FORMAT,
238 policy.getMetadata().getName(),
239 policy.getMetadata().getNamespace(),
240 policy.getSpec().getPolicyTypes().isEmpty() ?
241 "" : policy.getSpec().getPolicyTypes());
242 }
243
Jian Li3d1111e2019-02-22 02:02:13 +0900244 private void syncPortFromPod(Pod pod, K8sNetworkAdminService adminService) {
245 Map<String, String> annotations = pod.getMetadata().getAnnotations();
246 if (annotations != null && !annotations.isEmpty() &&
247 annotations.get(PORT_ID) != null) {
248 String portId = annotations.get(PORT_ID);
249
250 K8sPort oldPort = adminService.port(portId);
251
252 String networkId = annotations.get(NETWORK_ID);
253 DeviceId deviceId = DeviceId.deviceId(annotations.get(DEVICE_ID));
254 PortNumber portNumber = PortNumber.portNumber(annotations.get(PORT_NUMBER));
255 IpAddress ipAddress = IpAddress.valueOf(annotations.get(IP_ADDRESS));
256 MacAddress macAddress = MacAddress.valueOf(annotations.get(MAC_ADDRESS));
257
258 K8sPort newPort = DefaultK8sPort.builder()
259 .portId(portId)
260 .networkId(networkId)
261 .deviceId(deviceId)
262 .ipAddress(ipAddress)
263 .macAddress(macAddress)
264 .portNumber(portNumber)
265 .state(INACTIVE)
266 .build();
267
268 if (oldPort == null) {
269 adminService.createPort(newPort);
270 } else {
271 adminService.updatePort(newPort);
272 }
273 }
274 }
Jian Lib1cd0b02019-02-21 16:41:40 +0900275}