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.s13n; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.fhirpath.FhirPathExecutionException; 024import ca.uhn.fhir.fhirpath.IFhirPath; 025import ca.uhn.fhir.i18n.Msg; 026import ca.uhn.fhir.interceptor.api.Hook; 027import ca.uhn.fhir.interceptor.api.Interceptor; 028import ca.uhn.fhir.interceptor.api.Pointcut; 029import ca.uhn.fhir.rest.api.server.RequestDetails; 030import ca.uhn.fhir.rest.server.interceptor.ConfigLoader; 031import ca.uhn.fhir.rest.server.interceptor.s13n.standardizers.EmailStandardizer; 032import ca.uhn.fhir.rest.server.interceptor.s13n.standardizers.FirstNameStandardizer; 033import ca.uhn.fhir.rest.server.interceptor.s13n.standardizers.IStandardizer; 034import ca.uhn.fhir.rest.server.interceptor.s13n.standardizers.LastNameStandardizer; 035import ca.uhn.fhir.rest.server.interceptor.s13n.standardizers.PhoneStandardizer; 036import ca.uhn.fhir.rest.server.interceptor.s13n.standardizers.TextStandardizer; 037import ca.uhn.fhir.rest.server.interceptor.s13n.standardizers.TitleStandardizer; 038import org.hl7.fhir.instance.model.api.IBase; 039import org.hl7.fhir.instance.model.api.IBaseResource; 040import org.hl7.fhir.instance.model.api.IPrimitiveType; 041import org.slf4j.Logger; 042import org.slf4j.LoggerFactory; 043 044import java.util.HashMap; 045import java.util.List; 046import java.util.Map; 047 048@Interceptor 049public class StandardizingInterceptor { 050 051 /** 052 * Pre-defined standardizers 053 */ 054 public enum StandardizationType { 055 NAME_FAMILY, 056 NAME_GIVEN, 057 EMAIL, 058 TITLE, 059 PHONE, 060 TEXT; 061 } 062 063 public static final String STANDARDIZATION_DISABLED_HEADER = "HAPI-Standardization-Disabled"; 064 065 private static final Logger ourLog = LoggerFactory.getLogger(StandardizingInterceptor.class); 066 067 private Map<String, Map<String, String>> myConfig; 068 private Map<String, IStandardizer> myStandardizers = new HashMap<>(); 069 070 public StandardizingInterceptor() { 071 super(); 072 073 ourLog.info("Starting StandardizingInterceptor {}", this); 074 075 myConfig = ConfigLoader.loadJson("classpath:field-s13n-rules.json", Map.class); 076 initStandardizers(); 077 } 078 079 public StandardizingInterceptor(Map<String, Map<String, String>> theConfig) { 080 super(); 081 myConfig = theConfig; 082 initStandardizers(); 083 } 084 085 public void initStandardizers() { 086 myStandardizers.put(StandardizationType.NAME_FAMILY.name(), new LastNameStandardizer()); 087 myStandardizers.put(StandardizationType.NAME_GIVEN.name(), new FirstNameStandardizer()); 088 myStandardizers.put(StandardizationType.EMAIL.name(), new EmailStandardizer()); 089 myStandardizers.put(StandardizationType.TITLE.name(), new TitleStandardizer()); 090 myStandardizers.put(StandardizationType.PHONE.name(), new PhoneStandardizer()); 091 myStandardizers.put(StandardizationType.TEXT.name(), new TextStandardizer()); 092 093 ourLog.info("Initialized standardizers {}", myStandardizers); 094 } 095 096 @Hook(Pointcut.STORAGE_PRESTORAGE_RESOURCE_CREATED) 097 public void resourcePreCreate(RequestDetails theRequest, IBaseResource theResource) { 098 ourLog.debug("Standardizing on pre-create for - {}, {}", theRequest, theResource); 099 standardize(theRequest, theResource); 100 } 101 102 @Hook(Pointcut.STORAGE_PRESTORAGE_RESOURCE_UPDATED) 103 public void resourcePreUpdate( 104 RequestDetails theRequest, IBaseResource theOldResource, IBaseResource theNewResource) { 105 ourLog.debug("Standardizing on pre-update for - {}, {}, {}", theRequest, theOldResource, theNewResource); 106 standardize(theRequest, theNewResource); 107 } 108 109 private void standardize(RequestDetails theRequest, IBaseResource theResource) { 110 if (theRequest == null) { 111 ourLog.debug("RequestDetails is null - unable to standardize {}", theResource); 112 return; 113 } 114 115 if (!theRequest.getHeaders(STANDARDIZATION_DISABLED_HEADER).isEmpty()) { 116 ourLog.debug( 117 "Standardization for {} is disabled via header {}", theResource, STANDARDIZATION_DISABLED_HEADER); 118 return; 119 } 120 121 if (theResource == null) { 122 ourLog.debug("Nothing to standardize for {}", theRequest); 123 return; 124 } 125 126 FhirContext ctx = theRequest.getFhirContext(); 127 128 String resourceType = ctx.getResourceType(theResource); 129 IFhirPath fhirPath = ctx.newFhirPath(); 130 131 for (Map.Entry<String, Map<String, String>> rule : myConfig.entrySet()) { 132 String resourceFromConfig = rule.getKey(); 133 if (!appliesToResource(resourceFromConfig, resourceType)) { 134 continue; 135 } 136 137 standardize(theResource, rule.getValue(), fhirPath); 138 } 139 } 140 141 private void standardize(IBaseResource theResource, Map<String, String> theRules, IFhirPath theFhirPath) { 142 for (Map.Entry<String, String> rule : theRules.entrySet()) { 143 IStandardizer std = getStandardizer(rule); 144 List<IBase> values; 145 try { 146 values = theFhirPath.evaluate(theResource, rule.getKey(), IBase.class); 147 } catch (FhirPathExecutionException e) { 148 ourLog.warn("Unable to evaluate path at {} for {}", rule.getKey(), theResource); 149 return; 150 } 151 152 for (IBase v : values) { 153 if (!(v instanceof IPrimitiveType)) { 154 ourLog.warn( 155 "Value at path {} is of type {}, which is not of primitive type - skipping", 156 rule.getKey(), 157 v.fhirType()); 158 continue; 159 } 160 IPrimitiveType<?> value = (IPrimitiveType<?>) v; 161 String valueString = value.getValueAsString(); 162 String standardizedValueString = std.standardize(valueString); 163 value.setValueAsString(standardizedValueString); 164 ourLog.debug("Standardized {} to {}", valueString, standardizedValueString); 165 } 166 } 167 } 168 169 private IStandardizer getStandardizer(Map.Entry<String, String> rule) { 170 String standardizerName = rule.getValue(); 171 if (myStandardizers.containsKey(standardizerName)) { 172 return myStandardizers.get(standardizerName); 173 } 174 175 IStandardizer standardizer; 176 try { 177 standardizer = (IStandardizer) 178 Class.forName(standardizerName).getDeclaredConstructor().newInstance(); 179 } catch (Exception e) { 180 throw new RuntimeException( 181 Msg.code(349) + String.format("Unable to create standardizer %s", standardizerName), e); 182 } 183 184 myStandardizers.put(standardizerName, standardizer); 185 return standardizer; 186 } 187 188 private boolean appliesToResource(String theResourceFromConfig, String theActualResourceType) { 189 return theResourceFromConfig.equals(theActualResourceType); 190 } 191 192 public Map<String, Map<String, String>> getConfig() { 193 return myConfig; 194 } 195 196 public void setConfig(Map<String, Map<String, String>> theConfig) { 197 myConfig = theConfig; 198 } 199 200 public Map<String, IStandardizer> getStandardizers() { 201 return myStandardizers; 202 } 203 204 public void setStandardizers(Map<String, IStandardizer> theStandardizers) { 205 myStandardizers = theStandardizers; 206 } 207}