001/*- 002 * #%L 003 * HAPI FHIR - Master Data Management 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.mdm.rules.svc; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.fhirpath.IFhirPath; 024import ca.uhn.fhir.i18n.Msg; 025import ca.uhn.fhir.mdm.api.MdmMatchEvaluation; 026import ca.uhn.fhir.mdm.rules.json.MdmFieldMatchJson; 027import ca.uhn.fhir.mdm.rules.json.MdmMatcherJson; 028import ca.uhn.fhir.mdm.rules.json.MdmRulesJson; 029import ca.uhn.fhir.mdm.rules.json.MdmSimilarityJson; 030import ca.uhn.fhir.mdm.rules.matcher.IMatcherFactory; 031import ca.uhn.fhir.mdm.rules.matcher.models.IMdmFieldMatcher; 032import ca.uhn.fhir.mdm.rules.matcher.models.MatchTypeEnum; 033import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 034import ca.uhn.fhir.util.FhirTerser; 035import org.apache.commons.lang3.Validate; 036import org.hl7.fhir.instance.model.api.IBase; 037import org.hl7.fhir.instance.model.api.IBaseResource; 038 039import java.util.List; 040import java.util.stream.Collectors; 041 042import static ca.uhn.fhir.mdm.api.MdmConstants.ALL_RESOURCE_SEARCH_PARAM_TYPE; 043 044/** 045 * This class is responsible for performing matching between raw-typed values of a left record and a right record. 046 */ 047public class MdmResourceFieldMatcher { 048 049 private final FhirContext myFhirContext; 050 private final MdmFieldMatchJson myMdmFieldMatchJson; 051 private final String myResourceType; 052 private final String myResourcePath; 053 private final String myFhirPath; 054 private final MdmRulesJson myMdmRulesJson; 055 private final String myName; 056 private final boolean myIsFhirPathExpression; 057 058 private final IMatcherFactory myIMatcherFactory; 059 060 public MdmResourceFieldMatcher( 061 FhirContext theFhirContext, 062 IMatcherFactory theIMatcherFactory, 063 MdmFieldMatchJson theMdmFieldMatchJson, 064 MdmRulesJson theMdmRulesJson) { 065 myIMatcherFactory = theIMatcherFactory; 066 067 myFhirContext = theFhirContext; 068 myMdmFieldMatchJson = theMdmFieldMatchJson; 069 myResourceType = theMdmFieldMatchJson.getResourceType(); 070 myResourcePath = theMdmFieldMatchJson.getResourcePath(); 071 myFhirPath = theMdmFieldMatchJson.getFhirPath(); 072 myName = theMdmFieldMatchJson.getName(); 073 myMdmRulesJson = theMdmRulesJson; 074 myIsFhirPathExpression = myFhirPath != null; 075 } 076 077 /** 078 * Compares two {@link IBaseResource}s and determines if they match, using the algorithm defined in this object's 079 * {@link MdmFieldMatchJson}. 080 * <p> 081 * In this implementation, it determines whether a given field matches between two resources. Internally this is evaluated using FhirPath. If any of the elements of theLeftResource 082 * match any of the elements of theRightResource, will return true. Otherwise, false. 083 * 084 * @param theLeftResource the first {@link IBaseResource} 085 * @param theRightResource the second {@link IBaseResource} 086 * @return A boolean indicating whether they match. 087 */ 088 public MdmMatchEvaluation match(IBaseResource theLeftResource, IBaseResource theRightResource) { 089 validate(theLeftResource); 090 validate(theRightResource); 091 092 List<IBase> leftValues; 093 List<IBase> rightValues; 094 095 if (myIsFhirPathExpression) { 096 IFhirPath fhirPath = myFhirContext.newFhirPath(); 097 leftValues = fhirPath.evaluate(theLeftResource, myFhirPath, IBase.class); 098 rightValues = fhirPath.evaluate(theRightResource, myFhirPath, IBase.class); 099 } else { 100 FhirTerser fhirTerser = myFhirContext.newTerser(); 101 leftValues = fhirTerser.getValues(theLeftResource, myResourcePath, IBase.class); 102 rightValues = fhirTerser.getValues(theRightResource, myResourcePath, IBase.class); 103 } 104 return match(leftValues, rightValues); 105 } 106 107 private MdmMatchEvaluation match(List<IBase> theLeftValues, List<IBase> theRightValues) { 108 MdmMatchEvaluation retval = new MdmMatchEvaluation(false, 0.0); 109 110 boolean isMatchingEmptyFieldValues = (theLeftValues.isEmpty() && theRightValues.isEmpty()); 111 IMdmFieldMatcher matcher = getFieldMatcher(); 112 if (isMatchingEmptyFieldValues && (matcher != null && matcher.isMatchingEmptyFields())) { 113 return match((IBase) null, (IBase) null); 114 } 115 116 for (IBase leftValue : theLeftValues) { 117 for (IBase rightValue : theRightValues) { 118 MdmMatchEvaluation nextMatch = match(leftValue, rightValue); 119 retval = MdmMatchEvaluation.max(retval, nextMatch); 120 } 121 } 122 123 return retval; 124 } 125 126 private MdmMatchEvaluation match(IBase theLeftValue, IBase theRightValue) { 127 IMdmFieldMatcher matcher = getFieldMatcher(); 128 if (matcher != null) { 129 boolean isMatches = matcher.matches(theLeftValue, theRightValue, myMdmFieldMatchJson.getMatcher()); 130 return new MdmMatchEvaluation(isMatches, isMatches ? 1.0 : 0.0); 131 } 132 133 MdmSimilarityJson similarity = myMdmFieldMatchJson.getSimilarity(); 134 if (similarity != null) { 135 return similarity.match(myFhirContext, theLeftValue, theRightValue); 136 } 137 138 throw new InternalErrorException( 139 Msg.code(1522) + "Field Match " + myName + " has neither a matcher nor a similarity."); 140 } 141 142 private void validate(IBaseResource theResource) { 143 String resourceType = myFhirContext.getResourceType(theResource); 144 Validate.notNull(resourceType, "Resource type may not be null"); 145 146 if (ALL_RESOURCE_SEARCH_PARAM_TYPE.equals(myResourceType)) { 147 boolean isMdmType = 148 myMdmRulesJson.getMdmTypes().stream().anyMatch(mdmType -> mdmType.equalsIgnoreCase(resourceType)); 149 Validate.isTrue( 150 isMdmType, 151 "Expecting resource type %s, got resource type %s", 152 myMdmRulesJson.getMdmTypes().stream().collect(Collectors.joining(",")), 153 resourceType); 154 } else { 155 Validate.isTrue( 156 myResourceType.equals(resourceType), 157 "Expecting resource type %s got resource type %s", 158 myResourceType, 159 resourceType); 160 } 161 } 162 163 public String getResourceType() { 164 return myResourceType; 165 } 166 167 public String getResourcePath() { 168 return myResourcePath; 169 } 170 171 public String getName() { 172 return myName; 173 } 174 175 private IMdmFieldMatcher getFieldMatcher() { 176 MdmMatcherJson matcherJson = myMdmFieldMatchJson.getMatcher(); 177 MatchTypeEnum matchTypeEnum = null; 178 if (matcherJson != null) { 179 matchTypeEnum = matcherJson.getAlgorithm(); 180 } 181 if (matchTypeEnum == null) { 182 return null; 183 } 184 185 return myIMatcherFactory.getFieldMatcherForMatchType(matchTypeEnum); 186 } 187}