5eca4ee13a6a3ed27a100a59edd61bf95d85a8dc
[mdsal.git] / binding / mdsal-binding-generator / src / main / java / org / opendaylight / mdsal / binding / generator / impl / reactor / ListGenerator.java
1 /*
2  * Copyright (c) 2021 PANTHEON.tech, s.r.o. 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.generator.impl.reactor;
9
10 import static org.opendaylight.mdsal.binding.model.ri.BindingTypes.identifiable;
11
12 import org.eclipse.jdt.annotation.Nullable;
13 import org.opendaylight.mdsal.binding.model.api.GeneratedType;
14 import org.opendaylight.mdsal.binding.model.api.MethodSignature.ValueMechanics;
15 import org.opendaylight.mdsal.binding.model.api.Type;
16 import org.opendaylight.mdsal.binding.model.api.type.builder.GeneratedTypeBuilder;
17 import org.opendaylight.mdsal.binding.model.api.type.builder.GeneratedTypeBuilderBase;
18 import org.opendaylight.mdsal.binding.model.api.type.builder.MethodSignatureBuilder;
19 import org.opendaylight.mdsal.binding.model.ri.Types;
20 import org.opendaylight.mdsal.binding.spec.naming.BindingMapping;
21 import org.opendaylight.yangtools.yang.common.Ordering;
22 import org.opendaylight.yangtools.yang.model.api.stmt.KeyEffectiveStatement;
23 import org.opendaylight.yangtools.yang.model.api.stmt.ListEffectiveStatement;
24 import org.opendaylight.yangtools.yang.model.api.stmt.OrderedByEffectiveStatement;
25 import org.opendaylight.yangtools.yang.model.util.SchemaInferenceStack;
26
27 /**
28  * Generator corresponding to a {@code list} statement.
29  */
30 final class ListGenerator extends AbstractCompositeGenerator<ListEffectiveStatement> {
31     private final @Nullable KeyGenerator keyGen;
32
33     ListGenerator(final ListEffectiveStatement statement, final AbstractCompositeGenerator<?> parent) {
34         super(statement, parent);
35         keyGen = statement.findFirstEffectiveSubstatement(KeyEffectiveStatement.class)
36             .map(key -> new KeyGenerator(key, parent, this))
37             .orElse(null);
38     }
39
40     @Nullable KeyGenerator keyGenerator() {
41         return keyGen;
42     }
43
44     @Override
45     void pushToInference(final SchemaInferenceStack dataTree) {
46         dataTree.enterDataTree(statement().getIdentifier());
47     }
48
49     @Override
50     GeneratedType createTypeImpl(final TypeBuilderFactory builderFactory) {
51         final GeneratedTypeBuilder builder = builderFactory.newGeneratedTypeBuilder(typeName());
52         addImplementsChildOf(builder);
53         addAugmentable(builder);
54         addUsesInterfaces(builder, builderFactory);
55         addConcreteInterfaceMethods(builder);
56
57         final ModuleGenerator module = currentModule();
58         module.addQNameConstant(builder, localName());
59
60         if (keyGen != null) {
61             // Add yang.binding.Identifiable and its key() method
62             final GeneratedType keyType = keyGen.getGeneratedType(builderFactory);
63             builder.addImplementsType(identifiable(keyType));
64             builder.addMethod(BindingMapping.IDENTIFIABLE_KEY_NAME)
65                 .setReturnType(keyType)
66                 .addAnnotation(OVERRIDE_ANNOTATION);
67         }
68
69         addGetterMethods(builder, builderFactory);
70
71         annotateDeprecatedIfNecessary(builder);
72         builderFactory.addCodegenInformation(module, statement(), builder);
73         builder.setModuleName(module.statement().argument().getLocalName());
74         //    builder.setSchemaPath(node.getPath());
75
76         return builder.build();
77     }
78
79     @Override
80     Type methodReturnType(final TypeBuilderFactory builderFactory) {
81         final Type generatedType = super.methodReturnType(builderFactory);
82         // We are wrapping the generated type in either a List or a Map based on presence of the key
83         if (keyGen != null) {
84             final Ordering ordering = statement()
85                 .findFirstEffectiveSubstatementArgument(OrderedByEffectiveStatement.class)
86                 .orElse(Ordering.SYSTEM);
87             if (ordering == Ordering.SYSTEM) {
88                 return Types.mapTypeFor(keyGen.getGeneratedType(builderFactory), generatedType);
89             }
90         }
91
92         return Types.listTypeFor(generatedType);
93     }
94
95     @Override
96     MethodSignatureBuilder constructGetter(final GeneratedTypeBuilderBase<?> builder, final Type returnType) {
97         final MethodSignatureBuilder ret = super.constructGetter(builder, returnType)
98             .setMechanics(ValueMechanics.NULLIFY_EMPTY);
99
100         final MethodSignatureBuilder nonnull = builder
101             .addMethod(BindingMapping.getNonnullMethodName(localName().getLocalName()))
102             .setReturnType(returnType)
103             .setDefault(true);
104         annotateDeprecatedIfNecessary(nonnull);
105
106         return ret;
107     }
108 }