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.json; 021 022import ca.uhn.fhir.mdm.api.MdmMatchResultEnum; 023import ca.uhn.fhir.model.api.IModelJson; 024import com.fasterxml.jackson.annotation.JsonProperty; 025import com.fasterxml.jackson.databind.annotation.JsonDeserialize; 026import com.fasterxml.jackson.databind.util.StdConverter; 027import com.google.common.annotations.VisibleForTesting; 028import org.apache.commons.lang3.StringUtils; 029import org.apache.commons.lang3.Validate; 030 031import java.util.ArrayList; 032import java.util.Collections; 033import java.util.HashMap; 034import java.util.List; 035import java.util.Map; 036import java.util.Set; 037import java.util.stream.Collectors; 038 039import static ca.uhn.fhir.mdm.api.MdmConstants.ALL_RESOURCE_SEARCH_PARAM_TYPE; 040 041@JsonDeserialize(converter = MdmRulesJson.MdmRulesJsonConverter.class) 042public class MdmRulesJson implements IModelJson { 043 044 @JsonProperty(value = "version", required = true) 045 String myVersion; 046 047 @JsonProperty(value = "candidateSearchParams", required = true) 048 List<MdmResourceSearchParamJson> myCandidateSearchParams = new ArrayList<>(); 049 050 @JsonProperty(value = "candidateFilterSearchParams", required = true) 051 List<MdmFilterSearchParamJson> myCandidateFilterSearchParams = new ArrayList<>(); 052 053 @JsonProperty(value = "matchFields", required = true) 054 List<MdmFieldMatchJson> myMatchFieldJsonList = new ArrayList<>(); 055 056 @JsonProperty(value = "matchResultMap", required = true) 057 Map<String, MdmMatchResultEnum> myMatchResultMap = new HashMap<>(); 058 059 /** 060 * This field is deprecated, use eidSystems instead. 061 */ 062 @Deprecated 063 @JsonProperty(value = "eidSystem") 064 String myEnterpriseEIDSystem; 065 066 @JsonProperty(value = "eidSystems") 067 Map<String, String> myEnterpriseEidSystems = new HashMap<>(); 068 069 @JsonProperty(value = "mdmTypes") 070 List<String> myMdmTypes; 071 072 transient VectorMatchResultMap myVectorMatchResultMap; 073 074 public void addMatchField(MdmFieldMatchJson theMatchRuleName) { 075 myMatchFieldJsonList.add(theMatchRuleName); 076 } 077 078 public void addResourceSearchParam(MdmResourceSearchParamJson theSearchParam) { 079 myCandidateSearchParams.add(theSearchParam); 080 } 081 082 public void addFilterSearchParam(MdmFilterSearchParamJson theSearchParam) { 083 myCandidateFilterSearchParams.add(theSearchParam); 084 } 085 086 int size() { 087 return myMatchFieldJsonList.size(); 088 } 089 090 MdmFieldMatchJson get(int theIndex) { 091 return myMatchFieldJsonList.get(theIndex); 092 } 093 094 MdmMatchResultEnum getMatchResult(String theFieldMatchNames) { 095 return myMatchResultMap.get(theFieldMatchNames); 096 } 097 098 public MdmMatchResultEnum getMatchResult(Long theMatchVector) { 099 return myVectorMatchResultMap.get(theMatchVector); 100 } 101 102 public void putMatchResult(String theFieldMatchNames, MdmMatchResultEnum theMatchResult) { 103 myMatchResultMap.put(theFieldMatchNames, theMatchResult); 104 initialize(); 105 } 106 107 Map<String, MdmMatchResultEnum> getMatchResultMap() { 108 return Collections.unmodifiableMap(myMatchResultMap); 109 } 110 111 /** 112 * Must call initialize() before calling getMatchResult(Long) 113 */ 114 public void initialize() { 115 myVectorMatchResultMap = new VectorMatchResultMap(this); 116 } 117 118 public List<MdmFieldMatchJson> getMatchFields() { 119 return Collections.unmodifiableList(myMatchFieldJsonList); 120 } 121 122 public List<MdmResourceSearchParamJson> getCandidateSearchParams() { 123 return Collections.unmodifiableList(myCandidateSearchParams); 124 } 125 126 public List<MdmFilterSearchParamJson> getCandidateFilterSearchParams() { 127 return Collections.unmodifiableList(myCandidateFilterSearchParams); 128 } 129 130 /** 131 * Use {@link this#getEnterpriseEIDSystemForResourceType(String)} instead. 132 */ 133 @Deprecated 134 public String getEnterpriseEIDSystem() { 135 return myEnterpriseEIDSystem; 136 } 137 138 /** 139 * Use {@link this#setEnterpriseEIDSystems(Map)} (String)} or {@link this#addEnterpriseEIDSystem(String, String)} instead. 140 */ 141 @Deprecated 142 public void setEnterpriseEIDSystem(String theEnterpriseEIDSystem) { 143 myEnterpriseEIDSystem = theEnterpriseEIDSystem; 144 } 145 146 public void setEnterpriseEIDSystems(Map<String, String> theEnterpriseEIDSystems) { 147 myEnterpriseEidSystems = theEnterpriseEIDSystems; 148 } 149 150 public void addEnterpriseEIDSystem(String theResourceType, String theEidSystem) { 151 if (myEnterpriseEidSystems == null) { 152 myEnterpriseEidSystems = new HashMap<>(); 153 } 154 myEnterpriseEidSystems.put(theResourceType, theEidSystem); 155 } 156 157 public Map<String, String> getEnterpriseEIDSystems() { 158 // First try the new property. 159 if (myEnterpriseEidSystems != null && !myEnterpriseEidSystems.isEmpty()) { 160 return myEnterpriseEidSystems; 161 // If that fails, fall back to our deprecated property. 162 } else if (!StringUtils.isBlank(myEnterpriseEIDSystem)) { 163 HashMap<String, String> retVal = new HashMap<>(); 164 retVal.put(ALL_RESOURCE_SEARCH_PARAM_TYPE, myEnterpriseEIDSystem); 165 return retVal; 166 // Otherwise, return an empty map. 167 } else { 168 return Collections.emptyMap(); 169 } 170 } 171 172 public String getEnterpriseEIDSystemForResourceType(String theResourceType) { 173 Map<String, String> enterpriseEIDSystems = getEnterpriseEIDSystems(); 174 if (enterpriseEIDSystems.containsKey(ALL_RESOURCE_SEARCH_PARAM_TYPE)) { 175 return enterpriseEIDSystems.get(ALL_RESOURCE_SEARCH_PARAM_TYPE); 176 } else { 177 return enterpriseEIDSystems.get(theResourceType); 178 } 179 } 180 181 public String getVersion() { 182 return myVersion; 183 } 184 185 public MdmRulesJson setVersion(String theVersion) { 186 myVersion = theVersion; 187 return this; 188 } 189 190 private void validate() { 191 Validate.notBlank(myVersion, "version may not be blank"); 192 193 Map<String, String> enterpriseEIDSystems = getEnterpriseEIDSystems(); 194 195 // If we have a * eid system, there should only be one. 196 if (enterpriseEIDSystems.containsKey(ALL_RESOURCE_SEARCH_PARAM_TYPE)) { 197 Validate.isTrue(enterpriseEIDSystems.size() == 1); 198 } 199 } 200 201 public String getSummary() { 202 return myCandidateSearchParams.size() + " Candidate Search Params, " + myCandidateFilterSearchParams.size() 203 + " Filter Search Params, " + myMatchFieldJsonList.size() 204 + " Match Fields, " + myMatchResultMap.size() 205 + " Match Result Entries"; 206 } 207 208 public String getFieldMatchNamesForVector(long theVector) { 209 return myVectorMatchResultMap.getFieldMatchNames(theVector); 210 } 211 212 public Set<Map.Entry<String, MdmMatchResultEnum>> getMatchedRulesFromVectorMap(Long theLong) { 213 Set<String> matchedRules = myVectorMatchResultMap.getMatchedRules(theLong); 214 return myMatchResultMap.entrySet().stream() 215 .filter(e -> matchedRules.contains(e.getKey())) 216 .collect(Collectors.toSet()); 217 } 218 219 public String getDetailedFieldMatchResultWithSuccessInformation(long theVector) { 220 List<String> fieldMatchResult = new ArrayList<>(); 221 for (int i = 0; i < myMatchFieldJsonList.size(); ++i) { 222 if ((theVector & (1 << i)) == 0) { 223 fieldMatchResult.add(myMatchFieldJsonList.get(i).getName() + ": NO"); 224 } else { 225 fieldMatchResult.add(myMatchFieldJsonList.get(i).getName() + ": YES"); 226 } 227 } 228 return String.join("\n", fieldMatchResult); 229 } 230 231 @VisibleForTesting 232 VectorMatchResultMap getVectorMatchResultMapForUnitTest() { 233 return myVectorMatchResultMap; 234 } 235 236 /** 237 * Ensure the vector map is initialized after we deserialize 238 */ 239 static class MdmRulesJsonConverter extends StdConverter<MdmRulesJson, MdmRulesJson> { 240 241 /** 242 * This empty constructor is required by Jackson 243 */ 244 public MdmRulesJsonConverter() {} 245 246 @Override 247 public MdmRulesJson convert(MdmRulesJson theMdmRulesJson) { 248 theMdmRulesJson.validate(); 249 theMdmRulesJson.initialize(); 250 return theMdmRulesJson; 251 } 252 } 253 254 public List<String> getMdmTypes() { 255 return myMdmTypes; 256 } 257 258 public void setMdmTypes(List<String> theMdmTypes) { 259 myMdmTypes = theMdmTypes; 260 } 261}