001/* 002 * SPDX-FileCopyrightText: none 003 * SPDX-License-Identifier: CC0-1.0 004 */ 005 006package gov.nist.secauto.metaschema.databind.model.annotations; 007 008import gov.nist.secauto.metaschema.core.datatype.IDataTypeAdapter; 009import gov.nist.secauto.metaschema.core.datatype.adapter.MetaschemaDataTypeProvider; 010import gov.nist.secauto.metaschema.core.datatype.markup.MarkupLine; 011import gov.nist.secauto.metaschema.core.datatype.markup.MarkupMultiline; 012import gov.nist.secauto.metaschema.core.model.IAttributable; 013import gov.nist.secauto.metaschema.core.model.IBoundObject; 014import gov.nist.secauto.metaschema.core.model.IMetaschemaData; 015import gov.nist.secauto.metaschema.core.model.IModule; 016import gov.nist.secauto.metaschema.core.util.CollectionUtil; 017import gov.nist.secauto.metaschema.databind.IBindingContext; 018import gov.nist.secauto.metaschema.databind.model.IGroupAs; 019import gov.nist.secauto.metaschema.databind.model.impl.DefaultGroupAs; 020 021import java.lang.annotation.Annotation; 022import java.lang.reflect.Field; 023import java.net.URI; 024import java.util.Arrays; 025import java.util.LinkedHashSet; 026import java.util.List; 027import java.util.Map; 028import java.util.Set; 029 030import edu.umd.cs.findbugs.annotations.NonNull; 031import edu.umd.cs.findbugs.annotations.Nullable; 032 033public final class ModelUtil { 034 // TODO: replace NO_STRING_VALUE with NULL_VALUE where possible. URIs will not 035 // allow NULL_VALUE. 036 public static final String NO_STRING_VALUE = "##none"; 037 public static final String DEFAULT_STRING_VALUE = "##default"; 038 /** 039 * A placeholder for a {@code null} value for use in annotations, which cannot 040 * be null by default. 041 * <p> 042 * Use of {@code "\u0000"} simple substitute for {@code null} to allow 043 * implementations to recognize the "no default value" state. 044 */ 045 public static final String NULL_VALUE = "\u0000"; 046 047 private ModelUtil() { 048 // disable construction 049 } 050 051 /** 052 * Get the requested annotation from the provided Java class. 053 * 054 * @param <A> 055 * the annotation Java type 056 * @param clazz 057 * the Java class to get the annotation from 058 * @param annotationClass 059 * the annotation class instance 060 * @return the annotation 061 * @throws IllegalArgumentException 062 * if the annotation was not present on the class 063 */ 064 @NonNull 065 public static <A extends Annotation> A getAnnotation( 066 @NonNull Class<?> clazz, 067 Class<A> annotationClass) { 068 A annotation = clazz.getAnnotation(annotationClass); 069 if (annotation == null) { 070 throw new IllegalArgumentException( 071 String.format("Class '%s' is missing the '%s' annotation.", 072 clazz.getName(), 073 annotationClass.getName())); 074 } 075 return annotation; 076 } 077 078 /** 079 * Get the requested annotation from the provided Java field. 080 * 081 * @param <A> 082 * the annotation Java type 083 * @param javaField 084 * the Java field to get the annotation from 085 * @param annotationClass 086 * the annotation class instance 087 * @return the annotation 088 * @throws IllegalArgumentException 089 * if the annotation was not present on the field 090 */ 091 @NonNull 092 public static <A extends Annotation> A getAnnotation( 093 @NonNull Field javaField, 094 Class<A> annotationClass) { 095 A annotation = javaField.getAnnotation(annotationClass); 096 if (annotation == null) { 097 throw new IllegalArgumentException( 098 String.format("Field '%s' is missing the '%s' annotation.", 099 javaField.toGenericString(), 100 annotationClass.getName())); 101 } 102 return annotation; 103 } 104 105 /** 106 * Resolves a string value. If the value is {@code null} or "##default", then 107 * the provided default value will be used instead. If the value is "##none", 108 * then the value will be {@code null}. Otherwise, the value is returned. 109 * 110 * @param value 111 * the requested value 112 * @param defaultValue 113 * the default value 114 * @return the resolved value or {@code null} 115 */ 116 @Nullable 117 public static String resolveNoneOrDefault(@Nullable String value, @Nullable String defaultValue) { 118 String retval; 119 if (value == null || DEFAULT_STRING_VALUE.equals(value)) { 120 retval = defaultValue; 121 } else if (NO_STRING_VALUE.equals(value)) { 122 retval = null; // NOPMD - intentional 123 } else { 124 retval = value; 125 } 126 return retval; 127 } 128 129 /** 130 * Get the processed value of a string. If the value is "##none", then the value 131 * will be {@code null}. Otherwise the value is returned. 132 * 133 * @param value 134 * text or {@code "##none"} if no text is provided 135 * @return the resolved value or {@code null} 136 */ 137 @Nullable 138 public static String resolveNoneOrValue(@NonNull String value) { 139 return NO_STRING_VALUE.equals(value) ? null : value; 140 } 141 142 /** 143 * Get the markup value of a markdown string. 144 * 145 * @param value 146 * markdown text or {@code "##none"} if no text is provided 147 * @return the markup line content or {@code null} if no markup content was 148 * provided 149 */ 150 @Nullable 151 public static MarkupLine resolveToMarkupLine(@NonNull String value) { 152 return resolveNoneOrValue(value) == null ? null : MarkupLine.fromMarkdown(value); 153 } 154 155 /** 156 * Get the markup value of a markdown string. 157 * 158 * @param value 159 * markdown text or {@code "##none"} if no text is provided 160 * @return the markup line content or {@code null} if no markup content was 161 * provided 162 */ 163 @Nullable 164 public static MarkupMultiline resolveToMarkupMultiline(@NonNull String value) { 165 return resolveNoneOrValue(value) == null ? null : MarkupMultiline.fromMarkdown(value); 166 } 167 168 /** 169 * Get the data type adapter instance of the provided adapter class. 170 * <p> 171 * If the provided adapter Java class is the {@link NullJavaTypeAdapter} class, 172 * then the default data type adapter will be returned. 173 * 174 * @param adapterClass 175 * the data type adapter class to get the data type adapter instance 176 * for 177 * @param bindingContext 178 * the Metaschema binding context used to lookup the data type adapter 179 * @return the data type adapter 180 * @throws IllegalArgumentException 181 * if the provided adapter is not registered with the binding context 182 */ 183 @NonNull 184 public static IDataTypeAdapter<?> getDataTypeAdapter( 185 @NonNull Class<? extends IDataTypeAdapter<?>> adapterClass, 186 @NonNull IBindingContext bindingContext) { 187 IDataTypeAdapter<?> retval; 188 if (NullJavaTypeAdapter.class.equals(adapterClass)) { 189 retval = MetaschemaDataTypeProvider.DEFAULT_DATA_TYPE; 190 } else { 191 retval = bindingContext.getDataTypeAdapterInstance(adapterClass); 192 if (retval == null) { 193 throw new IllegalArgumentException("Unable to get type adapter instance for class: " + adapterClass.getName()); 194 } 195 } 196 return retval; 197 } 198 199 /** 200 * Given a provided default value string, get the data type specific default 201 * value using the provided data type adapter. 202 * <p> 203 * If the provided default value is {@link ModelUtil#NULL_VALUE}, then this 204 * method will return a {@code null} value. 205 * 206 * @param defaultValue 207 * the string representation of the default value 208 * @param adapter 209 * the data type adapter instance used to cast the default string value 210 * to a data type specific object 211 * @return the data type specific object or {@code null} if the provided default 212 * value was {@link ModelUtil#NULL_VALUE} 213 */ 214 @Nullable 215 public static Object resolveDefaultValue(@NonNull String defaultValue, IDataTypeAdapter<?> adapter) { 216 Object retval = null; 217 if (!NULL_VALUE.equals(defaultValue)) { 218 retval = adapter.parse(defaultValue); 219 } 220 return retval; 221 } 222 223 /** 224 * Resolves an integer value by determining if an actual value is provided or 225 * -2^31, which indicates that no actual value was provided. 226 * <p> 227 * The integer value -2^31 cannot be used, since this indicates no value. 228 * 229 * @param value 230 * the integer value to resolve 231 * @return the integer value or {@code null} if the provided value was -2^31 232 */ 233 public static Integer resolveDefaultInteger(int value) { 234 return value == Integer.MIN_VALUE ? null : value; 235 } 236 237 /** 238 * Resolves a {@link GroupAs} annotation determining if an actual value is 239 * provided or if the value is the default, which indicates that no actual 240 * GroupAs was provided. 241 * 242 * @param groupAs 243 * the GroupAs value to resolve 244 * @param module 245 * the containing module instance 246 * @return a new {@link IGroupAs} instance or a singleton group as if the 247 * provided value was the default value 248 */ 249 @NonNull 250 public static IGroupAs resolveDefaultGroupAs( 251 @NonNull GroupAs groupAs, 252 @NonNull IModule module) { 253 return NULL_VALUE.equals(groupAs.name()) 254 ? IGroupAs.SINGLETON_GROUP_AS 255 : new DefaultGroupAs(groupAs, module); 256 } 257 258 public static String toLocation(@NonNull IBoundObject obj) { 259 IMetaschemaData data = obj.getMetaschemaData(); 260 261 String retval = ""; 262 if (data != null) { 263 int line = data.getLine(); 264 if (line > -1) { 265 retval = line + ":" + data.getColumn(); 266 } 267 } 268 return retval; 269 } 270 271 public static String toLocation(@NonNull IBoundObject obj, @Nullable URI uri) { 272 String retval = uri == null ? "" : uri.toASCIIString(); 273 274 String location = toLocation(obj); 275 if (!location.isEmpty()) { 276 retval = retval.isEmpty() ? location : retval + "@" + location; 277 } 278 return retval; 279 } 280 281 public static Map.Entry<IAttributable.Key, Set<String>> toPropertyEntry(@NonNull Property property) { 282 String name = property.name(); 283 String namespace = property.namespace(); 284 @SuppressWarnings("PMD.AvoidInstantiatingObjectsInLoops") // ok 285 IAttributable.Key key = IAttributable.key(namespace, name); 286 287 String[] values = property.values(); 288 List<String> valueList = Arrays.asList(values); 289 @SuppressWarnings("PMD.AvoidInstantiatingObjectsInLoops") // ok 290 Set<String> valueSet = new LinkedHashSet<>(valueList); 291 292 return Map.entry(key, CollectionUtil.unmodifiableSet(valueSet)); 293 } 294}