1 /* 2 * ************************************************************************************************************************************************************* 3 * 4 * TheseFoolishThings: Miscellaneous utilities 5 * http://tidalwave.it/projects/thesefoolishthings 6 * 7 * Copyright (C) 2009 - 2025 by Tidalwave s.a.s. (http://tidalwave.it) 8 * 9 * ************************************************************************************************************************************************************* 10 * 11 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. 12 * You may obtain a copy of the License at 13 * 14 * http://www.apache.org/licenses/LICENSE-2.0 15 * 16 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR 17 * CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. 18 * 19 * ************************************************************************************************************************************************************* 20 * 21 * git clone https://bitbucket.org/tidalwave/thesefoolishthings-src 22 * git clone https://github.com/tidalwave-it/thesefoolishthings-src 23 * 24 * ************************************************************************************************************************************************************* 25 */ 26 package it.tidalwave.messagebus.spi; 27 28 import java.lang.annotation.Annotation; 29 import java.lang.reflect.Method; 30 import jakarta.annotation.Nonnull; 31 import java.util.ArrayList; 32 import java.util.Collections; 33 import java.util.List; 34 35 /*************************************************************************************************************************************************************** 36 * 37 * @author Fabrizio Giudici 38 * 39 **************************************************************************************************************************************************************/ 40 public class ReflectionUtils 41 { 42 /*********************************************************************************************************************************************************** 43 * 44 **********************************************************************************************************************************************************/ 45 public static interface MethodProcessor 46 { 47 enum FilterResult { ACCEPT, IGNORE } 48 49 public FilterResult filter (@Nonnull Class<?> clazz); 50 51 public void process (@Nonnull Method method); 52 } 53 54 /*********************************************************************************************************************************************************** 55 * 56 **********************************************************************************************************************************************************/ 57 public static class MethodProcessorSupport implements MethodProcessor 58 { 59 @Override @Nonnull 60 public FilterResult filter (@Nonnull final Class<?> clazz) 61 { 62 return FilterResult.ACCEPT; 63 } 64 65 @Override 66 public void process (@Nonnull final Method method) 67 { 68 } 69 } 70 71 /*********************************************************************************************************************************************************** 72 * Navigates the hierarchy of the given object, top down, and applies the {@link MethodProcessor} to all the 73 * methods of each class, if not filtered out by the processor itself. 74 * 75 * @param object the object at the bottom of the hierarchy 76 * @param processor the processor 77 **********************************************************************************************************************************************************/ 78 public static void forEachMethodInTopDownHierarchy (@Nonnull final Object object, 79 @Nonnull final MethodProcessor processor) 80 { 81 for (final var clazz : getClassHierarchy(object.getClass())) 82 { 83 for (final var method : clazz.getDeclaredMethods()) 84 { 85 processor.process(method); 86 } 87 } 88 } 89 90 /*********************************************************************************************************************************************************** 91 * Navigates the hierarchy of the given object, bottom up, and applies the {@link MethodProcessor} to all the 92 * methods of each class, if not filtered out by the processor itself. 93 * 94 * @param object the object at the bottom of the hierarchy 95 * @param processor the processor 96 **********************************************************************************************************************************************************/ 97 public static void forEachMethodInBottomUpHierarchy (@Nonnull final Object object, 98 @Nonnull final MethodProcessor processor) 99 { 100 final var hierarchy = getClassHierarchy(object.getClass()); 101 Collections.reverse(hierarchy); 102 103 for (final var clazz : hierarchy) 104 { 105 if (processor.filter(clazz) == MethodProcessor.FilterResult.ACCEPT) 106 { 107 for (final var method : clazz.getDeclaredMethods()) 108 { 109 processor.process(method); 110 } 111 } 112 } 113 } 114 115 /*********************************************************************************************************************************************************** 116 * Returns the hierarchy of the given class, top down. 117 * 118 * @param clazz the clazz at the bottom of the hierarchy 119 * @return the hierarchy 120 **********************************************************************************************************************************************************/ 121 @Nonnull 122 private static List<Class<?>> getClassHierarchy (@Nonnull final Class<?> clazz) 123 { 124 final List<Class<?>> hierarchy = new ArrayList<>(); 125 126 for (var ancestor = clazz; ancestor != null; ancestor = ancestor.getSuperclass()) 127 { 128 hierarchy.add(0, ancestor); 129 } 130 131 return hierarchy; 132 } 133 134 /*********************************************************************************************************************************************************** 135 * Checks whether an array of annotations contains an annotation of the given type. 136 * 137 * @param annotations the annotations to check 138 * @param annotationClass the type of the required annotation 139 * @return true if the annotation is found 140 **********************************************************************************************************************************************************/ 141 public static boolean containsAnnotation (@Nonnull final Annotation[] annotations, 142 @Nonnull final Class<?> annotationClass) 143 { 144 for (final var annotation : annotations) 145 { 146 if (annotationClass.isAssignableFrom(annotation.getClass())) 147 { 148 return true; 149 } 150 } 151 152 return false; 153 } 154 }