From e76eb78aa9e815e7dd23ed916ebd63c3fc2ec22d Mon Sep 17 00:00:00 2001 From: Boris Stanojevic Date: Tue, 12 Dec 2023 16:12:35 +0100 Subject: [PATCH] implement the #437 SimplifiedPublish iti-105 transaction --- .../fhir/audit/codes/FhirEventTypeCode.java | 1 + .../ihe/fhir/iti105/Iti105AuditDataset.java | 49 + .../ihe/fhir/iti105/Iti105AuditStrategy.java | 59 + .../iti105/Iti105ClientAuditStrategy.java | 51 + ...i105DocumentReferenceResourceProvider.java | 53 + .../ihe/fhir/iti105/Iti105RequestFactory.java | 43 + .../iti105/Iti105ServerAuditStrategy.java | 49 + .../Iti105TransactionConfiguration.java | 41 + .../ihe/fhir/iti105/Iti105Validator.java | 45 + .../ipf/commons/ihe/fhir/mhd/MHD.java | 12 + ...HD_SimplifiedPublish_DocumentReference.xml | 2493 +++++++++++++++++ .../ihe/fhir/iti105/Iti105ValidatorTest.java | 102 + .../ihe/fhir/iti105/Iti105Component.java | 49 + .../camel/ihe/fhir/iti105/Iti105Endpoint.java | 37 + .../org/apache/camel/component/mhd-iti105 | 12 + .../fhir/iti105/Iti105TestRouteBuilder.java | 81 + .../ihe/fhir/iti105/TestIti105Success.java | 199 ++ .../r4/mhd/src/test/resources/iti-105.xml | 30 + 18 files changed, 3406 insertions(+) create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105AuditDataset.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105AuditStrategy.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ClientAuditStrategy.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105DocumentReferenceResourceProvider.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105RequestFactory.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ServerAuditStrategy.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105TransactionConfiguration.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105Validator.java create mode 100644 commons/ihe/fhir/r4/mhd/src/main/resources/META-INF/profiles/IHE_MHD_SimplifiedPublish_DocumentReference.xml create mode 100644 commons/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ValidatorTest.java create mode 100644 platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Component.java create mode 100644 platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Endpoint.java create mode 100644 platform-camel/ihe/fhir/r4/mhd/src/main/resources/META-INF/services/org/apache/camel/component/mhd-iti105 create mode 100644 platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105TestRouteBuilder.java create mode 100644 platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/TestIti105Success.java create mode 100644 platform-camel/ihe/fhir/r4/mhd/src/test/resources/iti-105.xml diff --git a/commons/ihe/fhir/core/src/main/java/org/openehealth/ipf/commons/ihe/fhir/audit/codes/FhirEventTypeCode.java b/commons/ihe/fhir/core/src/main/java/org/openehealth/ipf/commons/ihe/fhir/audit/codes/FhirEventTypeCode.java index 62cc758c82..f831f25d6c 100644 --- a/commons/ihe/fhir/core/src/main/java/org/openehealth/ipf/commons/ihe/fhir/audit/codes/FhirEventTypeCode.java +++ b/commons/ihe/fhir/core/src/main/java/org/openehealth/ipf/commons/ihe/fhir/audit/codes/FhirEventTypeCode.java @@ -36,6 +36,7 @@ public enum FhirEventTypeCode implements EventType, EnumeratedCodedValue { + + public Iti105AuditStrategy(boolean serverSide) { + super(serverSide); + } + + @Override + public Iti105AuditDataset createAuditDataset() { + return new Iti105AuditDataset(isServerSide()); + } + + @Override + public Iti105AuditDataset enrichAuditDatasetFromRequest(Iti105AuditDataset auditDataset, Object request, + Map parameters) { + var dataset = super.enrichAuditDatasetFromRequest(auditDataset, request, parameters); + if (request instanceof DocumentReference) { + dataset.enrichDatasetFromDocumentReference((DocumentReference) request); + } + return dataset; + } + + @Override + public boolean enrichAuditDatasetFromResponse(Iti105AuditDataset auditDataset, Object response, AuditContext auditContext) { + var methodOutcome = (MethodOutcome) response; + if (methodOutcome.getResource() != null && methodOutcome.getResource().getIdElement() != null) { + auditDataset.setDocumentReferenceId(methodOutcome.getResource().getIdElement().getValue()); + } + return super.enrichAuditDatasetFromResponse(auditDataset, response, auditContext); + } + +} diff --git a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ClientAuditStrategy.java b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ClientAuditStrategy.java new file mode 100644 index 0000000000..f176fb035d --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ClientAuditStrategy.java @@ -0,0 +1,51 @@ +/* + * Copyright 2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.commons.ihe.fhir.iti105; + +import org.openehealth.ipf.commons.audit.AuditContext; +import org.openehealth.ipf.commons.audit.codes.ParticipantObjectIdTypeCode; +import org.openehealth.ipf.commons.audit.codes.ParticipantObjectTypeCodeRole; +import org.openehealth.ipf.commons.audit.model.AuditMessage; +import org.openehealth.ipf.commons.ihe.core.atna.event.PHIExportBuilder; +import org.openehealth.ipf.commons.ihe.fhir.audit.codes.FhirEventTypeCode; + +import java.util.Collections; + +/** + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105ClientAuditStrategy extends Iti105AuditStrategy { + + public Iti105ClientAuditStrategy() { + super(false); + } + + @Override + public AuditMessage[] makeAuditMessage(AuditContext auditContext, Iti105AuditDataset auditDataset) { + return new PHIExportBuilder<>(auditContext, auditDataset, FhirEventTypeCode.SimplifiedPublish) + .setPatient(auditDataset.getPatientId()) + .addExportedEntity( + auditDataset.getDocumentReferenceId() != null ? + auditDataset.getDocumentReferenceId() : + auditContext.getAuditValueIfMissing(), + ParticipantObjectIdTypeCode.XdsMetadata, + ParticipantObjectTypeCodeRole.Job, + Collections.emptyList()) + .getMessages(); + } +} diff --git a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105DocumentReferenceResourceProvider.java b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105DocumentReferenceResourceProvider.java new file mode 100644 index 0000000000..5b5e354ead --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105DocumentReferenceResourceProvider.java @@ -0,0 +1,53 @@ +/* + * Copyright 2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.commons.ihe.fhir.iti105; + +import ca.uhn.fhir.rest.annotation.Create; +import ca.uhn.fhir.rest.annotation.ResourceParam; +import ca.uhn.fhir.rest.api.MethodOutcome; +import ca.uhn.fhir.rest.api.server.RequestDetails; +import org.hl7.fhir.instance.model.api.IBaseResource; +import org.hl7.fhir.r4.model.DocumentReference; +import org.openehealth.ipf.commons.ihe.fhir.AbstractResourceProvider; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; + +/** + * Resource Provider for MHD (ITI-105) based on creating a DocumentReference resource + * + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105DocumentReferenceResourceProvider extends AbstractResourceProvider { + + @SuppressWarnings("unused") + @Create + public MethodOutcome createDocumentReference( + @ResourceParam DocumentReference documentReference, + RequestDetails requestDetails, + HttpServletRequest httpServletRequest, + HttpServletResponse httpServletResponse) { + + return requestAction(documentReference, null, httpServletRequest, httpServletResponse, requestDetails); + } + + @Override + public Class getResourceType() { + return DocumentReference.class; + } +} diff --git a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105RequestFactory.java b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105RequestFactory.java new file mode 100644 index 0000000000..5b6685ca66 --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105RequestFactory.java @@ -0,0 +1,43 @@ +/* + * Copyright 2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.commons.ihe.fhir.iti105; + +import ca.uhn.fhir.rest.api.MethodOutcome; +import ca.uhn.fhir.rest.client.api.IGenericClient; +import ca.uhn.fhir.rest.gclient.IClientExecutable; +import org.hl7.fhir.r4.model.DocumentReference; +import org.openehealth.ipf.commons.ihe.fhir.ClientRequestFactory; + +import java.util.Map; + +/** + * A {@link ClientRequestFactory} for ITI-105 requests + * + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105RequestFactory implements ClientRequestFactory> { + + @Override + public IClientExecutable getClientExecutable( + IGenericClient client, + Object requestData, + Map parameters) { + return client.create().resource((DocumentReference) requestData); + } + +} diff --git a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ServerAuditStrategy.java b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ServerAuditStrategy.java new file mode 100644 index 0000000000..117c39dacd --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ServerAuditStrategy.java @@ -0,0 +1,49 @@ +/* + * Copyright 2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.commons.ihe.fhir.iti105; + +import org.openehealth.ipf.commons.audit.AuditContext; +import org.openehealth.ipf.commons.audit.codes.ParticipantObjectIdTypeCode; +import org.openehealth.ipf.commons.audit.codes.ParticipantObjectTypeCodeRole; +import org.openehealth.ipf.commons.audit.model.AuditMessage; +import org.openehealth.ipf.commons.ihe.core.atna.event.PHIImportBuilder; +import org.openehealth.ipf.commons.ihe.fhir.audit.codes.FhirEventTypeCode; + +import java.util.Collections; + +/** + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105ServerAuditStrategy extends Iti105AuditStrategy { + + public Iti105ServerAuditStrategy() { + super(true); + } + + @Override + public AuditMessage[] makeAuditMessage(AuditContext auditContext, Iti105AuditDataset auditDataset) { + return new PHIImportBuilder<>(auditContext, auditDataset, FhirEventTypeCode.SimplifiedPublish) + .setPatient(auditDataset.getPatientId()) + .addImportedEntity( + auditDataset.getDocumentReferenceId(), + ParticipantObjectIdTypeCode.XdsMetadata, + ParticipantObjectTypeCodeRole.Job, + Collections.emptyList()) + .getMessages(); + } +} diff --git a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105TransactionConfiguration.java b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105TransactionConfiguration.java new file mode 100644 index 0000000000..b203b48621 --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105TransactionConfiguration.java @@ -0,0 +1,41 @@ +/* + * Copyright 2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openehealth.ipf.commons.ihe.fhir.iti105; + +import ca.uhn.fhir.context.FhirVersionEnum; +import org.openehealth.ipf.commons.ihe.fhir.FhirTransactionConfiguration; + +/** + * Standard Configuration for Iti105Component. + * + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105TransactionConfiguration extends FhirTransactionConfiguration { + + public Iti105TransactionConfiguration() { + super("mhd-iti105", + "Simplified Publish", + false, + new Iti105ClientAuditStrategy(), + new Iti105ServerAuditStrategy(), + FhirVersionEnum.R4, + new Iti105DocumentReferenceResourceProvider(), + new Iti105RequestFactory(), + Iti105Validator::new); + } + +} diff --git a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105Validator.java b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105Validator.java new file mode 100644 index 0000000000..e1702bbfe7 --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105Validator.java @@ -0,0 +1,45 @@ +/* + * Copyright 2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.commons.ihe.fhir.iti105; + +import ca.uhn.fhir.context.FhirContext; +import org.hl7.fhir.r4.model.Resource; +import org.openehealth.ipf.commons.ihe.fhir.support.IgBasedInstanceValidator; + +import java.util.Map; + +/** + * Validator for ITI-105 transactions. + * + * + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105Validator extends IgBasedInstanceValidator { + + public static final String ITI105_PROFILE = + "https://profiles.ihe.net/ITI/MHD/StructureDefinition/IHE.MHD.SimplifiedPublish.DocumentReference"; + + public Iti105Validator(FhirContext fhirContext) { + super(fhirContext); + } + + @Override + public void validateRequest(Object payload, Map parameters) { + handleOperationOutcome(validateProfileConformance((Resource) payload, ITI105_PROFILE)); + } +} diff --git a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/mhd/MHD.java b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/mhd/MHD.java index 8142106ea4..5bc26ef26d 100644 --- a/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/mhd/MHD.java +++ b/commons/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/commons/ihe/fhir/mhd/MHD.java @@ -28,6 +28,8 @@ import org.openehealth.ipf.commons.ihe.fhir.FhirTransactionValidator; import org.openehealth.ipf.commons.ihe.fhir.audit.FhirAuditDataset; import org.openehealth.ipf.commons.ihe.fhir.audit.FhirQueryAuditDataset; +import org.openehealth.ipf.commons.ihe.fhir.iti105.Iti105AuditDataset; +import org.openehealth.ipf.commons.ihe.fhir.iti105.Iti105TransactionConfiguration; import org.openehealth.ipf.commons.ihe.fhir.iti65.Iti65AuditDataset; import org.openehealth.ipf.commons.ihe.fhir.iti65.Iti65TransactionConfiguration; import org.openehealth.ipf.commons.ihe.fhir.iti66.Iti66ClientRequestFactory; @@ -117,6 +119,14 @@ public enum RetrieveBinaryInteractions implements InteractionId { private final TransactionConfiguration transactionConfiguration; } + @AllArgsConstructor + public enum SimplifiedPublishInteractions implements FhirInteractionId { + ITI_105(ITI_105_CONFIG); + + @Getter + private final FhirTransactionConfiguration fhirTransactionConfiguration; + } + @Override public List getInteractionIds() { var interactions = new ArrayList(); @@ -131,4 +141,6 @@ public List getInteractionIds() { private static final Iti67TransactionConfiguration ITI_67_CONFIG = new Iti67TransactionConfiguration(); private static final Iti68TransactionConfiguration ITI_68_CONFIG = new Iti68TransactionConfiguration(); private static final Iti68BinaryTransactionConfiguration ITI_68_BIN_CONFIG = new Iti68BinaryTransactionConfiguration(); + private static final Iti105TransactionConfiguration ITI_105_CONFIG = new Iti105TransactionConfiguration(); } + diff --git a/commons/ihe/fhir/r4/mhd/src/main/resources/META-INF/profiles/IHE_MHD_SimplifiedPublish_DocumentReference.xml b/commons/ihe/fhir/r4/mhd/src/main/resources/META-INF/profiles/IHE_MHD_SimplifiedPublish_DocumentReference.xml new file mode 100644 index 0000000000..9b859ccbc3 --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/main/resources/META-INF/profiles/IHE_MHD_SimplifiedPublish_DocumentReference.xml @@ -0,0 +1,2493 @@ + + + + + + + + + <status value="active"/> + <date value="2023-08-02T10:59:15-05:00"/> + <publisher value="IHE IT Infrastructure Technical Committee"/> + <contact> + <telecom> + <system value="url"/> + <value value="https://www.ihe.net/ihe_domains/it_infrastructure/"/> + </telecom> + </contact> + <contact> + <telecom> + <system value="email"/> + <value value="iti@ihe.net"/> + </telecom> + </contact> + <contact> + <name value="IHE IT Infrastructure Technical Committee"/> + <telecom> + <system value="email"/> + <value value="iti@ihe.net"/> + </telecom> + </contact> + <description value="A profile on the DocumentReference resource for MHD Simplified Publish constraints. - MHD is based on the [IHE Document Sharing](https://profiles.ihe.net/ITI/HIE-Whitepaper/index.html) model, - the [3:4.1 Abstract Metadata Model](https://profiles.ihe.net/ITI/TF/Volume3/ch-4.1.html#4.1), and - the use defined here is FHIR DocumentReference implementation of the - ebRIM implementation at [3:4.2.3.2 Document Entry](https://profiles.ihe.net/ITI/TF/Volume3/ch-4.2.html#4.2.3.2). - with use-cases and constraints found in [3:4.3 Additional Document Sharing Requirements](https://profiles.ihe.net/ITI/TF/Volume3/ch-4.3.html#4.3) Simplified Publish - Similar to Minimal Metadata constraints - must be status current - must have a patient indicated - uses attachment.data to carry the document, rather than attachment.url - so that the Simplified Publish is a simple POST of the DocumentReference - Document Recipient is expected to extract the .data, use .url"/> + <jurisdiction> + <coding> + <system value="http://unstats.un.org/unsd/methods/m49/m49.htm"/> + <code value="001"/> + </coding> + </jurisdiction> + <fhirVersion value="4.0.1"/> + <mapping> + <identity value="workflow"/> + <uri value="http://hl7.org/fhir/workflow"/> + <name value="Workflow Pattern"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <uri value="http://hl7.org/fhir/composition"/> + <name value="FHIR Composition"/> + </mapping> + <mapping> + <identity value="rim"/> + <uri value="http://hl7.org/v3"/> + <name value="RIM Mapping"/> + </mapping> + <mapping> + <identity value="cda"/> + <uri value="http://hl7.org/v3/cda"/> + <name value="CDA (R2)"/> + </mapping> + <mapping> + <identity value="w5"/> + <uri value="http://hl7.org/fhir/fivews"/> + <name value="FiveWs Pattern Mapping"/> + </mapping> + <mapping> + <identity value="v2"/> + <uri value="http://hl7.org/v2"/> + <name value="HL7 v2 Mapping"/> + </mapping> + <mapping> + <identity value="xds"/> + <uri value="http://ihe.net/xds"/> + <name value="XDS metadata equivalent"/> + </mapping> + <kind value="resource"/> + <abstract value="false"/> + <type value="DocumentReference"/> + <baseDefinition value="http://hl7.org/fhir/StructureDefinition/DocumentReference"/> + <derivation value="constraint"/> + <snapshot> + <element id="DocumentReference"> + <path value="DocumentReference"/> + <short value="A reference to a document"/> + <definition value="A reference to a document of any kind for any purpose. Provides metadata about the document so that the document can be discovered and managed. The scope of a document is any seralized object with a mime-type, so includes formal patient centric documents (CDA), cliical notes, scanned paper, and non-patient specific documents like policy text."/> + <comment value="Usually, this is used for documents other than those defined by FHIR."/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DocumentReference"/> + <min value="0"/> + <max value="*"/> + </base> + <constraint> + <key value="dom-2"/> + <severity value="error"/> + <human value="If the resource is contained in another resource, it SHALL NOT contain nested Resources"/> + <expression value="contained.contained.empty()"/> + <xpath value="not(parent::f:contained and f:contained)"/> + <source value="http://hl7.org/fhir/StructureDefinition/DomainResource"/> + </constraint> + <constraint> + <key value="dom-3"/> + <severity value="error"/> + <human value="If the resource is contained in another resource, it SHALL be referred to from elsewhere in the resource or SHALL refer to the containing resource"/> + <expression value="contained.where((('#'+id in (%resource.descendants().reference | %resource.descendants().as(canonical) | %resource.descendants().as(uri) | %resource.descendants().as(url))) or descendants().where(reference = '#').exists() or descendants().where(as(canonical) = '#').exists() or descendants().where(as(canonical) = '#').exists()).not()).trace('unmatched', id).empty()"/> + <xpath value="not(exists(for $id in f:contained/*/f:id/@value return $contained[not(parent::*/descendant::f:reference/@value=concat('#', $contained/*/id/@value) or descendant::f:reference[@value='#'])]))"/> + <source value="http://hl7.org/fhir/StructureDefinition/DomainResource"/> + </constraint> + <constraint> + <key value="dom-4"/> + <severity value="error"/> + <human value="If a resource is contained in another resource, it SHALL NOT have a meta.versionId or a meta.lastUpdated"/> + <expression value="contained.meta.versionId.empty() and contained.meta.lastUpdated.empty()"/> + <xpath value="not(exists(f:contained/*/f:meta/f:versionId)) and not(exists(f:contained/*/f:meta/f:lastUpdated))"/> + <source value="http://hl7.org/fhir/StructureDefinition/DomainResource"/> + </constraint> + <constraint> + <key value="dom-5"/> + <severity value="error"/> + <human value="If a resource is contained in another resource, it SHALL NOT have a security label"/> + <expression value="contained.meta.security.empty()"/> + <xpath value="not(exists(f:contained/*/f:meta/f:security))"/> + <source value="http://hl7.org/fhir/StructureDefinition/DomainResource"/> + </constraint> + <constraint> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bestpractice"> + <valueBoolean value="true"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bestpractice-explanation"> + <valueMarkdown value="When a resource has no narrative, only systems that fully understand the data can display the resource to a human safely. Including a human readable representation in the resource makes for a much more robust eco-system and cheaper handling of resources by intermediary systems. Some ecosystems restrict distribution of resources to only those systems that do fully understand the resources, and as a consequence implementers may believe that the narrative is superfluous. However experience shows that such eco-systems often open up to new participants over time."/> + </extension> + <key value="dom-6"/> + <severity value="warning"/> + <human value="A resource should have narrative for robust management"/> + <expression value="text.`div`.exists()"/> + <xpath value="exists(f:text/h:div)"/> + <source value="http://hl7.org/fhir/StructureDefinition/DomainResource"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="Entity. Role, or Act"/> + </mapping> + <mapping> + <identity value="workflow"/> + <map value="Event"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="when describing a Composition"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="Document[classCode="DOC" and moodCode="EVN"]"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="when describing a CDA"/> + </mapping> + </element> + <element id="DocumentReference.id"> + <path value="DocumentReference.id"/> + <short value="Logical id of this artifact"/> + <definition value="The logical id of the resource, as used in the URL for the resource. Once assigned, this value never changes."/> + <comment value="The only time that a resource does not have an id is when it is being submitted to the server using a create operation."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Resource.id"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <extension url="http://hl7.org/fhir/StructureDefinition/structuredefinition-fhir-type"> + <valueUrl value="id"/> + </extension> + <code value="http://hl7.org/fhirpath/System.String"/> + </type> + <isModifier value="false"/> + <isSummary value="true"/> + </element> + <element id="DocumentReference.meta"> + <path value="DocumentReference.meta"/> + <short value="Metadata about the resource"/> + <definition value="The metadata about the resource. This is content that is maintained by the infrastructure. Changes to the content might not always be associated with version changes to the resource."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Resource.meta"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Meta"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + </element> + <element id="DocumentReference.implicitRules"> + <path value="DocumentReference.implicitRules"/> + <short value="A set of rules under which this content was created"/> + <definition value="A reference to a set of rules that were followed when the resource was constructed, and which must be understood when processing the content. Often, this is a reference to an implementation guide that defines the special rules along with other profiles etc."/> + <comment value="Asserting this rule set restricts the content to be only understood by a limited set of trading partners. This inherently limits the usefulness of the data in the long term. However, the existing health eco-system is highly fractured, and not yet ready to define, collect, and exchange data in a generally computable sense. Wherever possible, implementers and/or specification writers should avoid using this element. Often, when used, the URL is a reference to an implementation guide that defines these special rules as part of it's narrative along with other profiles, value sets, etc."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Resource.implicitRules"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="uri"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="true"/> + <isModifierReason value="This element is labeled as a modifier because the implicit rules may provide additional knowledge about the resource that modifies it's meaning or interpretation"/> + <isSummary value="true"/> + </element> + <element id="DocumentReference.language"> + <path value="DocumentReference.language"/> + <short value="Language of the resource content"/> + <definition value="The base language in which the resource is written."/> + <comment value="Language is provided to support indexing and accessibility (typically, services such as text to speech use the language tag). The html language tag in the narrative applies to the narrative. The language tag on the resource may be used to specify the language of other presentations generated from the data in the resource. Not all the content has to be in the base language. The Resource.language should not be assumed to apply to the narrative automatically. If a language is specified, it should it also be specified on the div element in the html (see rules in HTML5 for information about the relationship between xml:lang and the html lang attribute)."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Resource.language"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="code"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-maxValueSet"> + <valueCanonical value="http://hl7.org/fhir/ValueSet/all-languages"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="Language"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-isCommonBinding"> + <valueBoolean value="true"/> + </extension> + <strength value="preferred"/> + <description value="A human language."/> + <valueSet value="http://hl7.org/fhir/ValueSet/languages"/> + </binding> + </element> + <element id="DocumentReference.text"> + <path value="DocumentReference.text"/> + <short value="Text summary of the resource, for human interpretation"/> + <definition value="A human-readable narrative that contains a summary of the resource and can be used to represent the content of the resource to a human. The narrative need not encode all the structured data, but is required to contain sufficient detail to make it "clinically safe" for a human to just read the narrative. Resource definitions may define what content should be represented in the narrative to ensure clinical safety."/> + <comment value="Contained resources do not have narrative. Resources that are not contained SHOULD have a narrative. In some cases, a resource may only have text with little or no additional discrete data (as long as all minOccurs=1 elements are satisfied). This may be necessary for data from legacy systems where information is captured as a "text blob" or where text is additionally entered raw or narrated and encoded information is added later."/> + <alias value="narrative"/> + <alias value="html"/> + <alias value="xhtml"/> + <alias value="display"/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DomainResource.text"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Narrative"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="Act.text?"/> + </mapping> + </element> + <element id="DocumentReference.contained"> + <path value="DocumentReference.contained"/> + <short value="Contained, inline Resources"/> + <definition value="These resources do not have an independent existence apart from the resource that contains them - they cannot be identified independently, and nor can they have their own independent transaction scope."/> + <comment value="This should never be done when the content can be identified properly, as once identification is lost, it is extremely difficult (and context dependent) to restore it again. Contained resources may have profiles and tags In their meta elements, but SHALL NOT have security labels."/> + <alias value="inline resources"/> + <alias value="anonymous resources"/> + <alias value="contained resources"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DomainResource.contained"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Resource"/> + </type> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="N/A"/> + </mapping> + </element> + <element id="DocumentReference.extension"> + <path value="DocumentReference.extension"/> + <short value="Additional content defined by implementations"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the resource. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <alias value="extensions"/> + <alias value="user content"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DomainResource.extension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="N/A"/> + </mapping> + </element> + <element id="DocumentReference.modifierExtension"> + <path value="DocumentReference.modifierExtension"/> + <short value="Extension"/> + <definition value="An Extension"/> + <min value="0"/> + <max value="0"/> + <base> + <path value="DomainResource.modifierExtension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="true"/> + <isModifierReason value="Modifier extensions are expected to modify the meaning or interpretation of the resource that contains them"/> + <isSummary value="false"/> + </element> + <element id="DocumentReference.masterIdentifier"> + <path value="DocumentReference.masterIdentifier"/> + <short value="Master Version Specific Identifier"/> + <definition value="Document identifier as assigned by the source of the document. This identifier is specific to this version of the document. This unique identifier may be used elsewhere to identify this version of the document."/> + <comment value="CDA Document Id extension and root."/> + <requirements value="The structure and format of this Id shall be consistent with the specification corresponding to the formatCode attribute. (e.g. for a DICOM standard document a 64-character numeric UID, for an HL7 CDA format a serialization of the CDA Document Id extension and root in the form "oid^extension", where OID is a 64 digits max, and the Id is a 16 UTF-8 char max. If the OID is coded without the extension then the '^' character shall not be included.)."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="DocumentReference.masterIdentifier"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Identifier"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="workflow"/> + <map value="Event.identifier"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.identifier"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.identifier"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-12"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".id"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.uniqueId"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/id"/> + </mapping> + </element> + <element id="DocumentReference.identifier"> + <path value="DocumentReference.identifier"/> + <short value="Other identifiers for the document"/> + <definition value="Other identifiers associated with the document, including version independent identifiers."/> + <min value="0"/> + <max value="0"/> + <base> + <path value="DocumentReference.identifier"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Identifier"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="workflow"/> + <map value="Event.identifier"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.identifier"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-16?"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".id / .setId"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.entryUUID"/> + </mapping> + </element> + <element id="DocumentReference.status"> + <path value="DocumentReference.status"/> + <short value="current | superseded | entered-in-error"/> + <definition value="The status of this document reference."/> + <comment value="This is the status of the DocumentReference object, which might be independent from the docStatus element. This element is labeled as a modifier because the status contains the codes that mark the document or reference as not currently valid."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="DocumentReference.status"/> + <min value="1"/> + <max value="1"/> + </base> + <type> + <code value="code"/> + </type> + <patternCode value="current"/> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="true"/> + <isModifierReason value="This element is labelled as a modifier because it is a status element that contains status entered-in-error which means that the resource should not be treated as valid"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentReferenceStatus"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-isCommonBinding"> + <valueBoolean value="true"/> + </extension> + <strength value="required"/> + <description value="The status of the document reference."/> + <valueSet value="http://hl7.org/fhir/ValueSet/document-reference-status|4.0.1"/> + </binding> + <mapping> + <identity value="workflow"/> + <map value="Event.status"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.status"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-19"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="interim: .completionCode="IN" & ./statusCode[isNormalDatatype()]="active"; final: .completionCode="AU" && ./statusCode[isNormalDatatype()]="complete" and not(./inboundRelationship[typeCode="SUBJ" and isNormalActRelationship()]/source[subsumesCode("ActClass#CACT") and moodCode="EVN" and domainMember("ReviseDocument", code) and isNormalAct()]); amended: .completionCode="AU" && ./statusCode[isNormalDatatype()]="complete" and ./inboundRelationship[typeCode="SUBJ" and isNormalActRelationship()]/source[subsumesCode("ActClass#CACT") and moodCode="EVN" and domainMember("ReviseDocument", code) and isNormalAct() and statusCode="completed"]; withdrawn : .completionCode=NI && ./statusCode[isNormalDatatype()]="obsolete""/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.availabilityStatus"/> + </mapping> + </element> + <element id="DocumentReference.docStatus"> + <path value="DocumentReference.docStatus"/> + <short value="preliminary | final | amended | entered-in-error"/> + <definition value="The status of the underlying document."/> + <comment value="The document that is pointed to might be in various lifecycle states."/> + <min value="0"/> + <max value="0"/> + <base> + <path value="DocumentReference.docStatus"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="code"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="ReferredDocumentStatus"/> + </extension> + <strength value="required"/> + <description value="Status of the underlying document."/> + <valueSet value="http://hl7.org/fhir/ValueSet/composition-status|4.0.1"/> + </binding> + <mapping> + <identity value="w5"/> + <map value="FiveWs.status"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.status"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-17"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".statusCode"/> + </mapping> + </element> + <element id="DocumentReference.type"> + <path value="DocumentReference.type"/> + <short value="Kind of document (LOINC if possible)"/> + <definition value="Specifies the particular kind of document referenced (e.g. History and Physical, Discharge Summary, Progress Note). This usually equates to the purpose of making the document referenced."/> + <comment value="Key metadata element describing the document that describes he exact type of document. Helps humans to assess whether the document is of interest when viewing a list of documents."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.type"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="CodeableConcept"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentC80Type"/> + </extension> + <strength value="preferred"/> + <description value="Precise type of clinical document."/> + <valueSet value="http://hl7.org/fhir/ValueSet/c80-doc-typecodes"/> + </binding> + <mapping> + <identity value="workflow"/> + <map value="Event.code"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.class"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.type"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-2"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="./code"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.type"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/code/@code The typeCode should be mapped from the ClinicalDocument/code element to a set of document type codes configured in the affinity domain. One suggested coding system to use for typeCode is LOINC, in which case the mapping step can be omitted."/> + </mapping> + </element> + <element id="DocumentReference.category"> + <path value="DocumentReference.category"/> + <short value="Categorization of document"/> + <definition value="A categorization for the type of document referenced - helps for indexing and searching. This may be implied by or derived from the code specified in the DocumentReference.type."/> + <comment value="Key metadata element describing the the category or classification of the document. This is a broader perspective that groups similar documents based on how they would be used. This is a primary key used in searching."/> + <alias value="claxs"/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.category"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="CodeableConcept"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentC80Class"/> + </extension> + <strength value="example"/> + <description value="High-level kind of a clinical document at a macro level."/> + <valueSet value="http://hl7.org/fhir/ValueSet/document-classcodes"/> + </binding> + <mapping> + <identity value="w5"/> + <map value="FiveWs.class"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.class"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".outboundRelationship[typeCode="COMP].target[classCode="LIST", moodCode="EVN"].code"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.class"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="Derived from a mapping of /ClinicalDocument/code/@code to an Affinity Domain specified coded value to use and coding system. Affinity Domains are encouraged to use the appropriate value for Type of Service, based on the LOINC Type of Service (see Page 53 of the LOINC User's Manual). Must be consistent with /ClinicalDocument/code/@code"/> + </mapping> + </element> + <element id="DocumentReference.subject"> + <path value="DocumentReference.subject"/> + <short value="Who/what is the subject of the document"/> + <definition value="Who or what the document is about. The document can be about a person, (patient or healthcare practitioner), a device (e.g. a machine) or even a group of subjects (such as a document about a herd of farm animals, or a set of patients that share a common exposure)."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="DocumentReference.subject"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Patient"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="workflow"/> + <map value="Event.subject"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.subject[x]"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.subject"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="PID-3 (No standard way to define a Practitioner or Group subject in HL7 v2 MDM message)"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".participation[typeCode="SBJ"].role[typeCode="PAT"]"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.patientId"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/recordTarget/"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.subject"/> + </mapping> + </element> + <element id="DocumentReference.date"> + <path value="DocumentReference.date"/> + <short value="When this document reference was created"/> + <definition value="When the document reference was created."/> + <comment value="Referencing/indexing time is used for tracking, organizing versions and searching."/> + <alias value="indexed"/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.date"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="instant"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="workflow"/> + <map value="Event.occurrence[x]"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.recorded"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.date"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".availabilityTime[type="TS"]"/> + </mapping> + </element> + <element id="DocumentReference.author"> + <path value="DocumentReference.author"/> + <short value="Who and/or what authored the document"/> + <definition value="Identifies who is responsible for adding the information to the document."/> + <comment value="Not necessarily who did the actual data entry (i.e. typist) or who was the source (informant)."/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DocumentReference.author"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Practitioner"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/PractitionerRole"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Organization"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Device"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Patient"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/RelatedPerson"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="workflow"/> + <map value="Event.performer.actor"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.author"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-9 (No standard way to indicate a Device in HL7 v2 MDM message)"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".participation[typeCode="AUT"].role[classCode="ASSIGNED"]"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.author"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/author"/> + </mapping> + </element> + <element id="DocumentReference.authenticator"> + <path value="DocumentReference.authenticator"/> + <short value="Who/what authenticated the document"/> + <definition value="Which person or organization authenticates that this document is valid."/> + <comment value="Represents a participant within the author institution who has legally authenticated or attested the document. Legal authentication implies that a document has been signed manually or electronically by the legal Authenticator."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.authenticator"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Practitioner"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/PractitionerRole"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Organization"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="workflow"/> + <map value="Event.performer.actor"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.witness"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.attester"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-10"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".participation[typeCode="AUTHEN"].role[classCode="ASSIGNED"]"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.legalAuthenticator"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/legalAuthenticator"/> + </mapping> + </element> + <element id="DocumentReference.custodian"> + <path value="DocumentReference.custodian"/> + <short value="Organization which maintains the document"/> + <definition value="Identifies the organization or group who is responsible for ongoing maintenance of and access to the document."/> + <comment value="Identifies the logical organization (software system, vendor, or department) to go to find the current version, where to report issues, etc. This is different from the physical location (URL, disk drive, or server) of the document, which is the technical location of the document, which host may be delegated to the management of some other organization."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.custodian"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Organization"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="workflow"/> + <map value="Event.performer.actor"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.custodian"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".participation[typeCode="RCV"].role[classCode="CUST"].scoper[classCode="ORG" and determinerCode="INST"]"/> + </mapping> + </element> + <element id="DocumentReference.relatesTo"> + <path value="DocumentReference.relatesTo"/> + <short value="Relationships to other documents"/> + <definition value="Relationships that this document has with other document references that already exist."/> + <comment value="This element is labeled as a modifier because documents that append to other documents are incomplete on their own."/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DocumentReference.relatesTo"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="BackboneElement"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.relatesTo"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".outboundRelationship"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry Associations"/> + </mapping> + </element> + <element id="DocumentReference.relatesTo.id"> + <path value="DocumentReference.relatesTo.id"/> + <representation value="xmlAttr"/> + <short value="Unique id for inter-element referencing"/> + <definition value="Unique id for the element within a resource (for internal references). This may be any string value that does not contain spaces."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Element.id"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <extension url="http://hl7.org/fhir/StructureDefinition/structuredefinition-fhir-type"> + <valueUrl value="string"/> + </extension> + <code value="http://hl7.org/fhirpath/System.String"/> + </type> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.relatesTo.extension"> + <path value="DocumentReference.relatesTo.extension"/> + <short value="Additional content defined by implementations"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the element. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <alias value="extensions"/> + <alias value="user content"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="Element.extension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.relatesTo.modifierExtension"> + <path value="DocumentReference.relatesTo.modifierExtension"/> + <short value="Extensions that cannot be ignored even if unrecognized"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the element and that modifies the understanding of the element in which it is contained and/or the understanding of the containing element's descendants. Usually modifier elements provide negation or qualification. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension. Applications processing a resource are required to check for modifier extensions. Modifier extensions SHALL NOT change the meaning of any elements on Resource or DomainResource (including cannot change the meaning of modifierExtension itself)."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <requirements value="Modifier extensions allow for extensions that *cannot* be safely ignored to be clearly distinguished from the vast majority of extensions which can be safely ignored. This promotes interoperability by eliminating the need for implementers to prohibit the presence of extensions. For further information, see the [definition of modifier extensions](http://hl7.org/fhir/R4/extensibility.html#modifierExtension)."/> + <alias value="extensions"/> + <alias value="user content"/> + <alias value="modifiers"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="BackboneElement.modifierExtension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="true"/> + <isModifierReason value="Modifier extensions are expected to modify the meaning or interpretation of the element that contains them"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value="N/A"/> + </mapping> + </element> + <element id="DocumentReference.relatesTo.code"> + <path value="DocumentReference.relatesTo.code"/> + <short value="replaces | transforms | signs | appends"/> + <definition value="The type of relationship that this document has with anther document."/> + <comment value="If this document appends another document, then the document cannot be fully understood without also accessing the referenced document."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="DocumentReference.relatesTo.code"/> + <min value="1"/> + <max value="1"/> + </base> + <type> + <code value="code"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentRelationshipType"/> + </extension> + <strength value="required"/> + <description value="The type of relationship between documents."/> + <valueSet value="http://hl7.org/fhir/ValueSet/document-relationship-type|4.0.1"/> + </binding> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.relatesTo.code"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".outboundRelationship.typeCode"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry Associations type"/> + </mapping> + </element> + <element id="DocumentReference.relatesTo.target"> + <path value="DocumentReference.relatesTo.target"/> + <short value="Target of the relationship"/> + <definition value="The target document of this relationship."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="DocumentReference.relatesTo.target"/> + <min value="1"/> + <max value="1"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/DocumentReference"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.relatesTo.target"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".target[classCode="DOC", moodCode="EVN"].id"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry Associations reference"/> + </mapping> + </element> + <element id="DocumentReference.description"> + <path value="DocumentReference.description"/> + <short value="Human-readable description"/> + <definition value="Human-readable description of the source document."/> + <comment value="What the document is about, a terse summary of the document."/> + <requirements value="Helps humans to assess whether the document is of interest."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.description"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="string"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="v2"/> + <map value="TXA-25"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".outboundRelationship[typeCode="SUBJ"].target.text"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.comments"/> + </mapping> + </element> + <element id="DocumentReference.securityLabel"> + <path value="DocumentReference.securityLabel"/> + <short value="Document security-tags"/> + <definition value="A set of Security-Tag codes specifying the level of privacy/security of the Document. Note that DocumentReference.meta.security contains the security labels of the "reference" to the document, while DocumentReference.securityLabel contains a snapshot of the security labels on the document the reference refers to."/> + <comment value="The confidentiality codes can carry multiple vocabulary items. HL7 has developed an understanding of security and privacy tags that might be desirable in a Document Sharing environment, called HL7 Healthcare Privacy and Security Classification System (HCS). The following specification is recommended but not mandated, as the vocabulary bindings are an administrative domain responsibility. The use of this method is up to the policy domain such as the XDS Affinity Domain or other Trust Domain where all parties including sender and recipients are trusted to appropriately tag and enforce. In the HL7 Healthcare Privacy and Security Classification (HCS) there are code systems specific to Confidentiality, Sensitivity, Integrity, and Handling Caveats. Some values would come from a local vocabulary as they are related to workflow roles and special projects."/> + <requirements value="Use of the Health Care Privacy/Security Classification (HCS) system of security-tag use is recommended."/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DocumentReference.securityLabel"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="CodeableConcept"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="SecurityLabels"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-isCommonBinding"> + <valueBoolean value="true"/> + </extension> + <strength value="extensible"/> + <description value="Security Labels from the Healthcare Privacy and Security Classification System."/> + <valueSet value="http://hl7.org/fhir/ValueSet/security-labels"/> + </binding> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.confidentiality, Composition.meta.security"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-18"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".confidentialityCode"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.confidentialityCode"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/confidentialityCode/@code"/> + </mapping> + </element> + <element id="DocumentReference.content"> + <path value="DocumentReference.content"/> + <short value="Document referenced"/> + <definition value="The document and format referenced."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="DocumentReference.content"/> + <min value="1"/> + <max value="*"/> + </base> + <type> + <code value="BackboneElement"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="fhircomposition"/> + <map value="Bundle(Composition+*)"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="document.text"/> + </mapping> + </element> + <element id="DocumentReference.content.id"> + <path value="DocumentReference.content.id"/> + <representation value="xmlAttr"/> + <short value="Unique id for inter-element referencing"/> + <definition value="Unique id for the element within a resource (for internal references). This may be any string value that does not contain spaces."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Element.id"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <extension url="http://hl7.org/fhir/StructureDefinition/structuredefinition-fhir-type"> + <valueUrl value="string"/> + </extension> + <code value="http://hl7.org/fhirpath/System.String"/> + </type> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.content.extension"> + <path value="DocumentReference.content.extension"/> + <short value="Additional content defined by implementations"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the element. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <alias value="extensions"/> + <alias value="user content"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="Element.extension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.content.modifierExtension"> + <path value="DocumentReference.content.modifierExtension"/> + <short value="Extensions that cannot be ignored even if unrecognized"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the element and that modifies the understanding of the element in which it is contained and/or the understanding of the containing element's descendants. Usually modifier elements provide negation or qualification. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension. Applications processing a resource are required to check for modifier extensions. Modifier extensions SHALL NOT change the meaning of any elements on Resource or DomainResource (including cannot change the meaning of modifierExtension itself)."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <requirements value="Modifier extensions allow for extensions that *cannot* be safely ignored to be clearly distinguished from the vast majority of extensions which can be safely ignored. This promotes interoperability by eliminating the need for implementers to prohibit the presence of extensions. For further information, see the [definition of modifier extensions](http://hl7.org/fhir/R4/extensibility.html#modifierExtension)."/> + <alias value="extensions"/> + <alias value="user content"/> + <alias value="modifiers"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="BackboneElement.modifierExtension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="true"/> + <isModifierReason value="Modifier extensions are expected to modify the meaning or interpretation of the element that contains them"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value="N/A"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment"> + <path value="DocumentReference.content.attachment"/> + <short value="Where to access the document"/> + <definition value="The document or URL of the document along with critical metadata to prove content has integrity."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="DocumentReference.content.attachment"/> + <min value="1"/> + <max value="1"/> + </base> + <type> + <code value="Attachment"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.language, Composition.title, Composition.date"/> + </mapping> + <mapping> + <identity value="v2"/> + <map value="TXA-3 for mime type"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="document.text"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.mimeType, DocumentEntry.languageCode, DocumentEntry.URI, DocumentEntry.size, DocumentEntry.hash, DocumentEntry.title, DocumentEntry.creationTime"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/languageCode, ClinicalDocument/title, ClinicalDocument/date"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.id"> + <path value="DocumentReference.content.attachment.id"/> + <representation value="xmlAttr"/> + <short value="Unique id for inter-element referencing"/> + <definition value="Unique id for the element within a resource (for internal references). This may be any string value that does not contain spaces."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Element.id"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <extension url="http://hl7.org/fhir/StructureDefinition/structuredefinition-fhir-type"> + <valueUrl value="string"/> + </extension> + <code value="http://hl7.org/fhirpath/System.String"/> + </type> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.extension"> + <path value="DocumentReference.content.attachment.extension"/> + <slicing> + <discriminator> + <type value="value"/> + <path value="url"/> + </discriminator> + <description value="Extensions are always sliced by (at least) url"/> + <rules value="open"/> + </slicing> + <short value="Additional content defined by implementations"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the element. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <alias value="extensions"/> + <alias value="user content"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="Element.extension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.contentType"> + <path value="DocumentReference.content.attachment.contentType"/> + <short value="Mime type of the content, with charset etc."/> + <definition value="Identifies the type of the data in the attachment and allows a method to be chosen to interpret or render the data. Includes mime type parameters such as charset where appropriate."/> + <requirements value="Processors of the data need to be able to know how to interpret the data."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="Attachment.contentType"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="code"/> + </type> + <example> + <label value="General"/> + <valueCode value="text/plain; charset=UTF-8, image/png"/> + </example> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="MimeType"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-isCommonBinding"> + <valueBoolean value="true"/> + </extension> + <strength value="required"/> + <description value="The mime type of an attachment. Any valid mime type is allowed."/> + <valueSet value="http://hl7.org/fhir/ValueSet/mimetypes|4.0.1"/> + </binding> + <mapping> + <identity value="v2"/> + <map value="ED.2+ED.3/RP.2+RP.3. Note conversion may be needed if old style values are being used"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="./mediaType, ./charset"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.language"> + <path value="DocumentReference.content.attachment.language"/> + <short value="Human language of the content (BCP-47)"/> + <definition value="The human language of the content. The value can be any valid value according to BCP 47."/> + <requirements value="Users need to be able to choose between the languages in a set of attachments."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Attachment.language"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="code"/> + </type> + <example> + <label value="General"/> + <valueCode value="en-AU"/> + </example> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-maxValueSet"> + <valueCanonical value="http://hl7.org/fhir/ValueSet/all-languages"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="Language"/> + </extension> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-isCommonBinding"> + <valueBoolean value="true"/> + </extension> + <strength value="preferred"/> + <description value="A human language."/> + <valueSet value="http://hl7.org/fhir/ValueSet/languages"/> + </binding> + <mapping> + <identity value="rim"/> + <map value="./language"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.data"> + <path value="DocumentReference.content.attachment.data"/> + <short value="The document is contained in the .data element, base64 encoded"/> + <definition value="The actual data of the attachment - a sequence of bytes, base64 encoded."/> + <comment value="The base64-encoded data SHALL be expressed in the same character set as the base resource XML or JSON."/> + <requirements value="The data needs to able to be transmitted inline."/> + <min value="1"/> + <max value="1"/> + <base> + <path value="Attachment.data"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="base64Binary"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="v2"/> + <map value="ED.5"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="./data"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.url"> + <path value="DocumentReference.content.attachment.url"/> + <short value="Uri where the data can be found"/> + <definition value="A location where the data can be accessed."/> + <comment value="If both data and url are provided, the url SHALL point to the same content as the data contains. Urls may be relative references or may reference transient locations such as a wrapping envelope using cid: though this has ramifications for using signatures. Relative URLs are interpreted relative to the service url, like a resource reference, rather than relative to the resource itself. If a URL is provided, it SHALL resolve to actual data."/> + <requirements value="The data needs to be transmitted by reference."/> + <min value="0"/> + <max value="0"/> + <base> + <path value="Attachment.url"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="url"/> + </type> + <example> + <label value="General"/> + <valueUrl value="http://www.acme.com/logo-small.png"/> + </example> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="v2"/> + <map value="RP.1+RP.2 - if they refer to a URL (see v2.6)"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="./reference/literal"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.size"> + <path value="DocumentReference.content.attachment.size"/> + <short value="Number of bytes of content (if url provided)"/> + <definition value="The number of bytes of data that make up this attachment (before base64 encoding, if that is done)."/> + <comment value="The number of bytes is redundant if the data is provided as a base64binary, but is useful if the data is provided as a url reference."/> + <requirements value="Representing the size allows applications to determine whether they should fetch the content automatically in advance, or refuse to fetch it at all."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Attachment.size"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="unsignedInt"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value="N/A (needs data type R3 proposal)"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.hash"> + <path value="DocumentReference.content.attachment.hash"/> + <short value="Hash of the data (sha-1, base64ed)"/> + <definition value="The calculated hash of the data using SHA-1. Represented using base64."/> + <comment value="The hash is calculated on the data prior to base64 encoding, if the data is based64 encoded. The hash is not intended to support digital signatures. Where protection against malicious threats a digital signature should be considered, see [Provenance.signature](http://hl7.org/fhir/R4/provenance-definitions.html#Provenance.signature) for mechanism to protect a resource with a digital signature."/> + <requirements value="Included so that applications can verify that the contents of a location have not changed due to technical failures (e.g., storage rot, transport glitch, incorrect version)."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Attachment.hash"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="base64Binary"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value=".integrityCheck[parent::ED/integrityCheckAlgorithm="SHA-1"]"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.title"> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-translatable"> + <valueBoolean value="true"/> + </extension> + <path value="DocumentReference.content.attachment.title"/> + <short value="Label to display in place of the data"/> + <definition value="A label or set of text to display in place of the data."/> + <requirements value="Applications need a label to display to a human user in place of the actual data if the data cannot be rendered or perceived by the viewer."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Attachment.title"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="string"/> + </type> + <example> + <label value="General"/> + <valueString value="Official Corporate Logo"/> + </example> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value="./title/data"/> + </mapping> + </element> + <element id="DocumentReference.content.attachment.creation"> + <path value="DocumentReference.content.attachment.creation"/> + <short value="Date attachment was first created"/> + <definition value="The date that the attachment was first created."/> + <requirements value="This is often tracked as an integrity issue for use of the attachment."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Attachment.creation"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="dateTime"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value="N/A (needs data type R3 proposal)"/> + </mapping> + </element> + <element id="DocumentReference.content.format"> + <path value="DocumentReference.content.format"/> + <short value="Format/content rules for the document"/> + <definition value="An identifier of the document encoding, structure, and template that the document conforms to beyond the base format indicated in the mimeType."/> + <comment value="Note that while IHE mostly issues URNs for format types, not all documents can be identified by a URI."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.content.format"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Coding"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentFormat"/> + </extension> + <strength value="preferred"/> + <description value="Document Format Codes."/> + <valueSet value="http://hl7.org/fhir/ValueSet/formatcodes"/> + </binding> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.meta.profile"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="document.text"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.formatCode"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="derived from the IHE Profile or Implementation Guide templateID"/> + </mapping> + </element> + <element id="DocumentReference.context"> + <path value="DocumentReference.context"/> + <short value="Clinical context of document"/> + <definition value="The clinical context in which the document was prepared."/> + <comment value="These values are primarily added to help with searching for interesting/relevant documents."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.context"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="BackboneElement"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value="outboundRelationship[typeCode="SUBJ"].target[classCode<'ACT']"/> + </mapping> + </element> + <element id="DocumentReference.context.id"> + <path value="DocumentReference.context.id"/> + <representation value="xmlAttr"/> + <short value="Unique id for inter-element referencing"/> + <definition value="Unique id for the element within a resource (for internal references). This may be any string value that does not contain spaces."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="Element.id"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <extension url="http://hl7.org/fhir/StructureDefinition/structuredefinition-fhir-type"> + <valueUrl value="string"/> + </extension> + <code value="http://hl7.org/fhirpath/System.String"/> + </type> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.context.extension"> + <path value="DocumentReference.context.extension"/> + <short value="Additional content defined by implementations"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the element. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <alias value="extensions"/> + <alias value="user content"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="Element.extension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="rim"/> + <map value="n/a"/> + </mapping> + </element> + <element id="DocumentReference.context.modifierExtension"> + <path value="DocumentReference.context.modifierExtension"/> + <short value="Extensions that cannot be ignored even if unrecognized"/> + <definition value="May be used to represent additional information that is not part of the basic definition of the element and that modifies the understanding of the element in which it is contained and/or the understanding of the containing element's descendants. Usually modifier elements provide negation or qualification. To make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer can define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension. Applications processing a resource are required to check for modifier extensions. Modifier extensions SHALL NOT change the meaning of any elements on Resource or DomainResource (including cannot change the meaning of modifierExtension itself)."/> + <comment value="There can be no stigma associated with the use of extensions by any application, project, or standard - regardless of the institution or jurisdiction that uses or defines the extensions. The use of extensions is what allows the FHIR specification to retain a core level of simplicity for everyone."/> + <requirements value="Modifier extensions allow for extensions that *cannot* be safely ignored to be clearly distinguished from the vast majority of extensions which can be safely ignored. This promotes interoperability by eliminating the need for implementers to prohibit the presence of extensions. For further information, see the [definition of modifier extensions](http://hl7.org/fhir/R4/extensibility.html#modifierExtension)."/> + <alias value="extensions"/> + <alias value="user content"/> + <alias value="modifiers"/> + <min value="0"/> + <max value="*"/> + <base> + <path value="BackboneElement.modifierExtension"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Extension"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <constraint> + <key value="ext-1"/> + <severity value="error"/> + <human value="Must have either extensions or value[x], not both"/> + <expression value="extension.exists() != value.exists()"/> + <xpath value="exists(f:extension)!=exists(f:*[starts-with(local-name(.), "value")])"/> + <source value="http://hl7.org/fhir/StructureDefinition/Extension"/> + </constraint> + <isModifier value="true"/> + <isModifierReason value="Modifier extensions are expected to modify the meaning or interpretation of the element that contains them"/> + <isSummary value="true"/> + <mapping> + <identity value="rim"/> + <map value="N/A"/> + </mapping> + </element> + <element id="DocumentReference.context.encounter"> + <path value="DocumentReference.context.encounter"/> + <short value="Context of the document content"/> + <definition value="Describes the clinical encounter or type of care that the document content is associated with."/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DocumentReference.context.encounter"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Encounter"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/EpisodeOfCare"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="workflow"/> + <map value="Event.context"/> + </mapping> + <mapping> + <identity value="w5"/> + <map value="FiveWs.context"/> + </mapping> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.encounter"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="unique(highest(./outboundRelationship[typeCode="SUBJ" and isNormalActRelationship()], priorityNumber)/target[moodCode="EVN" and classCode=("ENC", "PCPR") and isNormalAct])"/> + </mapping> + </element> + <element id="DocumentReference.context.event"> + <path value="DocumentReference.context.event"/> + <short value="Main clinical acts documented"/> + <definition value="This list of codes represents the main clinical acts, such as a colonoscopy or an appendectomy, being documented. In some cases, the event is inherent in the type Code, such as a "History and Physical Report" in which the procedure being documented is necessarily a "History and Physical" act."/> + <comment value="An event can further specialize the act inherent in the type, such as where it is simply "Procedure Report" and the procedure was a "colonoscopy". If one or more event codes are included, they shall not conflict with the values inherent in the class or type elements as such a conflict would create an ambiguous situation."/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DocumentReference.context.event"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="CodeableConcept"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentEventType"/> + </extension> + <strength value="example"/> + <description value="This list of codes represents the main clinical acts being documented."/> + <valueSet value="http://terminology.hl7.org/ValueSet/v3-ActCode"/> + </binding> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.event.code"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".code"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.eventCodeList"/> + </mapping> + </element> + <element id="DocumentReference.context.period"> + <path value="DocumentReference.context.period"/> + <short value="Time of service that is being documented"/> + <definition value="The time period over which the service that is described by the document was provided."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.context.period"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Period"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="true"/> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.event.period"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".effectiveTime"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.serviceStartTime, DocumentEntry.serviceStopTime"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/documentationOf/ serviceEvent/effectiveTime/low/ @value --> ClinicalDocument/documentationOf/ serviceEvent/effectiveTime/high/ @value"/> + </mapping> + </element> + <element id="DocumentReference.context.facilityType"> + <path value="DocumentReference.context.facilityType"/> + <short value="Kind of facility where patient was seen"/> + <definition value="The kind of facility where the patient was seen."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.context.facilityType"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="CodeableConcept"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="false"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentC80FacilityType"/> + </extension> + <strength value="example"/> + <description value="XDS Facility Type."/> + <valueSet value="http://hl7.org/fhir/ValueSet/c80-facilitycodes"/> + </binding> + <mapping> + <identity value="fhircomposition"/> + <map value="usually from a mapping to a local ValueSet"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".participation[typeCode="LOC"].role[classCode="DSDLOC"].code"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.healthcareFacilityTypeCode"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="usually a mapping to a local ValueSet. Must be consistent with /clinicalDocument/code"/> + </mapping> + </element> + <element id="DocumentReference.context.practiceSetting"> + <path value="DocumentReference.context.practiceSetting"/> + <short value="Additional details about where the content was created (e.g. clinical specialty)"/> + <definition value="This property may convey specifics about the practice setting where the content was created, often reflecting the clinical specialty."/> + <comment value="This element should be based on a coarse classification system for the class of specialty practice. Recommend the use of the classification system for Practice Setting, such as that described by the Subject Matter Domain in LOINC."/> + <requirements value="This is an important piece of metadata that providers often rely upon to quickly sort and/or filter out to find specific content."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.context.practiceSetting"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="CodeableConcept"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="false"/> + <binding> + <extension url="http://hl7.org/fhir/StructureDefinition/elementdefinition-bindingName"> + <valueString value="DocumentC80PracticeSetting"/> + </extension> + <strength value="example"/> + <description value="Additional details about where the content was created (e.g. clinical specialty)."/> + <valueSet value="http://hl7.org/fhir/ValueSet/c80-practice-codes"/> + </binding> + <mapping> + <identity value="fhircomposition"/> + <map value="usually from a mapping to a local ValueSet"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".participation[typeCode="LOC"].role[classCode="DSDLOC"].code"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.practiceSettingCode"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="usually from a mapping to a local ValueSet"/> + </mapping> + </element> + <element id="DocumentReference.context.sourcePatientInfo"> + <path value="DocumentReference.context.sourcePatientInfo"/> + <short value="Patient demographics from source"/> + <definition value="The Patient Information as known when the document was published. May be a reference to a version specific, or contained."/> + <min value="0"/> + <max value="1"/> + <base> + <path value="DocumentReference.context.sourcePatientInfo"/> + <min value="0"/> + <max value="1"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Patient"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <mustSupport value="true"/> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.subject"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value=".participation[typeCode="SBJ"].role[typeCode="PAT"]"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.sourcePatientInfo, DocumentEntry.sourcePatientId"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/recordTarget/"/> + </mapping> + </element> + <element id="DocumentReference.context.related"> + <path value="DocumentReference.context.related"/> + <short value="Related identifiers or resources"/> + <definition value="Related identifiers or resources associated with the DocumentReference."/> + <comment value="May be identifiers or resources that caused the DocumentReference or referenced Document to be created."/> + <min value="0"/> + <max value="*"/> + <base> + <path value="DocumentReference.context.related"/> + <min value="0"/> + <max value="*"/> + </base> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Resource"/> + </type> + <constraint> + <key value="ele-1"/> + <severity value="error"/> + <human value="All FHIR elements must have a @value or children"/> + <expression value="hasValue() or (children().count() > id.count())"/> + <xpath value="@value|f:*|h:div"/> + <source value="http://hl7.org/fhir/StructureDefinition/Element"/> + </constraint> + <isModifier value="false"/> + <isSummary value="false"/> + <mapping> + <identity value="fhircomposition"/> + <map value="Composition.event.detail"/> + </mapping> + <mapping> + <identity value="rim"/> + <map value="./outboundRelationship[typeCode="PERT" and isNormalActRelationship()] / target[isNormalAct]"/> + </mapping> + <mapping> + <identity value="xds"/> + <map value="DocumentEntry.referenceIdList"/> + </mapping> + <mapping> + <identity value="cda"/> + <map value="ClinicalDocument/relatedDocument"/> + </mapping> + </element> + </snapshot> + <differential> + <element id="DocumentReference"> + <path value="DocumentReference"/> + </element> + <element id="DocumentReference.modifierExtension"> + <path value="DocumentReference.modifierExtension"/> + <max value="0"/> + </element> + <element id="DocumentReference.masterIdentifier"> + <path value="DocumentReference.masterIdentifier"/> + <min value="1"/> + </element> + <element id="DocumentReference.identifier"> + <path value="DocumentReference.identifier"/> + <max value="0"/> + </element> + <element id="DocumentReference.status"> + <path value="DocumentReference.status"/> + <patternCode value="current"/> + </element> + <element id="DocumentReference.docStatus"> + <path value="DocumentReference.docStatus"/> + <max value="0"/> + </element> + <element id="DocumentReference.type"> + <path value="DocumentReference.type"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.category"> + <path value="DocumentReference.category"/> + <max value="1"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.subject"> + <path value="DocumentReference.subject"/> + <min value="1"/> + <type> + <code value="Reference"/> + <targetProfile value="http://hl7.org/fhir/StructureDefinition/Patient"/> + </type> + </element> + <element id="DocumentReference.date"> + <path value="DocumentReference.date"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.author"> + <path value="DocumentReference.author"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.relatesTo"> + <path value="DocumentReference.relatesTo"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.securityLabel"> + <path value="DocumentReference.securityLabel"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.content"> + <path value="DocumentReference.content"/> + <definition value="The document and format referenced."/> + <max value="1"/> + </element> + <element id="DocumentReference.content.attachment.contentType"> + <path value="DocumentReference.content.attachment.contentType"/> + <min value="1"/> + </element> + <element id="DocumentReference.content.attachment.language"> + <path value="DocumentReference.content.attachment.language"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.content.attachment.data"> + <path value="DocumentReference.content.attachment.data"/> + <short value="The document is contained in the .data element, base64 encoded"/> + <min value="1"/> + </element> + <element id="DocumentReference.content.attachment.url"> + <path value="DocumentReference.content.attachment.url"/> + <max value="0"/> + </element> + <element id="DocumentReference.content.attachment.creation"> + <path value="DocumentReference.content.attachment.creation"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.content.format"> + <path value="DocumentReference.content.format"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.context.period"> + <path value="DocumentReference.context.period"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.context.facilityType"> + <path value="DocumentReference.context.facilityType"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.context.practiceSetting"> + <path value="DocumentReference.context.practiceSetting"/> + <mustSupport value="true"/> + </element> + <element id="DocumentReference.context.sourcePatientInfo"> + <path value="DocumentReference.context.sourcePatientInfo"/> + <mustSupport value="true"/> + </element> + </differential> +</StructureDefinition> \ No newline at end of file diff --git a/commons/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ValidatorTest.java b/commons/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ValidatorTest.java new file mode 100644 index 0000000000..2270c73f0c --- /dev/null +++ b/commons/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/commons/ihe/fhir/iti105/Iti105ValidatorTest.java @@ -0,0 +1,102 @@ +package org.openehealth.ipf.commons.ihe.fhir.iti105; + +import ca.uhn.fhir.context.FhirContext; +import ca.uhn.fhir.rest.server.exceptions.UnprocessableEntityException; +import org.hl7.fhir.r4.model.Attachment; +import org.hl7.fhir.r4.model.Coding; +import org.hl7.fhir.r4.model.DocumentReference; +import org.hl7.fhir.r4.model.Enumerations; +import org.hl7.fhir.r4.model.Identifier; +import org.hl7.fhir.r4.model.Narrative; +import org.hl7.fhir.r4.model.OperationOutcome; +import org.hl7.fhir.r4.model.Practitioner; +import org.hl7.fhir.r4.model.Reference; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; +import java.time.LocalDateTime; +import java.time.ZoneOffset; +import java.util.Date; +import java.util.HashMap; + +import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.openehealth.ipf.commons.ihe.fhir.Constants.URN_IETF_RFC_3986; + +public class Iti105ValidatorTest { + + private static final Logger LOG = LoggerFactory.getLogger(Iti105ValidatorTest.class); + + static Iti105Validator iti105Validator; + + @BeforeAll + static void beforeAll() { + var context = FhirContext.forR4(); + iti105Validator = new Iti105Validator(context); + } + + @Test + void testValidConformance() { + assertDoesNotThrow(() -> iti105Validator.validateRequest(validDocumentreference(), new HashMap<>())); + } + + @Test + void testInvalidConformance() throws Exception { + var documentReference = invalidDocumentreference(); + UnprocessableEntityException exception = assertThrows(UnprocessableEntityException.class, () -> + iti105Validator.validateRequest(documentReference, new HashMap<>()) + ); + assertNotNull(exception); + var oo = (OperationOutcome) exception.getOperationOutcome(); + oo.getIssue().forEach(ooc -> LOG.error(ooc.getSeverity().getDisplay() + " : " + ooc.getDiagnostics())); + } + + private static DocumentReference validDocumentreference() throws NoSuchAlgorithmException { + var reference = invalidDocumentreference(); + reference.getMeta().addProfile(Iti105Validator.ITI105_PROFILE); + return reference; + } + + private static DocumentReference invalidDocumentreference() throws NoSuchAlgorithmException { + var documentContent = "Hello IHE World".getBytes(); + var practitioner = new Practitioner(); + var reference = new DocumentReference(); + var timestamp = Date.from(LocalDateTime.now().toInstant(ZoneOffset.UTC)); + reference.getMeta().setLastUpdated(timestamp); + + reference + .setMasterIdentifier( + new Identifier() + .setSystem(URN_IETF_RFC_3986) + .setValue("urn:oid:129.6.58.92.88336")) + .addIdentifier(new Identifier() + .setSystem(URN_IETF_RFC_3986) + .setValue("urn:oid:129.6.58.92.88336")) + .setDate(timestamp) // creation of document reference resource + .setDescription("Physical") + .setSubject(new Reference("http://server/Patient/a2")) + .addAuthor(new Reference(practitioner)) + .setStatus(Enumerations.DocumentReferenceStatus.CURRENT); + reference.getText().setStatus(Narrative.NarrativeStatus.EMPTY); + reference.getText().setDivAsString("<div>empty</div>"); + reference.getType().addCoding() + .setSystem("http://ihe.net/connectathon/classCodes") + .setCode("History and Physical") + .setDisplay("History and Physical"); + reference.addContent() + .setAttachment( + new Attachment() + .setContentType("text/plain") + .setLanguage("en/us") + .setSize(documentContent.length) + .setHash(MessageDigest.getInstance("SHA-1").digest(documentContent)) + .setUrl("urn:uuid:8da1cfcc-05db-4aca-86ad-82aa756a64bb")) + .setFormat(new Coding("urn:oid:1.3.6.1.4.1.19376.1.2.3", "urn:ihe:pcc:handp:2008", null)); + return reference; + } +} diff --git a/platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Component.java b/platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Component.java new file mode 100644 index 0000000000..988f066f6b --- /dev/null +++ b/platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Component.java @@ -0,0 +1,49 @@ +/* + * Copyright 2016 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.platform.camel.ihe.fhir.iti105; + +import org.apache.camel.CamelContext; +import org.openehealth.ipf.commons.ihe.fhir.iti105.Iti105AuditDataset; +import org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirComponent; +import org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirEndpointConfiguration; + +import static org.openehealth.ipf.commons.ihe.fhir.mhd.MHD.SimplifiedPublishInteractions.ITI_105; + +/** + * Component for MHD Simplified Publish (ITI-105) + * + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105Component extends FhirComponent<Iti105AuditDataset> { + + + public Iti105Component() { + super(ITI_105); + } + + public Iti105Component(CamelContext context) { + super(context, ITI_105); + } + + @Override + protected Iti105Endpoint doCreateEndpoint(String uri, FhirEndpointConfiguration<Iti105AuditDataset> config) { + return new Iti105Endpoint(uri, this, config); + } + + +} diff --git a/platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Endpoint.java b/platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Endpoint.java new file mode 100644 index 0000000000..95d1129395 --- /dev/null +++ b/platform-camel/ihe/fhir/r4/mhd/src/main/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105Endpoint.java @@ -0,0 +1,37 @@ +/* + * Copyright 2016 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.platform.camel.ihe.fhir.iti105; + +import org.openehealth.ipf.commons.ihe.fhir.iti105.Iti105AuditDataset; +import org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirEndpoint; +import org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirEndpointConfiguration; + +/** + * @author Boris Stanojevic + * @since 4.8 + */ +public class Iti105Endpoint extends FhirEndpoint<Iti105AuditDataset, Iti105Component> { + + public Iti105Endpoint(String uri, Iti105Component fhirComponent, FhirEndpointConfiguration<Iti105AuditDataset> config) { + super(uri, fhirComponent, config); + } + + @Override + protected String createEndpointUri() { + return "mhd-iti65:" + "not-implemented yet"; + } +} diff --git a/platform-camel/ihe/fhir/r4/mhd/src/main/resources/META-INF/services/org/apache/camel/component/mhd-iti105 b/platform-camel/ihe/fhir/r4/mhd/src/main/resources/META-INF/services/org/apache/camel/component/mhd-iti105 new file mode 100644 index 0000000000..bb3eb4e336 --- /dev/null +++ b/platform-camel/ihe/fhir/r4/mhd/src/main/resources/META-INF/services/org/apache/camel/component/mhd-iti105 @@ -0,0 +1,12 @@ +# Copyright 2023 the original author or authors. Licensed under the Apache +# License, Version 2.0 (the "License"); you may not use this file except +# in compliance with the License. You may obtain a copy of the License at +# http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable +# law or agreed to in writing, software distributed under the License is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +# Camel registration for the mhd-iti65 component + +class=org.openehealth.ipf.platform.camel.ihe.fhir.iti105.Iti105Component diff --git a/platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105TestRouteBuilder.java b/platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105TestRouteBuilder.java new file mode 100644 index 0000000000..9fa10f1456 --- /dev/null +++ b/platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/Iti105TestRouteBuilder.java @@ -0,0 +1,81 @@ +/* + * Copyright 2016 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.platform.camel.ihe.fhir.iti105; + +import ca.uhn.fhir.rest.api.MethodOutcome; +import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; +import org.apache.camel.Exchange; +import org.apache.camel.builder.RouteBuilder; +import org.apache.camel.component.http.HttpConstants; +import org.apache.camel.support.ExpressionAdapter; +import org.hl7.fhir.r4.model.IdType; +import org.hl7.fhir.r4.model.OperationOutcome; +import org.openehealth.ipf.commons.ihe.fhir.Constants; +import org.openehealth.ipf.platform.camel.core.adapter.ValidatorAdapter; +import org.openehealth.ipf.platform.camel.ihe.fhir.test.FhirTestContainer; + +import java.util.UUID; + +import static org.apache.camel.component.http.HttpMethods.POST; +import static org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirCamelValidators.SCHEMA; +import static org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirCamelValidators.SCHEMATRON; +import static org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirCamelValidators.VALIDATION_MODE; +import static org.openehealth.ipf.platform.camel.ihe.fhir.core.FhirCamelValidators.itiRequestValidator; + +/** + * + */ +public class Iti105TestRouteBuilder extends RouteBuilder { + + private final boolean returnError; + + public Iti105TestRouteBuilder(boolean returnError) { + this.returnError = returnError; + } + + @Override + public void configure() { + + from("direct:input") + .setHeader(Constants.HTTP_METHOD, POST) + .toF("mhd-iti105:localhost:%d", FhirTestContainer.DEMO_APP_PORT); + + from("mhd-iti105:stub?audit=true&fhirContext=#fhirContext") + .errorHandler(noErrorHandler()) + .setHeader(ValidatorAdapter.NEED_VALIDATION_HEADER_NAME, constant(true)) + .setHeader(VALIDATION_MODE, constant(SCHEMA | SCHEMATRON )) // | MODEL)) + .process(itiRequestValidator()) + .transform(new Responder()); + } + + + private class Responder extends ExpressionAdapter { + + @Override + public Object evaluate(Exchange exchange) { + + if (returnError) throw new InternalErrorException("Something went wrong"); + + var methodOutcome = new MethodOutcome(new IdType(UUID.randomUUID().toString()), true); + var outcome = new OperationOutcome(); + outcome.setId(new IdType(UUID.randomUUID().toString())); + methodOutcome.setOperationOutcome(outcome); + return methodOutcome; + } + + } +} diff --git a/platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/TestIti105Success.java b/platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/TestIti105Success.java new file mode 100644 index 0000000000..969dda5526 --- /dev/null +++ b/platform-camel/ihe/fhir/r4/mhd/src/test/java/org/openehealth/ipf/platform/camel/ihe/fhir/iti105/TestIti105Success.java @@ -0,0 +1,199 @@ +/* + * Copyright 2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.openehealth.ipf.platform.camel.ihe.fhir.iti105; + +import ca.uhn.fhir.context.FhirVersionEnum; +import ca.uhn.fhir.rest.api.MethodOutcome; +import ca.uhn.fhir.rest.client.interceptor.LoggingInterceptor; +import org.hl7.fhir.r4.model.*; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.Test; +import org.openehealth.ipf.commons.audit.codes.EventActionCode; +import org.openehealth.ipf.commons.audit.codes.EventIdCode; +import org.openehealth.ipf.commons.audit.codes.EventOutcomeIndicator; +import org.openehealth.ipf.commons.audit.codes.NetworkAccessPointTypeCode; +import org.openehealth.ipf.commons.audit.codes.ParticipantObjectIdTypeCode; +import org.openehealth.ipf.commons.audit.codes.ParticipantObjectTypeCode; +import org.openehealth.ipf.commons.audit.codes.ParticipantObjectTypeCodeRole; +import org.openehealth.ipf.commons.audit.utils.AuditUtils; +import org.openehealth.ipf.commons.ihe.fhir.IpfFhirServlet; +import org.openehealth.ipf.commons.ihe.fhir.SslAwareMethanolRestfulClientFactory; +import org.openehealth.ipf.commons.ihe.fhir.audit.codes.FhirEventTypeCode; +import org.openehealth.ipf.platform.camel.ihe.fhir.test.FhirTestContainer; + +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; +import java.time.LocalDateTime; +import java.time.ZoneOffset; +import java.util.Collections; +import java.util.Date; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertTrue; +import static org.openehealth.ipf.commons.ihe.fhir.Constants.URN_IETF_RFC_3986; +import static org.openehealth.ipf.commons.ihe.fhir.iti105.Iti105Validator.ITI105_PROFILE; + +/** + * + */ +public class TestIti105Success extends FhirTestContainer { + + private static final String CONTEXT_DESCRIPTOR = "iti-105.xml"; + + private static final String RESOURCE_NAME = ResourceType.DocumentReference.name(); + + @BeforeAll + public static void setUpClass() { + startServer(CONTEXT_DESCRIPTOR); + } + + public static void startServer(String contextDescriptor) { + var servlet = new IpfFhirServlet(FhirVersionEnum.R4); + startServer(servlet, contextDescriptor, false, DEMO_APP_PORT, "FhirServlet"); + + var loggingInterceptor = new LoggingInterceptor(); + loggingInterceptor.setLogRequestSummary(false); + loggingInterceptor.setLogRequestBody(true); + loggingInterceptor.setLogResponseBody(true); + startClient(String.format("http://localhost:%d/", DEMO_APP_PORT), fhirContext -> { + var clientFactory = new SslAwareMethanolRestfulClientFactory(fhirContext); + clientFactory.setAsync(true); + fhirContext.setRestfulClientFactory(clientFactory); + }).registerInterceptor(loggingInterceptor); + } + + @Test + public void testGetConformance() { + var conf = client.capabilities().ofType(CapabilityStatement.class).execute(); + var component = conf.getRest().iterator().next(); + assertTrue(component.getResource().stream().anyMatch(r -> r.getType().equals(RESOURCE_NAME))); + } + + @Test + public void testSendRemoteSimplifiedPublish() throws Exception { + var result = client.create() + .resource(documentReference()) + .encodedXml() + .execute(); + assertNotNull(result); + assertTrue(result.getCreated()); + + // Check ATNA Audit + var sender = getAuditSender(); + assertEquals(1, sender.getMessages().size()); + var event = sender.getMessages().get(0); + + // Event + assertEquals( + EventOutcomeIndicator.Success, + event.getEventIdentification().getEventOutcomeIndicator()); + assertEquals( + EventActionCode.Create, + event.getEventIdentification().getEventActionCode()); + + assertEquals(EventIdCode.Import, event.getEventIdentification().getEventID()); + assertEquals(FhirEventTypeCode.SimplifiedPublish, event.getEventIdentification().getEventTypeCode().get(0)); + + // ActiveParticipant Source + var source = event.getActiveParticipants().get(0); + assertTrue(source.isUserIsRequestor()); + assertEquals("127.0.0.1", source.getNetworkAccessPointID()); + assertEquals(NetworkAccessPointTypeCode.IPAddress, source.getNetworkAccessPointTypeCode()); + + // ActiveParticipant Destination + var destination = event.getActiveParticipants().get(1); + assertFalse(destination.isUserIsRequestor()); + assertEquals("http://localhost:" + DEMO_APP_PORT + "/" + RESOURCE_NAME, destination.getUserID()); + assertEquals(AuditUtils.getLocalIPAddress(), destination.getNetworkAccessPointID()); + + // Patient + var patient = event.getParticipantObjectIdentifications().get(0); + assertEquals(ParticipantObjectTypeCode.Person, patient.getParticipantObjectTypeCode()); + assertEquals(ParticipantObjectTypeCodeRole.Patient, patient.getParticipantObjectTypeCodeRole()); + assertEquals(ParticipantObjectIdTypeCode.PatientNumber, patient.getParticipantObjectIDTypeCode()); + assertEquals("Patient/123", patient.getParticipantObjectID()); + + // SubmissionSet + var poit = event.getParticipantObjectIdentifications().get(1); + assertEquals(ParticipantObjectTypeCode.System, poit.getParticipantObjectTypeCode()); + assertEquals(ParticipantObjectTypeCodeRole.Job, poit.getParticipantObjectTypeCodeRole()); + + // No real instructions how this should look like, so for now we take the XDS stuff + var poitTypeCode = poit.getParticipantObjectIDTypeCode(); + assertEquals("urn:uuid:a54d6aa5-d40d-43f9-88c5-b4633d873bdd", poitTypeCode.getCode()); + assertEquals("IHE XDS Metadata", poitTypeCode.getCodeSystemName()); + } + + @Test + public void testSendOverDirectEndpointSimplifiedPublish() throws Exception { + producerTemplate.requestBody("direct:input", documentReference(), MethodOutcome.class); + var sender = getAuditSender(); + assertEquals(2, sender.getMessages().size()); + } + + private static DocumentReference documentReference() throws NoSuchAlgorithmException { + var documentContent = "Hello IHE World".getBytes(); + var practitioner = new Practitioner(); + practitioner.setId("987"); + practitioner.addName(new HumanName() + .setFamily("Soo") + .setGiven(Collections.singletonList(new StringType("Yun")))); + + var patient = new Patient(); + patient.setId("123"); + patient.getName().add(new HumanName() + .setFamily("Foo") + .setGiven(Collections.singletonList(new StringType("Barbara")))); + var reference = new DocumentReference(); + reference.getMeta().addProfile(ITI105_PROFILE); + var timestamp = Date.from(LocalDateTime.now().toInstant(ZoneOffset.UTC)); + reference.getMeta().setLastUpdated(timestamp); + + reference + .setMasterIdentifier( + new Identifier() + .setSystem(URN_IETF_RFC_3986) + .setValue("urn:oid:129.6.58.92.88336")) + .addIdentifier(new Identifier() + .setSystem(URN_IETF_RFC_3986) + .setValue("urn:oid:129.6.58.92.88336")) + .setDate(timestamp) // creation of document reference resource + .setDescription("Physical") + .setSubject(new Reference(patient)) + .addAuthor(new Reference(practitioner)) + .setStatus(Enumerations.DocumentReferenceStatus.CURRENT); + reference.getText().setStatus(Narrative.NarrativeStatus.EMPTY); + reference.getText().setDivAsString("<div>empty</div>"); + reference.getType().addCoding() + .setSystem("http://ihe.net/connectathon/classCodes") + .setCode("History and Physical") + .setDisplay("History and Physical"); + reference.addContent() + .setAttachment( + new Attachment() + .setContentType("text/plain") + .setLanguage("en/us") + .setSize(documentContent.length) + .setHash(MessageDigest.getInstance("SHA-1").digest(documentContent)) + .setUrl("urn:uuid:8da1cfcc-05db-4aca-86ad-82aa756a64bb")) + .setFormat(new Coding("urn:oid:1.3.6.1.4.1.19376.1.2.3", "urn:ihe:pcc:handp:2008", null)); + return reference; + } + +} diff --git a/platform-camel/ihe/fhir/r4/mhd/src/test/resources/iti-105.xml b/platform-camel/ihe/fhir/r4/mhd/src/test/resources/iti-105.xml new file mode 100644 index 0000000000..d0cfc7db3b --- /dev/null +++ b/platform-camel/ihe/fhir/r4/mhd/src/test/resources/iti-105.xml @@ -0,0 +1,30 @@ +<!-- + ~ Copyright 2023 the original author or authors. + ~ + ~ Licensed under the Apache License, Version 2.0 (the "License"); + ~ you may not use this file except in compliance with the License. + ~ You may obtain a copy of the License at + ~ + ~ http://www.apache.org/licenses/LICENSE-2.0 + ~ + ~ Unless required by applicable law or agreed to in writing, software + ~ distributed under the License is distributed on an "AS IS" BASIS, + ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + ~ See the License for the specific language governing permissions and + ~ limitations under the License. + --> + +<beans xmlns="http://www.springframework.org/schema/beans" + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xsi:schemaLocation=" +http://www.springframework.org/schema/beans +http://www.springframework.org/schema/beans/spring-beans.xsd"> + + <import resource="common-fhir-beans.xml"/> + + <bean id="routeBuilder" + class="org.openehealth.ipf.platform.camel.ihe.fhir.iti105.Iti105TestRouteBuilder"> + <constructor-arg value="false"/> + </bean> + +</beans> \ No newline at end of file