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.context.FhirContext; 023import ca.uhn.fhir.context.support.IValidationSupport; 024import ca.uhn.fhir.rest.api.server.RequestDetails; 025import ca.uhn.fhir.rest.server.interceptor.consent.ConsentOutcome; 026import ca.uhn.fhir.rest.server.interceptor.consent.IConsentContextServices; 027import ca.uhn.fhir.rest.server.interceptor.consent.IConsentService; 028import ca.uhn.fhir.rest.server.util.FhirContextSearchParamRegistry; 029import ca.uhn.fhir.rest.server.util.ISearchParamRegistry; 030import jakarta.annotation.Nonnull; 031import org.apache.commons.lang3.Validate; 032import org.hl7.fhir.instance.model.api.IBaseResource; 033import org.slf4j.Logger; 034import org.slf4j.LoggerFactory; 035 036import java.util.List; 037 038public class SearchNarrowingConsentService implements IConsentService { 039 private static final Logger ourLog = LoggerFactory.getLogger(SearchNarrowingConsentService.class); 040 041 private final IValidationSupport myValidationSupport; 042 private final ISearchParamRegistry mySearchParamRegistry; 043 private Logger myTroubleshootingLog = ourLog; 044 045 /** 046 * Constructor (use this only if no {@link ISearchParamRegistry} is available 047 * 048 * @param theValidationSupport The validation support module 049 */ 050 public SearchNarrowingConsentService(IValidationSupport theValidationSupport, FhirContext theFhirContext) { 051 this(theValidationSupport, new FhirContextSearchParamRegistry(theFhirContext)); 052 } 053 054 /** 055 * Constructor 056 * 057 * @param theValidationSupport The validation support module 058 * @param theSearchParamRegistry The search param registry 059 */ 060 public SearchNarrowingConsentService( 061 IValidationSupport theValidationSupport, ISearchParamRegistry theSearchParamRegistry) { 062 myValidationSupport = theValidationSupport; 063 mySearchParamRegistry = theSearchParamRegistry; 064 } 065 066 /** 067 * Provides a log that will be apppended to for troubleshooting messages 068 * 069 * @param theTroubleshootingLog The logger (must not be <code>null</code>) 070 */ 071 public void setTroubleshootingLog(@Nonnull Logger theTroubleshootingLog) { 072 Validate.notNull(theTroubleshootingLog, "theTroubleshootingLog must not be null"); 073 myTroubleshootingLog = theTroubleshootingLog; 074 } 075 076 @Override 077 public boolean shouldProcessCanSeeResource( 078 RequestDetails theRequestDetails, IConsentContextServices theContextServices) { 079 List<AllowedCodeInValueSet> postFilteringList = 080 SearchNarrowingInterceptor.getPostFilteringListOrNull(theRequestDetails); 081 return postFilteringList != null && !postFilteringList.isEmpty(); 082 } 083 084 @Override 085 public ConsentOutcome canSeeResource( 086 RequestDetails theRequestDetails, IBaseResource theResource, IConsentContextServices theContextServices) { 087 return applyFilterForResource(theRequestDetails, theResource); 088 } 089 090 @Override 091 public ConsentOutcome willSeeResource( 092 RequestDetails theRequestDetails, IBaseResource theResource, IConsentContextServices theContextServices) { 093 return applyFilterForResource(theRequestDetails, theResource); 094 } 095 096 @Nonnull 097 private ConsentOutcome applyFilterForResource(RequestDetails theRequestDetails, IBaseResource theResource) { 098 List<AllowedCodeInValueSet> postFilteringList = 099 SearchNarrowingInterceptor.getPostFilteringListOrNull(theRequestDetails); 100 if (postFilteringList == null) { 101 return ConsentOutcome.PROCEED; 102 } 103 104 String resourceType = myValidationSupport.getFhirContext().getResourceType(theResource); 105 106 boolean allPositiveRulesMatched = true; 107 for (AllowedCodeInValueSet next : postFilteringList) { 108 if (!next.getResourceName().equals(resourceType)) { 109 continue; 110 } 111 112 boolean returnOnFirstMatch = true; 113 String searchParamName = next.getSearchParameterName(); 114 String valueSetUrl = next.getValueSetUrl(); 115 116 SearchParameterAndValueSetRuleImpl.CodeMatchCount outcome = 117 SearchParameterAndValueSetRuleImpl.countMatchingCodesInValueSetForSearchParameter( 118 theResource, 119 myValidationSupport, 120 mySearchParamRegistry, 121 returnOnFirstMatch, 122 searchParamName, 123 valueSetUrl, 124 myTroubleshootingLog, 125 "Search Narrowing"); 126 if (outcome.isAtLeastOneUnableToValidate()) { 127 myTroubleshootingLog.warn("Terminology Services failed to validate value from " + next.getResourceName() 128 + ":" + next.getSearchParameterName() + " in ValueSet " + next.getValueSetUrl() 129 + " - Assuming REJECT"); 130 return ConsentOutcome.REJECT; 131 } 132 133 if (next.isNegate()) { 134 if (outcome.getMatchingCodeCount() > 0) { 135 return ConsentOutcome.REJECT; 136 } 137 } else { 138 if (outcome.getMatchingCodeCount() == 0) { 139 allPositiveRulesMatched = false; 140 break; 141 } 142 } 143 } 144 145 if (!allPositiveRulesMatched) { 146 return ConsentOutcome.REJECT; 147 } 148 149 return ConsentOutcome.PROCEED; 150 } 151}