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.auth;
021
022import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
023import ca.uhn.fhir.rest.api.server.RequestDetails;
024import jakarta.annotation.Nonnull;
025import jakarta.annotation.Nullable;
026import org.apache.commons.lang3.Validate;
027import org.hl7.fhir.instance.model.api.IBaseResource;
028import org.hl7.fhir.instance.model.api.IIdType;
029
030/**
031 * Allows user-supplied logic for authorization rules.
032 * <p>
033 * THIS IS AN EXPERIMENTAL API! Feedback is welcome, and this API
034 * may change.
035 *
036 * @since 3.4.0
037 */
038public interface IAuthRuleTester {
039        /**
040         * A request object to make this easier to extend.
041         */
042        class RuleTestRequest {
043                // fake record pattern
044                /** the mode of the calling rule context */
045                @Nonnull
046                public final PolicyEnum mode;
047                /**
048                 * The FHIR operation being performed.
049                 * Note that this is not necessarily the same as the value obtained from invoking
050                 * {@link RequestDetails#getRestOperationType()} on {@literal requestDetails}
051                 * because multiple operations can be nested within
052                 * an HTTP request using FHIR transaction and batch operations
053                 */
054                @Nonnull
055                public final RestOperationTypeEnum operation;
056
057                @Nonnull
058                public final RequestDetails requestDetails;
059
060                @Nullable
061                public final IIdType resourceId;
062
063                @Nullable
064                public final IBaseResource resource;
065                /** supplier for support services */
066                @Nonnull
067                public final IRuleApplier ruleApplier;
068
069                public RuleTestRequest(
070                                PolicyEnum theMode,
071                                @Nonnull RestOperationTypeEnum theOperation,
072                                @Nonnull RequestDetails theRequestDetails,
073                                @Nullable IIdType theResourceId,
074                                @Nullable IBaseResource theResource,
075                                @Nonnull IRuleApplier theRuleApplier) {
076                        Validate.notNull(theMode);
077                        Validate.notNull(theOperation);
078                        Validate.notNull(theRequestDetails);
079                        Validate.notNull(theRuleApplier);
080                        mode = theMode;
081                        operation = theOperation;
082                        requestDetails = theRequestDetails;
083                        resource = theResource;
084                        if (theResourceId == null && resource != null) {
085                                resourceId = resource.getIdElement();
086                        } else {
087                                resourceId = theResourceId;
088                        }
089                        ruleApplier = theRuleApplier;
090                }
091        }
092
093        /**
094         * User supplied logic called just before the parent rule renders a verdict on the operation
095         * or input resource.
096         *
097         * Returning true will allow the verdict continue.
098         * Returning false will block the verdict and cause the rule to abstain (i.e. return null).
099         *
100         * <p>
101         * THIS IS AN EXPERIMENTAL API! Feedback is welcome, and this API
102         * may change.
103         *
104         * @param theRequest The details of the operation or an INPUT resource to evaluate
105         * @return true if the verdict should continue
106         * @since 6.1.0
107         */
108        default boolean matches(RuleTestRequest theRequest) {
109                return this.matches(
110                                theRequest.operation, theRequest.requestDetails, theRequest.resourceId, theRequest.resource);
111        }
112
113        /**
114         * DO NOT IMPLEMENT - Old api.  {@link #matches(RuleTestRequest)} instead.
115         * @deprecated
116         */
117        @Deprecated(since = "6.1.0")
118        default boolean matches(
119                        RestOperationTypeEnum theOperation,
120                        RequestDetails theRequestDetails,
121                        IIdType theInputResourceId,
122                        IBaseResource theInputResource) {
123                return true;
124        }
125
126        /**
127         * User supplied logic called just before the parent rule renders a verdict on an output resource.
128         *
129         * Returning true will allow the verdict continue.
130         * Returning false will block the verdict and cause the rule to abstain (i.e. return null).
131         *
132         * <p>
133         * THIS IS AN EXPERIMENTAL API! Feedback is welcome, and this API
134         * may change.
135         *
136         * @param theRequest The details of the operation or an INPUT resource to evaluate
137         * @return true if the verdict should continue
138         * @since 6.1.0  */
139        default boolean matchesOutput(RuleTestRequest theRequest) {
140                return this.matchesOutput(theRequest.operation, theRequest.requestDetails, theRequest.resource);
141        }
142
143        /**
144         * DO NOT IMPLEMENT - Old api.  {@link #matches(RuleTestRequest)} instead.
145         * @deprecated
146         */
147        @Deprecated(since = "6.1.0")
148        default boolean matchesOutput(
149                        RestOperationTypeEnum theOperation, RequestDetails theRequestDetails, IBaseResource theOutputResource) {
150                return true;
151        }
152}