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.annotation.Description; 026import ca.uhn.fhir.model.valueset.BundleTypeEnum; 027import ca.uhn.fhir.rest.annotation.Operation; 028import ca.uhn.fhir.rest.api.RestOperationTypeEnum; 029import ca.uhn.fhir.rest.client.impl.BaseHttpClientInvocation; 030import ca.uhn.fhir.rest.param.ParameterUtil; 031import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 032import ca.uhn.fhir.util.FhirTerser; 033import ca.uhn.fhir.util.ParametersUtil; 034import org.hl7.fhir.instance.model.api.IBase; 035import org.hl7.fhir.instance.model.api.IBaseBundle; 036import org.hl7.fhir.instance.model.api.IBaseDatatype; 037import org.hl7.fhir.instance.model.api.IBaseParameters; 038import org.hl7.fhir.instance.model.api.IBaseResource; 039import org.hl7.fhir.instance.model.api.IIdType; 040import org.hl7.fhir.instance.model.api.IPrimitiveType; 041 042import java.lang.reflect.Method; 043import java.lang.reflect.Modifier; 044import java.util.ArrayList; 045import java.util.LinkedHashMap; 046import java.util.List; 047import java.util.Map; 048 049import static org.apache.commons.lang3.StringUtils.isBlank; 050import static org.apache.commons.lang3.StringUtils.isNotBlank; 051 052public class OperationMethodBinding extends BaseResourceReturningMethodBinding { 053 054 private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(OperationMethodBinding.class); 055 private final boolean myIdempotent; 056 private final Integer myIdParamIndex; 057 private final String myName; 058 private final RestOperationTypeEnum myOtherOperationType; 059 private final ReturnTypeEnum myReturnType; 060 private BundleTypeEnum myBundleType; 061 private String myDescription; 062 063 protected OperationMethodBinding( 064 Class<?> theReturnResourceType, 065 Class<? extends IBaseResource> theReturnTypeFromRp, 066 Method theMethod, 067 FhirContext theContext, 068 Object theProvider, 069 boolean theIdempotent, 070 String theOperationName, 071 Class<? extends IBaseResource> theOperationType, 072 BundleTypeEnum theBundleType) { 073 super(theReturnResourceType, theMethod, theContext, theProvider); 074 075 myBundleType = theBundleType; 076 myIdempotent = theIdempotent; 077 myIdParamIndex = ParameterUtil.findIdParameterIndex(theMethod, getContext()); 078 079 Description description = theMethod.getAnnotation(Description.class); 080 if (description != null) { 081 myDescription = ParametersUtil.extractDescription(description); 082 } 083 if (isBlank(myDescription)) { 084 myDescription = null; 085 } 086 087 if (isBlank(theOperationName)) { 088 throw new ConfigurationException(Msg.code(1452) + "Method '" + theMethod.getName() + "' on type " 089 + theMethod.getDeclaringClass().getName() + " is annotated with @" + Operation.class.getSimpleName() 090 + " but this annotation has no name defined"); 091 } 092 if (theOperationName.startsWith("$") == false) { 093 theOperationName = "$" + theOperationName; 094 } 095 myName = theOperationName; 096 097 if (theReturnTypeFromRp != null) { 098 setResourceName(theContext.getResourceType(theReturnTypeFromRp)); 099 } else { 100 if (Modifier.isAbstract(theOperationType.getModifiers()) == false) { 101 setResourceName(theContext.getResourceType(theOperationType)); 102 } else { 103 setResourceName(null); 104 } 105 } 106 107 myReturnType = ReturnTypeEnum.RESOURCE; 108 109 if (getResourceName() == null) { 110 myOtherOperationType = RestOperationTypeEnum.EXTENDED_OPERATION_SERVER; 111 } else if (myIdParamIndex == null) { 112 myOtherOperationType = RestOperationTypeEnum.EXTENDED_OPERATION_TYPE; 113 } else { 114 myOtherOperationType = RestOperationTypeEnum.EXTENDED_OPERATION_INSTANCE; 115 } 116 } 117 118 public OperationMethodBinding( 119 Class<?> theReturnResourceType, 120 Class<? extends IBaseResource> theReturnTypeFromRp, 121 Method theMethod, 122 FhirContext theContext, 123 Object theProvider, 124 Operation theAnnotation) { 125 this( 126 theReturnResourceType, 127 theReturnTypeFromRp, 128 theMethod, 129 theContext, 130 theProvider, 131 theAnnotation.idempotent(), 132 theAnnotation.name(), 133 theAnnotation.type(), 134 theAnnotation.bundleType()); 135 } 136 137 public String getDescription() { 138 return myDescription; 139 } 140 141 public void setDescription(String theDescription) { 142 myDescription = theDescription; 143 } 144 145 /** 146 * Returns the name of the operation, starting with "$" 147 */ 148 public String getName() { 149 return myName; 150 } 151 152 @Override 153 protected BundleTypeEnum getResponseBundleType() { 154 return myBundleType; 155 } 156 157 @Override 158 public RestOperationTypeEnum getRestOperationType() { 159 return myOtherOperationType; 160 } 161 162 @Override 163 public ReturnTypeEnum getReturnType() { 164 return myReturnType; 165 } 166 167 @Override 168 public BaseHttpClientInvocation invokeClient(Object[] theArgs) throws InternalErrorException { 169 String id = null; 170 if (myIdParamIndex != null) { 171 IIdType idDt = (IIdType) theArgs[myIdParamIndex]; 172 id = idDt.getValue(); 173 } 174 IBaseParameters parameters = (IBaseParameters) 175 getContext().getResourceDefinition("Parameters").newInstance(); 176 177 if (theArgs != null) { 178 for (int idx = 0; idx < theArgs.length; idx++) { 179 IParameter nextParam = getParameters().get(idx); 180 nextParam.translateClientArgumentIntoQueryArgument(getContext(), theArgs[idx], null, parameters); 181 } 182 } 183 184 return createOperationInvocation(getContext(), getResourceName(), id, null, myName, parameters, false); 185 } 186 187 public boolean isIdempotent() { 188 return myIdempotent; 189 } 190 191 public static BaseHttpClientInvocation createOperationInvocation( 192 FhirContext theContext, 193 String theResourceName, 194 String theId, 195 String theVersion, 196 String theOperationName, 197 IBaseParameters theInput, 198 boolean theUseHttpGet) { 199 StringBuilder b = new StringBuilder(); 200 if (theResourceName != null) { 201 b.append(theResourceName); 202 if (isNotBlank(theId)) { 203 b.append('/'); 204 b.append(theId); 205 if (isNotBlank(theVersion)) { 206 b.append("/_history/"); 207 b.append(theVersion); 208 } 209 } 210 } 211 if (b.length() > 0) { 212 b.append('/'); 213 } 214 if (!theOperationName.startsWith("$")) { 215 b.append("$"); 216 } 217 b.append(theOperationName); 218 219 if (!theUseHttpGet) { 220 return new HttpPostClientInvocation(theContext, theInput, b.toString()); 221 } 222 FhirTerser t = theContext.newTerser(); 223 List<IBase> parameters = t.getValues(theInput, "Parameters.parameter"); 224 225 Map<String, List<String>> params = new LinkedHashMap<>(); 226 for (Object nextParameter : parameters) { 227 IPrimitiveType<?> nextNameDt = (IPrimitiveType<?>) t.getSingleValueOrNull((IBase) nextParameter, "name"); 228 if (nextNameDt == null || nextNameDt.isEmpty()) { 229 ourLog.warn( 230 "Ignoring input parameter with no value in Parameters.parameter.name in operation client invocation"); 231 continue; 232 } 233 String nextName = nextNameDt.getValueAsString(); 234 if (!params.containsKey(nextName)) { 235 params.put(nextName, new ArrayList<>()); 236 } 237 238 IBaseDatatype value = (IBaseDatatype) t.getSingleValueOrNull((IBase) nextParameter, "value[x]"); 239 if (value == null) { 240 continue; 241 } 242 if (!(value instanceof IPrimitiveType)) { 243 throw new IllegalArgumentException(Msg.code(1453) 244 + "Can not invoke operation as HTTP GET when it has parameters with a composite (non priitive) datatype as the value. Found value: " 245 + value.getClass().getName()); 246 } 247 IPrimitiveType<?> primitive = (IPrimitiveType<?>) value; 248 params.get(nextName).add(primitive.getValueAsString()); 249 } 250 return new HttpGetClientInvocation(theContext, params, b.toString()); 251 } 252 253 public static BaseHttpClientInvocation createProcessMsgInvocation( 254 FhirContext theContext, 255 String theOperationName, 256 IBaseBundle theInput, 257 Map<String, List<String>> urlParams) { 258 StringBuilder b = new StringBuilder(); 259 260 if (b.length() > 0) { 261 b.append('/'); 262 } 263 if (!theOperationName.startsWith("$")) { 264 b.append("$"); 265 } 266 b.append(theOperationName); 267 268 BaseHttpClientInvocation.appendExtraParamsWithQuestionMark(urlParams, b, b.indexOf("?") == -1); 269 270 return new HttpPostClientInvocation(theContext, theInput, b.toString()); 271 } 272}