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.narrative;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.fhirpath.IFhirPath;
024import ca.uhn.fhir.fhirpath.IFhirPathEvaluationContext;
025import ca.uhn.fhir.i18n.Msg;
026import ca.uhn.fhir.narrative2.BaseNarrativeGenerator;
027import ca.uhn.fhir.narrative2.INarrativeTemplate;
028import ca.uhn.fhir.narrative2.NarrativeGeneratorTemplateUtils;
029import ca.uhn.fhir.narrative2.TemplateTypeEnum;
030import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
031import com.google.common.collect.Sets;
032import org.hl7.fhir.instance.model.api.IBase;
033import org.thymeleaf.IEngineConfiguration;
034import org.thymeleaf.TemplateEngine;
035import org.thymeleaf.cache.AlwaysValidCacheEntryValidity;
036import org.thymeleaf.cache.ICacheEntryValidity;
037import org.thymeleaf.context.Context;
038import org.thymeleaf.context.IExpressionContext;
039import org.thymeleaf.context.ITemplateContext;
040import org.thymeleaf.dialect.IDialect;
041import org.thymeleaf.dialect.IExpressionObjectDialect;
042import org.thymeleaf.engine.AttributeName;
043import org.thymeleaf.expression.IExpressionObjectFactory;
044import org.thymeleaf.messageresolver.IMessageResolver;
045import org.thymeleaf.model.IProcessableElementTag;
046import org.thymeleaf.processor.IProcessor;
047import org.thymeleaf.processor.element.AbstractAttributeTagProcessor;
048import org.thymeleaf.processor.element.AbstractElementTagProcessor;
049import org.thymeleaf.processor.element.IElementTagStructureHandler;
050import org.thymeleaf.standard.StandardDialect;
051import org.thymeleaf.standard.expression.IStandardExpression;
052import org.thymeleaf.standard.expression.IStandardExpressionParser;
053import org.thymeleaf.standard.expression.StandardExpressions;
054import org.thymeleaf.templatemode.TemplateMode;
055import org.thymeleaf.templateresolver.DefaultTemplateResolver;
056import org.thymeleaf.templateresolver.ITemplateResolver;
057import org.thymeleaf.templateresource.ITemplateResource;
058import org.thymeleaf.templateresource.StringTemplateResource;
059
060import java.util.EnumSet;
061import java.util.List;
062import java.util.Map;
063import java.util.Optional;
064import java.util.Set;
065
066import static org.apache.commons.lang3.StringUtils.isNotBlank;
067
068public abstract class BaseThymeleafNarrativeGenerator extends BaseNarrativeGenerator {
069
070        public static final String FHIRPATH = "fhirpath";
071        private IMessageResolver myMessageResolver;
072        private IFhirPathEvaluationContext myFhirPathEvaluationContext;
073
074        /**
075         * Constructor
076         */
077        protected BaseThymeleafNarrativeGenerator() {
078                super();
079        }
080
081        public void setFhirPathEvaluationContext(IFhirPathEvaluationContext theFhirPathEvaluationContext) {
082                myFhirPathEvaluationContext = theFhirPathEvaluationContext;
083        }
084
085        private TemplateEngine getTemplateEngine(FhirContext theFhirContext) {
086                TemplateEngine engine = new TemplateEngine();
087                ITemplateResolver resolver = new NarrativeTemplateResolver(theFhirContext);
088                engine.setTemplateResolver(resolver);
089                if (myMessageResolver != null) {
090                        engine.setMessageResolver(myMessageResolver);
091                }
092                StandardDialect dialect = new StandardDialect() {
093                        @Override
094                        public Set<IProcessor> getProcessors(String theDialectPrefix) {
095                                Set<IProcessor> retVal = super.getProcessors(theDialectPrefix);
096                                retVal.add(new NarrativeTagProcessor(theFhirContext, theDialectPrefix));
097                                retVal.add(new NarrativeAttributeProcessor(theDialectPrefix, theFhirContext));
098                                return retVal;
099                        }
100                };
101                engine.setDialect(dialect);
102
103                engine.addDialect(new NarrativeGeneratorDialect(theFhirContext));
104                return engine;
105        }
106
107        @Override
108        protected String applyTemplate(FhirContext theFhirContext, INarrativeTemplate theTemplate, IBase theTargetContext) {
109
110                Context context = new Context();
111                context.setVariable("resource", theTargetContext);
112                context.setVariable("context", theTargetContext);
113                context.setVariable("narrativeUtil", NarrativeGeneratorTemplateUtils.INSTANCE);
114                context.setVariable(
115                                "fhirVersion", theFhirContext.getVersion().getVersion().name());
116
117                return getTemplateEngine(theFhirContext).process(theTemplate.getTemplateName(), context);
118        }
119
120        @Override
121        protected EnumSet<TemplateTypeEnum> getStyle() {
122                return EnumSet.of(TemplateTypeEnum.THYMELEAF);
123        }
124
125        private String applyTemplateWithinTag(
126                        FhirContext theFhirContext, ITemplateContext theTemplateContext, String theName, String theElement) {
127                IEngineConfiguration configuration = theTemplateContext.getConfiguration();
128                IStandardExpressionParser expressionParser = StandardExpressions.getExpressionParser(configuration);
129                final IStandardExpression expression = expressionParser.parseExpression(theTemplateContext, theElement);
130                Object elementValueObj = expression.execute(theTemplateContext);
131                final IBase elementValue = (IBase) elementValueObj;
132                if (elementValue == null) {
133                        return "";
134                }
135
136                List<INarrativeTemplate> templateOpt;
137                if (isNotBlank(theName)) {
138                        templateOpt = getManifest().getTemplateByName(theFhirContext, getStyle(), theName);
139                        if (templateOpt.isEmpty()) {
140                                throw new InternalErrorException(Msg.code(1863) + "Unknown template name: " + theName);
141                        }
142                } else {
143                        templateOpt = getManifest().getTemplateByElement(theFhirContext, getStyle(), elementValue);
144                        if (templateOpt.isEmpty()) {
145                                throw new InternalErrorException(Msg.code(1864) + "No template for type: " + elementValue.getClass());
146                        }
147                }
148
149                return applyTemplate(theFhirContext, templateOpt.get(0), elementValue);
150        }
151
152        public void setMessageResolver(IMessageResolver theMessageResolver) {
153                myMessageResolver = theMessageResolver;
154        }
155
156        private class NarrativeTemplateResolver extends DefaultTemplateResolver {
157                private final FhirContext myFhirContext;
158
159                private NarrativeTemplateResolver(FhirContext theFhirContext) {
160                        myFhirContext = theFhirContext;
161                }
162
163                @Override
164                protected boolean computeResolvable(
165                                IEngineConfiguration theConfiguration,
166                                String theOwnerTemplate,
167                                String theTemplate,
168                                Map<String, Object> theTemplateResolutionAttributes) {
169                        if (theOwnerTemplate == null) {
170                                return getManifest()
171                                                                .getTemplateByName(myFhirContext, getStyle(), theTemplate)
172                                                                .size()
173                                                > 0;
174                        } else {
175                                return getManifest()
176                                                                .getTemplateByFragmentName(myFhirContext, getStyle(), theTemplate)
177                                                                .size()
178                                                > 0;
179                        }
180                }
181
182                @Override
183                protected TemplateMode computeTemplateMode(
184                                IEngineConfiguration theConfiguration,
185                                String theOwnerTemplate,
186                                String theTemplate,
187                                Map<String, Object> theTemplateResolutionAttributes) {
188                        return TemplateMode.XML;
189                }
190
191                @Override
192                protected ITemplateResource computeTemplateResource(
193                                IEngineConfiguration theConfiguration,
194                                String theOwnerTemplate,
195                                String theTemplate,
196                                Map<String, Object> theTemplateResolutionAttributes) {
197                        if (theOwnerTemplate == null) {
198                                return getManifest().getTemplateByName(myFhirContext, getStyle(), theTemplate).stream()
199                                                .findFirst()
200                                                .map(t -> new StringTemplateResource(t.getTemplateText()))
201                                                .orElseThrow(() -> new IllegalArgumentException("Unknown template: " + theTemplate));
202                        } else {
203                                return getManifest().getTemplateByFragmentName(myFhirContext, getStyle(), theTemplate).stream()
204                                                .findFirst()
205                                                .map(t -> new StringTemplateResource(t.getTemplateText()))
206                                                .orElseThrow(() -> new IllegalArgumentException("Unknown template: " + theTemplate));
207                        }
208                }
209
210                @Override
211                protected ICacheEntryValidity computeValidity(
212                                IEngineConfiguration theConfiguration,
213                                String theOwnerTemplate,
214                                String theTemplate,
215                                Map<String, Object> theTemplateResolutionAttributes) {
216                        return AlwaysValidCacheEntryValidity.INSTANCE;
217                }
218        }
219
220        private class NarrativeTagProcessor extends AbstractElementTagProcessor {
221
222                private final FhirContext myFhirContext;
223
224                NarrativeTagProcessor(FhirContext theFhirContext, String dialectPrefix) {
225                        super(TemplateMode.XML, dialectPrefix, "narrative", true, null, true, 0);
226                        myFhirContext = theFhirContext;
227                }
228
229                @Override
230                protected void doProcess(
231                                ITemplateContext theTemplateContext,
232                                IProcessableElementTag theTag,
233                                IElementTagStructureHandler theStructureHandler) {
234                        String name = theTag.getAttributeValue("th:name");
235                        String element = theTag.getAttributeValue("th:element");
236
237                        String appliedTemplate = applyTemplateWithinTag(myFhirContext, theTemplateContext, name, element);
238                        theStructureHandler.replaceWith(appliedTemplate, false);
239                }
240        }
241
242        /**
243         * This is a thymeleaf extension that allows people to do things like
244         * <th:block th:narrative="${result}"/>
245         */
246        private class NarrativeAttributeProcessor extends AbstractAttributeTagProcessor {
247
248                private final FhirContext myFhirContext;
249
250                NarrativeAttributeProcessor(String theDialectPrefix, FhirContext theFhirContext) {
251                        super(TemplateMode.XML, theDialectPrefix, null, false, "narrative", true, 0, true);
252                        myFhirContext = theFhirContext;
253                }
254
255                @Override
256                protected void doProcess(
257                                ITemplateContext theContext,
258                                IProcessableElementTag theTag,
259                                AttributeName theAttributeName,
260                                String theAttributeValue,
261                                IElementTagStructureHandler theStructureHandler) {
262                        String text = applyTemplateWithinTag(myFhirContext, theContext, null, theAttributeValue);
263                        theStructureHandler.setBody(text, false);
264                }
265        }
266
267        private class NarrativeGeneratorDialect implements IDialect, IExpressionObjectDialect {
268
269                private final FhirContext myFhirContext;
270
271                public NarrativeGeneratorDialect(FhirContext theFhirContext) {
272                        myFhirContext = theFhirContext;
273                }
274
275                @Override
276                public String getName() {
277                        return "NarrativeGeneratorDialect";
278                }
279
280                @Override
281                public IExpressionObjectFactory getExpressionObjectFactory() {
282                        return new NarrativeGeneratorExpressionObjectFactory(myFhirContext);
283                }
284        }
285
286        private class NarrativeGeneratorExpressionObjectFactory implements IExpressionObjectFactory {
287
288                private final FhirContext myFhirContext;
289
290                public NarrativeGeneratorExpressionObjectFactory(FhirContext theFhirContext) {
291                        myFhirContext = theFhirContext;
292                }
293
294                @Override
295                public Set<String> getAllExpressionObjectNames() {
296                        return Sets.newHashSet(FHIRPATH);
297                }
298
299                @Override
300                public Object buildObject(IExpressionContext context, String expressionObjectName) {
301                        if (FHIRPATH.equals(expressionObjectName)) {
302                                return new NarrativeGeneratorFhirPathExpressionObject(myFhirContext);
303                        }
304                        return null;
305                }
306
307                @Override
308                public boolean isCacheable(String expressionObjectName) {
309                        return false;
310                }
311        }
312
313        private class NarrativeGeneratorFhirPathExpressionObject {
314
315                private final FhirContext myFhirContext;
316
317                public NarrativeGeneratorFhirPathExpressionObject(FhirContext theFhirContext) {
318                        myFhirContext = theFhirContext;
319                }
320
321                public IBase evaluateFirst(IBase theInput, String theExpression) {
322                        IFhirPath fhirPath = newFhirPath();
323                        Optional<IBase> output = fhirPath.evaluateFirst(theInput, theExpression, IBase.class);
324                        return output.orElse(null);
325                }
326
327                public List<IBase> evaluate(IBase theInput, String theExpression) {
328                        IFhirPath fhirPath = newFhirPath();
329                        return fhirPath.evaluate(theInput, theExpression, IBase.class);
330                }
331
332                private IFhirPath newFhirPath() {
333                        IFhirPath fhirPath = myFhirContext.newFhirPath();
334                        if (myFhirPathEvaluationContext != null) {
335                                fhirPath.setEvaluationContext(myFhirPathEvaluationContext);
336                        }
337                        return fhirPath;
338                }
339        }
340}