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.model.api.IElement; 023import ca.uhn.fhir.model.api.annotation.Child; 024import ca.uhn.fhir.model.api.annotation.Description; 025import ca.uhn.fhir.model.api.annotation.Extension; 026import ca.uhn.fhir.util.ReflectionUtil; 027import org.hl7.fhir.instance.model.api.IBase; 028 029import java.lang.reflect.Field; 030import java.lang.reflect.Modifier; 031import java.util.ArrayList; 032import java.util.Collections; 033import java.util.HashMap; 034import java.util.List; 035import java.util.Map; 036 037import static org.apache.commons.lang3.StringUtils.isNotBlank; 038 039public class RuntimeChildDeclaredExtensionDefinition extends RuntimeChildChoiceDefinition { 040 041 private boolean myDefinedLocally; 042 private String myExtensionUrl; 043 private boolean myModifier; 044 private Map<String, RuntimeChildDeclaredExtensionDefinition> myUrlToChildExtension; 045 private volatile Object myInstanceConstructorArguments; 046 private Class<?> myEnumerationType; 047 private Class<? extends IBase> myChildType; 048 private RuntimeResourceBlockDefinition myChildResourceBlock; 049 private BaseRuntimeElementDefinition<?> myChildDef; 050 051 /** 052 * @param theBoundTypeBinder 053 * If the child is of a type that requires a constructor argument to instantiate, this is the argument to 054 * use 055 * @param theDefinedLocally 056 * See {@link Extension#definedLocally()} 057 */ 058 RuntimeChildDeclaredExtensionDefinition( 059 Field theField, 060 Child theChild, 061 Description theDescriptionAnnotation, 062 Extension theExtension, 063 String theElementName, 064 String theExtensionUrl, 065 Class<? extends IBase> theChildType, 066 Object theBoundTypeBinder) 067 throws ConfigurationException { 068 super(theField, theElementName, theChild, theDescriptionAnnotation); 069 assert isNotBlank(theExtensionUrl); 070 myExtensionUrl = theExtensionUrl; 071 myChildType = theChildType; 072 myDefinedLocally = theExtension.definedLocally(); 073 myModifier = theExtension.isModifier(); 074 myInstanceConstructorArguments = theBoundTypeBinder; 075 076 List<Class<? extends IBase>> choiceTypes = new ArrayList<Class<? extends IBase>>(); 077 for (Class<? extends IElement> next : theChild.type()) { 078 choiceTypes.add(next); 079 } 080 081 if (Modifier.isAbstract(theChildType.getModifiers()) == false) { 082 choiceTypes.add(theChildType); 083 } 084 085 setChoiceTypes(choiceTypes, Collections.emptyList()); 086 } 087 088 @Override 089 public String getElementName() { 090 return "value"; 091 } 092 093 @Override 094 public Object getInstanceConstructorArguments() { 095 Object retVal = myInstanceConstructorArguments; 096 if (retVal == null && myEnumerationType != null) { 097 retVal = RuntimeChildPrimitiveEnumerationDatatypeDefinition.toEnumFactory(myEnumerationType); 098 myInstanceConstructorArguments = retVal; 099 } 100 return retVal; 101 } 102 103 public void setEnumerationType(Class<?> theEnumerationType) { 104 myEnumerationType = theEnumerationType; 105 } 106 107 @Override 108 public String getChildNameByDatatype(Class<? extends IBase> theDatatype) { 109 110 String retVal = super.getChildNameByDatatype(theDatatype); 111 112 if (retVal != null) { 113 BaseRuntimeElementDefinition<?> childDef = super.getChildElementDefinitionByDatatype(theDatatype); 114 if (childDef instanceof RuntimeResourceBlockDefinition) { 115 // Child is a newted extension 116 retVal = null; 117 } 118 } 119 120 if (retVal == null) { 121 if (myModifier) { 122 return "modifierExtension"; 123 } 124 return "extension"; 125 } 126 return retVal; 127 } 128 129 @Override 130 public BaseRuntimeElementDefinition<?> getChildByName(String theName) { 131 String name = theName; 132 if ("extension".equals(name) || "modifierExtension".equals(name)) { 133 if (myChildResourceBlock != null) { 134 return myChildResourceBlock; 135 } 136 if (myChildDef != null) { 137 return myChildDef; 138 } 139 } 140 141 if (getValidChildNames().contains(name) == false) { 142 return null; 143 } 144 145 return super.getChildByName(name); 146 } 147 148 @Override 149 public BaseRuntimeElementDefinition<?> getChildElementDefinitionByDatatype(Class<? extends IBase> theDatatype) { 150 if (myChildResourceBlock != null) { 151 if (myChildResourceBlock.getImplementingClass().equals(theDatatype)) { 152 return myChildResourceBlock; 153 } 154 } 155 return super.getChildElementDefinitionByDatatype(theDatatype); 156 } 157 158 public RuntimeChildDeclaredExtensionDefinition getChildExtensionForUrl(String theUrl) { 159 return myUrlToChildExtension.get(theUrl); 160 } 161 162 @Override 163 public String getExtensionUrl() { 164 return myExtensionUrl; 165 } 166 167 public boolean isDefinedLocally() { 168 return myDefinedLocally; 169 } 170 171 @Override 172 public boolean isModifier() { 173 return myModifier; 174 } 175 176 @Override 177 void sealAndInitialize( 178 FhirContext theContext, 179 Map<Class<? extends IBase>, BaseRuntimeElementDefinition<?>> theClassToElementDefinitions) { 180 myUrlToChildExtension = new HashMap<String, RuntimeChildDeclaredExtensionDefinition>(); 181 182 BaseRuntimeElementDefinition<?> elementDef = theClassToElementDefinitions.get(myChildType); 183 184 /* 185 * This will happen for any type that isn't defined in the base set of 186 * built-in types, e.g. custom structures or custom extensions 187 */ 188 if (elementDef == null) { 189 if (Modifier.isAbstract(myChildType.getModifiers()) == false) { 190 elementDef = theContext.getElementDefinition(myChildType); 191 } 192 } 193 194 if (elementDef instanceof RuntimePrimitiveDatatypeDefinition 195 || elementDef instanceof RuntimeCompositeDatatypeDefinition) { 196 // myDatatypeChildName = "value" + elementDef.getName().substring(0, 1).toUpperCase() + 197 // elementDef.getName().substring(1); 198 // if ("valueResourceReference".equals(myDatatypeChildName)) { 199 // Per one of the examples here: http://hl7.org/implement/standards/fhir/extensibility.html#extension 200 // myDatatypeChildName = "valueResource"; 201 // List<Class<? extends IBaseResource>> types = new ArrayList<Class<? extends IBaseResource>>(); 202 // types.add(IBaseResource.class); 203 // myChildDef = findResourceReferenceDefinition(theClassToElementDefinitions); 204 // } else { 205 myChildDef = elementDef; 206 // } 207 } else if (elementDef instanceof RuntimeResourceBlockDefinition) { 208 RuntimeResourceBlockDefinition extDef = ((RuntimeResourceBlockDefinition) elementDef); 209 for (RuntimeChildDeclaredExtensionDefinition next : extDef.getExtensions()) { 210 myUrlToChildExtension.put(next.getExtensionUrl(), next); 211 } 212 myChildResourceBlock = (RuntimeResourceBlockDefinition) elementDef; 213 } 214 215 myUrlToChildExtension = Collections.unmodifiableMap(myUrlToChildExtension); 216 217 super.sealAndInitialize(theContext, theClassToElementDefinitions); 218 } 219 220 public IBase newInstance() { 221 return ReflectionUtil.newInstance(myChildType); 222 } 223 224 public Class<? extends IBase> getChildType() { 225 return myChildType; 226 } 227}