|
16 | 16 |
|
17 | 17 | package org.springframework.boot.configurationprocessor;
|
18 | 18 |
|
| 19 | +import java.util.Arrays; |
19 | 20 | import java.util.List;
|
20 |
| -import java.util.Map; |
21 |
| -import java.util.function.Function; |
22 | 21 |
|
23 |
| -import javax.lang.model.element.AnnotationMirror; |
24 | 22 | import javax.lang.model.element.Element;
|
25 | 23 | import javax.lang.model.element.ExecutableElement;
|
26 | 24 | import javax.lang.model.element.TypeElement;
|
27 | 25 | import javax.lang.model.element.VariableElement;
|
28 |
| -import javax.lang.model.type.PrimitiveType; |
29 | 26 | import javax.lang.model.type.TypeMirror;
|
30 |
| -import javax.lang.model.util.TypeKindVisitor8; |
31 |
| -import javax.tools.Diagnostic.Kind; |
32 | 27 |
|
33 | 28 | /**
|
34 | 29 | * A {@link PropertyDescriptor} for a constructor parameter.
|
35 | 30 | *
|
36 | 31 | * @author Stephane Nicoll
|
| 32 | + * @author Phillip Webb |
37 | 33 | */
|
38 |
| -class ConstructorParameterPropertyDescriptor extends PropertyDescriptor<VariableElement> { |
| 34 | +class ConstructorParameterPropertyDescriptor extends ParameterPropertyDescriptor { |
39 | 35 |
|
40 |
| - ConstructorParameterPropertyDescriptor(TypeElement ownerElement, ExecutableElement factoryMethod, |
41 |
| - VariableElement source, String name, TypeMirror type, VariableElement field, ExecutableElement getter, |
42 |
| - ExecutableElement setter) { |
43 |
| - super(ownerElement, factoryMethod, source, name, type, field, getter, setter); |
44 |
| - } |
| 36 | + private final ExecutableElement setter; |
45 | 37 |
|
46 |
| - @Override |
47 |
| - protected boolean isProperty(MetadataGenerationEnvironment env) { |
48 |
| - // If it's a constructor parameter, it doesn't matter as we must be able to bind |
49 |
| - // it to build the object. |
50 |
| - return !isNested(env); |
51 |
| - } |
| 38 | + private final VariableElement field; |
52 | 39 |
|
53 |
| - @Override |
54 |
| - protected Object resolveDefaultValue(MetadataGenerationEnvironment environment) { |
55 |
| - Object defaultValue = getDefaultValueFromAnnotation(environment, getSource()); |
56 |
| - if (defaultValue != null) { |
57 |
| - return defaultValue; |
58 |
| - } |
59 |
| - return getSource().asType().accept(DefaultPrimitiveTypeVisitor.INSTANCE, null); |
| 40 | + ConstructorParameterPropertyDescriptor(String name, TypeMirror type, VariableElement parameter, |
| 41 | + TypeElement declaringElement, ExecutableElement getter, ExecutableElement setter, VariableElement field) { |
| 42 | + super(name, type, parameter, declaringElement, getter); |
| 43 | + this.setter = setter; |
| 44 | + this.field = field; |
60 | 45 | }
|
61 | 46 |
|
62 |
| - private Object getDefaultValueFromAnnotation(MetadataGenerationEnvironment environment, Element element) { |
63 |
| - AnnotationMirror annotation = environment.getDefaultValueAnnotation(element); |
64 |
| - List<String> defaultValue = getDefaultValue(environment, annotation); |
65 |
| - if (defaultValue != null) { |
66 |
| - try { |
67 |
| - TypeMirror specificType = determineSpecificType(environment); |
68 |
| - if (defaultValue.size() == 1) { |
69 |
| - return coerceValue(specificType, defaultValue.get(0)); |
70 |
| - } |
71 |
| - return defaultValue.stream().map((value) -> coerceValue(specificType, value)).toList(); |
72 |
| - } |
73 |
| - catch (IllegalArgumentException ex) { |
74 |
| - environment.getMessager().printMessage(Kind.ERROR, ex.getMessage(), element, annotation); |
75 |
| - } |
76 |
| - } |
77 |
| - return null; |
78 |
| - } |
79 |
| - |
80 |
| - @SuppressWarnings("unchecked") |
81 |
| - private List<String> getDefaultValue(MetadataGenerationEnvironment environment, AnnotationMirror annotation) { |
82 |
| - if (annotation == null) { |
83 |
| - return null; |
84 |
| - } |
85 |
| - Map<String, Object> values = environment.getAnnotationElementValues(annotation); |
86 |
| - return (List<String>) values.get("value"); |
87 |
| - } |
88 |
| - |
89 |
| - private TypeMirror determineSpecificType(MetadataGenerationEnvironment environment) { |
90 |
| - TypeMirror candidate = getSource().asType(); |
91 |
| - TypeMirror elementCandidate = environment.getTypeUtils().extractElementType(candidate); |
92 |
| - if (elementCandidate != null) { |
93 |
| - candidate = elementCandidate; |
94 |
| - } |
95 |
| - PrimitiveType primitiveType = environment.getTypeUtils().getPrimitiveType(candidate); |
96 |
| - return (primitiveType != null) ? primitiveType : candidate; |
97 |
| - } |
98 |
| - |
99 |
| - private Object coerceValue(TypeMirror type, String value) { |
100 |
| - Object coercedValue = type.accept(DefaultValueCoercionTypeVisitor.INSTANCE, value); |
101 |
| - return (coercedValue != null) ? coercedValue : value; |
| 47 | + @Override |
| 48 | + protected List<Element> getDeprecatableElements() { |
| 49 | + return Arrays.asList(getGetter(), this.setter, this.field); |
102 | 50 | }
|
103 | 51 |
|
104 |
| - private static final class DefaultValueCoercionTypeVisitor extends TypeKindVisitor8<Object, String> { |
105 |
| - |
106 |
| - private static final DefaultValueCoercionTypeVisitor INSTANCE = new DefaultValueCoercionTypeVisitor(); |
107 |
| - |
108 |
| - private <T extends Number> T parseNumber(String value, Function<String, T> parser, |
109 |
| - PrimitiveType primitiveType) { |
110 |
| - try { |
111 |
| - return parser.apply(value); |
112 |
| - } |
113 |
| - catch (NumberFormatException ex) { |
114 |
| - throw new IllegalArgumentException( |
115 |
| - String.format("Invalid %s representation '%s'", primitiveType, value)); |
116 |
| - } |
117 |
| - } |
118 |
| - |
119 |
| - @Override |
120 |
| - public Object visitPrimitiveAsBoolean(PrimitiveType t, String value) { |
121 |
| - return Boolean.parseBoolean(value); |
122 |
| - } |
123 |
| - |
124 |
| - @Override |
125 |
| - public Object visitPrimitiveAsByte(PrimitiveType t, String value) { |
126 |
| - return parseNumber(value, Byte::parseByte, t); |
127 |
| - } |
128 |
| - |
129 |
| - @Override |
130 |
| - public Object visitPrimitiveAsShort(PrimitiveType t, String value) { |
131 |
| - return parseNumber(value, Short::parseShort, t); |
132 |
| - } |
133 |
| - |
134 |
| - @Override |
135 |
| - public Object visitPrimitiveAsInt(PrimitiveType t, String value) { |
136 |
| - return parseNumber(value, Integer::parseInt, t); |
137 |
| - } |
138 |
| - |
139 |
| - @Override |
140 |
| - public Object visitPrimitiveAsLong(PrimitiveType t, String value) { |
141 |
| - return parseNumber(value, Long::parseLong, t); |
142 |
| - } |
143 |
| - |
144 |
| - @Override |
145 |
| - public Object visitPrimitiveAsChar(PrimitiveType t, String value) { |
146 |
| - if (value.length() > 1) { |
147 |
| - throw new IllegalArgumentException(String.format("Invalid character representation '%s'", value)); |
148 |
| - } |
149 |
| - return value; |
150 |
| - } |
151 |
| - |
152 |
| - @Override |
153 |
| - public Object visitPrimitiveAsFloat(PrimitiveType t, String value) { |
154 |
| - return parseNumber(value, Float::parseFloat, t); |
155 |
| - } |
156 |
| - |
157 |
| - @Override |
158 |
| - public Object visitPrimitiveAsDouble(PrimitiveType t, String value) { |
159 |
| - return parseNumber(value, Double::parseDouble, t); |
160 |
| - } |
161 |
| - |
| 52 | + @Override |
| 53 | + protected boolean isMarkedAsNested(MetadataGenerationEnvironment environment) { |
| 54 | + return environment.getNestedConfigurationPropertyAnnotation(this.field) != null; |
162 | 55 | }
|
163 | 56 |
|
164 |
| - private static final class DefaultPrimitiveTypeVisitor extends TypeKindVisitor8<Object, Void> { |
165 |
| - |
166 |
| - private static final DefaultPrimitiveTypeVisitor INSTANCE = new DefaultPrimitiveTypeVisitor(); |
167 |
| - |
168 |
| - @Override |
169 |
| - public Object visitPrimitiveAsBoolean(PrimitiveType t, Void ignore) { |
170 |
| - return false; |
171 |
| - } |
172 |
| - |
173 |
| - @Override |
174 |
| - public Object visitPrimitiveAsByte(PrimitiveType t, Void ignore) { |
175 |
| - return (byte) 0; |
176 |
| - } |
177 |
| - |
178 |
| - @Override |
179 |
| - public Object visitPrimitiveAsShort(PrimitiveType t, Void ignore) { |
180 |
| - return (short) 0; |
181 |
| - } |
182 |
| - |
183 |
| - @Override |
184 |
| - public Object visitPrimitiveAsInt(PrimitiveType t, Void ignore) { |
185 |
| - return 0; |
186 |
| - } |
187 |
| - |
188 |
| - @Override |
189 |
| - public Object visitPrimitiveAsLong(PrimitiveType t, Void ignore) { |
190 |
| - return 0L; |
191 |
| - } |
192 |
| - |
193 |
| - @Override |
194 |
| - public Object visitPrimitiveAsChar(PrimitiveType t, Void ignore) { |
195 |
| - return null; |
196 |
| - } |
197 |
| - |
198 |
| - @Override |
199 |
| - public Object visitPrimitiveAsFloat(PrimitiveType t, Void ignore) { |
200 |
| - return 0F; |
201 |
| - } |
202 |
| - |
203 |
| - @Override |
204 |
| - public Object visitPrimitiveAsDouble(PrimitiveType t, Void ignore) { |
205 |
| - return 0D; |
206 |
| - } |
207 |
| - |
| 57 | + @Override |
| 58 | + protected String resolveDescription(MetadataGenerationEnvironment environment) { |
| 59 | + return environment.getTypeUtils().getJavaDoc(this.field); |
208 | 60 | }
|
209 | 61 |
|
210 | 62 | }
|
0 commit comments