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.FhirContext; 023import ca.uhn.fhir.model.primitive.IdDt; 024import ca.uhn.fhir.rest.api.Constants; 025import ca.uhn.fhir.util.CoverageIgnore; 026import org.apache.commons.lang3.builder.ToStringBuilder; 027import org.apache.commons.lang3.builder.ToStringStyle; 028import org.hl7.fhir.instance.model.api.IBaseResource; 029import org.hl7.fhir.instance.model.api.IIdType; 030 031import java.math.BigDecimal; 032 033import static ca.uhn.fhir.model.primitive.IdDt.isValidLong; 034import static org.apache.commons.lang3.StringUtils.isBlank; 035import static org.apache.commons.lang3.StringUtils.isNotBlank; 036 037public class ReferenceParam extends BaseParam /*implements IQueryParameterType*/ { 038 039 private String myChain; 040 private String myResourceType; 041 private String myBaseUrl; 042 private String myValue; 043 private String myIdPart; 044 private Boolean myMdmExpand; 045 046 /** 047 * Constructor 048 */ 049 public ReferenceParam() { 050 super(); 051 } 052 053 /** 054 * Constructor 055 */ 056 public ReferenceParam(String theValue) { 057 setValueAsQueryToken(null, null, null, theValue); 058 } 059 060 /** 061 * Constructor 062 */ 063 public ReferenceParam(String theChain, String theValue) { 064 setValueAsQueryToken(null, null, null, theValue); 065 setChain(theChain); 066 } 067 068 /** 069 * Constructor 070 */ 071 public ReferenceParam(String theResourceType, String theChain, String theValue) { 072 String qualifier = ""; 073 if (isNotBlank(theResourceType)) { 074 qualifier = ":" + theResourceType; 075 } 076 if (isNotBlank(theChain)) { 077 qualifier = qualifier + "." + theChain; 078 } 079 080 setValueAsQueryToken(null, null, qualifier, theValue); 081 } 082 083 /** 084 * Constructor 085 * 086 * @since 5.0.0 087 */ 088 public ReferenceParam(IIdType theValue) { 089 if (theValue != null) { 090 setValueAsQueryToken(null, null, null, theValue.getValue()); 091 } 092 } 093 094 private String defaultGetQueryParameterQualifier() { 095 StringBuilder b = new StringBuilder(); 096 if (isNotBlank(myChain)) { 097 if (isNotBlank(getResourceType())) { 098 b.append(':'); 099 b.append(getResourceType()); 100 } 101 b.append('.'); 102 b.append(myChain); 103 } 104 if (b.length() != 0) { 105 return b.toString(); 106 } 107 return null; 108 } 109 110 @Override 111 String doGetQueryParameterQualifier() { 112 return this.myMdmExpand != null ? ":mdm" : defaultGetQueryParameterQualifier(); 113 } 114 115 @Override 116 String doGetValueAsQueryToken(FhirContext theContext) { 117 if (isBlank(getResourceType())) { 118 return myValue; // e.g. urn:asdjd or 123 or cid:wieiuru or #1 119 } else { 120 if (isBlank(getChain()) && isNotBlank(getResourceType())) { 121 return getResourceType() + "/" + getIdPart(); 122 } 123 return myValue; 124 } 125 } 126 127 @Override 128 void doSetValueAsQueryToken(FhirContext theContext, String theParamName, String theQualifier, String theValue) { 129 if (Constants.PARAMQUALIFIER_MDM.equals(theQualifier)) { 130 myMdmExpand = true; 131 theQualifier = ""; 132 } 133 134 String q = theQualifier; 135 if (isNotBlank(q)) { 136 if (q.startsWith(":")) { 137 int nextIdx = q.indexOf('.'); 138 if (nextIdx != -1) { 139 myChain = q.substring(nextIdx + 1); 140 myResourceType = q.substring(1, nextIdx); 141 } else { 142 myChain = null; 143 myResourceType = q.substring(1); 144 } 145 146 myValue = theValue; 147 myIdPart = theValue; 148 149 IdDt id = new IdDt(theValue); 150 if (!id.hasBaseUrl() && id.hasIdPart() && id.hasResourceType()) { 151 if (id.getResourceType().equals(myResourceType)) { 152 myIdPart = id.getIdPart(); 153 } 154 } 155 156 } else if (q.startsWith(".")) { 157 myChain = q.substring(1); 158 myResourceType = null; 159 myValue = theValue; 160 myIdPart = theValue; 161 } 162 } else { 163 myChain = null; 164 myValue = theValue; 165 IdDt id = new IdDt(theValue); 166 myResourceType = id.getResourceType(); 167 myIdPart = id.getIdPart(); 168 myBaseUrl = id.getBaseUrl(); 169 } 170 } 171 172 @CoverageIgnore 173 public String getBaseUrl() { 174 return myBaseUrl; 175 } 176 177 public boolean isMdmExpand() { 178 return myMdmExpand != null && myMdmExpand; 179 } 180 181 public ReferenceParam setMdmExpand(boolean theMdmExpand) { 182 myMdmExpand = theMdmExpand; 183 return this; 184 } 185 186 public String getChain() { 187 return myChain; 188 } 189 190 public ReferenceParam setChain(String theChain) { 191 myChain = theChain; 192 return this; 193 } 194 195 @CoverageIgnore 196 public String getIdPart() { 197 return myIdPart; 198 } 199 200 @CoverageIgnore 201 public BigDecimal getIdPartAsBigDecimal() { 202 return new IdDt(myValue).getIdPartAsBigDecimal(); 203 } 204 205 @CoverageIgnore 206 public Long getIdPartAsLong() { 207 return new IdDt(myValue).getIdPartAsLong(); 208 } 209 210 public String getResourceType() { 211 if (isNotBlank(myResourceType)) { 212 return myResourceType; 213 } 214 if (isBlank(myChain)) { 215 return new IdDt(myValue).getResourceType(); 216 } 217 return null; 218 } 219 220 public Class<? extends IBaseResource> getResourceType(FhirContext theCtx) { 221 if (isBlank(getResourceType())) { 222 return null; 223 } 224 return theCtx.getResourceDefinition(getResourceType()).getImplementingClass(); 225 } 226 227 public String getValue() { 228 return myValue; 229 } 230 231 /** 232 * Note that the parameter to this method <b>must</b> be a resource reference, e.g 233 * <code>123</code> or <code>Patient/123</code> or <code>http://example.com/fhir/Patient/123</code> 234 * or something like this. This is not appropriate for cases where a chain is being used and 235 * the value is for a different type of parameter (e.g. a token). In that case, use one of the 236 * setter constructors. 237 */ 238 public ReferenceParam setValue(String theValue) { 239 IdDt id = new IdDt(theValue); 240 String qualifier = null; 241 if (id.hasResourceType()) { 242 qualifier = ":" + id.getResourceType(); 243 } 244 setValueAsQueryToken(null, null, qualifier, id.getIdPart()); 245 return this; 246 } 247 248 public boolean hasResourceType() { 249 return isNotBlank(myResourceType); 250 } 251 252 @Override 253 protected boolean isSupportsChain() { 254 return true; 255 } 256 257 /** 258 * Returns a new param containing the same value as this param, but with the type copnverted 259 * to {@link DateParam}. This is useful if you are using reference parameters and want to handle 260 * chained parameters of different types in a single method. 261 * <p> 262 * See <a href="https://hapifhir.io/hapi-fhir/docs/server_plain/rest_operations_search.html#chained-resource-references">Dynamic Chains</a> 263 * in the HAPI FHIR documentation for an example of how to use this method. 264 * </p> 265 */ 266 public DateParam toDateParam(FhirContext theContext) { 267 DateParam retVal = new DateParam(); 268 retVal.setValueAsQueryToken(theContext, null, null, getValueAsQueryToken(theContext)); 269 return retVal; 270 } 271 272 /** 273 * Returns a new param containing the same value as this param, but with the type copnverted 274 * to {@link NumberParam}. This is useful if you are using reference parameters and want to handle 275 * chained parameters of different types in a single method. 276 * <p> 277 * See <a href="https://hapifhir.io/hapi-fhir/docs/server_plain/rest_operations_search.html#chained-resource-references">Dynamic Chains</a> 278 * in the HAPI FHIR documentation for an example of how to use this method. 279 * </p> 280 */ 281 public NumberParam toNumberParam(FhirContext theContext) { 282 NumberParam retVal = new NumberParam(); 283 retVal.setValueAsQueryToken(theContext, null, null, getValueAsQueryToken(theContext)); 284 return retVal; 285 } 286 287 /** 288 * Returns a new param containing the same value as this param, but with the type copnverted 289 * to {@link QuantityParam}. This is useful if you are using reference parameters and want to handle 290 * chained parameters of different types in a single method. 291 * <p> 292 * See <a href="https://hapifhir.io/hapi-fhir/docs/server_plain/rest_operations_search.html#chained-resource-references">Dynamic Chains</a> 293 * in the HAPI FHIR documentation for an example of how to use this method. 294 * </p> 295 */ 296 public QuantityParam toQuantityParam(FhirContext theContext) { 297 QuantityParam retVal = new QuantityParam(); 298 retVal.setValueAsQueryToken(theContext, null, null, getValueAsQueryToken(theContext)); 299 return retVal; 300 } 301 302 @Override 303 public String toString() { 304 ToStringBuilder b = new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE); 305 if (isNotBlank(myChain)) { 306 b.append("chain", myChain); 307 } 308 b.append("value", getValue()); 309 return b.build(); 310 } 311 312 /** 313 * Returns a new param containing the same value as this param, but with the type copnverted 314 * to {@link StringParam}. This is useful if you are using reference parameters and want to handle 315 * chained parameters of different types in a single method. 316 * <p> 317 * See <a href="https://hapifhir.io/hapi-fhir/docs/server_plain/rest_operations_search.html#chained-resource-references">Dynamic Chains</a> 318 * in the HAPI FHIR documentation for an example of how to use this method. 319 * </p> 320 */ 321 public StringParam toStringParam(FhirContext theContext) { 322 StringParam retVal = new StringParam(); 323 retVal.setValueAsQueryToken(theContext, null, null, getValueAsQueryToken(theContext)); 324 return retVal; 325 } 326 327 /** 328 * Returns a new param containing the same value as this param, but with the type copnverted 329 * to {@link TokenParam}. This is useful if you are using reference parameters and want to handle 330 * chained parameters of different types in a single method. 331 * <p> 332 * See <a href="https://hapifhir.io/hapi-fhir/docs/server_plain/rest_operations_search.html#chained-resource-references">Dynamic Chains</a> 333 * in the HAPI FHIR documentation for an example of how to use this method. 334 * </p> 335 */ 336 public TokenParam toTokenParam(FhirContext theContext) { 337 TokenParam retVal = new TokenParam(); 338 retVal.setValueAsQueryToken(theContext, null, null, getValueAsQueryToken(theContext)); 339 return retVal; 340 } 341 342 public boolean isIdPartValidLong() { 343 return isValidLong(getIdPart()); 344 } 345}