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; 021 022import ca.uhn.fhir.context.BaseRuntimeChildDefinition; 023import ca.uhn.fhir.context.BaseRuntimeElementCompositeDefinition; 024import ca.uhn.fhir.context.BaseRuntimeElementDefinition; 025import ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum; 026import ca.uhn.fhir.context.ConfigurationException; 027import ca.uhn.fhir.context.FhirContext; 028import ca.uhn.fhir.context.FhirVersionEnum; 029import ca.uhn.fhir.context.RuntimeChildContainedResources; 030import ca.uhn.fhir.context.RuntimeChildDeclaredExtensionDefinition; 031import ca.uhn.fhir.context.RuntimeChildNarrativeDefinition; 032import ca.uhn.fhir.context.RuntimeChildUndeclaredExtensionDefinition; 033import ca.uhn.fhir.context.RuntimeResourceDefinition; 034import ca.uhn.fhir.i18n.Msg; 035import ca.uhn.fhir.model.api.ExtensionDt; 036import ca.uhn.fhir.model.api.IPrimitiveDatatype; 037import ca.uhn.fhir.model.api.IResource; 038import ca.uhn.fhir.model.api.ISupportsUndeclaredExtensions; 039import ca.uhn.fhir.model.api.ResourceMetadataKeyEnum; 040import ca.uhn.fhir.model.api.Tag; 041import ca.uhn.fhir.model.api.TagList; 042import ca.uhn.fhir.model.base.composite.BaseCodingDt; 043import ca.uhn.fhir.model.base.composite.BaseContainedDt; 044import ca.uhn.fhir.model.primitive.IdDt; 045import ca.uhn.fhir.model.primitive.InstantDt; 046import ca.uhn.fhir.narrative.INarrativeGenerator; 047import ca.uhn.fhir.parser.json.BaseJsonLikeArray; 048import ca.uhn.fhir.parser.json.BaseJsonLikeObject; 049import ca.uhn.fhir.parser.json.BaseJsonLikeValue; 050import ca.uhn.fhir.parser.json.BaseJsonLikeValue.ScalarType; 051import ca.uhn.fhir.parser.json.BaseJsonLikeValue.ValueType; 052import ca.uhn.fhir.parser.json.BaseJsonLikeWriter; 053import ca.uhn.fhir.parser.json.JsonLikeStructure; 054import ca.uhn.fhir.parser.json.jackson.JacksonStructure; 055import ca.uhn.fhir.rest.api.EncodingEnum; 056import ca.uhn.fhir.util.ElementUtil; 057import ca.uhn.fhir.util.FhirTerser; 058import org.apache.commons.lang3.StringUtils; 059import org.apache.commons.lang3.Validate; 060import org.apache.commons.text.WordUtils; 061import org.hl7.fhir.instance.model.api.IBase; 062import org.hl7.fhir.instance.model.api.IBaseBooleanDatatype; 063import org.hl7.fhir.instance.model.api.IBaseDecimalDatatype; 064import org.hl7.fhir.instance.model.api.IBaseExtension; 065import org.hl7.fhir.instance.model.api.IBaseHasExtensions; 066import org.hl7.fhir.instance.model.api.IBaseHasModifierExtensions; 067import org.hl7.fhir.instance.model.api.IBaseIntegerDatatype; 068import org.hl7.fhir.instance.model.api.IBaseResource; 069import org.hl7.fhir.instance.model.api.IDomainResource; 070import org.hl7.fhir.instance.model.api.IIdType; 071import org.hl7.fhir.instance.model.api.INarrative; 072import org.hl7.fhir.instance.model.api.IPrimitiveType; 073 074import java.io.IOException; 075import java.io.Reader; 076import java.io.Writer; 077import java.math.BigDecimal; 078import java.util.ArrayList; 079import java.util.Collections; 080import java.util.Iterator; 081import java.util.List; 082import java.util.Map; 083 084import static ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum.ID_DATATYPE; 085import static ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum.PRIMITIVE_DATATYPE; 086import static org.apache.commons.lang3.StringUtils.defaultString; 087import static org.apache.commons.lang3.StringUtils.isBlank; 088import static org.apache.commons.lang3.StringUtils.isNotBlank; 089 090/** 091 * This class is the FHIR JSON parser/encoder. Users should not interact with this class directly, but should use 092 * {@link FhirContext#newJsonParser()} to get an instance. 093 */ 094public class JsonParser extends BaseParser implements IJsonLikeParser { 095 096 private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(JsonParser.HeldExtension.class); 097 098 private boolean myPrettyPrint; 099 100 private Boolean myIsSupportsFhirComment; 101 102 /** 103 * Do not use this constructor, the recommended way to obtain a new instance of the JSON parser is to invoke 104 * {@link FhirContext#newJsonParser()}. 105 * 106 * @param theParserErrorHandler 107 */ 108 public JsonParser(FhirContext theContext, IParserErrorHandler theParserErrorHandler) { 109 super(theContext, theParserErrorHandler); 110 } 111 112 private boolean addToHeldComments( 113 int valueIdx, List<String> theCommentsToAdd, ArrayList<ArrayList<String>> theListToAddTo) { 114 if (theCommentsToAdd.size() > 0) { 115 theListToAddTo.ensureCapacity(valueIdx); 116 while (theListToAddTo.size() <= valueIdx) { 117 theListToAddTo.add(null); 118 } 119 if (theListToAddTo.get(valueIdx) == null) { 120 theListToAddTo.set(valueIdx, new ArrayList<>()); 121 } 122 theListToAddTo.get(valueIdx).addAll(theCommentsToAdd); 123 return true; 124 } 125 return false; 126 } 127 128 private boolean addToHeldExtensions( 129 int valueIdx, 130 List<? extends IBaseExtension<?, ?>> ext, 131 ArrayList<ArrayList<HeldExtension>> list, 132 boolean theIsModifier, 133 CompositeChildElement theChildElem, 134 CompositeChildElement theParent, 135 EncodeContext theEncodeContext, 136 boolean theContainedResource, 137 IBase theContainingElement) { 138 boolean retVal = false; 139 if (ext.size() > 0) { 140 Boolean encodeExtension = null; 141 for (IBaseExtension<?, ?> next : ext) { 142 143 if (next.isEmpty()) { 144 continue; 145 } 146 147 // Make sure we respect _summary and _elements 148 if (encodeExtension == null) { 149 encodeExtension = 150 isEncodeExtension(theParent, theEncodeContext, theContainedResource, theContainingElement); 151 } 152 153 if (encodeExtension) { 154 HeldExtension extension = new HeldExtension(next, theIsModifier, theChildElem, theParent); 155 list.ensureCapacity(valueIdx); 156 while (list.size() <= valueIdx) { 157 list.add(null); 158 } 159 ArrayList<HeldExtension> extensionList = list.get(valueIdx); 160 if (extensionList == null) { 161 extensionList = new ArrayList<>(); 162 list.set(valueIdx, extensionList); 163 } 164 extensionList.add(extension); 165 retVal = true; 166 } 167 } 168 } 169 return retVal; 170 } 171 172 private void addToHeldIds(int theValueIdx, ArrayList<String> theListToAddTo, String theId) { 173 theListToAddTo.ensureCapacity(theValueIdx); 174 while (theListToAddTo.size() <= theValueIdx) { 175 theListToAddTo.add(null); 176 } 177 if (theListToAddTo.get(theValueIdx) == null) { 178 theListToAddTo.set(theValueIdx, theId); 179 } 180 } 181 182 // private void assertObjectOfType(JsonLikeValue theResourceTypeObj, Object theValueType, String thePosition) { 183 // if (theResourceTypeObj == null) { 184 // throw new DataFormatException(Msg.code(1836) + "Invalid JSON content detected, missing required element: '" + 185 // thePosition + "'"); 186 // } 187 // 188 // if (theResourceTypeObj.getValueType() != theValueType) { 189 // throw new DataFormatException(Msg.code(1837) + "Invalid content of element " + thePosition + ", expected " + 190 // theValueType); 191 // } 192 // } 193 194 private void beginArray(BaseJsonLikeWriter theEventWriter, String arrayName) throws IOException { 195 theEventWriter.beginArray(arrayName); 196 } 197 198 private void beginObject(BaseJsonLikeWriter theEventWriter, String arrayName) throws IOException { 199 theEventWriter.beginObject(arrayName); 200 } 201 202 private BaseJsonLikeWriter createJsonWriter(Writer theWriter) throws IOException { 203 JsonLikeStructure jsonStructure = new JacksonStructure(); 204 return jsonStructure.getJsonLikeWriter(theWriter); 205 } 206 207 public void doEncodeResourceToJsonLikeWriter( 208 IBaseResource theResource, BaseJsonLikeWriter theEventWriter, EncodeContext theEncodeContext) 209 throws IOException { 210 if (myPrettyPrint) { 211 theEventWriter.setPrettyPrint(myPrettyPrint); 212 } 213 theEventWriter.init(); 214 215 RuntimeResourceDefinition resDef = getContext().getResourceDefinition(theResource); 216 encodeResourceToJsonStreamWriter(resDef, theResource, theEventWriter, null, false, theEncodeContext); 217 theEventWriter.flush(); 218 } 219 220 @Override 221 protected void doEncodeResourceToWriter(IBaseResource theResource, Writer theWriter, EncodeContext theEncodeContext) 222 throws IOException { 223 BaseJsonLikeWriter eventWriter = createJsonWriter(theWriter); 224 doEncodeResourceToJsonLikeWriter(theResource, eventWriter, theEncodeContext); 225 eventWriter.close(); 226 } 227 228 @Override 229 protected void doEncodeToWriter(IBase theElement, Writer theWriter, EncodeContext theEncodeContext) 230 throws IOException, DataFormatException { 231 BaseJsonLikeWriter eventWriter = createJsonWriter(theWriter); 232 eventWriter.beginObject(); 233 encodeCompositeElementToStreamWriter(null, null, theElement, eventWriter, false, null, theEncodeContext); 234 eventWriter.endObject(); 235 eventWriter.close(); 236 } 237 238 @Override 239 public <T extends IBaseResource> T doParseResource(Class<T> theResourceType, Reader theReader) { 240 JsonLikeStructure jsonStructure = new JacksonStructure(); 241 jsonStructure.load(theReader); 242 243 T retVal = doParseResource(theResourceType, jsonStructure); 244 245 return retVal; 246 } 247 248 public <T extends IBaseResource> T doParseResource(Class<T> theResourceType, JsonLikeStructure theJsonStructure) { 249 BaseJsonLikeObject object = theJsonStructure.getRootObject(); 250 251 BaseJsonLikeValue resourceTypeObj = object.get("resourceType"); 252 if (resourceTypeObj == null || !resourceTypeObj.isString() || isBlank(resourceTypeObj.getAsString())) { 253 throw new DataFormatException( 254 Msg.code(1838) + "Invalid JSON content detected, missing required element: 'resourceType'"); 255 } 256 257 String resourceType = resourceTypeObj.getAsString(); 258 259 ParserState<? extends IBaseResource> state = 260 ParserState.getPreResourceInstance(this, theResourceType, getContext(), true, getErrorHandler()); 261 state.enteringNewElement(null, resourceType); 262 263 parseChildren(object, state); 264 265 state.endingElement(); 266 state.endingElement(); 267 268 @SuppressWarnings("unchecked") 269 T retVal = (T) state.getObject(); 270 271 return retVal; 272 } 273 274 private void encodeChildElementToStreamWriter( 275 RuntimeResourceDefinition theResDef, 276 IBaseResource theResource, 277 BaseJsonLikeWriter theEventWriter, 278 IBase theNextValue, 279 BaseRuntimeElementDefinition<?> theChildDef, 280 String theChildName, 281 boolean theContainedResource, 282 CompositeChildElement theChildElem, 283 boolean theForceEmpty, 284 EncodeContext theEncodeContext) 285 throws IOException { 286 287 switch (theChildDef.getChildType()) { 288 case EXTENSION_DECLARED: 289 break; 290 case ID_DATATYPE: { 291 IIdType value = (IIdType) theNextValue; 292 String encodedValue = "id".equals(theChildName) ? value.getIdPart() : value.getValue(); 293 if (isBlank(encodedValue)) { 294 break; 295 } 296 if (theChildName != null) { 297 write(theEventWriter, theChildName, encodedValue); 298 } else { 299 theEventWriter.write(encodedValue); 300 } 301 break; 302 } 303 case PRIMITIVE_DATATYPE: { 304 final IPrimitiveType<?> value = (IPrimitiveType<?>) theNextValue; 305 final String valueStr = value.getValueAsString(); 306 if (isBlank(valueStr)) { 307 if (theForceEmpty) { 308 theEventWriter.writeNull(); 309 } 310 break; 311 } 312 313 // check for the common case first - String value types 314 Object valueObj = value.getValue(); 315 if (valueObj instanceof String) { 316 if (theChildName != null) { 317 theEventWriter.write(theChildName, valueStr); 318 } else { 319 theEventWriter.write(valueStr); 320 } 321 break; 322 } else if (valueObj instanceof Long) { 323 if (theChildName != null) { 324 theEventWriter.write(theChildName, (long) valueObj); 325 } else { 326 theEventWriter.write((long) valueObj); 327 } 328 break; 329 } 330 331 if (value instanceof IBaseIntegerDatatype) { 332 if (theChildName != null) { 333 write(theEventWriter, theChildName, ((IBaseIntegerDatatype) value).getValue()); 334 } else { 335 theEventWriter.write(((IBaseIntegerDatatype) value).getValue()); 336 } 337 } else if (value instanceof IBaseDecimalDatatype) { 338 BigDecimal decimalValue = ((IBaseDecimalDatatype) value).getValue(); 339 decimalValue = new BigDecimal(decimalValue.toString()) { 340 private static final long serialVersionUID = 1L; 341 342 @Override 343 public String toString() { 344 return value.getValueAsString(); 345 } 346 }; 347 if (theChildName != null) { 348 write(theEventWriter, theChildName, decimalValue); 349 } else { 350 theEventWriter.write(decimalValue); 351 } 352 } else if (value instanceof IBaseBooleanDatatype) { 353 if (theChildName != null) { 354 write(theEventWriter, theChildName, ((IBaseBooleanDatatype) value).getValue()); 355 } else { 356 Boolean booleanValue = ((IBaseBooleanDatatype) value).getValue(); 357 if (booleanValue != null) { 358 theEventWriter.write(booleanValue.booleanValue()); 359 } 360 } 361 } else { 362 if (theChildName != null) { 363 write(theEventWriter, theChildName, valueStr); 364 } else { 365 theEventWriter.write(valueStr); 366 } 367 } 368 break; 369 } 370 case RESOURCE_BLOCK: 371 case COMPOSITE_DATATYPE: { 372 if (theChildName != null) { 373 theEventWriter.beginObject(theChildName); 374 } else { 375 theEventWriter.beginObject(); 376 } 377 encodeCompositeElementToStreamWriter( 378 theResDef, 379 theResource, 380 theNextValue, 381 theEventWriter, 382 theContainedResource, 383 theChildElem, 384 theEncodeContext); 385 theEventWriter.endObject(); 386 break; 387 } 388 case CONTAINED_RESOURCE_LIST: 389 case CONTAINED_RESOURCES: { 390 List<IBaseResource> containedResources = 391 theEncodeContext.getContainedResources().getContainedResources(); 392 if (containedResources.size() > 0) { 393 beginArray(theEventWriter, theChildName); 394 395 for (IBaseResource next : containedResources) { 396 IIdType resourceId = 397 theEncodeContext.getContainedResources().getResourceId(next); 398 String value = resourceId.getValue(); 399 encodeResourceToJsonStreamWriter( 400 theResDef, 401 next, 402 theEventWriter, 403 null, 404 true, 405 fixContainedResourceId(value), 406 theEncodeContext); 407 } 408 409 theEventWriter.endArray(); 410 } 411 break; 412 } 413 case PRIMITIVE_XHTML_HL7ORG: 414 case PRIMITIVE_XHTML: { 415 if (!isSuppressNarratives()) { 416 IPrimitiveType<?> dt = (IPrimitiveType<?>) theNextValue; 417 if (theChildName != null) { 418 write(theEventWriter, theChildName, dt.getValueAsString()); 419 } else { 420 theEventWriter.write(dt.getValueAsString()); 421 } 422 } else { 423 if (theChildName != null) { 424 // do nothing 425 } else { 426 theEventWriter.writeNull(); 427 } 428 } 429 break; 430 } 431 case RESOURCE: 432 IBaseResource resource = (IBaseResource) theNextValue; 433 RuntimeResourceDefinition def = getContext().getResourceDefinition(resource); 434 435 theEncodeContext.pushPath(def.getName(), true); 436 encodeResourceToJsonStreamWriter( 437 def, resource, theEventWriter, theChildName, theContainedResource, theEncodeContext); 438 theEncodeContext.popPath(); 439 440 break; 441 case UNDECL_EXT: 442 default: 443 throw new IllegalStateException(Msg.code(1839) + "Should not have this state here: " 444 + theChildDef.getChildType().name()); 445 } 446 } 447 448 private void encodeCompositeElementChildrenToStreamWriter( 449 RuntimeResourceDefinition theResDef, 450 IBaseResource theResource, 451 IBase theElement, 452 BaseJsonLikeWriter theEventWriter, 453 boolean theContainedResource, 454 CompositeChildElement theParent, 455 EncodeContext theEncodeContext) 456 throws IOException { 457 458 { 459 String elementId = getCompositeElementId(theElement); 460 if (isNotBlank(elementId)) { 461 write(theEventWriter, "id", elementId); 462 } 463 } 464 465 boolean haveWrittenExtensions = false; 466 Iterable<CompositeChildElement> compositeChildElements = 467 super.compositeChildIterator(theElement, theContainedResource, theParent, theEncodeContext); 468 for (CompositeChildElement nextChildElem : compositeChildElements) { 469 470 BaseRuntimeChildDefinition nextChild = nextChildElem.getDef(); 471 472 if (nextChildElem.getDef().getElementName().equals("extension") 473 || nextChildElem.getDef().getElementName().equals("modifierExtension") 474 || nextChild instanceof RuntimeChildDeclaredExtensionDefinition) { 475 if (!haveWrittenExtensions) { 476 extractAndWriteExtensionsAsDirectChild( 477 theElement, 478 theEventWriter, 479 getContext().getElementDefinition(theElement.getClass()), 480 theResDef, 481 theResource, 482 nextChildElem, 483 theParent, 484 theEncodeContext, 485 theContainedResource); 486 haveWrittenExtensions = true; 487 } 488 continue; 489 } 490 491 if (nextChild instanceof RuntimeChildNarrativeDefinition) { 492 INarrativeGenerator gen = getContext().getNarrativeGenerator(); 493 if (gen != null) { 494 INarrative narr; 495 if (theResource instanceof IResource) { 496 narr = ((IResource) theResource).getText(); 497 } else if (theResource instanceof IDomainResource) { 498 narr = ((IDomainResource) theResource).getText(); 499 } else { 500 narr = null; 501 } 502 if (narr != null && narr.isEmpty()) { 503 gen.populateResourceNarrative(getContext(), theResource); 504 if (!narr.isEmpty()) { 505 RuntimeChildNarrativeDefinition child = (RuntimeChildNarrativeDefinition) nextChild; 506 String childName = nextChild.getChildNameByDatatype(child.getDatatype()); 507 BaseRuntimeElementDefinition<?> type = child.getChildByName(childName); 508 encodeChildElementToStreamWriter( 509 theResDef, 510 theResource, 511 theEventWriter, 512 narr, 513 type, 514 childName, 515 theContainedResource, 516 nextChildElem, 517 false, 518 theEncodeContext); 519 continue; 520 } 521 } 522 } 523 } else if (nextChild instanceof RuntimeChildContainedResources) { 524 String childName = nextChild.getValidChildNames().iterator().next(); 525 BaseRuntimeElementDefinition<?> child = nextChild.getChildByName(childName); 526 encodeChildElementToStreamWriter( 527 theResDef, 528 theResource, 529 theEventWriter, 530 null, 531 child, 532 childName, 533 theContainedResource, 534 nextChildElem, 535 false, 536 theEncodeContext); 537 continue; 538 } 539 540 List<? extends IBase> values = nextChild.getAccessor().getValues(theElement); 541 values = preProcessValues(nextChild, theResource, values, nextChildElem, theEncodeContext); 542 543 if (values == null || values.isEmpty()) { 544 continue; 545 } 546 547 String currentChildName = null; 548 boolean inArray = false; 549 550 ArrayList<ArrayList<HeldExtension>> extensions = new ArrayList<>(0); 551 ArrayList<ArrayList<HeldExtension>> modifierExtensions = new ArrayList<>(0); 552 ArrayList<ArrayList<String>> comments = new ArrayList<>(0); 553 ArrayList<String> ids = new ArrayList<>(0); 554 555 int valueIdx = 0; 556 for (IBase nextValue : values) { 557 558 if (nextValue == null || nextValue.isEmpty()) { 559 if (nextValue instanceof BaseContainedDt) { 560 if (theContainedResource 561 || theEncodeContext.getContainedResources().isEmpty()) { 562 continue; 563 } 564 } else { 565 continue; 566 } 567 } 568 569 BaseParser.ChildNameAndDef childNameAndDef = super.getChildNameAndDef(nextChild, nextValue); 570 if (childNameAndDef == null) { 571 continue; 572 } 573 574 /* 575 * Often the two values below will be the same thing. There are cases though 576 * where they will not be. An example would be Observation.value, which is 577 * a choice type. If the value contains a Quantity, then: 578 * nextChildGenericName = "value" 579 * nextChildSpecificName = "valueQuantity" 580 */ 581 String nextChildSpecificName = childNameAndDef.getChildName(); 582 String nextChildGenericName = nextChild.getElementName(); 583 584 theEncodeContext.pushPath(nextChildGenericName, false); 585 586 BaseRuntimeElementDefinition<?> childDef = childNameAndDef.getChildDef(); 587 boolean primitive = childDef.getChildType() == ChildTypeEnum.PRIMITIVE_DATATYPE; 588 589 if ((childDef.getChildType() == ChildTypeEnum.CONTAINED_RESOURCES 590 || childDef.getChildType() == ChildTypeEnum.CONTAINED_RESOURCE_LIST) 591 && theContainedResource) { 592 continue; 593 } 594 595 boolean force = false; 596 if (primitive) { 597 if (nextValue instanceof ISupportsUndeclaredExtensions) { 598 List<ExtensionDt> ext = ((ISupportsUndeclaredExtensions) nextValue).getUndeclaredExtensions(); 599 force |= addToHeldExtensions( 600 valueIdx, 601 ext, 602 extensions, 603 false, 604 nextChildElem, 605 theParent, 606 theEncodeContext, 607 theContainedResource, 608 theElement); 609 610 ext = ((ISupportsUndeclaredExtensions) nextValue).getUndeclaredModifierExtensions(); 611 force |= addToHeldExtensions( 612 valueIdx, 613 ext, 614 modifierExtensions, 615 true, 616 nextChildElem, 617 theParent, 618 theEncodeContext, 619 theContainedResource, 620 theElement); 621 } else { 622 if (nextValue instanceof IBaseHasExtensions) { 623 IBaseHasExtensions element = (IBaseHasExtensions) nextValue; 624 List<? extends IBaseExtension<?, ?>> ext = element.getExtension(); 625 force |= addToHeldExtensions( 626 valueIdx, 627 ext, 628 extensions, 629 false, 630 nextChildElem, 631 theParent, 632 theEncodeContext, 633 theContainedResource, 634 theElement); 635 } 636 if (nextValue instanceof IBaseHasModifierExtensions) { 637 IBaseHasModifierExtensions element = (IBaseHasModifierExtensions) nextValue; 638 List<? extends IBaseExtension<?, ?>> ext = element.getModifierExtension(); 639 force |= addToHeldExtensions( 640 valueIdx, 641 ext, 642 modifierExtensions, 643 true, 644 nextChildElem, 645 theParent, 646 theEncodeContext, 647 theContainedResource, 648 theElement); 649 } 650 } 651 if (nextValue.hasFormatComment()) { 652 force |= addToHeldComments(valueIdx, nextValue.getFormatCommentsPre(), comments); 653 force |= addToHeldComments(valueIdx, nextValue.getFormatCommentsPost(), comments); 654 } 655 String elementId = getCompositeElementId(nextValue); 656 if (isNotBlank(elementId)) { 657 force = true; 658 addToHeldIds(valueIdx, ids, elementId); 659 } 660 } 661 662 if (currentChildName == null || !currentChildName.equals(nextChildSpecificName)) { 663 if (inArray) { 664 theEventWriter.endArray(); 665 } 666 BaseRuntimeChildDefinition replacedParentDefinition = nextChild.getReplacedParentDefinition(); 667 if (nextChild.isMultipleCardinality() 668 || (replacedParentDefinition != null && replacedParentDefinition.isMultipleCardinality())) { 669 beginArray(theEventWriter, nextChildSpecificName); 670 inArray = true; 671 encodeChildElementToStreamWriter( 672 theResDef, 673 theResource, 674 theEventWriter, 675 nextValue, 676 childDef, 677 null, 678 theContainedResource, 679 nextChildElem, 680 force, 681 theEncodeContext); 682 } else { 683 encodeChildElementToStreamWriter( 684 theResDef, 685 theResource, 686 theEventWriter, 687 nextValue, 688 childDef, 689 nextChildSpecificName, 690 theContainedResource, 691 nextChildElem, 692 false, 693 theEncodeContext); 694 } 695 currentChildName = nextChildSpecificName; 696 } else { 697 encodeChildElementToStreamWriter( 698 theResDef, 699 theResource, 700 theEventWriter, 701 nextValue, 702 childDef, 703 null, 704 theContainedResource, 705 nextChildElem, 706 force, 707 theEncodeContext); 708 } 709 710 valueIdx++; 711 theEncodeContext.popPath(); 712 } 713 714 if (inArray) { 715 theEventWriter.endArray(); 716 } 717 718 if (!extensions.isEmpty() 719 || !modifierExtensions.isEmpty() 720 || (!comments.isEmpty() && isSupportsFhirComment())) { 721 if (inArray) { 722 // If this is a repeatable field, the extensions go in an array too 723 beginArray(theEventWriter, '_' + currentChildName); 724 } else { 725 beginObject(theEventWriter, '_' + currentChildName); 726 } 727 728 for (int i = 0; i < valueIdx; i++) { 729 boolean haveContent = false; 730 731 List<HeldExtension> heldExts = Collections.emptyList(); 732 List<HeldExtension> heldModExts = Collections.emptyList(); 733 if (extensions.size() > i 734 && extensions.get(i) != null 735 && !extensions.get(i).isEmpty()) { 736 haveContent = true; 737 heldExts = extensions.get(i); 738 } 739 740 if (modifierExtensions.size() > i 741 && modifierExtensions.get(i) != null 742 && !modifierExtensions.get(i).isEmpty()) { 743 haveContent = true; 744 heldModExts = modifierExtensions.get(i); 745 } 746 747 ArrayList<String> nextComments; 748 if (comments.size() > i) { 749 nextComments = comments.get(i); 750 } else { 751 nextComments = null; 752 } 753 if (nextComments != null && !nextComments.isEmpty()) { 754 haveContent = true; 755 } 756 757 String elementId = null; 758 if (ids.size() > i) { 759 elementId = ids.get(i); 760 haveContent |= isNotBlank(elementId); 761 } 762 763 if (!haveContent) { 764 theEventWriter.writeNull(); 765 } else { 766 if (inArray) { 767 theEventWriter.beginObject(); 768 } 769 if (isNotBlank(elementId)) { 770 write(theEventWriter, "id", elementId); 771 } 772 if (nextComments != null && !nextComments.isEmpty()) { 773 if (isSupportsFhirComment()) { 774 beginArray(theEventWriter, "fhir_comments"); 775 for (String next : nextComments) { 776 theEventWriter.write(next); 777 } 778 theEventWriter.endArray(); 779 } 780 } 781 writeExtensionsAsDirectChild( 782 theResource, 783 theEventWriter, 784 theResDef, 785 heldExts, 786 heldModExts, 787 theEncodeContext, 788 theContainedResource); 789 if (inArray) { 790 theEventWriter.endObject(); 791 } 792 } 793 } 794 795 if (inArray) { 796 theEventWriter.endArray(); 797 } else { 798 theEventWriter.endObject(); 799 } 800 } 801 } 802 } 803 804 private boolean isSupportsFhirComment() { 805 if (myIsSupportsFhirComment == null) { 806 myIsSupportsFhirComment = !getContext().getVersion().getVersion().isNewerThan(FhirVersionEnum.DSTU2_1); 807 } 808 return myIsSupportsFhirComment; 809 } 810 811 private void encodeCompositeElementToStreamWriter( 812 RuntimeResourceDefinition theResDef, 813 IBaseResource theResource, 814 IBase theNextValue, 815 BaseJsonLikeWriter theEventWriter, 816 boolean theContainedResource, 817 CompositeChildElement theParent, 818 EncodeContext theEncodeContext) 819 throws IOException, DataFormatException { 820 821 writeCommentsPreAndPost(theNextValue, theEventWriter); 822 encodeCompositeElementChildrenToStreamWriter( 823 theResDef, 824 theResource, 825 theNextValue, 826 theEventWriter, 827 theContainedResource, 828 theParent, 829 theEncodeContext); 830 } 831 832 @Override 833 public void encodeResourceToJsonLikeWriter(IBaseResource theResource, BaseJsonLikeWriter theJsonLikeWriter) 834 throws IOException, DataFormatException { 835 Validate.notNull(theResource, "theResource can not be null"); 836 Validate.notNull(theJsonLikeWriter, "theJsonLikeWriter can not be null"); 837 838 if (theResource.getStructureFhirVersionEnum() 839 != getContext().getVersion().getVersion()) { 840 throw new IllegalArgumentException(Msg.code(1840) + "This parser is for FHIR version " 841 + getContext().getVersion().getVersion() + " - Can not encode a structure for version " 842 + theResource.getStructureFhirVersionEnum()); 843 } 844 845 EncodeContext encodeContext = 846 new EncodeContext(this, getContext().getParserOptions(), new FhirTerser.ContainedResources()); 847 String resourceName = getContext().getResourceType(theResource); 848 encodeContext.pushPath(resourceName, true); 849 doEncodeResourceToJsonLikeWriter(theResource, theJsonLikeWriter, encodeContext); 850 } 851 852 private void encodeResourceToJsonStreamWriter( 853 RuntimeResourceDefinition theResDef, 854 IBaseResource theResource, 855 BaseJsonLikeWriter theEventWriter, 856 String theObjectNameOrNull, 857 boolean theContainedResource, 858 EncodeContext theEncodeContext) 859 throws IOException { 860 IIdType resourceId = null; 861 862 if (StringUtils.isNotBlank(theResource.getIdElement().getIdPart())) { 863 resourceId = theResource.getIdElement(); 864 if (theResource.getIdElement().getValue().startsWith("urn:")) { 865 resourceId = null; 866 } 867 } 868 869 if (!theContainedResource) { 870 if (!super.shouldEncodeResourceId(theResource, theEncodeContext)) { 871 resourceId = null; 872 } else if (theEncodeContext.getResourcePath().size() == 1 && getEncodeForceResourceId() != null) { 873 resourceId = getEncodeForceResourceId(); 874 } 875 } 876 877 encodeResourceToJsonStreamWriter( 878 theResDef, 879 theResource, 880 theEventWriter, 881 theObjectNameOrNull, 882 theContainedResource, 883 resourceId, 884 theEncodeContext); 885 } 886 887 private void encodeResourceToJsonStreamWriter( 888 RuntimeResourceDefinition theResDef, 889 IBaseResource theResource, 890 BaseJsonLikeWriter theEventWriter, 891 String theObjectNameOrNull, 892 boolean theContainedResource, 893 IIdType theResourceId, 894 EncodeContext theEncodeContext) 895 throws IOException { 896 897 if (!super.shouldEncodeResource(theResDef.getName(), theEncodeContext)) { 898 return; 899 } 900 901 if (!theContainedResource) { 902 containResourcesInReferences(theResource, theEncodeContext); 903 } 904 905 RuntimeResourceDefinition resDef = getContext().getResourceDefinition(theResource); 906 907 if (theObjectNameOrNull == null) { 908 theEventWriter.beginObject(); 909 } else { 910 beginObject(theEventWriter, theObjectNameOrNull); 911 } 912 913 write(theEventWriter, "resourceType", resDef.getName()); 914 if (theResourceId != null && theResourceId.hasIdPart()) { 915 write(theEventWriter, "id", theResourceId.getIdPart()); 916 final List<HeldExtension> extensions = new ArrayList<>(0); 917 final List<HeldExtension> modifierExtensions = new ArrayList<>(0); 918 // Undeclared extensions 919 extractUndeclaredExtensions( 920 theResourceId, extensions, modifierExtensions, null, null, theEncodeContext, theContainedResource); 921 boolean haveExtension = !extensions.isEmpty(); 922 923 if (theResourceId.hasFormatComment() || haveExtension) { 924 beginObject(theEventWriter, "_id"); 925 if (theResourceId.hasFormatComment()) { 926 writeCommentsPreAndPost(theResourceId, theEventWriter); 927 } 928 if (haveExtension) { 929 writeExtensionsAsDirectChild( 930 theResource, 931 theEventWriter, 932 theResDef, 933 extensions, 934 modifierExtensions, 935 theEncodeContext, 936 theContainedResource); 937 } 938 theEventWriter.endObject(); 939 } 940 } 941 942 if (theResource instanceof IResource) { 943 parseMetaForDSTU2(theResDef, theResource, theEventWriter, theContainedResource, theEncodeContext, resDef); 944 } 945 946 encodeCompositeElementToStreamWriter( 947 theResDef, 948 theResource, 949 theResource, 950 theEventWriter, 951 theContainedResource, 952 new CompositeChildElement(resDef, theEncodeContext), 953 theEncodeContext); 954 955 theEventWriter.endObject(); 956 } 957 958 private void parseMetaForDSTU2( 959 RuntimeResourceDefinition theResDef, 960 IBaseResource theResource, 961 BaseJsonLikeWriter theEventWriter, 962 boolean theContainedResource, 963 EncodeContext theEncodeContext, 964 RuntimeResourceDefinition resDef) 965 throws IOException { 966 IResource resource = (IResource) theResource; 967 // Object securityLabelRawObj = 968 969 List<BaseCodingDt> securityLabels = 970 extractMetadataListNotNull(resource, ResourceMetadataKeyEnum.SECURITY_LABELS); 971 List<? extends IIdType> profiles = extractMetadataListNotNull(resource, ResourceMetadataKeyEnum.PROFILES); 972 profiles = super.getProfileTagsForEncoding(resource, profiles); 973 974 TagList tags = getMetaTagsForEncoding(resource, theEncodeContext); 975 InstantDt updated = (InstantDt) resource.getResourceMetadata().get(ResourceMetadataKeyEnum.UPDATED); 976 IdDt resourceId = resource.getId(); 977 String versionIdPart = resourceId.getVersionIdPart(); 978 if (isBlank(versionIdPart)) { 979 versionIdPart = ResourceMetadataKeyEnum.VERSION.get(resource); 980 } 981 List<Map.Entry<ResourceMetadataKeyEnum<?>, Object>> extensionMetadataKeys = getExtensionMetadataKeys(resource); 982 983 if (super.shouldEncodeResourceMeta(resource, theEncodeContext) 984 && (ElementUtil.isEmpty(versionIdPart, updated, securityLabels, tags, profiles) == false) 985 || !extensionMetadataKeys.isEmpty()) { 986 beginObject(theEventWriter, "meta"); 987 988 if (shouldEncodePath(resource, "meta.versionId", theEncodeContext)) { 989 writeOptionalTagWithTextNode(theEventWriter, "versionId", versionIdPart); 990 } 991 if (shouldEncodePath(resource, "meta.lastUpdated", theEncodeContext)) { 992 writeOptionalTagWithTextNode(theEventWriter, "lastUpdated", updated); 993 } 994 995 if (profiles != null && profiles.isEmpty() == false) { 996 beginArray(theEventWriter, "profile"); 997 for (IIdType profile : profiles) { 998 if (profile != null && isNotBlank(profile.getValue())) { 999 theEventWriter.write(profile.getValue()); 1000 } 1001 } 1002 theEventWriter.endArray(); 1003 } 1004 1005 if (securityLabels.isEmpty() == false) { 1006 beginArray(theEventWriter, "security"); 1007 for (BaseCodingDt securityLabel : securityLabels) { 1008 theEventWriter.beginObject(); 1009 theEncodeContext.pushPath("security", false); 1010 encodeCompositeElementChildrenToStreamWriter( 1011 resDef, 1012 resource, 1013 securityLabel, 1014 theEventWriter, 1015 theContainedResource, 1016 null, 1017 theEncodeContext); 1018 theEncodeContext.popPath(); 1019 theEventWriter.endObject(); 1020 } 1021 theEventWriter.endArray(); 1022 } 1023 1024 if (tags != null && tags.isEmpty() == false) { 1025 beginArray(theEventWriter, "tag"); 1026 for (Tag tag : tags) { 1027 if (tag.isEmpty()) { 1028 continue; 1029 } 1030 theEventWriter.beginObject(); 1031 writeOptionalTagWithTextNode(theEventWriter, "system", tag.getScheme()); 1032 writeOptionalTagWithTextNode(theEventWriter, "code", tag.getTerm()); 1033 writeOptionalTagWithTextNode(theEventWriter, "display", tag.getLabel()); 1034 writeOptionalTagWithTextNode(theEventWriter, "version", tag.getVersion()); 1035 write(theEventWriter, "userSelected", tag.getUserSelectedBoolean()); 1036 theEventWriter.endObject(); 1037 } 1038 theEventWriter.endArray(); 1039 } 1040 1041 addExtensionMetadata( 1042 theResDef, 1043 theResource, 1044 theContainedResource, 1045 extensionMetadataKeys, 1046 resDef, 1047 theEventWriter, 1048 theEncodeContext); 1049 1050 theEventWriter.endObject(); // end meta 1051 } 1052 } 1053 1054 private void addExtensionMetadata( 1055 RuntimeResourceDefinition theResDef, 1056 IBaseResource theResource, 1057 boolean theContainedResource, 1058 List<Map.Entry<ResourceMetadataKeyEnum<?>, Object>> extensionMetadataKeys, 1059 RuntimeResourceDefinition resDef, 1060 BaseJsonLikeWriter theEventWriter, 1061 EncodeContext theEncodeContext) 1062 throws IOException { 1063 if (extensionMetadataKeys.isEmpty()) { 1064 return; 1065 } 1066 1067 ExtensionDt metaResource = new ExtensionDt(); 1068 for (Map.Entry<ResourceMetadataKeyEnum<?>, Object> entry : extensionMetadataKeys) { 1069 metaResource.addUndeclaredExtension((ExtensionDt) entry.getValue()); 1070 } 1071 encodeCompositeElementToStreamWriter( 1072 theResDef, 1073 theResource, 1074 metaResource, 1075 theEventWriter, 1076 theContainedResource, 1077 new CompositeChildElement(resDef, theEncodeContext), 1078 theEncodeContext); 1079 } 1080 1081 /** 1082 * This is useful only for the two cases where extensions are encoded as direct children (e.g. not in some object 1083 * called _name): resource extensions, and extension extensions 1084 */ 1085 private void extractAndWriteExtensionsAsDirectChild( 1086 IBase theElement, 1087 BaseJsonLikeWriter theEventWriter, 1088 BaseRuntimeElementDefinition<?> theElementDef, 1089 RuntimeResourceDefinition theResDef, 1090 IBaseResource theResource, 1091 CompositeChildElement theChildElem, 1092 CompositeChildElement theParent, 1093 EncodeContext theEncodeContext, 1094 boolean theContainedResource) 1095 throws IOException { 1096 List<HeldExtension> extensions = new ArrayList<>(0); 1097 List<HeldExtension> modifierExtensions = new ArrayList<>(0); 1098 1099 // Undeclared extensions 1100 extractUndeclaredExtensions( 1101 theElement, 1102 extensions, 1103 modifierExtensions, 1104 theChildElem, 1105 theParent, 1106 theEncodeContext, 1107 theContainedResource); 1108 1109 // Declared extensions 1110 if (theElementDef != null) { 1111 extractDeclaredExtensions(theElement, theElementDef, extensions, modifierExtensions, theChildElem); 1112 } 1113 1114 // Write the extensions 1115 writeExtensionsAsDirectChild( 1116 theResource, 1117 theEventWriter, 1118 theResDef, 1119 extensions, 1120 modifierExtensions, 1121 theEncodeContext, 1122 theContainedResource); 1123 } 1124 1125 private void extractDeclaredExtensions( 1126 IBase theResource, 1127 BaseRuntimeElementDefinition<?> resDef, 1128 List<HeldExtension> extensions, 1129 List<HeldExtension> modifierExtensions, 1130 CompositeChildElement theChildElem) { 1131 for (RuntimeChildDeclaredExtensionDefinition nextDef : resDef.getExtensionsNonModifier()) { 1132 for (IBase nextValue : nextDef.getAccessor().getValues(theResource)) { 1133 if (nextValue != null) { 1134 if (nextValue.isEmpty()) { 1135 continue; 1136 } 1137 extensions.add(new HeldExtension(nextDef, nextValue, theChildElem)); 1138 } 1139 } 1140 } 1141 for (RuntimeChildDeclaredExtensionDefinition nextDef : resDef.getExtensionsModifier()) { 1142 for (IBase nextValue : nextDef.getAccessor().getValues(theResource)) { 1143 if (nextValue != null) { 1144 if (nextValue.isEmpty()) { 1145 continue; 1146 } 1147 modifierExtensions.add(new HeldExtension(nextDef, nextValue, theChildElem)); 1148 } 1149 } 1150 } 1151 } 1152 1153 private void extractUndeclaredExtensions( 1154 IBase theElement, 1155 List<HeldExtension> extensions, 1156 List<HeldExtension> modifierExtensions, 1157 CompositeChildElement theChildElem, 1158 CompositeChildElement theParent, 1159 EncodeContext theEncodeContext, 1160 boolean theContainedResource) { 1161 if (theElement instanceof ISupportsUndeclaredExtensions) { 1162 ISupportsUndeclaredExtensions element = (ISupportsUndeclaredExtensions) theElement; 1163 List<ExtensionDt> ext = element.getUndeclaredExtensions(); 1164 for (ExtensionDt next : ext) { 1165 if (next == null || next.isEmpty()) { 1166 continue; 1167 } 1168 extensions.add(new HeldExtension(next, false, theChildElem, theParent)); 1169 } 1170 1171 ext = element.getUndeclaredModifierExtensions(); 1172 for (ExtensionDt next : ext) { 1173 if (next == null || next.isEmpty()) { 1174 continue; 1175 } 1176 modifierExtensions.add(new HeldExtension(next, true, theChildElem, theParent)); 1177 } 1178 } else { 1179 if (theElement instanceof IBaseHasExtensions) { 1180 IBaseHasExtensions element = (IBaseHasExtensions) theElement; 1181 List<? extends IBaseExtension<?, ?>> ext = element.getExtension(); 1182 Boolean encodeExtension = null; 1183 for (IBaseExtension<?, ?> next : ext) { 1184 if (next == null 1185 || (ElementUtil.isEmpty(next.getValue()) 1186 && next.getExtension().isEmpty())) { 1187 continue; 1188 } 1189 1190 // Make sure we respect _elements and _summary 1191 if (encodeExtension == null) { 1192 encodeExtension = isEncodeExtension(theParent, theEncodeContext, theContainedResource, element); 1193 } 1194 if (encodeExtension) { 1195 HeldExtension extension = new HeldExtension(next, false, theChildElem, theParent); 1196 extensions.add(extension); 1197 } 1198 } 1199 } 1200 if (theElement instanceof IBaseHasModifierExtensions) { 1201 IBaseHasModifierExtensions element = (IBaseHasModifierExtensions) theElement; 1202 List<? extends IBaseExtension<?, ?>> ext = element.getModifierExtension(); 1203 for (IBaseExtension<?, ?> next : ext) { 1204 if (next == null || next.isEmpty()) { 1205 continue; 1206 } 1207 1208 HeldExtension extension = new HeldExtension(next, true, theChildElem, theParent); 1209 modifierExtensions.add(extension); 1210 } 1211 } 1212 } 1213 } 1214 1215 private boolean isEncodeExtension( 1216 CompositeChildElement theParent, 1217 EncodeContext theEncodeContext, 1218 boolean theContainedResource, 1219 IBase theElement) { 1220 BaseRuntimeElementDefinition<?> runtimeElementDefinition = 1221 getContext().getElementDefinition(theElement.getClass()); 1222 boolean retVal = true; 1223 if (runtimeElementDefinition instanceof BaseRuntimeElementCompositeDefinition) { 1224 BaseRuntimeElementCompositeDefinition definition = 1225 (BaseRuntimeElementCompositeDefinition) runtimeElementDefinition; 1226 BaseRuntimeChildDefinition childDef = definition.getChildByName("extension"); 1227 CompositeChildElement c = new CompositeChildElement(theParent, childDef, theEncodeContext); 1228 retVal = c.shouldBeEncoded(theContainedResource); 1229 } 1230 return retVal; 1231 } 1232 1233 @Override 1234 public EncodingEnum getEncoding() { 1235 return EncodingEnum.JSON; 1236 } 1237 1238 private BaseJsonLikeArray grabJsonArray(BaseJsonLikeObject theObject, String nextName, String thePosition) { 1239 BaseJsonLikeValue object = theObject.get(nextName); 1240 if (object == null || object.isNull()) { 1241 return null; 1242 } 1243 if (!object.isArray()) { 1244 throw new DataFormatException( 1245 Msg.code(1841) + "Syntax error parsing JSON FHIR structure: Expected ARRAY at element '" 1246 + thePosition + "', found '" + object.getJsonType() + "'"); 1247 } 1248 return object.getAsArray(); 1249 } 1250 1251 private void parseAlternates( 1252 BaseJsonLikeValue theAlternateVal, 1253 ParserState<?> theState, 1254 String theElementName, 1255 String theAlternateName) { 1256 if (theAlternateVal == null || theAlternateVal.isNull()) { 1257 return; 1258 } 1259 1260 if (theAlternateVal.isArray()) { 1261 BaseJsonLikeArray array = theAlternateVal.getAsArray(); 1262 if (array.size() > 1) { 1263 throw new DataFormatException(Msg.code(1842) + "Unexpected array of length " + array.size() 1264 + " (expected 0 or 1) for element: " + theElementName); 1265 } 1266 if (array.size() == 0) { 1267 return; 1268 } 1269 parseAlternates(array.get(0), theState, theElementName, theAlternateName); 1270 return; 1271 } 1272 1273 BaseJsonLikeValue alternateVal = theAlternateVal; 1274 if (alternateVal.isObject() == false) { 1275 getErrorHandler() 1276 .incorrectJsonType( 1277 null, theAlternateName, ValueType.OBJECT, null, alternateVal.getJsonType(), null); 1278 return; 1279 } 1280 1281 BaseJsonLikeObject alternate = alternateVal.getAsObject(); 1282 1283 for (Iterator<String> keyIter = alternate.keyIterator(); keyIter.hasNext(); ) { 1284 String nextKey = keyIter.next(); 1285 BaseJsonLikeValue nextVal = alternate.get(nextKey); 1286 if ("extension".equals(nextKey)) { 1287 boolean isModifier = false; 1288 BaseJsonLikeArray array = nextVal.getAsArray(); 1289 parseExtension(theState, array, isModifier); 1290 } else if ("modifierExtension".equals(nextKey)) { 1291 boolean isModifier = true; 1292 BaseJsonLikeArray array = nextVal.getAsArray(); 1293 parseExtension(theState, array, isModifier); 1294 } else if ("id".equals(nextKey)) { 1295 if (nextVal.isString()) { 1296 theState.attributeValue("id", nextVal.getAsString()); 1297 } else { 1298 getErrorHandler() 1299 .incorrectJsonType( 1300 null, 1301 "id", 1302 ValueType.SCALAR, 1303 ScalarType.STRING, 1304 nextVal.getJsonType(), 1305 nextVal.getDataType()); 1306 } 1307 } else if ("fhir_comments".equals(nextKey)) { 1308 parseFhirComments(nextVal, theState); 1309 } 1310 } 1311 } 1312 1313 private void parseChildren(BaseJsonLikeObject theObject, ParserState<?> theState) { 1314 int allUnderscoreNames = 0; 1315 int handledUnderscoreNames = 0; 1316 1317 for (Iterator<String> keyIter = theObject.keyIterator(); keyIter.hasNext(); ) { 1318 String nextName = keyIter.next(); 1319 if ("resourceType".equals(nextName)) { 1320 if (theState.isToplevelResourceElement()) { 1321 continue; 1322 } 1323 } else if ("extension".equals(nextName)) { 1324 BaseJsonLikeArray array = grabJsonArray(theObject, nextName, "extension"); 1325 parseExtension(theState, array, false); 1326 continue; 1327 } else if ("modifierExtension".equals(nextName)) { 1328 BaseJsonLikeArray array = grabJsonArray(theObject, nextName, "modifierExtension"); 1329 parseExtension(theState, array, true); 1330 continue; 1331 } else if (nextName.equals("fhir_comments")) { 1332 parseFhirComments(theObject.get(nextName), theState); 1333 continue; 1334 } else if (nextName.charAt(0) == '_') { 1335 allUnderscoreNames++; 1336 continue; 1337 } 1338 1339 BaseJsonLikeValue nextVal = theObject.get(nextName); 1340 String alternateName = '_' + nextName; 1341 BaseJsonLikeValue alternateVal = theObject.get(alternateName); 1342 if (alternateVal != null) { 1343 handledUnderscoreNames++; 1344 } 1345 1346 parseChildren(theState, nextName, nextVal, alternateVal, alternateName, false); 1347 } 1348 1349 // if (elementId != null) { 1350 // IBase object = (IBase) theState.getObject(); 1351 // if (object instanceof IIdentifiableElement) { 1352 // ((IIdentifiableElement) object).setElementSpecificId(elementId); 1353 // } else if (object instanceof IBaseResource) { 1354 // ((IBaseResource) object).getIdElement().setValue(elementId); 1355 // } 1356 // } 1357 1358 /* 1359 * This happens if an element has an extension but no actual value. I.e. 1360 * if a resource has a "_status" element but no corresponding "status" 1361 * element. This could be used to handle a null value with an extension 1362 * for example. 1363 */ 1364 if (allUnderscoreNames > handledUnderscoreNames) { 1365 for (Iterator<String> keyIter = theObject.keyIterator(); keyIter.hasNext(); ) { 1366 String alternateName = keyIter.next(); 1367 if (alternateName.startsWith("_") && alternateName.length() > 1) { 1368 BaseJsonLikeValue nextValue = theObject.get(alternateName); 1369 String nextName = alternateName.substring(1); 1370 1371 if (nextValue != null) { 1372 BaseJsonLikeValue nonAlternativeValue = theObject.get(nextName); 1373 1374 // Only alternate values with no corresponding "normal" value is unhandled from previous step. 1375 if (nonAlternativeValue != null) { 1376 continue; 1377 } 1378 1379 if (nextValue.isObject()) { 1380 if (theObject.get(nextName) == null) { 1381 theState.enteringNewElement(null, nextName); 1382 parseAlternates(nextValue, theState, alternateName, alternateName); 1383 theState.endingElement(); 1384 } 1385 } else { 1386 getErrorHandler() 1387 .incorrectJsonType( 1388 null, alternateName, ValueType.OBJECT, null, nextValue.getJsonType(), null); 1389 } 1390 } 1391 } 1392 } 1393 } 1394 } 1395 1396 private void parseChildren( 1397 ParserState<?> theState, 1398 String theName, 1399 BaseJsonLikeValue theJsonVal, 1400 BaseJsonLikeValue theAlternateVal, 1401 String theAlternateName, 1402 boolean theInArray) { 1403 if (theName.equals("id")) { 1404 if (!theJsonVal.isString()) { 1405 getErrorHandler() 1406 .incorrectJsonType( 1407 null, 1408 "id", 1409 ValueType.SCALAR, 1410 ScalarType.STRING, 1411 theJsonVal.getJsonType(), 1412 theJsonVal.getDataType()); 1413 } 1414 } 1415 1416 if (theJsonVal.isArray()) { 1417 BaseJsonLikeArray nextArray = theJsonVal.getAsArray(); 1418 1419 BaseJsonLikeValue alternateVal = theAlternateVal; 1420 if (alternateVal != null && alternateVal.isArray() == false) { 1421 getErrorHandler() 1422 .incorrectJsonType( 1423 null, theAlternateName, ValueType.ARRAY, null, alternateVal.getJsonType(), null); 1424 alternateVal = null; 1425 } 1426 1427 BaseJsonLikeArray nextAlternateArray = BaseJsonLikeValue.asArray(alternateVal); // could be null 1428 for (int i = 0; i < nextArray.size(); i++) { 1429 BaseJsonLikeValue nextObject = nextArray.get(i); 1430 BaseJsonLikeValue nextAlternate = null; 1431 if (nextAlternateArray != null && nextAlternateArray.size() >= (i + 1)) { 1432 nextAlternate = nextAlternateArray.get(i); 1433 } 1434 parseChildren(theState, theName, nextObject, nextAlternate, theAlternateName, true); 1435 } 1436 } else if (theJsonVal.isObject()) { 1437 if (!theInArray && theState.elementIsRepeating(theName)) { 1438 getErrorHandler().incorrectJsonType(null, theName, ValueType.ARRAY, null, ValueType.OBJECT, null); 1439 } 1440 1441 theState.enteringNewElement(null, theName); 1442 parseAlternates(theAlternateVal, theState, theAlternateName, theAlternateName); 1443 BaseJsonLikeObject nextObject = theJsonVal.getAsObject(); 1444 boolean preResource = false; 1445 if (theState.isPreResource()) { 1446 BaseJsonLikeValue resType = nextObject.get("resourceType"); 1447 if (resType == null || !resType.isString()) { 1448 throw new DataFormatException(Msg.code(1843) 1449 + "Missing required element 'resourceType' from JSON resource object, unable to parse"); 1450 } 1451 theState.enteringNewElement(null, resType.getAsString()); 1452 preResource = true; 1453 } 1454 parseChildren(nextObject, theState); 1455 if (preResource) { 1456 theState.endingElement(); 1457 } 1458 theState.endingElement(); 1459 } else if (theJsonVal.isNull()) { 1460 theState.enteringNewElement(null, theName); 1461 parseAlternates(theAlternateVal, theState, theAlternateName, theAlternateName); 1462 theState.endingElement(); 1463 } else { 1464 // must be a SCALAR 1465 theState.enteringNewElement(null, theName); 1466 String asString = theJsonVal.getAsString(); 1467 theState.attributeValue("value", asString); 1468 parseAlternates(theAlternateVal, theState, theAlternateName, theAlternateName); 1469 theState.endingElement(); 1470 } 1471 } 1472 1473 private void parseExtension(ParserState<?> theState, BaseJsonLikeArray theValues, boolean theIsModifier) { 1474 int allUnderscoreNames = 0; 1475 int handledUnderscoreNames = 0; 1476 1477 for (int i = 0; i < theValues.size(); i++) { 1478 BaseJsonLikeObject nextExtObj = BaseJsonLikeValue.asObject(theValues.get(i)); 1479 BaseJsonLikeValue jsonElement = nextExtObj.get("url"); 1480 String url; 1481 if (null == jsonElement || !(jsonElement.isScalar())) { 1482 String parentElementName; 1483 if (theIsModifier) { 1484 parentElementName = "modifierExtension"; 1485 } else { 1486 parentElementName = "extension"; 1487 } 1488 getErrorHandler() 1489 .missingRequiredElement(new ParseLocation().setParentElementName(parentElementName), "url"); 1490 url = null; 1491 } else { 1492 url = getExtensionUrl(jsonElement.getAsString()); 1493 } 1494 theState.enteringNewElementExtension(null, url, theIsModifier, getServerBaseUrl()); 1495 for (Iterator<String> keyIter = nextExtObj.keyIterator(); keyIter.hasNext(); ) { 1496 String next = keyIter.next(); 1497 if ("url".equals(next)) { 1498 continue; 1499 } else if ("extension".equals(next)) { 1500 BaseJsonLikeArray jsonVal = BaseJsonLikeValue.asArray(nextExtObj.get(next)); 1501 parseExtension(theState, jsonVal, false); 1502 } else if ("modifierExtension".equals(next)) { 1503 BaseJsonLikeArray jsonVal = BaseJsonLikeValue.asArray(nextExtObj.get(next)); 1504 parseExtension(theState, jsonVal, true); 1505 } else if (next.charAt(0) == '_') { 1506 allUnderscoreNames++; 1507 continue; 1508 } else { 1509 BaseJsonLikeValue jsonVal = nextExtObj.get(next); 1510 String alternateName = '_' + next; 1511 BaseJsonLikeValue alternateVal = nextExtObj.get(alternateName); 1512 if (alternateVal != null) { 1513 handledUnderscoreNames++; 1514 } 1515 parseChildren(theState, next, jsonVal, alternateVal, alternateName, false); 1516 } 1517 } 1518 1519 /* 1520 * This happens if an element has an extension but no actual value. I.e. 1521 * if a resource has a "_status" element but no corresponding "status" 1522 * element. This could be used to handle a null value with an extension 1523 * for example. 1524 */ 1525 if (allUnderscoreNames > handledUnderscoreNames) { 1526 for (Iterator<String> keyIter = nextExtObj.keyIterator(); keyIter.hasNext(); ) { 1527 String alternateName = keyIter.next(); 1528 if (alternateName.startsWith("_") && alternateName.length() > 1) { 1529 BaseJsonLikeValue nextValue = nextExtObj.get(alternateName); 1530 if (nextValue != null) { 1531 if (nextValue.isObject()) { 1532 String nextName = alternateName.substring(1); 1533 if (nextExtObj.get(nextName) == null) { 1534 theState.enteringNewElement(null, nextName); 1535 parseAlternates(nextValue, theState, alternateName, alternateName); 1536 theState.endingElement(); 1537 } 1538 } else { 1539 getErrorHandler() 1540 .incorrectJsonType( 1541 null, 1542 alternateName, 1543 ValueType.OBJECT, 1544 null, 1545 nextValue.getJsonType(), 1546 null); 1547 } 1548 } 1549 } 1550 } 1551 } 1552 theState.endingElement(); 1553 } 1554 } 1555 1556 private void parseFhirComments(BaseJsonLikeValue theObject, ParserState<?> theState) { 1557 if (isSupportsFhirComment()) { 1558 if (theObject.isArray()) { 1559 BaseJsonLikeArray comments = theObject.getAsArray(); 1560 for (int i = 0; i < comments.size(); i++) { 1561 BaseJsonLikeValue nextComment = comments.get(i); 1562 if (nextComment.isString()) { 1563 String commentText = nextComment.getAsString(); 1564 if (commentText != null) { 1565 theState.commentPre(commentText); 1566 } 1567 } 1568 } 1569 } 1570 } 1571 } 1572 1573 @Override 1574 public <T extends IBaseResource> T parseResource(Class<T> theResourceType, JsonLikeStructure theJsonLikeStructure) 1575 throws DataFormatException { 1576 1577 /***************************************************** 1578 * ************************************************* * 1579 * ** NOTE: this duplicates most of the code in ** * 1580 * ** BaseParser.parseResource(Class<T>, Reader). ** * 1581 * ** Unfortunately, there is no way to avoid ** * 1582 * ** this without doing some refactoring of the ** * 1583 * ** BaseParser class. ** * 1584 * ************************************************* * 1585 *****************************************************/ 1586 1587 /* 1588 * We do this so that the context can verify that the structure is for 1589 * the correct FHIR version 1590 */ 1591 if (theResourceType != null) { 1592 getContext().getResourceDefinition(theResourceType); 1593 } 1594 1595 // Actually do the parse 1596 T retVal = doParseResource(theResourceType, theJsonLikeStructure); 1597 1598 RuntimeResourceDefinition def = getContext().getResourceDefinition(retVal); 1599 if ("Bundle".equals(def.getName())) { 1600 1601 BaseRuntimeChildDefinition entryChild = def.getChildByName("entry"); 1602 BaseRuntimeElementCompositeDefinition<?> entryDef = 1603 (BaseRuntimeElementCompositeDefinition<?>) entryChild.getChildByName("entry"); 1604 List<IBase> entries = entryChild.getAccessor().getValues(retVal); 1605 if (entries != null) { 1606 for (IBase nextEntry : entries) { 1607 1608 /** 1609 * If Bundle.entry.fullUrl is populated, set the resource ID to that 1610 */ 1611 // TODO: should emit a warning and maybe notify the error handler if the resource ID doesn't match 1612 // the 1613 // fullUrl idPart 1614 BaseRuntimeChildDefinition fullUrlChild = entryDef.getChildByName("fullUrl"); 1615 if (fullUrlChild == null) { 1616 continue; // TODO: remove this once the data model in tinder plugin catches up to 1.2 1617 } 1618 List<IBase> fullUrl = fullUrlChild.getAccessor().getValues(nextEntry); 1619 if (fullUrl != null && !fullUrl.isEmpty()) { 1620 IPrimitiveType<?> value = (IPrimitiveType<?>) fullUrl.get(0); 1621 if (value.isEmpty() == false) { 1622 List<IBase> entryResources = entryDef.getChildByName("resource") 1623 .getAccessor() 1624 .getValues(nextEntry); 1625 if (entryResources != null && entryResources.size() > 0) { 1626 IBaseResource res = (IBaseResource) entryResources.get(0); 1627 String versionId = res.getIdElement().getVersionIdPart(); 1628 res.setId(value.getValueAsString()); 1629 if (isNotBlank(versionId) && res.getIdElement().hasVersionIdPart() == false) { 1630 res.setId(res.getIdElement().withVersion(versionId)); 1631 } 1632 } 1633 } 1634 } 1635 } 1636 } 1637 } 1638 1639 return retVal; 1640 } 1641 1642 @Override 1643 public IBaseResource parseResource(JsonLikeStructure theJsonLikeStructure) throws DataFormatException { 1644 return parseResource(null, theJsonLikeStructure); 1645 } 1646 1647 @Override 1648 public IParser setPrettyPrint(boolean thePrettyPrint) { 1649 myPrettyPrint = thePrettyPrint; 1650 return this; 1651 } 1652 1653 private void write(BaseJsonLikeWriter theEventWriter, String theChildName, Boolean theValue) throws IOException { 1654 if (theValue != null) { 1655 theEventWriter.write(theChildName, theValue.booleanValue()); 1656 } 1657 } 1658 1659 // private void parseExtensionInDstu2Style(boolean theModifier, ParserState<?> theState, String 1660 // theParentExtensionUrl, String theExtensionUrl, JsonArray theValues) { 1661 // String extUrl = UrlUtil.constructAbsoluteUrl(theParentExtensionUrl, theExtensionUrl); 1662 // theState.enteringNewElementExtension(null, extUrl, theModifier); 1663 // 1664 // for (int extIdx = 0; extIdx < theValues.size(); extIdx++) { 1665 // JsonObject nextExt = theValues.getJsonObject(extIdx); 1666 // for (String nextKey : nextExt.keySet()) { 1667 // // if (nextKey.startsWith("value") && nextKey.length() > 5 && 1668 // // myContext.getRuntimeChildUndeclaredExtensionDefinition().getChildByName(nextKey) != null) { 1669 // JsonElement jsonVal = nextExt.get(nextKey); 1670 // if (jsonVal.getValueType() == ValueType.ARRAY) { 1671 // /* 1672 // * Extension children which are arrays are sub-extensions. Any other value type should be treated as a value. 1673 // */ 1674 // JsonArray arrayValue = (JsonArray) jsonVal; 1675 // parseExtensionInDstu2Style(theModifier, theState, extUrl, nextKey, arrayValue); 1676 // } else { 1677 // parseChildren(theState, nextKey, jsonVal, null, null); 1678 // } 1679 // } 1680 // } 1681 // 1682 // theState.endingElement(); 1683 // } 1684 1685 private void write(BaseJsonLikeWriter theEventWriter, String theChildName, BigDecimal theDecimalValue) 1686 throws IOException { 1687 theEventWriter.write(theChildName, theDecimalValue); 1688 } 1689 1690 private void write(BaseJsonLikeWriter theEventWriter, String theChildName, Integer theValue) throws IOException { 1691 theEventWriter.write(theChildName, theValue); 1692 } 1693 1694 private void writeCommentsPreAndPost(IBase theNextValue, BaseJsonLikeWriter theEventWriter) throws IOException { 1695 if (theNextValue.hasFormatComment()) { 1696 if (isSupportsFhirComment()) { 1697 beginArray(theEventWriter, "fhir_comments"); 1698 List<String> pre = theNextValue.getFormatCommentsPre(); 1699 if (pre.isEmpty() == false) { 1700 for (String next : pre) { 1701 theEventWriter.write(next); 1702 } 1703 } 1704 List<String> post = theNextValue.getFormatCommentsPost(); 1705 if (post.isEmpty() == false) { 1706 for (String next : post) { 1707 theEventWriter.write(next); 1708 } 1709 } 1710 theEventWriter.endArray(); 1711 } 1712 } 1713 } 1714 1715 private void writeExtensionsAsDirectChild( 1716 IBaseResource theResource, 1717 BaseJsonLikeWriter theEventWriter, 1718 RuntimeResourceDefinition resDef, 1719 List<HeldExtension> extensions, 1720 List<HeldExtension> modifierExtensions, 1721 EncodeContext theEncodeContext, 1722 boolean theContainedResource) 1723 throws IOException { 1724 // Write Extensions 1725 if (extensions.isEmpty() == false) { 1726 theEncodeContext.pushPath("extension", false); 1727 beginArray(theEventWriter, "extension"); 1728 for (HeldExtension next : extensions) { 1729 next.write(resDef, theResource, theEventWriter, theEncodeContext, theContainedResource); 1730 } 1731 theEventWriter.endArray(); 1732 theEncodeContext.popPath(); 1733 } 1734 1735 // Write ModifierExtensions 1736 if (modifierExtensions.isEmpty() == false) { 1737 theEncodeContext.pushPath("modifierExtension", false); 1738 beginArray(theEventWriter, "modifierExtension"); 1739 for (HeldExtension next : modifierExtensions) { 1740 next.write(resDef, theResource, theEventWriter, theEncodeContext, theContainedResource); 1741 } 1742 theEventWriter.endArray(); 1743 theEncodeContext.popPath(); 1744 } 1745 } 1746 1747 private void writeOptionalTagWithTextNode( 1748 BaseJsonLikeWriter theEventWriter, String theElementName, IPrimitiveDatatype<?> thePrimitive) 1749 throws IOException { 1750 if (thePrimitive == null) { 1751 return; 1752 } 1753 String str = thePrimitive.getValueAsString(); 1754 writeOptionalTagWithTextNode(theEventWriter, theElementName, str); 1755 } 1756 1757 private void writeOptionalTagWithTextNode(BaseJsonLikeWriter theEventWriter, String theElementName, String theValue) 1758 throws IOException { 1759 if (StringUtils.isNotBlank(theValue)) { 1760 write(theEventWriter, theElementName, theValue); 1761 } 1762 } 1763 1764 private static void write(BaseJsonLikeWriter theWriter, String theName, String theValue) throws IOException { 1765 theWriter.write(theName, theValue); 1766 } 1767 1768 private class HeldExtension implements Comparable<HeldExtension> { 1769 1770 private CompositeChildElement myChildElem; 1771 private RuntimeChildDeclaredExtensionDefinition myDef; 1772 private boolean myModifier; 1773 private IBaseExtension<?, ?> myUndeclaredExtension; 1774 private IBase myValue; 1775 private CompositeChildElement myParent; 1776 1777 public HeldExtension( 1778 IBaseExtension<?, ?> theUndeclaredExtension, 1779 boolean theModifier, 1780 CompositeChildElement theChildElem, 1781 CompositeChildElement theParent) { 1782 assert theUndeclaredExtension != null; 1783 myUndeclaredExtension = theUndeclaredExtension; 1784 myModifier = theModifier; 1785 myChildElem = theChildElem; 1786 myParent = theParent; 1787 } 1788 1789 public HeldExtension( 1790 RuntimeChildDeclaredExtensionDefinition theDef, IBase theValue, CompositeChildElement theChildElem) { 1791 assert theDef != null; 1792 assert theValue != null; 1793 myDef = theDef; 1794 myValue = theValue; 1795 myChildElem = theChildElem; 1796 } 1797 1798 @Override 1799 public int compareTo(HeldExtension theArg0) { 1800 String url1 = myDef != null ? myDef.getExtensionUrl() : myUndeclaredExtension.getUrl(); 1801 String url2 = 1802 theArg0.myDef != null ? theArg0.myDef.getExtensionUrl() : theArg0.myUndeclaredExtension.getUrl(); 1803 url1 = defaultString(getExtensionUrl(url1)); 1804 url2 = defaultString(getExtensionUrl(url2)); 1805 return url1.compareTo(url2); 1806 } 1807 1808 private void managePrimitiveExtension( 1809 final IBase theValue, 1810 final RuntimeResourceDefinition theResDef, 1811 final IBaseResource theResource, 1812 final BaseJsonLikeWriter theEventWriter, 1813 final BaseRuntimeElementDefinition<?> def, 1814 final String childName, 1815 EncodeContext theEncodeContext, 1816 boolean theContainedResource) 1817 throws IOException { 1818 if (def.getChildType().equals(ID_DATATYPE) || def.getChildType().equals(PRIMITIVE_DATATYPE)) { 1819 final List<HeldExtension> extensions = new ArrayList<HeldExtension>(0); 1820 final List<HeldExtension> modifierExtensions = new ArrayList<HeldExtension>(0); 1821 // Undeclared extensions 1822 extractUndeclaredExtensions( 1823 theValue, 1824 extensions, 1825 modifierExtensions, 1826 myParent, 1827 null, 1828 theEncodeContext, 1829 theContainedResource); 1830 // Declared extensions 1831 extractDeclaredExtensions(theValue, def, extensions, modifierExtensions, myParent); 1832 boolean haveContent = false; 1833 if (!extensions.isEmpty() || !modifierExtensions.isEmpty()) { 1834 haveContent = true; 1835 } 1836 if (haveContent) { 1837 beginObject(theEventWriter, '_' + childName); 1838 writeExtensionsAsDirectChild( 1839 theResource, 1840 theEventWriter, 1841 theResDef, 1842 extensions, 1843 modifierExtensions, 1844 theEncodeContext, 1845 theContainedResource); 1846 theEventWriter.endObject(); 1847 } 1848 } 1849 } 1850 1851 public void write( 1852 RuntimeResourceDefinition theResDef, 1853 IBaseResource theResource, 1854 BaseJsonLikeWriter theEventWriter, 1855 EncodeContext theEncodeContext, 1856 boolean theContainedResource) 1857 throws IOException { 1858 if (myUndeclaredExtension != null) { 1859 writeUndeclaredExtension( 1860 theResDef, 1861 theResource, 1862 theEventWriter, 1863 myUndeclaredExtension, 1864 theEncodeContext, 1865 theContainedResource); 1866 } else { 1867 theEventWriter.beginObject(); 1868 1869 writeCommentsPreAndPost(myValue, theEventWriter); 1870 1871 JsonParser.write(theEventWriter, "url", getExtensionUrl(myDef.getExtensionUrl())); 1872 1873 /* 1874 * This makes sure that even if the extension contains a reference to a contained 1875 * resource which has a HAPI-assigned ID we'll still encode that ID. 1876 * 1877 * See #327 1878 */ 1879 List<? extends IBase> preProcessedValue = preProcessValues( 1880 myDef, theResource, Collections.singletonList(myValue), myChildElem, theEncodeContext); 1881 1882 // // Check for undeclared extensions on the declared extension 1883 // // (grrrrrr....) 1884 // if (myValue instanceof ISupportsUndeclaredExtensions) { 1885 // ISupportsUndeclaredExtensions value = (ISupportsUndeclaredExtensions)myValue; 1886 // List<ExtensionDt> exts = value.getUndeclaredExtensions(); 1887 // if (exts.size() > 0) { 1888 // ArrayList<IBase> newValueList = new ArrayList<IBase>(); 1889 // newValueList.addAll(preProcessedValue); 1890 // newValueList.addAll(exts); 1891 // preProcessedValue = newValueList; 1892 // } 1893 // } 1894 1895 myValue = preProcessedValue.get(0); 1896 1897 BaseRuntimeElementDefinition<?> def = myDef.getChildElementDefinitionByDatatype(myValue.getClass()); 1898 if (def.getChildType() == ChildTypeEnum.RESOURCE_BLOCK) { 1899 extractAndWriteExtensionsAsDirectChild( 1900 myValue, 1901 theEventWriter, 1902 def, 1903 theResDef, 1904 theResource, 1905 myChildElem, 1906 null, 1907 theEncodeContext, 1908 theContainedResource); 1909 } else { 1910 String childName = myDef.getChildNameByDatatype(myValue.getClass()); 1911 encodeChildElementToStreamWriter( 1912 theResDef, 1913 theResource, 1914 theEventWriter, 1915 myValue, 1916 def, 1917 childName, 1918 false, 1919 myParent, 1920 false, 1921 theEncodeContext); 1922 managePrimitiveExtension( 1923 myValue, 1924 theResDef, 1925 theResource, 1926 theEventWriter, 1927 def, 1928 childName, 1929 theEncodeContext, 1930 theContainedResource); 1931 } 1932 1933 theEventWriter.endObject(); 1934 } 1935 } 1936 1937 private void writeUndeclaredExtension( 1938 RuntimeResourceDefinition theResDef, 1939 IBaseResource theResource, 1940 BaseJsonLikeWriter theEventWriter, 1941 IBaseExtension<?, ?> ext, 1942 EncodeContext theEncodeContext, 1943 boolean theContainedResource) 1944 throws IOException { 1945 IBase value = ext.getValue(); 1946 final String extensionUrl = getExtensionUrl(ext.getUrl()); 1947 1948 theEventWriter.beginObject(); 1949 1950 writeCommentsPreAndPost(myUndeclaredExtension, theEventWriter); 1951 1952 String elementId = getCompositeElementId(ext); 1953 if (isNotBlank(elementId)) { 1954 JsonParser.write(theEventWriter, "id", getCompositeElementId(ext)); 1955 } 1956 1957 if (isBlank(extensionUrl)) { 1958 ParseLocation loc = new ParseLocation(theEncodeContext.toString()); 1959 getErrorHandler().missingRequiredElement(loc, "url"); 1960 } 1961 1962 JsonParser.write(theEventWriter, "url", extensionUrl); 1963 1964 boolean noValue = value == null || value.isEmpty(); 1965 if (noValue && ext.getExtension().isEmpty()) { 1966 1967 ParseLocation loc = new ParseLocation(theEncodeContext.toString()); 1968 getErrorHandler().missingRequiredElement(loc, "value"); 1969 ourLog.debug("Extension with URL[{}] has no value", extensionUrl); 1970 1971 } else { 1972 1973 if (!noValue && !ext.getExtension().isEmpty()) { 1974 ParseLocation loc = new ParseLocation(theEncodeContext.toString()); 1975 getErrorHandler().extensionContainsValueAndNestedExtensions(loc); 1976 } 1977 1978 // Write child extensions 1979 if (!ext.getExtension().isEmpty()) { 1980 1981 if (myModifier) { 1982 beginArray(theEventWriter, "modifierExtension"); 1983 } else { 1984 beginArray(theEventWriter, "extension"); 1985 } 1986 1987 for (Object next : ext.getExtension()) { 1988 writeUndeclaredExtension( 1989 theResDef, 1990 theResource, 1991 theEventWriter, 1992 (IBaseExtension<?, ?>) next, 1993 theEncodeContext, 1994 theContainedResource); 1995 } 1996 theEventWriter.endArray(); 1997 } 1998 1999 // Write value 2000 if (!noValue) { 2001 theEncodeContext.pushPath("value", false); 2002 2003 /* 2004 * Pre-process value - This is called in case the value is a reference 2005 * since we might modify the text 2006 */ 2007 value = preProcessValues( 2008 myDef, theResource, Collections.singletonList(value), myChildElem, theEncodeContext) 2009 .get(0); 2010 2011 RuntimeChildUndeclaredExtensionDefinition extDef = 2012 getContext().getRuntimeChildUndeclaredExtensionDefinition(); 2013 String childName = extDef.getChildNameByDatatype(value.getClass()); 2014 if (childName == null) { 2015 childName = "value" 2016 + WordUtils.capitalize(getContext() 2017 .getElementDefinition(value.getClass()) 2018 .getName()); 2019 } 2020 BaseRuntimeElementDefinition<?> childDef = 2021 extDef.getChildElementDefinitionByDatatype(value.getClass()); 2022 if (childDef == null) { 2023 throw new ConfigurationException( 2024 Msg.code(1844) + "Unable to encode extension, unrecognized child element type: " 2025 + value.getClass().getCanonicalName()); 2026 } 2027 encodeChildElementToStreamWriter( 2028 theResDef, 2029 theResource, 2030 theEventWriter, 2031 value, 2032 childDef, 2033 childName, 2034 false, 2035 myParent, 2036 false, 2037 theEncodeContext); 2038 managePrimitiveExtension( 2039 value, 2040 theResDef, 2041 theResource, 2042 theEventWriter, 2043 childDef, 2044 childName, 2045 theEncodeContext, 2046 theContainedResource); 2047 2048 theEncodeContext.popPath(); 2049 } 2050 } 2051 2052 // theEventWriter.name(myUndeclaredExtension.get); 2053 2054 theEventWriter.endObject(); 2055 } 2056 } 2057}