001/*-
002 * #%L
003 * HAPI FHIR Storage api
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.jpa.patch;
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.i18n.Msg;
027import ca.uhn.fhir.parser.path.EncodeContextPath;
028import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
029import ca.uhn.fhir.util.IModelVisitor2;
030import ca.uhn.fhir.util.ParametersUtil;
031import jakarta.annotation.Nonnull;
032import jakarta.annotation.Nullable;
033import org.apache.commons.lang3.StringUtils;
034import org.apache.commons.lang3.Validate;
035import org.hl7.fhir.instance.model.api.IBase;
036import org.hl7.fhir.instance.model.api.IBaseEnumeration;
037import org.hl7.fhir.instance.model.api.IBaseParameters;
038import org.hl7.fhir.instance.model.api.IBaseResource;
039import org.hl7.fhir.instance.model.api.IIdType;
040import org.hl7.fhir.instance.model.api.IPrimitiveType;
041import org.hl7.fhir.utilities.xhtml.XhtmlNode;
042
043import java.util.ArrayList;
044import java.util.Collections;
045import java.util.HashSet;
046import java.util.List;
047import java.util.Objects;
048import java.util.Optional;
049import java.util.Set;
050
051import static org.apache.commons.lang3.StringUtils.defaultString;
052
053public class FhirPatch {
054
055        public static final String OPERATION_ADD = "add";
056        public static final String OPERATION_DELETE = "delete";
057        public static final String OPERATION_INSERT = "insert";
058        public static final String OPERATION_MOVE = "move";
059        public static final String OPERATION_REPLACE = "replace";
060        public static final String PARAMETER_DESTINATION = "destination";
061        public static final String PARAMETER_INDEX = "index";
062        public static final String PARAMETER_NAME = "name";
063        public static final String PARAMETER_OPERATION = "operation";
064        public static final String PARAMETER_PATH = "path";
065        public static final String PARAMETER_SOURCE = "source";
066        public static final String PARAMETER_TYPE = "type";
067        public static final String PARAMETER_VALUE = "value";
068
069        private final FhirContext myContext;
070        private boolean myIncludePreviousValueInDiff;
071        private Set<EncodeContextPath> myIgnorePaths = Collections.emptySet();
072
073        public FhirPatch(FhirContext theContext) {
074                myContext = theContext;
075        }
076
077        /**
078         * Adds a path element that will not be included in generated diffs. Values can take the form
079         * <code>ResourceName.fieldName.fieldName</code> and wildcards are supported, such
080         * as <code>*.meta</code>.
081         */
082        public void addIgnorePath(String theIgnorePath) {
083                Validate.notBlank(theIgnorePath, "theIgnorePath must not be null or empty");
084
085                if (myIgnorePaths.isEmpty()) {
086                        myIgnorePaths = new HashSet<>();
087                }
088                myIgnorePaths.add(new EncodeContextPath(theIgnorePath));
089        }
090
091        public void setIncludePreviousValueInDiff(boolean theIncludePreviousValueInDiff) {
092                myIncludePreviousValueInDiff = theIncludePreviousValueInDiff;
093        }
094
095        public void apply(IBaseResource theResource, IBaseResource thePatch) {
096
097                List<IBase> opParameters = ParametersUtil.getNamedParameters(myContext, thePatch, PARAMETER_OPERATION);
098                for (IBase nextOperation : opParameters) {
099                        String type = ParametersUtil.getParameterPartValueAsString(myContext, nextOperation, PARAMETER_TYPE);
100                        type = defaultString(type);
101
102                        if (OPERATION_DELETE.equals(type)) {
103                                handleDeleteOperation(theResource, nextOperation);
104                        } else if (OPERATION_ADD.equals(type)) {
105                                handleAddOperation(theResource, nextOperation);
106                        } else if (OPERATION_REPLACE.equals(type)) {
107                                handleReplaceOperation(theResource, nextOperation);
108                        } else if (OPERATION_INSERT.equals(type)) {
109                                handleInsertOperation(theResource, nextOperation);
110                        } else if (OPERATION_MOVE.equals(type)) {
111                                handleMoveOperation(theResource, nextOperation);
112                        } else {
113                                throw new InvalidRequestException(Msg.code(1267) + "Unknown patch operation type: " + type);
114                        }
115                }
116        }
117
118        private void handleAddOperation(IBaseResource theResource, IBase theParameters) {
119
120                String path = ParametersUtil.getParameterPartValueAsString(myContext, theParameters, PARAMETER_PATH);
121                String elementName = ParametersUtil.getParameterPartValueAsString(myContext, theParameters, PARAMETER_NAME);
122
123                String containingPath = defaultString(path);
124
125                List<IBase> containingElements = myContext.newFhirPath().evaluate(theResource, containingPath, IBase.class);
126                for (IBase nextElement : containingElements) {
127                        ChildDefinition childDefinition = findChildDefinition(nextElement, elementName);
128
129                        IBase newValue = getNewValue(theParameters, nextElement, childDefinition);
130
131                        childDefinition.getChildDef().getMutator().addValue(nextElement, newValue);
132                }
133        }
134
135        private void handleInsertOperation(IBaseResource theResource, IBase theParameters) {
136
137                String path = ParametersUtil.getParameterPartValueAsString(myContext, theParameters, PARAMETER_PATH);
138                path = defaultString(path);
139
140                int lastDot = path.lastIndexOf(".");
141                String containingPath = path.substring(0, lastDot);
142                String elementName = path.substring(lastDot + 1);
143                Integer insertIndex = ParametersUtil.getParameterPartValueAsInteger(myContext, theParameters, PARAMETER_INDEX)
144                                .orElseThrow(() -> new InvalidRequestException("No index supplied for insert operation"));
145
146                List<IBase> containingElements = myContext.newFhirPath().evaluate(theResource, containingPath, IBase.class);
147                for (IBase nextElement : containingElements) {
148
149                        ChildDefinition childDefinition = findChildDefinition(nextElement, elementName);
150
151                        IBase newValue = getNewValue(theParameters, nextElement, childDefinition);
152
153                        List<IBase> existingValues =
154                                        new ArrayList<>(childDefinition.getChildDef().getAccessor().getValues(nextElement));
155                        if (insertIndex == null || insertIndex < 0 || insertIndex > existingValues.size()) {
156                                String msg = myContext
157                                                .getLocalizer()
158                                                .getMessage(FhirPatch.class, "invalidInsertIndex", insertIndex, path, existingValues.size());
159                                throw new InvalidRequestException(Msg.code(1270) + msg);
160                        }
161                        existingValues.add(insertIndex, newValue);
162
163                        childDefinition.getChildDef().getMutator().setValue(nextElement, null);
164                        for (IBase nextNewValue : existingValues) {
165                                childDefinition.getChildDef().getMutator().addValue(nextElement, nextNewValue);
166                        }
167                }
168        }
169
170        private void handleDeleteOperation(IBaseResource theResource, IBase theParameters) {
171
172                String path = ParametersUtil.getParameterPartValueAsString(myContext, theParameters, PARAMETER_PATH);
173                path = defaultString(path);
174
175                String containingPath;
176                String elementName;
177
178                if (path.endsWith(")")) {
179                        // This is probably a filter, so we're probably dealing with a list
180                        int filterArgsIndex = path.lastIndexOf('('); // Let's hope there aren't nested parentheses
181                        int lastDotIndex = path.lastIndexOf(
182                                        '.', filterArgsIndex); // There might be a dot inside the parentheses, so look to the left of that
183                        int secondLastDotIndex = path.lastIndexOf('.', lastDotIndex - 1);
184                        containingPath = path.substring(0, secondLastDotIndex);
185                        elementName = path.substring(secondLastDotIndex + 1, lastDotIndex);
186                } else if (path.endsWith("]")) {
187                        // This is almost definitely a list
188                        int openBracketIndex = path.lastIndexOf('[');
189                        int lastDotIndex = path.lastIndexOf('.', openBracketIndex);
190                        containingPath = path.substring(0, lastDotIndex);
191                        elementName = path.substring(lastDotIndex + 1, openBracketIndex);
192                } else {
193                        containingPath = path;
194                        elementName = null;
195                }
196
197                List<IBase> containingElements = myContext.newFhirPath().evaluate(theResource, containingPath, IBase.class);
198                for (IBase nextElement : containingElements) {
199                        if (elementName == null) {
200                                deleteSingleElement(nextElement);
201                        } else {
202                                deleteFromList(theResource, nextElement, elementName, path);
203                        }
204                }
205        }
206
207        private void deleteFromList(
208                        IBaseResource theResource,
209                        IBase theContainingElement,
210                        String theListElementName,
211                        String theElementToDeletePath) {
212                ChildDefinition childDefinition = findChildDefinition(theContainingElement, theListElementName);
213
214                List<IBase> existingValues =
215                                new ArrayList<>(childDefinition.getChildDef().getAccessor().getValues(theContainingElement));
216                List<IBase> elementsToRemove =
217                                myContext.newFhirPath().evaluate(theResource, theElementToDeletePath, IBase.class);
218                existingValues.removeAll(elementsToRemove);
219
220                childDefinition.getChildDef().getMutator().setValue(theContainingElement, null);
221                for (IBase nextNewValue : existingValues) {
222                        childDefinition.getChildDef().getMutator().addValue(theContainingElement, nextNewValue);
223                }
224        }
225
226        private void handleReplaceOperation(IBaseResource theResource, IBase theParameters) {
227                String path = ParametersUtil.getParameterPartValueAsString(myContext, theParameters, PARAMETER_PATH);
228                path = defaultString(path);
229
230                int lastDot = path.lastIndexOf(".");
231                String containingPath = path.substring(0, lastDot);
232                String elementName = path.substring(lastDot + 1);
233
234                List<IBase> containingElements = myContext.newFhirPath().evaluate(theResource, containingPath, IBase.class);
235                for (IBase nextElement : containingElements) {
236
237                        ChildDefinition childDefinition = findChildDefinition(nextElement, elementName);
238
239                        IBase newValue = getNewValue(theParameters, nextElement, childDefinition);
240
241                        childDefinition.getChildDef().getMutator().setValue(nextElement, newValue);
242                }
243        }
244
245        private void handleMoveOperation(IBaseResource theResource, IBase theParameters) {
246                String path = ParametersUtil.getParameterPartValueAsString(myContext, theParameters, PARAMETER_PATH);
247                path = defaultString(path);
248
249                int lastDot = path.lastIndexOf(".");
250                String containingPath = path.substring(0, lastDot);
251                String elementName = path.substring(lastDot + 1);
252                Integer insertIndex = ParametersUtil.getParameterPartValueAsInteger(
253                                                myContext, theParameters, PARAMETER_DESTINATION)
254                                .orElseThrow(() -> new InvalidRequestException("No index supplied for move operation"));
255                Integer removeIndex = ParametersUtil.getParameterPartValueAsInteger(myContext, theParameters, PARAMETER_SOURCE)
256                                .orElseThrow(() -> new InvalidRequestException("No index supplied for move operation"));
257
258                List<IBase> containingElements = myContext.newFhirPath().evaluate(theResource, containingPath, IBase.class);
259                for (IBase nextElement : containingElements) {
260
261                        ChildDefinition childDefinition = findChildDefinition(nextElement, elementName);
262
263                        List<IBase> existingValues =
264                                        new ArrayList<>(childDefinition.getChildDef().getAccessor().getValues(nextElement));
265                        if (removeIndex == null || removeIndex < 0 || removeIndex >= existingValues.size()) {
266                                String msg = myContext
267                                                .getLocalizer()
268                                                .getMessage(
269                                                                FhirPatch.class, "invalidMoveSourceIndex", removeIndex, path, existingValues.size());
270                                throw new InvalidRequestException(Msg.code(1268) + msg);
271                        }
272                        IBase newValue = existingValues.remove(removeIndex.intValue());
273
274                        if (insertIndex == null || insertIndex < 0 || insertIndex > existingValues.size()) {
275                                String msg = myContext
276                                                .getLocalizer()
277                                                .getMessage(
278                                                                FhirPatch.class,
279                                                                "invalidMoveDestinationIndex",
280                                                                insertIndex,
281                                                                path,
282                                                                existingValues.size());
283                                throw new InvalidRequestException(Msg.code(1269) + msg);
284                        }
285                        existingValues.add(insertIndex, newValue);
286
287                        childDefinition.getChildDef().getMutator().setValue(nextElement, null);
288                        for (IBase nextNewValue : existingValues) {
289                                childDefinition.getChildDef().getMutator().addValue(nextElement, nextNewValue);
290                        }
291                }
292        }
293
294        private ChildDefinition findChildDefinition(IBase theContainingElement, String theElementName) {
295                BaseRuntimeElementDefinition<?> elementDef = myContext.getElementDefinition(theContainingElement.getClass());
296
297                String childName = theElementName;
298                BaseRuntimeChildDefinition childDef = elementDef.getChildByName(childName);
299                BaseRuntimeElementDefinition<?> childElement;
300                if (childDef == null) {
301                        childName = theElementName + "[x]";
302                        childDef = elementDef.getChildByName(childName);
303                        childElement = childDef.getChildByName(
304                                        childDef.getValidChildNames().iterator().next());
305                } else {
306                        childElement = childDef.getChildByName(childName);
307                }
308
309                return new ChildDefinition(childDef, childElement);
310        }
311
312        private IBase getNewValue(IBase theParameters, IBase theElement, ChildDefinition theChildDefinition) {
313                Optional<IBase> valuePart = ParametersUtil.getParameterPart(myContext, theParameters, PARAMETER_VALUE);
314                Optional<IBase> valuePartValue =
315                                ParametersUtil.getParameterPartValue(myContext, theParameters, PARAMETER_VALUE);
316
317                IBase newValue;
318                if (valuePartValue.isPresent()) {
319                        newValue = valuePartValue.get();
320                } else {
321                        List<IBase> partParts = valuePart.map(this::extractPartsFromPart).orElse(Collections.emptyList());
322
323                        newValue = createAndPopulateNewElement(theChildDefinition, partParts);
324                }
325
326                if (IBaseEnumeration.class.isAssignableFrom(
327                                                theChildDefinition.getChildElement().getImplementingClass())
328                                || XhtmlNode.class.isAssignableFrom(
329                                                theChildDefinition.getChildElement().getImplementingClass())) {
330                        // If the compositeElementDef is an IBaseEnumeration, we will use the actual compositeElementDef definition
331                        // to build one, since
332                        // it needs the right factory object passed to its constructor
333                        IPrimitiveType<?> newValueInstance;
334                        if (theChildDefinition.getChildDef().getInstanceConstructorArguments() != null) {
335                                newValueInstance = (IPrimitiveType<?>) theChildDefinition
336                                                .getChildElement()
337                                                .newInstance(theChildDefinition.getChildDef().getInstanceConstructorArguments());
338                        } else {
339                                newValueInstance =
340                                                (IPrimitiveType<?>) theChildDefinition.getChildElement().newInstance();
341                        }
342                        newValueInstance.setValueAsString(((IPrimitiveType<?>) newValue).getValueAsString());
343                        theChildDefinition.getChildDef().getMutator().setValue(theElement, newValueInstance);
344                        newValue = newValueInstance;
345                }
346                return newValue;
347        }
348
349        @Nonnull
350        private List<IBase> extractPartsFromPart(IBase theParametersParameterComponent) {
351                return myContext.newTerser().getValues(theParametersParameterComponent, "part");
352        }
353
354        /**
355         * this method will instantiate an element according to the provided Definition and it according to
356         * the properties found in thePartParts.  a part usually represent a datatype as a name/value[X] pair.
357         * it may also represent a complex type like an Extension.
358         *
359         * @param theDefinition wrapper around the runtime definition of the element to be populated
360         * @param thePartParts list of Part to populate the element that will be created from theDefinition
361         * @return an element that was created from theDefinition and populated with the parts
362         */
363        private IBase createAndPopulateNewElement(ChildDefinition theDefinition, List<IBase> thePartParts) {
364                IBase newElement = theDefinition.getChildElement().newInstance();
365
366                for (IBase nextValuePartPart : thePartParts) {
367
368                        String name = myContext
369                                        .newTerser()
370                                        .getSingleValue(nextValuePartPart, PARAMETER_NAME, IPrimitiveType.class)
371                                        .map(IPrimitiveType::getValueAsString)
372                                        .orElse(null);
373
374                        if (StringUtils.isBlank(name)) {
375                                continue;
376                        }
377
378                        Optional<IBase> value = myContext.newTerser().getSingleValue(nextValuePartPart, "value[x]", IBase.class);
379
380                        if (value.isPresent()) {
381                                // we have a dataType. let's extract its value and assign it.
382                                BaseRuntimeChildDefinition partChildDef =
383                                                theDefinition.getChildElement().getChildByName(name);
384                                if (partChildDef == null) {
385                                        name = name + "[x]";
386                                        partChildDef = theDefinition.getChildElement().getChildByName(name);
387                                }
388                                partChildDef.getMutator().addValue(newElement, value.get());
389
390                                // a part represent a datatype or a complexType but not both at the same time.
391                                continue;
392                        }
393
394                        List<IBase> part = extractPartsFromPart(nextValuePartPart);
395
396                        if (!part.isEmpty()) {
397                                // we have a complexType.  let's find its definition and recursively process
398                                // them till all complexTypes are processed.
399                                ChildDefinition childDefinition = findChildDefinition(newElement, name);
400
401                                IBase childNewValue = createAndPopulateNewElement(childDefinition, part);
402
403                                childDefinition.getChildDef().getMutator().setValue(newElement, childNewValue);
404                        }
405                }
406
407                return newElement;
408        }
409
410        private void deleteSingleElement(IBase theElementToDelete) {
411                myContext.newTerser().visit(theElementToDelete, new IModelVisitor2() {
412                        @Override
413                        public boolean acceptElement(
414                                        IBase theElement,
415                                        List<IBase> theContainingElementPath,
416                                        List<BaseRuntimeChildDefinition> theChildDefinitionPath,
417                                        List<BaseRuntimeElementDefinition<?>> theElementDefinitionPath) {
418                                if (theElement instanceof IPrimitiveType) {
419                                        ((IPrimitiveType<?>) theElement).setValueAsString(null);
420                                }
421                                return true;
422                        }
423                });
424        }
425
426        public IBaseParameters diff(@Nullable IBaseResource theOldValue, @Nonnull IBaseResource theNewValue) {
427                IBaseParameters retVal = ParametersUtil.newInstance(myContext);
428                String newValueTypeName = myContext.getResourceDefinition(theNewValue).getName();
429
430                if (theOldValue == null) {
431
432                        IBase operation = ParametersUtil.addParameterToParameters(myContext, retVal, PARAMETER_OPERATION);
433                        ParametersUtil.addPartCode(myContext, operation, PARAMETER_TYPE, OPERATION_INSERT);
434                        ParametersUtil.addPartString(myContext, operation, PARAMETER_PATH, newValueTypeName);
435                        ParametersUtil.addPart(myContext, operation, PARAMETER_VALUE, theNewValue);
436
437                } else {
438
439                        String oldValueTypeName =
440                                        myContext.getResourceDefinition(theOldValue).getName();
441                        Validate.isTrue(oldValueTypeName.equalsIgnoreCase(newValueTypeName), "Resources must be of same type");
442
443                        BaseRuntimeElementCompositeDefinition<?> def =
444                                        myContext.getResourceDefinition(theOldValue).getBaseDefinition();
445                        String path = def.getName();
446
447                        EncodeContextPath contextPath = new EncodeContextPath();
448                        contextPath.pushPath(path, true);
449
450                        compare(retVal, contextPath, def, path, path, theOldValue, theNewValue);
451
452                        contextPath.popPath();
453                        assert contextPath.getPath().isEmpty();
454                }
455
456                return retVal;
457        }
458
459        private void compare(
460                        IBaseParameters theDiff,
461                        EncodeContextPath theSourceEncodeContext,
462                        BaseRuntimeElementDefinition<?> theDef,
463                        String theSourcePath,
464                        String theTargetPath,
465                        IBase theOldField,
466                        IBase theNewField) {
467
468                boolean pathIsIgnored = pathIsIgnored(theSourceEncodeContext);
469                if (pathIsIgnored) {
470                        return;
471                }
472
473                BaseRuntimeElementDefinition<?> sourceDef = myContext.getElementDefinition(theOldField.getClass());
474                BaseRuntimeElementDefinition<?> targetDef = myContext.getElementDefinition(theNewField.getClass());
475                if (!sourceDef.getName().equals(targetDef.getName())) {
476                        IBase operation = ParametersUtil.addParameterToParameters(myContext, theDiff, PARAMETER_OPERATION);
477                        ParametersUtil.addPartCode(myContext, operation, PARAMETER_TYPE, OPERATION_REPLACE);
478                        ParametersUtil.addPartString(myContext, operation, PARAMETER_PATH, theTargetPath);
479                        addValueToDiff(operation, theOldField, theNewField);
480                } else {
481                        if (theOldField instanceof IPrimitiveType) {
482                                IPrimitiveType<?> oldPrimitive = (IPrimitiveType<?>) theOldField;
483                                IPrimitiveType<?> newPrimitive = (IPrimitiveType<?>) theNewField;
484                                String oldValueAsString = toValue(oldPrimitive);
485                                String newValueAsString = toValue(newPrimitive);
486                                if (!Objects.equals(oldValueAsString, newValueAsString)) {
487                                        IBase operation = ParametersUtil.addParameterToParameters(myContext, theDiff, PARAMETER_OPERATION);
488                                        ParametersUtil.addPartCode(myContext, operation, PARAMETER_TYPE, OPERATION_REPLACE);
489                                        ParametersUtil.addPartString(myContext, operation, PARAMETER_PATH, theTargetPath);
490                                        addValueToDiff(operation, oldPrimitive, newPrimitive);
491                                }
492                        }
493
494                        List<BaseRuntimeChildDefinition> children = theDef.getChildren();
495                        for (BaseRuntimeChildDefinition nextChild : children) {
496                                compareField(
497                                                theDiff,
498                                                theSourceEncodeContext,
499                                                theSourcePath,
500                                                theTargetPath,
501                                                theOldField,
502                                                theNewField,
503                                                nextChild);
504                        }
505                }
506        }
507
508        private void compareField(
509                        IBaseParameters theDiff,
510                        EncodeContextPath theSourceEncodePath,
511                        String theSourcePath,
512                        String theTargetPath,
513                        IBase theOldField,
514                        IBase theNewField,
515                        BaseRuntimeChildDefinition theChildDef) {
516                String elementName = theChildDef.getElementName();
517                boolean repeatable = theChildDef.getMax() != 1;
518                theSourceEncodePath.pushPath(elementName, false);
519                if (pathIsIgnored(theSourceEncodePath)) {
520                        theSourceEncodePath.popPath();
521                        return;
522                }
523
524                List<? extends IBase> sourceValues = theChildDef.getAccessor().getValues(theOldField);
525                List<? extends IBase> targetValues = theChildDef.getAccessor().getValues(theNewField);
526
527                int sourceIndex = 0;
528                int targetIndex = 0;
529                while (sourceIndex < sourceValues.size() && targetIndex < targetValues.size()) {
530
531                        IBase sourceChildField = sourceValues.get(sourceIndex);
532                        Validate.notNull(sourceChildField); // not expected to happen, but just in case
533                        BaseRuntimeElementDefinition<?> def = myContext.getElementDefinition(sourceChildField.getClass());
534                        IBase targetChildField = targetValues.get(targetIndex);
535                        Validate.notNull(targetChildField); // not expected to happen, but just in case
536                        String sourcePath = theSourcePath + "." + elementName + (repeatable ? "[" + sourceIndex + "]" : "");
537                        String targetPath = theSourcePath + "." + elementName + (repeatable ? "[" + targetIndex + "]" : "");
538
539                        compare(theDiff, theSourceEncodePath, def, sourcePath, targetPath, sourceChildField, targetChildField);
540
541                        sourceIndex++;
542                        targetIndex++;
543                }
544
545                // Find newly inserted items
546                while (targetIndex < targetValues.size()) {
547                        String path = theTargetPath + "." + elementName;
548                        addInsertItems(theDiff, targetValues, targetIndex, path, theChildDef);
549                        targetIndex++;
550                }
551
552                // Find deleted items
553                while (sourceIndex < sourceValues.size()) {
554                        IBase operation = ParametersUtil.addParameterToParameters(myContext, theDiff, PARAMETER_OPERATION);
555                        ParametersUtil.addPartCode(myContext, operation, PARAMETER_TYPE, OPERATION_DELETE);
556                        ParametersUtil.addPartString(
557                                        myContext,
558                                        operation,
559                                        PARAMETER_PATH,
560                                        theTargetPath + "." + elementName + (repeatable ? "[" + targetIndex + "]" : ""));
561
562                        sourceIndex++;
563                        targetIndex++;
564                }
565
566                theSourceEncodePath.popPath();
567        }
568
569        private void addInsertItems(
570                        IBaseParameters theDiff,
571                        List<? extends IBase> theTargetValues,
572                        int theTargetIndex,
573                        String thePath,
574                        BaseRuntimeChildDefinition theChildDefinition) {
575                IBase operation = ParametersUtil.addParameterToParameters(myContext, theDiff, PARAMETER_OPERATION);
576                ParametersUtil.addPartCode(myContext, operation, PARAMETER_TYPE, OPERATION_INSERT);
577                ParametersUtil.addPartString(myContext, operation, PARAMETER_PATH, thePath);
578                ParametersUtil.addPartInteger(myContext, operation, PARAMETER_INDEX, theTargetIndex);
579
580                IBase value = theTargetValues.get(theTargetIndex);
581                BaseRuntimeElementDefinition<?> valueDef = myContext.getElementDefinition(value.getClass());
582
583                /*
584                 * If the value is a Resource or a datatype, we can put it into the part.value and that will cover
585                 * all of its children. If it's an infrastructure element though, such as Patient.contact we can't
586                 * just put it into part.value because it isn't an actual type. So we have to put all of its
587                 * children in instead.
588                 */
589                if (valueDef.isStandardType()) {
590                        ParametersUtil.addPart(myContext, operation, PARAMETER_VALUE, value);
591                } else {
592                        for (BaseRuntimeChildDefinition nextChild : valueDef.getChildren()) {
593                                List<IBase> childValues = nextChild.getAccessor().getValues(value);
594                                for (int index = 0; index < childValues.size(); index++) {
595                                        boolean childRepeatable = theChildDefinition.getMax() != 1;
596                                        String elementName = nextChild.getChildNameByDatatype(
597                                                        childValues.get(index).getClass());
598                                        String targetPath = thePath + (childRepeatable ? "[" + index + "]" : "") + "." + elementName;
599                                        addInsertItems(theDiff, childValues, index, targetPath, nextChild);
600                                }
601                        }
602                }
603        }
604
605        private void addValueToDiff(IBase theOperationPart, IBase theOldValue, IBase theNewValue) {
606
607                if (myIncludePreviousValueInDiff) {
608                        IBase oldValue = massageValueForDiff(theOldValue);
609                        ParametersUtil.addPart(myContext, theOperationPart, "previousValue", oldValue);
610                }
611
612                IBase newValue = massageValueForDiff(theNewValue);
613                ParametersUtil.addPart(myContext, theOperationPart, PARAMETER_VALUE, newValue);
614        }
615
616        private boolean pathIsIgnored(EncodeContextPath theSourceEncodeContext) {
617                boolean pathIsIgnored = false;
618                for (EncodeContextPath next : myIgnorePaths) {
619                        if (theSourceEncodeContext.startsWith(next, false)) {
620                                pathIsIgnored = true;
621                                break;
622                        }
623                }
624                return pathIsIgnored;
625        }
626
627        private IBase massageValueForDiff(IBase theNewValue) {
628                IBase massagedValue = theNewValue;
629
630                // XHTML content is dealt with by putting it in a string
631                if (theNewValue instanceof XhtmlNode) {
632                        String xhtmlString = ((XhtmlNode) theNewValue).getValueAsString();
633                        massagedValue = myContext.getElementDefinition("string").newInstance(xhtmlString);
634                }
635
636                // IIdType can hold a fully qualified ID, but we just want the ID part to show up in diffs
637                if (theNewValue instanceof IIdType) {
638                        String idPart = ((IIdType) theNewValue).getIdPart();
639                        massagedValue = myContext.getElementDefinition("id").newInstance(idPart);
640                }
641
642                return massagedValue;
643        }
644
645        private String toValue(IPrimitiveType<?> theOldPrimitive) {
646                if (theOldPrimitive instanceof IIdType) {
647                        return ((IIdType) theOldPrimitive).getIdPart();
648                }
649                return theOldPrimitive.getValueAsString();
650        }
651
652        private static class ChildDefinition {
653                private final BaseRuntimeChildDefinition myChildDef;
654                private final BaseRuntimeElementDefinition<?> myChildElement;
655
656                public ChildDefinition(
657                                BaseRuntimeChildDefinition theChildDef, BaseRuntimeElementDefinition<?> theChildElement) {
658                        this.myChildDef = theChildDef;
659                        this.myChildElement = theChildElement;
660                }
661
662                public BaseRuntimeChildDefinition getChildDef() {
663                        return myChildDef;
664                }
665
666                public BaseRuntimeElementDefinition<?> getChildElement() {
667                        return myChildElement;
668                }
669        }
670}