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.api; 021 022import com.google.common.collect.ArrayListMultimap; 023import com.google.common.collect.ListMultimap; 024import com.google.common.collect.Multimaps; 025import jakarta.annotation.Nonnull; 026import org.apache.commons.lang3.Validate; 027 028import java.util.Collection; 029import java.util.Collections; 030import java.util.List; 031import java.util.function.Supplier; 032import java.util.stream.Collectors; 033 034public class HookParams { 035 036 private ListMultimap<Class<?>, Object> myParams = ArrayListMultimap.create(); 037 038 /** 039 * Constructor 040 */ 041 public HookParams() {} 042 043 /** 044 * Constructor 045 */ 046 public HookParams(Object... theParams) { 047 for (Object next : theParams) { 048 add(next); 049 } 050 } 051 052 @SuppressWarnings("unchecked") 053 public <T> HookParams add(@Nonnull T theNext) { 054 Class<T> nextClass = (Class<T>) theNext.getClass(); 055 add(nextClass, theNext); 056 return this; 057 } 058 059 public <T> HookParams add(Class<T> theType, T theParam) { 060 return doAdd(theType, theParam); 061 } 062 063 // /** 064 // * This is useful for providing a lazy-loaded (generally expensive to create) 065 // * parameters 066 // */ 067 // public <T> HookParams addSupplier(Class<T> theType, Supplier<T> theParam) { 068 // return doAdd(theType, theParam); 069 // } 070 071 private <T> HookParams doAdd(Class<T> theType, Object theParam) { 072 Validate.isTrue(theType.equals(Supplier.class) == false, "Can not add parameters of type Supplier"); 073 myParams.put(theType, theParam); 074 return this; 075 } 076 077 public <T> T get(Class<T> theParamType) { 078 return get(theParamType, 0); 079 } 080 081 @SuppressWarnings("unchecked") 082 public <T> T get(Class<T> theParamType, int theIndex) { 083 List<Object> objects = myParams.get(theParamType); 084 Object retVal = null; 085 if (objects.size() > theIndex) { 086 retVal = objects.get(theIndex); 087 } 088 089 retVal = unwrapValue(retVal); 090 091 return (T) retVal; 092 } 093 094 private Object unwrapValue(Object theValue) { 095 if (theValue instanceof Supplier) { 096 theValue = ((Supplier) theValue).get(); 097 } 098 return theValue; 099 } 100 101 /** 102 * Returns an unmodifiable multimap of the params, where the 103 * key is the param type and the value is the actual instance 104 */ 105 public ListMultimap<Class<?>, Object> getParamsForType() { 106 ArrayListMultimap<Class<?>, Object> retVal = ArrayListMultimap.create(); 107 myParams.entries().forEach(entry -> retVal.put(entry.getKey(), unwrapValue(entry.getValue()))); 108 return Multimaps.unmodifiableListMultimap(retVal); 109 } 110 111 public Collection<Object> values() { 112 return Collections.unmodifiableCollection(myParams.values()).stream() 113 .map(t -> unwrapValue(t)) 114 .collect(Collectors.toList()); 115 } 116 117 @SuppressWarnings("unchecked") 118 public <T> HookParams addIfMatchesType(Class<T> theType, Object theParam) { 119 if (theParam == null) { 120 add(theType, null); 121 } else { 122 if (theType.isAssignableFrom(theParam.getClass())) { 123 T param = (T) theParam; 124 add(theType, param); 125 } else { 126 add(theType, null); 127 } 128 } 129 return this; 130 } 131 132 @Override 133 public String toString() { 134 StringBuilder b = new StringBuilder(); 135 myParams.forEach((key, value) -> { 136 b.append(" ") 137 .append(key.getSimpleName()) 138 .append(": ") 139 .append(value) 140 .append("\n"); 141 }); 142 return b.toString(); 143 } 144}