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.model.api; 021 022import ca.uhn.fhir.i18n.Msg; 023import org.apache.commons.lang3.builder.ToStringBuilder; 024 025import java.io.Serializable; 026 027import static org.apache.commons.lang3.StringUtils.defaultString; 028import static org.apache.commons.lang3.StringUtils.isBlank; 029import static org.apache.commons.lang3.StringUtils.isNotBlank; 030 031/** 032 * Represents a FHIR resource path specification, e.g. <code>Patient:name</code> 033 * <p> 034 * Note on equality: This class uses {@link #getValue() value} and the {@link #isRecurse() recurse} properties to test 035 * equality. Prior to HAPI 1.2 (and FHIR DSTU2) the recurse property did not exist, so this may merit consideration when 036 * upgrading servers. 037 * </p> 038 * <p> 039 * Note on thread safety: This class is not thread safe. 040 * </p> 041 */ 042public class Include implements Serializable { 043 044 private static final long serialVersionUID = 1L; 045 046 private final boolean myImmutable; 047 private boolean myIterate; 048 private String myValue; 049 private String myParamType; 050 private String myParamName; 051 private String myParamTargetType; 052 053 /** 054 * Constructor for <b>non-recursive</b> include 055 * 056 * @param theValue 057 * The <code>_include</code> value, e.g. "Patient:name" 058 */ 059 public Include(String theValue) { 060 this(theValue, false); 061 } 062 063 /** 064 * Constructor for an include 065 * 066 * @param theValue 067 * The <code>_include</code> value, e.g. "Patient:name" 068 * @param theIterate 069 * Should the include recurse 070 */ 071 public Include(String theValue, boolean theIterate) { 072 this(theValue, theIterate, false); 073 } 074 075 /** 076 * Constructor for an include 077 * 078 * @param theValue 079 * The <code>_include</code> value, e.g. "Patient:name" 080 * @param theIterate 081 * Should the include recurse 082 */ 083 public Include(String theValue, boolean theIterate, boolean theImmutable) { 084 setValue(theValue); 085 myIterate = theIterate; 086 myImmutable = theImmutable; 087 } 088 089 /** 090 * Creates a copy of this include with non-recurse behaviour 091 */ 092 public Include asNonRecursive() { 093 return new Include(myValue, false); 094 } 095 096 /** 097 * Creates a copy of this include with recurse behaviour 098 */ 099 public Include asRecursive() { 100 return new Include(myValue, true); 101 } 102 103 /** 104 * See the note on equality on the {@link Include class documentation} 105 */ 106 @Override 107 public boolean equals(Object obj) { 108 if (this == obj) { 109 return true; 110 } 111 if (obj == null) { 112 return false; 113 } 114 if (getClass() != obj.getClass()) { 115 return false; 116 } 117 Include other = (Include) obj; 118 if (myIterate != other.myIterate) { 119 return false; 120 } 121 if (myValue == null) { 122 if (other.myValue != null) { 123 return false; 124 } 125 } else if (!myValue.equals(other.myValue)) { 126 return false; 127 } 128 return true; 129 } 130 131 /** 132 * Returns the portion of the value before the first colon 133 */ 134 public String getParamType() { 135 return myParamType; 136 } 137 138 /** 139 * Returns the portion of the value after the first colon but before the second colon 140 */ 141 public String getParamName() { 142 return myParamName; 143 } 144 145 /** 146 * Returns the portion of the string after the second colon, or null if there are not two colons in the value. 147 */ 148 public String getParamTargetType() { 149 return myParamTargetType; 150 } 151 152 public String getValue() { 153 return myValue; 154 } 155 156 /** 157 * See the note on equality on the {@link Include class documentation} 158 */ 159 @Override 160 public int hashCode() { 161 final int prime = 31; 162 int result = 1; 163 result = prime * result + (myIterate ? 1231 : 1237); 164 result = prime * result + ((myValue == null) ? 0 : myValue.hashCode()); 165 return result; 166 } 167 168 /** 169 * Is this object {@link #toLocked() locked}? 170 */ 171 public boolean isLocked() { 172 return myImmutable; 173 } 174 175 public boolean isRecurse() { 176 return myIterate; 177 } 178 179 /** 180 * Should this include recurse 181 * 182 * @return Returns a reference to <code>this</code> for easy method chaining 183 */ 184 public Include setRecurse(boolean theRecurse) { 185 myIterate = theRecurse; 186 return this; 187 } 188 189 public void setValue(String theValue) { 190 if (myImmutable) { 191 throw new IllegalStateException(Msg.code(1888) + "Can not change the value of this include"); 192 } 193 194 String value = defaultString(theValue); 195 196 int firstColon = value.indexOf(':'); 197 String paramType; 198 String paramName; 199 String paramTargetType; 200 if (firstColon == -1 || firstColon == value.length() - 1) { 201 paramType = null; 202 paramName = null; 203 paramTargetType = null; 204 } else { 205 paramType = value.substring(0, firstColon); 206 int secondColon = value.indexOf(':', firstColon + 1); 207 if (secondColon == -1) { 208 paramName = value.substring(firstColon + 1); 209 paramTargetType = null; 210 } else { 211 paramName = value.substring(firstColon + 1, secondColon); 212 paramTargetType = value.substring(secondColon + 1); 213 } 214 } 215 216 myParamType = paramType; 217 myParamName = paramName; 218 myParamTargetType = paramTargetType; 219 myValue = theValue; 220 } 221 222 /** 223 * Return a new 224 */ 225 public Include toLocked() { 226 Include retVal = new Include(myValue, myIterate, true); 227 return retVal; 228 } 229 230 @Override 231 public String toString() { 232 ToStringBuilder builder = new ToStringBuilder(this); 233 builder.append("value", myValue); 234 builder.append("iterate", myIterate); 235 return builder.toString(); 236 } 237 238 /** 239 * Creates and returns a new copy of this Include with the given type. The following table shows what will be 240 * returned: 241 * <table> 242 * <tr> 243 * <th>Initial Contents</th> 244 * <th>theResourceType</th> 245 * <th>Output</th> 246 * </tr> 247 * <tr> 248 * <td>Patient:careProvider</th> 249 * <th>Organization</th> 250 * <th>Patient:careProvider:Organization</th> 251 * </tr> 252 * <tr> 253 * <td>Patient:careProvider:Practitioner</th> 254 * <th>Organization</th> 255 * <th>Patient:careProvider:Organization</th> 256 * </tr> 257 * <tr> 258 * <td>Patient</th> 259 * <th>(any)</th> 260 * <th>{@link IllegalStateException}</th> 261 * </tr> 262 * </table> 263 * 264 * @param theResourceType 265 * The resource type (e.g. "Organization") 266 * @return A new copy of the include. Note that if this include is {@link #toLocked() locked}, the returned include 267 * will be too 268 */ 269 public Include withType(String theResourceType) { 270 StringBuilder b = new StringBuilder(); 271 272 String paramType = getParamType(); 273 String paramName = getParamName(); 274 if (isBlank(paramType) || isBlank(paramName)) { 275 throw new IllegalStateException( 276 Msg.code(1889) + "This include does not contain a value in the format [ResourceType]:[paramName]"); 277 } 278 b.append(paramType); 279 b.append(":"); 280 b.append(paramName); 281 282 if (isNotBlank(theResourceType)) { 283 b.append(':'); 284 b.append(theResourceType); 285 } 286 Include retVal = new Include(b.toString(), myIterate, myImmutable); 287 return retVal; 288 } 289}