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 org.apache.commons.lang3.ObjectUtils;
013import org.hl7.fhir.instance.model.api.IBaseResource;
014import org.slf4j.Logger;
015import org.slf4j.LoggerFactory;
016
017import java.util.Collection;
018import java.util.Collections;
019import java.util.HashMap;
020import java.util.List;
021import java.util.Map;
022import java.util.stream.Collectors;
023
024import static java.util.Collections.emptyList;
025import static java.util.Objects.nonNull;
026
027public class OperationResponsesAccumulator implements IResultsAccumulator<OperationResponse> {
028
029        private static final Logger ourLog = LoggerFactory.getLogger(OperationResponsesAccumulator.class);
030
031        private final int myMaxResponsesCount;
032
033        private final Map<String, OperationResponse> myTargetIdToResponseMap = Collections.synchronizedMap(new HashMap<>());
034
035        public OperationResponsesAccumulator() {
036                this(Integer.MAX_VALUE);
037        }
038
039        public OperationResponsesAccumulator(int theMaxResponsesCount) {
040                myMaxResponsesCount = theMaxResponsesCount;
041        }
042
043        public boolean isAcceptingMoreOperationResponses() {
044                return getResults().size() < myMaxResponsesCount;
045        }
046
047        public Collection<OperationResponse> getResults() {
048                return myTargetIdToResponseMap.values();
049        }
050
051        /**
052         * Accumulates a list of resources that was originally obtained from the RESPONSE
053         * and processed by interceptors if applicable.
054         *
055         * @param theResults the list of resources processed by interceptors to accumulate
056         * @param theResponse the original response from which a list of resources was obtained and submitted to interceptors
057         * @param theTargetId the id of the target that returned the RESPONSE
058         */
059        @Override
060        public void accumulate(List<IBaseResource> theResults, OperationResponse theResponse, String theTargetId) {
061                // at this point, theResults is a list of resources (originally obtained from theResponse) that was processed
062                // by interceptors.  if applicable, we need to:
063                // 1- sanitize the list;
064                // 2- store it in theResponse;
065                // 3- accumulate theResponse;
066
067                if (!isAcceptingMoreOperationResponses()) {
068                        return;
069                }
070
071                List<IBaseResource> nonNullResultList = ObjectUtils.defaultIfNull(theResults, emptyList());
072                List<IBaseResource> sanitizedResultList = nonNullResultList.stream()
073                                .filter(theIBaseResource -> nonNull(theIBaseResource))
074                                .collect(Collectors.toList());
075
076                // the target response for an operation is one IBaseResource or nothing if there was an exception thrown by
077                // the invocation.  if we have anything to return, it will be at the first index.
078                IBaseResource resourceToReturn = sanitizedResultList.isEmpty() ? null : sanitizedResultList.get(0);
079
080                theResponse.setResponse(resourceToReturn);
081
082                myTargetIdToResponseMap.put(theTargetId, theResponse);
083        }
084
085        @Override
086        public void addResults(String theTargetId, List<IBaseResource> theResults, int theTotal) {
087                ourLog.debug("Can't add {} Resources to targetId {}.  Operation not supported", theResults.size(), theTargetId);
088        }
089
090        @Override
091        public List<IBaseResource> getResults(String theTargetId) {
092                ourLog.debug("Can't provide a list of Resources for targetId {}.  Operation not supported", theTargetId);
093                return emptyList();
094        }
095
096        @Override
097        public List<IBaseResource> getMatchedResults(String theTargetId) {
098                ourLog.debug("Can't provide matched results for targetId {}.  Operation not supported", theTargetId);
099                return emptyList();
100        }
101
102        @Override
103        public List<IBaseResource> getPageableResults(String theTargetId) {
104                ourLog.debug("Can't provide pageable results for targetId {}.  Operation not supported", theTargetId);
105                return emptyList();
106        }
107
108        @Override
109        public List<IBaseResource> getIncludeResults(String theTargetId) {
110                ourLog.debug("Can't provide a list of included results targetId {}.  Operation not supported", theTargetId);
111                return emptyList();
112        }
113
114        @Override
115        public List<IBaseResource> getOutcomeResults(String theTargetId) {
116                ourLog.debug("Can't provide a list of outcome results targetId {}.  Operation not supported", theTargetId);
117                return emptyList();
118        }
119
120        @Override
121        public List<IBaseResource> getUncategorizedResults(String theTargetId) {
122                ourLog.debug(
123                                "Can't provide a list of uncategorized included results targetId {}.  Operation not supported",
124                                theTargetId);
125                return emptyList();
126        }
127
128        @Override
129        public void removeResults(String theTargetId, List<IBaseResource> theResultsToBeRemoved) {
130                ourLog.debug("Can't remove results for targetId {}.  Operation not supported", theTargetId);
131        }
132
133        @Override
134        public int getTotal() {
135                ourLog.debug("Can't provide a total count.  A placeholder value of 0 is returned.  Operation not supported");
136                return 0;
137        }
138
139        @Override
140        public boolean getTotalOmitted() {
141                ourLog.debug(
142                                "Can't provide a total omitted count. A placeholder value of false is returned.  Operation not supported");
143                return false;
144        }
145
146        @Override
147        public void setTotalOmitted(boolean theTotalOmitted) {
148                ourLog.debug("Can't set the total omitted count to {}.  Operation not supported", theTotalOmitted);
149        }
150}