2 * Copyright (c) 2015, 2016 Brocade Communications Systems, Inc. and others. All rights reserved.
4 * This program and the accompanying materials are made available under the
5 * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6 * and is available at http://www.eclipse.org/legal/epl-v10.html
8 package org.opendaylight.ovsdb.southbound.ovsdb.transact;
10 import static org.opendaylight.ovsdb.lib.operations.Operations.op;
11 import static org.opendaylight.ovsdb.southbound.SouthboundUtil.schemaMismatchLog;
13 import java.util.Collection;
14 import java.util.HashMap;
15 import java.util.HashSet;
16 import java.util.List;
18 import java.util.Map.Entry;
20 import java.util.concurrent.ExecutionException;
22 import org.opendaylight.controller.md.sal.binding.api.DataTreeModification;
23 import org.opendaylight.controller.md.sal.binding.api.ReadOnlyTransaction;
24 import org.opendaylight.controller.md.sal.common.api.data.AsyncDataChangeEvent;
25 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
26 import org.opendaylight.controller.md.sal.common.api.data.ReadFailedException;
27 import org.opendaylight.ovsdb.lib.error.SchemaVersionMismatchException;
28 import org.opendaylight.ovsdb.lib.notation.UUID;
29 import org.opendaylight.ovsdb.lib.operations.TransactionBuilder;
30 import org.opendaylight.ovsdb.lib.schema.typed.TyperUtils;
31 import org.opendaylight.ovsdb.schema.openvswitch.Interface;
32 import org.opendaylight.ovsdb.schema.openvswitch.Port;
33 import org.opendaylight.ovsdb.southbound.OvsdbConnectionInstance;
34 import org.opendaylight.ovsdb.southbound.SouthboundConstants;
35 import org.opendaylight.ovsdb.southbound.SouthboundProvider;
36 import org.opendaylight.ovsdb.utils.yang.YangUtils;
37 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev100924.Uri;
38 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.yang.types.rev130715.Uuid;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.OvsdbPortInterfaceAttributes.VlanMode;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.OvsdbBridgeAugmentation;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.OvsdbNodeAugmentation;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.OvsdbQosRef;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.OvsdbTerminationPointAugmentation;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.node.attributes.QosEntries;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.node.attributes.Queues;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.InterfaceBfd;
47 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.InterfaceExternalIds;
48 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.InterfaceLldp;
49 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.InterfaceOtherConfigs;
50 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.Options;
51 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.PortExternalIds;
52 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.PortOtherConfigs;
53 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.Trunks;
54 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.Node;
55 import org.opendaylight.yangtools.yang.binding.DataObject;
56 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
57 import org.slf4j.Logger;
58 import org.slf4j.LoggerFactory;
60 import com.google.common.base.Optional;
61 import com.google.common.collect.Sets;
62 import com.google.common.util.concurrent.CheckedFuture;
64 public class TerminationPointUpdateCommand implements TransactCommand {
66 private static final Logger LOG = LoggerFactory.getLogger(TerminationPointUpdateCommand.class);
69 public void execute(TransactionBuilder transaction, BridgeOperationalState state,
70 AsyncDataChangeEvent<InstanceIdentifier<?>, DataObject> events) {
71 execute(transaction, state, TransactUtils.extractCreatedOrUpdated(events, OvsdbTerminationPointAugmentation.class));
75 public void execute(TransactionBuilder transaction, BridgeOperationalState state,
76 Collection<DataTreeModification<Node>> modifications) {
77 execute(transaction, state,
78 TransactUtils.extractCreatedOrUpdated(modifications, OvsdbTerminationPointAugmentation.class));
81 private void execute(TransactionBuilder transaction, BridgeOperationalState state,
82 Map<InstanceIdentifier<OvsdbTerminationPointAugmentation>,
83 OvsdbTerminationPointAugmentation> createdOrUpdated) {
84 LOG.trace("TerminationPointUpdateCommand called");
85 for (Entry<InstanceIdentifier<OvsdbTerminationPointAugmentation>,
86 OvsdbTerminationPointAugmentation> terminationPointEntry : createdOrUpdated.entrySet()) {
87 updateTerminationPoint(transaction, state, terminationPointEntry.getKey(), terminationPointEntry.getValue());
91 public void updateTerminationPoint(TransactionBuilder transaction, BridgeOperationalState state,
92 InstanceIdentifier<OvsdbTerminationPointAugmentation> iid,
93 OvsdbTerminationPointAugmentation terminationPoint) {
94 if (terminationPoint != null) {
95 LOG.debug("Received request to update termination point {}",
96 terminationPoint.getName());
99 Interface ovsInterface =
100 TyperUtils.getTypedRowWrapper(transaction.getDatabaseSchema(), Interface.class);
101 updateInterface(terminationPoint, ovsInterface);
102 Interface extraInterface = TyperUtils.getTypedRowWrapper(
103 transaction.getDatabaseSchema(), Interface.class);
104 extraInterface.setName("");
105 transaction.add(op.update(ovsInterface)
106 .where(extraInterface.getNameColumn().getSchema().opEqual(terminationPoint.getName()))
109 TerminationPointCreateCommand.stampInstanceIdentifier(transaction,
110 iid.firstIdentifierOf(OvsdbTerminationPointAugmentation.class), terminationPoint.getName());
113 OvsdbBridgeAugmentation operBridge = state.getBridgeNode(iid).get().getAugmentation(OvsdbBridgeAugmentation.class);
114 Port port = TyperUtils.getTypedRowWrapper(
115 transaction.getDatabaseSchema(), Port.class);
116 updatePort(terminationPoint, port, operBridge);
117 Port extraPort = TyperUtils.getTypedRowWrapper(
118 transaction.getDatabaseSchema(), Port.class);
119 extraPort.setName("");
120 transaction.add(op.update(port)
121 .where(extraPort.getNameColumn().getSchema().opEqual(terminationPoint.getName()))
126 private void updateInterface(
127 final OvsdbTerminationPointAugmentation terminationPoint,
128 final Interface ovsInterface) {
129 updateOfPort(terminationPoint, ovsInterface);
130 updateOfPortRequest(terminationPoint, ovsInterface);
131 updateInterfaceOptions(terminationPoint, ovsInterface);
132 updateInterfaceOtherConfig(terminationPoint, ovsInterface);
133 updateInterfaceExternalIds(terminationPoint, ovsInterface);
134 updateInterfaceLldp(terminationPoint, ovsInterface);
135 updateInterfaceBfd(terminationPoint, ovsInterface);
136 updateInterfacePolicing(terminationPoint, ovsInterface);
139 private void updatePort(
140 final OvsdbTerminationPointAugmentation terminationPoint,
142 final OvsdbBridgeAugmentation operBridge) {
144 updatePortOtherConfig(terminationPoint, port);
145 updatePortVlanTag(terminationPoint, port);
146 updatePortVlanTrunk(terminationPoint, port);
147 updatePortVlanMode(terminationPoint, port);
148 updatePortExternalIds(terminationPoint, port);
149 updatePortQos(terminationPoint, port, operBridge);
152 private void updatePortQos(
153 final OvsdbTerminationPointAugmentation terminationPoint,
155 final OvsdbBridgeAugmentation operBridge) {
157 Set<UUID> uuidSet = Sets.newHashSet();
159 // First check if QosEntry is present and use that
160 if (terminationPoint.getQosEntry() != null && !terminationPoint.getQosEntry().isEmpty()) {
161 OvsdbQosRef qosRef = terminationPoint.getQosEntry().iterator().next().getQosRef();
162 Uri qosId = qosRef.getValue().firstKeyOf(QosEntries.class).getQosId();
163 OvsdbNodeAugmentation operNode = getOperNode(operBridge);
164 if (operNode != null && operNode.getQosEntries() != null &&
165 !operNode.getQosEntries().isEmpty()) {
166 for (QosEntries qosEntry : operNode.getQosEntries()) {
167 if (qosEntry.getQosId().equals(qosId)) {
168 uuidSet.add(new UUID(qosEntry.getQosUuid().getValue()));
172 if (uuidSet.size() == 0) {
173 uuidSet.add(new UUID(SouthboundConstants.QOS_NAMED_UUID_PREFIX +
174 TransactUtils.bytesToHexString(qosId.getValue().getBytes())));
177 // Second check if Qos is present and use that (deprecated)
178 // Do not bother to check if QosEntry and Qos are consistent if both are present
179 Uuid qosUuid = terminationPoint.getQos();
180 if (qosUuid != null) {
181 uuidSet.add(new UUID(qosUuid.getValue()));
184 port.setQos(uuidSet);
187 private OvsdbNodeAugmentation getOperNode(final OvsdbBridgeAugmentation operBridge) {
188 @SuppressWarnings("unchecked")
189 InstanceIdentifier<Node> iidNode = (InstanceIdentifier<Node>)operBridge.getManagedBy().getValue();
190 OvsdbNodeAugmentation operNode = null;
191 ReadOnlyTransaction transaction = SouthboundProvider.getDb().newReadOnlyTransaction();
192 CheckedFuture<Optional<Node>, ReadFailedException> future =
193 transaction.read(LogicalDatastoreType.OPERATIONAL, iidNode);
195 Optional<Node> nodeOptional = future.get();
196 if (nodeOptional.isPresent()) {
197 operNode = nodeOptional.get().getAugmentation(OvsdbNodeAugmentation.class);
199 } catch (InterruptedException | ExecutionException e) {
200 LOG.warn("Error reading from datastore", e);
205 private void updateOfPort(
206 final OvsdbTerminationPointAugmentation terminationPoint,
207 final Interface ovsInterface) {
209 Long ofPort = terminationPoint.getOfport();
210 if (ofPort != null) {
211 ovsInterface.setOpenFlowPort(Sets.newHashSet(ofPort));
215 private void updateOfPortRequest(
216 final OvsdbTerminationPointAugmentation terminationPoint,
217 final Interface ovsInterface) {
219 Integer ofPortRequest = terminationPoint.getOfportRequest();
220 if (ofPortRequest != null) {
221 ovsInterface.setOpenFlowPortRequest(Sets.newHashSet(ofPortRequest.longValue()));
225 private void updateInterfaceOptions(
226 final OvsdbTerminationPointAugmentation terminationPoint,
227 final Interface ovsInterface) {
229 //Configure optional input
230 if (terminationPoint.getOptions() != null) {
232 ovsInterface.setOptions(YangUtils.convertYangKeyValueListToMap(terminationPoint.getOptions(),
233 Options::getOption, Options::getValue));
234 } catch (NullPointerException e) {
235 LOG.warn("Incomplete OVSDB interface options", e);
240 private void updateInterfaceExternalIds(
241 final OvsdbTerminationPointAugmentation terminationPoint,
242 final Interface ovsInterface) {
244 List<InterfaceExternalIds> interfaceExternalIds =
245 terminationPoint.getInterfaceExternalIds();
246 if (interfaceExternalIds != null && !interfaceExternalIds.isEmpty()) {
248 ovsInterface.setExternalIds(YangUtils.convertYangKeyValueListToMap(interfaceExternalIds,
249 InterfaceExternalIds::getExternalIdKey, InterfaceExternalIds::getExternalIdValue));
250 } catch (NullPointerException e) {
251 LOG.warn("Incomplete OVSDB interface external_ids", e);
256 private void updateInterfaceLldp(
257 final OvsdbTerminationPointAugmentation terminationPoint,
258 final Interface ovsInterface) {
261 List<InterfaceLldp> interfaceLldpList =
262 terminationPoint.getInterfaceLldp();
263 if (interfaceLldpList != null && !interfaceLldpList.isEmpty()) {
265 ovsInterface.setLldp(YangUtils.convertYangKeyValueListToMap(interfaceLldpList,
266 InterfaceLldp::getLldpKey, InterfaceLldp::getLldpValue));
267 } catch (NullPointerException e) {
268 LOG.warn("Incomplete OVSDB interface lldp", e);
271 } catch (SchemaVersionMismatchException e) {
272 schemaMismatchLog("lldp", "Interface", e);
276 private void updateInterfaceOtherConfig(
277 final OvsdbTerminationPointAugmentation terminationPoint,
278 final Interface ovsInterface) {
280 List<InterfaceOtherConfigs> interfaceOtherConfigs =
281 terminationPoint.getInterfaceOtherConfigs();
282 if (interfaceOtherConfigs != null && !interfaceOtherConfigs.isEmpty()) {
283 Map<String, String> otherConfigsMap = new HashMap<>();
284 for (InterfaceOtherConfigs interfaceOtherConfig : interfaceOtherConfigs) {
285 otherConfigsMap.put(interfaceOtherConfig.getOtherConfigKey(),
286 interfaceOtherConfig.getOtherConfigValue());
289 ovsInterface.setOtherConfig(otherConfigsMap);
290 } catch (NullPointerException e) {
291 LOG.warn("Incomplete OVSDB interface other_config", e);
296 private void updateInterfaceBfd(
297 final OvsdbTerminationPointAugmentation terminationPoint,
298 final Interface ovsInterface) {
301 List<InterfaceBfd> interfaceBfdList =
302 terminationPoint.getInterfaceBfd();
303 if (interfaceBfdList != null && !interfaceBfdList.isEmpty()) {
305 ovsInterface.setBfd(YangUtils.convertYangKeyValueListToMap(interfaceBfdList,
306 InterfaceBfd::getBfdKey, InterfaceBfd::getBfdValue));
307 } catch (NullPointerException e) {
308 LOG.warn("Incomplete OVSDB interface bfd", e);
311 } catch (SchemaVersionMismatchException e) {
312 schemaMismatchLog("bfd", "Interface", e);
316 private void updateInterfacePolicing(
317 final OvsdbTerminationPointAugmentation terminationPoint,
318 final Interface ovsInterface) {
320 Long ingressPolicingRate = terminationPoint.getIngressPolicingRate();
321 if (ingressPolicingRate != null) {
322 ovsInterface.setIngressPolicingRate(ingressPolicingRate);
324 Long ingressPolicingBurst = terminationPoint.getIngressPolicingBurst();
325 if (ingressPolicingBurst != null) {
326 ovsInterface.setIngressPolicingBurst(ingressPolicingBurst);
330 private void updatePortExternalIds(
331 final OvsdbTerminationPointAugmentation terminationPoint,
334 List<PortExternalIds> portExternalIds = terminationPoint.getPortExternalIds();
335 if (portExternalIds != null && !portExternalIds.isEmpty()) {
337 port.setExternalIds(YangUtils.convertYangKeyValueListToMap(portExternalIds,
338 PortExternalIds::getExternalIdKey, PortExternalIds::getExternalIdValue));
339 } catch (NullPointerException e) {
340 LOG.warn("Incomplete OVSDB port external_ids", e);
345 private void updatePortVlanTag(
346 final OvsdbTerminationPointAugmentation terminationPoint,
349 if (terminationPoint.getVlanTag() != null) {
350 Set<Long> vlanTag = new HashSet<>();
351 vlanTag.add(terminationPoint.getVlanTag().getValue().longValue());
352 port.setTag(vlanTag);
356 private void updatePortVlanTrunk(
357 final OvsdbTerminationPointAugmentation terminationPoint,
360 if (terminationPoint.getTrunks() != null && terminationPoint.getTrunks().size() > 0) {
361 Set<Long> portTrunks = new HashSet<>();
362 List<Trunks> modelTrunks = terminationPoint.getTrunks();
363 for (Trunks trunk : modelTrunks) {
364 if (trunk.getTrunk() != null) {
365 portTrunks.add(trunk.getTrunk().getValue().longValue());
368 port.setTrunks(portTrunks);
372 private void updatePortVlanMode(
373 final OvsdbTerminationPointAugmentation terminationPoint,
375 if (terminationPoint.getVlanMode() != null) {
376 Set<String> portVlanMode = new HashSet<>();
377 VlanMode modelVlanMode = terminationPoint.getVlanMode();
378 portVlanMode.add(SouthboundConstants.VLANMODES.values()[modelVlanMode.getIntValue() - 1].getMode());
379 port.setVlanMode(portVlanMode);
383 private void updatePortOtherConfig(
384 final OvsdbTerminationPointAugmentation terminationPoint,
385 final Port ovsPort) {
386 List<PortOtherConfigs> portOtherConfigs =
387 terminationPoint.getPortOtherConfigs();
388 if (portOtherConfigs != null && !portOtherConfigs.isEmpty()) {
390 ovsPort.setOtherConfig(YangUtils.convertYangKeyValueListToMap(portOtherConfigs,
391 PortOtherConfigs::getOtherConfigKey, PortOtherConfigs::getOtherConfigValue));
392 } catch (NullPointerException e) {
393 LOG.warn("Incomplete OVSDB port other_config", e);