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.binary;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.interceptor.api.Hook;
025import ca.uhn.fhir.interceptor.api.Interceptor;
026import ca.uhn.fhir.interceptor.api.Pointcut;
027import ca.uhn.fhir.rest.api.server.IPreResourceShowDetails;
028import ca.uhn.fhir.rest.api.server.RequestDetails;
029import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
030import ca.uhn.fhir.rest.server.exceptions.ForbiddenOperationException;
031import ca.uhn.fhir.rest.server.interceptor.auth.AuthorizationConstants;
032import ca.uhn.fhir.util.FhirTerser;
033import org.apache.commons.lang3.Validate;
034import org.hl7.fhir.instance.model.api.IBaseBinary;
035import org.hl7.fhir.instance.model.api.IBaseResource;
036
037import static org.apache.commons.lang3.StringUtils.isNotBlank;
038
039/**
040 * This security interceptor checks any Binary resources that are being exposed to
041 * a user and can forbid the user from accessing them based on the security context
042 * found in <code>Binary.securityContext.identifier</code>.
043 * <p>
044 * This interceptor is intended to be subclassed. The default implementation if it
045 * is not subclassed will reject any access to a Binary resource unless the
046 * request is a system request (using {@link SystemRequestDetails} or the Binary
047 * resource has no value in <code>Binary.securityContext.identifier</code>.
048 * </p>
049 * <p>
050 * Override {@link #securityContextIdentifierAllowed(String, String, RequestDetails)} in order
051 * to allow the user to access specific context values.
052 * </p>
053 *
054 * @since 6.8.0
055 */
056@SuppressWarnings("unused")
057@Interceptor(order = AuthorizationConstants.ORDER_BINARY_SECURITY_INTERCEPTOR)
058public class BinarySecurityContextInterceptor {
059
060        private final FhirContext myFhirContext;
061
062        /**
063         * Constructor
064         *
065         * @param theFhirContext The FHIR context
066         */
067        public BinarySecurityContextInterceptor(FhirContext theFhirContext) {
068                Validate.notNull(theFhirContext, "theFhirContext must not be null");
069                myFhirContext = theFhirContext;
070        }
071
072        /**
073         * Interceptor hook method. Do not call this method directly.
074         */
075        @Hook(Pointcut.STORAGE_PRESHOW_RESOURCES)
076        public void preShowResources(IPreResourceShowDetails theShowDetails, RequestDetails theRequestDetails) {
077                for (IBaseResource nextResource : theShowDetails.getAllResources()) {
078                        if (nextResource instanceof IBaseBinary) {
079                                applyAccessControl((IBaseBinary) nextResource, theRequestDetails);
080                        }
081                }
082        }
083
084        /**
085         * Interceptor hook method. Do not call this method directly.
086         */
087        @Hook(Pointcut.STORAGE_PRECOMMIT_RESOURCE_UPDATED)
088        public void preShowResources(
089                        IBaseResource theOldValue, IBaseResource theNewValue, RequestDetails theRequestDetails) {
090                if (theOldValue instanceof IBaseBinary) {
091                        applyAccessControl((IBaseBinary) theOldValue, theRequestDetails);
092                }
093        }
094
095        /**
096         * This method applies security to a given Binary resource. It is not typically
097         * overridden but you could override it if you wanted to completely replace the
098         * security logic in this interceptor.
099         *
100         * @param theBinary         The Binary resource being checked
101         * @param theRequestDetails The request details associated with this request
102         */
103        protected void applyAccessControl(IBaseBinary theBinary, RequestDetails theRequestDetails) {
104                FhirTerser terser = myFhirContext.newTerser();
105                String securityContextSystem =
106                                terser.getSinglePrimitiveValueOrNull(theBinary, "securityContext.identifier.system");
107                String securityContextValue =
108                                terser.getSinglePrimitiveValueOrNull(theBinary, "securityContext.identifier.value");
109
110                if (isNotBlank(securityContextSystem) || isNotBlank(securityContextValue)) {
111                        applyAccessControl(theBinary, securityContextSystem, securityContextValue, theRequestDetails);
112                }
113        }
114
115        /**
116         * This method applies access controls to a Binary resource containing the
117         * given identifier system and value in the Binary.securityContext element.
118         *
119         * @param theBinary                The binary resource
120         * @param theSecurityContextSystem The identifier system
121         * @param theSecurityContextValue  The identifier value
122         * @param theRequestDetails        The request details
123         */
124        protected void applyAccessControl(
125                        IBaseBinary theBinary,
126                        String theSecurityContextSystem,
127                        String theSecurityContextValue,
128                        RequestDetails theRequestDetails) {
129                if (theRequestDetails instanceof SystemRequestDetails) {
130                        return;
131                }
132                if (securityContextIdentifierAllowed(theSecurityContextSystem, theSecurityContextValue, theRequestDetails)) {
133                        return;
134                }
135
136                handleForbidden(theBinary);
137        }
138
139        /**
140         * Handles a non-permitted operation. This method throws a {@link ForbiddenOperationException}
141         * but you could override it to change that behaviour.
142         */
143        protected void handleForbidden(IBaseBinary theBinary) {
144                throw new ForbiddenOperationException(Msg.code(2369) + "Security context not permitted");
145        }
146
147        /**
148         * Determines whether the current user has access to the given security
149         * context identifier. This method is intended to be overridden, the default
150         * implementation simply always returns <code>false</code>.
151         *
152         * @param theSecurityContextSystem The <code>Binary.securityContext.identifier.system</code> value
153         * @param theSecurityContextValue  The <code>Binary.securityContext.identifier.value</code> value
154         * @param theRequestDetails        The request details associated with this request
155         * @return Returns <code>true</code> if the request should be permitted, and <code>false</code> otherwise
156         */
157        protected boolean securityContextIdentifierAllowed(
158                        String theSecurityContextSystem, String theSecurityContextValue, RequestDetails theRequestDetails) {
159                return false;
160        }
161}