001/*-
002 * #%L
003 * HAPI FHIR - Master Data Management
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.mdm.interceptor;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.interceptor.api.Hook;
024import ca.uhn.fhir.interceptor.api.Pointcut;
025import ca.uhn.fhir.jpa.api.dao.DaoRegistry;
026import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao;
027import ca.uhn.fhir.jpa.searchparam.SearchParameterMap;
028import ca.uhn.fhir.mdm.api.MdmConstants;
029import ca.uhn.fhir.mdm.log.Logs;
030import ca.uhn.fhir.mdm.svc.MdmSearchExpansionResults;
031import ca.uhn.fhir.mdm.svc.MdmSearchExpansionSvc;
032import ca.uhn.fhir.rest.api.server.IPreResourceShowDetails;
033import ca.uhn.fhir.rest.api.server.RequestDetails;
034import ca.uhn.fhir.rest.api.server.storage.IResourcePersistentId;
035import ca.uhn.fhir.rest.server.util.ICachedSearchDetails;
036import ca.uhn.fhir.util.FhirTerser;
037import ca.uhn.fhir.util.ResourceReferenceInfo;
038import org.hl7.fhir.instance.model.api.IAnyResource;
039import org.hl7.fhir.instance.model.api.IBaseResource;
040import org.hl7.fhir.instance.model.api.IIdType;
041import org.slf4j.Logger;
042import org.springframework.beans.factory.annotation.Autowired;
043
044import java.util.HashSet;
045import java.util.List;
046import java.util.Optional;
047import java.util.Set;
048import java.util.stream.Collectors;
049
050/**
051 * <b>This class is experimental and subject to change. Use with caution.</b>
052 * <p>
053 * This interceptor provides an "MDM Virtualized" endpoint, meaning that
054 * searches are expanded to include MDM-linked resources (including any
055 * linked golden resource, and also including any other resources linked
056 * to that golden resource). Searches for non-MDM resources which have
057 * a reference to an MDM resource will have their reference parameter
058 * expanded to include the golden and linked resources.
059 * </p>
060 * <p>
061 * In addition, responses are cleaned up so that only the golden resource
062 * is included in responses, and references to non-golden resources
063 * are rewritten.
064 * </p>
065 * <p>
066 * This interceptor does not modify data that is being stored/written
067 * in any way, it only modifies data that is being returned by the
068 * server.
069 * </p>
070 *
071 * @since 8.0.0
072 */
073public class MdmReadVirtualizationInterceptor<P extends IResourcePersistentId<?>> {
074        private static final Logger ourMdmTroubleshootingLog = Logs.getMdmTroubleshootingLog();
075
076        private static final String CURRENTLY_PROCESSING_FLAG =
077                        MdmReadVirtualizationInterceptor.class.getName() + "_CURRENTLY_PROCESSING";
078        private static final MdmSearchExpansionSvc.IParamTester PARAM_TESTER_NO_RES_ID =
079                        (paramName, param) -> !IAnyResource.SP_RES_ID.equals(paramName);
080        private static final MdmSearchExpansionSvc.IParamTester PARAM_TESTER_ALL = (paramName, param) -> true;
081
082        @Autowired
083        private FhirContext myFhirContext;
084
085        @Autowired
086        private DaoRegistry myDaoRegistry;
087
088        @Autowired
089        private MdmSearchExpansionSvc myMdmSearchExpansionSvc;
090
091        @Hook(
092                        value = Pointcut.STORAGE_PRESEARCH_REGISTERED,
093                        order = MdmConstants.ORDER_PRESEARCH_REGISTERED_MDM_READ_VIRTUALIZATION_INTERCEPTOR)
094        public void preSearchRegistered(
095                        RequestDetails theRequestDetails,
096                        SearchParameterMap theSearchParameterMap,
097                        ICachedSearchDetails theSearchDetails) {
098                ourMdmTroubleshootingLog
099                                .atTrace()
100                                .setMessage("MDM virtualization original search: {}{}")
101                                .addArgument(theRequestDetails.getResourceName())
102                                .addArgument(() -> theSearchParameterMap.toNormalizedQueryString(myFhirContext))
103                                .log();
104
105                String resourceType = theSearchDetails.getResourceType();
106
107                if (theSearchParameterMap.hasIncludes() || theSearchParameterMap.hasRevIncludes()) {
108                        myMdmSearchExpansionSvc.expandSearchAndStoreInRequestDetails(
109                                        resourceType, theRequestDetails, theSearchParameterMap, PARAM_TESTER_ALL);
110                } else {
111                        // If we don't have any includes, it's not worth auto-expanding the _id parameter since we'll only end
112                        // up filtering out the extra resources afterward
113                        myMdmSearchExpansionSvc.expandSearchAndStoreInRequestDetails(
114                                        resourceType, theRequestDetails, theSearchParameterMap, PARAM_TESTER_NO_RES_ID);
115                }
116
117                ourMdmTroubleshootingLog
118                                .atDebug()
119                                .setMessage("MDM virtualization remapped search: {}{}")
120                                .addArgument(theRequestDetails.getResourceName())
121                                .addArgument(() -> theSearchParameterMap.toNormalizedQueryString(myFhirContext))
122                                .log();
123        }
124
125        @Hook(Pointcut.STORAGE_PRESHOW_RESOURCES)
126        public void preShowResources(RequestDetails theRequestDetails, IPreResourceShowDetails theDetails) {
127                MdmSearchExpansionResults expansionResults = MdmSearchExpansionSvc.getCachedExpansionResults(theRequestDetails);
128                if (expansionResults == null) {
129                        // This means the PRESEARCH hook didn't save anything, which probably means
130                        // no RequestDetails is available
131                        return;
132                }
133
134                if (theRequestDetails.getUserData().get(CURRENTLY_PROCESSING_FLAG) != null) {
135                        // Avoid recursive calls
136                        return;
137                }
138
139                /*
140                 * If a resource being returned is a resource that was mdm-expanded,
141                 * we'll replace that resource with the originally requested resource,
142                 * making sure to avoid adding duplicates to the results.
143                 */
144                Set<IIdType> resourcesInBundle = new HashSet<>();
145                for (int resourceIdx = 0; resourceIdx < theDetails.size(); resourceIdx++) {
146                        IBaseResource resource = theDetails.getResource(resourceIdx);
147                        IIdType id = resource.getIdElement().toUnqualifiedVersionless();
148                        Optional<IIdType> originalIdOpt = expansionResults.getOriginalIdForExpandedId(id);
149                        if (originalIdOpt.isPresent()) {
150                                IIdType originalId = originalIdOpt.get();
151                                if (resourcesInBundle.add(originalId)) {
152                                        IBaseResource originalResource = fetchResourceFromRepository(theRequestDetails, originalId);
153                                        theDetails.setResource(resourceIdx, originalResource);
154                                } else {
155                                        theDetails.setResource(resourceIdx, null);
156                                }
157                        } else {
158                                if (!resourcesInBundle.add(id)) {
159                                        theDetails.setResource(resourceIdx, null);
160                                }
161                        }
162                }
163
164                FhirTerser terser = myFhirContext.newTerser();
165                for (int resourceIdx = 0; resourceIdx < theDetails.size(); resourceIdx++) {
166                        IBaseResource resource = theDetails.getResource(resourceIdx);
167                        if (resource != null) {
168
169                                // Extract all the references in the resources we're returning
170                                // in case we need to remap them to golden equivalents
171                                List<ResourceReferenceInfo> referenceInfos = terser.getAllResourceReferences(resource);
172                                for (ResourceReferenceInfo referenceInfo : referenceInfos) {
173                                        IIdType referenceId = referenceInfo
174                                                        .getResourceReference()
175                                                        .getReferenceElement()
176                                                        .toUnqualifiedVersionless();
177                                        if (referenceId.hasResourceType()
178                                                        && referenceId.hasIdPart()
179                                                        && !referenceId.isLocal()
180                                                        && !referenceId.isUuid()) {
181                                                Optional<IIdType> nonExpandedId = expansionResults.getOriginalIdForExpandedId(referenceId);
182                                                if (nonExpandedId != null && nonExpandedId.isPresent()) {
183                                                        ourMdmTroubleshootingLog.debug(
184                                                                        "MDM virtualization is replacing reference at {} value {} with {}",
185                                                                        referenceInfo.getName(),
186                                                                        referenceInfo.getResourceReference().getReferenceElement(),
187                                                                        nonExpandedId.get().getValue());
188                                                        referenceInfo
189                                                                        .getResourceReference()
190                                                                        .setReference(nonExpandedId.get().getValue());
191                                                }
192                                        }
193                                }
194                        }
195                }
196
197                ourMdmTroubleshootingLog
198                                .atTrace()
199                                .setMessage("Returning resources: {}")
200                                .addArgument(() -> theDetails.getAllResources().stream()
201                                                .map(t -> t.getIdElement().toUnqualifiedVersionless().getValue())
202                                                .sorted()
203                                                .collect(Collectors.toList()))
204                                .log();
205        }
206
207        private IBaseResource fetchResourceFromRepository(RequestDetails theRequestDetails, IIdType originalId) {
208                IFhirResourceDao<?> dao = myDaoRegistry.getResourceDao(originalId.getResourceType());
209                theRequestDetails.getUserData().put(CURRENTLY_PROCESSING_FLAG, Boolean.TRUE);
210                IBaseResource originalResource;
211                try {
212                        originalResource = dao.read(originalId, theRequestDetails);
213                } finally {
214                        theRequestDetails.getUserData().remove(CURRENTLY_PROCESSING_FLAG);
215                }
216                return originalResource;
217        }
218}