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.context.support; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.context.FhirVersionEnum; 024import ca.uhn.fhir.util.ILockable; 025import ca.uhn.fhir.util.ReflectionUtil; 026import jakarta.annotation.Nonnull; 027import jakarta.annotation.Nullable; 028import org.apache.commons.lang3.Validate; 029import org.hl7.fhir.instance.model.api.IBase; 030import org.hl7.fhir.instance.model.api.IBaseResource; 031import org.hl7.fhir.instance.model.api.IPrimitiveType; 032 033import java.util.Collections; 034import java.util.HashMap; 035import java.util.List; 036import java.util.Map; 037import java.util.Optional; 038 039/** 040 * This class returns the vocabulary that is shipped with the base FHIR 041 * specification. 042 * 043 * Note that this class is version aware. For example, a request for 044 * <code>http://foo-codesystem|123</code> will only return a value if 045 * the built in resource if the version matches. Unversioned URLs 046 * should generally be used, and will return whatever version is 047 * present. 048 */ 049public class DefaultProfileValidationSupport implements IValidationSupport { 050 051 private static final Map<FhirVersionEnum, IValidationSupport> ourImplementations = 052 Collections.synchronizedMap(new HashMap<>()); 053 054 /** 055 * Userdata key indicating the source package ID for this package 056 */ 057 public static final String SOURCE_PACKAGE_ID = 058 DefaultProfileValidationSupport.class.getName() + "_SOURCE_PACKAGE_ID"; 059 060 private final FhirContext myCtx; 061 /** 062 * This module just delegates all calls to a concrete implementation which will 063 * be in this field. Which implementation gets used depends on the FHIR version. 064 */ 065 private final IValidationSupport myDelegate; 066 067 private final Runnable myFlush; 068 069 /** 070 * Constructor 071 * 072 * @param theFhirContext The context to use 073 */ 074 public DefaultProfileValidationSupport(@Nonnull FhirContext theFhirContext) { 075 Validate.notNull(theFhirContext, "FhirContext must not be null"); 076 myCtx = theFhirContext; 077 078 IValidationSupport strategy; 079 synchronized (ourImplementations) { 080 strategy = ourImplementations.get(theFhirContext.getVersion().getVersion()); 081 082 if (strategy == null) { 083 if (theFhirContext.getVersion().getVersion().isEqualOrNewerThan(FhirVersionEnum.R5)) { 084 /* 085 * I don't love that we use reflection here, but this class is in 086 * hapi-fhir-base, and the class we're creating is in 087 * hapi-fhir-validation. There are complicated dependency chains that 088 * make this hard to clean up. At some point it'd be nice to figure out 089 * a cleaner solution though. 090 */ 091 strategy = ReflectionUtil.newInstance( 092 "org.hl7.fhir.common.hapi.validation.support.DefaultProfileValidationSupportNpmStrategy", 093 IValidationSupport.class, 094 new Class[] {FhirContext.class}, 095 new Object[] {theFhirContext}); 096 ((ILockable) strategy).lock(); 097 } else { 098 strategy = new DefaultProfileValidationSupportBundleStrategy(theFhirContext); 099 } 100 ourImplementations.put(theFhirContext.getVersion().getVersion(), strategy); 101 } 102 } 103 104 myDelegate = strategy; 105 if (myDelegate instanceof DefaultProfileValidationSupportBundleStrategy) { 106 myFlush = () -> ((DefaultProfileValidationSupportBundleStrategy) myDelegate).flush(); 107 } else { 108 myFlush = () -> {}; 109 } 110 } 111 112 @Override 113 public String getName() { 114 return myCtx.getVersion().getVersion() + " FHIR Standard Profile Validation Support"; 115 } 116 117 @Override 118 public List<IBaseResource> fetchAllConformanceResources() { 119 List<IBaseResource> retVal = myDelegate.fetchAllConformanceResources(); 120 addPackageInformation(retVal); 121 return retVal; 122 } 123 124 @Override 125 public <T extends IBaseResource> List<T> fetchAllStructureDefinitions() { 126 List<T> retVal = myDelegate.fetchAllStructureDefinitions(); 127 addPackageInformation(retVal); 128 return retVal; 129 } 130 131 @Nullable 132 @Override 133 public <T extends IBaseResource> List<T> fetchAllNonBaseStructureDefinitions() { 134 List<T> retVal = myDelegate.fetchAllNonBaseStructureDefinitions(); 135 addPackageInformation(retVal); 136 return retVal; 137 } 138 139 @Override 140 public IBaseResource fetchCodeSystem(String theSystem) { 141 IBaseResource retVal = myDelegate.fetchCodeSystem(theSystem); 142 addPackageInformation(retVal); 143 return retVal; 144 } 145 146 @Override 147 public IBaseResource fetchStructureDefinition(String theUrl) { 148 IBaseResource retVal = myDelegate.fetchStructureDefinition(theUrl); 149 addPackageInformation(retVal); 150 return retVal; 151 } 152 153 @Override 154 public IBaseResource fetchValueSet(String theUrl) { 155 IBaseResource retVal = myDelegate.fetchValueSet(theUrl); 156 addPackageInformation(retVal); 157 return retVal; 158 } 159 160 public void flush() { 161 myFlush.run(); 162 } 163 164 @Override 165 public FhirContext getFhirContext() { 166 return myCtx; 167 } 168 169 @Nullable 170 public static String getConformanceResourceUrl(FhirContext theFhirContext, IBaseResource theResource) { 171 String urlValueString = null; 172 Optional<IBase> urlValue = theFhirContext 173 .getResourceDefinition(theResource) 174 .getChildByName("url") 175 .getAccessor() 176 .getFirstValueOrNull(theResource); 177 if (urlValue.isPresent()) { 178 IPrimitiveType<?> urlValueType = (IPrimitiveType<?>) urlValue.get(); 179 urlValueString = urlValueType.getValueAsString(); 180 } 181 return urlValueString; 182 } 183 184 private <T extends IBaseResource> void addPackageInformation(List<T> theResources) { 185 if (theResources != null) { 186 theResources.forEach(this::addPackageInformation); 187 } 188 } 189 190 private void addPackageInformation(IBaseResource theResource) { 191 if (theResource != null) { 192 String sourcePackageId = null; 193 switch (myCtx.getVersion().getVersion()) { 194 case DSTU2: 195 case DSTU2_HL7ORG: 196 sourcePackageId = "hl7.fhir.r2.core"; 197 break; 198 case DSTU2_1: 199 return; 200 case DSTU3: 201 sourcePackageId = "hl7.fhir.r3.core"; 202 break; 203 case R4: 204 sourcePackageId = "hl7.fhir.r4.core"; 205 break; 206 case R4B: 207 sourcePackageId = "hl7.fhir.r4b.core"; 208 break; 209 case R5: 210 sourcePackageId = "hl7.fhir.r5.core"; 211 break; 212 } 213 214 Validate.notNull( 215 sourcePackageId, 216 "Don't know how to handle package ID: %s", 217 myCtx.getVersion().getVersion()); 218 219 theResource.setUserData(SOURCE_PACKAGE_ID, sourcePackageId); 220 } 221 } 222}