001/* 002 * #%L 003 * HAPI FHIR - Client Framework 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.client.method; 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.IDatatype; 026import ca.uhn.fhir.model.api.IQueryParameterAnd; 027import ca.uhn.fhir.model.api.IQueryParameterOr; 028import ca.uhn.fhir.model.api.IQueryParameterType; 029import ca.uhn.fhir.rest.annotation.OperationParam; 030import ca.uhn.fhir.rest.api.ValidationModeEnum; 031import ca.uhn.fhir.rest.param.DateRangeParam; 032import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 033import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 034import ca.uhn.fhir.util.ParametersUtil; 035import org.apache.commons.lang3.Validate; 036import org.hl7.fhir.instance.model.api.IBase; 037import org.hl7.fhir.instance.model.api.IBaseParameters; 038import org.hl7.fhir.instance.model.api.IBaseResource; 039import org.hl7.fhir.instance.model.api.IPrimitiveType; 040 041import java.lang.reflect.Method; 042import java.lang.reflect.Modifier; 043import java.util.Collection; 044import java.util.List; 045import java.util.Map; 046 047public class OperationParameter implements IParameter { 048 049 @SuppressWarnings("unchecked") 050 private static final Class<? extends IQueryParameterType>[] COMPOSITE_TYPES = new Class[0]; 051 052 static final String REQUEST_CONTENTS_USERDATA_KEY = OperationParam.class.getName() + "_PARSED_RESOURCE"; 053 054 private final FhirContext myContext; 055 private IOperationParamConverter myConverter; 056 057 @SuppressWarnings("rawtypes") 058 private int myMax; 059 060 private int myMin; 061 private final String myName; 062 private Class<?> myParameterType; 063 private String myParamType; 064 private SearchParameter mySearchParameterBinding; 065 066 public OperationParameter(FhirContext theCtx, String theOperationName, OperationParam theOperationParam) { 067 this(theCtx, theOperationName, theOperationParam.name(), theOperationParam.min(), theOperationParam.max()); 068 } 069 070 OperationParameter(FhirContext theCtx, String theOperationName, String theParameterName, int theMin, int theMax) { 071 myName = theParameterName; 072 myMin = theMin; 073 myMax = theMax; 074 myContext = theCtx; 075 } 076 077 protected FhirContext getContext() { 078 return myContext; 079 } 080 081 public String getName() { 082 return myName; 083 } 084 085 public String getParamType() { 086 return myParamType; 087 } 088 089 public String getSearchParamType() { 090 if (mySearchParameterBinding != null) { 091 return mySearchParameterBinding.getParamType().getCode(); 092 } 093 return null; 094 } 095 096 @SuppressWarnings("unchecked") 097 @Override 098 public void initializeTypes( 099 Method theMethod, 100 Class<? extends Collection<?>> theOuterCollectionType, 101 Class<? extends Collection<?>> theInnerCollectionType, 102 Class<?> theParameterType) { 103 if (getContext().getVersion().getVersion().isRi()) { 104 if (IDatatype.class.isAssignableFrom(theParameterType)) { 105 throw new ConfigurationException(Msg.code(1408) + "Incorrect use of type " 106 + theParameterType.getSimpleName() 107 + " as parameter type for method when context is for version " 108 + getContext().getVersion().getVersion().name() + " in method: " + theMethod.toString()); 109 } 110 } 111 112 myParameterType = theParameterType; 113 if (theInnerCollectionType != null) { 114 if (myMax == OperationParam.MAX_DEFAULT) { 115 myMax = OperationParam.MAX_UNLIMITED; 116 } 117 } else if (IQueryParameterAnd.class.isAssignableFrom(myParameterType)) { 118 if (myMax == OperationParam.MAX_DEFAULT) { 119 myMax = OperationParam.MAX_UNLIMITED; 120 } 121 } else { 122 if (myMax == OperationParam.MAX_DEFAULT) { 123 myMax = 1; 124 } 125 } 126 127 boolean typeIsConcrete = !myParameterType.isInterface() && !Modifier.isAbstract(myParameterType.getModifiers()); 128 129 // @formatter:off 130 boolean isSearchParam = IQueryParameterType.class.isAssignableFrom(myParameterType) 131 || IQueryParameterOr.class.isAssignableFrom(myParameterType) 132 || IQueryParameterAnd.class.isAssignableFrom(myParameterType); 133 // @formatter:off 134 135 /* 136 * Note: We say here !IBase.class.isAssignableFrom because a bunch of DSTU1/2 datatypes also 137 * extend this interface. I'm not sure if they should in the end.. but they do, so we 138 * exclude them. 139 */ 140 isSearchParam &= typeIsConcrete && !IBase.class.isAssignableFrom(myParameterType); 141 142 /* 143 * The parameter can be of type string for validation methods - This is a bit weird. See ValidateDstu2Test. We 144 * should probably clean this up.. 145 */ 146 if (!myParameterType.equals(IBase.class) && !myParameterType.equals(String.class)) { 147 if (IBaseResource.class.isAssignableFrom(myParameterType) && myParameterType.isInterface()) { 148 myParamType = "Resource"; 149 } else if (DateRangeParam.class.isAssignableFrom(myParameterType)) { 150 myParamType = "date"; 151 myMax = 2; 152 } else if (myParameterType.equals(ValidationModeEnum.class)) { 153 myParamType = "code"; 154 } else if (IBase.class.isAssignableFrom(myParameterType) && typeIsConcrete) { 155 myParamType = myContext 156 .getElementDefinition((Class<? extends IBase>) myParameterType) 157 .getName(); 158 } else if (isSearchParam) { 159 myParamType = "string"; 160 mySearchParameterBinding = new SearchParameter(myName, myMin > 0); 161 mySearchParameterBinding.setCompositeTypes(COMPOSITE_TYPES); 162 mySearchParameterBinding.setType( 163 myContext, theParameterType, theInnerCollectionType, theOuterCollectionType); 164 myConverter = new OperationParamConverter(); 165 } else { 166 throw new ConfigurationException( 167 Msg.code(1409) + "Invalid type for @OperationParam: " + myParameterType.getName()); 168 } 169 } 170 } 171 172 public OperationParameter setConverter(IOperationParamConverter theConverter) { 173 myConverter = theConverter; 174 return this; 175 } 176 177 @Override 178 public void translateClientArgumentIntoQueryArgument( 179 FhirContext theContext, 180 Object theSourceClientArgument, 181 Map<String, List<String>> theTargetQueryArguments, 182 IBaseResource theTargetResource) 183 throws InternalErrorException { 184 assert theTargetResource != null; 185 Object sourceClientArgument = theSourceClientArgument; 186 if (sourceClientArgument == null) { 187 return; 188 } 189 190 if (myConverter != null) { 191 sourceClientArgument = myConverter.outgoingClient(sourceClientArgument); 192 } 193 194 ParametersUtil.addParameterToParameters( 195 theContext, (IBaseParameters) theTargetResource, myName, sourceClientArgument); 196 } 197 198 public static void throwInvalidMode(String paramValues) { 199 throw new InvalidRequestException(Msg.code(1410) + "Invalid mode value: \"" + paramValues + "\""); 200 } 201 202 interface IOperationParamConverter { 203 204 Object outgoingClient(Object theObject); 205 } 206 207 class OperationParamConverter implements IOperationParamConverter { 208 209 public OperationParamConverter() { 210 Validate.isTrue(mySearchParameterBinding != null); 211 } 212 213 @Override 214 public Object outgoingClient(Object theObject) { 215 IQueryParameterType obj = (IQueryParameterType) theObject; 216 IPrimitiveType<?> retVal = 217 (IPrimitiveType<?>) myContext.getElementDefinition("string").newInstance(); 218 retVal.setValueAsString(obj.getValueAsQueryToken(myContext)); 219 return retVal; 220 } 221 } 222}