001/*- 002 * #%L 003 * HAPI FHIR - Core Library 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.util; 021 022import ca.uhn.fhir.context.phonetic.ApacheEncoder; 023import ca.uhn.fhir.context.phonetic.IPhoneticEncoder; 024import ca.uhn.fhir.context.phonetic.NumericEncoder; 025import ca.uhn.fhir.context.phonetic.PhoneticEncoderEnum; 026import org.apache.commons.codec.language.Caverphone1; 027import org.apache.commons.codec.language.Caverphone2; 028import org.apache.commons.codec.language.ColognePhonetic; 029import org.apache.commons.codec.language.DoubleMetaphone; 030import org.apache.commons.codec.language.MatchRatingApproachEncoder; 031import org.apache.commons.codec.language.Metaphone; 032import org.apache.commons.codec.language.Nysiis; 033import org.apache.commons.codec.language.RefinedSoundex; 034import org.apache.commons.codec.language.Soundex; 035import org.apache.commons.lang3.EnumUtils; 036import org.slf4j.Logger; 037import org.slf4j.LoggerFactory; 038 039public final class PhoneticEncoderUtil { 040 041 // embedded class only for parameter returns 042 private static class ParsedValues { 043 private final Integer maxCodeLength; 044 private final String encoderString; 045 046 public ParsedValues(String theString, Integer theMaxCode) { 047 maxCodeLength = theMaxCode; 048 encoderString = theString; 049 } 050 051 public Integer getMaxCodeLength() { 052 return maxCodeLength; 053 } 054 055 public String getEncoderString() { 056 return encoderString; 057 } 058 } 059 060 private static final Logger ourLog = LoggerFactory.getLogger(PhoneticEncoderUtil.class); 061 062 private PhoneticEncoderUtil() {} 063 064 /** 065 * Creates the phonetic encoder wrapper from 066 * an input string. 067 * 068 * <p> 069 * String must be in the format of... 070 * </p> 071 * 072 * PhoneticEncoderEnum(MAX_LENGTH) 073 * 074 * @return The IPhoneticEncoder 075 */ 076 public static IPhoneticEncoder getEncoder(String theString) { 077 ParsedValues values = parseIntValue(theString); 078 String encoderType = values.getEncoderString(); 079 Integer encoderMaxString = values.getMaxCodeLength(); 080 081 IPhoneticEncoder encoder = getEncoderFromString(encoderType, encoderMaxString); 082 if (encoder != null) { 083 return encoder; 084 } else { 085 ourLog.warn("Invalid phonetic param string " + theString); 086 return null; 087 } 088 } 089 090 private static ParsedValues parseIntValue(String theString) { 091 String encoderType = null; 092 Integer encoderMaxString = null; 093 094 int braceIndex = theString.indexOf("("); 095 if (braceIndex != -1) { 096 int len = theString.length(); 097 if (theString.charAt(len - 1) == ')') { 098 encoderType = theString.substring(0, braceIndex); 099 String num = theString.substring(braceIndex + 1, len - 1); 100 try { 101 encoderMaxString = Integer.parseInt(num); 102 } catch (NumberFormatException ex) { 103 // invalid number parse error 104 } 105 106 if (encoderMaxString == null || encoderMaxString < 0) { 107 // parse error 108 ourLog.error("Invalid encoder max character length: " + num); 109 encoderType = null; 110 } 111 } 112 // else - parse error 113 } else { 114 encoderType = theString; 115 } 116 117 return new ParsedValues(encoderType, encoderMaxString); 118 } 119 120 private static IPhoneticEncoder getEncoderFromString(String theName, Integer theMax) { 121 IPhoneticEncoder encoder = null; 122 PhoneticEncoderEnum enumVal = EnumUtils.getEnum(PhoneticEncoderEnum.class, theName); 123 124 if (enumVal != null) { 125 switch (enumVal) { 126 case CAVERPHONE1: 127 Caverphone1 caverphone1 = new Caverphone1(); 128 encoder = new ApacheEncoder(theName, caverphone1); 129 break; 130 case CAVERPHONE2: 131 Caverphone2 caverphone2 = new Caverphone2(); 132 encoder = new ApacheEncoder(theName, caverphone2); 133 break; 134 case COLOGNE: 135 ColognePhonetic colognePhonetic = new ColognePhonetic(); 136 encoder = new ApacheEncoder(theName, colognePhonetic); 137 break; 138 case DOUBLE_METAPHONE: 139 DoubleMetaphone doubleMetaphone = new DoubleMetaphone(); 140 if (theMax != null) { 141 doubleMetaphone.setMaxCodeLen(theMax); 142 } 143 encoder = new ApacheEncoder(theName, doubleMetaphone); 144 break; 145 case MATCH_RATING_APPROACH: 146 MatchRatingApproachEncoder matchRatingApproachEncoder = new MatchRatingApproachEncoder(); 147 encoder = new ApacheEncoder(theName, matchRatingApproachEncoder); 148 break; 149 case METAPHONE: 150 Metaphone metaphone = new Metaphone(); 151 if (theMax != null) { 152 metaphone.setMaxCodeLen(theMax); 153 } 154 encoder = new ApacheEncoder(theName, metaphone); 155 break; 156 case NYSIIS: 157 Nysiis nysiis = new Nysiis(); 158 encoder = new ApacheEncoder(theName, nysiis); 159 break; 160 case NYSIIS_LONG: 161 Nysiis nysiis1_long = new Nysiis(false); 162 encoder = new ApacheEncoder(theName, nysiis1_long); 163 break; 164 case REFINED_SOUNDEX: 165 RefinedSoundex refinedSoundex = new RefinedSoundex(); 166 encoder = new ApacheEncoder(theName, refinedSoundex); 167 break; 168 case SOUNDEX: 169 Soundex soundex = new Soundex(); 170 // soundex has deprecated setting the max size 171 encoder = new ApacheEncoder(theName, soundex); 172 break; 173 case NUMERIC: 174 encoder = new NumericEncoder(); 175 break; 176 default: 177 // we don't ever expect to be here 178 // this log message is purely for devs who update this 179 // enum, but not this method 180 ourLog.error("Unhandled PhoneticParamEnum value " + enumVal.name()); 181 break; 182 } 183 } 184 return encoder; 185 } 186}