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.util;
021
022import ca.uhn.fhir.context.BaseRuntimeChildDefinition;
023import ca.uhn.fhir.context.BaseRuntimeElementCompositeDefinition;
024import ca.uhn.fhir.context.BaseRuntimeElementDefinition;
025import ca.uhn.fhir.context.FhirContext;
026import ca.uhn.fhir.context.RuntimeResourceDefinition;
027import ca.uhn.fhir.i18n.Msg;
028import ca.uhn.fhir.model.primitive.IdDt;
029import ca.uhn.fhir.model.valueset.BundleTypeEnum;
030import ca.uhn.fhir.rest.api.PatchTypeEnum;
031import ca.uhn.fhir.rest.api.RequestTypeEnum;
032import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
033import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
034import ca.uhn.fhir.util.bundle.BundleEntryMutator;
035import ca.uhn.fhir.util.bundle.BundleEntryParts;
036import ca.uhn.fhir.util.bundle.EntryListAccumulator;
037import ca.uhn.fhir.util.bundle.ModifiableBundleEntry;
038import ca.uhn.fhir.util.bundle.SearchBundleEntryParts;
039import com.google.common.collect.Sets;
040import jakarta.annotation.Nonnull;
041import jakarta.annotation.Nullable;
042import org.apache.commons.lang3.Validate;
043import org.apache.commons.lang3.tuple.Pair;
044import org.hl7.fhir.instance.model.api.IBase;
045import org.hl7.fhir.instance.model.api.IBaseBinary;
046import org.hl7.fhir.instance.model.api.IBaseBundle;
047import org.hl7.fhir.instance.model.api.IBaseReference;
048import org.hl7.fhir.instance.model.api.IBaseResource;
049import org.hl7.fhir.instance.model.api.IPrimitiveType;
050import org.slf4j.Logger;
051import org.slf4j.LoggerFactory;
052
053import java.math.BigDecimal;
054import java.util.ArrayList;
055import java.util.HashMap;
056import java.util.LinkedHashSet;
057import java.util.List;
058import java.util.Map;
059import java.util.Objects;
060import java.util.Set;
061import java.util.function.Consumer;
062import java.util.stream.Collectors;
063
064import static org.apache.commons.lang3.StringUtils.defaultString;
065import static org.apache.commons.lang3.StringUtils.isBlank;
066import static org.apache.commons.lang3.StringUtils.isNotBlank;
067import static org.hl7.fhir.instance.model.api.IBaseBundle.LINK_PREV;
068
069/**
070 * Fetch resources from a bundle
071 */
072public class BundleUtil {
073
074        /** Non instantiable */
075        private BundleUtil() {
076                // nothing
077        }
078
079        private static final Logger ourLog = LoggerFactory.getLogger(BundleUtil.class);
080
081        private static final String PREVIOUS = LINK_PREV;
082        private static final String PREV = "prev";
083        private static final Set<String> previousOrPrev = Sets.newHashSet(PREVIOUS, PREV);
084
085        public static final String DIFFERENT_LINK_ERROR_MSG =
086                        "Mismatching 'previous' and 'prev' links exist. 'previous' " + "is: '$PREVIOUS' and 'prev' is: '$PREV'.";
087
088        /**
089         * @return Returns <code>null</code> if the link isn't found or has no value
090         */
091        public static String getLinkUrlOfType(FhirContext theContext, IBaseBundle theBundle, String theLinkRelation) {
092                return getLinkUrlOfType(theContext, theBundle, theLinkRelation, true);
093        }
094
095        private static String getLinkUrlOfType(
096                        FhirContext theContext, IBaseBundle theBundle, String theLinkRelation, boolean isPreviousCheck) {
097                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
098                BaseRuntimeChildDefinition entryChild = def.getChildByName("link");
099                List<IBase> links = entryChild.getAccessor().getValues(theBundle);
100                for (IBase nextLink : links) {
101
102                        boolean isRightRel = false;
103                        BaseRuntimeElementCompositeDefinition<?> relDef =
104                                        (BaseRuntimeElementCompositeDefinition<?>) theContext.getElementDefinition(nextLink.getClass());
105                        BaseRuntimeChildDefinition relChild = relDef.getChildByName("relation");
106                        List<IBase> relValues = relChild.getAccessor().getValues(nextLink);
107                        for (IBase next : relValues) {
108                                IPrimitiveType<?> nextValue = (IPrimitiveType<?>) next;
109                                if (isRelationMatch(
110                                                theContext, theBundle, theLinkRelation, nextValue.getValueAsString(), isPreviousCheck)) {
111                                        isRightRel = true;
112                                }
113                        }
114
115                        if (!isRightRel) {
116                                continue;
117                        }
118
119                        BaseRuntimeElementCompositeDefinition<?> linkDef =
120                                        (BaseRuntimeElementCompositeDefinition<?>) theContext.getElementDefinition(nextLink.getClass());
121                        BaseRuntimeChildDefinition urlChild = linkDef.getChildByName("url");
122                        List<IBase> values = urlChild.getAccessor().getValues(nextLink);
123                        for (IBase nextUrl : values) {
124                                IPrimitiveType<?> nextValue = (IPrimitiveType<?>) nextUrl;
125                                if (isNotBlank(nextValue.getValueAsString())) {
126                                        return nextValue.getValueAsString();
127                                }
128                        }
129                }
130
131                return null;
132        }
133
134        private static boolean isRelationMatch(
135                        FhirContext theContext, IBaseBundle theBundle, String value, String matching, boolean theIsPreviousCheck) {
136                if (!theIsPreviousCheck) {
137                        return value.equals(matching);
138                }
139
140                if (previousOrPrev.contains(value)) {
141                        validateUniqueOrMatchingPreviousValues(theContext, theBundle);
142                        if (previousOrPrev.contains(matching)) {
143                                return true;
144                        }
145                }
146                return (value.equals(matching));
147        }
148
149        private static void validateUniqueOrMatchingPreviousValues(FhirContext theContext, IBaseBundle theBundle) {
150                String previousLink = getLinkNoCheck(theContext, theBundle, PREVIOUS);
151                String prevLink = getLinkNoCheck(theContext, theBundle, PREV);
152                if (prevLink != null && previousLink != null) {
153                        if (!previousLink.equals(prevLink)) {
154                                String msg = DIFFERENT_LINK_ERROR_MSG
155                                                .replace("$PREVIOUS", previousLink)
156                                                .replace("$PREV", prevLink);
157                                throw new InternalErrorException(Msg.code(2368) + msg);
158                        }
159                }
160        }
161
162        private static String getLinkNoCheck(FhirContext theContext, IBaseBundle theBundle, String theLinkRelation) {
163                return getLinkUrlOfType(theContext, theBundle, theLinkRelation, false);
164        }
165
166        /**
167         * Returns a collection of Pairs, one for each entry in the bundle. Each pair will contain
168         * the values of Bundle.entry.fullUrl, and Bundle.entry.resource respectively. Nulls
169         * are possible in either or both values in the Pair.
170         *
171         * @since 7.0.0
172         */
173        @SuppressWarnings("unchecked")
174        public static List<Pair<String, IBaseResource>> getBundleEntryFullUrlsAndResources(
175                        FhirContext theContext, IBaseBundle theBundle) {
176                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
177                BaseRuntimeChildDefinition entryChild = def.getChildByName("entry");
178                List<IBase> entries = entryChild.getAccessor().getValues(theBundle);
179
180                BaseRuntimeElementCompositeDefinition<?> entryChildElem =
181                                (BaseRuntimeElementCompositeDefinition<?>) entryChild.getChildByName("entry");
182                BaseRuntimeChildDefinition resourceChild = entryChildElem.getChildByName("resource");
183
184                BaseRuntimeChildDefinition urlChild = entryChildElem.getChildByName("fullUrl");
185
186                List<Pair<String, IBaseResource>> retVal = new ArrayList<>(entries.size());
187                for (IBase nextEntry : entries) {
188
189                        String fullUrl = urlChild.getAccessor()
190                                        .getFirstValueOrNull(nextEntry)
191                                        .map(t -> (((IPrimitiveType<?>) t).getValueAsString()))
192                                        .orElse(null);
193                        IBaseResource resource = (IBaseResource)
194                                        resourceChild.getAccessor().getFirstValueOrNull(nextEntry).orElse(null);
195
196                        retVal.add(Pair.of(fullUrl, resource));
197                }
198
199                return retVal;
200        }
201
202        public static List<Pair<String, IBaseResource>> getBundleEntryUrlsAndResources(
203                        FhirContext theContext, IBaseBundle theBundle) {
204                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
205                BaseRuntimeChildDefinition entryChild = def.getChildByName("entry");
206                List<IBase> entries = entryChild.getAccessor().getValues(theBundle);
207
208                BaseRuntimeElementCompositeDefinition<?> entryChildElem =
209                                (BaseRuntimeElementCompositeDefinition<?>) entryChild.getChildByName("entry");
210                BaseRuntimeChildDefinition resourceChild = entryChildElem.getChildByName("resource");
211
212                BaseRuntimeChildDefinition requestChild = entryChildElem.getChildByName("request");
213                BaseRuntimeElementCompositeDefinition<?> requestDef =
214                                (BaseRuntimeElementCompositeDefinition<?>) requestChild.getChildByName("request");
215
216                BaseRuntimeChildDefinition urlChild = requestDef.getChildByName("url");
217
218                List<Pair<String, IBaseResource>> retVal = new ArrayList<>(entries.size());
219                for (IBase nextEntry : entries) {
220
221                        String url = requestChild
222                                        .getAccessor()
223                                        .getFirstValueOrNull(nextEntry)
224                                        .flatMap(e -> urlChild.getAccessor().getFirstValueOrNull(e))
225                                        .map(t -> ((IPrimitiveType<?>) t).getValueAsString())
226                                        .orElse(null);
227
228                        IBaseResource resource = (IBaseResource)
229                                        resourceChild.getAccessor().getFirstValueOrNull(nextEntry).orElse(null);
230
231                        retVal.add(Pair.of(url, resource));
232                }
233
234                return retVal;
235        }
236
237        public static String getBundleType(FhirContext theContext, IBaseBundle theBundle) {
238                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
239                BaseRuntimeChildDefinition entryChild = def.getChildByName("type");
240                List<IBase> entries = entryChild.getAccessor().getValues(theBundle);
241                if (entries.size() > 0) {
242                        IPrimitiveType<?> typeElement = (IPrimitiveType<?>) entries.get(0);
243                        return typeElement.getValueAsString();
244                }
245                return null;
246        }
247
248        public static BundleTypeEnum getBundleTypeEnum(FhirContext theContext, IBaseBundle theBundle) {
249                String bundleTypeCode = BundleUtil.getBundleType(theContext, theBundle);
250                if (isBlank(bundleTypeCode)) {
251                        return null;
252                }
253                return BundleTypeEnum.forCode(bundleTypeCode);
254        }
255
256        public static void setBundleType(FhirContext theContext, IBaseBundle theBundle, String theType) {
257                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
258                BaseRuntimeChildDefinition entryChild = def.getChildByName("type");
259                BaseRuntimeElementDefinition<?> element = entryChild.getChildByName("type");
260                IPrimitiveType<?> typeInstance =
261                                (IPrimitiveType<?>) element.newInstance(entryChild.getInstanceConstructorArguments());
262                typeInstance.setValueAsString(theType);
263
264                entryChild.getMutator().setValue(theBundle, typeInstance);
265        }
266
267        public static Integer getTotal(FhirContext theContext, IBaseBundle theBundle) {
268                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
269                BaseRuntimeChildDefinition entryChild = def.getChildByName("total");
270                List<IBase> entries = entryChild.getAccessor().getValues(theBundle);
271                if (entries.size() > 0) {
272                        @SuppressWarnings("unchecked")
273                        IPrimitiveType<Number> typeElement = (IPrimitiveType<Number>) entries.get(0);
274                        if (typeElement != null && typeElement.getValue() != null) {
275                                return typeElement.getValue().intValue();
276                        }
277                }
278                return null;
279        }
280
281        public static void setTotal(FhirContext theContext, IBaseBundle theBundle, Integer theTotal) {
282                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
283                BaseRuntimeChildDefinition entryChild = def.getChildByName("total");
284                @SuppressWarnings("unchecked")
285                IPrimitiveType<Integer> value =
286                                (IPrimitiveType<Integer>) entryChild.getChildByName("total").newInstance();
287                value.setValue(theTotal);
288                entryChild.getMutator().setValue(theBundle, value);
289        }
290
291        /**
292         * Extract all of the resources from a given bundle
293         */
294        public static List<BundleEntryParts> toListOfEntries(FhirContext theContext, IBaseBundle theBundle) {
295                EntryListAccumulator entryListAccumulator = new EntryListAccumulator();
296                processEntries(theContext, theBundle, entryListAccumulator);
297                return entryListAccumulator.getList();
298        }
299
300        static int WHITE = 1;
301        static int GRAY = 2;
302        static int BLACK = 3;
303
304        /**
305         * Function which will do an in-place sort of a bundles' entries, to the correct processing order, which is:
306         * 1. Deletes
307         * 2. Creates
308         * 3. Updates
309         * <p>
310         * Furthermore, within these operation types, the entries will be sorted based on the order in which they should be processed
311         * e.g. if you have 2 CREATEs, one for a Patient, and one for an Observation which has this Patient as its Subject,
312         * the patient will come first, then the observation.
313         * <p>
314         * In cases of there being a cyclic dependency (e.g. Organization/1 is partOf Organization/2 and Organization/2 is partOf Organization/1)
315         * this function will throw an IllegalStateException.
316         *
317         * @param theContext The FhirContext.
318         * @param theBundle The {@link IBaseBundle} which contains the entries you would like sorted into processing order.
319         */
320        public static void sortEntriesIntoProcessingOrder(FhirContext theContext, IBaseBundle theBundle)
321                        throws IllegalStateException {
322                Map<BundleEntryParts, IBase> partsToIBaseMap = getPartsToIBaseMap(theContext, theBundle);
323
324                // Get all deletions.
325                LinkedHashSet<IBase> deleteParts =
326                                sortEntriesOfTypeIntoProcessingOrder(theContext, RequestTypeEnum.DELETE, partsToIBaseMap);
327                validatePartsNotNull(deleteParts);
328                LinkedHashSet<IBase> retVal = new LinkedHashSet<>(deleteParts);
329
330                // Get all Creations
331                LinkedHashSet<IBase> createParts =
332                                sortEntriesOfTypeIntoProcessingOrder(theContext, RequestTypeEnum.POST, partsToIBaseMap);
333                validatePartsNotNull(createParts);
334                retVal.addAll(createParts);
335
336                // Get all Updates
337                LinkedHashSet<IBase> updateParts =
338                                sortEntriesOfTypeIntoProcessingOrder(theContext, RequestTypeEnum.PUT, partsToIBaseMap);
339                validatePartsNotNull(updateParts);
340                retVal.addAll(updateParts);
341
342                // Once we are done adding all DELETE, POST, PUT operations, add everything else.
343                // Since this is a set, it will just fail to add already-added operations.
344                retVal.addAll(partsToIBaseMap.values());
345
346                // Blow away the entries and reset them in the right order.
347                TerserUtil.clearField(theContext, theBundle, "entry");
348                TerserUtil.setField(theContext, "entry", theBundle, retVal.toArray(new IBase[0]));
349        }
350
351        /**
352         * Converts a Bundle containing resources into a FHIR transaction which
353         * creates/updates the resources. This method does not modify the original
354         * bundle, but returns a new copy.
355         * <p>
356         * This method is mostly intended for test scenarios where you have a Bundle
357         * containing search results or other sourced resources, and want to upload
358         * these resources to a server using a single FHIR transaction.
359         * </p>
360         * <p>
361         * The Bundle is converted using the following logic:
362         * <ul>
363         *     <li>Bundle.type is changed to <code>transaction</code></li>
364         *     <li>Bundle.request.method is changed to <code>PUT</code></li>
365         *     <li>Bundle.request.url is changed to <code>[resourceType]/[id]</code></li>
366         *     <li>Bundle.fullUrl is changed to <code>[resourceType]/[id]</code></li>
367         * </ul>
368         * </p>
369         *
370         * @param theContext The FhirContext to use with the bundle
371         * @param theBundle The Bundle to modify. All resources in the Bundle should have an ID.
372         * @param thePrefixIdsOrNull If not <code>null</code>, all resource IDs and all references in the Bundle will be
373         *                           modified to such that their IDs contain the given prefix. For example, for a value
374         *                           of "A", the resource "Patient/123" will be changed to be "Patient/A123". If set to
375         *                           <code>null</code>, resource IDs are unchanged.
376         * @since 7.4.0
377         */
378        public static <T extends IBaseBundle> T convertBundleIntoTransaction(
379                        @Nonnull FhirContext theContext, @Nonnull T theBundle, @Nullable String thePrefixIdsOrNull) {
380                String prefix = defaultString(thePrefixIdsOrNull);
381
382                BundleBuilder bb = new BundleBuilder(theContext);
383
384                FhirTerser terser = theContext.newTerser();
385                List<IBase> entries = terser.getValues(theBundle, "Bundle.entry");
386                for (var entry : entries) {
387                        IBaseResource resource = terser.getSingleValueOrNull(entry, "resource", IBaseResource.class);
388                        if (resource != null) {
389                                Validate.isTrue(resource.getIdElement().hasIdPart(), "Resource in bundle has no ID");
390                                String newId = theContext.getResourceType(resource) + "/" + prefix
391                                                + resource.getIdElement().getIdPart();
392
393                                IBaseResource resourceClone = terser.clone(resource);
394                                resourceClone.setId(newId);
395
396                                if (isNotBlank(prefix)) {
397                                        for (var ref : terser.getAllResourceReferences(resourceClone)) {
398                                                var refElement = ref.getResourceReference().getReferenceElement();
399                                                ref.getResourceReference()
400                                                                .setReference(refElement.getResourceType() + "/" + prefix + refElement.getIdPart());
401                                        }
402                                }
403
404                                bb.addTransactionUpdateEntry(resourceClone);
405                        }
406                }
407
408                return bb.getBundleTyped();
409        }
410
411        private static void validatePartsNotNull(LinkedHashSet<IBase> theDeleteParts) {
412                if (theDeleteParts == null) {
413                        throw new IllegalStateException(
414                                        Msg.code(1745) + "This transaction contains a cycle, so it cannot be sorted.");
415                }
416        }
417
418        private static LinkedHashSet<IBase> sortEntriesOfTypeIntoProcessingOrder(
419                        FhirContext theContext,
420                        RequestTypeEnum theRequestTypeEnum,
421                        Map<BundleEntryParts, IBase> thePartsToIBaseMap) {
422                SortLegality legality = new SortLegality();
423                HashMap<String, Integer> color = new HashMap<>();
424                HashMap<String, List<String>> adjList = new HashMap<>();
425                List<String> topologicalOrder = new ArrayList<>();
426                Set<BundleEntryParts> bundleEntryParts = thePartsToIBaseMap.keySet().stream()
427                                .filter(part -> part.getRequestType().equals(theRequestTypeEnum))
428                                .collect(Collectors.toSet());
429                HashMap<String, BundleEntryParts> resourceIdToBundleEntryMap = new HashMap<>();
430
431                for (BundleEntryParts bundleEntryPart : bundleEntryParts) {
432                        IBaseResource resource = bundleEntryPart.getResource();
433                        if (resource != null) {
434                                String resourceId = resource.getIdElement().toVersionless().toString();
435                                resourceIdToBundleEntryMap.put(resourceId, bundleEntryPart);
436                                if (resourceId == null) {
437                                        if (bundleEntryPart.getFullUrl() != null) {
438                                                resourceId = bundleEntryPart.getFullUrl();
439                                        }
440                                }
441
442                                color.put(resourceId, WHITE);
443                        }
444                }
445
446                for (BundleEntryParts bundleEntryPart : bundleEntryParts) {
447                        IBaseResource resource = bundleEntryPart.getResource();
448                        if (resource != null) {
449                                String resourceId = resource.getIdElement().toVersionless().toString();
450                                resourceIdToBundleEntryMap.put(resourceId, bundleEntryPart);
451                                if (resourceId == null) {
452                                        if (bundleEntryPart.getFullUrl() != null) {
453                                                resourceId = bundleEntryPart.getFullUrl();
454                                        }
455                                }
456                                List<ResourceReferenceInfo> allResourceReferences =
457                                                theContext.newTerser().getAllResourceReferences(resource);
458                                String finalResourceId = resourceId;
459                                allResourceReferences.forEach(refInfo -> {
460                                        String referencedResourceId = refInfo.getResourceReference()
461                                                        .getReferenceElement()
462                                                        .toVersionless()
463                                                        .getValue();
464                                        if (color.containsKey(referencedResourceId)) {
465                                                if (!adjList.containsKey(finalResourceId)) {
466                                                        adjList.put(finalResourceId, new ArrayList<>());
467                                                }
468                                                adjList.get(finalResourceId).add(referencedResourceId);
469                                        }
470                                });
471                        }
472                }
473
474                for (Map.Entry<String, Integer> entry : color.entrySet()) {
475                        if (entry.getValue() == WHITE) {
476                                depthFirstSearch(entry.getKey(), color, adjList, topologicalOrder, legality);
477                        }
478                }
479
480                if (legality.isLegal()) {
481                        if (ourLog.isDebugEnabled()) {
482                                ourLog.debug("Topological order is: {}", String.join(",", topologicalOrder));
483                        }
484
485                        LinkedHashSet<IBase> orderedEntries = new LinkedHashSet<>();
486                        for (int i = 0; i < topologicalOrder.size(); i++) {
487                                BundleEntryParts bep;
488                                if (theRequestTypeEnum.equals(RequestTypeEnum.DELETE)) {
489                                        int index = topologicalOrder.size() - i - 1;
490                                        bep = resourceIdToBundleEntryMap.get(topologicalOrder.get(index));
491                                } else {
492                                        bep = resourceIdToBundleEntryMap.get(topologicalOrder.get(i));
493                                }
494                                IBase base = thePartsToIBaseMap.get(bep);
495                                orderedEntries.add(base);
496                        }
497
498                        return orderedEntries;
499
500                } else {
501                        return null;
502                }
503        }
504
505        private static void depthFirstSearch(
506                        String theResourceId,
507                        HashMap<String, Integer> theResourceIdToColor,
508                        HashMap<String, List<String>> theAdjList,
509                        List<String> theTopologicalOrder,
510                        SortLegality theLegality) {
511
512                if (!theLegality.isLegal()) {
513                        ourLog.debug("Found a cycle while trying to sort bundle entries. This bundle is not sortable.");
514                        return;
515                }
516
517                // We are currently recursing over this node (gray)
518                theResourceIdToColor.put(theResourceId, GRAY);
519
520                for (String neighbourResourceId : theAdjList.getOrDefault(theResourceId, new ArrayList<>())) {
521                        if (theResourceIdToColor.get(neighbourResourceId) == WHITE) {
522                                depthFirstSearch(
523                                                neighbourResourceId, theResourceIdToColor, theAdjList, theTopologicalOrder, theLegality);
524                        } else if (theResourceIdToColor.get(neighbourResourceId) == GRAY) {
525                                theLegality.setLegal(false);
526                                return;
527                        }
528                }
529                // Mark the node as black
530                theResourceIdToColor.put(theResourceId, BLACK);
531                theTopologicalOrder.add(theResourceId);
532        }
533
534        private static Map<BundleEntryParts, IBase> getPartsToIBaseMap(FhirContext theContext, IBaseBundle theBundle) {
535                RuntimeResourceDefinition bundleDef = theContext.getResourceDefinition(theBundle);
536                BaseRuntimeChildDefinition entryChildDef = bundleDef.getChildByName("entry");
537                List<IBase> entries = entryChildDef.getAccessor().getValues(theBundle);
538
539                BaseRuntimeElementCompositeDefinition<?> entryChildContentsDef =
540                                (BaseRuntimeElementCompositeDefinition<?>) entryChildDef.getChildByName("entry");
541                BaseRuntimeChildDefinition fullUrlChildDef = entryChildContentsDef.getChildByName("fullUrl");
542                BaseRuntimeChildDefinition resourceChildDef = entryChildContentsDef.getChildByName("resource");
543                BaseRuntimeChildDefinition requestChildDef = entryChildContentsDef.getChildByName("request");
544                BaseRuntimeElementCompositeDefinition<?> requestChildContentsDef =
545                                (BaseRuntimeElementCompositeDefinition<?>) requestChildDef.getChildByName("request");
546                BaseRuntimeChildDefinition requestUrlChildDef = requestChildContentsDef.getChildByName("url");
547                BaseRuntimeChildDefinition requestIfNoneExistChildDef = requestChildContentsDef.getChildByName("ifNoneExist");
548                BaseRuntimeChildDefinition methodChildDef = requestChildContentsDef.getChildByName("method");
549                Map<BundleEntryParts, IBase> map = new HashMap<>();
550                for (IBase nextEntry : entries) {
551                        BundleEntryParts parts = getBundleEntryParts(
552                                        fullUrlChildDef,
553                                        resourceChildDef,
554                                        requestChildDef,
555                                        requestUrlChildDef,
556                                        requestIfNoneExistChildDef,
557                                        methodChildDef,
558                                        nextEntry);
559                        /*
560                         * All 3 might be null - That's ok because we still want to know the
561                         * order in the original bundle.
562                         */
563                        map.put(parts, nextEntry);
564                }
565                return map;
566        }
567
568        public static List<SearchBundleEntryParts> getSearchBundleEntryParts(
569                        FhirContext theContext, IBaseBundle theBundle) {
570                RuntimeResourceDefinition bundleDef = theContext.getResourceDefinition(theBundle);
571                BaseRuntimeChildDefinition entryChildDef = bundleDef.getChildByName("entry");
572                List<IBase> entries = entryChildDef.getAccessor().getValues(theBundle);
573
574                BaseRuntimeElementCompositeDefinition<?> entryChildContentsDef =
575                                (BaseRuntimeElementCompositeDefinition<?>) entryChildDef.getChildByName("entry");
576                BaseRuntimeChildDefinition fullUrlChildDef = entryChildContentsDef.getChildByName("fullUrl");
577                BaseRuntimeChildDefinition resourceChildDef = entryChildContentsDef.getChildByName("resource");
578                BaseRuntimeChildDefinition searchChildDef = entryChildContentsDef.getChildByName("search");
579                BaseRuntimeElementCompositeDefinition<?> searchChildContentsDef =
580                                (BaseRuntimeElementCompositeDefinition<?>) searchChildDef.getChildByName("search");
581                BaseRuntimeChildDefinition searchModeChildDef = searchChildContentsDef.getChildByName("mode");
582                BaseRuntimeChildDefinition searchScoreChildDef = searchChildContentsDef.getChildByName("score");
583
584                List<SearchBundleEntryParts> retVal = new ArrayList<>();
585                for (IBase nextEntry : entries) {
586                        SearchBundleEntryParts parts = getSearchBundleEntryParts(
587                                        fullUrlChildDef,
588                                        resourceChildDef,
589                                        searchChildDef,
590                                        searchModeChildDef,
591                                        searchScoreChildDef,
592                                        nextEntry);
593                        retVal.add(parts);
594                }
595                return retVal;
596        }
597
598        private static SearchBundleEntryParts getSearchBundleEntryParts(
599                        BaseRuntimeChildDefinition theFullUrlChildDef,
600                        BaseRuntimeChildDefinition theResourceChildDef,
601                        BaseRuntimeChildDefinition theSearchChildDef,
602                        BaseRuntimeChildDefinition theSearchModeChildDef,
603                        BaseRuntimeChildDefinition theSearchScoreChildDef,
604                        IBase entry) {
605                IBaseResource resource = null;
606                String matchMode = null;
607                BigDecimal searchScore = null;
608
609                String fullUrl = theFullUrlChildDef
610                                .getAccessor()
611                                .getFirstValueOrNull(entry)
612                                .map(t -> ((IPrimitiveType<?>) t).getValueAsString())
613                                .orElse(null);
614
615                for (IBase nextResource : theResourceChildDef.getAccessor().getValues(entry)) {
616                        resource = (IBaseResource) nextResource;
617                }
618
619                for (IBase nextSearch : theSearchChildDef.getAccessor().getValues(entry)) {
620                        for (IBase nextUrl : theSearchModeChildDef.getAccessor().getValues(nextSearch)) {
621                                matchMode = ((IPrimitiveType<?>) nextUrl).getValueAsString();
622                        }
623                        for (IBase nextUrl : theSearchScoreChildDef.getAccessor().getValues(nextSearch)) {
624                                searchScore = (BigDecimal) ((IPrimitiveType<?>) nextUrl).getValue();
625                        }
626                }
627
628                return new SearchBundleEntryParts(fullUrl, resource, matchMode, searchScore);
629        }
630
631        /**
632         * Given a bundle, and a consumer, apply the consumer to each entry in the bundle.
633         * @param theContext The FHIR Context
634         * @param theBundle The bundle to have its entries processed.
635         * @param theProcessor a {@link Consumer} which will operate on all the entries of a bundle.
636         */
637        public static void processEntries(
638                        FhirContext theContext, IBaseBundle theBundle, Consumer<ModifiableBundleEntry> theProcessor) {
639                RuntimeResourceDefinition bundleDef = theContext.getResourceDefinition(theBundle);
640                BaseRuntimeChildDefinition entryChildDef = bundleDef.getChildByName("entry");
641                List<IBase> entries = entryChildDef.getAccessor().getValues(theBundle);
642
643                BaseRuntimeElementCompositeDefinition<?> entryChildContentsDef =
644                                (BaseRuntimeElementCompositeDefinition<?>) entryChildDef.getChildByName("entry");
645                BaseRuntimeChildDefinition fullUrlChildDef = entryChildContentsDef.getChildByName("fullUrl");
646                BaseRuntimeChildDefinition resourceChildDef = entryChildContentsDef.getChildByName("resource");
647                BaseRuntimeChildDefinition requestChildDef = entryChildContentsDef.getChildByName("request");
648                BaseRuntimeElementCompositeDefinition<?> requestChildContentsDef =
649                                (BaseRuntimeElementCompositeDefinition<?>) requestChildDef.getChildByName("request");
650                BaseRuntimeChildDefinition requestUrlChildDef = requestChildContentsDef.getChildByName("url");
651                BaseRuntimeChildDefinition requestIfNoneExistChildDef = requestChildContentsDef.getChildByName("ifNoneExist");
652                BaseRuntimeChildDefinition methodChildDef = requestChildContentsDef.getChildByName("method");
653
654                for (IBase nextEntry : entries) {
655                        BundleEntryParts parts = getBundleEntryParts(
656                                        fullUrlChildDef,
657                                        resourceChildDef,
658                                        requestChildDef,
659                                        requestUrlChildDef,
660                                        requestIfNoneExistChildDef,
661                                        methodChildDef,
662                                        nextEntry);
663                        /*
664                         * All 3 might be null - That's ok because we still want to know the
665                         * order in the original bundle.
666                         */
667                        BundleEntryMutator mutator = new BundleEntryMutator(
668                                        theContext, nextEntry, requestChildDef, requestChildContentsDef, entryChildContentsDef);
669                        ModifiableBundleEntry entry = new ModifiableBundleEntry(parts, mutator);
670                        theProcessor.accept(entry);
671                }
672        }
673
674        private static BundleEntryParts getBundleEntryParts(
675                        BaseRuntimeChildDefinition fullUrlChildDef,
676                        BaseRuntimeChildDefinition resourceChildDef,
677                        BaseRuntimeChildDefinition requestChildDef,
678                        BaseRuntimeChildDefinition requestUrlChildDef,
679                        BaseRuntimeChildDefinition requestIfNoneExistChildDef,
680                        BaseRuntimeChildDefinition methodChildDef,
681                        IBase nextEntry) {
682                IBaseResource resource = null;
683                String url = null;
684                RequestTypeEnum requestType = null;
685                String conditionalUrl = null;
686                String fullUrl = fullUrlChildDef
687                                .getAccessor()
688                                .getFirstValueOrNull(nextEntry)
689                                .map(t -> ((IPrimitiveType<?>) t).getValueAsString())
690                                .orElse(null);
691
692                for (IBase nextResource : resourceChildDef.getAccessor().getValues(nextEntry)) {
693                        resource = (IBaseResource) nextResource;
694                }
695                for (IBase nextRequest : requestChildDef.getAccessor().getValues(nextEntry)) {
696                        for (IBase nextUrl : requestUrlChildDef.getAccessor().getValues(nextRequest)) {
697                                url = ((IPrimitiveType<?>) nextUrl).getValueAsString();
698                        }
699                        for (IBase nextMethod : methodChildDef.getAccessor().getValues(nextRequest)) {
700                                String methodString = ((IPrimitiveType<?>) nextMethod).getValueAsString();
701                                if (isNotBlank(methodString)) {
702                                        requestType = RequestTypeEnum.valueOf(methodString);
703                                }
704                        }
705
706                        if (requestType != null) {
707                                //noinspection EnumSwitchStatementWhichMissesCases
708                                switch (requestType) {
709                                        case PUT:
710                                        case DELETE:
711                                        case PATCH:
712                                                conditionalUrl = url != null && url.contains("?") ? url : null;
713                                                break;
714                                        case POST:
715                                                List<IBase> ifNoneExistReps =
716                                                                requestIfNoneExistChildDef.getAccessor().getValues(nextRequest);
717                                                if (ifNoneExistReps.size() > 0) {
718                                                        IPrimitiveType<?> ifNoneExist = (IPrimitiveType<?>) ifNoneExistReps.get(0);
719                                                        conditionalUrl = ifNoneExist.getValueAsString();
720                                                }
721                                                break;
722                                }
723                        }
724                }
725                return new BundleEntryParts(fullUrl, requestType, url, resource, conditionalUrl);
726        }
727
728        /**
729         * Extract all of the resources from a given bundle
730         */
731        public static List<IBaseResource> toListOfResources(FhirContext theContext, IBaseBundle theBundle) {
732                return toListOfResourcesOfType(theContext, theBundle, IBaseResource.class);
733        }
734
735        /**
736         * Extract all of ids of all the resources from a given bundle
737         */
738        public static List<String> toListOfResourceIds(FhirContext theContext, IBaseBundle theBundle) {
739                return toListOfResourcesOfType(theContext, theBundle, IBaseResource.class).stream()
740                                .map(resource -> resource.getIdElement().getIdPart())
741                                .collect(Collectors.toList());
742        }
743
744        /**
745         * Extract all of the resources of a given type from a given bundle
746         */
747        @SuppressWarnings("unchecked")
748        public static <T extends IBaseResource> List<T> toListOfResourcesOfType(
749                        FhirContext theContext, IBaseBundle theBundle, Class<T> theTypeToInclude) {
750                Objects.requireNonNull(theTypeToInclude, "ResourceType must not be null");
751                List<T> retVal = new ArrayList<>();
752
753                RuntimeResourceDefinition def = theContext.getResourceDefinition(theBundle);
754                BaseRuntimeChildDefinition entryChild = def.getChildByName("entry");
755                List<IBase> entries = entryChild.getAccessor().getValues(theBundle);
756
757                BaseRuntimeElementCompositeDefinition<?> entryChildElem =
758                                (BaseRuntimeElementCompositeDefinition<?>) entryChild.getChildByName("entry");
759                BaseRuntimeChildDefinition resourceChild = entryChildElem.getChildByName("resource");
760                for (IBase nextEntry : entries) {
761                        for (IBase next : resourceChild.getAccessor().getValues(nextEntry)) {
762                                if (theTypeToInclude.isAssignableFrom(next.getClass())) {
763                                        retVal.add((T) next);
764                                }
765                        }
766                }
767                return retVal;
768        }
769
770        public static IBase getReferenceInBundle(
771                        @Nonnull FhirContext theFhirContext, @Nonnull String theUrl, @Nullable Object theAppContext) {
772                if (!(theAppContext instanceof IBaseBundle) || isBlank(theUrl) || theUrl.startsWith("#")) {
773                        return null;
774                }
775
776                /*
777                 * If this is a reference that is a UUID, we must be looking for local references within a Bundle
778                 */
779                IBaseBundle bundle = (IBaseBundle) theAppContext;
780
781                final boolean isPlaceholderReference = theUrl.startsWith("urn:");
782                final String unqualifiedVersionlessReference =
783                                new IdDt(theUrl).toUnqualifiedVersionless().getValue();
784
785                for (BundleEntryParts next : BundleUtil.toListOfEntries(theFhirContext, bundle)) {
786                        IBaseResource nextResource = next.getResource();
787                        if (nextResource == null) {
788                                continue;
789                        }
790                        if (isPlaceholderReference) {
791                                if (theUrl.equals(next.getFullUrl())
792                                                || theUrl.equals(nextResource.getIdElement().getValue())) {
793                                        return nextResource;
794                                }
795                        } else {
796                                if (unqualifiedVersionlessReference.equals(
797                                                nextResource.getIdElement().toUnqualifiedVersionless().getValue())) {
798                                        return nextResource;
799                                }
800                        }
801                }
802                return null;
803        }
804
805        /**
806         * DSTU3 did not allow the PATCH verb for transaction bundles- so instead we infer that a bundle
807         * is a patch if the payload is a binary resource containing a patch. This method
808         * tests whether a resource (which should have come from
809         * <code>Bundle.entry.resource</code> is a Binary resource with a patch
810         * payload type.
811         */
812        public static boolean isDstu3TransactionPatch(FhirContext theContext, IBaseResource thePayloadResource) {
813                boolean isPatch = false;
814                if (thePayloadResource instanceof IBaseBinary) {
815                        String contentType = ((IBaseBinary) thePayloadResource).getContentType();
816                        try {
817                                PatchTypeEnum.forContentTypeOrThrowInvalidRequestException(theContext, contentType);
818                                isPatch = true;
819                        } catch (InvalidRequestException e) {
820                                // ignore
821                        }
822                }
823                return isPatch;
824        }
825
826        /**
827         * create a new bundle entry and set a value for a single field
828         * @param theContext     Context holding resource definition
829         * @param theFieldName   Child field name of the bundle entry to set
830         * @param theValues      The values to set on the bundle entry child field name
831         * @return the new bundle entry
832         */
833        public static IBase createNewBundleEntryWithSingleField(
834                        FhirContext theContext, String theFieldName, IBase... theValues) {
835                IBaseBundle newBundle = TerserUtil.newResource(theContext, "Bundle");
836                BaseRuntimeChildDefinition entryChildDef =
837                                theContext.getResourceDefinition(newBundle).getChildByName("entry");
838
839                BaseRuntimeElementCompositeDefinition<?> entryChildElem =
840                                (BaseRuntimeElementCompositeDefinition<?>) entryChildDef.getChildByName("entry");
841                BaseRuntimeChildDefinition resourceChild = entryChildElem.getChildByName(theFieldName);
842                IBase bundleEntry = entryChildElem.newInstance();
843                for (IBase value : theValues) {
844                        try {
845                                resourceChild.getMutator().addValue(bundleEntry, value);
846                        } catch (UnsupportedOperationException e) {
847                                ourLog.warn(
848                                                "Resource {} does not support multiple values, but an attempt to set {} was made. Setting the first item only",
849                                                bundleEntry,
850                                                theValues);
851                                resourceChild.getMutator().setValue(bundleEntry, value);
852                                break;
853                        }
854                }
855                return bundleEntry;
856        }
857
858        /**
859         * Get resource from bundle by resource type and reference
860         * @param theContext   FhirContext
861         * @param theBundle    IBaseBundle
862         * @param theReference IBaseReference
863         * @return IBaseResource if found and null if not found.
864         */
865        @Nonnull
866        public static IBaseResource getResourceByReferenceAndResourceType(
867                        @Nonnull FhirContext theContext, @Nonnull IBaseBundle theBundle, @Nonnull IBaseReference theReference) {
868                return toListOfResources(theContext, theBundle).stream()
869                                .filter(theResource -> theReference
870                                                .getReferenceElement()
871                                                .getIdPart()
872                                                .equals(theResource.getIdElement().getIdPart()))
873                                .findFirst()
874                                .orElse(null);
875        }
876
877        private static class SortLegality {
878                private boolean myIsLegal;
879
880                SortLegality() {
881                        this.myIsLegal = true;
882                }
883
884                private void setLegal(boolean theLegal) {
885                        myIsLegal = theLegal;
886                }
887
888                public boolean isLegal() {
889                        return myIsLegal;
890                }
891        }
892}