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.BaseRuntimeDeclaredChildDefinition; 024import ca.uhn.fhir.context.BaseRuntimeElementDefinition; 025import ca.uhn.fhir.context.ConfigurationException; 026import ca.uhn.fhir.context.FhirContext; 027import ca.uhn.fhir.context.RuntimeChildContainedResources; 028import ca.uhn.fhir.context.RuntimeChildDirectResource; 029import ca.uhn.fhir.context.RuntimeChildExtension; 030import ca.uhn.fhir.context.RuntimeChildNarrativeDefinition; 031import ca.uhn.fhir.context.RuntimeResourceDefinition; 032import ca.uhn.fhir.i18n.Msg; 033import ca.uhn.fhir.model.api.IResource; 034import ca.uhn.fhir.narrative.INarrativeGenerator; 035import ca.uhn.fhir.rest.api.EncodingEnum; 036import ca.uhn.fhir.util.rdf.RDFUtil; 037import org.apache.commons.lang3.StringUtils; 038import org.apache.jena.datatypes.xsd.XSDDatatype; 039import org.apache.jena.irix.IRIs; 040import org.apache.jena.rdf.model.Literal; 041import org.apache.jena.rdf.model.Model; 042import org.apache.jena.rdf.model.RDFNode; 043import org.apache.jena.rdf.model.Resource; 044import org.apache.jena.rdf.model.Statement; 045import org.apache.jena.rdf.model.StmtIterator; 046import org.apache.jena.riot.Lang; 047import org.apache.jena.vocabulary.RDF; 048import org.hl7.fhir.instance.model.api.IAnyResource; 049import org.hl7.fhir.instance.model.api.IBase; 050import org.hl7.fhir.instance.model.api.IBaseBackboneElement; 051import org.hl7.fhir.instance.model.api.IBaseDatatypeElement; 052import org.hl7.fhir.instance.model.api.IBaseElement; 053import org.hl7.fhir.instance.model.api.IBaseExtension; 054import org.hl7.fhir.instance.model.api.IBaseHasExtensions; 055import org.hl7.fhir.instance.model.api.IBaseResource; 056import org.hl7.fhir.instance.model.api.IBaseXhtml; 057import org.hl7.fhir.instance.model.api.IDomainResource; 058import org.hl7.fhir.instance.model.api.IIdType; 059import org.hl7.fhir.instance.model.api.INarrative; 060import org.hl7.fhir.instance.model.api.IPrimitiveType; 061 062import java.io.Reader; 063import java.io.Writer; 064import java.util.Arrays; 065import java.util.Comparator; 066import java.util.HashMap; 067import java.util.List; 068import java.util.Map; 069 070import static ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum.ID_DATATYPE; 071import static ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum.PRIMITIVE_DATATYPE; 072import static ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum.PRIMITIVE_XHTML; 073import static ca.uhn.fhir.context.BaseRuntimeElementDefinition.ChildTypeEnum.PRIMITIVE_XHTML_HL7ORG; 074 075/** 076 * This class is the FHIR RDF parser/encoder. Users should not interact with this class directly, but should use 077 * {@link FhirContext#newRDFParser()} to get an instance. 078 */ 079public class RDFParser extends BaseParser { 080 081 private static final String VALUE = "value"; 082 private static final String FHIR_INDEX = "index"; 083 private static final String FHIR_PREFIX = "fhir"; 084 private static final String FHIR_NS = "http://hl7.org/fhir/"; 085 private static final String RDF_PREFIX = "rdf"; 086 private static final String RDF_NS = "http://www.w3.org/1999/02/22-rdf-syntax-ns#"; 087 private static final String RDFS_PREFIX = "rdfs"; 088 private static final String RDFS_NS = "http://www.w3.org/2000/01/rdf-schema#"; 089 private static final String XSD_PREFIX = "xsd"; 090 private static final String XSD_NS = "http://www.w3.org/2001/XMLSchema#"; 091 private static final String SCT_PREFIX = "sct"; 092 private static final String SCT_NS = "http://snomed.info/id#"; 093 private static final String EXTENSION_URL = "Extension.url"; 094 private static final String ELEMENT_EXTENSION = "Element.extension"; 095 096 private static final org.slf4j.Logger logger = org.slf4j.LoggerFactory.getLogger(RDFParser.class); 097 098 public static final String NODE_ROLE = "nodeRole"; 099 private static final List<String> ignoredPredicates = 100 Arrays.asList(RDF.type.getURI(), FHIR_NS + FHIR_INDEX, FHIR_NS + NODE_ROLE); 101 public static final String TREE_ROOT = "treeRoot"; 102 public static final String RESOURCE_ID = "Resource.id"; 103 public static final String ID = "id"; 104 public static final String ELEMENT_ID = "Element.id"; 105 public static final String DOMAIN_RESOURCE_CONTAINED = "DomainResource.contained"; 106 public static final String EXTENSION = "extension"; 107 public static final String CONTAINED = "contained"; 108 public static final String MODIFIER_EXTENSION = "modifierExtension"; 109 private final Map<Class, String> classToFhirTypeMap = new HashMap<>(); 110 111 private final Lang lang; 112 113 /** 114 * Do not use this constructor, the recommended way to obtain a new instance of the RDF parser is to invoke 115 * {@link FhirContext#newRDFParser()}. 116 * 117 * @param parserErrorHandler the Parser Error Handler 118 */ 119 public RDFParser(final FhirContext context, final IParserErrorHandler parserErrorHandler, final Lang lang) { 120 super(context, parserErrorHandler); 121 this.lang = lang; 122 } 123 124 @Override 125 public EncodingEnum getEncoding() { 126 return EncodingEnum.RDF; 127 } 128 129 @Override 130 public IParser setPrettyPrint(final boolean prettyPrint) { 131 return this; 132 } 133 134 /** 135 * Writes the provided resource to the writer. This should only be called for the top-level resource being encoded. 136 * @param resource FHIR resource for writing 137 * @param writer The writer to write to -- Note: Jena prefers streams over writers 138 * @param encodeContext encoding content from parent 139 */ 140 @Override 141 protected void doEncodeResourceToWriter( 142 final IBaseResource resource, final Writer writer, final EncodeContext encodeContext) { 143 Model rdfModel = RDFUtil.initializeRDFModel(); 144 145 // Establish the namespaces and prefixes needed 146 HashMap<String, String> prefixes = new HashMap<>(); 147 prefixes.put(RDF_PREFIX, RDF_NS); 148 prefixes.put(RDFS_PREFIX, RDFS_NS); 149 prefixes.put(XSD_PREFIX, XSD_NS); 150 prefixes.put(FHIR_PREFIX, FHIR_NS); 151 prefixes.put(SCT_PREFIX, SCT_NS); 152 153 for (String key : prefixes.keySet()) { 154 rdfModel.setNsPrefix(key, prefixes.get(key)); 155 } 156 157 IIdType resourceId = processResourceID(resource, encodeContext); 158 159 encodeResourceToRDFStreamWriter(resource, rdfModel, false, resourceId, encodeContext, true, null); 160 161 RDFUtil.writeRDFModel(writer, rdfModel, lang); 162 } 163 164 /** 165 * Parses RDF content to a FHIR resource using Apache Jena 166 * @param resourceType Class of FHIR resource being deserialized 167 * @param reader Reader containing RDF (turtle) content 168 * @param <T> Type parameter denoting which resource is being parsed 169 * @return Populated FHIR resource 170 * @throws DataFormatException Exception that can be thrown from parser 171 */ 172 @Override 173 protected <T extends IBaseResource> T doParseResource(final Class<T> resourceType, final Reader reader) 174 throws DataFormatException { 175 Model model = RDFUtil.readRDFToModel(reader, this.lang); 176 return parseResource(resourceType, model); 177 } 178 179 private Resource encodeResourceToRDFStreamWriter( 180 final IBaseResource resource, 181 final Model rdfModel, 182 final boolean containedResource, 183 final IIdType resourceId, 184 final EncodeContext encodeContext, 185 final boolean rootResource, 186 Resource parentResource) { 187 188 RuntimeResourceDefinition resDef = getContext().getResourceDefinition(resource); 189 if (resDef == null) { 190 throw new ConfigurationException(Msg.code(1845) + "Unknown resource type: " + resource.getClass()); 191 } 192 193 if (!containedResource) { 194 containResourcesInReferences(resource, encodeContext); 195 } 196 197 if (!(resource instanceof IAnyResource)) { 198 throw new IllegalStateException(Msg.code(1846) + "Unsupported resource found: " 199 + resource.getClass().getName()); 200 } 201 202 // Create absolute IRI for the resource 203 String uriBase = resource.getIdElement().getBaseUrl(); 204 if (uriBase == null) { 205 uriBase = getServerBaseUrl(); 206 } 207 if (uriBase == null) { 208 uriBase = FHIR_NS; 209 } 210 if (!uriBase.endsWith("/")) { 211 uriBase = uriBase + "/"; 212 } 213 214 if (parentResource == null) { 215 if (!resource.getIdElement().toUnqualified().hasIdPart()) { 216 parentResource = rdfModel.getResource(null); 217 } else { 218 219 String resourceUri = IRIs.resolve( 220 uriBase, resource.getIdElement().toUnqualified().toString()) 221 .toString(); 222 parentResource = rdfModel.getResource(resourceUri); 223 } 224 // If the resource already exists and has statements, return that existing resource. 225 if (parentResource != null 226 && parentResource.listProperties().toList().size() > 0) { 227 return parentResource; 228 } else if (parentResource == null) { 229 return null; 230 } 231 } 232 233 parentResource.addProperty(RDF.type, rdfModel.createProperty(FHIR_NS + resDef.getName())); 234 235 // Only the top-level resource should have the nodeRole set to treeRoot 236 if (rootResource) { 237 parentResource.addProperty( 238 rdfModel.createProperty(FHIR_NS + NODE_ROLE), rdfModel.createProperty(FHIR_NS + TREE_ROOT)); 239 } 240 241 if (resourceId != null 242 && resourceId.getIdPart() != null 243 && !resourceId.getValue().startsWith("urn:")) { 244 parentResource.addProperty( 245 rdfModel.createProperty(FHIR_NS + RESOURCE_ID), 246 createFhirValueBlankNode(rdfModel, resourceId.getIdPart())); 247 } 248 249 encodeCompositeElementToStreamWriter( 250 resource, 251 resource, 252 rdfModel, 253 parentResource, 254 containedResource, 255 new CompositeChildElement(resDef, encodeContext), 256 encodeContext); 257 258 return parentResource; 259 } 260 261 /** 262 * Utility method to create a blank node with a fhir:value predicate 263 * @param rdfModel Model to create node within 264 * @param value value object - assumed to be xsd:string 265 * @return Blank node resource containing fhir:value 266 */ 267 private Resource createFhirValueBlankNode(Model rdfModel, String value) { 268 return createFhirValueBlankNode(rdfModel, value, XSDDatatype.XSDstring, null); 269 } 270 /** 271 * Utility method to create a blank node with a fhir:value predicate accepting a specific data type and index 272 * @param rdfModel Model to create node within 273 * @param value value object 274 * @param xsdDataType data type for value 275 * @param cardinalityIndex if a collection, this value is written as a fhir:index predicate 276 * @return Blank node resource containing fhir:value (and possibly fhir:index) 277 */ 278 private Resource createFhirValueBlankNode( 279 Model rdfModel, String value, XSDDatatype xsdDataType, Integer cardinalityIndex) { 280 Resource fhirValueBlankNodeResource = rdfModel.createResource() 281 .addProperty(rdfModel.createProperty(FHIR_NS + VALUE), rdfModel.createTypedLiteral(value, xsdDataType)); 282 283 if (cardinalityIndex != null && cardinalityIndex > -1) { 284 fhirValueBlankNodeResource.addProperty( 285 rdfModel.createProperty(FHIR_NS + FHIR_INDEX), 286 rdfModel.createTypedLiteral(cardinalityIndex, XSDDatatype.XSDinteger)); 287 } 288 return fhirValueBlankNodeResource; 289 } 290 291 /** 292 * Builds the predicate name based on field definition 293 * @param resource Resource being interrogated 294 * @param definition field definition 295 * @param childName childName which been massaged for different data types 296 * @return String of predicate name 297 */ 298 private String constructPredicateName( 299 IBaseResource resource, BaseRuntimeChildDefinition definition, String childName, IBase parentElement) { 300 String basePropertyName = FHIR_NS + resource.fhirType() + "." + childName; 301 String classBasedPropertyName; 302 303 if (definition instanceof BaseRuntimeDeclaredChildDefinition) { 304 BaseRuntimeDeclaredChildDefinition declaredDef = (BaseRuntimeDeclaredChildDefinition) definition; 305 Class declaringClass = declaredDef.getField().getDeclaringClass(); 306 if (declaringClass != resource.getClass()) { 307 String property = null; 308 if (IBaseBackboneElement.class.isAssignableFrom(declaringClass) 309 || IBaseDatatypeElement.class.isAssignableFrom(declaringClass)) { 310 if (classToFhirTypeMap.containsKey(declaringClass)) { 311 property = classToFhirTypeMap.get(declaringClass); 312 } else { 313 try { 314 IBase elem = (IBase) 315 declaringClass.getDeclaredConstructor().newInstance(); 316 property = elem.fhirType(); 317 classToFhirTypeMap.put(declaringClass, property); 318 } catch (Exception ex) { 319 logger.debug("Error instantiating an " + declaringClass.getSimpleName() 320 + " to retrieve its FhirType"); 321 } 322 } 323 } else { 324 if ("MetadataResource".equals(declaringClass.getSimpleName())) { 325 property = resource.getClass().getSimpleName(); 326 } else { 327 property = declaredDef.getField().getDeclaringClass().getSimpleName(); 328 } 329 } 330 classBasedPropertyName = FHIR_NS + property + "." + childName; 331 return classBasedPropertyName; 332 } 333 } 334 return basePropertyName; 335 } 336 337 private Model encodeChildElementToStreamWriter( 338 final IBaseResource resource, 339 IBase parentElement, 340 Model rdfModel, 341 Resource rdfResource, 342 final BaseRuntimeChildDefinition childDefinition, 343 final IBase element, 344 final String childName, 345 final BaseRuntimeElementDefinition<?> childDef, 346 final boolean includedResource, 347 final CompositeChildElement parent, 348 final EncodeContext theEncodeContext, 349 final Integer cardinalityIndex) { 350 351 String childGenericName = childDefinition.getElementName(); 352 353 theEncodeContext.pushPath(childGenericName, false); 354 try { 355 356 if (element == null || element.isEmpty()) { 357 if (!isChildContained(childDef, includedResource, theEncodeContext)) { 358 return rdfModel; 359 } 360 } 361 362 switch (childDef.getChildType()) { 363 case ID_DATATYPE: { 364 IIdType value = (IIdType) element; 365 assert value != null; 366 String encodedValue = ID.equals(childName) ? value.getIdPart() : value.getValue(); 367 if (StringUtils.isNotBlank(encodedValue) || !hasNoExtensions(value)) { 368 if (StringUtils.isNotBlank(encodedValue)) { 369 370 String propertyName = 371 constructPredicateName(resource, childDefinition, childName, parentElement); 372 if (element != null) { 373 XSDDatatype dataType = getXSDDataTypeForFhirType(element.fhirType(), encodedValue); 374 rdfResource.addProperty( 375 rdfModel.createProperty(propertyName), 376 this.createFhirValueBlankNode( 377 rdfModel, encodedValue, dataType, cardinalityIndex)); 378 } 379 } 380 } 381 break; 382 } 383 case PRIMITIVE_DATATYPE: { 384 IPrimitiveType<?> pd = (IPrimitiveType<?>) element; 385 assert pd != null; 386 String value = pd.getValueAsString(); 387 if (value != null || !hasNoExtensions(pd)) { 388 if (value != null) { 389 String propertyName = 390 constructPredicateName(resource, childDefinition, childName, parentElement); 391 XSDDatatype dataType = getXSDDataTypeForFhirType(pd.fhirType(), value); 392 Resource valueResource = 393 this.createFhirValueBlankNode(rdfModel, value, dataType, cardinalityIndex); 394 if (!hasNoExtensions(pd)) { 395 IBaseHasExtensions hasExtension = (IBaseHasExtensions) pd; 396 if (hasExtension.getExtension() != null 397 && hasExtension.getExtension().size() > 0) { 398 int i = 0; 399 for (IBaseExtension extension : hasExtension.getExtension()) { 400 RuntimeResourceDefinition resDef = 401 getContext().getResourceDefinition(resource); 402 Resource extensionResource = rdfModel.createResource(); 403 extensionResource.addProperty( 404 rdfModel.createProperty(FHIR_NS + FHIR_INDEX), 405 rdfModel.createTypedLiteral(i, XSDDatatype.XSDinteger)); 406 valueResource.addProperty( 407 rdfModel.createProperty(FHIR_NS + ELEMENT_EXTENSION), 408 extensionResource); 409 encodeCompositeElementToStreamWriter( 410 resource, 411 extension, 412 rdfModel, 413 extensionResource, 414 false, 415 new CompositeChildElement(resDef, theEncodeContext), 416 theEncodeContext); 417 } 418 } 419 } 420 421 rdfResource.addProperty(rdfModel.createProperty(propertyName), valueResource); 422 } 423 } 424 break; 425 } 426 case RESOURCE_BLOCK: 427 case COMPOSITE_DATATYPE: { 428 String idString = null; 429 String idPredicate = null; 430 if (element instanceof IBaseResource) { 431 idPredicate = FHIR_NS + RESOURCE_ID; 432 IIdType resourceId = processResourceID((IBaseResource) element, theEncodeContext); 433 if (resourceId != null) { 434 idString = resourceId.getIdPart(); 435 } 436 } else if (element instanceof IBaseElement) { 437 idPredicate = FHIR_NS + ELEMENT_ID; 438 if (((IBaseElement) element).getId() != null) { 439 idString = ((IBaseElement) element).getId(); 440 } 441 } 442 if (idString != null) { 443 rdfResource.addProperty( 444 rdfModel.createProperty(idPredicate), createFhirValueBlankNode(rdfModel, idString)); 445 } 446 rdfModel = encodeCompositeElementToStreamWriter( 447 resource, element, rdfModel, rdfResource, includedResource, parent, theEncodeContext); 448 break; 449 } 450 case CONTAINED_RESOURCE_LIST: 451 case CONTAINED_RESOURCES: { 452 if (element != null) { 453 IIdType resourceId = ((IBaseResource) element).getIdElement(); 454 Resource containedResource = rdfModel.createResource(); 455 rdfResource.addProperty( 456 rdfModel.createProperty(FHIR_NS + DOMAIN_RESOURCE_CONTAINED), containedResource); 457 if (cardinalityIndex != null) { 458 containedResource.addProperty( 459 rdfModel.createProperty(FHIR_NS + FHIR_INDEX), 460 cardinalityIndex.toString(), 461 XSDDatatype.XSDinteger); 462 } 463 encodeResourceToRDFStreamWriter( 464 (IBaseResource) element, 465 rdfModel, 466 true, 467 super.fixContainedResourceId(resourceId.getValue()), 468 theEncodeContext, 469 false, 470 containedResource); 471 } 472 break; 473 } 474 case RESOURCE: { 475 IBaseResource baseResource = (IBaseResource) element; 476 String resourceName = getContext().getResourceType(baseResource); 477 if (!super.shouldEncodeResource(resourceName, theEncodeContext)) { 478 break; 479 } 480 theEncodeContext.pushPath(resourceName, true); 481 IIdType resourceId = processResourceID(resource, theEncodeContext); 482 encodeResourceToRDFStreamWriter( 483 resource, rdfModel, false, resourceId, theEncodeContext, false, null); 484 theEncodeContext.popPath(); 485 break; 486 } 487 case PRIMITIVE_XHTML: 488 case PRIMITIVE_XHTML_HL7ORG: { 489 IBaseXhtml xHtmlNode = (IBaseXhtml) element; 490 if (xHtmlNode != null) { 491 String value = xHtmlNode.getValueAsString(); 492 String propertyName = 493 constructPredicateName(resource, childDefinition, childName, parentElement); 494 rdfResource.addProperty(rdfModel.createProperty(propertyName), value); 495 } 496 break; 497 } 498 case EXTENSION_DECLARED: 499 case UNDECL_EXT: 500 default: { 501 throw new IllegalStateException( 502 Msg.code(1847) + "Unexpected node - should not happen: " + childDef.getName()); 503 } 504 } 505 } finally { 506 theEncodeContext.popPath(); 507 } 508 509 return rdfModel; 510 } 511 512 /** 513 * Maps hapi internal fhirType attribute to XSDDatatype enumeration 514 * @param fhirType hapi field type 515 * @return XSDDatatype value 516 */ 517 private XSDDatatype getXSDDataTypeForFhirType(String fhirType, String value) { 518 switch (fhirType) { 519 case "boolean": 520 return XSDDatatype.XSDboolean; 521 case "uri": 522 return XSDDatatype.XSDanyURI; 523 case "decimal": 524 return XSDDatatype.XSDdecimal; 525 case "date": 526 return XSDDatatype.XSDdate; 527 case "dateTime": 528 case "instant": 529 switch (value.length()) { // assumes valid lexical value 530 case 4: 531 return XSDDatatype.XSDgYear; 532 case 7: 533 return XSDDatatype.XSDgYearMonth; 534 case 10: 535 return XSDDatatype.XSDdate; 536 default: 537 return XSDDatatype.XSDdateTime; 538 } 539 case "code": 540 case "string": 541 default: 542 return XSDDatatype.XSDstring; 543 } 544 } 545 546 private IIdType processResourceID(final IBaseResource resource, final EncodeContext encodeContext) { 547 IIdType resourceId = null; 548 549 if (StringUtils.isNotBlank(resource.getIdElement().getIdPart())) { 550 resourceId = resource.getIdElement(); 551 if (resource.getIdElement().getValue().startsWith("urn:")) { 552 resourceId = null; 553 } 554 } 555 556 if (!super.shouldEncodeResourceId(resource, encodeContext)) { 557 resourceId = null; 558 } else if (encodeContext.getResourcePath().size() == 1 && super.getEncodeForceResourceId() != null) { 559 resourceId = super.getEncodeForceResourceId(); 560 } 561 562 return resourceId; 563 } 564 565 private Model encodeExtension( 566 final IBaseResource resource, 567 Model rdfModel, 568 Resource rdfResource, 569 final boolean containedResource, 570 final CompositeChildElement nextChildElem, 571 final BaseRuntimeChildDefinition nextChild, 572 final IBase nextValue, 573 final String childName, 574 final BaseRuntimeElementDefinition<?> childDef, 575 final EncodeContext encodeContext, 576 Integer cardinalityIndex) { 577 BaseRuntimeDeclaredChildDefinition extDef = (BaseRuntimeDeclaredChildDefinition) nextChild; 578 579 Resource childResource = rdfModel.createResource(); 580 String extensionPredicateName = constructPredicateName(resource, extDef, extDef.getElementName(), null); 581 rdfResource.addProperty(rdfModel.createProperty(extensionPredicateName), childResource); 582 if (cardinalityIndex != null && cardinalityIndex > -1) { 583 childResource.addProperty( 584 rdfModel.createProperty(FHIR_NS + FHIR_INDEX), cardinalityIndex.toString(), XSDDatatype.XSDinteger); 585 } 586 587 rdfModel = encodeChildElementToStreamWriter( 588 resource, 589 null, 590 rdfModel, 591 childResource, 592 nextChild, 593 nextValue, 594 childName, 595 childDef, 596 containedResource, 597 nextChildElem, 598 encodeContext, 599 cardinalityIndex); 600 601 return rdfModel; 602 } 603 604 private Model encodeCompositeElementToStreamWriter( 605 final IBaseResource resource, 606 final IBase element, 607 Model rdfModel, 608 Resource rdfResource, 609 final boolean containedResource, 610 final CompositeChildElement parent, 611 final EncodeContext encodeContext) { 612 613 for (CompositeChildElement nextChildElem : 614 super.compositeChildIterator(element, containedResource, parent, encodeContext)) { 615 616 BaseRuntimeChildDefinition nextChild = nextChildElem.getDef(); 617 618 if (nextChild instanceof RuntimeChildNarrativeDefinition) { 619 INarrativeGenerator gen = getContext().getNarrativeGenerator(); 620 if (gen != null) { 621 INarrative narrative; 622 if (resource instanceof IResource) { 623 narrative = ((IResource) resource).getText(); 624 } else if (resource instanceof IDomainResource) { 625 narrative = ((IDomainResource) resource).getText(); 626 } else { 627 narrative = null; 628 } 629 assert narrative != null; 630 if (narrative.isEmpty()) { 631 gen.populateResourceNarrative(getContext(), resource); 632 } else { 633 RuntimeChildNarrativeDefinition child = (RuntimeChildNarrativeDefinition) nextChild; 634 635 // This is where we populate the parent of the narrative 636 Resource childResource = rdfModel.createResource(); 637 638 String propertyName = constructPredicateName(resource, child, child.getElementName(), element); 639 rdfResource.addProperty(rdfModel.createProperty(propertyName), childResource); 640 641 String childName = nextChild.getChildNameByDatatype(child.getDatatype()); 642 BaseRuntimeElementDefinition<?> type = child.getChildByName(childName); 643 rdfModel = encodeChildElementToStreamWriter( 644 resource, 645 element, 646 rdfModel, 647 childResource, 648 nextChild, 649 narrative, 650 childName, 651 type, 652 containedResource, 653 nextChildElem, 654 encodeContext, 655 null); 656 continue; 657 } 658 } 659 } 660 661 if (nextChild instanceof RuntimeChildDirectResource) { 662 663 List<? extends IBase> values = nextChild.getAccessor().getValues(element); 664 if (values == null || values.isEmpty()) { 665 continue; 666 } 667 668 IBaseResource directChildResource = (IBaseResource) values.get(0); 669 // If it is a direct resource, we need to create a new subject for it. 670 Resource childResource = encodeResourceToRDFStreamWriter( 671 directChildResource, 672 rdfModel, 673 false, 674 directChildResource.getIdElement(), 675 encodeContext, 676 false, 677 null); 678 String propertyName = constructPredicateName(resource, nextChild, nextChild.getElementName(), element); 679 rdfResource.addProperty(rdfModel.createProperty(propertyName), childResource); 680 681 continue; 682 } 683 684 if (nextChild instanceof RuntimeChildContainedResources) { 685 List<? extends IBase> values = nextChild.getAccessor().getValues(element); 686 int i = 0; 687 for (IBase containedResourceEntity : values) { 688 rdfModel = encodeChildElementToStreamWriter( 689 resource, 690 element, 691 rdfModel, 692 rdfResource, 693 nextChild, 694 containedResourceEntity, 695 nextChild.getChildNameByDatatype(null), 696 nextChild.getChildElementDefinitionByDatatype(null), 697 containedResource, 698 nextChildElem, 699 encodeContext, 700 i); 701 i++; 702 } 703 } else { 704 705 List<? extends IBase> values = nextChild.getAccessor().getValues(element); 706 values = super.preProcessValues(nextChild, resource, values, nextChildElem, encodeContext); 707 708 if (values == null || values.isEmpty()) { 709 continue; 710 } 711 712 Integer cardinalityIndex = null; 713 int indexCounter = 0; 714 715 for (IBase nextValue : values) { 716 if (nextChild.getMax() != 1) { 717 cardinalityIndex = indexCounter; 718 indexCounter++; 719 } 720 if ((nextValue == null || nextValue.isEmpty())) { 721 continue; 722 } 723 724 ChildNameAndDef childNameAndDef = super.getChildNameAndDef(nextChild, nextValue); 725 if (childNameAndDef == null) { 726 continue; 727 } 728 729 String childName = childNameAndDef.getChildName(); 730 BaseRuntimeElementDefinition<?> childDef = childNameAndDef.getChildDef(); 731 String extensionUrl = getExtensionUrl(nextChild.getExtensionUrl()); 732 733 if (extensionUrl != null && !childName.equals(EXTENSION)) { 734 rdfModel = encodeExtension( 735 resource, 736 rdfModel, 737 rdfResource, 738 containedResource, 739 nextChildElem, 740 nextChild, 741 nextValue, 742 childName, 743 childDef, 744 encodeContext, 745 cardinalityIndex); 746 } else if (nextChild instanceof RuntimeChildExtension) { 747 IBaseExtension<?, ?> extension = (IBaseExtension<?, ?>) nextValue; 748 if ((extension.getValue() == null 749 || extension.getValue().isEmpty())) { 750 if (extension.getExtension().isEmpty()) { 751 continue; 752 } 753 } 754 rdfModel = encodeExtension( 755 resource, 756 rdfModel, 757 rdfResource, 758 containedResource, 759 nextChildElem, 760 nextChild, 761 nextValue, 762 childName, 763 childDef, 764 encodeContext, 765 cardinalityIndex); 766 } else if (!(nextChild instanceof RuntimeChildNarrativeDefinition) || !containedResource) { 767 768 // If the child is not a value type, create a child object (blank node) for subordinate 769 // predicates to be attached to 770 if (childDef.getChildType() != PRIMITIVE_DATATYPE 771 && childDef.getChildType() != PRIMITIVE_XHTML_HL7ORG 772 && childDef.getChildType() != PRIMITIVE_XHTML 773 && childDef.getChildType() != ID_DATATYPE) { 774 Resource childResource = rdfModel.createResource(); 775 776 String propertyName = constructPredicateName(resource, nextChild, childName, nextValue); 777 rdfResource.addProperty(rdfModel.createProperty(propertyName), childResource); 778 if (cardinalityIndex != null && cardinalityIndex > -1) { 779 childResource.addProperty( 780 rdfModel.createProperty(FHIR_NS + FHIR_INDEX), 781 cardinalityIndex.toString(), 782 XSDDatatype.XSDinteger); 783 } 784 rdfModel = encodeChildElementToStreamWriter( 785 resource, 786 element, 787 rdfModel, 788 childResource, 789 nextChild, 790 nextValue, 791 childName, 792 childDef, 793 containedResource, 794 nextChildElem, 795 encodeContext, 796 cardinalityIndex); 797 } else { 798 rdfModel = encodeChildElementToStreamWriter( 799 resource, 800 element, 801 rdfModel, 802 rdfResource, 803 nextChild, 804 nextValue, 805 childName, 806 childDef, 807 containedResource, 808 nextChildElem, 809 encodeContext, 810 cardinalityIndex); 811 } 812 } 813 } 814 } 815 } 816 return rdfModel; 817 } 818 819 private <T extends IBaseResource> T parseResource(Class<T> resourceType, Model rdfModel) { 820 // jsonMode of true is passed in so that the xhtml parser state behaves as expected 821 // Push PreResourceState 822 ParserState<T> parserState = 823 ParserState.getPreResourceInstance(this, resourceType, getContext(), true, getErrorHandler()); 824 return parseRootResource(rdfModel, parserState, resourceType); 825 } 826 827 private <T> T parseRootResource(Model rdfModel, ParserState<T> parserState, Class<T> resourceType) { 828 logger.trace("Entering parseRootResource with state: {}", parserState); 829 830 StmtIterator rootStatementIterator = rdfModel.listStatements( 831 null, rdfModel.getProperty(FHIR_NS + NODE_ROLE), rdfModel.getProperty(FHIR_NS + TREE_ROOT)); 832 833 Resource rootResource; 834 String fhirResourceType, fhirTypeString; 835 while (rootStatementIterator.hasNext()) { 836 Statement rootStatement = rootStatementIterator.next(); 837 rootResource = rootStatement.getSubject(); 838 839 // If a resourceType is not provided via the server framework, discern it based on the rdf:type Arc 840 if (resourceType == null) { 841 Statement resourceTypeStatement = rootResource.getProperty(RDF.type); 842 fhirTypeString = resourceTypeStatement.getObject().toString(); 843 if (fhirTypeString.startsWith(FHIR_NS)) { 844 fhirTypeString = fhirTypeString.replace(FHIR_NS, ""); 845 } 846 } else { 847 fhirTypeString = resourceType.getSimpleName(); 848 } 849 850 RuntimeResourceDefinition definition = getContext().getResourceDefinition(fhirTypeString); 851 fhirResourceType = definition.getName(); 852 853 parseResource(parserState, fhirResourceType, rootResource); 854 855 // Pop PreResourceState 856 parserState.endingElement(); 857 } 858 return parserState.getObject(); 859 } 860 861 private <T> void parseResource(ParserState<T> parserState, String resourceType, RDFNode rootNode) { 862 // Push top-level entity 863 parserState.enteringNewElement(FHIR_NS, resourceType); 864 865 if (rootNode instanceof Resource) { 866 Resource rootResource = rootNode.asResource(); 867 List<Statement> statements = rootResource.listProperties().toList(); 868 statements.sort(new FhirIndexStatementComparator()); 869 for (Statement statement : statements) { 870 String predicateAttributeName = extractAttributeNameFromPredicate(statement); 871 if (predicateAttributeName != null) { 872 if (predicateAttributeName.equals(MODIFIER_EXTENSION)) { 873 processExtension(parserState, statement.getObject(), true); 874 } else if (predicateAttributeName.equals(EXTENSION)) { 875 processExtension(parserState, statement.getObject(), false); 876 } else { 877 processStatementObject(parserState, predicateAttributeName, statement.getObject()); 878 } 879 } 880 } 881 } else if (rootNode instanceof Literal) { 882 parserState.attributeValue(VALUE, rootNode.asLiteral().getString()); 883 } 884 885 // Pop top-level entity 886 parserState.endingElement(); 887 } 888 889 private String extractAttributeNameFromPredicate(Statement statement) { 890 String predicateUri = statement.getPredicate().getURI(); 891 892 // If the predicateURI is one we're ignoring, return null 893 // This minimizes 'Unknown Element' warnings in the parsing process 894 if (ignoredPredicates.contains(predicateUri)) { 895 return null; 896 } 897 898 String predicateObjectAttribute = predicateUri.substring(predicateUri.lastIndexOf("/") + 1); 899 String predicateAttributeName; 900 if (predicateObjectAttribute.contains(".")) { 901 predicateAttributeName = predicateObjectAttribute.substring(predicateObjectAttribute.lastIndexOf(".") + 1); 902 } else { 903 predicateAttributeName = predicateObjectAttribute; 904 } 905 return predicateAttributeName; 906 } 907 908 private <T> void processStatementObject( 909 ParserState<T> parserState, String predicateAttributeName, RDFNode statementObject) { 910 logger.trace( 911 "Entering processStatementObject with state: {}, for attribute {}", 912 parserState, 913 predicateAttributeName); 914 // Push attribute element 915 parserState.enteringNewElement(FHIR_NS, predicateAttributeName); 916 917 if (statementObject != null) { 918 if (statementObject.isLiteral()) { 919 // If the object is a literal, apply the value directly 920 parserState.attributeValue(VALUE, statementObject.asLiteral().getLexicalForm()); 921 } else if (statementObject.isAnon()) { 922 // If the object is a blank node, 923 Resource resourceObject = statementObject.asResource(); 924 925 boolean containedResource = false; 926 if (predicateAttributeName.equals(CONTAINED)) { 927 containedResource = true; 928 parserState.enteringNewElement( 929 FHIR_NS, 930 resourceObject 931 .getProperty(resourceObject.getModel().createProperty(RDF.type.getURI())) 932 .getObject() 933 .toString() 934 .replace(FHIR_NS, "")); 935 } 936 937 List<Statement> objectStatements = 938 resourceObject.listProperties().toList(); 939 objectStatements.sort(new FhirIndexStatementComparator()); 940 for (Statement objectProperty : objectStatements) { 941 if (objectProperty.getPredicate().hasURI(FHIR_NS + VALUE)) { 942 predicateAttributeName = VALUE; 943 parserState.attributeValue( 944 predicateAttributeName, 945 objectProperty.getObject().asLiteral().getLexicalForm()); 946 } else { 947 // Otherwise, process it as a net-new node 948 predicateAttributeName = extractAttributeNameFromPredicate(objectProperty); 949 if (predicateAttributeName != null) { 950 if (predicateAttributeName.equals(EXTENSION)) { 951 processExtension(parserState, objectProperty.getObject(), false); 952 } else if (predicateAttributeName.equals(MODIFIER_EXTENSION)) { 953 processExtension(parserState, objectProperty.getObject(), true); 954 } else { 955 processStatementObject(parserState, predicateAttributeName, objectProperty.getObject()); 956 } 957 } 958 } 959 } 960 961 if (containedResource) { 962 // Leave the contained resource element we created 963 parserState.endingElement(); 964 } 965 } else if (statementObject.isResource()) { 966 Resource innerResource = statementObject.asResource(); 967 Statement resourceTypeStatement = innerResource.getProperty(RDF.type); 968 String fhirTypeString = resourceTypeStatement.getObject().toString(); 969 if (fhirTypeString.startsWith(FHIR_NS)) { 970 fhirTypeString = fhirTypeString.replace(FHIR_NS, ""); 971 } 972 parseResource(parserState, fhirTypeString, innerResource); 973 } 974 } 975 976 // Pop attribute element 977 parserState.endingElement(); 978 } 979 980 private <T> void processExtension(ParserState<T> parserState, RDFNode statementObject, boolean isModifier) { 981 logger.trace("Entering processExtension with state: {}", parserState); 982 Resource resource = statementObject.asResource(); 983 Statement urlProperty = resource.getProperty(resource.getModel().createProperty(FHIR_NS + EXTENSION_URL)); 984 Resource urlPropertyResource = urlProperty.getObject().asResource(); 985 String extensionUrl = urlPropertyResource 986 .getProperty(resource.getModel().createProperty(FHIR_NS + VALUE)) 987 .getObject() 988 .asLiteral() 989 .getString(); 990 991 List<Statement> extensionStatements = resource.listProperties().toList(); 992 String extensionValueType = null; 993 RDFNode extensionValueResource = null; 994 for (Statement statement : extensionStatements) { 995 String propertyUri = statement.getPredicate().getURI(); 996 if (propertyUri.contains("Extension.value")) { 997 extensionValueType = propertyUri.replace(FHIR_NS + "Extension.", ""); 998 BaseRuntimeElementDefinition<?> target = getContext() 999 .getRuntimeChildUndeclaredExtensionDefinition() 1000 .getChildByName(extensionValueType); 1001 if (target.getChildType().equals(ID_DATATYPE) 1002 || target.getChildType().equals(PRIMITIVE_DATATYPE)) { 1003 extensionValueResource = statement 1004 .getObject() 1005 .asResource() 1006 .getProperty(resource.getModel().createProperty(FHIR_NS + VALUE)) 1007 .getObject() 1008 .asLiteral(); 1009 } else { 1010 extensionValueResource = statement.getObject().asResource(); 1011 } 1012 break; 1013 } 1014 } 1015 1016 parserState.enteringNewElementExtension(null, extensionUrl, isModifier, null); 1017 // Some extensions don't have their own values - they then have more extensions inside of them 1018 if (extensionValueType != null) { 1019 parseResource(parserState, extensionValueType, extensionValueResource); 1020 } 1021 1022 for (Statement statement : extensionStatements) { 1023 String propertyUri = statement.getPredicate().getURI(); 1024 if (propertyUri.equals(FHIR_NS + ELEMENT_EXTENSION)) { 1025 processExtension(parserState, statement.getObject(), false); 1026 } 1027 } 1028 1029 parserState.endingElement(); 1030 } 1031 1032 static class FhirIndexStatementComparator implements Comparator<Statement> { 1033 1034 @Override 1035 public int compare(Statement arg0, Statement arg1) { 1036 int result = 1037 arg0.getPredicate().getURI().compareTo(arg1.getPredicate().getURI()); 1038 if (result == 0) { 1039 if (arg0.getObject().isResource() && arg1.getObject().isResource()) { 1040 Resource resource0 = arg0.getObject().asResource(); 1041 Resource resource1 = arg1.getObject().asResource(); 1042 1043 result = Integer.compare(getFhirIndex(resource0), getFhirIndex(resource1)); 1044 } 1045 } 1046 return result; 1047 } 1048 1049 private int getFhirIndex(Resource resource) { 1050 if (resource.hasProperty(resource.getModel().createProperty(FHIR_NS + FHIR_INDEX))) { 1051 return resource.getProperty(resource.getModel().createProperty(FHIR_NS + FHIR_INDEX)) 1052 .getInt(); 1053 } 1054 return -1; 1055 } 1056 } 1057}