2 * Copyright (c) 2016 Cisco and/or its affiliates.
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:
8 * http://www.apache.org/licenses/LICENSE-2.0
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.
17 package io.fd.honeycomb.translate.v3po.interfaces.ip;
19 import static org.junit.Assert.assertTrue;
20 import static org.junit.Assert.fail;
21 import static org.mockito.Matchers.any;
22 import static org.mockito.Matchers.argThat;
23 import static org.mockito.Mockito.doNothing;
24 import static org.mockito.Mockito.doReturn;
25 import static org.mockito.Mockito.doThrow;
26 import static org.mockito.Mockito.times;
27 import static org.mockito.Mockito.verify;
28 import static org.mockito.Mockito.when;
30 import com.google.common.base.Optional;
31 import io.fd.honeycomb.translate.v3po.interfaces.ip.subnet.validation.SubnetValidationException;
32 import io.fd.honeycomb.translate.v3po.interfaces.ip.subnet.validation.SubnetValidator;
33 import io.fd.honeycomb.translate.v3po.util.NamingContext;
34 import io.fd.honeycomb.translate.write.WriteFailedException;
35 import io.fd.honeycomb.vpp.test.write.WriterCustomizerTest;
36 import java.util.Arrays;
37 import java.util.Collections;
38 import java.util.List;
39 import org.junit.Before;
40 import org.junit.Test;
41 import org.mockito.ArgumentMatcher;
42 import org.mockito.Mock;
43 import org.mockito.Mockito;
44 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.Ipv4Address;
45 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.Ipv4AddressNoZone;
46 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.Interfaces;
47 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.Interface;
48 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.InterfaceKey;
49 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.Interface1;
50 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.Ipv4;
51 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.Ipv4Builder;
52 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.ipv4.Address;
53 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.ipv4.AddressBuilder;
54 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.ipv4.address.subnet.Netmask;
55 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.ipv4.address.subnet.NetmaskBuilder;
56 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.ipv4.address.subnet.PrefixLength;
57 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.ip.rev140616.interfaces._interface.ipv4.address.subnet.PrefixLengthBuilder;
58 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.yang.types.rev130715.DottedQuad;
59 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
60 import org.openvpp.jvpp.VppBaseCallException;
61 import org.openvpp.jvpp.core.dto.IpAddressDetailsReplyDump;
62 import org.openvpp.jvpp.core.dto.SwInterfaceAddDelAddress;
63 import org.openvpp.jvpp.core.dto.SwInterfaceAddDelAddressReply;
65 public class Ipv4AddressCustomizerTest extends WriterCustomizerTest {
67 private static final String IFC_CTX_NAME = "ifc-test-instance";
68 private static final String IFACE_NAME = "eth0";
69 private static final int IFACE_ID = 123;
72 private SubnetValidator subnetValidator;
74 private NamingContext interfaceContext;
75 private Ipv4AddressCustomizer customizer;
78 public void setUp() throws Exception {
79 interfaceContext = new NamingContext("generatedIfaceName", IFC_CTX_NAME);
81 customizer = new Ipv4AddressCustomizer(api, interfaceContext, subnetValidator);
83 doReturn(future(new IpAddressDetailsReplyDump())).when(api).ipAddressDump(any());
84 when(writeContext.readAfter(Mockito.any()))
85 .thenReturn(Optional.of(new Ipv4Builder().setAddress(Collections.emptyList()).build()));
88 private static InstanceIdentifier<Address> getAddressId(final String ifaceName) {
89 return InstanceIdentifier.builder(Interfaces.class)
90 .child(Interface.class, new InterfaceKey(ifaceName))
91 .augmentation(Interface1.class)
97 private void whenSwInterfaceAddDelAddressThenSuccess() {
98 doReturn(future(new SwInterfaceAddDelAddressReply())).when(api).swInterfaceAddDelAddress(any(SwInterfaceAddDelAddress.class));
101 private void whenSwInterfaceAddDelAddressThenFailure() {
102 doReturn(failedFuture()).when(api).swInterfaceAddDelAddress(any(SwInterfaceAddDelAddress.class));
106 public void testAddPrefixLengthIpv4Address() throws Exception {
107 doNothing().when(subnetValidator).checkNotAddingToSameSubnet(Mockito.anyList());
109 final InstanceIdentifier<Address> id = getAddressId(IFACE_NAME);
110 when(writeContext.readBefore(id)).thenReturn(Optional.absent());
112 Ipv4AddressNoZone noZoneIp = new Ipv4AddressNoZone(new Ipv4Address("192.168.2.1"));
113 PrefixLength length = new PrefixLengthBuilder().setPrefixLength(new Integer(24).shortValue()).build();
114 Address data = new AddressBuilder().setIp(noZoneIp).setSubnet(length).build();
116 defineMapping(mappingContext, IFACE_NAME, IFACE_ID, IFC_CTX_NAME);
117 whenSwInterfaceAddDelAddressThenSuccess();
119 customizer.writeCurrentAttributes(id, data, writeContext);
121 verify(api).swInterfaceAddDelAddress(generateSwInterfaceAddDelAddressRequest(new byte[]{-64, -88, 2, 1},
122 (byte) 1, (byte) 24));
126 public void testAddPrefixLengthIpv4AddressFailed() throws Exception {
127 final InstanceIdentifier<Address> id = getAddressId(IFACE_NAME);
128 when(writeContext.readBefore(id)).thenReturn(Optional.absent());
130 Ipv4AddressNoZone noZoneIp = new Ipv4AddressNoZone(new Ipv4Address("192.168.2.1"));
131 PrefixLength length = new PrefixLengthBuilder().setPrefixLength(new Integer(24).shortValue()).build();
132 Address data = new AddressBuilder().setIp(noZoneIp).setSubnet(length).build();
134 defineMapping(mappingContext, IFACE_NAME, IFACE_ID, IFC_CTX_NAME);
135 whenSwInterfaceAddDelAddressThenFailure();
138 customizer.writeCurrentAttributes(id, data, writeContext);
139 } catch (WriteFailedException e) {
140 assertTrue(e.getCause() instanceof VppBaseCallException);
141 verify(api).swInterfaceAddDelAddress(
142 generateSwInterfaceAddDelAddressRequest(new byte[]{-64, -88, 2, 1},
143 (byte) 1, (byte) 24));
146 fail("WriteFailedException was expected");
150 public void testAddPrefixLengthIpv4AddressConflicted() throws Exception {
152 final InstanceIdentifier<Address> id = getAddressId(IFACE_NAME);
153 when(writeContext.readBefore(id)).thenReturn(Optional.absent());
155 Ipv4AddressNoZone noZoneIp = new Ipv4AddressNoZone(new Ipv4Address("192.168.2.1"));
156 PrefixLength length = new PrefixLengthBuilder().setPrefixLength(new Integer(24).shortValue()).build();
157 Address data = new AddressBuilder().setIp(noZoneIp).setSubnet(length).build();
158 final List<Address> addressList = Arrays.asList(data);
160 //throws when validation invoked
161 doThrow(SubnetValidationException.forConflictingData((short) 24, Arrays.asList(data))).when(subnetValidator)
162 .checkNotAddingToSameSubnet(addressList);
164 //fake data return from WriteContext
165 doReturn(Optional.of(new Ipv4Builder().setAddress(addressList).build())).when(writeContext)
166 .readAfter(argThat(matchInstanceIdentifier(Ipv4.class)));
168 defineMapping(mappingContext, IFACE_NAME, IFACE_ID, IFC_CTX_NAME);
171 customizer.writeCurrentAttributes(id, data, writeContext);
172 } catch (WriteFailedException e) {
173 //verifies if cause of exception is correct type
174 assertTrue(e.getCause() instanceof SubnetValidationException);
176 //verify that validation call was invoked with data from writeContext
177 verify(subnetValidator, times(1)).checkNotAddingToSameSubnet(addressList);
182 private static ArgumentMatcher<InstanceIdentifier<?>> matchInstanceIdentifier(
183 Class<?> desiredClass) {
184 return new ArgumentMatcher<InstanceIdentifier<?>>() {
186 public boolean matches(final Object o) {
187 return o instanceof InstanceIdentifier && ((InstanceIdentifier) o).getTargetType().equals(desiredClass);
192 private SwInterfaceAddDelAddress generateSwInterfaceAddDelAddressRequest(final byte[] address, final byte isAdd,
193 final byte prefixLength) {
194 final SwInterfaceAddDelAddress request = new SwInterfaceAddDelAddress();
195 request.swIfIndex = IFACE_ID;
196 request.isAdd = isAdd;
199 request.addressLength = prefixLength;
200 request.address = address;
205 public void testDeletePrefixLengthIpv4Address() throws Exception {
206 final InstanceIdentifier<Address> id = getAddressId(IFACE_NAME);
208 Ipv4AddressNoZone noZoneIp = new Ipv4AddressNoZone(new Ipv4Address("192.168.2.1"));
209 PrefixLength length = new PrefixLengthBuilder().setPrefixLength(new Integer(24).shortValue()).build();
210 Address data = new AddressBuilder().setIp(noZoneIp).setSubnet(length).build();
212 defineMapping(mappingContext, IFACE_NAME, IFACE_ID, IFC_CTX_NAME);
213 whenSwInterfaceAddDelAddressThenSuccess();
215 customizer.deleteCurrentAttributes(id, data, writeContext);
217 verify(api).swInterfaceAddDelAddress(generateSwInterfaceAddDelAddressRequest(new byte[]{-64, -88, 2, 1},
218 (byte) 0, (byte) 24));
222 public void testDeletePrefixLengthIpv4AddressFailed() throws Exception {
223 final InstanceIdentifier<Address> id = getAddressId(IFACE_NAME);
225 Ipv4AddressNoZone noZoneIp = new Ipv4AddressNoZone(new Ipv4Address("192.168.2.1"));
226 PrefixLength length = new PrefixLengthBuilder().setPrefixLength(new Integer(24).shortValue()).build();
227 Address data = new AddressBuilder().setIp(noZoneIp).setSubnet(length).build();
229 defineMapping(mappingContext, IFACE_NAME, IFACE_ID, IFC_CTX_NAME);
230 whenSwInterfaceAddDelAddressThenFailure();
233 customizer.deleteCurrentAttributes(id, data, writeContext);
234 } catch (WriteFailedException e) {
235 assertTrue(e.getCause() instanceof VppBaseCallException);
236 verify(api).swInterfaceAddDelAddress(
237 generateSwInterfaceAddDelAddressRequest(new byte[]{-64, -88, 2, 1},
238 (byte) 0, (byte) 24));
241 fail("WriteFailedException was expec16ted");
244 private void testSingleNetmask(final int expectedPrefixLength, final String stringMask) throws Exception {
245 final InstanceIdentifier<Address> id = getAddressId(IFACE_NAME);
246 when(writeContext.readBefore(id)).thenReturn(Optional.absent());
248 Ipv4AddressNoZone noZoneIp = new Ipv4AddressNoZone(new Ipv4Address("192.168.2.1"));
249 Netmask subnet = new NetmaskBuilder().setNetmask(new DottedQuad(stringMask)).build();
250 Address data = new AddressBuilder().setIp(noZoneIp).setSubnet(subnet).build();
252 defineMapping(mappingContext, IFACE_NAME, IFACE_ID, IFC_CTX_NAME);
253 whenSwInterfaceAddDelAddressThenSuccess();
255 customizer.writeCurrentAttributes(id, data, writeContext);
257 verify(api).swInterfaceAddDelAddress(generateSwInterfaceAddDelAddressRequest(new byte[]{-64, -88, 2, 1},
258 (byte) 1, (byte) expectedPrefixLength));
261 private void testSingleIllegalNetmask(final String stringMask) throws Exception {
263 final InstanceIdentifier<Address> id = getAddressId(IFACE_NAME);
264 when(writeContext.readBefore(id)).thenReturn(Optional.absent());
266 Ipv4AddressNoZone noZoneIp = new Ipv4AddressNoZone(new Ipv4Address("192.168.2.1"));
267 Netmask subnet = new NetmaskBuilder().setNetmask(new DottedQuad(stringMask)).build();
268 Address data = new AddressBuilder().setIp(noZoneIp).setSubnet(subnet).build();
270 defineMapping(mappingContext, IFACE_NAME, IFACE_ID, IFC_CTX_NAME);
271 whenSwInterfaceAddDelAddressThenSuccess();
273 customizer.writeCurrentAttributes(id, data, writeContext);
274 } catch (IllegalArgumentException e) {
277 fail("IllegalArgumentException expected");
282 * Test contiguous netmask length from QuadDotted notation
285 public void testNetmaskLength() throws Exception {
286 testSingleNetmask(1, "128.0.0.0");
287 testSingleNetmask(2, "192.0.0.0");
288 testSingleNetmask(8, "255.0.0.0");
289 testSingleNetmask(9, "255.128.0.0");
290 testSingleNetmask(16, "255.255.0.0");
291 testSingleNetmask(24, "255.255.255.0");
295 public void testNetmaskIllegal() throws Exception {
296 testSingleIllegalNetmask("");
297 testSingleIllegalNetmask(".");
298 testSingleIllegalNetmask(".255");
299 testSingleIllegalNetmask("255");
300 testSingleIllegalNetmask("255.");
301 testSingleIllegalNetmask("255.255");
302 testSingleIllegalNetmask("255.255.0");
303 testSingleIllegalNetmask("255.255.255.");
304 testSingleIllegalNetmask("255.255.255.256");
305 testSingleIllegalNetmask("0.0.0.0");
306 testSingleIllegalNetmask("10.10.10.10");
307 testSingleIllegalNetmask("255.1.255.0");
308 testSingleIllegalNetmask("255.255.255.255");