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.narrative2;
021
022import ca.uhn.fhir.context.BaseRuntimeChildDefinition;
023import ca.uhn.fhir.context.BaseRuntimeElementCompositeDefinition;
024import ca.uhn.fhir.context.FhirContext;
025import ca.uhn.fhir.context.FhirVersionEnum;
026import ca.uhn.fhir.fhirpath.IFhirPath;
027import ca.uhn.fhir.i18n.Msg;
028import ca.uhn.fhir.narrative.INarrativeGenerator;
029import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
030import ca.uhn.fhir.util.Logs;
031import jakarta.annotation.Nullable;
032import org.hl7.fhir.instance.model.api.IBase;
033import org.hl7.fhir.instance.model.api.IBaseResource;
034import org.hl7.fhir.instance.model.api.INarrative;
035
036import java.util.Collections;
037import java.util.EnumSet;
038import java.util.List;
039import java.util.Set;
040import java.util.stream.Collectors;
041
042import static org.apache.commons.lang3.StringUtils.defaultIfEmpty;
043import static org.apache.commons.lang3.StringUtils.isNotBlank;
044
045public abstract class BaseNarrativeGenerator implements INarrativeGenerator {
046
047        @Override
048        public boolean populateResourceNarrative(FhirContext theFhirContext, IBaseResource theResource) {
049                INarrativeTemplate template = selectTemplate(theFhirContext, theResource);
050                if (template != null) {
051                        applyTemplate(theFhirContext, template, theResource);
052                        return true;
053                }
054
055                return false;
056        }
057
058        @Nullable
059        private INarrativeTemplate selectTemplate(FhirContext theFhirContext, IBaseResource theResource) {
060                List<INarrativeTemplate> templates = getTemplateForElement(theFhirContext, theResource);
061                INarrativeTemplate template = null;
062                if (templates.isEmpty()) {
063                        Logs.getNarrativeGenerationTroubleshootingLog()
064                                        .debug("No templates match for resource of type {}", theResource.getClass());
065                } else {
066                        if (templates.size() > 1) {
067                                Logs.getNarrativeGenerationTroubleshootingLog()
068                                                .debug(
069                                                                "Multiple templates match for resource of type {} - Picking first from: {}",
070                                                                theResource.getClass(),
071                                                                templates);
072                        }
073                        template = templates.get(0);
074                        Logs.getNarrativeGenerationTroubleshootingLog().debug("Selected template: {}", template);
075                }
076                return template;
077        }
078
079        @Override
080        public String generateResourceNarrative(FhirContext theFhirContext, IBaseResource theResource) {
081                INarrativeTemplate template = selectTemplate(theFhirContext, theResource);
082                if (template != null) {
083                        String narrative = applyTemplate(theFhirContext, template, (IBase) theResource);
084                        return cleanWhitespace(narrative);
085                }
086
087                return null;
088        }
089
090        protected List<INarrativeTemplate> getTemplateForElement(FhirContext theFhirContext, IBase theElement) {
091                return getManifest().getTemplateByElement(theFhirContext, getStyle(), theElement);
092        }
093
094        private boolean applyTemplate(
095                        FhirContext theFhirContext, INarrativeTemplate theTemplate, IBaseResource theResource) {
096                if (templateDoesntApplyToResource(theTemplate, theResource)) {
097                        return false;
098                }
099
100                boolean retVal = false;
101                String resourceName = theFhirContext.getResourceType(theResource);
102                String contextPath = defaultIfEmpty(theTemplate.getContextPath(), resourceName);
103
104                // Narrative templates define a path within the resource that they apply to. Here, we're
105                // finding anywhere in the resource that gets a narrative
106                List<IBase> targets = findElementsInResourceRequiringNarratives(theFhirContext, theResource, contextPath);
107                for (IBase nextTargetContext : targets) {
108
109                        // Extract [element].text of type Narrative
110                        INarrative nextTargetNarrative = getOrCreateNarrativeChildElement(theFhirContext, nextTargetContext);
111
112                        // Create the actual narrative text
113                        String narrative = applyTemplate(theFhirContext, theTemplate, nextTargetContext);
114                        narrative = cleanWhitespace(narrative);
115
116                        if (isNotBlank(narrative)) {
117                                try {
118                                        nextTargetNarrative.setDivAsString(narrative);
119                                        nextTargetNarrative.setStatusAsString("generated");
120                                        retVal = true;
121                                } catch (Exception e) {
122                                        throw new InternalErrorException(Msg.code(1865) + e);
123                                }
124                        }
125                }
126                return retVal;
127        }
128
129        private INarrative getOrCreateNarrativeChildElement(FhirContext theFhirContext, IBase nextTargetContext) {
130                BaseRuntimeElementCompositeDefinition<?> targetElementDef = (BaseRuntimeElementCompositeDefinition<?>)
131                                theFhirContext.getElementDefinition(nextTargetContext.getClass());
132                BaseRuntimeChildDefinition targetTextChild = targetElementDef.getChildByName("text");
133                List<IBase> existing = targetTextChild.getAccessor().getValues(nextTargetContext);
134                INarrative nextTargetNarrative;
135                if (existing.isEmpty()) {
136                        nextTargetNarrative = (INarrative)
137                                        theFhirContext.getElementDefinition("narrative").newInstance();
138                        targetTextChild.getMutator().addValue(nextTargetContext, nextTargetNarrative);
139                } else {
140                        nextTargetNarrative = (INarrative) existing.get(0);
141                }
142                return nextTargetNarrative;
143        }
144
145        private List<IBase> findElementsInResourceRequiringNarratives(
146                        FhirContext theFhirContext, IBaseResource theResource, String theContextPath) {
147                if (theFhirContext.getVersion().getVersion().isOlderThan(FhirVersionEnum.DSTU3)) {
148                        return Collections.singletonList(theResource);
149                }
150                IFhirPath fhirPath = theFhirContext.newFluentPath();
151                return fhirPath.evaluate(theResource, theContextPath, IBase.class);
152        }
153
154        protected abstract String applyTemplate(
155                        FhirContext theFhirContext, INarrativeTemplate theTemplate, IBase theTargetContext);
156
157        private boolean templateDoesntApplyToResource(INarrativeTemplate theTemplate, IBaseResource theResource) {
158                boolean retVal = false;
159                if (theTemplate.getAppliesToProfiles() != null
160                                && !theTemplate.getAppliesToProfiles().isEmpty()) {
161                        Set<String> resourceProfiles = theResource.getMeta().getProfile().stream()
162                                        .map(t -> t.getValueAsString())
163                                        .collect(Collectors.toSet());
164                        retVal = true;
165                        for (String next : theTemplate.getAppliesToProfiles()) {
166                                if (resourceProfiles.contains(next)) {
167                                        retVal = false;
168                                        break;
169                                }
170                        }
171                }
172                return retVal;
173        }
174
175        protected abstract EnumSet<TemplateTypeEnum> getStyle();
176
177        /**
178         * Trims the superfluous whitespace out of an HTML block
179         */
180        public static String cleanWhitespace(String theResult) {
181                StringBuilder b = new StringBuilder();
182                boolean inWhitespace = false;
183                boolean betweenTags = false;
184                boolean lastNonWhitespaceCharWasTagEnd = false;
185                boolean inPre = false;
186                for (int i = 0; i < theResult.length(); i++) {
187                        char nextChar = theResult.charAt(i);
188                        if (inPre) {
189                                b.append(nextChar);
190                                continue;
191                        } else if (nextChar == '>') {
192                                b.append(nextChar);
193                                betweenTags = true;
194                                lastNonWhitespaceCharWasTagEnd = true;
195                                continue;
196                        } else if (nextChar == '\n' || nextChar == '\r') {
197                                continue;
198                        }
199
200                        if (betweenTags) {
201                                if (Character.isWhitespace(nextChar)) {
202                                        inWhitespace = true;
203                                } else if (nextChar == '<') {
204                                        if (inWhitespace && !lastNonWhitespaceCharWasTagEnd) {
205                                                b.append(' ');
206                                        }
207                                        b.append(nextChar);
208                                        inWhitespace = false;
209                                        betweenTags = false;
210                                        lastNonWhitespaceCharWasTagEnd = false;
211                                        if (i + 3 < theResult.length()) {
212                                                char char1 = Character.toLowerCase(theResult.charAt(i + 1));
213                                                char char2 = Character.toLowerCase(theResult.charAt(i + 2));
214                                                char char3 = Character.toLowerCase(theResult.charAt(i + 3));
215                                                char char4 =
216                                                                Character.toLowerCase((i + 4 < theResult.length()) ? theResult.charAt(i + 4) : ' ');
217                                                if (char1 == 'p' && char2 == 'r' && char3 == 'e') {
218                                                        inPre = true;
219                                                } else if (char1 == '/' && char2 == 'p' && char3 == 'r' && char4 == 'e') {
220                                                        inPre = false;
221                                                }
222                                        }
223                                } else {
224                                        lastNonWhitespaceCharWasTagEnd = false;
225                                        if (inWhitespace) {
226                                                b.append(' ');
227                                                inWhitespace = false;
228                                        }
229                                        b.append(nextChar);
230                                }
231                        } else {
232                                b.append(nextChar);
233                        }
234                }
235                return b.toString();
236        }
237
238        protected abstract NarrativeTemplateManifest getManifest();
239}