001/*- 002 * #%L 003 * HAPI FHIR - Core Library 004 * %% 005 * Copyright (C) 2014 - 2025 Smile CDR, Inc. 006 * %% 007 * Licensed under the Apache License, Version 2.0 (the "License"); 008 * you may not use this file except in compliance with the License. 009 * You may obtain a copy of the License at 010 * 011 * http://www.apache.org/licenses/LICENSE-2.0 012 * 013 * Unless required by applicable law or agreed to in writing, software 014 * distributed under the License is distributed on an "AS IS" BASIS, 015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 * #L% 019 */ 020package ca.uhn.fhir.util; 021 022import ca.uhn.fhir.context.BaseRuntimeChildDefinition; 023import ca.uhn.fhir.context.BaseRuntimeElementCompositeDefinition; 024import ca.uhn.fhir.context.FhirContext; 025import ca.uhn.fhir.context.FhirVersionEnum; 026import ca.uhn.fhir.i18n.Msg; 027import org.apache.commons.lang3.StringUtils; 028import org.hl7.fhir.instance.model.api.IBase; 029import org.hl7.fhir.instance.model.api.IBaseExtension; 030import org.hl7.fhir.instance.model.api.IBaseHasExtensions; 031import org.hl7.fhir.instance.model.api.IBaseMetaType; 032import org.hl7.fhir.instance.model.api.IBaseResource; 033import org.hl7.fhir.instance.model.api.IPrimitiveType; 034import org.slf4j.Logger; 035import org.slf4j.LoggerFactory; 036 037import java.util.List; 038import java.util.Set; 039import java.util.stream.Collectors; 040 041import static org.apache.commons.lang3.StringUtils.defaultString; 042import static org.apache.commons.lang3.StringUtils.isNotBlank; 043 044public class MetaUtil { 045 private static final Logger ourLog = LoggerFactory.getLogger(MetaUtil.class); 046 047 private MetaUtil() { 048 // non-instantiable 049 } 050 051 public static String cleanProvenanceSourceUriOrEmpty(String theProvenanceSourceUri) { 052 String sanitizedProvenance = defaultString(theProvenanceSourceUri); 053 return StringUtils.substringBefore(sanitizedProvenance, "#"); 054 } 055 056 public static String getSource(FhirContext theContext, IBaseMetaType theMeta) { 057 if (theContext.getVersion().getVersion().isEqualOrNewerThan(FhirVersionEnum.R4)) { 058 return getSourceR4Plus(theContext, theMeta); 059 } else if (theContext.getVersion().getVersion().equals(FhirVersionEnum.DSTU3)) { 060 return getSourceDstu3((IBaseHasExtensions) theMeta); 061 } else { 062 throw new UnsupportedOperationException( 063 Msg.code(1782) + MetaUtil.class.getSimpleName() + ".getSource() not supported on FHIR Version " 064 + theContext.getVersion().getVersion()); 065 } 066 } 067 068 private static String getSourceDstu3(IBaseHasExtensions theMeta) { 069 IBaseHasExtensions metaWithExtensions = theMeta; 070 List<? extends IBaseExtension<?, ?>> extensions = metaWithExtensions.getExtension(); 071 for (IBaseExtension extension : extensions) { 072 if (HapiExtensions.EXT_META_SOURCE.equals(extension.getUrl())) { 073 IPrimitiveType<String> value = (IPrimitiveType<String>) extension.getValue(); 074 return value.getValueAsString(); 075 } 076 } 077 return null; 078 } 079 080 private static String getSourceR4Plus(FhirContext theFhirContext, IBaseMetaType theMeta) { 081 BaseRuntimeElementCompositeDefinition<?> elementDef = 082 (BaseRuntimeElementCompositeDefinition<?>) theFhirContext.getElementDefinition(theMeta.getClass()); 083 BaseRuntimeChildDefinition sourceChild = elementDef.getChildByName("source"); 084 if (sourceChild == null) { 085 return null; 086 } 087 List<IBase> sourceValues = sourceChild.getAccessor().getValues(theMeta); 088 String retVal = null; 089 if (sourceValues.size() > 0) { 090 retVal = ((IPrimitiveType<?>) sourceValues.get(0)).getValueAsString(); 091 } 092 return retVal; 093 } 094 095 public static <R extends IBaseResource> void populateResourceSource( 096 FhirContext theFhirContext, String theProvenanceSourceUri, String theProvenanceRequestId, R theRetVal) { 097 String sourceString = cleanProvenanceSourceUriOrEmpty(theProvenanceSourceUri); 098 if (isNotBlank(theProvenanceRequestId)) { 099 sourceString = sourceString + "#" + theProvenanceRequestId; 100 } 101 102 if (isNotBlank(sourceString)) { 103 setSource(theFhirContext, theRetVal, sourceString); 104 } 105 } 106 107 /** 108 * Sets the value for <code>Resource.meta.source</code> for R4+ resources, and places the value in 109 * an extension on <code>Resource.meta</code> 110 * with the URL <code>http://hapifhir.io/fhir/StructureDefinition/resource-meta-source</code> for DSTU3. 111 * 112 * @param theContext The FhirContext object 113 * @param theResource The resource to modify 114 * @param theValue The source URI 115 * @see <a href="http://hl7.org/fhir/resource-definitions.html#Resource.meta">Meta.source</a> 116 */ 117 @SuppressWarnings("unchecked") 118 public static void setSource(FhirContext theContext, IBaseResource theResource, String theValue) { 119 if (theContext.getVersion().getVersion().isEqualOrNewerThan(FhirVersionEnum.R4)) { 120 MetaUtil.setSource(theContext, theResource.getMeta(), theValue); 121 } else if (theContext.getVersion().getVersion().equals(FhirVersionEnum.DSTU3)) { 122 IBaseExtension<?, ?> sourceExtension = ((IBaseHasExtensions) theResource.getMeta()).addExtension(); 123 sourceExtension.setUrl(HapiExtensions.EXT_META_SOURCE); 124 IPrimitiveType<String> value = (IPrimitiveType<String>) 125 theContext.getElementDefinition("uri").newInstance(); 126 value.setValue(theValue); 127 sourceExtension.setValue(value); 128 } else { 129 ourLog.debug(MetaUtil.class.getSimpleName() + ".setSource() not supported on FHIR Version " 130 + theContext.getVersion().getVersion()); 131 } 132 } 133 134 public static void setSource(FhirContext theContext, IBaseMetaType theMeta, String theValue) { 135 BaseRuntimeElementCompositeDefinition<?> elementDef = 136 (BaseRuntimeElementCompositeDefinition<?>) theContext.getElementDefinition(theMeta.getClass()); 137 BaseRuntimeChildDefinition sourceChild = elementDef.getChildByName("source"); 138 List<IBase> sourceValues = sourceChild.getAccessor().getValues(theMeta); 139 IPrimitiveType<?> sourceElement; 140 if (sourceValues.size() > 0) { 141 sourceElement = ((IPrimitiveType<?>) sourceValues.get(0)); 142 } else { 143 sourceElement = 144 (IPrimitiveType<?>) theContext.getElementDefinition("uri").newInstance(); 145 sourceChild.getMutator().setValue(theMeta, sourceElement); 146 } 147 sourceElement.setValueAsString(theValue); 148 } 149 150 public static Set<String> getAutoVersionReferencesAtPath(IBaseMetaType theMeta, String theResourceType) { 151 return ExtensionUtil.getExtensionPrimitiveValues( 152 theMeta, HapiExtensions.EXTENSION_AUTO_VERSION_REFERENCES_AT_PATH) 153 .stream() 154 .map(path -> String.format("%s.%s", theResourceType, path)) 155 .collect(Collectors.toSet()); 156 } 157}