Map identities to interfaces, not abstract classes
[mdsal.git] / binding / mdsal-binding-java-api-generator / src / test / java / org / opendaylight / mdsal / binding / java / api / generator / test / CompilationTest.java
1 /*
2  * Copyright (c) 2016 Cisco Systems, Inc. and others.  All rights reserved.
3  *
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
7  */
8 package org.opendaylight.mdsal.binding.java.api.generator.test;
9
10 import static org.junit.Assert.assertEquals;
11 import static org.junit.Assert.assertFalse;
12 import static org.junit.Assert.assertTrue;
13
14 import com.google.common.collect.Collections2;
15 import com.google.common.collect.ImmutableList;
16 import com.google.common.collect.ImmutableSet;
17 import com.google.common.collect.Range;
18 import java.io.File;
19 import java.io.IOException;
20 import java.lang.annotation.Annotation;
21 import java.lang.reflect.Field;
22 import java.lang.reflect.Method;
23 import java.lang.reflect.Modifier;
24 import java.lang.reflect.ParameterizedType;
25 import java.lang.reflect.WildcardType;
26 import java.math.BigDecimal;
27 import java.math.BigInteger;
28 import java.net.URISyntaxException;
29 import java.net.URL;
30 import java.net.URLClassLoader;
31 import java.util.ArrayList;
32 import java.util.Arrays;
33 import java.util.Collection;
34 import java.util.Collections;
35 import java.util.List;
36 import org.junit.Test;
37 import org.opendaylight.mdsal.binding.java.api.generator.GeneratorJavaFile;
38 import org.opendaylight.mdsal.binding.model.api.Type;
39 import org.opendaylight.yangtools.yang.binding.ChildOf;
40 import org.opendaylight.yangtools.yang.binding.annotations.RoutingContext;
41 import org.opendaylight.yangtools.yang.model.api.SchemaContext;
42 import org.opendaylight.yangtools.yang.test.util.YangParserTestUtils;
43
44 /**
45  * Test correct code generation.
46  *
47  */
48 public class CompilationTest extends BaseCompilationTest {
49
50     /*
51      * Java 8 allows JaCoCo to hook onto interfaces, as well as
52      * generating a default implementation. We only want to check
53      * abstract methods.
54      */
55     private static Collection<Method> abstractMethods(final Class<?> clazz) {
56         // Filter out
57         return Collections2.filter(Arrays.asList(clazz.getDeclaredMethods()), input -> Modifier.isAbstract(input.getModifiers()));
58     }
59
60     @Test
61     public void testListGeneration() throws Exception {
62         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("list-gen");
63         final File compiledOutputDir = CompilationTestUtils.compilerOutput("list-gen");
64         generateTestSources("/compilation/list-gen", sourcesOutputDir);
65
66         // Test if all sources are generated
67         File parent = new File(sourcesOutputDir, CompilationTestUtils.NS_TEST);
68         final File keyArgs = new File(parent, "KeyArgs.java");
69         final File links = new File(parent, "Links.java");
70         final File linksBuilder = new File(parent, "LinksBuilder.java");
71         final File linksKey = new File(parent, "LinksKey.java");
72         final File testData = new File(parent, "TestData.java");
73         assertTrue(keyArgs.exists());
74         assertTrue(links.exists());
75         assertTrue(linksBuilder.exists());
76         assertTrue(linksKey.exists());
77         assertTrue(testData.exists());
78         CompilationTestUtils.assertFilesCount(parent, 6);
79
80         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_TEST + CompilationTestUtils.FS + "links");
81         final File level = new File(parent, "Level.java");
82         final File linkGroup = new File(parent, "LinkGroup.java");
83         final File node = new File(parent, "Node.java");
84         final File nodeBuilder = new File(parent, "NodeBuilder.java");
85         final File nodeList = new File(parent, "NodeList.java");
86         final File nodeListBuilder = new File(parent, "NodeListBuilder.java");
87         final File nodesType = new File(parent, "NodesType.java");
88         assertTrue(level.exists());
89         assertTrue(linkGroup.exists());
90         assertTrue(node.exists());
91         assertTrue(nodeBuilder.exists());
92         assertTrue(nodeList.exists());
93         assertTrue(nodeListBuilder.exists());
94         assertTrue(nodesType.exists());
95         CompilationTestUtils.assertFilesCount(parent, 7);
96
97         // Test if sources are compilable
98         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
99
100         final ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
101         final Class<?> keyArgsClass = Class.forName(CompilationTestUtils.BASE_PKG + ".urn.opendaylight.test.rev131008.KeyArgs", true, loader);
102         final Class<?> linksClass = Class.forName(CompilationTestUtils.BASE_PKG + ".urn.opendaylight.test.rev131008.Links", true, loader);
103         final Class<?> linksKeyClass = Class.forName(CompilationTestUtils.BASE_PKG + ".urn.opendaylight.test.rev131008.LinksKey", true, loader);
104
105         // Test generated 'grouping key-args'
106         assertTrue(keyArgsClass.isInterface());
107         CompilationTestUtils.assertContainsMethod(keyArgsClass, String.class, "getName");
108         CompilationTestUtils.assertContainsMethod(keyArgsClass, Integer.class, "getSize");
109         assertEquals(2, abstractMethods(keyArgsClass).size());
110
111         // Test generated 'list links'
112         assertTrue(linksClass.isInterface());
113         CompilationTestUtils.assertImplementsIfc(linksClass, keyArgsClass);
114         // TODO: anyxml
115         assertEquals(6, abstractMethods(linksClass).size());
116
117         // Test list key constructor arguments ordering
118         CompilationTestUtils.assertContainsConstructor(linksKeyClass, Byte.class, String.class, Integer.class);
119         // Test serialVersionUID generation
120         final Field suid = CompilationTestUtils.assertContainsField(linksKeyClass, "serialVersionUID", Long.TYPE);
121         suid.setAccessible(true);
122         assertEquals(-8829501012356283881L, suid.getLong(null));
123
124         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
125     }
126
127     @Test
128     public void testAugmentUnderUsesGeneration() throws Exception {
129         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("augment-under-uses");
130         final File compiledOutputDir = CompilationTestUtils.compilerOutput("augment-under-uses");
131         generateTestSources("/compilation/augment-under-uses", sourcesOutputDir);
132
133         // Test if all sources were generated from 'module foo'
134         File parent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO);
135         assertTrue(new File(parent, "Object.java").exists());
136         assertTrue(new File(parent, "ClosedObject.java").exists());
137         assertTrue(new File(parent, "OpenObject.java").exists());
138         assertTrue(new File(parent, "ExplicitRouteObject.java").exists());
139         assertTrue(new File(parent, "PathKeySubobject.java").exists());
140         CompilationTestUtils.assertFilesCount(parent, 9);
141
142         parent = new File(parent, "object");
143         assertTrue(new File(parent, "Nodes.java").exists());
144         assertTrue(new File(parent, "NodesBuilder.java").exists());
145         CompilationTestUtils.assertFilesCount(parent, 2);
146
147         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO + CompilationTestUtils.FS + "closed");
148         CompilationTestUtils.assertFilesCount(parent, 1);
149
150         parent = new File(parent, "object");
151         assertTrue(new File(parent, "Link1.java").exists());
152         assertTrue(new File(parent, "Link1Builder.java").exists());
153         CompilationTestUtils.assertFilesCount(parent, 2);
154
155         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO + CompilationTestUtils.FS + "open");
156         CompilationTestUtils.assertFilesCount(parent, 1);
157
158         parent = new File(parent, "object");
159         assertTrue(new File(parent, "Nodes1.java").exists());
160         assertTrue(new File(parent, "Nodes1Builder.java").exists());
161         CompilationTestUtils.assertFilesCount(parent, 3);
162
163         parent = new File(parent, "nodes");
164         assertTrue(new File(parent, "Links.java").exists());
165         assertTrue(new File(parent, "LinksBuilder.java").exists());
166         CompilationTestUtils.assertFilesCount(parent, 2);
167
168         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO + CompilationTestUtils.FS + "explicit");
169         CompilationTestUtils.assertFilesCount(parent, 1);
170         parent = new File(parent, "route");
171         CompilationTestUtils.assertFilesCount(parent, 1);
172         parent = new File(parent, "object");
173         assertTrue(new File(parent, "Subobjects.java").exists());
174         assertTrue(new File(parent, "SubobjectsBuilder.java").exists());
175         CompilationTestUtils.assertFilesCount(parent, 3);
176
177         parent = new File(parent, "subobjects");
178         CompilationTestUtils.assertFilesCount(parent, 1);
179         parent = new File(parent, "subobject");
180         CompilationTestUtils.assertFilesCount(parent, 1);
181         parent = new File(parent, "type");
182         assertTrue(new File(parent, "PathKey.java").exists());
183         assertTrue(new File(parent, "PathKeyBuilder.java").exists());
184         CompilationTestUtils.assertFilesCount(parent, 3);
185
186         parent = new File(parent, "path");
187         CompilationTestUtils.assertFilesCount(parent, 1);
188         parent = new File(parent, "key");
189         assertTrue(new File(parent, "PathKey.java").exists());
190         assertTrue(new File(parent, "PathKeyBuilder.java").exists());
191         CompilationTestUtils.assertFilesCount(parent, 2);
192
193         // Test if all sources were generated from 'module bar'
194         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_BAR);
195         assertTrue(new File(parent, "BasicExplicitRouteSubobjects.java").exists());
196         assertTrue(new File(parent, "ExplicitRouteSubobjects.java").exists());
197         assertTrue(new File(parent, "RouteSubobjects.java").exists());
198         CompilationTestUtils.assertFilesCount(parent, 5);
199
200         parent = new File(parent, "route");
201         CompilationTestUtils.assertFilesCount(parent, 1);
202         parent = new File(new File(sourcesOutputDir, CompilationTestUtils.NS_BAR), "basic");
203         CompilationTestUtils.assertFilesCount(parent, 1);
204         parent = new File(parent, "explicit");
205         CompilationTestUtils.assertFilesCount(parent, 1);
206         parent = new File(parent, "route");
207         CompilationTestUtils.assertFilesCount(parent, 1);
208
209         parent = new File(parent, "subobjects");
210         CompilationTestUtils.assertFilesCount(parent, 2);
211         assertTrue(new File(parent, "SubobjectType.java").exists());
212
213         parent = new File(parent, "subobject");
214         CompilationTestUtils.assertFilesCount(parent, 1);
215
216         parent = new File(parent, "type");
217         assertTrue(new File(parent, "IpPrefix.java").exists());
218         assertTrue(new File(parent, "IpPrefixBuilder.java").exists());
219         assertTrue(new File(parent, "Label.java").exists());
220         assertTrue(new File(parent, "LabelBuilder.java").exists());
221         CompilationTestUtils.assertFilesCount(parent, 4);
222
223         // Test if sources are compilable
224         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
225
226         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
227     }
228
229     @Test
230     public void testAugmentOfAugmentGeneration() throws Exception {
231         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("aug-of-aug");
232         final File compiledOutputDir = CompilationTestUtils.compilerOutput("aug-of-aug");
233         generateTestSources("/compilation/augment-of-augment", sourcesOutputDir);
234
235         // Test if all sources were generated from 'module foo'
236         File parent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO);
237         final File fooListener = new File(parent, "FooListener.java");
238         File pathAttributes = new File(parent, "PathAttributes.java");
239         final File update = new File(parent, "Update.java");
240         final File updateBuilder = new File(parent, "UpdateBuilder.java");
241         assertTrue(fooListener.exists());
242         assertTrue(pathAttributes.exists());
243         assertTrue(update.exists());
244         assertTrue(updateBuilder.exists());
245         CompilationTestUtils.assertFilesCount(parent, 6);
246
247         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO + CompilationTestUtils.FS + "path");
248         CompilationTestUtils.assertFilesCount(parent, 1);
249         parent = new File(parent, "attributes");
250         CompilationTestUtils.assertFilesCount(parent, 2);
251         final File origin = new File(parent, "Origin.java");
252         final File originBuilder = new File(parent, "OriginBuilder.java");
253         assertTrue(origin.exists());
254         assertTrue(originBuilder.exists());
255
256         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO + CompilationTestUtils.FS + "update");
257         CompilationTestUtils.assertFilesCount(parent, 2);
258         pathAttributes = new File(parent, "PathAttributes.java");
259         final File pathAttributesBuilder = new File(parent, "PathAttributesBuilder.java");
260         assertTrue(pathAttributes.exists());
261         assertTrue(pathAttributesBuilder.exists());
262
263         // Test if all sources were generated from 'module bar'
264         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_BAR);
265         final File destination = new File(parent, "Destination.java");
266         final File pathAttributes1 = new File(parent, "PathAttributes1.java");
267         final File pathAttributes1Builder = new File(parent, "PathAttributes1Builder.java");
268         assertTrue(destination.exists());
269         assertTrue(pathAttributes1.exists());
270         assertTrue(pathAttributes1Builder.exists());
271         CompilationTestUtils.assertFilesCount(parent, 5);
272
273         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_BAR + CompilationTestUtils.FS + "destination");
274         CompilationTestUtils.assertFilesCount(parent, 2);
275         final File destinationType = new File(parent, "DestinationType.java");
276         assertTrue(destinationType.exists());
277
278         parent = new File(parent, "destination");
279         CompilationTestUtils.assertFilesCount(parent, 1);
280         parent = new File(parent, "type");
281         CompilationTestUtils.assertFilesCount(parent, 2);
282         final File destinationIpv4 = new File(parent, "DestinationIp.java");
283         final File destinationIpv4Builder = new File(parent, "DestinationIpBuilder.java");
284         assertTrue(destinationIpv4.exists());
285         assertTrue(destinationIpv4Builder.exists());
286
287         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_BAR + CompilationTestUtils.FS + "update");
288         CompilationTestUtils.assertFilesCount(parent, 1);
289         parent = new File(parent, "path");
290         CompilationTestUtils.assertFilesCount(parent, 1);
291         parent = new File(parent, "attributes");
292         final File mpUnreachNlri = new File(parent, "MpUnreachNlri.java");
293         final File mpUnreachNlriBuilder = new File(parent, "MpUnreachNlriBuilder.java");
294         assertTrue(mpUnreachNlri.exists());
295         assertTrue(mpUnreachNlriBuilder.exists());
296         CompilationTestUtils.assertFilesCount(parent, 3);
297
298         parent = new File(parent, "mp");
299         CompilationTestUtils.assertFilesCount(parent, 1);
300         parent = new File(parent, "unreach");
301         CompilationTestUtils.assertFilesCount(parent, 1);
302         parent = new File(parent, "nlri");
303         final File withdrawnRoutes = new File(parent, "WithdrawnRoutes.java");
304         final File withdrawnRoutesBuilder = new File(parent, "WithdrawnRoutesBuilder.java");
305         assertTrue(withdrawnRoutes.exists());
306         assertTrue(withdrawnRoutesBuilder.exists());
307         CompilationTestUtils.assertFilesCount(parent, 2);
308
309         // Test if all sources were generated from 'module baz'
310         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_BAZ);
311         CompilationTestUtils.assertFilesCount(parent, 2);
312         final File linkstateDestination = new File(parent, "LinkstateDestination.java");
313         assertTrue(linkstateDestination.exists());
314
315         parent = new File(sourcesOutputDir, CompilationTestUtils.NS_BAZ + CompilationTestUtils.FS + "update");
316         CompilationTestUtils.assertFilesCount(parent, 1);
317         parent = new File(parent, "path");
318         CompilationTestUtils.assertFilesCount(parent, 1);
319         parent = new File(parent, "attributes");
320         CompilationTestUtils.assertFilesCount(parent, 1);
321         parent = new File(parent, "mp");
322         CompilationTestUtils.assertFilesCount(parent, 1);
323         parent = new File(parent, "unreach");
324         CompilationTestUtils.assertFilesCount(parent, 1);
325         parent = new File(parent, "nlri");
326         CompilationTestUtils.assertFilesCount(parent, 1);
327         parent = new File(parent, "withdrawn");
328         CompilationTestUtils.assertFilesCount(parent, 1);
329         parent = new File(parent, "routes");
330         CompilationTestUtils.assertFilesCount(parent, 1);
331         parent = new File(parent, "destination");
332         CompilationTestUtils.assertFilesCount(parent, 1);
333         parent = new File(parent, "type");
334         final File destinationLinkstate = new File(parent, "DestinationLinkstate.java");
335         final File destinationLinkstateBuilder = new File(parent, "DestinationLinkstateBuilder.java");
336         assertTrue(destinationLinkstate.exists());
337         assertTrue(destinationLinkstateBuilder.exists());
338         CompilationTestUtils.assertFilesCount(parent, 3);
339         parent = new File(parent, "destination");
340         CompilationTestUtils.assertFilesCount(parent, 1);
341         parent = new File(parent, "linkstate");
342         final File links = new File(parent, "Links.java");
343         final File linksBuilder = new File(parent, "LinksBuilder.java");
344         assertTrue(links.exists());
345         assertTrue(linksBuilder.exists());
346         CompilationTestUtils.assertFilesCount(parent, 3);
347         parent = new File(parent, "links");
348         final File source = new File(parent, "Source.java");
349         final File sourceBuilder = new File(parent, "SourceBuilder.java");
350         assertTrue(source.exists());
351         assertTrue(sourceBuilder.exists());
352         CompilationTestUtils.assertFilesCount(parent, 3);
353         parent = new File(parent, "source");
354         final File address = new File(parent, "Address.java");
355         final File addressBuilder = new File(parent, "AddressBuilder.java");
356         assertTrue(address.exists());
357         assertTrue(addressBuilder.exists());
358         CompilationTestUtils.assertFilesCount(parent, 2);
359
360         // Test if sources are compilable
361         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
362
363         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
364     }
365
366     @Test
367     public void testLeafReturnTypes() throws Exception {
368         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("leaf-return-types");
369         final File compiledOutputDir = CompilationTestUtils.compilerOutput("leaf-return-types");
370         generateTestSources("/compilation/leaf-return-types", sourcesOutputDir);
371
372         final File parent = new File(sourcesOutputDir, CompilationTestUtils.NS_TEST);
373         assertTrue(new File(parent, "TestData.java").exists());
374         assertTrue(new File(parent, "Nodes.java").exists());
375         assertTrue(new File(parent, "NodesBuilder.java").exists());
376         assertTrue(new File(parent, "Alg.java").exists());
377         assertTrue(new File(parent, "NodesIdUnionBuilder.java").exists());
378         CompilationTestUtils.assertFilesCount(parent, 5);
379
380         // Test if sources are compilable
381         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
382
383         final String pkg = CompilationTestUtils.BASE_PKG + ".urn.opendaylight.test.rev131008";
384         final ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
385         final Class<?> nodesClass = Class.forName(pkg + ".Nodes", true, loader);
386         final Class<?> builderClass = Class.forName(pkg + ".NodesBuilder", true, loader);
387
388         // Test methods return type
389         final byte[] b = new byte[] {};
390         CompilationTestUtils.assertContainsMethod(nodesClass, b.getClass(), "getIdBinary");
391         CompilationTestUtils.assertContainsMethod(nodesClass, pkg + ".Nodes$IdBits", "getIdBits", loader);
392         CompilationTestUtils.assertContainsMethod(nodesClass, Boolean.class, "isIdBoolean");
393         CompilationTestUtils.assertContainsMethod(nodesClass, BigDecimal.class, "getIdDecimal64");
394         CompilationTestUtils.assertContainsMethod(nodesClass, Boolean.class, "isIdEmpty");
395         CompilationTestUtils.assertContainsMethod(nodesClass, pkg + ".Nodes$IdEnumeration", "getIdEnumeration", loader);
396         testReturnTypeIdentityref(nodesClass, "getIdIdentityref", pkg + ".Alg");
397         testReturnTypeInstanceIdentitifer(loader, nodesClass, "getIdInstanceIdentifier");
398         CompilationTestUtils.assertContainsMethod(nodesClass, Byte.class, "getId8");
399         CompilationTestUtils.assertContainsMethod(nodesClass, Short.class, "getId16");
400         CompilationTestUtils.assertContainsMethod(nodesClass, Integer.class, "getId32");
401         CompilationTestUtils.assertContainsMethod(nodesClass, Long.class, "getId64");
402         CompilationTestUtils.assertContainsMethod(nodesClass, Long.class, "getIdLeafref");
403         CompilationTestUtils.assertContainsMethod(nodesClass, String.class, "getIdString");
404         CompilationTestUtils.assertContainsMethod(nodesClass, Short.class, "getIdU8");
405         CompilationTestUtils.assertContainsMethod(nodesClass, Integer.class, "getIdU16");
406         CompilationTestUtils.assertContainsMethod(nodesClass, Long.class, "getIdU32");
407         CompilationTestUtils.assertContainsMethod(nodesClass, BigInteger.class, "getIdU64");
408         CompilationTestUtils.assertContainsMethod(nodesClass, pkg + ".Nodes$IdUnion", "getIdUnion", loader);
409
410         final Object builderObj = builderClass.newInstance();
411
412         Method m = CompilationTestUtils.assertContainsMethod(builderClass, builderClass, "setIdBinary", b.getClass());
413         final List<Range<Integer>> lengthConstraints = new ArrayList<>();
414         lengthConstraints.add(Range.closed(1, 10));
415         byte[] arg = new byte[] {};
416         String expectedMsg = String.format("Invalid length: %s, expected: %s.", Arrays.toString(arg), lengthConstraints);
417         CompilationTestUtils.assertContainsRestrictionCheck(builderObj, m, expectedMsg, arg);
418
419         m = CompilationTestUtils.assertContainsMethod(builderClass, builderClass, "setIdDecimal64", BigDecimal.class);
420         final List<Range<BigDecimal>> rangeConstraints = new ArrayList<>();
421         rangeConstraints.add(Range.closed(new BigDecimal("1.5"), new BigDecimal("5.5")));
422         Object arg1 = new BigDecimal("1.4");
423         expectedMsg = String.format("Invalid range: %s, expected: %s.", arg1, rangeConstraints);
424         CompilationTestUtils.assertContainsRestrictionCheck(builderObj, m, expectedMsg, arg1);
425
426         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
427     }
428
429     @Test
430     public void testGenerationContextReferenceExtension() throws Exception {
431         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("context-reference");
432         final File compiledOutputDir = CompilationTestUtils.compilerOutput("context-reference");
433         generateTestSources("/compilation/context-reference", sourcesOutputDir);
434
435         // Test if all sources are generated
436         final File fooParent = new File(sourcesOutputDir, CompilationTestUtils.NS_FOO);
437         CompilationTestUtils.assertFilesCount(fooParent, 3);
438         assertTrue(new File(fooParent, "FooData.java").exists());
439         assertTrue(new File(fooParent, "Nodes.java").exists());
440         assertTrue(new File(fooParent, "NodesBuilder.java").exists());
441
442         final File barParent = new File(sourcesOutputDir, CompilationTestUtils.NS_BAR);
443         CompilationTestUtils.assertFilesCount(barParent, 1);
444         assertTrue(new File(barParent, "IdentityClass.java").exists());
445
446         // Test if sources are compilable
447         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
448
449         final ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
450         final Class<?> nodesClass = Class.forName(CompilationTestUtils.BASE_PKG + ".urn.opendaylight.foo.rev131008.Nodes", true, loader);
451         final Class<?> identityClass = Class
452                 .forName(CompilationTestUtils.BASE_PKG + ".urn.opendaylight.bar.rev131008.IdentityClass", true, loader);
453
454         // test identity
455         final Class<?> baseIdentity = Class.forName("org.opendaylight.yangtools.yang.binding.BaseIdentity", true, loader);
456         assertEquals(ImmutableList.of(baseIdentity), Arrays.asList(identityClass.getInterfaces()));
457
458         // Test annotation
459         try {
460             final Method getId = nodesClass.getMethod("getId");
461             final Annotation[] annotations = getId.getAnnotations();
462             assertEquals(1, annotations.length);
463             final Annotation routingContext = annotations[0];
464             assertEquals(RoutingContext.class, routingContext.annotationType());
465         } catch (final NoSuchMethodException e) {
466             throw new AssertionError("Method getId() not found");
467         }
468
469         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
470     }
471
472     @Test
473     public void compilationTest() throws Exception {
474         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("yang");
475         final File compiledOutputDir = CompilationTestUtils.compilerOutput("yang");
476         generateTestSources("/yang", sourcesOutputDir);
477
478         // Test if sources are compilable
479         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
480
481         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
482     }
483
484     @Test
485     public void testBug586() throws Exception {
486         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("bug586");
487         final File compiledOutputDir = CompilationTestUtils.compilerOutput("bug586");
488         generateTestSources("/compilation/bug586", sourcesOutputDir);
489
490         // Test if sources are compilable
491         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
492
493         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
494     }
495
496     @Test
497     public void testBug4760() throws Exception {
498         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("bug4760");
499         final File compiledOutputDir = CompilationTestUtils.compilerOutput("bug4760");
500         generateTestSources("/compilation/bug4760", sourcesOutputDir);
501
502         // Test if sources are compilable
503         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
504
505         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
506     }
507
508     /**
509      * Test handling nested uses-augmentations.
510      *
511      * @throws Exception
512      */
513     @Test
514     public void testBug1172() throws Exception {
515         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("bug1172");
516         final File compiledOutputDir = CompilationTestUtils.compilerOutput("bug1172");
517         generateTestSources("/compilation/bug1172", sourcesOutputDir);
518
519         // Test if sources are compilable
520         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
521
522         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
523     }
524
525     @Test
526     public void testBug5461() throws Exception {
527         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("bug5461");
528         final File compiledOutputDir = CompilationTestUtils.compilerOutput("bug5461");
529         generateTestSources("/compilation/bug5461", sourcesOutputDir);
530
531         // Test if sources are compilable
532         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
533
534         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
535     }
536
537     @Test
538     public void testBug5882() throws Exception {
539         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("bug5882");
540         final File compiledOutputDir = CompilationTestUtils.compilerOutput("bug5882");
541         generateTestSources("/compilation/bug5882", sourcesOutputDir);
542
543         // Test if sources are compilable
544         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
545
546         final File parent = new File(sourcesOutputDir, CompilationTestUtils.NS_BUG5882);
547         assertTrue(new File(parent, "FooData.java").exists());
548         assertTrue(new File(parent, "TypedefCurrent.java").exists());
549         assertTrue(new File(parent, "TypedefDeprecated.java").exists());
550
551         final String pkg = CompilationTestUtils.BASE_PKG + ".urn.yang.foo.rev160102";
552         final ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
553         final Class<?> cls = loader.loadClass(pkg + ".FooData");
554         final Class<?> clsContainer = loader.loadClass(pkg + ".ContainerMain");
555         final Class<?> clsTypedefDepr = loader.loadClass(pkg + ".TypedefDeprecated");
556         final Class<?> clsTypedefCur = loader.loadClass(pkg + ".TypedefCurrent");
557         final Class<?> clsGroupingDepr = loader.loadClass(pkg + ".GroupingDeprecated");
558         final Class<?> clsGroupingCur = loader.loadClass(pkg + ".GroupingCurrent");
559         final Class<?> clsTypeDef1 = loader.loadClass(pkg + ".Typedef1");
560         final Class<?> clsTypeDef2 = loader.loadClass(pkg + ".Typedef2");
561         final Class<?> clsTypeDef3 = loader.loadClass(pkg + ".Typedef3");
562         assertTrue(clsTypedefDepr.getAnnotations()[0].toString().contains("Deprecated"));
563         assertTrue(clsTypedefCur.getAnnotations().length == 0);
564         assertTrue(clsGroupingDepr.getAnnotations()[0].toString().contains("Deprecated"));
565         assertTrue(clsGroupingCur.getAnnotations().length == 0);
566         assertTrue(clsTypeDef1.getAnnotations().length == 0);
567         assertTrue(clsTypeDef3.getAnnotations().length == 0);
568         assertTrue(clsTypeDef2.getAnnotations()[0].toString().contains("Deprecated"));
569
570         /*methods inside container*/
571         assertTrue(clsContainer.getMethod("getContainerMainLeafDepr").isAnnotationPresent(Deprecated.class));
572         assertTrue(clsContainer.getMethod("getContainerMainListDepr").isAnnotationPresent(Deprecated.class));
573         assertTrue(clsContainer.getMethod("getContainerMainChoiceDepr").isAnnotationPresent(Deprecated.class));
574         assertFalse(clsContainer.getMethod("getContainerMainLeafCurrent").isAnnotationPresent(Deprecated.class));
575         assertFalse(clsContainer.getMethod("getContainerMainListCurrent").isAnnotationPresent(Deprecated.class));
576         assertFalse(clsContainer.getMethod("getContainerMainChoiceCur").isAnnotationPresent(Deprecated.class));
577
578         /*methods inside module*/
579         assertTrue(cls.getMethod("getContainerMainLeafDepr").isAnnotationPresent(Deprecated.class));
580         assertTrue(cls.getMethod("getContainerMainListDepr").isAnnotationPresent(Deprecated.class));
581         assertTrue(cls.getMethod("getContainerMainChoiceDepr").isAnnotationPresent(Deprecated.class));
582         assertFalse(cls.getMethod("getContainerMainLeafCurrent").isAnnotationPresent(Deprecated.class));
583         assertFalse(cls.getMethod("getContainerMainListCurrent").isAnnotationPresent(Deprecated.class));
584         assertFalse(cls.getMethod("getContainerMainChoiceCur").isAnnotationPresent(Deprecated.class));
585         assertTrue(cls.getMethod("getLeafDeprecated").isAnnotationPresent(Deprecated.class));
586
587         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
588     }
589
590     /**
591      * Test if class generated for node from grouping implements ChildOf.
592      *
593      * @throws Exception
594      */
595     @Test
596     public void testBug1377() throws Exception {
597         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("bug1377");
598         final File compiledOutputDir = CompilationTestUtils.compilerOutput("bug1377");
599
600         generateTestSources("/compilation/bug1377", sourcesOutputDir);
601
602         // Test if sources are compilable
603         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
604
605         final ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
606         final Class<?> outputActionClass = Class.forName(CompilationTestUtils.BASE_PKG
607                 + ".urn.test.foo.rev140717.action.action.output.action._case.OutputAction", true, loader);
608         final Class<?> actionClass = Class.forName(CompilationTestUtils.BASE_PKG + ".urn.test.foo.rev140717.Action", true, loader);
609
610         // Test generated 'container output-action'
611         assertTrue(outputActionClass.isInterface());
612         CompilationTestUtils.assertImplementsParameterizedIfc(outputActionClass, ChildOf.class.toString(), actionClass.getCanonicalName());
613
614         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
615     }
616
617     @Test
618     public void classNamesColisionTest() throws Exception {
619         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("class-name-collision");
620         final File compiledOutputDir = CompilationTestUtils.compilerOutput("class-name-collision");
621         generateTestSources("/compilation/class-name-collision", sourcesOutputDir);
622         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
623         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
624     }
625
626     @Test
627     public void innerEnumerationNameCollisionTest() throws Exception {
628         final File sourcesOutputDir = CompilationTestUtils.generatorOutput("mdsal321");
629         final File compiledOutputDir = CompilationTestUtils.compilerOutput("mdsal321");
630         generateTestSources("/compilation/mdsal321", sourcesOutputDir);
631         CompilationTestUtils.testCompilation(sourcesOutputDir, compiledOutputDir);
632         CompilationTestUtils.cleanUp(sourcesOutputDir, compiledOutputDir);
633     }
634
635     private void generateTestSources(final String resourceDirPath, final File sourcesOutputDir)
636             throws IOException, URISyntaxException {
637         final List<File> sourceFiles = CompilationTestUtils.getSourceFiles(resourceDirPath);
638         final SchemaContext context = YangParserTestUtils.parseYangFiles(sourceFiles);
639         final List<Type> types = bindingGenerator.generateTypes(context);
640         Collections.sort(types, (o1, o2) -> o2.getName().compareTo(o1.getName()));
641         final GeneratorJavaFile generator = new GeneratorJavaFile(ImmutableSet.copyOf(types));
642         generator.generateToFile(sourcesOutputDir);
643     }
644
645     private static void testReturnTypeIdentityref(final Class<?> clazz, final String methodName, final String returnTypeStr) throws Exception {
646         Method method;
647         java.lang.reflect.Type returnType;
648         try {
649             method = clazz.getMethod(methodName);
650             assertEquals(java.lang.Class.class, method.getReturnType());
651             returnType = method.getGenericReturnType();
652             assertTrue(returnType instanceof ParameterizedType);
653             final ParameterizedType pt = (ParameterizedType) returnType;
654             final java.lang.reflect.Type[] parameters = pt.getActualTypeArguments();
655             assertEquals(1, parameters.length);
656             final java.lang.reflect.Type parameter = parameters[0];
657             assertTrue(parameter instanceof WildcardType);
658             final WildcardType wildcardType = (WildcardType) parameter;
659             assertEquals("? extends " + returnTypeStr, wildcardType.toString());
660         } catch (final NoSuchMethodException e) {
661             throw new AssertionError("Method '" + methodName + "' not found");
662         }
663     }
664
665     private static void testReturnTypeInstanceIdentitifer(final ClassLoader loader, final Class<?> clazz, final String methodName)
666             throws Exception {
667         Method method;
668         Class<?> rawReturnType;
669         java.lang.reflect.Type returnType;
670         try {
671             method = clazz.getMethod(methodName);
672             rawReturnType = Class.forName("org.opendaylight.yangtools.yang.binding.InstanceIdentifier", true, loader);
673             assertEquals(rawReturnType, method.getReturnType());
674             returnType = method.getGenericReturnType();
675             assertTrue(returnType instanceof ParameterizedType);
676             final ParameterizedType pt = (ParameterizedType) returnType;
677             final java.lang.reflect.Type[] parameters = pt.getActualTypeArguments();
678             assertEquals(1, parameters.length);
679             final java.lang.reflect.Type parameter = parameters[0];
680             assertTrue(parameter instanceof WildcardType);
681             final WildcardType wildcardType = (WildcardType) parameter;
682             assertEquals("?", wildcardType.toString());
683         } catch (final NoSuchMethodException e) {
684             throw new AssertionError("Method '" + methodName + "' not found");
685         }
686     }
687
688 }