Fixed incorrect path construction in JUnit tests.
[yangtools.git] / code-generator / binding-java-api-generator / src / test / java / org / opendaylight / yangtools / sal / java / api / generator / test / CompilationTest.java
1 /*
2  * Copyright (c) 2013 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.yangtools.sal.java.api.generator.test;
9
10 import static org.junit.Assert.assertEquals;
11 import static org.junit.Assert.assertTrue;
12 import static org.opendaylight.yangtools.sal.java.api.generator.test.CompilationTestUtils.*;
13
14 import java.io.File;
15 import java.lang.annotation.Annotation;
16 import java.lang.reflect.Field;
17 import java.lang.reflect.Method;
18 import java.lang.reflect.ParameterizedType;
19 import java.lang.reflect.WildcardType;
20 import java.math.BigDecimal;
21 import java.math.BigInteger;
22 import java.net.URL;
23 import java.net.URLClassLoader;
24 import java.util.ArrayList;
25 import java.util.HashSet;
26 import java.util.List;
27 import java.util.Set;
28
29 import org.junit.Test;
30 import org.opendaylight.yangtools.sal.binding.model.api.Type;
31 import org.opendaylight.yangtools.sal.java.api.generator.GeneratorJavaFile;
32 import org.opendaylight.yangtools.yang.binding.annotations.RoutingContext;
33 import org.opendaylight.yangtools.yang.model.api.Module;
34 import org.opendaylight.yangtools.yang.model.api.SchemaContext;
35
36 import com.google.common.collect.Range;
37
38 /**
39  * Test correct code generation.
40  *
41  */
42 public class CompilationTest extends BaseCompilationTest {
43
44     @Test
45     public void testListGeneration() throws Exception {
46         final File sourcesOutputDir = new File(GENERATOR_OUTPUT_PATH + FS + "list-gen");
47         assertTrue("Failed to create test file '" + sourcesOutputDir + "'", sourcesOutputDir.mkdir());
48         final File compiledOutputDir = new File(COMPILER_OUTPUT_PATH + FS + "list-gen");
49         assertTrue("Failed to create test file '" + compiledOutputDir + "'", compiledOutputDir.mkdir());
50
51         final List<File> sourceFiles = getSourceFiles("/compilation/list-gen");
52         final Set<Module> modulesToBuild = parser.parseYangModels(sourceFiles);
53         final SchemaContext context = parser.resolveSchemaContext(modulesToBuild);
54         final List<Type> types = bindingGenerator.generateTypes(context);
55         final GeneratorJavaFile generator = new GeneratorJavaFile(new HashSet<>(types));
56         generator.generateToFile(sourcesOutputDir);
57
58         // Test if all sources are generated
59         File parent = new File(sourcesOutputDir, NS_TEST);
60         File keyArgs = new File(parent, "KeyArgs.java");
61         File links = new File(parent, "Links.java");
62         File linksBuilder = new File(parent, "LinksBuilder.java");
63         File linksKey = new File(parent, "LinksKey.java");
64         File testData = new File(parent, "TestData.java");
65         assertTrue(keyArgs.exists());
66         assertTrue(links.exists());
67         assertTrue(linksBuilder.exists());
68         assertTrue(linksKey.exists());
69         assertTrue(testData.exists());
70         assertFilesCount(parent, 6);
71
72         parent = new File(sourcesOutputDir, NS_TEST + FS + "links");
73         File level = new File(parent, "Level.java");
74         File linkGroup = new File(parent, "LinkGroup.java");
75         File node = new File(parent, "Node.java");
76         File nodeBuilder = new File(parent, "NodeBuilder.java");
77         File nodeList = new File(parent, "NodeList.java");
78         File nodeListBuilder = new File(parent, "NodeListBuilder.java");
79         File nodesType = new File(parent, "NodesType.java");
80         assertTrue(level.exists());
81         assertTrue(linkGroup.exists());
82         assertTrue(node.exists());
83         assertTrue(nodeBuilder.exists());
84         assertTrue(nodeList.exists());
85         assertTrue(nodeListBuilder.exists());
86         assertTrue(nodesType.exists());
87         assertFilesCount(parent, 7);
88
89         // Test if sources are compilable
90         testCompilation(sourcesOutputDir, compiledOutputDir);
91
92         ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
93         Class<?> keyArgsClass = Class.forName(BASE_PKG + ".urn.opendaylight.test.rev131008.KeyArgs", true, loader);
94         Class<?> linksClass = Class.forName(BASE_PKG + ".urn.opendaylight.test.rev131008.Links", true, loader);
95         Class<?> linksKeyClass = Class.forName(BASE_PKG + ".urn.opendaylight.test.rev131008.LinksKey", true, loader);
96
97         // Test generated 'grouping key-args'
98         assertTrue(keyArgsClass.isInterface());
99         assertEquals(2, keyArgsClass.getDeclaredMethods().length);
100         assertContainsMethod(keyArgsClass, String.class, "getName");
101         assertContainsMethod(keyArgsClass, Integer.class, "getSize");
102
103         // Test generated 'list links'
104         assertTrue(linksClass.isInterface());
105         // TODO: anyxml
106         assertEquals(6, linksClass.getDeclaredMethods().length);
107         assertImplementsIfc(linksClass, keyArgsClass);
108
109         // Test list key constructor arguments ordering
110         assertContainsConstructor(linksKeyClass, Byte.class, String.class, Integer.class);
111         // Test serialVersionUID generation
112         Field suid = assertContainsField(linksKeyClass, "serialVersionUID", Long.TYPE);
113         suid.setAccessible(true);
114         assertEquals(-8829501012356283881L, suid.getLong(null));
115
116         cleanUp(sourcesOutputDir, compiledOutputDir);
117     }
118
119     @Test
120     public void testAugmentUnderUsesGeneration() throws Exception {
121         final File sourcesOutputDir = new File(GENERATOR_OUTPUT_PATH + FS + "augment-under-uses");
122         assertTrue("Failed to create test file '" + sourcesOutputDir + "'", sourcesOutputDir.mkdir());
123         final File compiledOutputDir = new File(COMPILER_OUTPUT_PATH + FS + "augment-under-uses");
124         assertTrue("Failed to create test file '" + compiledOutputDir + "'", compiledOutputDir.mkdir());
125
126         final List<File> sourceFiles = getSourceFiles("/compilation/augment-under-uses");
127         final Set<Module> modulesToBuild = parser.parseYangModels(sourceFiles);
128         final SchemaContext context = parser.resolveSchemaContext(modulesToBuild);
129         final List<Type> types = bindingGenerator.generateTypes(context);
130         final GeneratorJavaFile generator = new GeneratorJavaFile(new HashSet<>(types));
131         generator.generateToFile(sourcesOutputDir);
132
133         // Test if all sources were generated from 'module foo'
134         File parent = new File(sourcesOutputDir, NS_FOO);
135         assertTrue(new File(parent, "Object.java").exists());
136         assertTrue(new File(parent, "OpenObject.java").exists());
137         assertTrue(new File(parent, "ExplicitRouteObject.java").exists());
138         assertTrue(new File(parent, "PathKeySubobject.java").exists());
139         assertFilesCount(parent, 7);
140
141         parent = new File(parent, "object");
142         assertTrue(new File(parent, "Nodes.java").exists());
143         assertTrue(new File(parent, "NodesBuilder.java").exists());
144         assertFilesCount(parent, 2);
145
146         parent = new File(sourcesOutputDir, NS_FOO + FS + "open");
147         assertFilesCount(parent, 1);
148
149         parent = new File(parent, "object");
150         assertTrue(new File(parent, "Nodes1.java").exists());
151         assertTrue(new File(parent, "Nodes1Builder.java").exists());
152         assertFilesCount(parent, 3);
153
154         parent = new File(parent, "nodes");
155         assertTrue(new File(parent, "Links.java").exists());
156         assertTrue(new File(parent, "LinksBuilder.java").exists());
157         assertFilesCount(parent, 2);
158
159         parent = new File(sourcesOutputDir, NS_FOO + FS + "explicit");
160         assertFilesCount(parent, 1);
161         parent = new File(parent, "route");
162         assertFilesCount(parent, 1);
163         parent = new File(parent, "object");
164         assertTrue(new File(parent, "Subobjects.java").exists());
165         assertTrue(new File(parent, "SubobjectsBuilder.java").exists());
166         assertFilesCount(parent, 3);
167
168         parent = new File(parent, "subobjects");
169         assertFilesCount(parent, 1);
170         parent = new File(parent, "subobject");
171         assertFilesCount(parent, 1);
172         parent = new File(parent, "type");
173         assertTrue(new File(parent, "PathKey.java").exists());
174         assertTrue(new File(parent, "PathKeyBuilder.java").exists());
175         assertFilesCount(parent, 3);
176
177         parent = new File(parent, "path");
178         assertFilesCount(parent, 1);
179         parent = new File(parent, "key");
180         assertTrue(new File(parent, "PathKey.java").exists());
181         assertTrue(new File(parent, "PathKeyBuilder.java").exists());
182         assertFilesCount(parent, 2);
183
184         // Test if all sources were generated from 'module bar'
185         parent = new File(sourcesOutputDir, NS_BAR);
186         assertTrue(new File(parent, "BasicExplicitRouteSubobjects.java").exists());
187         assertTrue(new File(parent, "ExplicitRouteSubobjects.java").exists());
188         assertFilesCount(parent, 3);
189
190         parent = new File(parent, "basic");
191         assertFilesCount(parent, 1);
192         parent = new File(parent, "explicit");
193         assertFilesCount(parent, 1);
194         parent = new File(parent, "route");
195         assertFilesCount(parent, 1);
196
197         parent = new File(parent, "subobjects");
198         assertFilesCount(parent, 2);
199         assertTrue(new File(parent, "SubobjectType.java").exists());
200
201         parent = new File(parent, "subobject");
202         assertFilesCount(parent, 1);
203
204         parent = new File(parent, "type");
205         assertTrue(new File(parent, "IpPrefix.java").exists());
206         assertTrue(new File(parent, "IpPrefixBuilder.java").exists());
207         assertTrue(new File(parent, "Label.java").exists());
208         assertTrue(new File(parent, "LabelBuilder.java").exists());
209         assertFilesCount(parent, 4);
210
211         // Test if sources are compilable
212         testCompilation(sourcesOutputDir, compiledOutputDir);
213
214         cleanUp(sourcesOutputDir, compiledOutputDir);
215     }
216
217     @Test
218     public void testAugmentOfAugmentGeneration() throws Exception {
219         final File sourcesOutputDir = new File(GENERATOR_OUTPUT_PATH + FS + "aug-of-aug");
220         assertTrue("Failed to create test file '" + sourcesOutputDir + "'", sourcesOutputDir.mkdir());
221         final File compiledOutputDir = new File(COMPILER_OUTPUT_PATH + FS + "aug-of-aug");
222         assertTrue("Failed to create test file '" + compiledOutputDir + "'", compiledOutputDir.mkdir());
223
224         final List<File> sourceFiles = getSourceFiles("/compilation/augment-of-augment");
225         final Set<Module> modulesToBuild = parser.parseYangModels(sourceFiles);
226         final SchemaContext context = parser.resolveSchemaContext(modulesToBuild);
227         final List<Type> types = bindingGenerator.generateTypes(context);
228         final GeneratorJavaFile generator = new GeneratorJavaFile(new HashSet<>(types));
229         generator.generateToFile(sourcesOutputDir);
230
231         // Test if all sources were generated from 'module foo'
232         File parent = new File(sourcesOutputDir, NS_FOO);
233         File fooListener = new File(parent, "FooListener.java");
234         File pathAttributes = new File(parent, "PathAttributes.java");
235         File update = new File(parent, "Update.java");
236         File updateBuilder = new File(parent, "UpdateBuilder.java");
237         assertTrue(fooListener.exists());
238         assertTrue(pathAttributes.exists());
239         assertTrue(update.exists());
240         assertTrue(updateBuilder.exists());
241         assertFilesCount(parent, 6);
242
243         parent = new File(sourcesOutputDir, NS_FOO + FS + "path");
244         assertFilesCount(parent, 1);
245         parent = new File(parent, "attributes");
246         assertFilesCount(parent, 2);
247         File origin = new File(parent, "Origin.java");
248         File originBuilder = new File(parent, "OriginBuilder.java");
249         assertTrue(origin.exists());
250         assertTrue(originBuilder.exists());
251
252         parent = new File(sourcesOutputDir, NS_FOO + FS + "update");
253         assertFilesCount(parent, 2);
254         pathAttributes = new File(parent, "PathAttributes.java");
255         File pathAttributesBuilder = new File(parent, "PathAttributesBuilder.java");
256         assertTrue(pathAttributes.exists());
257         assertTrue(pathAttributesBuilder.exists());
258
259         // Test if all sources were generated from 'module bar'
260         parent = new File(sourcesOutputDir, NS_BAR);
261         File destination = new File(parent, "Destination.java");
262         File pathAttributes1 = new File(parent, "PathAttributes1.java");
263         File pathAttributes1Builder = new File(parent, "PathAttributes1Builder.java");
264         assertTrue(destination.exists());
265         assertTrue(pathAttributes1.exists());
266         assertTrue(pathAttributes1Builder.exists());
267         assertFilesCount(parent, 5);
268
269         parent = new File(sourcesOutputDir, NS_BAR + FS + "destination");
270         assertFilesCount(parent, 2);
271         File destinationType = new File(parent, "DestinationType.java");
272         assertTrue(destinationType.exists());
273
274         parent = new File(parent, "destination");
275         assertFilesCount(parent, 1);
276         parent = new File(parent, "type");
277         assertFilesCount(parent, 2);
278         File destinationIpv4 = new File(parent, "DestinationIp.java");
279         File destinationIpv4Builder = new File(parent, "DestinationIpBuilder.java");
280         assertTrue(destinationIpv4.exists());
281         assertTrue(destinationIpv4Builder.exists());
282
283         parent = new File(sourcesOutputDir, NS_BAR + FS + "update");
284         assertFilesCount(parent, 1);
285         parent = new File(parent, "path");
286         assertFilesCount(parent, 1);
287         parent = new File(parent, "attributes");
288         File mpUnreachNlri = new File(parent, "MpUnreachNlri.java");
289         File mpUnreachNlriBuilder = new File(parent, "MpUnreachNlriBuilder.java");
290         assertTrue(mpUnreachNlri.exists());
291         assertTrue(mpUnreachNlriBuilder.exists());
292         assertFilesCount(parent, 3);
293
294         parent = new File(parent, "mp");
295         assertFilesCount(parent, 1);
296         parent = new File(parent, "unreach");
297         assertFilesCount(parent, 1);
298         parent = new File(parent, "nlri");
299         File withdrawnRoutes = new File(parent, "WithdrawnRoutes.java");
300         File withdrawnRoutesBuilder = new File(parent, "WithdrawnRoutesBuilder.java");
301         assertTrue(withdrawnRoutes.exists());
302         assertTrue(withdrawnRoutesBuilder.exists());
303         assertFilesCount(parent, 2);
304
305         // Test if all sources were generated from 'module baz'
306         parent = new File(sourcesOutputDir, NS_BAZ);
307         assertFilesCount(parent, 2);
308         File linkstateDestination = new File(parent, "LinkstateDestination.java");
309         assertTrue(linkstateDestination.exists());
310
311         parent = new File(sourcesOutputDir, NS_BAZ + FS + "update");
312         assertFilesCount(parent, 1);
313         parent = new File(parent, "path");
314         assertFilesCount(parent, 1);
315         parent = new File(parent, "attributes");
316         assertFilesCount(parent, 1);
317         parent = new File(parent, "mp");
318         assertFilesCount(parent, 1);
319         parent = new File(parent, "unreach");
320         assertFilesCount(parent, 1);
321         parent = new File(parent, "nlri");
322         assertFilesCount(parent, 1);
323         parent = new File(parent, "withdrawn");
324         assertFilesCount(parent, 1);
325         parent = new File(parent, "routes");
326         assertFilesCount(parent, 1);
327         parent = new File(parent, "destination");
328         assertFilesCount(parent, 1);
329         parent = new File(parent, "type");
330         File destinationLinkstate = new File(parent, "DestinationLinkstate.java");
331         File destinationLinkstateBuilder = new File(parent, "DestinationLinkstateBuilder.java");
332         assertTrue(destinationLinkstate.exists());
333         assertTrue(destinationLinkstateBuilder.exists());
334         assertFilesCount(parent, 3);
335         parent = new File(parent, "destination");
336         assertFilesCount(parent, 1);
337         parent = new File(parent, "linkstate");
338         File links = new File(parent, "Links.java");
339         File linksBuilder = new File(parent, "LinksBuilder.java");
340         assertTrue(links.exists());
341         assertTrue(linksBuilder.exists());
342         assertFilesCount(parent, 3);
343         parent = new File(parent, "links");
344         File source = new File(parent, "Source.java");
345         File sourceBuilder = new File(parent, "SourceBuilder.java");
346         assertTrue(source.exists());
347         assertTrue(sourceBuilder.exists());
348         assertFilesCount(parent, 3);
349         parent = new File(parent, "source");
350         File address = new File(parent, "Address.java");
351         File addressBuilder = new File(parent, "AddressBuilder.java");
352         assertTrue(address.exists());
353         assertTrue(addressBuilder.exists());
354         assertFilesCount(parent, 2);
355
356         // Test if sources are compilable
357         testCompilation(sourcesOutputDir, compiledOutputDir);
358
359         cleanUp(sourcesOutputDir, compiledOutputDir);
360     }
361
362     @Test
363     public void testLeafReturnTypes() throws Exception {
364         final File sourcesOutputDir = new File(GENERATOR_OUTPUT_PATH + FS + "leaf-return-types");
365         assertTrue("Failed to create test file '" + sourcesOutputDir + "'", sourcesOutputDir.mkdir());
366         final File compiledOutputDir = new File(COMPILER_OUTPUT_PATH + FS + "leaf-return-types");
367         assertTrue("Failed to create test file '" + compiledOutputDir + "'", compiledOutputDir.mkdir());
368
369         final List<File> sourceFiles = getSourceFiles("/compilation/leaf-return-types");
370         final Set<Module> modulesToBuild = parser.parseYangModels(sourceFiles);
371         final SchemaContext context = parser.resolveSchemaContext(modulesToBuild);
372         final List<Type> types = bindingGenerator.generateTypes(context);
373         final GeneratorJavaFile generator = new GeneratorJavaFile(new HashSet<>(types));
374         generator.generateToFile(sourcesOutputDir);
375
376         File parent = new File(sourcesOutputDir, NS_TEST);
377         assertTrue(new File(parent, "TestData.java").exists());
378         assertTrue(new File(parent, "Nodes.java").exists());
379         assertTrue(new File(parent, "NodesBuilder.java").exists());
380         assertTrue(new File(parent, "Alg.java").exists());
381         assertTrue(new File(parent, "IdUnionBuilder.java").exists());
382         assertFilesCount(parent, 5);
383
384         // Test if sources are compilable
385         testCompilation(sourcesOutputDir, compiledOutputDir);
386
387         String pkg = BASE_PKG + ".urn.opendaylight.test.rev131008";
388         ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
389         Class<?> nodesClass = Class.forName(pkg + ".Nodes", true, loader);
390         Class<?> builderClass = Class.forName(pkg + ".NodesBuilder", true, loader);
391
392         // Test methods return type
393         byte[] b = new byte[] {};
394         assertContainsMethod(nodesClass, b.getClass(), "getIdBinary");
395         assertContainsMethod(nodesClass, pkg + ".Nodes$IdBits", "getIdBits", loader);
396         assertContainsMethod(nodesClass, Boolean.class, "isIdBoolean");
397         assertContainsMethod(nodesClass, BigDecimal.class, "getIdDecimal64");
398         assertContainsMethod(nodesClass, Boolean.class, "isIdEmpty");
399         assertContainsMethod(nodesClass, pkg + ".Nodes$IdEnumeration", "getIdEnumeration", loader);
400         testReturnTypeIdentityref(nodesClass, "getIdIdentityref", pkg + ".Alg");
401         testReturnTypeInstanceIdentitifer(loader, nodesClass, "getIdInstanceIdentifier");
402         assertContainsMethod(nodesClass, Byte.class, "getId8");
403         assertContainsMethod(nodesClass, Short.class, "getId16");
404         assertContainsMethod(nodesClass, Integer.class, "getId32");
405         assertContainsMethod(nodesClass, Long.class, "getId64");
406         assertContainsMethod(nodesClass, Long.class, "getIdLeafref");
407         assertContainsMethod(nodesClass, String.class, "getIdString");
408         assertContainsMethod(nodesClass, Short.class, "getIdU8");
409         assertContainsMethod(nodesClass, Integer.class, "getIdU16");
410         assertContainsMethod(nodesClass, Long.class, "getIdU32");
411         assertContainsMethod(nodesClass, BigInteger.class, "getIdU64");
412         assertContainsMethod(nodesClass, pkg + ".Nodes$IdUnion", "getIdUnion", loader);
413
414         Object builderObj = builderClass.newInstance();
415
416         Method m = assertContainsMethod(builderClass, builderClass, "setIdBinary", b.getClass());
417         List<Range<Integer>> lengthConstraints = new ArrayList<>();
418         lengthConstraints.add(Range.closed(1, 10));
419         Object arg = new byte[] {};
420         String expectedMsg = String.format("Invalid length: {}, expected: {}.", arg, lengthConstraints);
421         assertContainsRestrictionCheck(builderObj, m, expectedMsg, arg);
422
423         m = assertContainsMethod(builderClass, builderClass, "setIdDecimal64", BigDecimal.class);
424         List<Range<BigDecimal>> rangeConstraints = new ArrayList<>();
425         rangeConstraints.add(Range.closed(new BigDecimal("1.5"), new BigDecimal("5.5")));
426         arg = new BigDecimal("1.4");
427         expectedMsg = String.format("Invalid range: %s, expected: %s.", arg, rangeConstraints);
428         assertContainsRestrictionCheck(builderObj, m, expectedMsg, arg);
429
430         cleanUp(sourcesOutputDir, compiledOutputDir);
431     }
432
433     @Test
434     public void testGenerationContextReferenceExtension() throws Exception {
435         final File sourcesOutputDir = new File(GENERATOR_OUTPUT_PATH + FS + "context-reference");
436         assertTrue("Failed to create test file '" + sourcesOutputDir + "'", sourcesOutputDir.mkdir());
437         final File compiledOutputDir = new File(COMPILER_OUTPUT_PATH + FS + "context-reference");
438         assertTrue("Failed to create test file '" + compiledOutputDir + "'", compiledOutputDir.mkdir());
439
440         final List<File> sourceFiles = getSourceFiles("/compilation/context-reference");
441         final Set<Module> modulesToBuild = parser.parseYangModels(sourceFiles);
442         final SchemaContext context = parser.resolveSchemaContext(modulesToBuild);
443         final List<Type> types = bindingGenerator.generateTypes(context);
444         final GeneratorJavaFile generator = new GeneratorJavaFile(new HashSet<>(types));
445         generator.generateToFile(sourcesOutputDir);
446
447         // Test if all sources are generated
448         File fooParent = new File(sourcesOutputDir, NS_FOO);
449         assertFilesCount(fooParent, 3);
450         assertTrue(new File(fooParent, "FooData.java").exists());
451         assertTrue(new File(fooParent, "Nodes.java").exists());
452         assertTrue(new File(fooParent, "NodesBuilder.java").exists());
453
454         File barParent = new File(sourcesOutputDir, NS_BAR);
455         assertFilesCount(barParent, 1);
456         assertTrue(new File(barParent, "IdentityClass.java").exists());
457
458         // Test if sources are compilable
459         testCompilation(sourcesOutputDir, compiledOutputDir);
460
461         ClassLoader loader = new URLClassLoader(new URL[] { compiledOutputDir.toURI().toURL() });
462         Class<?> nodesClass = Class.forName(BASE_PKG + ".urn.opendaylight.foo.rev131008.Nodes", true, loader);
463         Class<?> identityClass = Class
464                 .forName(BASE_PKG + ".urn.opendaylight.bar.rev131008.IdentityClass", true, loader);
465
466         // test identity
467         try {
468             identityClass.getConstructor();
469             Class<?> baseIdentity = Class.forName("org.opendaylight.yangtools.yang.binding.BaseIdentity", true, loader);
470             assertEquals(baseIdentity, identityClass.getSuperclass());
471         } catch (NoSuchMethodException e) {
472             throw new AssertionError("IdentityClass must have no-arg constructor");
473         }
474
475         // Test annotation
476         try {
477             Method getId = nodesClass.getMethod("getId");
478             Annotation[] annotations = getId.getAnnotations();
479             assertEquals(1, annotations.length);
480             Annotation routingContext = annotations[0];
481             assertEquals(RoutingContext.class, routingContext.annotationType());
482         } catch (NoSuchMethodException e) {
483             throw new AssertionError("Method getId() not found");
484         }
485
486         cleanUp(sourcesOutputDir, compiledOutputDir);
487     }
488
489     @Test
490     public void compilationTest() throws Exception {
491         final File sourcesOutputDir = new File(GENERATOR_OUTPUT_PATH + FS + "yang");
492         assertTrue("Failed to create test file '" + sourcesOutputDir + "'", sourcesOutputDir.mkdir());
493         final File compiledOutputDir = new File(COMPILER_OUTPUT_PATH + FS + "yang");
494         assertTrue("Failed to create test file '" + compiledOutputDir + "'", compiledOutputDir.mkdir());
495
496         final List<File> sourceFiles = getSourceFiles("/yang");
497         final Set<Module> modulesToBuild = parser.parseYangModels(sourceFiles);
498         final SchemaContext context = parser.resolveSchemaContext(modulesToBuild);
499         final List<Type> types = bindingGenerator.generateTypes(context);
500         final GeneratorJavaFile generator = new GeneratorJavaFile(new HashSet<>(types));
501         generator.generateToFile(sourcesOutputDir);
502
503         // Test if sources are compilable
504         testCompilation(sourcesOutputDir, compiledOutputDir);
505
506         cleanUp(sourcesOutputDir, compiledOutputDir);
507     }
508
509
510     private void testReturnTypeIdentityref(Class<?> clazz, String methodName, String returnTypeStr) throws Exception {
511         Method method;
512         java.lang.reflect.Type returnType;
513         try {
514             method = clazz.getMethod(methodName);
515             assertEquals(java.lang.Class.class, method.getReturnType());
516             returnType = method.getGenericReturnType();
517             assertTrue(returnType instanceof ParameterizedType);
518             ParameterizedType pt = (ParameterizedType) returnType;
519             java.lang.reflect.Type[] parameters = pt.getActualTypeArguments();
520             assertEquals(1, parameters.length);
521             java.lang.reflect.Type parameter = parameters[0];
522             assertTrue(parameter instanceof WildcardType);
523             WildcardType wildcardType = (WildcardType) parameter;
524             assertEquals("? extends " + returnTypeStr, wildcardType.toString());
525         } catch (NoSuchMethodException e) {
526             throw new AssertionError("Method '" + methodName + "' not found");
527         }
528     }
529
530     private void testReturnTypeInstanceIdentitifer(ClassLoader loader, Class<?> clazz, String methodName)
531             throws Exception {
532         Method method;
533         Class<?> rawReturnType;
534         java.lang.reflect.Type returnType;
535         try {
536             method = clazz.getMethod(methodName);
537             rawReturnType = Class.forName("org.opendaylight.yangtools.yang.binding.InstanceIdentifier", true, loader);
538             assertEquals(rawReturnType, method.getReturnType());
539             returnType = method.getGenericReturnType();
540             assertTrue(returnType instanceof ParameterizedType);
541             ParameterizedType pt = (ParameterizedType) returnType;
542             java.lang.reflect.Type[] parameters = pt.getActualTypeArguments();
543             assertEquals(1, parameters.length);
544             java.lang.reflect.Type parameter = parameters[0];
545             assertTrue(parameter instanceof WildcardType);
546             WildcardType wildcardType = (WildcardType) parameter;
547             assertEquals("?", wildcardType.toString());
548         } catch (NoSuchMethodException e) {
549             throw new AssertionError("Method '" + methodName + "' not found");
550         }
551     }
552
553 }