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;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.interceptor.api.Interceptor;
025import ca.uhn.fhir.parser.IParser;
026import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
027import ca.uhn.fhir.rest.api.server.RequestDetails;
028import ca.uhn.fhir.rest.server.exceptions.BaseServerResponseException;
029import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
030import ca.uhn.fhir.rest.server.exceptions.UnprocessableEntityException;
031import ca.uhn.fhir.util.OperationOutcomeUtil;
032import ca.uhn.fhir.validation.FhirValidator;
033import ca.uhn.fhir.validation.IValidatorModule;
034import ca.uhn.fhir.validation.ResultSeverityEnum;
035import ca.uhn.fhir.validation.SingleValidationMessage;
036import ca.uhn.fhir.validation.ValidationResult;
037import org.apache.commons.lang3.Validate;
038import org.apache.commons.lang3.text.StrLookup;
039import org.apache.commons.lang3.text.StrSubstitutor;
040import org.hl7.fhir.instance.model.api.IBaseOperationOutcome;
041import org.slf4j.Logger;
042import org.slf4j.LoggerFactory;
043
044import java.util.ArrayList;
045import java.util.List;
046
047import static org.apache.commons.lang3.StringUtils.isNotBlank;
048
049/**
050 * This interceptor intercepts each incoming request and if it contains a FHIR resource, validates that resource. The
051 * interceptor may be configured to run any validator modules, and will then add headers to the response or fail the
052 * request with an {@link UnprocessableEntityException HTTP 422 Unprocessable Entity}.
053 */
054@Interceptor
055public abstract class BaseValidatingInterceptor<T> extends ValidationResultEnrichingInterceptor {
056
057        /**
058         * Default value:<br/>
059         * <code>
060         * ${row}:${col} ${severity} ${message} (${location})
061         * </code>
062         */
063        public static final String DEFAULT_RESPONSE_HEADER_VALUE = "${row}:${col} ${severity} ${message} (${location})";
064
065        private static final Logger ourLog = LoggerFactory.getLogger(BaseValidatingInterceptor.class);
066
067        private Integer myAddResponseIssueHeaderOnSeverity = null;
068        private Integer myAddResponseOutcomeHeaderOnSeverity = null;
069        private Integer myFailOnSeverity = ResultSeverityEnum.ERROR.ordinal();
070        private boolean myIgnoreValidatorExceptions;
071        private int myMaximumHeaderLength = 200;
072        private String myResponseIssueHeaderName = provideDefaultResponseHeaderName();
073        private String myResponseIssueHeaderValue = DEFAULT_RESPONSE_HEADER_VALUE;
074        private String myResponseIssueHeaderValueNoIssues = null;
075        private String myResponseOutcomeHeaderName = provideDefaultResponseHeaderName();
076
077        private List<IValidatorModule> myValidatorModules;
078        private FhirValidator myValidator;
079
080        private void addResponseIssueHeader(RequestDetails theRequestDetails, SingleValidationMessage theNext) {
081                // Perform any string substitutions from the message format
082                StrLookup<?> lookup = new MyLookup(theNext);
083                StrSubstitutor subs = new StrSubstitutor(lookup, "${", "}", '\\');
084
085                // Log the header
086                String headerValue = subs.replace(myResponseIssueHeaderValue);
087                ourLog.trace("Adding header to response: {}", headerValue);
088
089                theRequestDetails.getResponse().addHeader(myResponseIssueHeaderName, headerValue);
090        }
091
092        /**
093         * Specify a validator module to use.
094         *
095         * @see #setValidator(FhirValidator)
096         */
097        public BaseValidatingInterceptor<T> addValidatorModule(IValidatorModule theModule) {
098                Validate.notNull(theModule, "theModule must not be null");
099                Validate.isTrue(
100                                myValidator == null,
101                                "Can not specify both a validator and validator modules. Only one needs to be supplied.");
102                if (getValidatorModules() == null) {
103                        setValidatorModules(new ArrayList<>());
104                }
105                getValidatorModules().add(theModule);
106                return this;
107        }
108
109        /**
110         * Provides the validator to use. This can be used as an alternative to {@link #addValidatorModule(IValidatorModule)}
111         *
112         * @see #addValidatorModule(IValidatorModule)
113         * @see #setValidatorModules(List)
114         */
115        public void setValidator(FhirValidator theValidator) {
116                Validate.isTrue(
117                                theValidator == null
118                                                || getValidatorModules() == null
119                                                || getValidatorModules().isEmpty(),
120                                "Can not specify both a validator and validator modules. Only one needs to be supplied.");
121                myValidator = theValidator;
122        }
123
124        abstract ValidationResult doValidate(FhirValidator theValidator, T theRequest);
125
126        /**
127         * Fail the request by throwing an {@link UnprocessableEntityException} as a result of a validation failure.
128         * Subclasses may change this behaviour by providing alternate behaviour.
129         */
130        protected void fail(RequestDetails theRequestDetails, ValidationResult theValidationResult) {
131                throw new UnprocessableEntityException(
132                                Msg.code(330) + theValidationResult.getMessages().get(0).getMessage(),
133                                theValidationResult.toOperationOutcome());
134        }
135
136        /**
137         * If the validation produces a result with at least the given severity, a header with the name
138         * specified by {@link #setResponseOutcomeHeaderName(String)} will be added containing a JSON encoded
139         * OperationOutcome resource containing the validation results.
140         */
141        public ResultSeverityEnum getAddResponseOutcomeHeaderOnSeverity() {
142                return myAddResponseOutcomeHeaderOnSeverity != null
143                                ? ResultSeverityEnum.values()[myAddResponseOutcomeHeaderOnSeverity]
144                                : null;
145        }
146
147        /**
148         * If the validation produces a result with at least the given severity, a header with the name
149         * specified by {@link #setResponseOutcomeHeaderName(String)} will be added containing a JSON encoded
150         * OperationOutcome resource containing the validation results.
151         */
152        public void setAddResponseOutcomeHeaderOnSeverity(ResultSeverityEnum theAddResponseOutcomeHeaderOnSeverity) {
153                myAddResponseOutcomeHeaderOnSeverity =
154                                theAddResponseOutcomeHeaderOnSeverity != null ? theAddResponseOutcomeHeaderOnSeverity.ordinal() : null;
155        }
156
157        /**
158         * The maximum length for an individual header. If an individual header would be written exceeding this length,
159         * the header value will be truncated.
160         */
161        public int getMaximumHeaderLength() {
162                return myMaximumHeaderLength;
163        }
164
165        /**
166         * The maximum length for an individual header. If an individual header would be written exceeding this length,
167         * the header value will be truncated. Value must be greater than 100.
168         */
169        public void setMaximumHeaderLength(int theMaximumHeaderLength) {
170                Validate.isTrue(theMaximumHeaderLength >= 100, "theMaximumHeadeerLength must be >= 100");
171                myMaximumHeaderLength = theMaximumHeaderLength;
172        }
173
174        /**
175         * The name of the header specified by {@link #setAddResponseOutcomeHeaderOnSeverity(ResultSeverityEnum)}
176         */
177        public String getResponseOutcomeHeaderName() {
178                return myResponseOutcomeHeaderName;
179        }
180
181        /**
182         * The name of the header specified by {@link #setAddResponseOutcomeHeaderOnSeverity(ResultSeverityEnum)}
183         */
184        public void setResponseOutcomeHeaderName(String theResponseOutcomeHeaderName) {
185                Validate.notEmpty(theResponseOutcomeHeaderName, "theResponseOutcomeHeaderName can not be empty or null");
186                myResponseOutcomeHeaderName = theResponseOutcomeHeaderName;
187        }
188
189        public List<IValidatorModule> getValidatorModules() {
190                return myValidatorModules;
191        }
192
193        public void setValidatorModules(List<IValidatorModule> theValidatorModules) {
194                Validate.isTrue(
195                                myValidator == null || theValidatorModules == null || theValidatorModules.isEmpty(),
196                                "Can not specify both a validator and validator modules. Only one needs to be supplied.");
197                myValidatorModules = theValidatorModules;
198        }
199
200        /**
201         * If set to <code>true</code> (default is <code>false</code>) this interceptor
202         * will exit immediately and allow processing to continue if the validator throws
203         * any exceptions.
204         * <p>
205         * This setting is mostly useful in testing situations
206         * </p>
207         */
208        public boolean isIgnoreValidatorExceptions() {
209                return myIgnoreValidatorExceptions;
210        }
211
212        /**
213         * If set to <code>true</code> (default is <code>false</code>) this interceptor
214         * will exit immediately and allow processing to continue if the validator throws
215         * any exceptions.
216         * <p>
217         * This setting is mostly useful in testing situations
218         * </p>
219         */
220        public void setIgnoreValidatorExceptions(boolean theIgnoreValidatorExceptions) {
221                myIgnoreValidatorExceptions = theIgnoreValidatorExceptions;
222        }
223
224        abstract String provideDefaultResponseHeaderName();
225
226        /**
227         * Sets the minimum severity at which an issue detected by the validator will result in a header being added to the
228         * response. Default is {@link ResultSeverityEnum#INFORMATION}. Set to <code>null</code> to disable this behaviour.
229         *
230         * @see #setResponseHeaderName(String)
231         * @see #setResponseHeaderValue(String)
232         */
233        public void setAddResponseHeaderOnSeverity(ResultSeverityEnum theSeverity) {
234                myAddResponseIssueHeaderOnSeverity = theSeverity != null ? theSeverity.ordinal() : null;
235        }
236
237        /**
238         * Sets the minimum severity at which an issue detected by the validator will fail/reject the request. Default is
239         * {@link ResultSeverityEnum#ERROR}. Set to <code>null</code> to disable this behaviour.
240         */
241        public void setFailOnSeverity(ResultSeverityEnum theSeverity) {
242                myFailOnSeverity = theSeverity != null ? theSeverity.ordinal() : null;
243        }
244
245        /**
246         * Sets the name of the response header to add validation failures to
247         *
248         * @see #setAddResponseHeaderOnSeverity(ResultSeverityEnum)
249         */
250        protected void setResponseHeaderName(String theResponseHeaderName) {
251                Validate.notBlank(theResponseHeaderName, "theResponseHeaderName must not be blank or null");
252                myResponseIssueHeaderName = theResponseHeaderName;
253        }
254
255        /**
256         * Sets the value to add to the response header with the name specified by {@link #setResponseHeaderName(String)}
257         * when validation produces a message of severity equal to or greater than
258         * {@link #setAddResponseHeaderOnSeverity(ResultSeverityEnum)}
259         * <p>
260         * This field allows the following substitutions:
261         * </p>
262         * <table>
263         * <tr>
264         * <td>Name</td>
265         * <td>Value</td>
266         * </tr>
267         * <tr>
268         * <td>${line}</td>
269         * <td>The line in the request</td>
270         * </tr>
271         * <tr>
272         * <td>${col}</td>
273         * <td>The column in the request</td>
274         * </tr>
275         * <tr>
276         * <td>${location}</td>
277         * <td>The location in the payload as a string (typically this will be a path)</td>
278         * </tr>
279         * <tr>
280         * <td>${severity}</td>
281         * <td>The severity of the issue</td>
282         * </tr>
283         * <tr>
284         * <td>${message}</td>
285         * <td>The validation message</td>
286         * </tr>
287         * </table>
288         *
289         * @see #DEFAULT_RESPONSE_HEADER_VALUE
290         * @see #setAddResponseHeaderOnSeverity(ResultSeverityEnum)
291         */
292        public void setResponseHeaderValue(String theResponseHeaderValue) {
293                Validate.notBlank(theResponseHeaderValue, "theResponseHeaderValue must not be blank or null");
294                myResponseIssueHeaderValue = theResponseHeaderValue;
295        }
296
297        /**
298         * Sets the header value to add when no issues are found at or exceeding the
299         * threshold specified by {@link #setAddResponseHeaderOnSeverity(ResultSeverityEnum)}
300         */
301        public void setResponseHeaderValueNoIssues(String theResponseHeaderValueNoIssues) {
302                myResponseIssueHeaderValueNoIssues = theResponseHeaderValueNoIssues;
303        }
304
305        /**
306         * Hook for subclasses (e.g. add a tag (coding) to an incoming resource when a given severity appears in the
307         * ValidationResult).
308         */
309        protected void postProcessResult(RequestDetails theRequestDetails, ValidationResult theValidationResult) {}
310
311        /**
312         * Hook for subclasses on failure (e.g. add a response header to an incoming resource upon rejection).
313         */
314        protected void postProcessResultOnFailure(RequestDetails theRequestDetails, ValidationResult theValidationResult) {}
315
316        /**
317         * Note: May return null
318         */
319        protected ValidationResult validate(T theRequest, RequestDetails theRequestDetails) {
320                if (theRequest == null || theRequestDetails == null) {
321                        return null;
322                }
323
324                RestOperationTypeEnum opType = theRequestDetails.getRestOperationType();
325                if (opType != null) {
326                        switch (opType) {
327                                case GRAPHQL_REQUEST:
328                                        return null;
329                                default:
330                                        break;
331                        }
332                }
333
334                FhirValidator validator;
335                if (myValidator != null) {
336                        validator = myValidator;
337                } else {
338                        validator = theRequestDetails.getServer().getFhirContext().newValidator();
339                        if (myValidatorModules != null) {
340                                for (IValidatorModule next : myValidatorModules) {
341                                        validator.registerValidatorModule(next);
342                                }
343                        }
344                }
345
346                ValidationResult validationResult;
347                try {
348                        validationResult = doValidate(validator, theRequest);
349                } catch (Exception e) {
350                        if (myIgnoreValidatorExceptions) {
351                                ourLog.warn("Validator threw an exception during validation", e);
352                                return null;
353                        }
354                        if (e instanceof BaseServerResponseException) {
355                                throw (BaseServerResponseException) e;
356                        }
357                        throw new InternalErrorException(Msg.code(331) + e);
358                }
359
360                if (myAddResponseIssueHeaderOnSeverity != null) {
361                        boolean found = false;
362                        for (SingleValidationMessage next : validationResult.getMessages()) {
363                                if (next.getSeverity().ordinal() >= myAddResponseIssueHeaderOnSeverity) {
364                                        addResponseIssueHeader(theRequestDetails, next);
365                                        found = true;
366                                }
367                        }
368                        if (!found) {
369                                if (isNotBlank(myResponseIssueHeaderValueNoIssues)) {
370                                        theRequestDetails
371                                                        .getResponse()
372                                                        .addHeader(myResponseIssueHeaderName, myResponseIssueHeaderValueNoIssues);
373                                }
374                        }
375                }
376
377                if (myFailOnSeverity != null) {
378                        for (SingleValidationMessage next : validationResult.getMessages()) {
379                                if (next.getSeverity().ordinal() >= myFailOnSeverity) {
380                                        postProcessResultOnFailure(theRequestDetails, validationResult);
381                                        fail(theRequestDetails, validationResult);
382                                        return validationResult;
383                                }
384                        }
385                }
386
387                if (myAddResponseOutcomeHeaderOnSeverity != null) {
388                        IBaseOperationOutcome outcome = null;
389                        for (SingleValidationMessage next : validationResult.getMessages()) {
390                                if (next.getSeverity().ordinal() >= myAddResponseOutcomeHeaderOnSeverity) {
391                                        outcome = validationResult.toOperationOutcome();
392                                        break;
393                                }
394                        }
395                        if (outcome == null
396                                        && myAddResponseOutcomeHeaderOnSeverity != null
397                                        && myAddResponseOutcomeHeaderOnSeverity == ResultSeverityEnum.INFORMATION.ordinal()) {
398                                FhirContext ctx = theRequestDetails.getServer().getFhirContext();
399                                outcome = OperationOutcomeUtil.newInstance(ctx);
400                                OperationOutcomeUtil.addIssue(ctx, outcome, "information", "No issues detected", "", "informational");
401                        }
402
403                        if (outcome != null) {
404                                IParser parser = theRequestDetails
405                                                .getServer()
406                                                .getFhirContext()
407                                                .newJsonParser()
408                                                .setPrettyPrint(false);
409                                String encoded = parser.encodeResourceToString(outcome);
410                                if (encoded.length() > getMaximumHeaderLength()) {
411                                        encoded = encoded.substring(0, getMaximumHeaderLength() - 3) + "...";
412                                }
413                                theRequestDetails.getResponse().addHeader(myResponseOutcomeHeaderName, encoded);
414                        }
415                }
416
417                postProcessResult(theRequestDetails, validationResult);
418
419                return validationResult;
420        }
421
422        private static class MyLookup extends StrLookup<String> {
423
424                private SingleValidationMessage myMessage;
425
426                public MyLookup(SingleValidationMessage theMessage) {
427                        myMessage = theMessage;
428                }
429
430                @Override
431                public String lookup(String theKey) {
432                        if ("line".equals(theKey)) {
433                                return toString(myMessage.getLocationLine());
434                        }
435                        if ("col".equals(theKey)) {
436                                return toString(myMessage.getLocationCol());
437                        }
438                        if ("message".equals(theKey)) {
439                                return toString(myMessage.getMessage());
440                        }
441                        if ("location".equals(theKey)) {
442                                return toString(myMessage.getLocationString());
443                        }
444                        if ("severity".equals(theKey)) {
445                                return myMessage.getSeverity() != null ? myMessage.getSeverity().name() : null;
446                        }
447
448                        return "";
449                }
450
451                private static String toString(Object theInt) {
452                        return theInt != null ? theInt.toString() : "";
453                }
454        }
455}