001/*-
002 * #%L
003 * Smile CDR - CDR
004 * %%
005 * Copyright (C) 2016 - 2025 Smile CDR, Inc.
006 * %%
007 * All rights reserved.
008 * #L%
009 */
010package ca.cdr.api.fhirgw.model;
011
012import ca.uhn.fhir.context.FhirContext;
013import ca.uhn.fhir.model.valueset.BundleEntrySearchModeEnum;
014import ca.uhn.fhir.util.BundleUtil;
015import org.hl7.fhir.instance.model.api.IBaseResource;
016
017import java.util.ArrayList;
018import java.util.Collection;
019import java.util.Collections;
020import java.util.HashMap;
021import java.util.List;
022import java.util.Map;
023import java.util.concurrent.atomic.AtomicBoolean;
024import java.util.concurrent.atomic.AtomicInteger;
025import java.util.stream.Stream;
026
027import static ca.uhn.fhir.model.api.ResourceMetadataKeyEnum.ENTRY_SEARCH_MODE;
028
029/**
030 * This class is used to collect the search results as they are fetched from
031 * individual targets from the gateway.
032 * An instance is created for each set of request sent from the gateway to target servers
033 */
034public class BundleResultsAccumulator implements IResultsAccumulator<SearchSingleTargetResponse> {
035
036        private final FhirContext myFhirCtx;
037
038        public BundleResultsAccumulator(FhirContext theFhirCtx) {
039                myFhirCtx = theFhirCtx;
040        }
041
042        private final Map<String, List<IBaseResource>> myTargetIdToMatchResults =
043                        Collections.synchronizedMap(new HashMap<>());
044        private final Map<String, List<IBaseResource>> myTargetIdToIncludeResults =
045                        Collections.synchronizedMap(new HashMap<>());
046
047        private final Map<String, List<IBaseResource>> myTargetIdToOutcomeResults =
048                        Collections.synchronizedMap(new HashMap<>());
049        private final Map<String, List<IBaseResource>> myTargetIdToUncategorizedResults =
050                        Collections.synchronizedMap(new HashMap<>());
051        private final AtomicInteger myTotal = new AtomicInteger();
052        private final AtomicBoolean myTotalOmitted = new AtomicBoolean();
053
054        @Override
055        public void addResults(String theTargetId, List<IBaseResource> theSearchResults, int theTotal) {
056                List<IBaseResource> matchList = myTargetIdToMatchResults.computeIfAbsent(
057                                theTargetId, t -> Collections.synchronizedList(new ArrayList<>()));
058                List<IBaseResource> includeList = myTargetIdToIncludeResults.computeIfAbsent(
059                                theTargetId, t -> Collections.synchronizedList(new ArrayList<>()));
060                List<IBaseResource> outcomeList = myTargetIdToOutcomeResults.computeIfAbsent(
061                                theTargetId, t -> Collections.synchronizedList(new ArrayList<>()));
062                List<IBaseResource> uncategorizedList = myTargetIdToUncategorizedResults.computeIfAbsent(
063                                theTargetId, t -> Collections.synchronizedList(new ArrayList<>()));
064
065                // Split into actual matched results, includes, outcomes, and unknown(missing)
066                for (IBaseResource theSearchResult : theSearchResults) {
067                        if (theSearchResult == null) {
068                                continue;
069                        }
070
071                        //                      BundleEntrySearchModeEnum searchMode = ENTRY_SEARCH_MODE.get(theSearchResult);
072                        // TODO GGG fix MetadataKeyEnum to support IBaseResource instead of IAnyResource
073                        BundleEntrySearchModeEnum searchMode =
074                                        (BundleEntrySearchModeEnum) theSearchResult.getUserData(ENTRY_SEARCH_MODE.name());
075                        if (searchMode == null) {
076                                uncategorizedList.add(theSearchResult);
077                        } else {
078                                switch (searchMode) {
079                                        case MATCH -> matchList.add(theSearchResult);
080                                        case INCLUDE -> includeList.add(theSearchResult);
081                                        case OUTCOME -> outcomeList.add(theSearchResult);
082                                        default -> uncategorizedList.add(theSearchResult);
083                                }
084                        }
085                }
086                if (!myTotalOmitted.get()) {
087                        myTotal.addAndGet(theTotal);
088                }
089        }
090
091        @Override
092        public List<IBaseResource> getResults(String theTargetId) {
093                return Stream.of(
094                                                getMatchedResults(theTargetId),
095                                                getIncludeResults(theTargetId),
096                                                getOutcomeResults(theTargetId),
097                                                getUncategorizedResults(theTargetId))
098                                .flatMap(Collection::stream)
099                                .toList();
100        }
101
102        @Override
103        public List<IBaseResource> getPageableResults(String theTargetId) {
104                return Collections.unmodifiableList(concatenateMatchedAndUncategorized(theTargetId));
105        }
106
107        private List<IBaseResource> concatenateMatchedAndUncategorized(String theTargetId) {
108                return Stream.concat(getMatchedResults(theTargetId).stream(), getUncategorizedResults(theTargetId).stream())
109                                .toList();
110        }
111
112        @Override
113        public List<IBaseResource> getMatchedResults(String theTargetId) {
114                List<IBaseResource> retVal = myTargetIdToMatchResults.getOrDefault(theTargetId, Collections.emptyList());
115                return Collections.unmodifiableList(retVal);
116        }
117
118        @Override
119        public List<IBaseResource> getIncludeResults(String theTargetId) {
120                List<IBaseResource> retVal = myTargetIdToIncludeResults.getOrDefault(theTargetId, Collections.emptyList());
121                return Collections.unmodifiableList(retVal);
122        }
123
124        @Override
125        public List<IBaseResource> getOutcomeResults(String theTargetId) {
126                List<IBaseResource> retVal = myTargetIdToOutcomeResults.getOrDefault(theTargetId, Collections.emptyList());
127                return Collections.unmodifiableList(retVal);
128        }
129
130        @Override
131        public List<IBaseResource> getUncategorizedResults(String theTargetId) {
132                List<IBaseResource> retVal =
133                                myTargetIdToUncategorizedResults.getOrDefault(theTargetId, Collections.emptyList());
134                return Collections.unmodifiableList(retVal);
135        }
136
137        @Override
138        public void removeResults(String theTargetId, List<IBaseResource> theResultsToBeRemoved) {
139                List.of(
140                                                myTargetIdToMatchResults,
141                                                myTargetIdToIncludeResults,
142                                                myTargetIdToUncategorizedResults,
143                                                myTargetIdToOutcomeResults)
144                                .forEach(map ->
145                                                map.getOrDefault(theTargetId, Collections.emptyList()).removeAll(theResultsToBeRemoved));
146        }
147
148        @Override
149        public int getTotal() {
150                return myTotal.get();
151        }
152
153        @Override
154        public boolean getTotalOmitted() {
155                return myTotalOmitted.get();
156        }
157
158        @Override
159        public void setTotalOmitted(boolean theTotalOmitted) {
160                myTotalOmitted.set(theTotalOmitted);
161        }
162
163        @Override
164        public void accumulate(List<IBaseResource> theResults, SearchSingleTargetResponse theResponse, String theId) {
165                Integer total = BundleUtil.getTotal(myFhirCtx, theResponse.getSearchResults());
166
167                if (total != null) {
168                        addResults(theId, theResults, total);
169                } else {
170                        setTotalOmitted(true);
171                        addResults(theId, theResults, 0);
172                }
173        }
174}