001/*- 002 * #%L 003 * HAPI FHIR Storage api 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.jpa.api.dao; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.context.RuntimeResourceDefinition; 024import ca.uhn.fhir.i18n.Msg; 025import ca.uhn.fhir.jpa.api.IDaoRegistry; 026import ca.uhn.fhir.model.dstu2.valueset.ResourceTypeEnum; 027import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 028import jakarta.annotation.Nullable; 029import org.apache.commons.lang3.Validate; 030import org.hl7.fhir.instance.model.api.IBaseResource; 031import org.springframework.beans.BeansException; 032import org.springframework.beans.factory.annotation.Autowired; 033import org.springframework.context.ApplicationContext; 034import org.springframework.context.ApplicationContextAware; 035 036import java.util.Arrays; 037import java.util.Collection; 038import java.util.Collections; 039import java.util.HashMap; 040import java.util.HashSet; 041import java.util.List; 042import java.util.Map; 043import java.util.Set; 044import java.util.stream.Collectors; 045 046public class DaoRegistry implements ApplicationContextAware, IDaoRegistry { 047 private ApplicationContext myAppCtx; 048 049 @Autowired 050 private FhirContext myFhirContext; 051 052 private volatile Map<String, IFhirResourceDao<?>> myResourceNameToResourceDao; 053 private volatile IFhirSystemDao<?, ?> mySystemDao; 054 private Set<String> mySupportedResourceTypes; 055 056 /** 057 * Constructor 058 */ 059 public DaoRegistry() { 060 this(null); 061 } 062 063 /** 064 * Constructor 065 */ 066 public DaoRegistry(FhirContext theFhirContext) { 067 super(); 068 myFhirContext = theFhirContext; 069 } 070 071 public void setSupportedResourceTypes(Collection<String> theSupportedResourceTypes) { 072 HashSet<String> supportedResourceTypes = new HashSet<>(); 073 if (theSupportedResourceTypes != null) { 074 supportedResourceTypes.addAll(theSupportedResourceTypes); 075 } 076 mySupportedResourceTypes = supportedResourceTypes; 077 myResourceNameToResourceDao = null; 078 } 079 080 @Override 081 public void setApplicationContext(ApplicationContext theApplicationContext) throws BeansException { 082 myAppCtx = theApplicationContext; 083 } 084 085 public IFhirSystemDao getSystemDao() { 086 IFhirSystemDao retVal = mySystemDao; 087 if (retVal == null) { 088 retVal = myAppCtx.getBean(IFhirSystemDao.class); 089 mySystemDao = retVal; 090 } 091 return retVal; 092 } 093 094 /** 095 * @throws InvalidRequestException If the given resource type is not supported 096 */ 097 public IFhirResourceDao getResourceDao(String theResourceName) { 098 IFhirResourceDao<IBaseResource> retVal = getResourceDaoOrNull(theResourceName); 099 if (retVal == null) { 100 List<String> supportedResourceTypes = 101 myResourceNameToResourceDao.keySet().stream().sorted().collect(Collectors.toList()); 102 throw new InvalidRequestException(Msg.code(572) 103 + "Unable to process request, this server does not know how to handle resources of type " 104 + theResourceName + " - Can handle: " + supportedResourceTypes); 105 } 106 return retVal; 107 } 108 109 @SuppressWarnings("unchecked") 110 public <R extends IBaseResource> IFhirResourceDao<R> getResourceDao(R theResource) { 111 return (IFhirResourceDao<R>) getResourceDao(theResource.getClass()); 112 } 113 114 public <R extends IBaseResource> IFhirResourceDao<R> getResourceDao(Class<R> theResourceType) { 115 IFhirResourceDao<R> retVal = getResourceDaoIfExists(theResourceType); 116 Validate.notNull( 117 retVal, "No DAO exists for resource type %s - Have: %s", theResourceType, myResourceNameToResourceDao); 118 return retVal; 119 } 120 121 /** 122 * Use getResourceDaoOrNull 123 */ 124 @Deprecated 125 public <T extends IBaseResource> IFhirResourceDao<T> getResourceDaoIfExists(Class<T> theResourceType) { 126 return getResourceDaoOrNull(theResourceType); 127 } 128 129 @Nullable 130 public <T extends IBaseResource> IFhirResourceDao<T> getResourceDaoOrNull(Class<T> theResourceType) { 131 String resourceName = myFhirContext.getResourceType(theResourceType); 132 try { 133 return (IFhirResourceDao<T>) getResourceDao(resourceName); 134 } catch (InvalidRequestException e) { 135 return null; 136 } 137 } 138 139 /** 140 * Use getResourceDaoOrNull 141 */ 142 @Deprecated 143 public <T extends IBaseResource> IFhirResourceDao<T> getResourceDaoIfExists(String theResourceType) { 144 return getResourceDaoOrNull(theResourceType); 145 } 146 147 @Nullable 148 public <T extends IBaseResource> IFhirResourceDao<T> getResourceDaoOrNull(String theResourceName) { 149 init(); 150 return (IFhirResourceDao<T>) myResourceNameToResourceDao.get(theResourceName); 151 } 152 153 @Override 154 public boolean isResourceTypeSupported(String theResourceType) { 155 if (mySupportedResourceTypes == null) { 156 return getResourceDaoOrNull(theResourceType) != null; 157 } 158 return mySupportedResourceTypes.contains(theResourceType); 159 } 160 161 private void init() { 162 if (myResourceNameToResourceDao != null && !myResourceNameToResourceDao.isEmpty()) { 163 return; 164 } 165 166 Map<String, IFhirResourceDao> resourceDaos = myAppCtx.getBeansOfType(IFhirResourceDao.class); 167 168 initializeMaps(resourceDaos.values()); 169 } 170 171 private void initializeMaps(Collection<IFhirResourceDao> theResourceDaos) { 172 173 myResourceNameToResourceDao = new HashMap<>(); 174 175 for (IFhirResourceDao nextResourceDao : theResourceDaos) { 176 Class resourceType = nextResourceDao.getResourceType(); 177 assert resourceType != null; 178 RuntimeResourceDefinition nextResourceDef = myFhirContext.getResourceDefinition(resourceType); 179 if (mySupportedResourceTypes == null || mySupportedResourceTypes.contains(nextResourceDef.getName())) { 180 myResourceNameToResourceDao.put(nextResourceDef.getName(), nextResourceDao); 181 } 182 } 183 } 184 185 public void register(IFhirResourceDao theResourceDao) { 186 RuntimeResourceDefinition resourceDef = myFhirContext.getResourceDefinition(theResourceDao.getResourceType()); 187 String resourceName = resourceDef.getName(); 188 myResourceNameToResourceDao.put(resourceName, theResourceDao); 189 } 190 191 public IFhirResourceDao getDaoOrThrowException(Class<? extends IBaseResource> theClass) { 192 IFhirResourceDao retVal = getResourceDao(theClass); 193 if (retVal == null) { 194 List<String> supportedResourceNames = myResourceNameToResourceDao.keySet().stream() 195 .map(t -> myFhirContext.getResourceType(t)) 196 .sorted() 197 .collect(Collectors.toList()); 198 throw new InvalidRequestException(Msg.code(573) 199 + "Unable to process request, this server does not know how to handle resources of type " 200 + myFhirContext.getResourceType(theClass) + " - Can handle: " + supportedResourceNames); 201 } 202 return retVal; 203 } 204 205 public void setResourceDaos(Collection<IFhirResourceDao> theResourceDaos) { 206 initializeMaps(theResourceDaos); 207 } 208 209 public IFhirResourceDao getSubscriptionDao() { 210 return getResourceDao(ResourceTypeEnum.SUBSCRIPTION.getCode()); 211 } 212 213 public void setSupportedResourceTypes(String... theResourceTypes) { 214 setSupportedResourceTypes(toCollection(theResourceTypes)); 215 } 216 217 private List<String> toCollection(String[] theResourceTypes) { 218 List<String> retVal = null; 219 if (theResourceTypes != null && theResourceTypes.length > 0) { 220 retVal = Arrays.asList(theResourceTypes); 221 } 222 return retVal; 223 } 224 225 public Set<String> getRegisteredDaoTypes() { 226 return Collections.unmodifiableSet(myResourceNameToResourceDao.keySet()); 227 } 228 229 // TODO KHS find all the places where FhirContext and DaoRegistry are both passed into constructors and 230 // remove the FhirContext parameter and pull it from the DaoRegistry parameter 231 public FhirContext getFhirContext() { 232 return myFhirContext; 233 } 234}