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.rest.param; 021 022import ca.uhn.fhir.context.ConfigurationException; 023import ca.uhn.fhir.context.FhirContext; 024import ca.uhn.fhir.i18n.Msg; 025import ca.uhn.fhir.model.api.IQueryParameterOr; 026import ca.uhn.fhir.model.api.IQueryParameterType; 027import ca.uhn.fhir.model.primitive.IdDt; 028import ca.uhn.fhir.model.primitive.IntegerDt; 029import ca.uhn.fhir.rest.annotation.IdParam; 030import ca.uhn.fhir.rest.api.Constants; 031import ca.uhn.fhir.rest.api.QualifiedParamList; 032import ca.uhn.fhir.util.ReflectionUtil; 033import ca.uhn.fhir.util.UrlUtil; 034import org.hl7.fhir.instance.model.api.IIdType; 035import org.hl7.fhir.instance.model.api.IPrimitiveType; 036 037import java.lang.annotation.Annotation; 038import java.lang.reflect.Method; 039import java.util.ArrayList; 040import java.util.Collection; 041import java.util.Collections; 042import java.util.List; 043import java.util.stream.Collectors; 044 045public class ParameterUtil { 046 047 @SuppressWarnings("unchecked") 048 public static <T extends IIdType> T convertIdToType(IIdType value, Class<T> theIdParamType) { 049 if (value != null && !theIdParamType.isAssignableFrom(value.getClass())) { 050 IIdType newValue = ReflectionUtil.newInstance(theIdParamType); 051 newValue.setValue(value.getValue()); 052 value = newValue; 053 } 054 return (T) value; 055 } 056 057 /** 058 * Removes :modifiers and .chains from URL parameter names 059 */ 060 public static String stripModifierPart(String theParam) { 061 for (int i = 0; i < theParam.length(); i++) { 062 char nextChar = theParam.charAt(i); 063 if (nextChar == ':' || nextChar == '.') { 064 return theParam.substring(0, i); 065 } 066 } 067 return theParam; 068 } 069 070 /** 071 * Escapes a string according to the rules for parameter escaping specified in the <a href="http://www.hl7.org/implement/standards/fhir/search.html#escaping">FHIR Specification Escaping 072 * Section</a> 073 */ 074 public static String escape(String theValue) { 075 if (theValue == null) { 076 return null; 077 } 078 StringBuilder b = new StringBuilder(); 079 080 for (int i = 0; i < theValue.length(); i++) { 081 char next = theValue.charAt(i); 082 switch (next) { 083 case '$': 084 case ',': 085 case '|': 086 case '\\': 087 b.append('\\'); 088 break; 089 default: 090 break; 091 } 092 b.append(next); 093 } 094 095 return b.toString(); 096 } 097 098 /** 099 * Escapes a string according to the rules for parameter escaping specified in the <a href="http://www.hl7.org/implement/standards/fhir/search.html#escaping">FHIR Specification Escaping 100 * Section</a> 101 */ 102 public static String escapeWithDefault(Object theValue) { 103 if (theValue == null) { 104 return ""; 105 } 106 return escape(theValue.toString()); 107 } 108 109 /** 110 * Applies {@link #escapeWithDefault(Object)} followed by {@link UrlUtil#escapeUrlParam(String)} 111 */ 112 public static String escapeAndUrlEncode(String theInput) { 113 return UrlUtil.escapeUrlParam(escapeWithDefault(theInput)); 114 } 115 116 public static Integer findIdParameterIndex(Method theMethod, FhirContext theContext) { 117 Integer index = findParamAnnotationIndex(theMethod, IdParam.class); 118 if (index != null) { 119 Class<?> paramType = theMethod.getParameterTypes()[index]; 120 if (IIdType.class.equals(paramType)) { 121 return index; 122 } 123 boolean isRi = theContext.getVersion().getVersion().isRi(); 124 boolean usesHapiId = IdDt.class.equals(paramType); 125 if (isRi == usesHapiId) { 126 throw new ConfigurationException(Msg.code(1936) 127 + "Method uses the wrong Id datatype (IdDt / IdType) for the given context FHIR version: " 128 + theMethod.toString()); 129 } 130 } 131 return index; 132 } 133 134 // public static Integer findSinceParameterIndex(Method theMethod) { 135 // return findParamIndex(theMethod, Since.class); 136 // } 137 138 public static Integer findParamAnnotationIndex(Method theMethod, Class<?> toFind) { 139 int paramIndex = 0; 140 for (Annotation[] annotations : theMethod.getParameterAnnotations()) { 141 for (Annotation nextAnnotation : annotations) { 142 Class<? extends Annotation> class1 = nextAnnotation.annotationType(); 143 if (toFind.isAssignableFrom(class1)) { 144 return paramIndex; 145 } 146 } 147 paramIndex++; 148 } 149 return null; 150 } 151 152 public static Object fromInteger(Class<?> theType, IntegerDt theArgument) { 153 if (theArgument == null) { 154 return null; 155 } 156 if (theType.equals(Integer.class)) { 157 return theArgument.getValue(); 158 } 159 IPrimitiveType<?> retVal = (IPrimitiveType<?>) ReflectionUtil.newInstance(theType); 160 retVal.setValueAsString(theArgument.getValueAsString()); 161 return retVal; 162 } 163 164 public static boolean isBindableIntegerType(Class<?> theClass) { 165 return Integer.class.isAssignableFrom(theClass) || IPrimitiveType.class.isAssignableFrom(theClass); 166 } 167 168 public static String escapeAndJoinOrList(Collection<String> theValues) { 169 return theValues.stream().map(ParameterUtil::escape).collect(Collectors.joining(",")); 170 } 171 172 public static int nonEscapedIndexOf(String theString, char theCharacter) { 173 for (int i = 0; i < theString.length(); i++) { 174 if (theString.charAt(i) == theCharacter) { 175 if (i == 0 || theString.charAt(i - 1) != '\\') { 176 return i; 177 } 178 } 179 } 180 return -1; 181 } 182 183 public static String parseETagValue(String value) { 184 String eTagVersion; 185 value = value.trim(); 186 if (value.length() > 1) { 187 if (value.charAt(value.length() - 1) == '"') { 188 if (value.charAt(0) == '"') { 189 eTagVersion = value.substring(1, value.length() - 1); 190 } else if (value.length() > 3 191 && value.charAt(0) == 'W' 192 && value.charAt(1) == '/' 193 && value.charAt(2) == '"') { 194 eTagVersion = value.substring(3, value.length() - 1); 195 } else { 196 eTagVersion = value; 197 } 198 } else { 199 eTagVersion = value; 200 } 201 } else { 202 eTagVersion = value; 203 } 204 return eTagVersion; 205 } 206 207 public static IQueryParameterOr<?> singleton(final IQueryParameterType theParam, final String theParamName) { 208 return new IQueryParameterOr<IQueryParameterType>() { 209 210 private static final long serialVersionUID = 1L; 211 212 @Override 213 public List<IQueryParameterType> getValuesAsQueryTokens() { 214 return Collections.singletonList(theParam); 215 } 216 217 @Override 218 public void setValuesAsQueryTokens( 219 FhirContext theContext, String theParamName, QualifiedParamList theParameters) { 220 if (theParameters.isEmpty()) { 221 return; 222 } 223 if (theParameters.size() > 1) { 224 throw new IllegalArgumentException(Msg.code(1937) + "Type " 225 + theParam.getClass().getCanonicalName() + " does not support multiple values"); 226 } 227 theParam.setValueAsQueryToken( 228 theContext, theParamName, theParameters.getQualifier(), theParameters.get(0)); 229 } 230 }; 231 } 232 233 static List<String> splitParameterString(String theInput, char theDelimiter, boolean theUnescapeComponents) { 234 ArrayList<String> retVal = new ArrayList<>(); 235 if (theInput != null) { 236 StringBuilder b = new StringBuilder(); 237 for (int i = 0; i < theInput.length(); i++) { 238 char next = theInput.charAt(i); 239 if (next == theDelimiter) { 240 if (i == 0) { 241 b.append(next); 242 } else { 243 char prevChar = theInput.charAt(i - 1); 244 if (prevChar == '\\') { 245 b.append(next); 246 } else { 247 if (b.length() > 0) { 248 retVal.add(b.toString()); 249 } else { 250 retVal.add(null); 251 } 252 b.setLength(0); 253 } 254 } 255 } else { 256 b.append(next); 257 } 258 } 259 if (b.length() > 0) { 260 retVal.add(b.toString()); 261 } 262 } 263 264 if (theUnescapeComponents) { 265 for (int i = 0; i < retVal.size(); i++) { 266 retVal.set(i, unescape(retVal.get(i))); 267 } 268 } 269 270 return retVal; 271 } 272 273 public static IntegerDt toInteger(Object theArgument) { 274 if (theArgument instanceof IntegerDt) { 275 return (IntegerDt) theArgument; 276 } 277 if (theArgument instanceof Integer) { 278 return new IntegerDt((Integer) theArgument); 279 } 280 if (theArgument instanceof IPrimitiveType) { 281 IPrimitiveType<?> pt = (IPrimitiveType<?>) theArgument; 282 return new IntegerDt(pt.getValueAsString()); 283 } 284 return null; 285 } 286 287 /** 288 * Unescapes a string according to the rules for parameter escaping specified in the <a href="http://www.hl7.org/implement/standards/fhir/search.html#escaping">FHIR Specification Escaping 289 * Section</a> 290 */ 291 public static String unescape(String theValue) { 292 if (theValue == null) { 293 return null; 294 } 295 if (theValue.indexOf('\\') == -1) { 296 return theValue; 297 } 298 299 StringBuilder b = new StringBuilder(); 300 301 for (int i = 0; i < theValue.length(); i++) { 302 char next = theValue.charAt(i); 303 if (next == '\\') { 304 if (i == theValue.length() - 1) { 305 b.append(next); 306 } else { 307 switch (theValue.charAt(i + 1)) { 308 case '$': 309 case ',': 310 case '|': 311 case '\\': 312 continue; 313 default: 314 b.append(next); 315 } 316 } 317 } else { 318 b.append(next); 319 } 320 } 321 322 return b.toString(); 323 } 324 325 /** 326 * Returns true if the value is :iterate or :recurse (the former name of :iterate) for an _include parameter 327 */ 328 public static boolean isIncludeIterate(String theQualifier) { 329 return Constants.PARAM_INCLUDE_QUALIFIER_RECURSE.equals(theQualifier) 330 || Constants.PARAM_INCLUDE_QUALIFIER_ITERATE.equals(theQualifier); 331 } 332}