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.rest.api.BundleLinks;
023import ca.uhn.fhir.rest.api.server.IBundleProvider;
024import ca.uhn.fhir.rest.server.RestfulServerUtils;
025import org.apache.commons.lang3.ObjectUtils;
026import org.apache.commons.lang3.StringUtils;
027import org.hl7.fhir.instance.model.api.IBaseResource;
028import org.slf4j.Logger;
029import org.slf4j.LoggerFactory;
030
031import java.util.List;
032
033/**
034 * This is an intermediate record object that holds all the fields required to make the final bundle that will be returned to the client.
035 */
036public class ResponsePage {
037        private static final Logger ourLog = LoggerFactory.getLogger(ResponsePage.class);
038
039        /**
040         * The id of the search used to page through search results
041         */
042        private final String mySearchId;
043        /**
044         * The list of resources that will be used to create the bundle
045         */
046        private final List<IBaseResource> myResourceList;
047        /**
048         * The total number of results that matched the search
049         */
050        private final Integer myNumTotalResults;
051        /**
052         * The number of resources that should be returned in each page
053         */
054        private final int myPageSize;
055        /**
056         * The number of resources that should be returned in the bundle.
057         * Can be smaller than pageSize when the bundleProvider
058         * has fewer results than the page size.
059         */
060        private final int myNumToReturn;
061
062        /**
063         * The count of resources included from the _include filter.
064         * These _include resources are otherwise included in the resourceList.
065         */
066        private final int myIncludedResourceCount;
067        /**
068         * This is the count of resources that have been omitted from results
069         * (typically because of consent interceptors).
070         * We track these because they shouldn't change paging results,
071         * even though it will change number of resources returned.
072         */
073        private final int myOmittedResourceCount;
074        /**
075         * This is the total count of requested resources
076         * (ie, non-omitted, non-_include'd resource count).
077         * We typically fetch (for offset queries) 1 more than
078         * we need so we know if there is an additional page
079         * to fetch.
080         * But this is determined by the implementers of
081         * IBundleProvider.
082         */
083        private final int myTotalRequestedResourcesFetched;
084
085        /**
086         * The bundle provider.
087         */
088        private final IBundleProvider myBundleProvider;
089
090        // Properties below here are set for calculation of pages;
091        // not part of the response pages in and of themselves
092
093        /**
094         * The response bundle request object
095         */
096        private ResponseBundleRequest myResponseBundleRequest;
097
098        /**
099         * Whether or not this page uses (non-cached) offset paging
100         */
101        private boolean myIsUsingOffsetPages = false;
102
103        /**
104         * The requested page object (should not be null for proper calculations)
105         */
106        private RequestedPage myRequestedPage;
107
108        /**
109         * The paging style being used.
110         * This is determined by a number of conditions,
111         * including what the bundleprovider provides.
112         */
113        private PagingStyle myPagingStyle;
114
115        ResponsePage(
116                        String theSearchId,
117                        List<IBaseResource> theResourceList,
118                        int thePageSize,
119                        int theNumToReturn,
120                        int theIncludedResourceCount,
121                        int theOmittedResourceCount,
122                        int theTotalRequestedResourcesFetched,
123                        IBundleProvider theBundleProvider) {
124                mySearchId = theSearchId;
125                myResourceList = theResourceList;
126                myPageSize = thePageSize;
127                myNumToReturn = theNumToReturn;
128                myIncludedResourceCount = theIncludedResourceCount;
129                myOmittedResourceCount = theOmittedResourceCount;
130                myTotalRequestedResourcesFetched = theTotalRequestedResourcesFetched;
131                myBundleProvider = theBundleProvider;
132
133                myNumTotalResults = myBundleProvider.size();
134        }
135
136        public int size() {
137                return myResourceList.size();
138        }
139
140        public List<IBaseResource> getResourceList() {
141                return myResourceList;
142        }
143
144        private boolean isBundleProviderOffsetPaging() {
145                if (myBundleProvider != null) {
146                        if (myBundleProvider.getCurrentPageOffset() != null) {
147                                // it's not enough that currentpageoffset is not null
148                                // (sometimes it's 0, even if it's not a currentpageoffset search)
149                                // so we have to make sure either next or prev links are not null
150                                return (StringUtils.isNotBlank(myBundleProvider.getNextPageId())
151                                                || StringUtils.isNotBlank(myBundleProvider.getPreviousPageId()));
152                        }
153                }
154
155                return false;
156        }
157
158        private void determinePagingStyle() {
159                if (myPagingStyle != null) {
160                        // already assigned
161                        return;
162                }
163
164                if (isBundleProviderOffsetPaging()) {
165                        myPagingStyle = PagingStyle.BUNDLE_PROVIDER_OFFSETS;
166                } else if (myIsUsingOffsetPages) {
167                        myPagingStyle = PagingStyle.NONCACHED_OFFSET;
168                } else if (myBundleProvider != null && StringUtils.isNotBlank(myBundleProvider.getCurrentPageId())) {
169                        myPagingStyle = PagingStyle.BUNDLE_PROVIDER_PAGE_IDS;
170                } else if (StringUtils.isNotBlank(mySearchId)) {
171                        myPagingStyle = PagingStyle.SAVED_SEARCH;
172                } else {
173                        myPagingStyle = PagingStyle.NONE;
174                        // only end up here if no paging is desired
175                        ourLog.debug(
176                                        "No accurate paging will be generated."
177                                                        + " If accurate paging is desired, ResponsePageBuilder must be provided with additioanl information.");
178                }
179        }
180
181        public void setRequestedPage(RequestedPage theRequestedPage) {
182                myRequestedPage = theRequestedPage;
183        }
184
185        public IBundleProvider getBundleProvider() {
186                return myBundleProvider;
187        }
188
189        public void setUseOffsetPaging(boolean theIsUsingOffsetPaging) {
190                myIsUsingOffsetPages = theIsUsingOffsetPaging;
191        }
192
193        public void setResponseBundleRequest(ResponseBundleRequest theRequest) {
194                myResponseBundleRequest = theRequest;
195        }
196
197        private boolean hasNextPage() {
198                determinePagingStyle();
199                switch (myPagingStyle) {
200                        case BUNDLE_PROVIDER_OFFSETS:
201                        case BUNDLE_PROVIDER_PAGE_IDS:
202                                return StringUtils.isNotBlank(myBundleProvider.getNextPageId());
203                        case NONCACHED_OFFSET:
204                                if (myNumTotalResults == null) {
205                                        if (hasNextPageWithoutKnowingTotal()) {
206                                                return true;
207                                        }
208                                } else if (myNumTotalResults > myNumToReturn + ObjectUtils.defaultIfNull(myRequestedPage.offset, 0)) {
209                                        return true;
210                                }
211                                break;
212                        case SAVED_SEARCH:
213                                if (myNumTotalResults == null) {
214                                        if (hasNextPageWithoutKnowingTotal()) {
215                                                return true;
216                                        }
217                                } else if (myResponseBundleRequest.offset + myNumToReturn < myNumTotalResults) {
218                                        return true;
219                                }
220                                break;
221                }
222
223                // fallthrough
224                return false;
225        }
226
227        /**
228         * If myNumTotalResults is null, it typically means we don't
229         * have an accurate total.
230         *
231         * Ie, we're in the middle of a set of pages (of non-named page results),
232         * and _total=accurate was not passed.
233         *
234         * This typically always means that a
235         * 'next' link definitely exists.
236         *
237         * But there are cases where this might not be true:
238         * * the last page of a search that also has an _include
239         *      query parameter where the total of resources + _include'd
240         *      resources is > the page size expected to be returned.
241         * * the last page of a search that returns the exact number
242         *      of resources requested
243         *
244         * In these case, we must check to see if the returned
245         * number of *requested* resources.
246         * If our bundleprovider has fetched > requested,
247         * we'll know that there are more resources already.
248         * But if it hasn't, we'll have to check pagesize compared to
249         * _include'd count, omitted count, and resource count.
250         */
251        private boolean hasNextPageWithoutKnowingTotal() {
252                // if we have totalRequestedResource count, and it's not equal to pagesize,
253                // then we can use this, alone, to determine if there are more pages
254                if (myTotalRequestedResourcesFetched >= 0) {
255                        if (myPageSize < myTotalRequestedResourcesFetched) {
256                                return true;
257                        }
258                } else {
259                        // otherwise we'll try and determine if there are next links based on the following
260                        // calculation:
261                        // resourceList.size - included resources + omitted resources == pagesize
262                        // -> we (most likely) have more resources
263                        if (myPageSize == myResourceList.size() - myIncludedResourceCount + myOmittedResourceCount) {
264                                ourLog.warn(
265                                                "Returning a next page based on calculated resource count."
266                                                                + " This could be inaccurate if the exact number of resources were fetched is equal to the pagesize requested. "
267                                                                + " Consider setting ResponseBundleBuilder.setTotalResourcesFetchedRequest after fetching resources.");
268                                return true;
269                        }
270                }
271                return false;
272        }
273
274        public void setNextPageIfNecessary(BundleLinks theLinks) {
275                if (hasNextPage()) {
276                        String next;
277                        switch (myPagingStyle) {
278                                case BUNDLE_PROVIDER_OFFSETS:
279                                        next = RestfulServerUtils.createOffsetPagingLink(
280                                                        theLinks,
281                                                        myResponseBundleRequest.requestDetails.getRequestPath(),
282                                                        myResponseBundleRequest.requestDetails.getTenantId(),
283                                                        myRequestedPage.offset + myRequestedPage.limit,
284                                                        myRequestedPage.limit,
285                                                        myResponseBundleRequest.getRequestParameters());
286                                        break;
287                                case NONCACHED_OFFSET:
288                                        next = RestfulServerUtils.createOffsetPagingLink(
289                                                        theLinks,
290                                                        myResponseBundleRequest.requestDetails.getRequestPath(),
291                                                        myResponseBundleRequest.requestDetails.getTenantId(),
292                                                        ObjectUtils.defaultIfNull(myRequestedPage.offset, 0) + myNumToReturn,
293                                                        myNumToReturn,
294                                                        myResponseBundleRequest.getRequestParameters());
295                                        break;
296                                case BUNDLE_PROVIDER_PAGE_IDS:
297                                        next = RestfulServerUtils.createPagingLink(
298                                                        theLinks,
299                                                        myResponseBundleRequest.requestDetails,
300                                                        myBundleProvider.getUuid(),
301                                                        myBundleProvider.getNextPageId(),
302                                                        myResponseBundleRequest.getRequestParameters());
303                                        break;
304                                case SAVED_SEARCH:
305                                        next = RestfulServerUtils.createPagingLink(
306                                                        theLinks,
307                                                        myResponseBundleRequest.requestDetails,
308                                                        mySearchId,
309                                                        myResponseBundleRequest.offset + myNumToReturn,
310                                                        myNumToReturn,
311                                                        myResponseBundleRequest.getRequestParameters());
312                                        break;
313                                default:
314                                        next = null;
315                                        break;
316                        }
317
318                        if (StringUtils.isNotBlank(next)) {
319                                theLinks.setNext(next);
320                        }
321                }
322        }
323
324        private boolean hasPreviousPage() {
325                determinePagingStyle();
326                switch (myPagingStyle) {
327                        case BUNDLE_PROVIDER_OFFSETS:
328                        case BUNDLE_PROVIDER_PAGE_IDS:
329                                return StringUtils.isNotBlank(myBundleProvider.getPreviousPageId());
330                        case NONCACHED_OFFSET:
331                                if (myRequestedPage != null && myRequestedPage.offset != null && myRequestedPage.offset > 0) {
332                                        return true;
333                                }
334                                break;
335                        case SAVED_SEARCH:
336                                return myResponseBundleRequest.offset > 0;
337                }
338
339                // fallthrough
340                return false;
341        }
342
343        public void setPreviousPageIfNecessary(BundleLinks theLinks) {
344                if (hasPreviousPage()) {
345                        String prev;
346                        switch (myPagingStyle) {
347                                case BUNDLE_PROVIDER_OFFSETS:
348                                        prev = RestfulServerUtils.createOffsetPagingLink(
349                                                        theLinks,
350                                                        myResponseBundleRequest.requestDetails.getRequestPath(),
351                                                        myResponseBundleRequest.requestDetails.getTenantId(),
352                                                        Math.max(ObjectUtils.defaultIfNull(myRequestedPage.offset, 0) - myRequestedPage.limit, 0),
353                                                        myRequestedPage.limit,
354                                                        myResponseBundleRequest.getRequestParameters());
355                                        break;
356                                case NONCACHED_OFFSET:
357                                        {
358                                                int start = Math.max(0, ObjectUtils.defaultIfNull(myRequestedPage.offset, 0) - myPageSize);
359                                                prev = RestfulServerUtils.createOffsetPagingLink(
360                                                                theLinks,
361                                                                myResponseBundleRequest.requestDetails.getRequestPath(),
362                                                                myResponseBundleRequest.requestDetails.getTenantId(),
363                                                                start,
364                                                                myPageSize,
365                                                                myResponseBundleRequest.getRequestParameters());
366                                        }
367                                        break;
368                                case BUNDLE_PROVIDER_PAGE_IDS:
369                                        prev = RestfulServerUtils.createPagingLink(
370                                                        theLinks,
371                                                        myResponseBundleRequest.requestDetails,
372                                                        myBundleProvider.getUuid(),
373                                                        myBundleProvider.getPreviousPageId(),
374                                                        myResponseBundleRequest.getRequestParameters());
375                                        break;
376                                case SAVED_SEARCH:
377                                        {
378                                                int start = Math.max(0, myResponseBundleRequest.offset - myPageSize);
379                                                prev = RestfulServerUtils.createPagingLink(
380                                                                theLinks,
381                                                                myResponseBundleRequest.requestDetails,
382                                                                mySearchId,
383                                                                start,
384                                                                myPageSize,
385                                                                myResponseBundleRequest.getRequestParameters());
386                                        }
387                                        break;
388                                default:
389                                        prev = null;
390                        }
391
392                        if (StringUtils.isNotBlank(prev)) {
393                                theLinks.setPrev(prev);
394                        }
395                }
396        }
397
398        /**
399         * A builder for constructing ResponsePage objects.
400         */
401        public static class ResponsePageBuilder {
402
403                private String mySearchId;
404                private List<IBaseResource> myResources;
405                private int myPageSize;
406                private int myNumToReturn;
407                private int myIncludedResourceCount;
408                private int myOmittedResourceCount;
409                private IBundleProvider myBundleProvider;
410                private int myTotalRequestedResourcesFetched = -1;
411
412                public ResponsePageBuilder setOmittedResourceCount(int theOmittedResourceCount) {
413                        myOmittedResourceCount = theOmittedResourceCount;
414                        return this;
415                }
416
417                public ResponsePageBuilder setIncludedResourceCount(int theIncludedResourceCount) {
418                        myIncludedResourceCount = theIncludedResourceCount;
419                        return this;
420                }
421
422                public ResponsePageBuilder setNumToReturn(int theNumToReturn) {
423                        myNumToReturn = theNumToReturn;
424                        return this;
425                }
426
427                public ResponsePageBuilder setPageSize(int thePageSize) {
428                        myPageSize = thePageSize;
429                        return this;
430                }
431
432                public ResponsePageBuilder setBundleProvider(IBundleProvider theBundleProvider) {
433                        myBundleProvider = theBundleProvider;
434                        return this;
435                }
436
437                public ResponsePageBuilder setResources(List<IBaseResource> theResources) {
438                        myResources = theResources;
439                        return this;
440                }
441
442                public ResponsePageBuilder setSearchId(String theSearchId) {
443                        mySearchId = theSearchId;
444                        return this;
445                }
446
447                public ResponsePageBuilder setTotalRequestedResourcesFetched(int theTotalRequestedResourcesFetched) {
448                        myTotalRequestedResourcesFetched = theTotalRequestedResourcesFetched;
449                        return this;
450                }
451
452                /**
453                 * Combine this builder with a second buider.
454                 * Useful if a second page is requested, but you do not wish to
455                 * overwrite the current values.
456                 *
457                 * Will not replace searchId, nor IBundleProvider (which should be
458                 * the exact same for any subsequent searches anyways).
459                 *
460                 * Will also not copy pageSize nor numToReturn, as these should be
461                 * the same for any single search result set.
462                 *
463                 * @param theSecondBuilder - a second builder (cannot be this one)
464                 */
465                public void combineWith(ResponsePageBuilder theSecondBuilder) {
466                        assert theSecondBuilder != this; // don't want to combine with itself
467
468                        if (myTotalRequestedResourcesFetched != -1 && theSecondBuilder.myTotalRequestedResourcesFetched != -1) {
469                                myTotalRequestedResourcesFetched += theSecondBuilder.myTotalRequestedResourcesFetched;
470                        }
471
472                        // primitives can always be added
473                        myIncludedResourceCount += theSecondBuilder.myIncludedResourceCount;
474                        myOmittedResourceCount += theSecondBuilder.myOmittedResourceCount;
475                }
476
477                public ResponsePage build() {
478                        return new ResponsePage(
479                                        mySearchId, // search id
480                                        myResources, // resource list
481                                        myPageSize, // page size
482                                        myNumToReturn, // num to return
483                                        myIncludedResourceCount, // included count
484                                        myOmittedResourceCount, // omitted resources
485                                        myTotalRequestedResourcesFetched, // total count of requested resources
486                                        myBundleProvider // the bundle provider
487                                        );
488                }
489        }
490
491        /**
492         * First we determine what kind of paging we use:
493         * * Bundle Provider Offsets - the bundle provider has offset counts that it uses
494         *                                                      to determine the page. For legacy reasons, it's not enough
495         *                                                      that the bundle provider has a currentOffsetPage. Sometimes
496         *                                                      this value is provided (often as a 0), but no nextPageId nor previousPageId
497         *                                                      is available. Typically this is the case in UnitTests.
498         * * non-cached offsets - if the server is not storing the search results (and it's not
499         *                                                      an everything operator) OR the Requested Page has an initial offset
500         *                                                      OR it is explicitly set to use non-cached offset
501         *                                                      (ResponseBundleBuilder.myIsOffsetModeHistory)
502         * * Bundle Provider Page Ids - the bundle provider knows the page ids and will
503         *                                                      provide them. bundle provider will have a currentPageId
504         * * Saved Search                       - the server has a saved search object with an id that it
505         *                                                      uses to page through results.
506         */
507        private enum PagingStyle {
508                /**
509                 * Paging is done by offsets; pages are not cached
510                 */
511                NONCACHED_OFFSET,
512                /**
513                 * Paging is done by offsets, but
514                 * the bundle provider provides the offsets
515                 */
516                BUNDLE_PROVIDER_OFFSETS,
517                /**
518                 * Paging is done by page ids,
519                 * but bundle provider provides the page ids
520                 */
521                BUNDLE_PROVIDER_PAGE_IDS,
522                /**
523                 * The server has a saved search object with an id
524                 * that is used to page through results.
525                 */
526                SAVED_SEARCH,
527                /**
528                 * No paging is done at all.
529                 * No previous nor next links will be available, even if previous or next
530                 * links exist.
531                 * If paging is required, a different paging method must be specified.
532                 */
533                NONE;
534        }
535}