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.context; 021 022import ca.uhn.fhir.context.phonetic.IPhoneticEncoder; 023import ca.uhn.fhir.rest.api.RestSearchParameterTypeEnum; 024import jakarta.annotation.Nonnull; 025import jakarta.annotation.Nullable; 026import org.apache.commons.lang3.builder.EqualsBuilder; 027import org.apache.commons.lang3.builder.HashCodeBuilder; 028import org.apache.commons.lang3.builder.ToStringBuilder; 029import org.apache.commons.lang3.builder.ToStringStyle; 030import org.hl7.fhir.instance.model.api.IBaseExtension; 031import org.hl7.fhir.instance.model.api.IIdType; 032 033import java.util.ArrayList; 034import java.util.Collection; 035import java.util.Collections; 036import java.util.HashMap; 037import java.util.HashSet; 038import java.util.List; 039import java.util.Map; 040import java.util.Set; 041import java.util.StringTokenizer; 042 043import static org.apache.commons.lang3.StringUtils.isNotBlank; 044import static org.apache.commons.lang3.StringUtils.trim; 045 046public class RuntimeSearchParam { 047 private final IIdType myId; 048 private final Set<String> myBase; 049 private final String myDescription; 050 private final String myName; 051 private final RestSearchParameterTypeEnum myParamType; 052 private final String myPath; 053 private final Set<String> myTargets; 054 private final Set<String> myProvidesMembershipInCompartments; 055 private final RuntimeSearchParamStatusEnum myStatus; 056 private final String myUri; 057 private final Map<String, List<IBaseExtension<?, ?>>> myExtensions = new HashMap<>(); 058 private final Map<String, String> myUpliftRefchains = new HashMap<>(); 059 private final ComboSearchParamType myComboSearchParamType; 060 private final List<Component> myComponents; 061 private final IIdType myIdUnqualifiedVersionless; 062 private IPhoneticEncoder myPhoneticEncoder; 063 private boolean myEnabledForSearching = true; 064 065 /** 066 * Constructor 067 */ 068 public RuntimeSearchParam( 069 IIdType theId, 070 String theUri, 071 String theName, 072 String theDescription, 073 String thePath, 074 RestSearchParameterTypeEnum theParamType, 075 Set<String> theProvidesMembershipInCompartments, 076 Set<String> theTargets, 077 RuntimeSearchParamStatusEnum theStatus, 078 Collection<String> theBase) { 079 this( 080 theId, 081 theUri, 082 theName, 083 theDescription, 084 thePath, 085 theParamType, 086 theProvidesMembershipInCompartments, 087 theTargets, 088 theStatus, 089 null, 090 Collections.emptyList(), 091 theBase); 092 } 093 094 /** 095 * Copy constructor 096 */ 097 public RuntimeSearchParam(RuntimeSearchParam theSp) { 098 this( 099 theSp.getId(), 100 theSp.getUri(), 101 theSp.getName(), 102 theSp.getDescription(), 103 theSp.getPath(), 104 theSp.getParamType(), 105 theSp.getProvidesMembershipInCompartments(), 106 theSp.getTargets(), 107 theSp.getStatus(), 108 theSp.getComboSearchParamType(), 109 theSp.getComponents(), 110 theSp.getBase()); 111 } 112 113 /** 114 * Constructor 115 */ 116 public RuntimeSearchParam( 117 IIdType theId, 118 String theUri, 119 String theName, 120 String theDescription, 121 String thePath, 122 RestSearchParameterTypeEnum theParamType, 123 Set<String> theProvidesMembershipInCompartments, 124 Set<String> theTargets, 125 RuntimeSearchParamStatusEnum theStatus, 126 ComboSearchParamType theComboSearchParamType, 127 List<Component> theComponents, 128 Collection<String> theBase) { 129 super(); 130 131 myId = theId; 132 myIdUnqualifiedVersionless = theId != null ? theId.toUnqualifiedVersionless() : null; 133 myUri = theUri; 134 myName = theName; 135 myDescription = theDescription; 136 myPath = thePath; 137 myParamType = theParamType; 138 myStatus = theStatus; 139 if (theProvidesMembershipInCompartments != null && !theProvidesMembershipInCompartments.isEmpty()) { 140 myProvidesMembershipInCompartments = Collections.unmodifiableSet(theProvidesMembershipInCompartments); 141 } else { 142 myProvidesMembershipInCompartments = null; 143 } 144 if (theTargets != null && theTargets.isEmpty() == false) { 145 myTargets = Collections.unmodifiableSet(theTargets); 146 } else { 147 myTargets = Collections.emptySet(); 148 } 149 150 if (theBase == null || theBase.isEmpty()) { 151 HashSet<String> base = new HashSet<>(); 152 if (isNotBlank(thePath)) { 153 int indexOf = thePath.indexOf('.'); 154 if (indexOf != -1) { 155 base.add(trim(thePath.substring(0, indexOf))); 156 } 157 } 158 myBase = Collections.unmodifiableSet(base); 159 } else { 160 myBase = Collections.unmodifiableSet(new HashSet<>(theBase)); 161 } 162 myComboSearchParamType = theComboSearchParamType; 163 if (theComponents != null) { 164 myComponents = Collections.unmodifiableList(theComponents); 165 } else { 166 myComponents = Collections.emptyList(); 167 } 168 } 169 170 /** 171 * Is this search parameter actually enabled for being used in searches (as opposed to only being used for 172 * generating indexes, which might be desired while the search parameter is still being indexed). This 173 * setting defaults to {@literal true} if it isn't set otherwise. 174 */ 175 public boolean isEnabledForSearching() { 176 return myEnabledForSearching; 177 } 178 179 /** 180 * Is this search parameter actually enabled for being used in searches (as opposed to only being used for 181 * generating indexes, which might be desired while the search parameter is still being indexed). This 182 * setting defaults to {@literal true} if it isn't set otherwise. 183 */ 184 public void setEnabledForSearching(boolean theEnabledForSearching) { 185 myEnabledForSearching = theEnabledForSearching; 186 } 187 188 public List<Component> getComponents() { 189 return myComponents; 190 } 191 192 /** 193 * Returns <code>null</code> if this is not a combo search param type 194 */ 195 @Nullable 196 public ComboSearchParamType getComboSearchParamType() { 197 return myComboSearchParamType; 198 } 199 200 /** 201 * Retrieve user data - This can be used to store any application-specific data 202 */ 203 @Nonnull 204 public List<IBaseExtension<?, ?>> getExtensions(String theKey) { 205 List<IBaseExtension<?, ?>> retVal = myExtensions.get(theKey); 206 if (retVal != null) { 207 retVal = Collections.unmodifiableList(retVal); 208 } else { 209 retVal = Collections.emptyList(); 210 } 211 return retVal; 212 } 213 214 /** 215 * Sets user data - This can be used to store any application-specific data 216 */ 217 public RuntimeSearchParam addExtension(String theKey, IBaseExtension<?, ?> theValue) { 218 List<IBaseExtension<?, ?>> valuesList = myExtensions.computeIfAbsent(theKey, k -> new ArrayList<>()); 219 valuesList.add(theValue); 220 return this; 221 } 222 223 @Override 224 public String toString() { 225 return new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE) 226 .append("base", myBase) 227 .append("name", myName) 228 .append("path", myPath) 229 .append("id", myId) 230 .append("uri", myUri) 231 .toString(); 232 } 233 234 public IIdType getId() { 235 return myId; 236 } 237 238 public IIdType getIdUnqualifiedVersionless() { 239 return myIdUnqualifiedVersionless; 240 } 241 242 public String getUri() { 243 return myUri; 244 } 245 246 @Override 247 public boolean equals(Object theO) { 248 if (this == theO) return true; 249 250 if (theO == null || getClass() != theO.getClass()) return false; 251 252 RuntimeSearchParam that = (RuntimeSearchParam) theO; 253 254 return new EqualsBuilder() 255 .append(getId(), that.getId()) 256 .append(getName(), that.getName()) 257 .append(getPath(), that.getPath()) 258 .append(getUri(), that.getUri()) 259 .isEquals(); 260 } 261 262 @Override 263 public int hashCode() { 264 return new HashCodeBuilder(17, 37) 265 .append(getId()) 266 .append(getName()) 267 .append(getPath()) 268 .append(getUri()) 269 .toHashCode(); 270 } 271 272 public Set<String> getBase() { 273 return myBase; 274 } 275 276 @Nonnull 277 public Set<String> getTargets() { 278 return myTargets; 279 } 280 281 public boolean hasTargets() { 282 return !myTargets.isEmpty(); 283 } 284 285 public RuntimeSearchParamStatusEnum getStatus() { 286 return myStatus; 287 } 288 289 public String getDescription() { 290 return myDescription; 291 } 292 293 public String getName() { 294 return myName; 295 } 296 297 public RestSearchParameterTypeEnum getParamType() { 298 return myParamType; 299 } 300 301 public String getPath() { 302 return myPath; 303 } 304 305 public List<String> getPathsSplit() { 306 return getPathsSplitForResourceType(null); 307 } 308 309 /** 310 * Can return null 311 */ 312 public Set<String> getProvidesMembershipInCompartments() { 313 return myProvidesMembershipInCompartments; 314 } 315 316 public RuntimeSearchParam setPhoneticEncoder(IPhoneticEncoder thePhoneticEncoder) { 317 myPhoneticEncoder = thePhoneticEncoder; 318 return this; 319 } 320 321 public String encode(String theString) { 322 if (myPhoneticEncoder == null || theString == null) { 323 return theString; 324 } 325 return myPhoneticEncoder.encode(theString); 326 } 327 328 public List<String> getPathsSplitForResourceType(@Nullable String theResourceName) { 329 String path = getPath(); 330 if (path.indexOf('|') == -1) { 331 if (theResourceName != null && !pathMatchesResourceType(theResourceName, path)) { 332 return Collections.emptyList(); 333 } 334 return Collections.singletonList(path); 335 } 336 337 List<String> retVal = new ArrayList<>(); 338 StringTokenizer tok = new StringTokenizer(path, "|"); 339 while (tok.hasMoreElements()) { 340 String nextPath = tok.nextToken().trim(); 341 if (theResourceName != null && !pathMatchesResourceType(theResourceName, nextPath)) { 342 continue; 343 } 344 retVal.add(nextPath.trim()); 345 } 346 return retVal; 347 } 348 349 public void addUpliftRefchain(@Nonnull String theCode, @Nonnull String theElementName) { 350 myUpliftRefchains.put(theCode, theElementName); 351 } 352 353 /** 354 * Does this search parameter have an uplift refchain definition for the given code? 355 * See the HAPI FHIR documentation for a description of how uplift refchains work. 356 * 357 * @since 6.6.0 358 */ 359 public boolean hasUpliftRefchain(String theCode) { 360 return myUpliftRefchains.containsKey(theCode); 361 } 362 363 /** 364 * Returns a set of all codes associated with uplift refchains for this search parameter. 365 * See the HAPI FHIR documentation for a description of how uplift refchains work. 366 * 367 * @since 6.6.0 368 */ 369 public Set<String> getUpliftRefchainCodes() { 370 return Collections.unmodifiableSet(myUpliftRefchains.keySet()); 371 } 372 373 /** 374 * Does this search parameter have any uplift refchain definitions? 375 * See the HAPI FHIR documentation for a description of how uplift refchains work. 376 * 377 * @since 6.6.0 378 */ 379 public boolean hasUpliftRefchains() { 380 return !myUpliftRefchains.isEmpty(); 381 } 382 383 /** 384 * This method tests whether a given FHIRPath expression <i>could</i> 385 * possibly apply to the given resource type. 386 * 387 * @param theResourceName 388 * @param thePath 389 * @return 390 */ 391 static boolean pathMatchesResourceType(String theResourceName, String thePath) { 392 for (int i = 0; i < thePath.length() - 1; i++) { 393 char nextChar = thePath.charAt(i); 394 if (Character.isLowerCase(nextChar)) { 395 return true; 396 } 397 if (Character.isLetter(nextChar)) { 398 if (fhirPathExpressionStartsWith(theResourceName, thePath, i)) { 399 return true; 400 } 401 if (fhirPathExpressionStartsWith("Resource", thePath, i)) { 402 return true; 403 } 404 if (fhirPathExpressionStartsWith("DomainResource", thePath, i)) { 405 return true; 406 } 407 return false; 408 } 409 } 410 411 return false; 412 } 413 414 private static boolean fhirPathExpressionStartsWith(String theResourceName, String thePath, int theStartingIndex) { 415 if (thePath.startsWith(theResourceName, theStartingIndex) && thePath.length() > theResourceName.length()) { 416 for (int i = theResourceName.length() + theStartingIndex; i < thePath.length(); i++) { 417 char nextChar = thePath.charAt(i); 418 if (nextChar == '.') { 419 return true; 420 } else if (nextChar != ' ') { 421 return false; 422 } 423 } 424 } 425 return false; 426 } 427 428 public enum RuntimeSearchParamStatusEnum { 429 ACTIVE, 430 DRAFT, 431 RETIRED, 432 UNKNOWN 433 } 434 435 public static class Component { 436 private final String myExpression; 437 private final String myReference; 438 439 /** 440 * Constructor 441 */ 442 public Component(String theExpression, String theReference) { 443 myExpression = theExpression; 444 myReference = theReference; 445 } 446 447 @Override 448 public String toString() { 449 return new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE) 450 .append("expression", myExpression) 451 .append("reference", myReference) 452 .toString(); 453 } 454 455 public String getExpression() { 456 return myExpression; 457 } 458 459 public String getReference() { 460 return myReference; 461 } 462 } 463}