blob: 843c2f8ebbd326a930989d9f689f5f84c01856f4 [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
Jian Lidad23432019-12-16 14:22:13 +090018import com.google.common.collect.ImmutableList;
Jian Lib1cd0b02019-02-21 16:41:40 +090019import com.google.common.collect.Lists;
20import io.fabric8.kubernetes.api.model.Endpoints;
Jian Li324d6dc2019-07-10 10:55:15 +090021import io.fabric8.kubernetes.api.model.Namespace;
Jian Lib1cd0b02019-02-21 16:41:40 +090022import io.fabric8.kubernetes.api.model.Pod;
Jian Li1cf51882019-02-26 14:41:20 +090023import io.fabric8.kubernetes.api.model.extensions.Ingress;
Jian Li9ac73952021-01-15 14:53:22 +090024import io.fabric8.kubernetes.api.model.networking.v1.NetworkPolicy;
Jian Lib1cd0b02019-02-21 16:41:40 +090025import io.fabric8.kubernetes.client.KubernetesClient;
Jian Lidad23432019-12-16 14:22:13 +090026import org.apache.commons.lang.StringUtils;
Jian Lib1cd0b02019-02-21 16:41:40 +090027import org.apache.karaf.shell.api.action.Command;
28import org.apache.karaf.shell.api.action.lifecycle.Service;
29import org.onosproject.cli.AbstractShellCommand;
30import org.onosproject.k8snetworking.api.K8sEndpointsAdminService;
Jian Li1cf51882019-02-26 14:41:20 +090031import org.onosproject.k8snetworking.api.K8sIngressAdminService;
Jian Li324d6dc2019-07-10 10:55:15 +090032import org.onosproject.k8snetworking.api.K8sNamespaceAdminService;
Jian Li3d1111e2019-02-22 02:02:13 +090033import org.onosproject.k8snetworking.api.K8sNetworkAdminService;
Jian Lif4523d82019-07-07 01:06:09 +090034import org.onosproject.k8snetworking.api.K8sNetworkPolicyAdminService;
Jian Lib1cd0b02019-02-21 16:41:40 +090035import org.onosproject.k8snetworking.api.K8sPodAdminService;
36import org.onosproject.k8snetworking.api.K8sServiceAdminService;
37import org.onosproject.k8snetworking.util.K8sNetworkingUtil;
38import org.onosproject.k8snode.api.K8sApiConfig;
39import org.onosproject.k8snode.api.K8sApiConfigService;
40
41import java.util.List;
Jian Li3d1111e2019-02-22 02:02:13 +090042
Jian Lidad23432019-12-16 14:22:13 +090043import static org.onosproject.k8snetworking.api.Constants.CLI_CONTAINERS_LENGTH;
44import static org.onosproject.k8snetworking.api.Constants.CLI_IP_ADDRESSES_LENGTH;
45import static org.onosproject.k8snetworking.api.Constants.CLI_IP_ADDRESS_LENGTH;
46import static org.onosproject.k8snetworking.api.Constants.CLI_LABELS_LENGTH;
47import static org.onosproject.k8snetworking.api.Constants.CLI_MARGIN_LENGTH;
48import static org.onosproject.k8snetworking.api.Constants.CLI_NAMESPACE_LENGTH;
49import static org.onosproject.k8snetworking.api.Constants.CLI_NAME_LENGTH;
50import static org.onosproject.k8snetworking.api.Constants.CLI_PHASE_LENGTH;
51import static org.onosproject.k8snetworking.api.Constants.CLI_PORTS_LENGTH;
52import static org.onosproject.k8snetworking.api.Constants.CLI_TYPES_LENGTH;
53import static org.onosproject.k8snetworking.util.K8sNetworkingUtil.genFormatString;
Jian Li4bd6f2b2019-08-16 21:39:27 +090054import static org.onosproject.k8snetworking.util.K8sNetworkingUtil.syncPortFromPod;
Jian Lib1cd0b02019-02-21 16:41:40 +090055
56/**
57 * Synchronizes kubernetes states.
58 */
59@Service
60@Command(scope = "onos", name = "k8s-sync-states",
61 description = "Synchronizes all kubernetes states")
62public class K8sSyncStateCommand extends AbstractShellCommand {
63
Jian Lidad23432019-12-16 14:22:13 +090064 private static final String POD_FORMAT = genFormatString(ImmutableList.of(
65 CLI_NAME_LENGTH, CLI_NAMESPACE_LENGTH, CLI_IP_ADDRESS_LENGTH,
66 CLI_CONTAINERS_LENGTH));
67 private static final String SERVICE_FORMAT = genFormatString(ImmutableList.of(
68 CLI_NAME_LENGTH, CLI_IP_ADDRESS_LENGTH, CLI_PORTS_LENGTH));
69 private static final String ENDPOINTS_FORMAT = genFormatString(ImmutableList.of(
70 CLI_NAME_LENGTH, CLI_IP_ADDRESSES_LENGTH, CLI_PORTS_LENGTH));
71 private static final String INGRESS_FORMAT = genFormatString(ImmutableList.of(
72 CLI_NAME_LENGTH, CLI_NAMESPACE_LENGTH, CLI_IP_ADDRESS_LENGTH));
73 private static final String NETWORK_POLICY_FORMAT = genFormatString(ImmutableList.of(
74 CLI_NAME_LENGTH, CLI_NAMESPACE_LENGTH, CLI_TYPES_LENGTH));
75 private static final String NAMESPACE_FORMAT = genFormatString(ImmutableList.of(
76 CLI_NAME_LENGTH, CLI_PHASE_LENGTH, CLI_LABELS_LENGTH));
Jian Lib1cd0b02019-02-21 16:41:40 +090077
78 @Override
79 protected void doExecute() {
80 K8sApiConfigService configService = get(K8sApiConfigService.class);
81 K8sPodAdminService podAdminService = get(K8sPodAdminService.class);
Jian Li324d6dc2019-07-10 10:55:15 +090082 K8sNamespaceAdminService namespaceAdminService =
83 get(K8sNamespaceAdminService.class);
Jian Lib1cd0b02019-02-21 16:41:40 +090084 K8sServiceAdminService serviceAdminService =
85 get(K8sServiceAdminService.class);
Jian Li1cf51882019-02-26 14:41:20 +090086 K8sIngressAdminService ingressAdminService =
87 get(K8sIngressAdminService.class);
Jian Lib1cd0b02019-02-21 16:41:40 +090088 K8sEndpointsAdminService endpointsAdminService =
89 get(K8sEndpointsAdminService.class);
Jian Li3d1111e2019-02-22 02:02:13 +090090 K8sNetworkAdminService networkAdminService =
91 get(K8sNetworkAdminService.class);
Jian Lif4523d82019-07-07 01:06:09 +090092 K8sNetworkPolicyAdminService networkPolicyAdminService =
93 get(K8sNetworkPolicyAdminService.class);
Jian Lib1cd0b02019-02-21 16:41:40 +090094
95 K8sApiConfig config =
96 configService.apiConfigs().stream().findAny().orElse(null);
97 if (config == null) {
98 log.error("Failed to find valid kubernetes API configuration.");
99 return;
100 }
101
102 KubernetesClient client = K8sNetworkingUtil.k8sClient(config);
103
104 if (client == null) {
105 log.error("Failed to connect to kubernetes API server.");
106 return;
107 }
108
Jian Li324d6dc2019-07-10 10:55:15 +0900109 print("\nSynchronizing kubernetes namespaces");
110 print(NAMESPACE_FORMAT, "Name", "Phase", "Labels");
111 client.namespaces().list().getItems().forEach(ns -> {
112 if (namespaceAdminService.namespace(ns.getMetadata().getUid()) != null) {
113 namespaceAdminService.updateNamespace(ns);
114 } else {
115 namespaceAdminService.createNamespace(ns);
116 }
117 printNamespace(ns);
118 });
119
Jian Lib1cd0b02019-02-21 16:41:40 +0900120 print("Synchronizing kubernetes services");
121 print(SERVICE_FORMAT, "Name", "Cluster IP", "Ports");
Jian Liaf081522019-06-19 20:53:07 +0900122 client.services().inAnyNamespace().list().getItems().forEach(svc -> {
Jian Lib1cd0b02019-02-21 16:41:40 +0900123 if (serviceAdminService.service(svc.getMetadata().getUid()) != null) {
124 serviceAdminService.updateService(svc);
125 } else {
126 serviceAdminService.createService(svc);
127 }
128 printService(svc);
129 });
130
131 print("\nSynchronizing kubernetes endpoints");
132 print(ENDPOINTS_FORMAT, "Name", "IP Addresses", "Ports");
Jian Liaf081522019-06-19 20:53:07 +0900133 client.endpoints().inAnyNamespace().list().getItems().forEach(ep -> {
Jian Lib1cd0b02019-02-21 16:41:40 +0900134 if (endpointsAdminService.endpoints(ep.getMetadata().getUid()) != null) {
135 endpointsAdminService.updateEndpoints(ep);
136 } else {
137 endpointsAdminService.createEndpoints(ep);
138 }
139 printEndpoints(ep);
140 });
141
142 print("\nSynchronizing kubernetes pods");
143 print(POD_FORMAT, "Name", "Namespace", "IP", "Containers");
Jian Liaf081522019-06-19 20:53:07 +0900144 client.pods().inAnyNamespace().list().getItems().forEach(pod -> {
Jian Lib1cd0b02019-02-21 16:41:40 +0900145 if (podAdminService.pod(pod.getMetadata().getUid()) != null) {
146 podAdminService.updatePod(pod);
147 } else {
148 podAdminService.createPod(pod);
149 }
Jian Li3d1111e2019-02-22 02:02:13 +0900150
151 syncPortFromPod(pod, networkAdminService);
152
Jian Lib1cd0b02019-02-21 16:41:40 +0900153 printPod(pod);
154 });
Jian Li1cf51882019-02-26 14:41:20 +0900155
156 print("\nSynchronizing kubernetes ingresses");
157 print(INGRESS_FORMAT, "Name", "Namespace", "LB Addresses");
Jian Liaf081522019-06-19 20:53:07 +0900158 client.extensions().ingresses().inAnyNamespace().list().getItems().forEach(ingress -> {
Jian Li1cf51882019-02-26 14:41:20 +0900159 if (ingressAdminService.ingress(ingress.getMetadata().getUid()) != null) {
160 ingressAdminService.updateIngress(ingress);
161 } else {
162 ingressAdminService.createIngress(ingress);
163 }
164 printIngresses(ingress);
165 });
166
Jian Lif4523d82019-07-07 01:06:09 +0900167 print("\nSynchronizing kubernetes network policies");
168 print(NETWORK_POLICY_FORMAT, "Name", "Namespace", "Types");
169 client.network().networkPolicies().inAnyNamespace().list().getItems().forEach(policy -> {
170 if (networkPolicyAdminService.networkPolicy(policy.getMetadata().getUid()) != null) {
171 networkPolicyAdminService.updateNetworkPolicy(policy);
172 } else {
173 networkPolicyAdminService.createNetworkPolicy(policy);
174 }
175 printNetworkPolicy(policy);
176 });
Jian Li1cf51882019-02-26 14:41:20 +0900177 }
178
179 private void printIngresses(Ingress ingress) {
180
181 List<String> lbIps = Lists.newArrayList();
182
183 ingress.getStatus().getLoadBalancer()
184 .getIngress().forEach(i -> lbIps.add(i.getIp()));
185
186 print(INGRESS_FORMAT,
Jian Lidad23432019-12-16 14:22:13 +0900187 StringUtils.substring(ingress.getMetadata().getName(),
188 0, CLI_NAME_LENGTH - CLI_MARGIN_LENGTH),
189 StringUtils.substring(ingress.getMetadata().getNamespace(),
190 0, CLI_NAMESPACE_LENGTH - CLI_MARGIN_LENGTH),
Jian Li1cf51882019-02-26 14:41:20 +0900191 lbIps.isEmpty() ? "" : lbIps);
Jian Lib1cd0b02019-02-21 16:41:40 +0900192 }
193
194 private void printEndpoints(Endpoints endpoints) {
195 List<String> ips = Lists.newArrayList();
196 List<Integer> ports = Lists.newArrayList();
197
198 endpoints.getSubsets().forEach(e -> {
199 e.getAddresses().forEach(a -> ips.add(a.getIp()));
200 e.getPorts().forEach(p -> ports.add(p.getPort()));
201 });
202
203 print(ENDPOINTS_FORMAT,
Jian Lidad23432019-12-16 14:22:13 +0900204 StringUtils.substring(endpoints.getMetadata().getName(),
205 0, CLI_NAME_LENGTH - CLI_MARGIN_LENGTH),
206 ips.isEmpty() ? "" : StringUtils.substring(ips.toString(),
207 0, CLI_IP_ADDRESSES_LENGTH - CLI_MARGIN_LENGTH),
Jian Lib1cd0b02019-02-21 16:41:40 +0900208 ports.isEmpty() ? "" : ports);
209 }
210
Jian Li324d6dc2019-07-10 10:55:15 +0900211 private void printNamespace(Namespace namespace) {
212 print(NAMESPACE_FORMAT,
Jian Lidad23432019-12-16 14:22:13 +0900213 StringUtils.substring(namespace.getMetadata().getName(),
214 0, CLI_NAME_LENGTH - CLI_MARGIN_LENGTH),
Jian Li324d6dc2019-07-10 10:55:15 +0900215 namespace.getStatus().getPhase(),
216 namespace.getMetadata() != null &&
217 namespace.getMetadata().getLabels() != null &&
218 !namespace.getMetadata().getLabels().isEmpty() ?
Jian Lidad23432019-12-16 14:22:13 +0900219 StringUtils.substring(namespace.getMetadata()
220 .getLabels().toString(), 0,
221 CLI_LABELS_LENGTH - CLI_MARGIN_LENGTH) : "");
Jian Li324d6dc2019-07-10 10:55:15 +0900222 }
223
Jian Lib1cd0b02019-02-21 16:41:40 +0900224 private void printService(io.fabric8.kubernetes.api.model.Service service) {
225
226 List<Integer> ports = Lists.newArrayList();
227
228 service.getSpec().getPorts().forEach(p -> ports.add(p.getPort()));
229
230 print(SERVICE_FORMAT,
Jian Lidad23432019-12-16 14:22:13 +0900231 StringUtils.substring(service.getMetadata().getName(),
232 0, CLI_NAME_LENGTH - CLI_MARGIN_LENGTH),
233 StringUtils.substring(service.getSpec().getClusterIP(),
234 0, CLI_IP_ADDRESS_LENGTH - CLI_MARGIN_LENGTH),
Jian Lib1cd0b02019-02-21 16:41:40 +0900235 ports.isEmpty() ? "" : ports);
236 }
237
238 private void printPod(Pod pod) {
239
240 List<String> containers = Lists.newArrayList();
241
242 pod.getSpec().getContainers().forEach(c -> containers.add(c.getName()));
243
244 print(POD_FORMAT,
Jian Lidad23432019-12-16 14:22:13 +0900245 StringUtils.substring(pod.getMetadata().getName(),
246 0, CLI_NAME_LENGTH - CLI_MARGIN_LENGTH),
247 StringUtils.substring(pod.getMetadata().getNamespace(),
248 0, CLI_NAMESPACE_LENGTH - CLI_MARGIN_LENGTH),
249 StringUtils.substring(pod.getStatus().getPodIP(),
250 0, CLI_IP_ADDRESS_LENGTH - CLI_MARGIN_LENGTH),
Jian Lib1cd0b02019-02-21 16:41:40 +0900251 containers.isEmpty() ? "" : containers);
252 }
Jian Li3d1111e2019-02-22 02:02:13 +0900253
Jian Lif4523d82019-07-07 01:06:09 +0900254 private void printNetworkPolicy(NetworkPolicy policy) {
255 print(NETWORK_POLICY_FORMAT,
Jian Lidad23432019-12-16 14:22:13 +0900256 StringUtils.substring(policy.getMetadata().getName(),
257 0, CLI_NAME_LENGTH - CLI_MARGIN_LENGTH),
258 StringUtils.substring(policy.getMetadata().getNamespace(),
259 0, CLI_NAMESPACE_LENGTH - CLI_MARGIN_LENGTH),
Jian Lif4523d82019-07-07 01:06:09 +0900260 policy.getSpec().getPolicyTypes().isEmpty() ?
261 "" : policy.getSpec().getPolicyTypes());
262 }
Jian Lib1cd0b02019-02-21 16:41:40 +0900263}