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.interceptor.api.Pointcut;
023import ca.uhn.fhir.model.primitive.IdDt;
024import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
025import ca.uhn.fhir.rest.api.server.RequestDetails;
026import ca.uhn.fhir.rest.api.server.bulk.BulkExportJobParameters;
027import com.google.common.annotations.VisibleForTesting;
028import org.hl7.fhir.instance.model.api.IBaseResource;
029import org.hl7.fhir.instance.model.api.IIdType;
030
031import java.util.ArrayList;
032import java.util.Collection;
033import java.util.Objects;
034import java.util.Set;
035import java.util.stream.Collectors;
036
037import static org.apache.commons.collections4.CollectionUtils.isEmpty;
038import static org.apache.commons.collections4.CollectionUtils.isNotEmpty;
039import static org.apache.commons.lang3.StringUtils.isNotBlank;
040
041public class RuleBulkExportImpl extends BaseRule {
042        private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(RuleBulkExportImpl.class);
043        private String myGroupId;
044        private final Collection<String> myPatientIds;
045        private boolean myAppliesToAllPatients;
046        private BulkExportJobParameters.ExportStyle myWantExportStyle;
047        private Collection<String> myResourceTypes;
048        private boolean myWantAnyStyle;
049
050        RuleBulkExportImpl(String theRuleName) {
051                super(theRuleName);
052                myPatientIds = new ArrayList<>();
053        }
054
055        @Override
056        public AuthorizationInterceptor.Verdict applyRule(
057                        RestOperationTypeEnum theOperation,
058                        RequestDetails theRequestDetails,
059                        IBaseResource theInputResource,
060                        IIdType theInputResourceId,
061                        IBaseResource theOutputResource,
062                        IRuleApplier theRuleApplier,
063                        Set<AuthorizationFlagsEnum> theFlags,
064                        Pointcut thePointcut) {
065                if (thePointcut != Pointcut.STORAGE_INITIATE_BULK_EXPORT) {
066                        return null;
067                }
068
069                if (theRequestDetails == null) {
070                        return null;
071                }
072
073                BulkExportJobParameters inboundBulkExportRequestOptions = (BulkExportJobParameters)
074                                theRequestDetails.getAttribute(AuthorizationInterceptor.REQUEST_ATTRIBUTE_BULK_DATA_EXPORT_OPTIONS);
075                // if style doesn't match - abstain
076                if (!myWantAnyStyle && inboundBulkExportRequestOptions.getExportStyle() != myWantExportStyle) {
077                        return null;
078                }
079
080                // Do we only authorize some types?  If so, make sure requested types are a subset
081                if (isNotEmpty(myResourceTypes)) {
082                        if (isEmpty(inboundBulkExportRequestOptions.getResourceTypes())) {
083                                return new AuthorizationInterceptor.Verdict(PolicyEnum.DENY, this);
084                        }
085                        if (!myResourceTypes.containsAll(inboundBulkExportRequestOptions.getResourceTypes())) {
086                                return new AuthorizationInterceptor.Verdict(PolicyEnum.DENY, this);
087                        }
088                }
089
090                // system only supports filtering by resource type.  So if we are system, or any(), then allow, since we have
091                // done resource type checking
092                // above
093                AuthorizationInterceptor.Verdict allowVerdict = newVerdict(
094                                theOperation,
095                                theRequestDetails,
096                                theInputResource,
097                                theInputResourceId,
098                                theOutputResource,
099                                theRuleApplier);
100
101                if (myWantAnyStyle || myWantExportStyle == BulkExportJobParameters.ExportStyle.SYSTEM) {
102                        return allowVerdict;
103                }
104
105                // assume myGroupId not empty->myStyle is group.  If target group matches, then allow.
106                if (isNotBlank(myGroupId) && inboundBulkExportRequestOptions.getGroupId() != null) {
107                        String expectedGroupId =
108                                        new IdDt(myGroupId).toUnqualifiedVersionless().getValue();
109                        String actualGroupId = new IdDt(inboundBulkExportRequestOptions.getGroupId())
110                                        .toUnqualifiedVersionless()
111                                        .getValue();
112                        if (Objects.equals(expectedGroupId, actualGroupId)) {
113                                return allowVerdict;
114                        }
115                }
116                // patient export mode - instance or type.  type can have 0..n patient ids.
117                // myPatientIds == the rules built by the auth interceptor rule builder
118                // options.getPatientIds() == the requested IDs in the export job.
119
120                // 1. If each of the requested resource IDs in the parameters are present in the users permissions, Approve
121                // 2. If any requested ID is not present in the users permissions, Deny.
122                if (myWantExportStyle == BulkExportJobParameters.ExportStyle.PATIENT)
123                        // Unfiltered Type Level
124                        if (myAppliesToAllPatients) {
125                                return allowVerdict;
126                        }
127
128                // Instance level, or filtered type level
129                if (isNotEmpty(myPatientIds)) {
130                        // If bulk export options defines no patient IDs, return null.
131                        if (inboundBulkExportRequestOptions.getPatientIds().isEmpty()) {
132                                return null;
133                        } else {
134                                ourLog.debug("options.getPatientIds() != null");
135                                Set<String> requestedPatientIds = sanitizeIds(inboundBulkExportRequestOptions.getPatientIds());
136                                Set<String> permittedPatientIds = sanitizeIds(myPatientIds);
137                                if (permittedPatientIds.containsAll(requestedPatientIds)) {
138                                        return allowVerdict;
139                                } else {
140                                        return new AuthorizationInterceptor.Verdict(PolicyEnum.DENY, this);
141                                }
142                        }
143                }
144                return null;
145        }
146
147        private Set<String> sanitizeIds(Collection<String> myPatientIds) {
148                return myPatientIds.stream()
149                                .map(id -> new IdDt(id).toUnqualifiedVersionless().getValue())
150                                .collect(Collectors.toSet());
151        }
152
153        public void setAppliesToGroupExportOnGroup(String theGroupId) {
154                myWantExportStyle = BulkExportJobParameters.ExportStyle.GROUP;
155                myGroupId = theGroupId;
156        }
157
158        public void setAppliesToPatientExportOnGroup(String theGroupId) {
159                myWantExportStyle = BulkExportJobParameters.ExportStyle.PATIENT;
160                myGroupId = theGroupId;
161        }
162
163        public void setAppliesToPatientExport(String thePatientId) {
164                myWantExportStyle = BulkExportJobParameters.ExportStyle.PATIENT;
165                myPatientIds.add(thePatientId);
166        }
167
168        public void setAppliesToPatientExport(Collection<String> thePatientIds) {
169                myWantExportStyle = BulkExportJobParameters.ExportStyle.PATIENT;
170                myPatientIds.addAll(thePatientIds);
171        }
172
173        public void setAppliesToPatientExportAllPatients() {
174                myWantExportStyle = BulkExportJobParameters.ExportStyle.PATIENT;
175                myAppliesToAllPatients = true;
176        }
177
178        public void setAppliesToSystem() {
179                myWantExportStyle = BulkExportJobParameters.ExportStyle.SYSTEM;
180        }
181
182        public void setResourceTypes(Collection<String> theResourceTypes) {
183                myResourceTypes = theResourceTypes;
184        }
185
186        public void setAppliesToAny() {
187                myWantAnyStyle = true;
188        }
189
190        String getGroupId() {
191                return myGroupId;
192        }
193
194        BulkExportJobParameters.ExportStyle getWantExportStyle() {
195                return myWantExportStyle;
196        }
197
198        @VisibleForTesting
199        Collection<String> getPatientIds() {
200                return myPatientIds;
201        }
202
203        @VisibleForTesting
204        Collection<String> getResourceTypes() {
205                return myResourceTypes;
206        }
207}