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.FhirContext; 023import ca.uhn.fhir.i18n.Msg; 024import org.apache.commons.lang3.StringUtils; 025import org.hl7.fhir.instance.model.api.IBase; 026 027import java.lang.reflect.Method; 028import java.util.Arrays; 029import java.util.List; 030import java.util.stream.Collectors; 031 032/** 033 * Helper class for handling updates of the instances that support property modification via <code>setProperty</code> 034 * and <code>getProperty</code> methods. 035 */ 036public class PropertyModifyingHelper { 037 038 public static final String GET_PROPERTY_METHOD_NAME = "getProperty"; 039 public static final String SET_PROPERTY_METHOD_NAME = "setProperty"; 040 public static final String DEFAULT_DELIMITER = ", "; 041 042 private IBase myBase; 043 044 private String myDelimiter = DEFAULT_DELIMITER; 045 046 private FhirContext myFhirContext; 047 048 /** 049 * Creates a new instance initializing the dependencies. 050 * 051 * @param theFhirContext FHIR context holding the resource definitions 052 * @param theBase The base class to set properties on 053 */ 054 public PropertyModifyingHelper(FhirContext theFhirContext, IBase theBase) { 055 if (findGetPropertyMethod(theBase) == null) { 056 throw new IllegalArgumentException( 057 Msg.code(1771) + "Specified base instance does not support property retrieval."); 058 } 059 myBase = theBase; 060 myFhirContext = theFhirContext; 061 } 062 063 /** 064 * Gets the method with the specified name and parameter types. 065 * 066 * @param theObject Non-null instance to get the method from 067 * @param theMethodName Name of the method to get 068 * @param theParamClasses Parameters types that method parameters should be assignable as 069 * @return Returns the method with the given name and parameters or null if it can't be found 070 */ 071 protected Method getMethod(Object theObject, String theMethodName, Class... theParamClasses) { 072 for (Method m : theObject.getClass().getDeclaredMethods()) { 073 if (m.getName().equals(theMethodName)) { 074 if (theParamClasses.length == 0) { 075 return m; 076 } 077 if (m.getParameterCount() != theParamClasses.length) { 078 continue; 079 } 080 for (int i = 0; i < theParamClasses.length; i++) { 081 if (!m.getParameterTypes()[i].isAssignableFrom(theParamClasses[i])) { 082 continue; 083 } 084 } 085 return m; 086 } 087 } 088 return null; 089 } 090 091 /** 092 * Gets all non-blank fields as a single string joined with the delimiter provided by {@link #getDelimiter()} 093 * 094 * @param theFiledNames Field names to retrieve values for 095 * @return Returns all specified non-blank fileds as a single string. 096 */ 097 public String getFields(String... theFiledNames) { 098 return Arrays.stream(theFiledNames) 099 .map(this::get) 100 .filter(s -> !StringUtils.isBlank(s)) 101 .collect(Collectors.joining(getDelimiter())); 102 } 103 104 /** 105 * Gets property with the specified name from the provided base class. 106 * 107 * @param thePropertyName Name of the property to get 108 * @return Returns property value converted to string. In case of multiple values, they are joined with the 109 * specified delimiter. 110 */ 111 public String get(String thePropertyName) { 112 return getMultiple(thePropertyName).stream().collect(Collectors.joining(getDelimiter())); 113 } 114 115 /** 116 * Sets property or adds to a collection of properties with the specified name from the provided base class. 117 * 118 * @param thePropertyName Name of the property to set or add element to in case property is a collection 119 */ 120 public void set(String thePropertyName, String theValue) { 121 if (theValue == null || theValue.isEmpty()) { 122 return; 123 } 124 125 try { 126 IBase value = myFhirContext.getElementDefinition("string").newInstance(theValue); 127 Method setPropertyMethod = findSetPropertyMethod(myBase, int.class, String.class, value.getClass()); 128 int hashCode = thePropertyName.hashCode(); 129 setPropertyMethod.invoke(myBase, hashCode, thePropertyName, value); 130 } catch (Exception e) { 131 throw new IllegalStateException( 132 Msg.code(1772) + String.format("Unable to set property %s on %s", thePropertyName, myBase), e); 133 } 134 } 135 136 /** 137 * Gets property values with the specified name from the provided base class. 138 * 139 * @param thePropertyName Name of the property to get 140 * @return Returns property values converted to string. 141 */ 142 public List<String> getMultiple(String thePropertyName) { 143 Method getPropertyMethod = findGetPropertyMethod(myBase); 144 Object[] values; 145 try { 146 values = (Object[]) getPropertyMethod.invoke(myBase, thePropertyName.hashCode(), thePropertyName, true); 147 } catch (Exception e) { 148 throw new IllegalStateException( 149 Msg.code(1773) + String.format("Instance %s does not supply property %s", myBase, thePropertyName), 150 e); 151 } 152 153 return Arrays.stream(values) 154 .map(String::valueOf) 155 .filter(s -> !StringUtils.isEmpty(s)) 156 .collect(Collectors.toList()); 157 } 158 159 private Method findGetPropertyMethod(IBase theAddress) { 160 return getMethod(theAddress, GET_PROPERTY_METHOD_NAME); 161 } 162 163 private Method findSetPropertyMethod(IBase theAddress, Class... theParamClasses) { 164 return getMethod(theAddress, SET_PROPERTY_METHOD_NAME, theParamClasses); 165 } 166 167 /** 168 * Gets the delimiter used when concatenating multiple field values 169 * 170 * @return Returns the delimiter 171 */ 172 public String getDelimiter() { 173 return myDelimiter; 174 } 175 176 /** 177 * Sets the delimiter used when concatenating multiple field values 178 * 179 * @param theDelimiter The delimiter to set 180 */ 181 public void setDelimiter(String theDelimiter) { 182 this.myDelimiter = theDelimiter; 183 } 184 185 /** 186 * Gets the base instance that this helper operates on 187 * 188 * @return Returns the base instance 189 */ 190 public IBase getBase() { 191 return myBase; 192 } 193}