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.interceptor.executor;
021
022import ca.uhn.fhir.i18n.Msg;
023import ca.uhn.fhir.interceptor.api.Hook;
024import ca.uhn.fhir.interceptor.api.HookParams;
025import ca.uhn.fhir.interceptor.api.IBaseInterceptorBroadcaster;
026import ca.uhn.fhir.interceptor.api.IBaseInterceptorService;
027import ca.uhn.fhir.interceptor.api.IPointcut;
028import ca.uhn.fhir.interceptor.api.Interceptor;
029import ca.uhn.fhir.interceptor.api.Pointcut;
030import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
031import ca.uhn.fhir.util.ReflectionUtil;
032import com.google.common.annotations.VisibleForTesting;
033import com.google.common.collect.ArrayListMultimap;
034import com.google.common.collect.ListMultimap;
035import io.opentelemetry.api.common.AttributeKey;
036import io.opentelemetry.api.trace.Span;
037import io.opentelemetry.instrumentation.annotations.WithSpan;
038import jakarta.annotation.Nonnull;
039import jakarta.annotation.Nullable;
040import org.apache.commons.lang3.Validate;
041import org.apache.commons.lang3.builder.ToStringBuilder;
042import org.apache.commons.lang3.builder.ToStringStyle;
043import org.apache.commons.lang3.reflect.MethodUtils;
044import org.slf4j.Logger;
045import org.slf4j.LoggerFactory;
046
047import java.lang.annotation.Annotation;
048import java.lang.reflect.AnnotatedElement;
049import java.lang.reflect.InvocationTargetException;
050import java.lang.reflect.Method;
051import java.util.ArrayList;
052import java.util.Arrays;
053import java.util.Collection;
054import java.util.Collections;
055import java.util.Comparator;
056import java.util.EnumSet;
057import java.util.HashMap;
058import java.util.IdentityHashMap;
059import java.util.List;
060import java.util.Map;
061import java.util.Optional;
062import java.util.concurrent.atomic.AtomicInteger;
063import java.util.function.Predicate;
064import java.util.stream.Collectors;
065
066import static org.apache.commons.lang3.ObjectUtils.defaultIfNull;
067
068public abstract class BaseInterceptorService<POINTCUT extends Enum<POINTCUT> & IPointcut>
069                implements IBaseInterceptorService<POINTCUT>, IBaseInterceptorBroadcaster<POINTCUT> {
070        private static final Logger ourLog = LoggerFactory.getLogger(BaseInterceptorService.class);
071        private static final AttributeKey<String> OTEL_INTERCEPTOR_POINTCUT_NAME_ATT_KEY =
072                        AttributeKey.stringKey("hapifhir.interceptor.pointcut_name");
073        private static final AttributeKey<String> OTEL_INTERCEPTOR_CLASS_NAME_ATT_KEY =
074                        AttributeKey.stringKey("hapifhir.interceptor.class_name");
075        private static final AttributeKey<String> OTEL_INTERCEPTOR_METHOD_NAME_ATT_KEY =
076                        AttributeKey.stringKey("hapifhir.interceptor.method_name");
077
078        private final List<Object> myInterceptors = new ArrayList<>();
079        private final ListMultimap<POINTCUT, IInvoker> myGlobalInvokers = ArrayListMultimap.create();
080        private final ListMultimap<POINTCUT, IInvoker> myAnonymousInvokers = ArrayListMultimap.create();
081        private final Object myRegistryMutex = new Object();
082        private final Class<POINTCUT> myPointcutType;
083        private volatile EnumSet<POINTCUT> myRegisteredPointcuts;
084        private boolean myWarnOnInterceptorWithNoHooks = true;
085
086        /**
087         * Constructor which uses a default name of "default"
088         */
089        public BaseInterceptorService(Class<POINTCUT> thePointcutType) {
090                this(thePointcutType, "default");
091        }
092
093        /**
094         * Constructor
095         *
096         * @param theName The name for this registry (useful for troubleshooting)
097         * @deprecated The name parameter is not used for anything
098         */
099        @Deprecated(since = "8.0.0", forRemoval = true)
100        public BaseInterceptorService(Class<POINTCUT> thePointcutType, String theName) {
101                super();
102                myPointcutType = thePointcutType;
103                rebuildRegisteredPointcutSet();
104        }
105
106        /**
107         * Should a warning be issued if an interceptor is registered and it has no hooks
108         */
109        public void setWarnOnInterceptorWithNoHooks(boolean theWarnOnInterceptorWithNoHooks) {
110                myWarnOnInterceptorWithNoHooks = theWarnOnInterceptorWithNoHooks;
111        }
112
113        @VisibleForTesting
114        List<Object> getGlobalInterceptorsForUnitTest() {
115                return myInterceptors;
116        }
117
118        /**
119         * @deprecated This value is not used anywhere
120         */
121        @Deprecated(since = "8.0.0", forRemoval = true)
122        public void setName(@SuppressWarnings("unused") String theName) {
123                // nothing
124        }
125
126        protected void registerAnonymousInterceptor(POINTCUT thePointcut, Object theInterceptor, BaseInvoker theInvoker) {
127                Validate.notNull(thePointcut, "thePointcut must not be null");
128                Validate.notNull(theInterceptor, "theInterceptor must not be null");
129                synchronized (myRegistryMutex) {
130                        myAnonymousInvokers.put(thePointcut, theInvoker);
131                        if (!isInterceptorAlreadyRegistered(theInterceptor)) {
132                                myInterceptors.add(theInterceptor);
133                        }
134
135                        rebuildRegisteredPointcutSet();
136                }
137        }
138
139        @Override
140        public List<Object> getAllRegisteredInterceptors() {
141                synchronized (myRegistryMutex) {
142                        List<Object> retVal = new ArrayList<>(myInterceptors);
143                        return Collections.unmodifiableList(retVal);
144                }
145        }
146
147        @Override
148        @VisibleForTesting
149        public void unregisterAllInterceptors() {
150                synchronized (myRegistryMutex) {
151                        unregisterInterceptors(myAnonymousInvokers.values());
152                        unregisterInterceptors(myGlobalInvokers.values());
153                        unregisterInterceptors(myInterceptors);
154                }
155        }
156
157        @Override
158        public void unregisterInterceptors(@Nullable Collection<?> theInterceptors) {
159                if (theInterceptors != null) {
160                        // We construct a new list before iterating because the service's internal
161                        // interceptor lists get passed into this method, and we get concurrent
162                        // modification errors if we modify them at the same time as we iterate them
163                        new ArrayList<>(theInterceptors).forEach(this::unregisterInterceptor);
164                }
165        }
166
167        @Override
168        public void registerInterceptors(@Nullable Collection<?> theInterceptors) {
169                if (theInterceptors != null) {
170                        theInterceptors.forEach(this::registerInterceptor);
171                }
172        }
173
174        @Override
175        public void unregisterAllAnonymousInterceptors() {
176                synchronized (myRegistryMutex) {
177                        unregisterInterceptorsIf(t -> true, myAnonymousInvokers);
178                }
179        }
180
181        @Override
182        public void unregisterInterceptorsIf(Predicate<Object> theShouldUnregisterFunction) {
183                unregisterInterceptorsIf(theShouldUnregisterFunction, myGlobalInvokers);
184                unregisterInterceptorsIf(theShouldUnregisterFunction, myAnonymousInvokers);
185        }
186
187        private void unregisterInterceptorsIf(
188                        Predicate<Object> theShouldUnregisterFunction, ListMultimap<POINTCUT, IInvoker> theGlobalInvokers) {
189                synchronized (myRegistryMutex) {
190                        for (Map.Entry<POINTCUT, IInvoker> nextInvoker : new ArrayList<>(theGlobalInvokers.entries())) {
191                                if (theShouldUnregisterFunction.test(nextInvoker.getValue().getInterceptor())) {
192                                        unregisterInterceptor(nextInvoker.getValue().getInterceptor());
193                                }
194                        }
195
196                        rebuildRegisteredPointcutSet();
197                }
198        }
199
200        @Override
201        public boolean registerInterceptor(Object theInterceptor) {
202                synchronized (myRegistryMutex) {
203                        if (isInterceptorAlreadyRegistered(theInterceptor)) {
204                                return false;
205                        }
206
207                        List<HookInvoker> addedInvokers = scanInterceptorAndAddToInvokerMultimap(theInterceptor, myGlobalInvokers);
208                        if (addedInvokers.isEmpty()) {
209                                if (myWarnOnInterceptorWithNoHooks) {
210                                        ourLog.warn(
211                                                        "Interceptor registered with no valid hooks - Type was: {}",
212                                                        theInterceptor.getClass().getName());
213                                }
214                                return false;
215                        }
216
217                        // Add to the global list
218                        myInterceptors.add(theInterceptor);
219                        sortByOrderAnnotation(myInterceptors);
220
221                        rebuildRegisteredPointcutSet();
222
223                        return true;
224                }
225        }
226
227        private void rebuildRegisteredPointcutSet() {
228                EnumSet<POINTCUT> registeredPointcuts = EnumSet.noneOf(myPointcutType);
229                registeredPointcuts.addAll(myAnonymousInvokers.keySet());
230                registeredPointcuts.addAll(myGlobalInvokers.keySet());
231                myRegisteredPointcuts = registeredPointcuts;
232        }
233
234        private boolean isInterceptorAlreadyRegistered(Object theInterceptor) {
235                for (Object next : myInterceptors) {
236                        if (next == theInterceptor) {
237                                return true;
238                        }
239                }
240                return false;
241        }
242
243        @Override
244        public boolean unregisterInterceptor(Object theInterceptor) {
245                synchronized (myRegistryMutex) {
246                        boolean removed = myInterceptors.removeIf(t -> t == theInterceptor);
247                        removed |= myGlobalInvokers.entries().removeIf(t -> t.getValue().getInterceptor() == theInterceptor);
248                        removed |= myAnonymousInvokers.entries().removeIf(t -> t.getValue().getInterceptor() == theInterceptor);
249                        rebuildRegisteredPointcutSet();
250                        return removed;
251                }
252        }
253
254        private void sortByOrderAnnotation(List<Object> theObjects) {
255                IdentityHashMap<Object, Integer> interceptorToOrder = new IdentityHashMap<>();
256                for (Object next : theObjects) {
257                        Interceptor orderAnnotation = next.getClass().getAnnotation(Interceptor.class);
258                        int order = orderAnnotation != null ? orderAnnotation.order() : 0;
259                        interceptorToOrder.put(next, order);
260                }
261
262                theObjects.sort((a, b) -> {
263                        Integer orderA = interceptorToOrder.get(a);
264                        Integer orderB = interceptorToOrder.get(b);
265                        return orderA - orderB;
266                });
267        }
268
269        @Override
270        public Object callHooksAndReturnObject(POINTCUT thePointcut, HookParams theParams) {
271                assert haveAppropriateParams(thePointcut, theParams);
272                assert thePointcut.getReturnType() != void.class;
273
274                return doCallHooks(thePointcut, theParams);
275        }
276
277        @Override
278        public boolean hasHooks(POINTCUT thePointcut) {
279                return myRegisteredPointcuts.contains(thePointcut);
280        }
281
282        protected Class<?> getBooleanReturnType() {
283                return boolean.class;
284        }
285
286        @Override
287        public boolean callHooks(POINTCUT thePointcut, HookParams theParams) {
288                assert haveAppropriateParams(thePointcut, theParams);
289                assert thePointcut.getReturnType() == void.class || thePointcut.getReturnType() == getBooleanReturnType();
290
291                Object retValObj = doCallHooks(thePointcut, theParams);
292                retValObj = defaultIfNull(retValObj, true);
293                return (Boolean) retValObj;
294        }
295
296        private Object doCallHooks(POINTCUT thePointcut, HookParams theParams) {
297                List<IInvoker> invokers = getInvokersForPointcut(thePointcut);
298                return callInvokers(thePointcut, theParams, invokers);
299        }
300
301        @VisibleForTesting
302        List<Object> getInterceptorsWithInvokersForPointcut(POINTCUT thePointcut) {
303                return getInvokersForPointcut(thePointcut).stream()
304                                .map(IInvoker::getInterceptor)
305                                .collect(Collectors.toList());
306        }
307
308        /**
309         * Returns a list of all invokers registered for the given pointcut. The list
310         * is ordered by the invoker order (specified on the {@link Interceptor#order()}
311         * and {@link Hook#order()} values.
312         *
313         * @return The list returned by this method will always be a newly created list, so it will be stable and can be modified.
314         */
315        @Override
316        public List<IInvoker> getInvokersForPointcut(POINTCUT thePointcut) {
317                List<IInvoker> invokers;
318
319                synchronized (myRegistryMutex) {
320                        List<IInvoker> globalInvokers = myGlobalInvokers.get(thePointcut);
321                        List<IInvoker> anonymousInvokers = myAnonymousInvokers.get(thePointcut);
322                        invokers = union(Arrays.asList(globalInvokers, anonymousInvokers));
323                }
324
325                return invokers;
326        }
327
328        /**
329         * Only call this when assertions are enabled, it's expensive
330         */
331        public static boolean haveAppropriateParams(IPointcut thePointcut, HookParams theParams) {
332                if (theParams.getParamsForType().values().size()
333                                != thePointcut.getParameterTypes().size()) {
334                        throw new IllegalArgumentException(Msg.code(1909)
335                                        + String.format(
336                                                        "Wrong number of params for pointcut %s - Wanted %s but found %s",
337                                                        thePointcut.name(),
338                                                        toErrorString(thePointcut.getParameterTypes()),
339                                                        theParams.getParamsForType().values().stream()
340                                                                        .map(t -> t != null ? t.getClass().getSimpleName() : "null")
341                                                                        .sorted()
342                                                                        .collect(Collectors.toList())));
343                }
344
345                List<String> wantedTypes = new ArrayList<>(thePointcut.getParameterTypes());
346
347                ListMultimap<Class<?>, Object> givenTypes = theParams.getParamsForType();
348                for (Class<?> nextTypeClass : givenTypes.keySet()) {
349                        String nextTypeName = nextTypeClass.getName();
350                        for (Object nextParamValue : givenTypes.get(nextTypeClass)) {
351                                Validate.isTrue(
352                                                nextParamValue == null || nextTypeClass.isAssignableFrom(nextParamValue.getClass()),
353                                                "Invalid params for pointcut %s - %s is not of type %s",
354                                                thePointcut.name(),
355                                                nextParamValue != null ? nextParamValue.getClass() : "null",
356                                                nextTypeClass);
357                                Validate.isTrue(
358                                                wantedTypes.remove(nextTypeName),
359                                                "Invalid params for pointcut %s - Wanted %s but found %s",
360                                                thePointcut.name(),
361                                                toErrorString(thePointcut.getParameterTypes()),
362                                                nextTypeName);
363                        }
364                }
365
366                return true;
367        }
368
369        private List<HookInvoker> scanInterceptorAndAddToInvokerMultimap(
370                        Object theInterceptor, ListMultimap<POINTCUT, IInvoker> theInvokers) {
371                Class<?> interceptorClass = theInterceptor.getClass();
372                int typeOrder = determineOrder(interceptorClass);
373
374                List<HookInvoker> addedInvokers = scanInterceptorForHookMethods(theInterceptor, typeOrder);
375
376                // Invoke the REGISTERED pointcut for any added hooks
377                addedInvokers.stream()
378                                .filter(t -> Pointcut.INTERCEPTOR_REGISTERED.equals(t.getPointcut()))
379                                .forEach(t -> t.invoke(new HookParams()));
380
381                // Register the interceptor and its various hooks
382                for (HookInvoker nextAddedHook : addedInvokers) {
383                        POINTCUT nextPointcut = nextAddedHook.getPointcut();
384                        if (nextPointcut.equals(Pointcut.INTERCEPTOR_REGISTERED)) {
385                                continue;
386                        }
387                        theInvokers.put(nextPointcut, nextAddedHook);
388                }
389
390                // Make sure we're always sorted according to the order declared in @Order
391                for (POINTCUT nextPointcut : theInvokers.keys()) {
392                        List<IInvoker> nextInvokerList = theInvokers.get(nextPointcut);
393                        nextInvokerList.sort(Comparator.naturalOrder());
394                }
395
396                return addedInvokers;
397        }
398
399        /**
400         * @return Returns a list of any added invokers
401         */
402        private List<HookInvoker> scanInterceptorForHookMethods(Object theInterceptor, int theTypeOrder) {
403                ArrayList<HookInvoker> retVal = new ArrayList<>();
404                for (Method nextMethod : ReflectionUtil.getDeclaredMethods(theInterceptor.getClass(), true)) {
405                        Optional<HookDescriptor> hook = scanForHook(nextMethod);
406
407                        if (hook.isPresent()) {
408                                int methodOrder = theTypeOrder;
409                                int methodOrderAnnotation = hook.get().getOrder();
410                                if (methodOrderAnnotation != Interceptor.DEFAULT_ORDER) {
411                                        methodOrder = methodOrderAnnotation;
412                                }
413
414                                retVal.add(new HookInvoker(hook.get(), theInterceptor, nextMethod, methodOrder));
415                        }
416                }
417
418                return retVal;
419        }
420
421        protected abstract Optional<HookDescriptor> scanForHook(Method nextMethod);
422
423        public static Object callInvokers(IPointcut thePointcut, HookParams theParams, List<IInvoker> invokers) {
424
425                Object retVal = null;
426
427                /*
428                 * Call each hook in order
429                 */
430                for (IInvoker nextInvoker : invokers) {
431                        Object nextOutcome = nextInvoker.invoke(theParams);
432                        Class<?> pointcutReturnType = thePointcut.getReturnType();
433                        if (pointcutReturnType.equals(thePointcut.getBooleanReturnTypeForEnum())) {
434                                Boolean nextOutcomeAsBoolean = (Boolean) nextOutcome;
435                                if (Boolean.FALSE.equals(nextOutcomeAsBoolean)) {
436                                        ourLog.trace("callHooks({}) for invoker({}) returned false", thePointcut, nextInvoker);
437                                        retVal = false;
438                                        break;
439                                } else {
440                                        retVal = true;
441                                }
442                        } else if (!pointcutReturnType.equals(void.class)) {
443                                if (nextOutcome != null) {
444                                        retVal = nextOutcome;
445                                        break;
446                                }
447                        }
448                }
449
450                return retVal;
451        }
452
453        /**
454         * First argument must be the global invoker list!!
455         */
456        public static List<IInvoker> union(List<List<IInvoker>> theInvokersLists) {
457                List<IInvoker> haveOne = null;
458                boolean haveMultiple = false;
459                for (List<IInvoker> nextInvokerList : theInvokersLists) {
460                        if (nextInvokerList == null || nextInvokerList.isEmpty()) {
461                                continue;
462                        }
463
464                        if (haveOne == null) {
465                                haveOne = nextInvokerList;
466                        } else {
467                                haveMultiple = true;
468                        }
469                }
470
471                if (haveOne == null) {
472                        return Collections.emptyList();
473                }
474
475                List<IInvoker> retVal;
476
477                if (!haveMultiple) {
478
479                        // The global list doesn't need to be sorted every time since it's sorted on
480                        // insertion each time. Doing so is a waste of cycles..
481                        if (haveOne == theInvokersLists.get(0)) {
482                                retVal = haveOne;
483                        } else {
484                                retVal = new ArrayList<>(haveOne);
485                                retVal.sort(Comparator.naturalOrder());
486                        }
487
488                } else {
489
490                        int totalSize = 0;
491                        for (List<IInvoker> list : theInvokersLists) {
492                                totalSize += list.size();
493                        }
494                        retVal = new ArrayList<>(totalSize);
495                        for (List<IInvoker> list : theInvokersLists) {
496                                retVal.addAll(list);
497                        }
498                        retVal.sort(Comparator.naturalOrder());
499                }
500
501                return retVal;
502        }
503
504        protected static <T extends Annotation> Optional<T> findAnnotation(
505                        AnnotatedElement theObject, Class<T> theHookClass) {
506                T annotation;
507                if (theObject instanceof Method) {
508                        annotation = MethodUtils.getAnnotation((Method) theObject, theHookClass, true, true);
509                } else {
510                        annotation = theObject.getAnnotation(theHookClass);
511                }
512                return Optional.ofNullable(annotation);
513        }
514
515        private static int determineOrder(Class<?> theInterceptorClass) {
516                return findAnnotation(theInterceptorClass, Interceptor.class)
517                                .map(Interceptor::order)
518                                .orElse(Interceptor.DEFAULT_ORDER);
519        }
520
521        private static String toErrorString(List<String> theParameterTypes) {
522                return theParameterTypes.stream().sorted().collect(Collectors.joining(","));
523        }
524
525        private class HookInvoker extends BaseInvoker {
526
527                private final Method myMethod;
528                private final Class<?>[] myParameterTypes;
529                private final int[] myParameterIndexes;
530                private final POINTCUT myPointcut;
531
532                /**
533                 * Constructor
534                 */
535                private HookInvoker(
536                                HookDescriptor theHook, @Nonnull Object theInterceptor, @Nonnull Method theHookMethod, int theOrder) {
537                        super(theInterceptor, theOrder);
538                        myPointcut = theHook.getPointcut();
539                        myParameterTypes = theHookMethod.getParameterTypes();
540                        myMethod = theHookMethod;
541
542                        Class<?> returnType = theHookMethod.getReturnType();
543                        if (myPointcut.getReturnType().equals(myPointcut.getBooleanReturnTypeForEnum())) {
544                                Validate.isTrue(
545                                                myPointcut.getBooleanReturnTypeForEnum().equals(returnType) || void.class.equals(returnType),
546                                                "Method does not return %s or void: %s",
547                                                myPointcut.getBooleanReturnTypeForEnum().getSimpleName(),
548                                                theHookMethod);
549                        } else if (myPointcut.getReturnType().equals(void.class)) {
550                                Validate.isTrue(void.class.equals(returnType), "Method does not return void: %s", theHookMethod);
551                        } else {
552                                Validate.isTrue(
553                                                myPointcut.getReturnType().isAssignableFrom(returnType) || void.class.equals(returnType),
554                                                "Method does not return %s or void: %s",
555                                                myPointcut.getReturnType(),
556                                                theHookMethod);
557                        }
558
559                        myParameterIndexes = new int[myParameterTypes.length];
560                        Map<Class<?>, AtomicInteger> typeToCount = new HashMap<>();
561                        for (int i = 0; i < myParameterTypes.length; i++) {
562                                AtomicInteger counter = typeToCount.computeIfAbsent(myParameterTypes[i], t -> new AtomicInteger(0));
563                                myParameterIndexes[i] = counter.getAndIncrement();
564                        }
565
566                        myMethod.setAccessible(true);
567                }
568
569                @Override
570                public String toString() {
571                        return new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE)
572                                        .append("method", myMethod)
573                                        .toString();
574                }
575
576                public POINTCUT getPointcut() {
577                        return myPointcut;
578                }
579
580                /**
581                 * @return Returns true/false if the hook method returns a boolean, returns true otherwise
582                 */
583                @Override
584                public Object invoke(HookParams theParams) {
585
586                        Object[] args = new Object[myParameterTypes.length];
587                        for (int i = 0; i < myParameterTypes.length; i++) {
588                                Class<?> nextParamType = myParameterTypes[i];
589                                if (nextParamType.equals(Pointcut.class)) {
590                                        args[i] = myPointcut;
591                                } else {
592                                        int nextParamIndex = myParameterIndexes[i];
593                                        Object nextParamValue = theParams.get(nextParamType, nextParamIndex);
594                                        args[i] = nextParamValue;
595                                }
596                        }
597
598                        // Invoke the method
599                        try {
600                                return invokeMethod(args);
601                        } catch (InvocationTargetException e) {
602                                Throwable targetException = e.getTargetException();
603                                if (myPointcut.isShouldLogAndSwallowException(targetException)) {
604                                        ourLog.error("Exception thrown by interceptor: " + targetException.toString(), targetException);
605                                        return null;
606                                }
607
608                                if (targetException instanceof RuntimeException) {
609                                        throw ((RuntimeException) targetException);
610                                } else {
611                                        throw new InternalErrorException(
612                                                        Msg.code(1910) + "Failure invoking interceptor for pointcut(s) " + getPointcut(),
613                                                        targetException);
614                                }
615                        } catch (Exception e) {
616                                throw new InternalErrorException(Msg.code(1911) + e);
617                        }
618                }
619
620                @WithSpan("hapifhir.interceptor")
621                private Object invokeMethod(Object[] args) throws InvocationTargetException, IllegalAccessException {
622                        // Add attributes to the opentelemetry span
623                        Span currentSpan = Span.current();
624                        currentSpan.setAttribute(OTEL_INTERCEPTOR_POINTCUT_NAME_ATT_KEY, myPointcut.name());
625                        currentSpan.setAttribute(
626                                        OTEL_INTERCEPTOR_CLASS_NAME_ATT_KEY,
627                                        myMethod.getDeclaringClass().getName());
628                        currentSpan.setAttribute(OTEL_INTERCEPTOR_METHOD_NAME_ATT_KEY, myMethod.getName());
629
630                        return myMethod.invoke(getInterceptor(), args);
631                }
632        }
633
634        protected class HookDescriptor {
635
636                private final POINTCUT myPointcut;
637                private final int myOrder;
638
639                public HookDescriptor(POINTCUT thePointcut, int theOrder) {
640                        myPointcut = thePointcut;
641                        myOrder = theOrder;
642                }
643
644                POINTCUT getPointcut() {
645                        return myPointcut;
646                }
647
648                int getOrder() {
649                        return myOrder;
650                }
651        }
652
653        public abstract static class BaseInvoker implements IInvoker {
654
655                private final int myOrder;
656                private final Object myInterceptor;
657
658                BaseInvoker(Object theInterceptor, int theOrder) {
659                        myInterceptor = theInterceptor;
660                        myOrder = theOrder;
661                }
662
663                @Override
664                public Object getInterceptor() {
665                        return myInterceptor;
666                }
667
668                @Override
669                public int getOrder() {
670                        return myOrder;
671                }
672
673                @Override
674                public int compareTo(IInvoker theInvoker) {
675                        return myOrder - theInvoker.getOrder();
676                }
677        }
678}