2 * Copyright (c) 2010-2023 Contributors to the openHAB project
4 * See the NOTICE file(s) distributed with this work for additional
7 * This program and the accompanying materials are made available under the
8 * terms of the Eclipse Public License 2.0 which is available at
9 * http://www.eclipse.org/legal/epl-2.0
11 * SPDX-License-Identifier: EPL-2.0
13 package org.openhab.binding.mielecloud.internal.util;
15 import java.lang.reflect.Field;
16 import java.lang.reflect.InvocationTargetException;
17 import java.lang.reflect.Method;
19 import org.eclipse.jdt.annotation.NonNullByDefault;
20 import org.eclipse.jdt.annotation.Nullable;
23 * Utility class for reflection operations such as accessing private fields or methods.
25 * @author Björn Lange - Initial contribution
28 public final class ReflectionUtil {
29 private ReflectionUtil() {
33 * Gets a private attribute.
35 * @param object The object to get the attribute from.
36 * @param fieldName The name of the field to get.
37 * @return The obtained value.
38 * @throws SecurityException if the operation is not allowed.
39 * @throws NoSuchFieldException if no field with the given name exists.
40 * @throws IllegalAccessException if the field is enforcing Java language access control and is inaccessible.
41 * @throws IllegalArgumentException if one of the passed parameters is invalid.
43 @SuppressWarnings("unchecked")
44 public static <T> T getPrivate(Object object, String fieldName)
45 throws IllegalArgumentException, IllegalAccessException, NoSuchFieldException, SecurityException {
46 Field field = getFieldFromClassHierarchy(object.getClass(), fieldName);
47 field.setAccessible(true);
48 return (T) field.get(object);
51 private static Field getFieldFromClassHierarchy(Class<?> clazz, String fieldName)
52 throws NoSuchFieldException, SecurityException {
53 Class<?> iteratedClass = clazz;
56 return iteratedClass.getDeclaredField(fieldName);
57 } catch (NoSuchFieldException e) {
59 iteratedClass = iteratedClass.getSuperclass();
60 } while (iteratedClass != null);
61 throw new NoSuchFieldException();
65 * Sets a private attribute.
67 * @param object The object to set the attribute on.
68 * @param fieldName The name of the field to set.
69 * @param value The value to set.
70 * @throws SecurityException if the operation is not allowed.
71 * @throws NoSuchFieldException if no field with the given name exists.
72 * @throws IllegalAccessException if the field is enforcing Java language access control and is inaccessible.
73 * @throws IllegalArgumentException if one of the passed parameters is invalid.
75 public static void setPrivate(Object object, String fieldName, @Nullable Object value)
76 throws NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException {
77 Field field = object.getClass().getDeclaredField(fieldName);
78 field.setAccessible(true);
79 field.set(object, value);
83 * Invokes a private method on an object.
85 * @param object The object to invoke the method on.
86 * @param methodName The name of the method to invoke.
87 * @param parameters The parameters of the method invocation.
88 * @return The method call's return value.
89 * @throws NoSuchMethodException if no method with the given parameters or name exists.
90 * @throws SecurityException if the operation is not allowed.
91 * @throws IllegalAccessException if the method is enforcing Java language access control and is inaccessible.
92 * @throws IllegalArgumentException if one of the passed parameters is invalid.
94 public static <T> T invokePrivate(Object object, String methodName, Object... parameters)
95 throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException {
96 Class<?>[] parameterTypes = new Class[parameters.length];
97 for (int i = 0; i < parameters.length; i++) {
98 parameterTypes[i] = parameters[i].getClass();
101 return invokePrivate(object, methodName, parameterTypes, parameters);
105 * Invokes a private method on an object.
107 * @param object The object to invoke the method on.
108 * @param methodName The name of the method to invoke.
109 * @param parameterTypes The types of the parameters.
110 * @param parameters The parameters of the method invocation.
111 * @return The method call's return value.
112 * @throws NoSuchMethodException if no method with the given parameters or name exists.
113 * @throws SecurityException if the operation is not allowed.
114 * @throws IllegalAccessException if the method is enforcing Java language access control and is inaccessible.
115 * @throws IllegalArgumentException if one of the passed parameters is invalid.
117 @SuppressWarnings("unchecked")
118 public static <T> T invokePrivate(Object object, String methodName, Class<?>[] parameterTypes, Object... parameters)
119 throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException {
120 Method method = getMethodFromClassHierarchy(object.getClass(), methodName, parameterTypes);
121 method.setAccessible(true);
123 return (T) method.invoke(object, parameters);
124 } catch (InvocationTargetException e) {
125 throw new IllegalStateException(e.getCause());
129 private static Method getMethodFromClassHierarchy(Class<?> clazz, String methodName, Class<?>[] parameterTypes)
130 throws NoSuchMethodException {
131 Class<?> iteratedClass = clazz;
134 return iteratedClass.getDeclaredMethod(methodName, parameterTypes);
135 } catch (NoSuchMethodException e) {
137 iteratedClass = iteratedClass.getSuperclass();
138 } while (iteratedClass != null);
139 throw new NoSuchMethodException();