blob: bae08bc870cc2456de498f6f9e52fe030b067e2a [file] [log] [blame]
Jian Li16eed162021-01-15 16:58:48 +09001/*
2 * Copyright 2021-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.kubevirtnetworking.impl;
17
18import io.fabric8.kubernetes.client.KubernetesClient;
19import io.fabric8.kubernetes.client.Watcher;
20import io.fabric8.kubernetes.client.WatcherException;
21import io.fabric8.kubernetes.client.dsl.base.CustomResourceDefinitionContext;
Jian Li97e6fc32021-02-01 20:36:45 +090022import org.apache.commons.lang.StringUtils;
23import org.json.JSONArray;
Jian Li16eed162021-01-15 16:58:48 +090024import org.json.JSONException;
25import org.json.JSONObject;
26import org.onlab.packet.IpAddress;
Jian Li97e6fc32021-02-01 20:36:45 +090027import org.onlab.packet.IpPrefix;
Jian Li16eed162021-01-15 16:58:48 +090028import org.onosproject.cluster.ClusterService;
29import org.onosproject.cluster.LeadershipService;
30import org.onosproject.cluster.NodeId;
31import org.onosproject.core.ApplicationId;
32import org.onosproject.core.CoreService;
33import org.onosproject.kubevirtnetworking.api.DefaultKubevirtNetwork;
Jian Li97e6fc32021-02-01 20:36:45 +090034import org.onosproject.kubevirtnetworking.api.KubevirtHostRoute;
Jian Li16eed162021-01-15 16:58:48 +090035import org.onosproject.kubevirtnetworking.api.KubevirtIpPool;
36import org.onosproject.kubevirtnetworking.api.KubevirtNetwork;
37import org.onosproject.kubevirtnetworking.api.KubevirtNetwork.Type;
38import org.onosproject.kubevirtnetworking.api.KubevirtNetworkAdminService;
39import org.onosproject.kubevirtnode.api.KubevirtApiConfigEvent;
40import org.onosproject.kubevirtnode.api.KubevirtApiConfigListener;
41import org.onosproject.kubevirtnode.api.KubevirtApiConfigService;
42import org.onosproject.mastership.MastershipService;
43import org.osgi.service.component.annotations.Activate;
44import org.osgi.service.component.annotations.Component;
45import org.osgi.service.component.annotations.Deactivate;
46import org.osgi.service.component.annotations.Reference;
47import org.osgi.service.component.annotations.ReferenceCardinality;
48import org.slf4j.Logger;
49
50import java.io.IOException;
Jian Li97e6fc32021-02-01 20:36:45 +090051import java.util.HashSet;
Jian Li16eed162021-01-15 16:58:48 +090052import java.util.Objects;
Jian Li97e6fc32021-02-01 20:36:45 +090053import java.util.Set;
Jian Li16eed162021-01-15 16:58:48 +090054import java.util.concurrent.ExecutorService;
55
56import static java.util.concurrent.Executors.newSingleThreadExecutor;
57import static org.onlab.util.Tools.groupedThreads;
58import static org.onosproject.kubevirtnetworking.api.Constants.KUBEVIRT_NETWORKING_APP_ID;
59import static org.onosproject.kubevirtnetworking.api.KubevirtNetwork.Type.FLAT;
60import static org.onosproject.kubevirtnetworking.util.KubevirtNetworkingUtil.k8sClient;
61import static org.slf4j.LoggerFactory.getLogger;
62
63/**
64 * Kubernetes network-attachment-definition watcher used for feeding kubevirt network information.
65 */
66@Component(immediate = true)
67public class NetworkAttachmentDefinitionWatcher {
68
69 private final Logger log = getLogger(getClass());
70
71 private static final String NETWORK_CONFIG = "network-config";
72 private static final String TYPE = "type";
73 private static final String MTU = "mtu";
74 private static final String SEGMENT_ID = "segmentId";
75 private static final String GATEWAY_IP = "gatewayIp";
76 private static final String CIDR = "cidr";
77 private static final String HOST_ROUTES = "hostRoutes";
Jian Li97e6fc32021-02-01 20:36:45 +090078 private static final String DESTINATION = "destination";
79 private static final String NEXTHOP = "nexthop";
Jian Li16eed162021-01-15 16:58:48 +090080 private static final String IP_POOL = "ipPool";
81 private static final String START = "start";
82 private static final String END = "end";
Jian Li97e6fc32021-02-01 20:36:45 +090083 private static final String DNSES = "dnses";
Jian Li16eed162021-01-15 16:58:48 +090084
85 @Reference(cardinality = ReferenceCardinality.MANDATORY)
86 protected CoreService coreService;
87
88 @Reference(cardinality = ReferenceCardinality.MANDATORY)
89 protected MastershipService mastershipService;
90
91 @Reference(cardinality = ReferenceCardinality.MANDATORY)
92 protected ClusterService clusterService;
93
94 @Reference(cardinality = ReferenceCardinality.MANDATORY)
95 protected LeadershipService leadershipService;
96
97 @Reference(cardinality = ReferenceCardinality.MANDATORY)
98 protected KubevirtNetworkAdminService adminService;
99
100 @Reference(cardinality = ReferenceCardinality.MANDATORY)
101 protected KubevirtApiConfigService configService;
102
103 private final ExecutorService eventExecutor = newSingleThreadExecutor(
104 groupedThreads(this.getClass().getSimpleName(), "event-handler"));
105
106 private final InternalNetworkAttachmentDefinitionWatcher
107 watcher = new InternalNetworkAttachmentDefinitionWatcher();
108 private final InternalKubevirtApiConfigListener
109 configListener = new InternalKubevirtApiConfigListener();
110
111 CustomResourceDefinitionContext nadCrdCxt = new CustomResourceDefinitionContext
112 .Builder()
113 .withGroup("k8s.cni.cncf.io")
114 .withScope("Namespaced")
115 .withVersion("v1")
116 .withPlural("network-attachment-definitions")
117 .build();
118
119 private ApplicationId appId;
120 private NodeId localNodeId;
121
122 @Activate
123 protected void activate() {
124 appId = coreService.registerApplication(KUBEVIRT_NETWORKING_APP_ID);
125 localNodeId = clusterService.getLocalNode().id();
126 leadershipService.runForLeadership(appId.name());
127 configService.addListener(configListener);
128
129 log.info("Started");
130 }
131
132 @Deactivate
133 protected void deactivate() {
134 configService.removeListener(configListener);
135 leadershipService.withdraw(appId.name());
136 eventExecutor.shutdown();
137
138 log.info("Stopped");
139 }
140
141 private class InternalKubevirtApiConfigListener implements KubevirtApiConfigListener {
142
143 private boolean isRelevantHelper() {
144 return Objects.equals(localNodeId, leadershipService.getLeader(appId.name()));
145 }
146
147 @Override
148 public void event(KubevirtApiConfigEvent event) {
149
150 switch (event.type()) {
151 case KUBEVIRT_API_CONFIG_UPDATED:
152 eventExecutor.execute(this::processConfigUpdate);
153 break;
154 case KUBEVIRT_API_CONFIG_CREATED:
155 case KUBEVIRT_API_CONFIG_REMOVED:
156 default:
157 // do nothing
158 break;
159 }
160 }
161
162 private void processConfigUpdate() {
163 if (!isRelevantHelper()) {
164 return;
165 }
166
167 KubernetesClient client = k8sClient(configService);
168
169 if (client != null) {
170 try {
171 client.customResource(nadCrdCxt).watch(watcher);
172 } catch (IOException e) {
173 e.printStackTrace();
174 }
175 }
176 }
177 }
178
179 private class InternalNetworkAttachmentDefinitionWatcher
180 implements Watcher<String> {
181
182 @Override
183 public void eventReceived(Action action, String resource) {
184 switch (action) {
185 case ADDED:
186 eventExecutor.execute(() -> processAddition(resource));
187 break;
188 case MODIFIED:
189 eventExecutor.execute(() -> processModification(resource));
190 break;
191 case DELETED:
192 eventExecutor.execute(() -> processDeletion(resource));
193 break;
194 case ERROR:
195 log.warn("Failures processing network-attachment-definition manipulation.");
196 break;
197 default:
198 break;
199 }
200 }
201
202 @Override
203 public void onClose(WatcherException e) {
204 log.warn("Network-attachment-definition watcher OnClose", e);
205 }
206
207 private void processAddition(String resource) {
208 if (!isMaster()) {
209 return;
210 }
211
212 String name = parseName(resource);
213
214 log.trace("Process NetworkAttachmentDefinition {} creating event from API server.",
215 name);
216
217 KubevirtNetwork network = parseKubevirtNetwork(resource);
218 if (network != null) {
Jian Li9ca07f52021-01-19 23:42:55 +0900219 if (adminService.network(network.networkId()) == null) {
220 adminService.createNetwork(network);
221 }
Jian Li16eed162021-01-15 16:58:48 +0900222 }
223 }
224
225 private void processModification(String resource) {
226 if (!isMaster()) {
227 return;
228 }
229
230 String name = parseName(resource);
231
232 log.trace("Process NetworkAttachmentDefinition {} updating event from API server.",
233 name);
234
235 KubevirtNetwork network = parseKubevirtNetwork(resource);
236 if (network != null) {
237 adminService.updateNetwork(network);
238 }
239 }
240
241 private void processDeletion(String resource) {
242 if (!isMaster()) {
243 return;
244 }
245
246 String name = parseName(resource);
247
248 log.trace("Process NetworkAttachmentDefinition {} removal event from API server.",
249 name);
250
251 adminService.removeNetwork(name);
252 }
253
254 private boolean isMaster() {
255 return Objects.equals(localNodeId, leadershipService.getLeader(appId.name()));
256 }
257
258 private String parseName(String resource) {
259 try {
260 JSONObject json = new JSONObject(resource);
261 return json.getJSONObject("metadata").getString("name");
262 } catch (JSONException e) {
263 log.error("");
264 }
265 return "";
266 }
267
268 private KubevirtNetwork parseKubevirtNetwork(String resource) {
269 try {
270 JSONObject json = new JSONObject(resource);
271 String name = parseName(resource);
272 JSONObject annots = json.getJSONObject("metadata").getJSONObject("annotations");
273 String networkConfig = annots.getString(NETWORK_CONFIG);
274 if (networkConfig != null) {
275 KubevirtNetwork.Builder builder = DefaultKubevirtNetwork.builder();
276
277 JSONObject configJson = new JSONObject(networkConfig);
278 String type = configJson.getString(TYPE);
279 Integer mtu = configJson.getInt(MTU);
280 String gatewayIp = configJson.getString(GATEWAY_IP);
281
282 if (!type.equalsIgnoreCase(FLAT.name())) {
283 builder.segmentId(configJson.getString(SEGMENT_ID));
284 }
285
286 String cidr = configJson.getString(CIDR);
287
288 JSONObject poolJson = configJson.getJSONObject(IP_POOL);
289 if (poolJson != null) {
290 String start = poolJson.getString(START);
291 String end = poolJson.getString(END);
292 builder.ipPool(new KubevirtIpPool(
293 IpAddress.valueOf(start), IpAddress.valueOf(end)));
294 }
295
Jian Li2417ab72021-02-02 17:35:12 +0900296 if (configJson.has(HOST_ROUTES)) {
297 JSONArray routesJson = configJson.getJSONArray(HOST_ROUTES);
298 Set<KubevirtHostRoute> hostRoutes = new HashSet<>();
299 if (routesJson != null) {
300 for (int i = 0; i < routesJson.length(); i++) {
301 JSONObject route = routesJson.getJSONObject(i);
302 String destinationStr = route.getString(DESTINATION);
303 String nexthopStr = route.getString(NEXTHOP);
Jian Li97e6fc32021-02-01 20:36:45 +0900304
Jian Li2417ab72021-02-02 17:35:12 +0900305 if (StringUtils.isNotEmpty(destinationStr) &&
306 StringUtils.isNotEmpty(nexthopStr)) {
307 hostRoutes.add(new KubevirtHostRoute(
308 IpPrefix.valueOf(destinationStr),
309 IpAddress.valueOf(nexthopStr)));
310 }
Jian Li97e6fc32021-02-01 20:36:45 +0900311 }
312 }
Jian Li2417ab72021-02-02 17:35:12 +0900313 builder.hostRoutes(hostRoutes);
Jian Li97e6fc32021-02-01 20:36:45 +0900314 }
Jian Li97e6fc32021-02-01 20:36:45 +0900315
Jian Li2417ab72021-02-02 17:35:12 +0900316 if (configJson.has(DNSES)) {
317 JSONArray dnsesJson = configJson.getJSONArray(DNSES);
318 Set<IpAddress> dnses = new HashSet<>();
319 if (dnsesJson != null) {
320 for (int i = 0; i < dnsesJson.length(); i++) {
321 String dns = dnsesJson.getString(i);
322 if (StringUtils.isNotEmpty(dns)) {
323 dnses.add(IpAddress.valueOf(dns));
324 }
325 }
Jian Li97e6fc32021-02-01 20:36:45 +0900326 }
Jian Li2417ab72021-02-02 17:35:12 +0900327 builder.dnses(dnses);
Jian Li97e6fc32021-02-01 20:36:45 +0900328 }
Jian Li97e6fc32021-02-01 20:36:45 +0900329
Jian Li16eed162021-01-15 16:58:48 +0900330 builder.networkId(name).name(name).type(Type.valueOf(type))
331 .mtu(mtu).gatewayIp(IpAddress.valueOf(gatewayIp)).cidr(cidr);
332
333 return builder.build();
334 }
335 } catch (JSONException e) {
336 log.error("Failed to parse network attachment definition object");
337 }
338
339 return null;
340 }
341 }
342}