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.api.paging; 021 022import ca.uhn.fhir.i18n.Msg; 023import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 024import jakarta.annotation.Nullable; 025import org.apache.commons.lang3.StringUtils; 026import org.hl7.fhir.instance.model.api.IPrimitiveType; 027import org.springframework.data.domain.PageRequest; 028 029import static ca.uhn.fhir.rest.api.Constants.PARAM_COUNT; 030import static ca.uhn.fhir.rest.api.Constants.PARAM_OFFSET; 031 032/** 033 * This class is essentially just a data clump of offset + count, as well as the ability to convert itself into a standard 034 * {@link PageRequest} for spring data to use. The reason we don't use PageRequest natively is because it is concerned with `pages` and `counts`, 035 * but we are using `offset` and `count` which requires some minor translation. 036 */ 037public class MdmPageRequest { 038 039 private final int myPage; 040 private final int myOffset; 041 private final int myCount; 042 043 public MdmPageRequest( 044 @Nullable IPrimitiveType<Integer> theOffset, 045 @Nullable IPrimitiveType<Integer> theCount, 046 int theDefaultPageSize, 047 int theMaximumPageSize) { 048 myOffset = theOffset == null ? 0 : theOffset.getValue(); 049 myCount = theCount == null ? theDefaultPageSize : Math.min(theCount.getValue(), theMaximumPageSize); 050 validatePagingParameters(myOffset, myCount); 051 052 this.myPage = myOffset / myCount; 053 } 054 055 public MdmPageRequest( 056 @Nullable Integer theOffset, @Nullable Integer theCount, int theDefaultPageSize, int theMaximumPageSize) { 057 myOffset = theOffset == null ? 0 : theOffset; 058 myCount = theCount == null ? theDefaultPageSize : Math.min(theCount, theMaximumPageSize); 059 validatePagingParameters(myOffset, myCount); 060 061 this.myPage = myOffset / myCount; 062 } 063 064 private void validatePagingParameters(int theOffset, int theCount) { 065 String errorMessage = ""; 066 067 if (theOffset < 0) { 068 errorMessage += PARAM_OFFSET + " must be greater than or equal to 0. "; 069 } 070 if (theCount <= 0) { 071 errorMessage += PARAM_COUNT + " must be greater than 0."; 072 } 073 if (StringUtils.isNotEmpty(errorMessage)) { 074 throw new InvalidRequestException(Msg.code(1524) + errorMessage); 075 } 076 } 077 078 public int getOffset() { 079 return myOffset; 080 } 081 082 public int getPage() { 083 return myPage; 084 } 085 086 public int getCount() { 087 return myCount; 088 } 089 090 public int getNextOffset() { 091 return myOffset + myCount; 092 } 093 094 public int getPreviousOffset() { 095 return myOffset - myCount; 096 } 097 098 public PageRequest toPageRequest() { 099 return PageRequest.of(this.myPage, this.myCount); 100 } 101}