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.parser.json.jackson; 021 022import ca.uhn.fhir.i18n.Msg; 023import ca.uhn.fhir.parser.DataFormatException; 024import ca.uhn.fhir.parser.json.BaseJsonLikeArray; 025import ca.uhn.fhir.parser.json.BaseJsonLikeObject; 026import ca.uhn.fhir.parser.json.BaseJsonLikeValue; 027import ca.uhn.fhir.parser.json.BaseJsonLikeWriter; 028import ca.uhn.fhir.parser.json.JsonLikeStructure; 029import com.fasterxml.jackson.core.JsonGenerator; 030import com.fasterxml.jackson.core.JsonParser; 031import com.fasterxml.jackson.core.JsonProcessingException; 032import com.fasterxml.jackson.core.StreamReadConstraints; 033import com.fasterxml.jackson.core.json.JsonReadFeature; 034import com.fasterxml.jackson.databind.DeserializationFeature; 035import com.fasterxml.jackson.databind.JsonNode; 036import com.fasterxml.jackson.databind.ObjectMapper; 037import com.fasterxml.jackson.databind.json.JsonMapper; 038import com.fasterxml.jackson.databind.node.ArrayNode; 039import com.fasterxml.jackson.databind.node.DecimalNode; 040import com.fasterxml.jackson.databind.node.JsonNodeFactory; 041import com.fasterxml.jackson.databind.node.ObjectNode; 042 043import java.io.IOException; 044import java.io.PushbackReader; 045import java.io.Reader; 046import java.io.Writer; 047import java.math.BigDecimal; 048import java.util.AbstractSet; 049import java.util.ArrayList; 050import java.util.Iterator; 051import java.util.LinkedHashMap; 052import java.util.Map; 053 054public class JacksonStructure implements JsonLikeStructure { 055 056 private static final ObjectMapper OBJECT_MAPPER = createObjectMapper(); 057 private JacksonWriter jacksonWriter; 058 private ROOT_TYPE rootType = null; 059 private JsonNode nativeRoot = null; 060 private JsonNode jsonLikeRoot = null; 061 062 public void setNativeObject(ObjectNode objectNode) { 063 this.rootType = ROOT_TYPE.OBJECT; 064 this.nativeRoot = objectNode; 065 } 066 067 public void setNativeArray(ArrayNode arrayNode) { 068 this.rootType = ROOT_TYPE.ARRAY; 069 this.nativeRoot = arrayNode; 070 } 071 072 @Override 073 public JsonLikeStructure getInstance() { 074 return new JacksonStructure(); 075 } 076 077 @Override 078 public void load(Reader theReader) throws DataFormatException { 079 this.load(theReader, false); 080 } 081 082 @Override 083 public void load(Reader theReader, boolean allowArray) throws DataFormatException { 084 PushbackReader pbr = new PushbackReader(theReader); 085 int nextInt; 086 try { 087 while (true) { 088 nextInt = pbr.read(); 089 if (nextInt == -1) { 090 throw new DataFormatException(Msg.code(1857) + "Did not find any content to parse"); 091 } 092 if (nextInt == '{') { 093 pbr.unread(nextInt); 094 break; 095 } 096 if (Character.isWhitespace(nextInt)) { 097 continue; 098 } 099 if (allowArray) { 100 if (nextInt == '[') { 101 pbr.unread(nextInt); 102 break; 103 } 104 throw new DataFormatException(Msg.code(1858) 105 + "Content does not appear to be FHIR JSON, first non-whitespace character was: '" 106 + (char) nextInt + "' (must be '{' or '[')"); 107 } 108 throw new DataFormatException(Msg.code(1859) 109 + "Content does not appear to be FHIR JSON, first non-whitespace character was: '" 110 + (char) nextInt + "' (must be '{')"); 111 } 112 113 if (nextInt == '{') { 114 setNativeObject((ObjectNode) OBJECT_MAPPER.readTree(pbr)); 115 } else { 116 setNativeArray((ArrayNode) OBJECT_MAPPER.readTree(pbr)); 117 } 118 } catch (Exception e) { 119 String message; 120 if (e instanceof JsonProcessingException) { 121 /* 122 * Currently there is no way of preventing Jackson from adding this 123 * annoying REDACTED message from certain messages we get back from 124 * the parser, so we just manually strip them. Hopefully Jackson 125 * will accept this request at some point: 126 * https://github.com/FasterXML/jackson-core/issues/1158 127 */ 128 JsonProcessingException jpe = (JsonProcessingException) e; 129 StringBuilder messageBuilder = new StringBuilder(); 130 String originalMessage = jpe.getOriginalMessage(); 131 originalMessage = originalMessage.replace( 132 "Source: REDACTED (`StreamReadFeature.INCLUDE_SOURCE_IN_LOCATION` disabled); ", ""); 133 messageBuilder.append(originalMessage); 134 if (jpe.getLocation() != null) { 135 messageBuilder.append("\n at ["); 136 jpe.getLocation().appendOffsetDescription(messageBuilder); 137 messageBuilder.append("]"); 138 } 139 message = messageBuilder.toString(); 140 } else { 141 message = e.getMessage(); 142 } 143 144 if (message.startsWith("Unexpected char 39")) { 145 throw new DataFormatException( 146 Msg.code(1860) + "Failed to parse JSON encoded FHIR content: " + message + " - " 147 + "This may indicate that single quotes are being used as JSON escapes where double quotes are required", 148 e); 149 } 150 throw new DataFormatException(Msg.code(1861) + "Failed to parse JSON encoded FHIR content: " + message, e); 151 } 152 } 153 154 @Override 155 public BaseJsonLikeWriter getJsonLikeWriter(Writer writer) throws IOException { 156 if (null == jacksonWriter) { 157 jacksonWriter = new JacksonWriter(OBJECT_MAPPER.getFactory(), writer); 158 } 159 160 return jacksonWriter; 161 } 162 163 @Override 164 public BaseJsonLikeWriter getJsonLikeWriter() { 165 if (null == jacksonWriter) { 166 jacksonWriter = new JacksonWriter(); 167 } 168 return jacksonWriter; 169 } 170 171 @Override 172 public BaseJsonLikeObject getRootObject() throws DataFormatException { 173 if (rootType == ROOT_TYPE.OBJECT) { 174 if (null == jsonLikeRoot) { 175 jsonLikeRoot = nativeRoot; 176 } 177 178 return new JacksonJsonObject((ObjectNode) jsonLikeRoot); 179 } 180 181 throw new DataFormatException(Msg.code(1862) + "Content must be a valid JSON Object. It must start with '{'."); 182 } 183 184 private enum ROOT_TYPE { 185 OBJECT, 186 ARRAY 187 } 188 189 private static class JacksonJsonObject extends BaseJsonLikeObject { 190 private final ObjectNode nativeObject; 191 192 public JacksonJsonObject(ObjectNode json) { 193 this.nativeObject = json; 194 } 195 196 @Override 197 public Object getValue() { 198 return null; 199 } 200 201 @Override 202 public Iterator<String> keyIterator() { 203 return nativeObject.fieldNames(); 204 } 205 206 @Override 207 public BaseJsonLikeValue get(String key) { 208 JsonNode child = nativeObject.get(key); 209 if (child != null) { 210 return new JacksonJsonValue(child); 211 } 212 return null; 213 } 214 } 215 216 private static class EntryOrderedSet<T> extends AbstractSet<T> { 217 private final transient ArrayList<T> data; 218 219 public EntryOrderedSet() { 220 data = new ArrayList<>(); 221 } 222 223 @Override 224 public int size() { 225 return data.size(); 226 } 227 228 @Override 229 public boolean contains(Object o) { 230 return data.contains(o); 231 } 232 233 public T get(int index) { 234 return data.get(index); 235 } 236 237 @Override 238 public boolean add(T element) { 239 if (data.contains(element)) { 240 return false; 241 } 242 return data.add(element); 243 } 244 245 @Override 246 public boolean remove(Object o) { 247 return data.remove(o); 248 } 249 250 @Override 251 public void clear() { 252 data.clear(); 253 } 254 255 @Override 256 public Iterator<T> iterator() { 257 return data.iterator(); 258 } 259 } 260 261 private static class JacksonJsonArray extends BaseJsonLikeArray { 262 private final ArrayNode nativeArray; 263 private final Map<Integer, BaseJsonLikeValue> jsonLikeMap = new LinkedHashMap<Integer, BaseJsonLikeValue>(); 264 265 public JacksonJsonArray(ArrayNode json) { 266 this.nativeArray = json; 267 } 268 269 @Override 270 public Object getValue() { 271 return null; 272 } 273 274 @Override 275 public int size() { 276 return nativeArray.size(); 277 } 278 279 @Override 280 public BaseJsonLikeValue get(int index) { 281 Integer key = index; 282 BaseJsonLikeValue result = null; 283 if (jsonLikeMap.containsKey(key)) { 284 result = jsonLikeMap.get(key); 285 } else { 286 JsonNode child = nativeArray.get(index); 287 if (child != null) { 288 result = new JacksonJsonValue(child); 289 } 290 jsonLikeMap.put(key, result); 291 } 292 return result; 293 } 294 } 295 296 private static class JacksonJsonValue extends BaseJsonLikeValue { 297 private final JsonNode nativeValue; 298 private BaseJsonLikeObject jsonLikeObject = null; 299 private BaseJsonLikeArray jsonLikeArray = null; 300 301 public JacksonJsonValue(JsonNode jsonNode) { 302 this.nativeValue = jsonNode; 303 } 304 305 @Override 306 public Object getValue() { 307 if (nativeValue != null && nativeValue.isValueNode()) { 308 if (nativeValue.isNumber()) { 309 return nativeValue.numberValue(); 310 } 311 312 if (nativeValue.isBoolean()) { 313 return nativeValue.booleanValue(); 314 } 315 316 return nativeValue.asText(); 317 } 318 return null; 319 } 320 321 @Override 322 public ValueType getJsonType() { 323 if (null == nativeValue) { 324 return ValueType.NULL; 325 } 326 327 switch (nativeValue.getNodeType()) { 328 case NULL: 329 case MISSING: 330 return ValueType.NULL; 331 case OBJECT: 332 return ValueType.OBJECT; 333 case ARRAY: 334 return ValueType.ARRAY; 335 case POJO: 336 case BINARY: 337 case STRING: 338 case NUMBER: 339 case BOOLEAN: 340 default: 341 break; 342 } 343 344 return ValueType.SCALAR; 345 } 346 347 @Override 348 public ScalarType getDataType() { 349 if (nativeValue != null && nativeValue.isValueNode()) { 350 if (nativeValue.isNumber()) { 351 return ScalarType.NUMBER; 352 } 353 if (nativeValue.isTextual()) { 354 return ScalarType.STRING; 355 } 356 if (nativeValue.isBoolean()) { 357 return ScalarType.BOOLEAN; 358 } 359 } 360 return null; 361 } 362 363 @Override 364 public BaseJsonLikeArray getAsArray() { 365 if (nativeValue != null && nativeValue.isArray()) { 366 if (null == jsonLikeArray) { 367 jsonLikeArray = new JacksonJsonArray((ArrayNode) nativeValue); 368 } 369 } 370 return jsonLikeArray; 371 } 372 373 @Override 374 public BaseJsonLikeObject getAsObject() { 375 if (nativeValue != null && nativeValue.isObject()) { 376 if (null == jsonLikeObject) { 377 jsonLikeObject = new JacksonJsonObject((ObjectNode) nativeValue); 378 } 379 } 380 return jsonLikeObject; 381 } 382 383 @Override 384 public Number getAsNumber() { 385 return nativeValue != null ? nativeValue.numberValue() : null; 386 } 387 388 @Override 389 public String getAsString() { 390 if (nativeValue != null) { 391 if (nativeValue instanceof DecimalNode) { 392 BigDecimal value = nativeValue.decimalValue(); 393 return value.toPlainString(); 394 } 395 return nativeValue.asText(); 396 } 397 return null; 398 } 399 400 @Override 401 public boolean getAsBoolean() { 402 if (nativeValue != null && nativeValue.isValueNode() && nativeValue.isBoolean()) { 403 return nativeValue.asBoolean(); 404 } 405 return super.getAsBoolean(); 406 } 407 } 408 409 private static ObjectMapper createObjectMapper() { 410 ObjectMapper retVal = JsonMapper.builder() 411 .enable(JsonReadFeature.ALLOW_LEADING_PLUS_SIGN_FOR_NUMBERS) 412 .build(); 413 retVal = retVal.setNodeFactory(new JsonNodeFactory(true)); 414 retVal = retVal.enable(DeserializationFeature.USE_BIG_DECIMAL_FOR_FLOATS); 415 retVal = retVal.enable(DeserializationFeature.FAIL_ON_TRAILING_TOKENS); 416 retVal = retVal.disable(JsonParser.Feature.INCLUDE_SOURCE_IN_LOCATION); 417 retVal = retVal.disable(JsonGenerator.Feature.AUTO_CLOSE_TARGET); 418 retVal = retVal.disable(JsonParser.Feature.AUTO_CLOSE_SOURCE); 419 retVal = retVal.configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true); 420 421 retVal.getFactory().setStreamReadConstraints(createStreamReadConstraints()); 422 423 return retVal; 424 } 425 426 private static StreamReadConstraints createStreamReadConstraints() { 427 return StreamReadConstraints.builder() 428 .maxStringLength(Integer.MAX_VALUE) 429 .build(); 430 } 431}