001/*- 002 * #%L 003 * Smile CDR - CDR 004 * %% 005 * Copyright (C) 2016 - 2025 Smile CDR, Inc. 006 * %% 007 * All rights reserved. 008 * #L% 009 */ 010package ca.cdr.api.fhirgw.model; 011 012import ca.uhn.fhir.model.api.Include; 013import ca.uhn.fhir.rest.api.SearchStyleEnum; 014import ca.uhn.fhir.rest.api.SearchTotalModeEnum; 015import ca.uhn.fhir.rest.api.SortSpec; 016import ca.uhn.fhir.rest.param.DateRangeParam; 017import ca.uhn.fhir.rest.param.UriAndListParam; 018import ca.uhn.fhir.rest.param.UriOrListParam; 019import com.google.common.collect.ArrayListMultimap; 020import com.google.common.collect.ListMultimap; 021import com.google.common.collect.Multimap; 022import com.google.common.collect.Multimaps; 023import jakarta.annotation.Nonnull; 024import jakarta.annotation.Nullable; 025import org.apache.commons.lang3.Validate; 026 027import java.util.Collections; 028import java.util.HashSet; 029import java.util.List; 030import java.util.Map; 031import java.util.Set; 032 033import static org.apache.commons.lang3.ObjectUtils.defaultIfNull; 034 035/** 036 * This class represents a FHIR Gateway Search request 037 */ 038public class SearchRequest extends BaseRequest<SearchRequest> { 039 040 private ListMultimap<String, String> myParameters = EMPTY_STRING_MULTIMAP; 041 private Set<Include> myIncludes = Collections.emptySet(); 042 private Set<Include> myRevIncludes = Collections.emptySet(); 043 private SortSpec mySort; 044 private Integer myCount; 045 private SearchTotalModeEnum myTotalMode; 046 private Integer myOffset; 047 private DateRangeParam myLastUpdated; 048 private UriAndListParam myProfile; 049 050 @Nullable 051 private SearchStyleEnum mySearchStyle; 052 053 /** 054 * Constructor 055 */ 056 public SearchRequest() { 057 // nothing 058 } 059 060 /** 061 * Copy constructor 062 */ 063 public SearchRequest(SearchRequest theSearchRequest) { 064 super(theSearchRequest); 065 066 if (!theSearchRequest.getParameters().isEmpty()) { 067 myParameters = ArrayListMultimap.create(); 068 myParameters.putAll(theSearchRequest.getParameters()); 069 } 070 071 if (!theSearchRequest.getIncludes().isEmpty()) { 072 myIncludes = new HashSet<>(); 073 for (Include next : theSearchRequest.getIncludes()) { 074 myIncludes.add(next.toLocked()); 075 } 076 } 077 078 if (!theSearchRequest.getRevIncludes().isEmpty()) { 079 myRevIncludes = new HashSet<>(); 080 for (Include next : theSearchRequest.getRevIncludes()) { 081 myRevIncludes.add(next.toLocked()); 082 } 083 } 084 085 if (theSearchRequest.getSort() != null) { 086 mySort = clone(theSearchRequest.getSort()); 087 } 088 089 if (theSearchRequest.getLastUpdated() != null) { 090 myLastUpdated = new DateRangeParam( 091 theSearchRequest.getLastUpdated().getLowerBound() != null 092 ? theSearchRequest.getLastUpdated().getLowerBound().getValueAsString() 093 : null, 094 theSearchRequest.getLastUpdated().getUpperBound() != null 095 ? theSearchRequest.getLastUpdated().getUpperBound().getValueAsString() 096 : null); 097 } 098 099 if (theSearchRequest.getProfile() != null) { 100 myProfile = new UriAndListParam(); 101 for (UriOrListParam uriOrListParam : theSearchRequest.getProfile().getValuesAsQueryTokens()) { 102 myProfile.addAnd(uriOrListParam); 103 } 104 } 105 106 myCount = theSearchRequest.getCount(); 107 myOffset = theSearchRequest.getOffset(); 108 myTotalMode = theSearchRequest.getTotalMode(); 109 } 110 111 private static SortSpec clone(SortSpec theSort) { 112 if (theSort == null) { 113 return null; 114 } 115 return new SortSpec(theSort.getParamName(), theSort.getOrder(), clone(theSort.getChain())); 116 } 117 118 /** 119 * Specifies the search form (GET vs POST) to use for this search. If set to 120 * <code>null</code>, the default will be used. 121 */ 122 @Nullable 123 public SearchStyleEnum getSearchStyle() { 124 return mySearchStyle; 125 } 126 127 /** 128 * Specifies the search form (GET vs POST) to use for this search. If set to 129 * <code>null</code>, the default will be used. 130 */ 131 public void setSearchStyle(@Nullable SearchStyleEnum theSearchStyle) { 132 mySearchStyle = theSearchStyle; 133 } 134 135 public Multimap<String, String> getParameters() { 136 return Multimaps.unmodifiableMultimap(myParameters); 137 } 138 139 public SearchRequest setParameters(Map<String, List<String>> theParameters) { 140 Validate.notNull(theParameters, "theParameters must not be null"); 141 myParameters = ArrayListMultimap.create(); 142 for (Map.Entry<String, List<String>> next : theParameters.entrySet()) { 143 myParameters.putAll(next.getKey(), next.getValue()); 144 } 145 return this; 146 } 147 148 /** 149 * Remove all parameters with the given name 150 * 151 * @return Returns any parameter values that were removed 152 */ 153 public List<String> removeParameters(String theName) { 154 if (!myParameters.isEmpty()) { 155 return Collections.unmodifiableList(myParameters.removeAll(theName)); 156 } else { 157 return Collections.emptyList(); 158 } 159 } 160 161 /** 162 * Remove the parameter with the given name and value, if found. Any other parameters with the same name are kept. 163 * 164 * @param theName The parameter name 165 * @param theValue The parameter value 166 * @return Returns <code>true</code> if a parameter value was removed 167 */ 168 public boolean removeParameter(String theName, String theValue) { 169 return myParameters.remove(theName, theValue); 170 } 171 172 /** 173 * Returns the first parameter value for a given parameter name (returns <code>null</code> if no parameters match the given name) 174 * 175 * @param theName The parameter name 176 * @return The value, or <code>null</code> 177 */ 178 @Nullable 179 public String getParameter(String theName) { 180 List<String> values = getParameters(theName); 181 if (!values.isEmpty()) { 182 return values.get(0); 183 } 184 return null; 185 } 186 187 /** 188 * Returns a List of parameter values for the given name. Will not return <code>null</code>. 189 * 190 * @param theName The parameter name 191 * @return A List of values, or an empty List 192 */ 193 @Nonnull 194 public List<String> getParameters(String theName) { 195 List<String> values = myParameters.get(theName); 196 values = defaultIfNull(values, Collections.emptyList()); 197 return Collections.unmodifiableList(values); 198 } 199 200 public Set<Include> getIncludes() { 201 return myIncludes; 202 } 203 204 public SearchRequest setIncludes(Set<Include> theIncludes) { 205 Validate.notNull(theIncludes, "theIncludes must not be null"); 206 myIncludes = theIncludes; 207 return this; 208 } 209 210 public Set<Include> getRevIncludes() { 211 return myRevIncludes; 212 } 213 214 public SearchRequest setRevIncludes(Set<Include> theRevIncludes) { 215 Validate.notNull(theRevIncludes, "theRevIncludes must not be null"); 216 myRevIncludes = theRevIncludes; 217 return this; 218 } 219 220 public SortSpec getSort() { 221 return mySort; 222 } 223 224 public SearchRequest setSort(SortSpec theSort) { 225 mySort = theSort; 226 return this; 227 } 228 229 public Integer getCount() { 230 return myCount; 231 } 232 233 public SearchRequest setCount(Integer theCount) { 234 myCount = theCount; 235 return this; 236 } 237 238 public Integer getOffset() { 239 return myOffset; 240 } 241 242 public SearchRequest setOffset(Integer theOffset) { 243 myOffset = theOffset; 244 return this; 245 } 246 247 public DateRangeParam getLastUpdated() { 248 return myLastUpdated; 249 } 250 251 public SearchRequest setLastUpdated(DateRangeParam theLastUpdated) { 252 myLastUpdated = theLastUpdated; 253 return this; 254 } 255 256 public UriAndListParam getProfile() { 257 return myProfile; 258 } 259 260 public SearchRequest setProfile(UriAndListParam theProfile) { 261 myProfile = theProfile; 262 return this; 263 } 264 265 public SearchTotalModeEnum getTotalMode() { 266 return myTotalMode; 267 } 268 269 public SearchRequest setTotalMode(SearchTotalModeEnum theTotalMode) { 270 myTotalMode = theTotalMode; 271 return this; 272 } 273 274 /** 275 * Add a new search parameter 276 */ 277 public void addParameter(String theName, String theValue) { 278 Validate.notBlank(theName, "theName must not be empty"); 279 Validate.notBlank(theValue, "theValue must not be empty"); 280 if (myParameters.isEmpty()) { 281 myParameters = ArrayListMultimap.create(); 282 } 283 myParameters.put(theName, theValue); 284 } 285}