001/*
002 * #%L
003 * HAPI FHIR - Core Library
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 * http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.util;
021
022import ca.uhn.fhir.context.ConfigurationException;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
025import jakarta.annotation.Nonnull;
026import org.apache.commons.lang3.Validate;
027
028import java.lang.reflect.Constructor;
029import java.lang.reflect.Field;
030import java.lang.reflect.InvocationTargetException;
031import java.lang.reflect.Method;
032import java.lang.reflect.Modifier;
033import java.lang.reflect.ParameterizedType;
034import java.lang.reflect.Type;
035import java.lang.reflect.TypeVariable;
036import java.lang.reflect.WildcardType;
037import java.util.ArrayList;
038import java.util.Arrays;
039import java.util.Comparator;
040import java.util.HashMap;
041import java.util.List;
042import java.util.concurrent.ConcurrentHashMap;
043
044public class ReflectionUtil {
045
046        public static final Object[] EMPTY_OBJECT_ARRAY = new Object[0];
047        public static final Class<?>[] EMPTY_CLASS_ARRAY = new Class[0];
048        private static final ConcurrentHashMap<String, Object> ourFhirServerVersions = new ConcurrentHashMap<>();
049        private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(ReflectionUtil.class);
050
051        /**
052         * Non instantiable
053         */
054        private ReflectionUtil() {
055                super();
056        }
057
058        /**
059         * Returns all methods declared against {@literal theClazz}. This method returns a predictable order, which is
060         * sorted by method name and then by parameters.
061         * <p>
062         * This method does not include superclass methods (see {@link #getDeclaredMethods(Class, boolean)} if you
063         * want to include those.
064         * </p>
065         */
066        public static List<Method> getDeclaredMethods(Class<?> theClazz) {
067                return getDeclaredMethods(theClazz, false);
068        }
069
070        /**
071         * Returns all methods declared against {@literal theClazz}. This method returns a predictable order, which is
072         * sorted by method name and then by parameters.
073         */
074        public static List<Method> getDeclaredMethods(Class<?> theClazz, boolean theIncludeMethodsFromSuperclasses) {
075                HashMap<String, Method> foundMethods = new HashMap<>();
076
077                populateDeclaredMethodsMap(theClazz, foundMethods, theIncludeMethodsFromSuperclasses);
078
079                List<Method> retVal = new ArrayList<>(foundMethods.values());
080                retVal.sort((Comparator.comparing(ReflectionUtil::describeMethodInSortFriendlyWay)));
081                return retVal;
082        }
083
084        private static void populateDeclaredMethodsMap(
085                        Class<?> theClazz, HashMap<String, Method> foundMethods, boolean theIncludeMethodsFromSuperclasses) {
086                Method[] declaredMethods = theClazz.getDeclaredMethods();
087                for (Method next : declaredMethods) {
088
089                        if (Modifier.isAbstract(next.getModifiers())
090                                        || Modifier.isStatic(next.getModifiers())
091                                        || Modifier.isPrivate(next.getModifiers())) {
092                                continue;
093                        }
094
095                        String description = next.getName() + Arrays.asList(next.getParameterTypes());
096
097                        if (!foundMethods.containsKey(description)) {
098                                try {
099                                        Method method = theClazz.getMethod(next.getName(), next.getParameterTypes());
100                                        foundMethods.put(description, method);
101                                } catch (NoSuchMethodException | SecurityException e) {
102                                        foundMethods.put(description, next);
103                                }
104                        }
105                }
106
107                if (theIncludeMethodsFromSuperclasses && !theClazz.getSuperclass().equals(Object.class)) {
108                        populateDeclaredMethodsMap(theClazz.getSuperclass(), foundMethods, theIncludeMethodsFromSuperclasses);
109                }
110        }
111
112        /**
113         * Returns a description like <code>startsWith params(java.lang.String, int) returns(boolean)</code>.
114         * The format is chosen in order to provide a predictable and useful sorting order.
115         */
116        public static String describeMethodInSortFriendlyWay(Method theMethod) {
117                StringBuilder b = new StringBuilder();
118                b.append(theMethod.getName());
119                b.append(" returns(");
120                b.append(theMethod.getReturnType().getName());
121                b.append(") params(");
122                Class<?>[] parameterTypes = theMethod.getParameterTypes();
123                for (int i = 0, parameterTypesLength = parameterTypes.length; i < parameterTypesLength; i++) {
124                        if (i > 0) {
125                                b.append(", ");
126                        }
127                        Class<?> next = parameterTypes[i];
128                        b.append(next.getName());
129                }
130                b.append(")");
131                return b.toString();
132        }
133
134        public static Class<?> getGenericCollectionTypeOfField(Field next) {
135                ParameterizedType collectionType = (ParameterizedType) next.getGenericType();
136                return getGenericCollectionTypeOf(collectionType.getActualTypeArguments()[0]);
137        }
138
139        /**
140         * For a field of type List<Enumeration<Foo>>, returns Foo
141         */
142        public static Class<?> getGenericCollectionTypeOfFieldWithSecondOrderForList(Field next) {
143                if (!List.class.isAssignableFrom(next.getType())) {
144                        return getGenericCollectionTypeOfField(next);
145                }
146
147                Class<?> type;
148                ParameterizedType collectionType = (ParameterizedType) next.getGenericType();
149                Type firstArg = collectionType.getActualTypeArguments()[0];
150                if (ParameterizedType.class.isAssignableFrom(firstArg.getClass())) {
151                        ParameterizedType pt = ((ParameterizedType) firstArg);
152                        Type pt2 = pt.getActualTypeArguments()[0];
153                        return (Class<?>) pt2;
154                }
155                type = (Class<?>) firstArg;
156                return type;
157        }
158
159        public static Class<?> getGenericCollectionTypeOfMethodParameter(Method theMethod, int theParamIndex) {
160                Type genericParameterType = theMethod.getGenericParameterTypes()[theParamIndex];
161                if (Class.class.equals(genericParameterType) || Class.class.equals(genericParameterType.getClass())) {
162                        return null;
163                }
164                ParameterizedType collectionType = (ParameterizedType) genericParameterType;
165                return getGenericCollectionTypeOf(collectionType.getActualTypeArguments()[0]);
166        }
167
168        public static Class<?> getGenericCollectionTypeOfMethodReturnType(Method theMethod) {
169                Type genericReturnType = theMethod.getGenericReturnType();
170                if (!(genericReturnType instanceof ParameterizedType)) {
171                        return null;
172                }
173                ParameterizedType collectionType = (ParameterizedType) genericReturnType;
174                return getGenericCollectionTypeOf(collectionType.getActualTypeArguments()[0]);
175        }
176
177        @SuppressWarnings({"rawtypes"})
178        private static Class<?> getGenericCollectionTypeOf(Type theType) {
179                Class<?> type;
180                if (ParameterizedType.class.isAssignableFrom(theType.getClass())) {
181                        ParameterizedType pt = ((ParameterizedType) theType);
182                        type = (Class<?>) pt.getRawType();
183                } else if (theType instanceof TypeVariable<?>) {
184                        Type decl = ((TypeVariable) theType).getBounds()[0];
185                        return (Class<?>) decl;
186                } else if (theType instanceof WildcardType) {
187                        Type decl = ((WildcardType) theType).getUpperBounds()[0];
188                        return (Class<?>) decl;
189                } else {
190                        type = (Class<?>) theType;
191                }
192                return type;
193        }
194
195        public static boolean isInstantiable(Class<?> theType) {
196                return !theType.isInterface() && !Modifier.isAbstract(theType.getModifiers());
197        }
198
199        /**
200         * Instantiate a class by no-arg constructor, throw {@link ConfigurationException} if we fail to do so
201         */
202        public static <T> T newInstance(Class<T> theType) {
203                Validate.notNull(theType, "theType must not be null");
204                try {
205                        return theType.getConstructor().newInstance();
206                } catch (Exception e) {
207                        throw new ConfigurationException(Msg.code(1784) + "Failed to instantiate " + theType.getName(), e);
208                }
209        }
210
211        public static <T> T newInstance(Class<T> theType, Class<?> theArgumentType, Object theArgument) {
212                Validate.notNull(theType, "theType must not be null");
213                try {
214                        Constructor<T> constructor = theType.getConstructor(theArgumentType);
215                        return constructor.newInstance(theArgument);
216                } catch (Exception e) {
217                        throw new ConfigurationException(Msg.code(1785) + "Failed to instantiate " + theType.getName(), e);
218                }
219        }
220
221        public static Object newInstanceOfFhirServerType(String theType) {
222                String errorMessage =
223                                "Unable to instantiate server framework. Please make sure that hapi-fhir-server library is on your classpath!";
224                String wantedType = "ca.uhn.fhir.rest.api.server.IFhirVersionServer";
225                return newInstanceOfType(theType, theType, errorMessage, wantedType, new Class[0], new Object[0]);
226        }
227
228        private static Object newInstanceOfType(
229                        String theKey,
230                        String theType,
231                        String errorMessage,
232                        String wantedType,
233                        Class<?>[] theParameterArgTypes,
234                        Object[] theConstructorArgs) {
235                Object fhirServerVersion = ourFhirServerVersions.get(theKey);
236                if (fhirServerVersion == null) {
237                        try {
238                                Class<?> type = Class.forName(theType);
239                                Class<?> serverType = Class.forName(wantedType);
240                                Validate.isTrue(serverType.isAssignableFrom(type));
241                                fhirServerVersion = type.getConstructor(theParameterArgTypes).newInstance(theConstructorArgs);
242                        } catch (Exception e) {
243                                throw new ConfigurationException(Msg.code(1786) + errorMessage, e);
244                        }
245
246                        ourFhirServerVersions.put(theKey, fhirServerVersion);
247                }
248                return fhirServerVersion;
249        }
250
251        public static <T> T newInstanceOrReturnNull(String theClassName, Class<T> theType) {
252                return newInstanceOrReturnNull(theClassName, theType, EMPTY_CLASS_ARRAY, EMPTY_OBJECT_ARRAY);
253        }
254
255        @SuppressWarnings("unchecked")
256        public static <T> T newInstanceOrReturnNull(
257                        String theClassName, Class<T> theType, Class<?>[] theArgTypes, Object[] theArgs) {
258                try {
259                        return newInstance(theClassName, theType, theArgTypes, theArgs);
260                } catch (ConfigurationException e) {
261                        throw e;
262                } catch (Exception e) {
263                        ourLog.info("Failed to instantiate {}: {}", theClassName, e.toString());
264                        return null;
265                }
266        }
267
268        @Nonnull
269        public static <T> T newInstance(String theClassName, Class<T> theType, Class<?>[] theArgTypes, Object[] theArgs) {
270                try {
271                        Class<?> clazz = Class.forName(theClassName);
272                        if (!theType.isAssignableFrom(clazz)) {
273                                throw new ConfigurationException(Msg.code(1787) + theClassName + " is not assignable to " + theType);
274                        }
275                        return (T) clazz.getConstructor(theArgTypes).newInstance(theArgs);
276                } catch (ClassNotFoundException
277                                | NoSuchMethodException
278                                | InstantiationException
279                                | IllegalAccessException
280                                | InvocationTargetException e) {
281                        throw new InternalErrorException(Msg.code(2330) + e.getMessage(), e);
282                }
283        }
284
285        public static boolean typeExists(String theName) {
286                try {
287                        Class.forName(theName);
288                        return true;
289                } catch (ClassNotFoundException theE) {
290                        return false;
291                }
292        }
293}