2 * Copyright (c) 2015 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;
12 import java.util.Collections;
13 import java.util.HashMap;
14 import java.util.HashSet;
15 import java.util.List;
17 import java.util.Map.Entry;
18 import java.util.concurrent.ExecutionException;
21 import org.opendaylight.controller.md.sal.binding.api.ReadOnlyTransaction;
22 import org.opendaylight.controller.md.sal.common.api.data.AsyncDataChangeEvent;
23 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
24 import org.opendaylight.controller.md.sal.common.api.data.ReadFailedException;
25 import org.opendaylight.ovsdb.lib.error.SchemaVersionMismatchException;
26 import org.opendaylight.ovsdb.lib.notation.Mutator;
27 import org.opendaylight.ovsdb.lib.notation.UUID;
28 import org.opendaylight.ovsdb.lib.operations.Mutate;
29 import org.opendaylight.ovsdb.lib.operations.TransactionBuilder;
30 import org.opendaylight.ovsdb.lib.schema.typed.TyperUtils;
31 import org.opendaylight.ovsdb.schema.openvswitch.Bridge;
32 import org.opendaylight.ovsdb.schema.openvswitch.Interface;
33 import org.opendaylight.ovsdb.schema.openvswitch.Port;
34 import org.opendaylight.ovsdb.southbound.SouthboundConstants;
35 import org.opendaylight.ovsdb.southbound.SouthboundMapper;
36 import org.opendaylight.ovsdb.southbound.SouthboundProvider;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.InterfaceTypeBase;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.OvsdbBridgeAugmentation;
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.OvsdbTerminationPointAugmentation;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.InterfaceExternalIds;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.InterfaceLldp;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.InterfaceOtherConfigs;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.Options;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.PortExternalIds;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.PortOtherConfigs;
47 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.ovsdb.rev150105.ovsdb.port._interface.attributes.Trunks;
48 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.Node;
49 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.node.TerminationPoint;
50 import org.opendaylight.yangtools.yang.binding.DataObject;
51 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
52 import org.slf4j.Logger;
53 import org.slf4j.LoggerFactory;
55 import com.google.common.base.Optional;
56 import com.google.common.collect.ImmutableMap;
57 import com.google.common.collect.Sets;
58 import com.google.common.util.concurrent.CheckedFuture;
60 public class TerminationPointCreateCommand extends AbstractTransactCommand {
62 private static final Logger LOG = LoggerFactory.getLogger(TerminationPointCreateCommand.class);
64 public TerminationPointCreateCommand(BridgeOperationalState state,
65 AsyncDataChangeEvent<InstanceIdentifier<?>, DataObject> changes) {
66 super(state, changes);
70 public void execute(TransactionBuilder transaction) {
71 for (Entry<InstanceIdentifier<?>, DataObject> entry: getChanges().getCreatedData().entrySet()) {
72 DataObject dataObject = entry.getValue();
73 if (dataObject instanceof OvsdbTerminationPointAugmentation) {
74 OvsdbTerminationPointAugmentation terminationPoint = (OvsdbTerminationPointAugmentation) dataObject;
75 LOG.debug("Received request to create termination point {}",
76 terminationPoint.getName());
77 InstanceIdentifier terminationPointIid = entry.getKey();
78 Optional<TerminationPoint> terminationPointOptional =
79 getOperationalState().getBridgeTerminationPoint(terminationPointIid);
80 if (!terminationPointOptional.isPresent()) {
81 // Configure interface
82 String interfaceUuid = "Interface_" + SouthboundMapper.getRandomUUID();
83 Interface ovsInterface =
84 TyperUtils.getTypedRowWrapper(transaction.getDatabaseSchema(), Interface.class);
85 createInterface(terminationPoint, ovsInterface);
86 transaction.add(op.insert(ovsInterface).withId(interfaceUuid));
88 stampInstanceIdentifier(transaction, (InstanceIdentifier<TerminationPoint>) entry.getKey(),
89 ovsInterface.getName());
91 // Configure port with the above interface details
92 String portUuid = "Port_" + SouthboundMapper.getRandomUUID();
93 Port port = TyperUtils.getTypedRowWrapper(transaction.getDatabaseSchema(), Port.class);
94 createPort(terminationPoint, port, interfaceUuid);
95 transaction.add(op.insert(port).withId(portUuid));
97 //Configure bridge with the above port details
98 Bridge bridge = TyperUtils.getTypedRowWrapper(transaction.getDatabaseSchema(), Bridge.class);
99 if (getBridge(entry.getKey()) != null) {
100 bridge.setName(getBridge(entry.getKey()).getBridgeName().getValue());
101 bridge.setPorts(Sets.newHashSet(new UUID(portUuid)));
103 transaction.add(op.mutate(bridge)
104 .addMutation(bridge.getPortsColumn().getSchema(),
105 Mutator.INSERT,bridge.getPortsColumn().getData())
106 .where(bridge.getNameColumn().getSchema()
107 .opEqual(bridge.getNameColumn().getData())).build());
115 private void createInterface(
116 final OvsdbTerminationPointAugmentation terminationPoint,
117 final Interface ovsInterface) {
118 ovsInterface.setName(terminationPoint.getName());
120 createInterfaceType(terminationPoint, ovsInterface);
121 createOfPort(terminationPoint, ovsInterface);
122 createOfPortRequest(terminationPoint, ovsInterface);
123 createInterfaceOptions(terminationPoint, ovsInterface);
124 createInterfaceOtherConfig(terminationPoint, ovsInterface);
125 createInterfaceExternalIds(terminationPoint, ovsInterface);
126 createInterfaceLldp(terminationPoint, ovsInterface);
129 private void createInterfaceType(final OvsdbTerminationPointAugmentation terminationPoint,
130 final Interface ovsInterface) {
132 Class<? extends InterfaceTypeBase> mdsaltype = terminationPoint.getInterfaceType();
133 if (mdsaltype != null) {
134 ovsInterface.setType(SouthboundMapper.createOvsdbInterfaceType(mdsaltype));
138 private void createPort(
139 final OvsdbTerminationPointAugmentation terminationPoint,
140 final Port port, final String interfaceUuid) {
142 port.setName(terminationPoint.getName());
143 port.setInterfaces(Sets.newHashSet(new UUID(interfaceUuid)));
144 createPortOtherConfig(terminationPoint, port);
145 createPortVlanTag(terminationPoint, port);
146 createPortVlanTrunk(terminationPoint, port);
147 createPortVlanMode(terminationPoint, port);
148 createPortExternalIds(terminationPoint, port);
151 private void createOfPort(
152 final OvsdbTerminationPointAugmentation terminationPoint,
153 final Interface ovsInterface) {
155 Long ofPort = terminationPoint.getOfport();
156 if (ofPort != null) {
157 ovsInterface.setOpenFlowPort(Sets.newHashSet(ofPort));
161 private void createOfPortRequest(
162 final OvsdbTerminationPointAugmentation terminationPoint,
163 final Interface ovsInterface) {
165 Integer ofPortRequest = terminationPoint.getOfportRequest();
166 if (ofPortRequest != null) {
167 ovsInterface.setOpenFlowPortRequest(Sets.newHashSet(ofPortRequest.longValue()));
171 private void createInterfaceOptions(
172 final OvsdbTerminationPointAugmentation terminationPoint,
173 final Interface ovsInterface) {
175 //Configure optional input
176 if (terminationPoint.getOptions() != null) {
177 Map<String, String> optionsMap = new HashMap<>();
178 for (Options option : terminationPoint.getOptions()) {
179 optionsMap.put(option.getOption(), option.getValue());
182 ovsInterface.setOptions(ImmutableMap.copyOf(optionsMap));
183 } catch (NullPointerException e) {
184 LOG.warn("Incomplete OVSDB interface options", e);
189 private void createInterfaceExternalIds(
190 final OvsdbTerminationPointAugmentation terminationPoint,
191 final Interface ovsInterface) {
193 List<InterfaceExternalIds> interfaceExternalIds =
194 terminationPoint.getInterfaceExternalIds();
195 if (interfaceExternalIds != null && !interfaceExternalIds.isEmpty()) {
196 Map<String, String> externalIdsMap = new HashMap<>();
197 for (InterfaceExternalIds externalId: interfaceExternalIds) {
198 externalIdsMap.put(externalId.getExternalIdKey(), externalId.getExternalIdValue());
201 ovsInterface.setExternalIds(ImmutableMap.copyOf(externalIdsMap));
202 } catch (NullPointerException e) {
203 LOG.warn("Incomplete OVSDB interface external_ids", e);
208 private void createInterfaceOtherConfig(
209 final OvsdbTerminationPointAugmentation terminationPoint,
210 final Interface ovsInterface) {
212 List<InterfaceOtherConfigs> interfaceOtherConfigs =
213 terminationPoint.getInterfaceOtherConfigs();
214 if (interfaceOtherConfigs != null && !interfaceOtherConfigs.isEmpty()) {
215 Map<String, String> otherConfigsMap = new HashMap<>();
216 for (InterfaceOtherConfigs interfaceOtherConfig : interfaceOtherConfigs) {
217 otherConfigsMap.put(interfaceOtherConfig.getOtherConfigKey(),
218 interfaceOtherConfig.getOtherConfigValue());
221 ovsInterface.setOtherConfig(otherConfigsMap);
222 } catch (NullPointerException e) {
223 LOG.warn("Incomplete OVSDB interface other_config", e);
228 private void createInterfaceLldp(
229 final OvsdbTerminationPointAugmentation terminationPoint,
230 final Interface ovsInterface) {
233 List<InterfaceLldp> interfaceLldpList =
234 terminationPoint.getInterfaceLldp();
235 if (interfaceLldpList != null && !interfaceLldpList.isEmpty()) {
236 Map<String, String> interfaceLldpMap = new HashMap<>();
237 for (InterfaceLldp interfaceLldp : interfaceLldpList) {
238 interfaceLldpMap.put(interfaceLldp.getLldpKey(), interfaceLldp.getLldpValue());
241 ovsInterface.setLldp(ImmutableMap.copyOf(interfaceLldpMap));
242 } catch (NullPointerException e) {
243 LOG.warn("Incomplete OVSDB interface lldp", e);
246 } catch (SchemaVersionMismatchException e) {
247 LOG.debug("lldp column for Interface Table unsupported for this version of ovsdb schema", e);
251 private void createPortExternalIds(
252 final OvsdbTerminationPointAugmentation terminationPoint,
255 List<PortExternalIds> portExternalIds = terminationPoint.getPortExternalIds();
256 if (portExternalIds != null && !portExternalIds.isEmpty()) {
257 Map<String, String> externalIdsMap = new HashMap<>();
258 for (PortExternalIds externalId: portExternalIds) {
259 externalIdsMap.put(externalId.getExternalIdKey(), externalId.getExternalIdValue());
262 port.setExternalIds(ImmutableMap.copyOf(externalIdsMap));
263 } catch (NullPointerException e) {
264 LOG.warn("Incomplete OVSDB port external_ids", e);
269 private void createPortVlanTag(
270 final OvsdbTerminationPointAugmentation terminationPoint,
273 if (terminationPoint.getVlanTag() != null) {
274 Set<Long> vlanTag = new HashSet<>();
275 vlanTag.add(terminationPoint.getVlanTag().getValue().longValue());
276 port.setTag(vlanTag);
280 private void createPortVlanTrunk(
281 final OvsdbTerminationPointAugmentation terminationPoint,
284 if (terminationPoint.getTrunks() != null && terminationPoint.getTrunks().size() > 0) {
285 Set<Long> portTrunks = new HashSet<>();
286 List<Trunks> modelTrunks = terminationPoint.getTrunks();
287 for (Trunks trunk: modelTrunks) {
288 if (trunk.getTrunk() != null) {
289 portTrunks.add(trunk.getTrunk().getValue().longValue());
292 port.setTrunks(portTrunks);
296 private void createPortVlanMode(
297 final OvsdbTerminationPointAugmentation terminationPoint,
299 if (terminationPoint.getVlanMode() != null) {
300 Set<String> portVlanMode = new HashSet<>();
301 VlanMode modelVlanMode = terminationPoint.getVlanMode();
302 portVlanMode.add(SouthboundConstants.VLANMODES.values()[modelVlanMode.getIntValue() - 1].getMode());
303 port.setVlanMode(portVlanMode);
307 private void createPortOtherConfig(
308 final OvsdbTerminationPointAugmentation terminationPoint,
309 final Port ovsPort) {
310 List<PortOtherConfigs> portOtherConfigs =
311 terminationPoint.getPortOtherConfigs();
312 if (portOtherConfigs != null && !portOtherConfigs.isEmpty()) {
313 Map<String, String> otherConfigsMap = new HashMap<>();
314 for (PortOtherConfigs portOtherConfig : portOtherConfigs) {
315 otherConfigsMap.put(portOtherConfig.getOtherConfigKey(),
316 portOtherConfig.getOtherConfigValue());
319 ovsPort.setOtherConfig(ImmutableMap.copyOf(otherConfigsMap));
320 } catch (NullPointerException e) {
321 LOG.warn("Incomplete OVSDB port other_config", e);
326 private OvsdbBridgeAugmentation getBridge(InstanceIdentifier<?> key) {
327 OvsdbBridgeAugmentation bridge = null;
328 InstanceIdentifier<Node> nodeIid = key.firstIdentifierOf(Node.class);
329 Map<InstanceIdentifier<Node>, Node> nodes =
330 TransactUtils.extractCreatedOrUpdated(getChanges(),Node.class);
331 if (nodes != null && nodes.get(nodeIid) != null) {
332 Node node = nodes.get(nodeIid);
333 bridge = node.getAugmentation(OvsdbBridgeAugmentation.class);
334 if (bridge == null) {
335 ReadOnlyTransaction transaction = SouthboundProvider.getDb().newReadOnlyTransaction();
336 CheckedFuture<Optional<Node>, ReadFailedException> future =
337 transaction.read(LogicalDatastoreType.OPERATIONAL, nodeIid);
339 Optional<Node> nodeOptional = future.get();
340 if (nodeOptional.isPresent()) {
341 bridge = nodeOptional.get().getAugmentation(OvsdbBridgeAugmentation.class);
343 } catch (InterruptedException | ExecutionException e) {
344 LOG.warn("Error reading from datastore",e);
352 public static void stampInstanceIdentifier(TransactionBuilder transaction,InstanceIdentifier<TerminationPoint> iid,
353 String interfaceName) {
354 Port port = TyperUtils.getTypedRowWrapper(transaction.getDatabaseSchema(), Port.class);
355 port.setName(interfaceName);
356 port.setExternalIds(Collections.<String,String>emptyMap());
357 Mutate mutate = TransactUtils.stampInstanceIdentifierMutation(transaction,
360 port.getExternalIdsColumn().getSchema());
361 transaction.add(mutate
362 .where(port.getNameColumn().getSchema().opEqual(interfaceName))