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.interceptor.api.Hook;
024import ca.uhn.fhir.interceptor.api.Interceptor;
025import ca.uhn.fhir.interceptor.api.Pointcut;
026import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
027import ca.uhn.fhir.rest.server.method.BaseMethodBinding;
028import ca.uhn.fhir.rest.server.method.OperationMethodBinding;
029import jakarta.annotation.Nonnull;
030import org.apache.commons.lang3.Validate;
031import org.slf4j.Logger;
032import org.slf4j.LoggerFactory;
033
034import java.util.Collections;
035import java.util.HashSet;
036import java.util.Set;
037import java.util.TreeSet;
038
039import static org.apache.commons.lang3.StringUtils.isBlank;
040
041/**
042 * This interceptor can be used to selectively block specific interactions/operations from
043 * the server's capabilities. This interceptor must be configured and registered to a
044 * {@link ca.uhn.fhir.rest.server.RestfulServer} prior to any resource provider
045 * classes being registered to it. This interceptor will then examine any
046 * provider classes being registered and may choose to discard some or all
047 * of the method bindings on each provider.
048 * <p>
049 * For example, if this interceptor is configured to block resource creation, then
050 * when a resource provider is registered that has both a
051 * {@link ca.uhn.fhir.rest.annotation.Read @Read} method and a
052 * {@link ca.uhn.fhir.rest.annotation.Create @Create} method, the
053 * create method will be ignored and not bound.
054 * </p>
055 * <p>
056 * Note: This interceptor is not a security interceptor! It can be used to remove
057 * writes capabilities from a FHIR endpoint (for example) but it does not guarantee
058 * that writes won't be possible. Security rules should be enforced using
059 * {@link ca.uhn.fhir.rest.server.interceptor.auth.AuthorizationInterceptor} or
060 * a similar strategy. However, this interceptor can be useful in order to
061 * clarify the intent of an endpoint to the outside world. Of particular note,
062 * even if a create method has been blocked from binding by this interceptor,
063 * it may still be possible to create resources via a FHIR transaction unless
064 * proper security has been implemented.
065 * </p>
066 * <p>
067 * Use {@link Builder new Builder()} to create a new instance of this class.
068 * </p>
069 *
070 * @see Builder#addAllowedSpec(String) to add allowed interactions
071 * @since 6.2.0
072 */
073@Interceptor
074public class InteractionBlockingInterceptor {
075
076        public static final Set<RestOperationTypeEnum> ALLOWED_OP_TYPES;
077        private static final Logger ourLog = LoggerFactory.getLogger(InteractionBlockingInterceptor.class);
078
079        static {
080                Set<RestOperationTypeEnum> allowedOpTypes = new TreeSet<>();
081                allowedOpTypes.add(RestOperationTypeEnum.META);
082                allowedOpTypes.add(RestOperationTypeEnum.META_ADD);
083                allowedOpTypes.add(RestOperationTypeEnum.META_DELETE);
084                allowedOpTypes.add(RestOperationTypeEnum.PATCH);
085                allowedOpTypes.add(RestOperationTypeEnum.READ);
086                allowedOpTypes.add(RestOperationTypeEnum.CREATE);
087                allowedOpTypes.add(RestOperationTypeEnum.UPDATE);
088                allowedOpTypes.add(RestOperationTypeEnum.DELETE);
089                allowedOpTypes.add(RestOperationTypeEnum.BATCH);
090                allowedOpTypes.add(RestOperationTypeEnum.TRANSACTION);
091                allowedOpTypes.add(RestOperationTypeEnum.VALIDATE);
092                allowedOpTypes.add(RestOperationTypeEnum.SEARCH_TYPE);
093                allowedOpTypes.add(RestOperationTypeEnum.HISTORY_TYPE);
094                allowedOpTypes.add(RestOperationTypeEnum.HISTORY_INSTANCE);
095                allowedOpTypes.add(RestOperationTypeEnum.HISTORY_SYSTEM);
096                ALLOWED_OP_TYPES = Collections.unmodifiableSet(allowedOpTypes);
097        }
098
099        private final Set<String> myAllowedKeys;
100
101        /**
102         * Constructor
103         */
104        private InteractionBlockingInterceptor(@Nonnull Builder theBuilder) {
105                myAllowedKeys = theBuilder.myAllowedKeys;
106        }
107
108        @Hook(Pointcut.SERVER_PROVIDER_METHOD_BOUND)
109        public BaseMethodBinding bindMethod(BaseMethodBinding theMethodBinding) {
110
111                boolean allowed = true;
112                String resourceName = theMethodBinding.getResourceName();
113                RestOperationTypeEnum restOperationType = theMethodBinding.getRestOperationType();
114                switch (restOperationType) {
115                        case EXTENDED_OPERATION_SERVER:
116                        case EXTENDED_OPERATION_TYPE:
117                        case EXTENDED_OPERATION_INSTANCE: {
118                                OperationMethodBinding operationMethodBinding = (OperationMethodBinding) theMethodBinding;
119                                if (!myAllowedKeys.isEmpty()) {
120                                        if (!myAllowedKeys.contains(operationMethodBinding.getName())) {
121                                                allowed = false;
122                                        }
123                                }
124                                break;
125                        }
126                        default: {
127                                if (restOperationType == RestOperationTypeEnum.VREAD) {
128                                        restOperationType = RestOperationTypeEnum.READ;
129                                }
130                                String key = toKey(resourceName, restOperationType);
131                                if (!myAllowedKeys.isEmpty()) {
132                                        if (!myAllowedKeys.contains(key)) {
133                                                allowed = false;
134                                        }
135                                }
136                                break;
137                        }
138                }
139
140                if (!allowed) {
141                        ourLog.info(
142                                        "Skipping method binding for {}:{} provided by {}",
143                                        resourceName,
144                                        restOperationType,
145                                        theMethodBinding.getMethod());
146                        return null;
147                }
148
149                return theMethodBinding;
150        }
151
152        private static String toKey(String theResourceType, RestOperationTypeEnum theRestOperationTypeEnum) {
153                if (isBlank(theResourceType)) {
154                        return theRestOperationTypeEnum.getCode();
155                }
156                return theResourceType + ":" + theRestOperationTypeEnum.getCode();
157        }
158
159        public static class Builder {
160
161                private final Set<String> myAllowedKeys = new HashSet<>();
162                private final FhirContext myCtx;
163
164                /**
165                 * Constructor
166                 */
167                public Builder(@Nonnull FhirContext theCtx) {
168                        Validate.notNull(theCtx, "theCtx must not be null");
169                        myCtx = theCtx;
170                }
171
172                /**
173                 * Adds an interaction or operation that will be permitted. Allowable formats
174                 * are:
175                 * <ul>
176                 *    <li>
177                 *       <b>[resourceType]:[interaction]</b> - Use this form to allow type- and instance-level interactions, such as
178                 *       <code>create</code>, <code>read</code>, and <code>patch</code>. For example, the spec <code>Patient:create</code>
179                 *       allows the Patient-level create operation (i.e. <code>POST /Patient</code>).
180                 *    </li>
181                 *    <li>
182                 *       <b>$[operation-name]</b> - Use this form to allow operations (at any level) by name. For example, the spec
183                 *       <code>$diff</code> permits the <a href="https://hapifhir.io/hapi-fhir/docs/server_jpa/diff.html">Diff Operation</a>
184                 *       to be applied at both the server- and instance-level.
185                 *    </li>
186                 * </ul>
187                 * <p>
188                 * Note that the spec does not differentiate between the <code>read</code> and <code>vread</code> interactions. If one
189                 * is permitted the other will also be permitted.
190                 * </p>
191                 *
192                 * @return
193                 */
194                public Builder addAllowedSpec(String theSpec) {
195                        Validate.notBlank(theSpec, "theSpec must not be null or blank");
196
197                        if (theSpec.startsWith("$")) {
198                                addAllowedOperation(theSpec);
199                                return this;
200                        }
201
202                        int colonIdx = theSpec.indexOf(':');
203                        Validate.isTrue(colonIdx > 0, "Invalid interaction allowed spec: %s", theSpec);
204
205                        String resourceName = theSpec.substring(0, colonIdx);
206                        String interactionName = theSpec.substring(colonIdx + 1);
207                        if (interactionName.equals("search")) {
208                                interactionName = "search-type";
209                                validateInteraction(interactionName, theSpec, resourceName);
210                        } else if (interactionName.equals("history")) {
211                                validateInteraction("history-instance", theSpec, resourceName);
212                                validateInteraction("history-type", theSpec, resourceName);
213                        } else {
214                                validateInteraction(interactionName, theSpec, resourceName);
215                        }
216                        return this;
217                }
218
219                private void validateInteraction(String theInteractionName, String theSpec, String theResourceName) {
220                        RestOperationTypeEnum interaction = RestOperationTypeEnum.forCode(theInteractionName);
221                        Validate.notNull(interaction, "Unknown interaction %s in spec %s", theInteractionName, theSpec);
222                        addAllowedInteraction(theResourceName, interaction);
223                }
224
225                /**
226                 * Adds an interaction that will be permitted.
227                 */
228                private void addAllowedInteraction(String theResourceType, RestOperationTypeEnum theInteractionType) {
229                        Validate.notBlank(theResourceType, "theResourceType must not be null or blank");
230                        Validate.notNull(theInteractionType, "theInteractionType must not be null");
231                        Validate.isTrue(
232                                        ALLOWED_OP_TYPES.contains(theInteractionType),
233                                        "Operation type %s can not be used as an allowable rule",
234                                        theInteractionType);
235                        Validate.isTrue(myCtx.getResourceType(theResourceType) != null, "Unknown resource type: %s");
236                        String key = toKey(theResourceType, theInteractionType);
237                        myAllowedKeys.add(key);
238                }
239
240                private void addAllowedOperation(String theOperationName) {
241                        Validate.notBlank(theOperationName, "theOperationName must not be null or blank");
242                        Validate.isTrue(theOperationName.startsWith("$"), "Invalid operation name: %s", theOperationName);
243                        myAllowedKeys.add(theOperationName);
244                }
245
246                public InteractionBlockingInterceptor build() {
247                        return new InteractionBlockingInterceptor(this);
248                }
249        }
250}