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.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import jakarta.annotation.Nonnull;
025import org.apache.commons.lang3.StringUtils;
026import org.apache.commons.lang3.Validate;
027import org.hl7.fhir.instance.model.api.IBase;
028import org.hl7.fhir.instance.model.api.IBaseDatatype;
029import org.hl7.fhir.instance.model.api.IBaseExtension;
030import org.hl7.fhir.instance.model.api.IBaseHasExtensions;
031import org.hl7.fhir.instance.model.api.IPrimitiveType;
032
033import java.util.Collections;
034import java.util.List;
035import java.util.Optional;
036import java.util.function.Predicate;
037import java.util.stream.Collectors;
038
039/**
040 * Utility for modifying with extensions in a FHIR version-independent approach.
041 */
042public class ExtensionUtil {
043
044        /**
045         * Non instantiable
046         */
047        private ExtensionUtil() {
048                // nothing
049        }
050
051        /**
052         * Returns an extension with the specified URL creating one if it doesn't exist.
053         *
054         * @param theBase Base resource to get extension from
055         * @param theUrl  URL for the extension
056         * @return Returns a extension with the specified URL.
057         * @throws IllegalArgumentException IllegalArgumentException is thrown in case resource doesn't support extensions
058         */
059        public static IBaseExtension<?, ?> getOrCreateExtension(IBase theBase, String theUrl) {
060                IBaseHasExtensions baseHasExtensions = validateExtensionSupport(theBase);
061                IBaseExtension<?, ?> extension = getExtensionByUrl(baseHasExtensions, theUrl);
062                if (extension == null) {
063                        extension = baseHasExtensions.addExtension();
064                        extension.setUrl(theUrl);
065                }
066                return extension;
067        }
068
069        /**
070         * Returns an new empty extension.
071         *
072         * @param theBase Base resource to add the extension to
073         * @return Returns a new extension
074         * @throws IllegalArgumentException IllegalArgumentException is thrown in case resource doesn't support extensions
075         */
076        public static IBaseExtension<?, ?> addExtension(IBase theBase) {
077                return addExtension(theBase, null);
078        }
079
080        /**
081         * Returns an extension with the specified URL
082         *
083         * @param theBase Base resource to add the extension to
084         * @param theUrl  URL for the extension
085         * @return Returns a new extension with the specified URL.
086         * @throws IllegalArgumentException IllegalArgumentException is thrown in case resource doesn't support extensions
087         */
088        public static IBaseExtension<?, ?> addExtension(IBase theBase, String theUrl) {
089                IBaseHasExtensions baseHasExtensions = validateExtensionSupport(theBase);
090                IBaseExtension<?, ?> extension = baseHasExtensions.addExtension();
091                if (theUrl != null) {
092                        extension.setUrl(theUrl);
093                }
094                return extension;
095        }
096
097        /**
098         * Adds an extension with the specified value
099         *
100         * @param theBase        The resource to update extension on
101         * @param theUrl         Extension URL
102         * @param theValueType   Type of the value to set in the extension
103         * @param theValue       Extension value
104         * @param theFhirContext The context containing FHIR resource definitions
105         */
106        public static void addExtension(
107                        FhirContext theFhirContext, IBase theBase, String theUrl, String theValueType, Object theValue) {
108                IBaseExtension<?, ?> ext = addExtension(theBase, theUrl);
109                setExtension(theFhirContext, ext, theValueType, theValue);
110        }
111
112        private static IBaseHasExtensions validateExtensionSupport(IBase theBase) {
113                if (!(theBase instanceof IBaseHasExtensions)) {
114                        throw new IllegalArgumentException(
115                                        Msg.code(1747) + String.format("Expected instance that supports extensions, but got %s", theBase));
116                }
117                return (IBaseHasExtensions) theBase;
118        }
119
120        /**
121         * Checks if the specified instance has an extension with the specified URL
122         *
123         * @param theBase         The base resource to check extensions on
124         * @param theExtensionUrl URL of the extension
125         * @return Returns true if extension is exists and false otherwise
126         */
127        public static boolean hasExtension(IBase theBase, String theExtensionUrl) {
128                IBaseHasExtensions baseHasExtensions;
129                try {
130                        baseHasExtensions = validateExtensionSupport(theBase);
131                } catch (Exception e) {
132                        return false;
133                }
134
135                return getExtensionByUrl(baseHasExtensions, theExtensionUrl) != null;
136        }
137
138        /**
139         * Checks if the specified instance has an extension with the specified URL
140         *
141         * @param theBase         The base resource to check extensions on
142         * @param theExtensionUrl URL of the extension
143         * @return Returns true if extension is exists and false otherwise
144         */
145        public static boolean hasExtension(IBase theBase, String theExtensionUrl, String theExtensionValue) {
146                if (!hasExtension(theBase, theExtensionUrl)) {
147                        return false;
148                }
149                IBaseDatatype value = getExtensionByUrl(theBase, theExtensionUrl).getValue();
150                if (value == null) {
151                        return theExtensionValue == null;
152                }
153                return value.toString().equals(theExtensionValue);
154        }
155
156        /**
157         * Gets the first extension with the specified URL
158         *
159         * @param theBase         The resource to get the extension for
160         * @param theExtensionUrl URL of the extension to get. Must be non-null
161         * @return Returns the first available extension with the specified URL, or null if such extension doesn't exist
162         */
163        public static IBaseExtension<?, ?> getExtensionByUrl(IBase theBase, String theExtensionUrl) {
164                Predicate<IBaseExtension<?, ?>> filter;
165                if (theExtensionUrl == null) {
166                        filter = (e -> true);
167                } else {
168                        filter = (e -> theExtensionUrl.equals(e.getUrl()));
169                }
170
171                return getExtensionsMatchingPredicate(theBase, filter).stream()
172                                .findFirst()
173                                .orElse(null);
174        }
175
176        /**
177         * Given a resource or other structure that can have direct extensions,
178         * pulls out any extensions that have the given theExtensionUrl and a primitive value type,
179         * and returns a list of the string version of the extension values.
180         */
181        public static List<String> getExtensionPrimitiveValues(IBase theBase, String theExtensionUrl) {
182                if (theBase instanceof IBaseHasExtensions) {
183                        return ((IBaseHasExtensions) theBase)
184                                        .getExtension().stream()
185                                                        .filter(t -> theExtensionUrl.equals(t.getUrl()))
186                                                        .filter(t -> t.getValue() instanceof IPrimitiveType<?>)
187                                                        .map(t -> (IPrimitiveType<?>) t.getValue())
188                                                        .map(IPrimitiveType::getValueAsString)
189                                                        .filter(StringUtils::isNotBlank)
190                                                        .collect(Collectors.toList());
191                }
192                return Collections.emptyList();
193        }
194
195        /**
196         * Gets all extensions that match the specified filter predicate
197         *
198         * @param theBase   The resource to get the extension for
199         * @param theFilter Predicate to match the extension against
200         * @return Returns all extension with the specified URL, or an empty list if such extensions do not exist
201         */
202        public static List<IBaseExtension<?, ?>> getExtensionsMatchingPredicate(
203                        IBase theBase, Predicate<? super IBaseExtension<?, ?>> theFilter) {
204                return validateExtensionSupport(theBase).getExtension().stream()
205                                .filter(theFilter)
206                                .collect(Collectors.toList());
207        }
208
209        /**
210         * Removes all extensions.
211         *
212         * @param theBase The resource to clear the extension for
213         * @return Returns all extension that were removed
214         */
215        public static List<IBaseExtension<?, ?>> clearAllExtensions(IBase theBase) {
216                return clearExtensionsMatchingPredicate(theBase, (e -> true));
217        }
218
219        /**
220         * Removes all extensions by URL.
221         *
222         * @param theBase The resource to clear the extension for
223         * @param theUrl  The url to clear extensions for
224         * @return Returns all extension that were removed
225         */
226        public static List<IBaseExtension<?, ?>> clearExtensionsByUrl(IBase theBase, String theUrl) {
227                return clearExtensionsMatchingPredicate(theBase, (e -> theUrl.equals(e.getUrl())));
228        }
229
230        /**
231         * Removes all extensions that match the specified predicate
232         *
233         * @param theBase   The base object to clear the extension for
234         * @param theFilter Defines which extensions should be cleared
235         * @return Returns all extension that were removed
236         */
237        private static List<IBaseExtension<?, ?>> clearExtensionsMatchingPredicate(
238                        IBase theBase, Predicate<? super IBaseExtension<?, ?>> theFilter) {
239                List<IBaseExtension<?, ?>> retVal = getExtensionsMatchingPredicate(theBase, theFilter);
240                validateExtensionSupport(theBase).getExtension().removeIf(theFilter);
241                return retVal;
242        }
243
244        /**
245         * Gets all extensions with the specified URL
246         *
247         * @param theBase         The resource to get the extension for
248         * @param theExtensionUrl URL of the extension to get. Must be non-null
249         * @return Returns all extension with the specified URL, or an empty list if such extensions do not exist
250         */
251        public static List<IBaseExtension<?, ?>> getExtensionsByUrl(IBaseHasExtensions theBase, String theExtensionUrl) {
252                Predicate<IBaseExtension<?, ?>> urlEqualityPredicate = e -> theExtensionUrl.equals(e.getUrl());
253                return getExtensionsMatchingPredicate(theBase, urlEqualityPredicate);
254        }
255
256        /**
257         * Sets value of the extension as a string
258         *
259         * @param theExtension   The extension to set the value on
260         * @param theValue       The value to set
261         * @param theFhirContext The context containing FHIR resource definitions
262         */
263        public static void setExtension(FhirContext theFhirContext, IBaseExtension<?, ?> theExtension, String theValue) {
264                setExtension(theFhirContext, theExtension, "string", theValue);
265        }
266
267        /**
268         * Sets value of the extension
269         *
270         * @param theExtension     The extension to set the value on
271         * @param theExtensionType Element type of the extension
272         * @param theValue         The value to set
273         * @param theFhirContext   The context containing FHIR resource definitions
274         */
275        public static void setExtension(
276                        FhirContext theFhirContext, IBaseExtension<?, ?> theExtension, String theExtensionType, Object theValue) {
277                theExtension.setValue(TerserUtil.newElement(theFhirContext, theExtensionType, theValue));
278        }
279
280        /**
281         * Sets or replaces existing extension with the specified value as a string
282         *
283         * @param theBase        The resource to update extension on
284         * @param theUrl         Extension URL
285         * @param theValue       Extension value
286         * @param theFhirContext The context containing FHIR resource definitions
287         */
288        public static void setExtensionAsString(FhirContext theFhirContext, IBase theBase, String theUrl, String theValue) {
289                IBaseExtension<?, ?> ext = getOrCreateExtension(theBase, theUrl);
290                setExtension(theFhirContext, ext, theValue);
291        }
292
293        /**
294         * Sets or replaces existing extension with the specified value
295         *
296         * @param theBase        The resource to update extension on
297         * @param theUrl         Extension URL
298         * @param theValueType   Type of the value to set in the extension
299         * @param theValue       Extension value
300         * @param theFhirContext The context containing FHIR resource definitions
301         */
302        public static void setExtension(
303                        FhirContext theFhirContext, IBase theBase, String theUrl, String theValueType, Object theValue) {
304                IBaseExtension<?, ?> ext = getOrCreateExtension(theBase, theUrl);
305                setExtension(theFhirContext, ext, theValueType, theValue);
306        }
307
308        /**
309         * Compares two extensions, returns true if they have the same value and url
310         *
311         * @param theLeftExtension  : Extension to be evaluated #1
312         * @param theRightExtension : Extension to be evaluated #2
313         * @return Result of the comparison
314         */
315        public static boolean equals(IBaseExtension<?, ?> theLeftExtension, IBaseExtension<?, ?> theRightExtension) {
316                return TerserUtil.equals(theLeftExtension, theRightExtension);
317        }
318
319        /**
320         * Given an extension, looks for the first child extension with the given URL of {@literal theChildExtensionUrl}
321         * and a primitive datatype value, and returns the String version of that value. E.g. if the
322         * value is a FHIR boolean, it would return the string "true" or "false. If the extension
323         * has no value, or the value is not a primitive datatype, or the URL is not found, the method
324         * will return {@literal null}.
325         *
326         * @param theExtension The parent extension. Must not be null.
327         * @param theChildExtensionUrl The child extension URL. Must not be null or blank.
328         * @since 6.6.0
329         */
330        public static <D, T extends IBaseExtension<T, D>> String extractChildPrimitiveExtensionValue(
331                        @Nonnull IBaseExtension<T, D> theExtension, @Nonnull String theChildExtensionUrl) {
332                Validate.notNull(theExtension, "theExtension must not be null");
333                Validate.notBlank(theChildExtensionUrl, "theChildExtensionUrl must not be null or blank");
334
335                Optional<T> codeExtension = theExtension.getExtension().stream()
336                                .filter(t -> theChildExtensionUrl.equals(t.getUrl()))
337                                .findFirst();
338                String retVal = null;
339                if (codeExtension.isPresent() && codeExtension.get().getValue() instanceof IPrimitiveType) {
340                        IPrimitiveType<?> codeValue =
341                                        (IPrimitiveType<?>) codeExtension.get().getValue();
342                        retVal = codeValue.getValueAsString();
343                }
344                return retVal;
345        }
346}