001/*-
002 * #%L
003 * HAPI FHIR - Core Library
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.interceptor.model;
021
022import ca.uhn.fhir.model.api.IModelJson;
023import ca.uhn.fhir.util.JsonUtil;
024import com.fasterxml.jackson.annotation.JsonProperty;
025import com.fasterxml.jackson.core.JsonProcessingException;
026import com.fasterxml.jackson.databind.ObjectMapper;
027import jakarta.annotation.Nonnull;
028import jakarta.annotation.Nullable;
029import org.apache.commons.lang3.Validate;
030import org.apache.commons.lang3.builder.EqualsBuilder;
031import org.apache.commons.lang3.builder.HashCodeBuilder;
032import org.apache.commons.lang3.builder.ToStringBuilder;
033import org.apache.commons.lang3.builder.ToStringStyle;
034
035import java.time.LocalDate;
036import java.util.ArrayList;
037import java.util.Arrays;
038import java.util.Collection;
039import java.util.Collections;
040import java.util.List;
041import java.util.Objects;
042import java.util.stream.Collectors;
043import java.util.stream.Stream;
044
045import static org.apache.commons.lang3.ObjectUtils.defaultIfNull;
046
047/**
048 * @since 5.0.0
049 */
050public class RequestPartitionId implements IModelJson {
051        private static final RequestPartitionId ALL_PARTITIONS = new RequestPartitionId();
052        private static final ObjectMapper ourObjectMapper =
053                        new ObjectMapper().registerModule(new com.fasterxml.jackson.datatype.jsr310.JavaTimeModule());
054
055        @JsonProperty("partitionDate")
056        private final LocalDate myPartitionDate;
057
058        @JsonProperty("allPartitions")
059        private final boolean myAllPartitions;
060
061        @JsonProperty("partitionIds")
062        private final List<Integer> myPartitionIds;
063
064        @JsonProperty("partitionNames")
065        private final List<String> myPartitionNames;
066
067        /**
068         * Constructor for a single partition
069         */
070        private RequestPartitionId(
071                        @Nullable String thePartitionName, @Nullable Integer thePartitionId, @Nullable LocalDate thePartitionDate) {
072                myPartitionIds = toListOrNull(thePartitionId);
073                myPartitionNames = toListOrNull(thePartitionName);
074                myPartitionDate = thePartitionDate;
075                myAllPartitions = false;
076        }
077
078        /**
079         * Constructor for a multiple partition
080         */
081        private RequestPartitionId(
082                        @Nullable List<String> thePartitionName,
083                        @Nullable List<Integer> thePartitionId,
084                        @Nullable LocalDate thePartitionDate) {
085                myPartitionIds = toListOrNull(thePartitionId);
086                myPartitionNames = toListOrNull(thePartitionName);
087                myPartitionDate = thePartitionDate;
088                myAllPartitions = false;
089        }
090
091        /**
092         * Constructor for all partitions
093         */
094        private RequestPartitionId() {
095                super();
096                myPartitionDate = null;
097                myPartitionNames = null;
098                myPartitionIds = null;
099                myAllPartitions = true;
100        }
101
102        /**
103         * Creates a new RequestPartitionId which includes all partition IDs from
104         * this {@link RequestPartitionId} but also includes all IDs from the given
105         * {@link RequestPartitionId}. Any duplicates are only included once, and
106         * partition names and dates are ignored and not returned. This {@link RequestPartitionId}
107         * and {@literal theOther} are not modified.
108         *
109         * @since 7.4.0
110         */
111        public RequestPartitionId mergeIds(RequestPartitionId theOther) {
112                if (isAllPartitions() || theOther.isAllPartitions()) {
113                        return RequestPartitionId.allPartitions();
114                }
115
116                // don't know why this is required - otherwise PartitionedStrictTransactionR4Test fails
117                if (this.equals(theOther)) {
118                        return this;
119                }
120
121                List<Integer> thisPartitionIds = getPartitionIds();
122                List<Integer> otherPartitionIds = theOther.getPartitionIds();
123                List<Integer> newPartitionIds = Stream.concat(thisPartitionIds.stream(), otherPartitionIds.stream())
124                                .distinct()
125                                .collect(Collectors.toList());
126                return RequestPartitionId.fromPartitionIds(newPartitionIds);
127        }
128
129        public static RequestPartitionId fromJson(String theJson) throws JsonProcessingException {
130                return ourObjectMapper.readValue(theJson, RequestPartitionId.class);
131        }
132
133        public boolean isAllPartitions() {
134                return myAllPartitions;
135        }
136
137        public boolean isPartitionCovered(Integer thePartitionId) {
138                return isAllPartitions() || getPartitionIds().contains(thePartitionId);
139        }
140
141        @Nullable
142        public LocalDate getPartitionDate() {
143                return myPartitionDate;
144        }
145
146        @Nullable
147        public List<String> getPartitionNames() {
148                return myPartitionNames;
149        }
150
151        @Nonnull
152        public List<Integer> getPartitionIds() {
153                Validate.notNull(myPartitionIds, "Partition IDs have not been set");
154                return myPartitionIds;
155        }
156
157        @Override
158        public String toString() {
159                ToStringBuilder b = new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE);
160                if (hasPartitionIds()) {
161                        b.append("ids", getPartitionIds());
162                }
163                if (hasPartitionNames()) {
164                        b.append("names", getPartitionNames());
165                }
166                if (myAllPartitions) {
167                        b.append("allPartitions", myAllPartitions);
168                }
169                return b.build();
170        }
171
172        @Override
173        public boolean equals(Object theO) {
174                if (this == theO) {
175                        return true;
176                }
177
178                if (theO == null || getClass() != theO.getClass()) {
179                        return false;
180                }
181
182                RequestPartitionId that = (RequestPartitionId) theO;
183
184                EqualsBuilder b = new EqualsBuilder();
185                b.append(myAllPartitions, that.myAllPartitions);
186                b.append(myPartitionDate, that.myPartitionDate);
187                b.append(myPartitionIds, that.myPartitionIds);
188                b.append(myPartitionNames, that.myPartitionNames);
189                return b.isEquals();
190        }
191
192        @Override
193        public int hashCode() {
194                return new HashCodeBuilder(17, 37)
195                                .append(myPartitionDate)
196                                .append(myAllPartitions)
197                                .append(myPartitionIds)
198                                .append(myPartitionNames)
199                                .toHashCode();
200        }
201
202        public String toJson() {
203                return JsonUtil.serializeOrInvalidRequest(this);
204        }
205
206        @Nullable
207        public Integer getFirstPartitionIdOrNull() {
208                if (myPartitionIds != null) {
209                        return myPartitionIds.get(0);
210                }
211                return null;
212        }
213
214        public String getFirstPartitionNameOrNull() {
215                if (myPartitionNames != null) {
216                        return myPartitionNames.get(0);
217                }
218                return null;
219        }
220
221        /**
222         * Returns true if this request partition contains only one partition ID and it is the DEFAULT partition ID (null)
223         */
224        public boolean isDefaultPartition() {
225                if (isAllPartitions()) {
226                        return false;
227                }
228                return hasPartitionIds()
229                                && getPartitionIds().size() == 1
230                                && getPartitionIds().get(0) == null;
231        }
232
233        public boolean hasPartitionId(Integer thePartitionId) {
234                Validate.notNull(myPartitionIds, "Partition IDs not set");
235                return myPartitionIds.contains(thePartitionId);
236        }
237
238        public boolean hasPartitionIds() {
239                return myPartitionIds != null;
240        }
241
242        public boolean hasPartitionNames() {
243                return myPartitionNames != null;
244        }
245
246        public boolean hasDefaultPartitionId() {
247                return getPartitionIds().contains(null);
248        }
249
250        public List<Integer> getPartitionIdsWithoutDefault() {
251                return getPartitionIds().stream().filter(Objects::nonNull).collect(Collectors.toList());
252        }
253
254        @Nullable
255        private static <T> List<T> toListOrNull(@Nullable Collection<T> theList) {
256                if (theList != null) {
257                        if (theList.size() == 1) {
258                                return Collections.singletonList(theList.iterator().next());
259                        }
260                        return Collections.unmodifiableList(new ArrayList<>(theList));
261                }
262                return null;
263        }
264
265        @Nullable
266        private static <T> List<T> toListOrNull(@Nullable T theObject) {
267                if (theObject != null) {
268                        return Collections.singletonList(theObject);
269                }
270                return null;
271        }
272
273        @SafeVarargs
274        @Nullable
275        private static <T> List<T> toListOrNull(@Nullable T... theObject) {
276                if (theObject != null) {
277                        return Arrays.asList(theObject);
278                }
279                return null;
280        }
281
282        @Nonnull
283        public static RequestPartitionId allPartitions() {
284                return ALL_PARTITIONS;
285        }
286
287        @Nonnull
288        public static RequestPartitionId defaultPartition() {
289                return fromPartitionIds(Collections.singletonList(null));
290        }
291
292        @Nonnull
293        public static RequestPartitionId defaultPartition(@Nullable LocalDate thePartitionDate) {
294                return fromPartitionIds(Collections.singletonList(null), thePartitionDate);
295        }
296
297        @Nonnull
298        public static RequestPartitionId fromPartitionId(@Nullable Integer thePartitionId) {
299                return fromPartitionIds(Collections.singletonList(thePartitionId));
300        }
301
302        @Nonnull
303        public static RequestPartitionId fromPartitionId(
304                        @Nullable Integer thePartitionId, @Nullable LocalDate thePartitionDate) {
305                return new RequestPartitionId(null, Collections.singletonList(thePartitionId), thePartitionDate);
306        }
307
308        @Nonnull
309        public static RequestPartitionId fromPartitionIds(@Nonnull Collection<Integer> thePartitionIds) {
310                return fromPartitionIds(thePartitionIds, null);
311        }
312
313        @Nonnull
314        public static RequestPartitionId fromPartitionIds(
315                        @Nonnull Collection<Integer> thePartitionIds, @Nullable LocalDate thePartitionDate) {
316                return new RequestPartitionId(null, toListOrNull(thePartitionIds), thePartitionDate);
317        }
318
319        @Nonnull
320        public static RequestPartitionId fromPartitionIds(Integer... thePartitionIds) {
321                return new RequestPartitionId(null, toListOrNull(thePartitionIds), null);
322        }
323
324        @Nonnull
325        public static RequestPartitionId fromPartitionName(@Nullable String thePartitionName) {
326                return fromPartitionName(thePartitionName, null);
327        }
328
329        @Nonnull
330        public static RequestPartitionId fromPartitionName(
331                        @Nullable String thePartitionName, @Nullable LocalDate thePartitionDate) {
332                return new RequestPartitionId(thePartitionName, null, thePartitionDate);
333        }
334
335        @Nonnull
336        public static RequestPartitionId fromPartitionNames(@Nullable List<String> thePartitionNames) {
337                return new RequestPartitionId(toListOrNull(thePartitionNames), null, null);
338        }
339
340        @Nonnull
341        public static RequestPartitionId fromPartitionNames(String... thePartitionNames) {
342                return new RequestPartitionId(toListOrNull(thePartitionNames), null, null);
343        }
344
345        @Nonnull
346        public static RequestPartitionId fromPartitionIdAndName(
347                        @Nullable Integer thePartitionId, @Nullable String thePartitionName) {
348                return new RequestPartitionId(thePartitionName, thePartitionId, null);
349        }
350
351        @Nonnull
352        public static RequestPartitionId forPartitionIdAndName(
353                        @Nullable Integer thePartitionId, @Nullable String thePartitionName, @Nullable LocalDate thePartitionDate) {
354                return new RequestPartitionId(thePartitionName, thePartitionId, thePartitionDate);
355        }
356
357        @Nonnull
358        public static RequestPartitionId forPartitionIdsAndNames(
359                        List<String> thePartitionNames, List<Integer> thePartitionIds, LocalDate thePartitionDate) {
360                return new RequestPartitionId(thePartitionNames, thePartitionIds, thePartitionDate);
361        }
362
363        public static boolean isDefaultPartition(@Nullable RequestPartitionId thePartitionId) {
364                if (thePartitionId == null) {
365                        return false;
366                }
367
368                return thePartitionId.isDefaultPartition();
369        }
370
371        /**
372         * Create a string representation suitable for use as a cache key. Null aware.
373         * <p>
374         * Returns the partition IDs (numeric) as a joined string with a space between, using the string "null" for any null values
375         */
376        public static String stringifyForKey(@Nonnull RequestPartitionId theRequestPartitionId) {
377                String retVal = "(all)";
378                if (!theRequestPartitionId.isAllPartitions()) {
379                        assert theRequestPartitionId.hasPartitionIds();
380                        retVal = theRequestPartitionId.getPartitionIds().stream()
381                                        .map(t -> defaultIfNull(t, "null").toString())
382                                        .collect(Collectors.joining(" "));
383                }
384                return retVal;
385        }
386
387        public String asJson() throws JsonProcessingException {
388                return ourObjectMapper.writeValueAsString(this);
389        }
390}