001/*-
002 * #%L
003 * HAPI FHIR - Server Framework
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.rest.server.method;
021
022import ca.uhn.fhir.i18n.Msg;
023import ca.uhn.fhir.rest.api.BundleLinks;
024import ca.uhn.fhir.rest.api.IVersionSpecificBundleFactory;
025import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
026import ca.uhn.fhir.rest.api.server.IBundleProvider;
027import ca.uhn.fhir.rest.api.server.IRestfulServer;
028import ca.uhn.fhir.rest.api.server.RequestDetails;
029import ca.uhn.fhir.rest.server.IPagingProvider;
030import ca.uhn.fhir.rest.server.RestfulServerUtils;
031import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
032import org.apache.commons.lang3.StringUtils;
033import org.hl7.fhir.instance.model.api.IBaseBundle;
034import org.hl7.fhir.instance.model.api.IBaseOperationOutcome;
035import org.hl7.fhir.instance.model.api.IBaseResource;
036import org.slf4j.Logger;
037import org.slf4j.LoggerFactory;
038
039import java.util.ArrayList;
040import java.util.Collections;
041import java.util.List;
042import java.util.Objects;
043
044/**
045 * Service to build a FHIR Bundle from a request and a Bundle Provider
046 */
047public class ResponseBundleBuilder {
048        private static final Logger ourLog = LoggerFactory.getLogger(ResponseBundleBuilder.class);
049
050        private final boolean myIsOffsetModeHistory;
051
052        public ResponseBundleBuilder(boolean theIsOffsetModeHistory) {
053                myIsOffsetModeHistory = theIsOffsetModeHistory;
054        }
055
056        IBaseBundle buildResponseBundle(ResponseBundleRequest theResponseBundleRequest) {
057                final ResponsePage responsePage = buildResponsePage(theResponseBundleRequest);
058
059                removeNulls(responsePage.getResourceList());
060                validateIds(responsePage.getResourceList());
061
062                BundleLinks links = buildLinks(theResponseBundleRequest, responsePage);
063
064                return buildBundle(theResponseBundleRequest, responsePage, links);
065        }
066
067        private static IBaseBundle buildBundle(
068                        ResponseBundleRequest theResponseBundleRequest, ResponsePage pageResponse, BundleLinks links) {
069                final IRestfulServer<?> server = theResponseBundleRequest.server;
070                final IVersionSpecificBundleFactory bundleFactory =
071                                server.getFhirContext().newBundleFactory();
072                final IBundleProvider bundleProvider = theResponseBundleRequest.bundleProvider;
073
074                bundleFactory.addRootPropertiesToBundle(
075                                bundleProvider.getUuid(), links, bundleProvider.size(), bundleProvider.getPublished());
076                bundleFactory.addResourcesToBundle(
077                                new ArrayList<>(pageResponse.getResourceList()),
078                                theResponseBundleRequest.bundleType,
079                                links.serverBase,
080                                server.getBundleInclusionRule(),
081                                theResponseBundleRequest.includes);
082
083                return (IBaseBundle) bundleFactory.getResourceBundle();
084        }
085
086        private ResponsePage buildResponsePage(ResponseBundleRequest theResponseBundleRequest) {
087                final IRestfulServer<?> server = theResponseBundleRequest.server;
088                final IBundleProvider bundleProvider = theResponseBundleRequest.bundleProvider;
089                final RequestedPage requestedPage = theResponseBundleRequest.requestedPage;
090                final List<IBaseResource> resourceList;
091                final int pageSize;
092
093                ResponsePage.ResponsePageBuilder responsePageBuilder = new ResponsePage.ResponsePageBuilder();
094
095                int numToReturn;
096                String searchId = null;
097
098                if (requestedPage.offset != null || !server.canStoreSearchResults()) {
099                        pageSize = offsetCalculatePageSize(server, requestedPage, bundleProvider.size());
100                        numToReturn = pageSize;
101
102                        resourceList = offsetBuildResourceList(bundleProvider, requestedPage, numToReturn, responsePageBuilder);
103                        RestfulServerUtils.validateResourceListNotNull(resourceList);
104                } else {
105                        pageSize = pagingCalculatePageSize(requestedPage, server.getPagingProvider());
106
107                        Integer size = bundleProvider.size();
108                        if (size == null) {
109                                numToReturn = pageSize;
110                        } else {
111                                numToReturn = Math.min(pageSize, size.intValue() - theResponseBundleRequest.offset);
112                        }
113
114                        resourceList =
115                                        pagingBuildResourceList(theResponseBundleRequest, bundleProvider, numToReturn, responsePageBuilder);
116                        RestfulServerUtils.validateResourceListNotNull(resourceList);
117
118                        searchId = pagingBuildSearchId(theResponseBundleRequest, numToReturn, bundleProvider.size());
119                }
120
121                // We should leave the IBundleProvider to populate these values (specifically resourceList).
122                // But since we haven't updated all such providers, we will
123                // build it here (this is at best 'duplicating' work).
124                responsePageBuilder
125                                .setSearchId(searchId)
126                                .setPageSize(pageSize)
127                                .setNumToReturn(numToReturn)
128                                .setBundleProvider(bundleProvider)
129                                .setResources(resourceList);
130
131                return responsePageBuilder.build();
132        }
133
134        private static String pagingBuildSearchId(
135                        ResponseBundleRequest theResponseBundleRequest, int theNumToReturn, Integer theNumTotalResults) {
136                final IPagingProvider pagingProvider = theResponseBundleRequest.server.getPagingProvider();
137                String retval = null;
138
139                if (theResponseBundleRequest.searchId != null) {
140                        retval = theResponseBundleRequest.searchId;
141                } else {
142                        if (theNumTotalResults == null || theNumTotalResults > theNumToReturn) {
143                                retval = pagingProvider.storeResultList(
144                                                theResponseBundleRequest.requestDetails, theResponseBundleRequest.bundleProvider);
145                                if (StringUtils.isBlank(retval)) {
146                                        ourLog.info(
147                                                        "Found {} results but paging provider did not provide an ID to use for paging",
148                                                        theNumTotalResults);
149                                        retval = null;
150                                }
151                        }
152                }
153                return retval;
154        }
155
156        private static List<IBaseResource> pagingBuildResourceList(
157                        ResponseBundleRequest theResponseBundleRequest,
158                        IBundleProvider theBundleProvider,
159                        int theNumToReturn,
160                        ResponsePage.ResponsePageBuilder theResponsePageBuilder) {
161                final List<IBaseResource> retval;
162                if (theNumToReturn > 0 || theBundleProvider.getCurrentPageId() != null) {
163                        retval = theBundleProvider.getResources(
164                                        theResponseBundleRequest.offset,
165                                        theNumToReturn + theResponseBundleRequest.offset,
166                                        theResponsePageBuilder);
167                } else {
168                        retval = Collections.emptyList();
169                }
170                return retval;
171        }
172
173        private static int pagingCalculatePageSize(RequestedPage theRequestedPage, IPagingProvider thePagingProvider) {
174                if (theRequestedPage.limit == null || theRequestedPage.limit.equals(0)) {
175                        return thePagingProvider.getDefaultPageSize();
176                } else {
177                        return Math.min(thePagingProvider.getMaximumPageSize(), theRequestedPage.limit);
178                }
179        }
180
181        private List<IBaseResource> offsetBuildResourceList(
182                        IBundleProvider theBundleProvider,
183                        RequestedPage theRequestedPage,
184                        int theNumToReturn,
185                        ResponsePage.ResponsePageBuilder theResponsePageBuilder) {
186                final List<IBaseResource> retval;
187                if ((theRequestedPage.offset != null && !myIsOffsetModeHistory)
188                                || theBundleProvider.getCurrentPageOffset() != null) {
189                        // When offset query is done theResult already contains correct amount (+ their includes etc.) so return
190                        // everything
191                        retval = theBundleProvider.getResources(0, Integer.MAX_VALUE, theResponsePageBuilder);
192                } else if (theNumToReturn > 0) {
193                        retval = theBundleProvider.getResources(0, theNumToReturn, theResponsePageBuilder);
194                } else {
195                        retval = Collections.emptyList();
196                }
197                return retval;
198        }
199
200        private static int offsetCalculatePageSize(
201                        IRestfulServer<?> server, RequestedPage theRequestedPage, Integer theNumTotalResults) {
202                final int retval;
203                if (theRequestedPage.limit != null) {
204                        retval = theRequestedPage.limit;
205                } else {
206                        if (server.getDefaultPageSize() != null) {
207                                retval = server.getDefaultPageSize();
208                        } else {
209                                retval = theNumTotalResults != null ? theNumTotalResults : Integer.MAX_VALUE;
210                        }
211                }
212                return retval;
213        }
214
215        private static void validateIds(List<IBaseResource> theResourceList) {
216                /*
217                 * Make sure all returned resources have an ID (if not, this is a bug
218                 * in the user server code)
219                 */
220                for (IBaseResource next : theResourceList) {
221                        if (next.getIdElement() == null || next.getIdElement().isEmpty()) {
222                                if (!(next instanceof IBaseOperationOutcome)) {
223                                        throw new InternalErrorException(Msg.code(435) + "Server method returned resource of type["
224                                                        + next.getClass().getSimpleName()
225                                                        + "] with no ID specified (IResource#setId(IdDt) must be called)");
226                                }
227                        }
228                }
229        }
230
231        private static void removeNulls(List<IBaseResource> resourceList) {
232                /*
233                 * Remove any null entries in the list - This generally shouldn't happen but can if
234                 * data has been manually purged from the JPA database
235                 */
236                boolean hasNull = false;
237                for (IBaseResource next : resourceList) {
238                        if (next == null) {
239                                hasNull = true;
240                                break;
241                        }
242                }
243                if (hasNull) {
244                        resourceList.removeIf(Objects::isNull);
245                }
246        }
247
248        private BundleLinks buildLinks(ResponseBundleRequest theResponseBundleRequest, ResponsePage theResponsePage) {
249                final IRestfulServer<?> server = theResponseBundleRequest.server;
250                final RequestedPage pageRequest = theResponseBundleRequest.requestedPage;
251
252                BundleLinks retval = new BundleLinks(
253                                theResponseBundleRequest.requestDetails.getFhirServerBase(),
254                                theResponseBundleRequest.includes,
255                                RestfulServerUtils.prettyPrintResponse(server, theResponseBundleRequest.requestDetails),
256                                theResponseBundleRequest.bundleType);
257
258                // set self link
259                retval.setSelf(theResponseBundleRequest.linkSelf);
260
261                // determine if we are using offset / uncached pages
262                theResponsePage.setUseOffsetPaging(pageRequest.offset != null
263                                || (!server.canStoreSearchResults() && !isEverythingOperation(theResponseBundleRequest.requestDetails))
264                                || myIsOffsetModeHistory);
265                theResponsePage.setResponseBundleRequest(theResponseBundleRequest);
266                theResponsePage.setRequestedPage(pageRequest);
267
268                // generate our links
269                theResponsePage.setNextPageIfNecessary(retval);
270                theResponsePage.setPreviousPageIfNecessary(retval);
271
272                return retval;
273        }
274
275        private boolean isEverythingOperation(RequestDetails theRequest) {
276                return (theRequest.getRestOperationType() == RestOperationTypeEnum.EXTENDED_OPERATION_TYPE
277                                                || theRequest.getRestOperationType() == RestOperationTypeEnum.EXTENDED_OPERATION_INSTANCE)
278                                && theRequest.getOperation() != null
279                                && theRequest.getOperation().equals("$everything");
280        }
281}