001/*-
002 * #%L
003 * HAPI FHIR - Server Framework
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.rest.server.interceptor.validation;
021
022import ca.uhn.fhir.interceptor.api.Hook;
023import ca.uhn.fhir.interceptor.api.Interceptor;
024import ca.uhn.fhir.interceptor.api.Pointcut;
025import ca.uhn.fhir.validation.SingleValidationMessage;
026import ca.uhn.fhir.validation.ValidationResult;
027
028import java.util.ArrayList;
029import java.util.Arrays;
030import java.util.List;
031import java.util.regex.Pattern;
032
033import static org.apache.commons.lang3.StringUtils.isNotBlank;
034
035@Interceptor
036public class ValidationMessageSuppressingInterceptor {
037
038        private List<Pattern> mySuppressPatterns = new ArrayList<>();
039
040        /**
041         * Constructor
042         */
043        public ValidationMessageSuppressingInterceptor() {
044                super();
045        }
046
047        /**
048         * Supplies one or more patterns to suppress. Any validation messages (of any severity) will be suppressed
049         * if they match this pattern. Patterns are in Java Regular Expression format (as defined by the {@link Pattern} class)
050         * and are treated as partial maches. They are also case insensitive.
051         * <p>
052         *    For example, a pattern of <code>loinc.*1234</code> would suppress the following message:<br/>
053         *    <code>The LOINC code 1234 is not valid</code>
054         * </p>
055         */
056        public ValidationMessageSuppressingInterceptor addMessageSuppressionPatterns(String... thePatterns) {
057                return addMessageSuppressionPatterns(Arrays.asList(thePatterns));
058        }
059
060        /**
061         * Supplies one or more patterns to suppress. Any validation messages (of any severity) will be suppressed
062         * if they match this pattern. Patterns are in Java Regular Expression format (as defined by the {@link Pattern} class)
063         * and are treated as partial maches. They are also case insensitive.
064         * <p>
065         *    For example, a pattern of <code>loinc.*1234</code> would suppress the following message:<br/>
066         *    <code>The LOINC code 1234 is not valid</code>
067         * </p>
068         */
069        public ValidationMessageSuppressingInterceptor addMessageSuppressionPatterns(List<String> thePatterns) {
070                for (String next : thePatterns) {
071                        if (isNotBlank(next)) {
072                                Pattern pattern = Pattern.compile(next, Pattern.CASE_INSENSITIVE);
073                                mySuppressPatterns.add(pattern);
074                        }
075                }
076                return this;
077        }
078
079        @Hook(Pointcut.VALIDATION_COMPLETED)
080        public ValidationResult handle(ValidationResult theResult) {
081
082                List<SingleValidationMessage> newMessages =
083                                new ArrayList<>(theResult.getMessages().size());
084                for (SingleValidationMessage next : theResult.getMessages()) {
085
086                        String nextMessage = next.getMessage();
087                        boolean suppress = false;
088                        for (Pattern nextSuppressPattern : mySuppressPatterns) {
089                                if (nextSuppressPattern.matcher(nextMessage).find()) {
090                                        suppress = true;
091                                        break;
092                                }
093                        }
094
095                        if (!suppress) {
096                                newMessages.add(next);
097                        }
098                }
099
100                if (newMessages.size() == theResult.getMessages().size()) {
101                        return null;
102                }
103
104                return new ValidationResult(theResult.getContext(), newMessages);
105        }
106}