001/*- 002 * #%L 003 * HAPI FHIR - Server Framework 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.rest.server.interceptor; 021 022import ca.uhn.fhir.context.BaseRuntimeChildDefinition; 023import ca.uhn.fhir.context.BaseRuntimeElementCompositeDefinition; 024import ca.uhn.fhir.context.BaseRuntimeElementDefinition; 025import ca.uhn.fhir.context.FhirContext; 026import ca.uhn.fhir.context.RuntimePrimitiveDatatypeDefinition; 027import ca.uhn.fhir.context.support.IValidationSupport; 028import ca.uhn.fhir.context.support.TranslateConceptResult; 029import ca.uhn.fhir.context.support.TranslateConceptResults; 030import ca.uhn.fhir.util.FhirTerser; 031import ca.uhn.fhir.util.IModelVisitor; 032import com.google.common.collect.ArrayListMultimap; 033import com.google.common.collect.Multimap; 034import jakarta.annotation.Nonnull; 035import org.apache.commons.lang3.StringUtils; 036import org.apache.commons.lang3.Validate; 037import org.hl7.fhir.instance.model.api.IBase; 038import org.hl7.fhir.instance.model.api.IBaseCoding; 039import org.hl7.fhir.instance.model.api.IBaseResource; 040import org.hl7.fhir.instance.model.api.IPrimitiveType; 041 042import java.util.ArrayList; 043import java.util.HashMap; 044import java.util.List; 045import java.util.Map; 046import java.util.Objects; 047 048public class ResponseTerminologyTranslationSvc { 049 private BaseRuntimeChildDefinition myCodingSystemChild; 050 private BaseRuntimeChildDefinition myCodingCodeChild; 051 private BaseRuntimeElementDefinition<IPrimitiveType<?>> myUriDefinition; 052 private BaseRuntimeElementDefinition<IPrimitiveType<?>> myCodeDefinition; 053 private Class<? extends IBase> myCodeableConceptType; 054 private Class<? extends IBase> myCodingType; 055 private BaseRuntimeChildDefinition myCodeableConceptCodingChild; 056 private BaseRuntimeElementCompositeDefinition<?> myCodingDefinition; 057 private RuntimePrimitiveDatatypeDefinition myStringDefinition; 058 private BaseRuntimeChildDefinition myCodingDisplayChild; 059 private Map<String, String> myMappingSpec; 060 private final IValidationSupport myValidationSupport; 061 private final FhirContext myFhirContext; 062 063 public ResponseTerminologyTranslationSvc(@Nonnull IValidationSupport theValidationSupport) { 064 myValidationSupport = theValidationSupport; 065 Validate.notNull(theValidationSupport, "The validation support must not be null"); 066 067 myFhirContext = theValidationSupport.getFhirContext(); 068 Validate.notNull(myFhirContext, "The validation support must not return a null context"); 069 070 BaseRuntimeElementCompositeDefinition<?> codeableConceptDef = (BaseRuntimeElementCompositeDefinition<?>) 071 Objects.requireNonNull(myFhirContext.getElementDefinition("CodeableConcept")); 072 073 myCodeableConceptType = codeableConceptDef.getImplementingClass(); 074 myCodeableConceptCodingChild = codeableConceptDef.getChildByName("coding"); 075 076 myCodingDefinition = (BaseRuntimeElementCompositeDefinition<?>) 077 Objects.requireNonNull(myFhirContext.getElementDefinition("Coding")); 078 myCodingType = myCodingDefinition.getImplementingClass(); 079 myCodingSystemChild = myCodingDefinition.getChildByName("system"); 080 myCodingCodeChild = myCodingDefinition.getChildByName("code"); 081 myCodingDisplayChild = myCodingDefinition.getChildByName("display"); 082 083 myUriDefinition = (RuntimePrimitiveDatatypeDefinition) myFhirContext.getElementDefinition("uri"); 084 myCodeDefinition = (RuntimePrimitiveDatatypeDefinition) myFhirContext.getElementDefinition("code"); 085 myStringDefinition = (RuntimePrimitiveDatatypeDefinition) myFhirContext.getElementDefinition("string"); 086 } 087 088 public void processResourcesForTerminologyTranslation(List<IBaseResource> resources) { 089 FhirTerser terser = myFhirContext.newTerser(); 090 for (IBaseResource nextResource : resources) { 091 terser.visit(nextResource, new MappingVisitor()); 092 } 093 } 094 095 public void addMappingSpecification(String theSourceCodeSystemUrl, String theTargetCodeSystemUrl) { 096 Validate.notBlank(theSourceCodeSystemUrl, "theSourceCodeSystemUrl must not be null or blank"); 097 Validate.notBlank(theTargetCodeSystemUrl, "theTargetCodeSystemUrl must not be null or blank"); 098 099 getMappingSpecifications().put(theSourceCodeSystemUrl, theTargetCodeSystemUrl); 100 } 101 102 public void clearMappingSpecifications() { 103 myMappingSpec.clear(); 104 } 105 106 public Map<String, String> getMappingSpecifications() { 107 if (myMappingSpec == null) { 108 myMappingSpec = new HashMap<>(); 109 } 110 return myMappingSpec; 111 } 112 113 private class MappingVisitor implements IModelVisitor { 114 115 @Override 116 public void acceptElement( 117 IBaseResource theResource, 118 IBase theElement, 119 List<String> thePathToElement, 120 BaseRuntimeChildDefinition theChildDefinition, 121 BaseRuntimeElementDefinition<?> theDefinition) { 122 if (myCodeableConceptType.isAssignableFrom(theElement.getClass())) { 123 124 // Find all existing Codings 125 Multimap<String, String> foundSystemsToCodes = ArrayListMultimap.create(); 126 List<IBase> nextCodeableConceptCodings = 127 myCodeableConceptCodingChild.getAccessor().getValues(theElement); 128 for (IBase nextCodeableConceptCoding : nextCodeableConceptCodings) { 129 String system = myCodingSystemChild 130 .getAccessor() 131 .getFirstValueOrNull(nextCodeableConceptCoding) 132 .map(t -> (IPrimitiveType<?>) t) 133 .map(IPrimitiveType::getValueAsString) 134 .orElse(null); 135 String code = myCodingCodeChild 136 .getAccessor() 137 .getFirstValueOrNull(nextCodeableConceptCoding) 138 .map(t -> (IPrimitiveType<?>) t) 139 .map(IPrimitiveType::getValueAsString) 140 .orElse(null); 141 if (StringUtils.isNotBlank(system) 142 && StringUtils.isNotBlank(code) 143 && !foundSystemsToCodes.containsKey(system)) { 144 foundSystemsToCodes.put(system, code); 145 } 146 } 147 148 // Look for mappings 149 for (String nextSourceSystem : foundSystemsToCodes.keySet()) { 150 String wantTargetSystem = getMappingSpecifications().get(nextSourceSystem); 151 if (wantTargetSystem != null) { 152 if (!foundSystemsToCodes.containsKey(wantTargetSystem)) { 153 154 for (String code : foundSystemsToCodes.get(nextSourceSystem)) { 155 List<IBaseCoding> codings = new ArrayList<>(); 156 codings.add(createCodingFromPrimitives(nextSourceSystem, code, null)); 157 TranslateConceptResults translateConceptResults = myValidationSupport.translateConcept( 158 new IValidationSupport.TranslateCodeRequest(codings, wantTargetSystem)); 159 if (translateConceptResults != null) { 160 List<TranslateConceptResult> mappings = translateConceptResults.getResults(); 161 for (TranslateConceptResult nextMapping : mappings) { 162 163 IBase newCoding = createCodingFromPrimitives( 164 nextMapping.getSystem(), 165 nextMapping.getCode(), 166 nextMapping.getDisplay()); 167 168 // Add coding to existing CodeableConcept 169 myCodeableConceptCodingChild 170 .getMutator() 171 .addValue(theElement, newCoding); 172 } 173 } 174 } 175 } 176 } 177 } 178 } 179 } 180 181 private IBaseCoding createCodingFromPrimitives(String system, String code, String display) { 182 assert myUriDefinition != null; 183 assert myCodeDefinition != null; 184 IBaseCoding newCoding = (IBaseCoding) myCodingDefinition.newInstance(); 185 IPrimitiveType<?> newSystem = myUriDefinition.newInstance(system); 186 myCodingSystemChild.getMutator().addValue(newCoding, newSystem); 187 IPrimitiveType<?> newCode = myCodeDefinition.newInstance(code); 188 myCodingCodeChild.getMutator().addValue(newCoding, newCode); 189 if (StringUtils.isNotBlank(display)) { 190 assert myStringDefinition != null; 191 IPrimitiveType<?> newDisplay = myStringDefinition.newInstance(display); 192 myCodingDisplayChild.getMutator().addValue(newCoding, newDisplay); 193 } 194 return newCoding; 195 } 196 } 197}