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.context; 021 022import ca.uhn.fhir.i18n.Msg; 023import ca.uhn.fhir.model.api.annotation.Child; 024import ca.uhn.fhir.model.api.annotation.Description; 025import ca.uhn.fhir.util.ParametersUtil; 026import ca.uhn.fhir.util.ValidateUtil; 027import org.apache.commons.lang3.Validate; 028import org.hl7.fhir.instance.model.api.IBase; 029 030import java.lang.reflect.Field; 031import java.util.ArrayList; 032import java.util.Collections; 033import java.util.List; 034import java.util.Optional; 035 036public abstract class BaseRuntimeDeclaredChildDefinition extends BaseRuntimeChildDefinition { 037 private final IAccessor myAccessor; 038 private final String myElementName; 039 private final Field myField; 040 private final String myFormalDefinition; 041 private final int myMax; 042 private final int myMin; 043 private final IMutator myMutator; 044 private final String myShortDefinition; 045 private String myBindingValueSet; 046 private boolean myModifier; 047 private boolean mySummary; 048 049 BaseRuntimeDeclaredChildDefinition( 050 Field theField, Child theChildAnnotation, Description theDescriptionAnnotation, String theElementName) 051 throws ConfigurationException { 052 super(); 053 Validate.notNull(theField, "No field specified"); 054 ValidateUtil.isGreaterThanOrEqualTo(theChildAnnotation.min(), 0, "Min must be >= 0"); 055 Validate.isTrue( 056 theChildAnnotation.max() == -1 || theChildAnnotation.max() >= theChildAnnotation.min(), 057 "Max must be >= Min (unless it is -1 / unlimited)"); 058 Validate.notBlank(theElementName, "Element name must not be blank"); 059 060 myField = theField; 061 myMin = theChildAnnotation.min(); 062 myMax = theChildAnnotation.max(); 063 mySummary = theChildAnnotation.summary(); 064 myModifier = theChildAnnotation.modifier(); 065 myElementName = theElementName; 066 if (theDescriptionAnnotation != null) { 067 myShortDefinition = theDescriptionAnnotation.shortDefinition(); 068 myFormalDefinition = ParametersUtil.extractDescription(theDescriptionAnnotation); 069 } else { 070 myShortDefinition = null; 071 myFormalDefinition = null; 072 } 073 074 myField.setAccessible(true); 075 if (List.class.equals(myField.getType())) { 076 // TODO: verify that generic type is IElement 077 myAccessor = new FieldListAccessor(); 078 myMutator = new FieldListMutator(); 079 } else { 080 myAccessor = new FieldPlainAccessor(); 081 myMutator = new FieldPlainMutator(); 082 } 083 } 084 085 @Override 086 public IAccessor getAccessor() { 087 return myAccessor; 088 } 089 090 public String getBindingValueSet() { 091 return myBindingValueSet; 092 } 093 094 void setBindingValueSet(String theBindingValueSet) { 095 myBindingValueSet = theBindingValueSet; 096 } 097 098 @Override 099 public String getElementName() { 100 return myElementName; 101 } 102 103 public Field getField() { 104 return myField; 105 } 106 107 public String getFormalDefinition() { 108 return myFormalDefinition; 109 } 110 111 @Override 112 public int getMax() { 113 return myMax; 114 } 115 116 @Override 117 public int getMin() { 118 return myMin; 119 } 120 121 @Override 122 public IMutator getMutator() { 123 return myMutator; 124 } 125 126 public String getShortDefinition() { 127 return myShortDefinition; 128 } 129 130 public boolean isModifier() { 131 return myModifier; 132 } 133 134 protected void setModifier(boolean theModifier) { 135 myModifier = theModifier; 136 } 137 138 @Override 139 public boolean isSummary() { 140 return mySummary; 141 } 142 143 private final class FieldListAccessor implements IAccessor { 144 @SuppressWarnings("unchecked") 145 @Override 146 public List<IBase> getValues(IBase theTarget) { 147 List<IBase> retVal = (List<IBase>) getFieldValue(theTarget, myField); 148 if (retVal == null) { 149 retVal = Collections.emptyList(); 150 } 151 return retVal; 152 } 153 } 154 155 protected final class FieldListMutator implements IMutator { 156 @Override 157 public void addValue(IBase theTarget, IBase theValue) { 158 addValue(theTarget, theValue, false); 159 } 160 161 private void addValue(IBase theTarget, IBase theValue, boolean theClear) { 162 @SuppressWarnings("unchecked") 163 List<IBase> existingList = (List<IBase>) getFieldValue(theTarget, myField); 164 if (existingList == null) { 165 existingList = new ArrayList<>(2); 166 setFieldValue(theTarget, existingList, myField); 167 } 168 if (theClear) { 169 existingList.clear(); 170 if (theValue == null) { 171 return; 172 } 173 } 174 existingList.add(theValue); 175 } 176 177 @Override 178 public void setValue(IBase theTarget, IBase theValue) { 179 addValue(theTarget, theValue, true); 180 } 181 182 @Override 183 public void remove(IBase theTarget, int theIndex) { 184 List<IBase> existingList = (List<IBase>) getFieldValue(theTarget, myField); 185 if (existingList == null) { 186 throw new IndexOutOfBoundsException( 187 Msg.code(2143) + "Can not remove element at index " + theIndex + " from list - List is null"); 188 } 189 if (theIndex >= existingList.size()) { 190 throw new IndexOutOfBoundsException(Msg.code(2144) + "Can not remove element at index " + theIndex 191 + " from list - List size is " + existingList.size()); 192 } 193 existingList.remove(theIndex); 194 } 195 } 196 197 private final class FieldPlainAccessor implements IAccessor { 198 @Override 199 public List<IBase> getValues(IBase theTarget) { 200 Object values = getFieldValue(theTarget, myField); 201 if (values == null) { 202 return Collections.emptyList(); 203 } 204 return Collections.singletonList((IBase) values); 205 } 206 207 @Override 208 public <T extends IBase> Optional<T> getFirstValueOrNull(IBase theTarget) { 209 return Optional.ofNullable(((T) getFieldValue(theTarget, myField))); 210 } 211 } 212 213 protected final class FieldPlainMutator implements IMutator { 214 @Override 215 public void addValue(IBase theTarget, IBase theValue) { 216 setFieldValue(theTarget, theValue, myField); 217 } 218 219 @Override 220 public void setValue(IBase theTarget, IBase theValue) { 221 addValue(theTarget, theValue); 222 } 223 224 @Override 225 public void remove(IBase theTarget, int theIndex) { 226 throw new UnsupportedOperationException( 227 Msg.code(2142) + "Remove by index can only be called on a list-valued field. '" + myField.getName() 228 + "' is a single-valued field."); 229 } 230 } 231 232 private static void setFieldValue(IBase theTarget, Object theValue, Field theField) { 233 try { 234 theField.set(theTarget, theValue); 235 } catch (IllegalAccessException e) { 236 throw new ConfigurationException(Msg.code(1736) + "Failed to set value", e); 237 } 238 } 239 240 private static Object getFieldValue(IBase theTarget, Field theField) { 241 try { 242 return theField.get(theTarget); 243 } catch (IllegalAccessException e) { 244 throw new ConfigurationException(Msg.code(1737) + "Failed to get value", e); 245 } 246 } 247}