diff --git a/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/QuteProcessor.java b/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/QuteProcessor.java index 724c4384efffe..f591dcfb3cf27 100644 --- a/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/QuteProcessor.java +++ b/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/QuteProcessor.java @@ -2433,6 +2433,7 @@ void initialize(BuildProducer syntheticBeans, QuteRecord List templates = new ArrayList<>(); List tags = new ArrayList<>(); + Map templateContents = new HashMap<>(); for (TemplatePathBuildItem templatePath : templatePaths) { if (templatePath.isTag()) { // tags/myTag.html -> myTag.html @@ -2441,6 +2442,9 @@ void initialize(BuildProducer syntheticBeans, QuteRecord } else { templates.add(templatePath.getPath()); } + if (!templatePath.isFileBased()) { + templateContents.put(templatePath.getPath(), templatePath.getContent()); + } } Map> variants; if (templateVariants.isPresent()) { @@ -2454,7 +2458,7 @@ void initialize(BuildProducer syntheticBeans, QuteRecord .map(GeneratedValueResolverBuildItem::getClassName).collect(Collectors.toList()), templates, tags, variants, templateInitializers.stream() .map(TemplateGlobalProviderBuildItem::getClassName).collect(Collectors.toList()), - templateRoots.getPaths().stream().map(p -> p + "/").collect(Collectors.toSet()))) + templateRoots.getPaths().stream().map(p -> p + "/").collect(Collectors.toSet()), templateContents)) .done()); } @@ -3423,8 +3427,10 @@ private void checkDuplicatePaths(List templatePaths) { if (!duplicates.isEmpty()) { StringBuilder builder = new StringBuilder("Duplicate templates found:"); for (Entry> e : duplicates.entrySet()) { - builder.append("\n\t- ").append(e.getKey()).append(": ") - .append(e.getValue().stream().map(TemplatePathBuildItem::getFullPath).collect(Collectors.toList())); + builder.append("\n\t- ") + .append(e.getKey()) + .append(": ") + .append(e.getValue().stream().map(TemplatePathBuildItem::getSourceInfo).collect(Collectors.toList())); } throw new IllegalStateException(builder.toString()); } diff --git a/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/TemplatePathBuildItem.java b/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/TemplatePathBuildItem.java index 85a4a9fdeeeae..eaa113941153e 100644 --- a/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/TemplatePathBuildItem.java +++ b/extensions/qute/deployment/src/main/java/io/quarkus/qute/deployment/TemplatePathBuildItem.java @@ -1,28 +1,59 @@ package io.quarkus.qute.deployment; import java.nio.file.Path; +import java.util.Objects; import io.quarkus.builder.item.MultiBuildItem; /** - * Represents a template path. + * Discovered template. + *

+ * Templates backed by files located in a template root are discovered automatically. Furthermore, extensions can produce this + * build item in order to provide a template that is not backed by a file. + * + * @see TemplateRootBuildItem */ public final class TemplatePathBuildItem extends MultiBuildItem { + /** + * + * @return a new builder instance + */ + public static Builder builder() { + return new Builder(); + } + static final String TAGS = "tags/"; private final String path; - private final Path fullPath; private final String content; + private final Path fullPath; + private final String extensionInfo; + /** + * + * @param path + * @param fullPath + * @param content + * @deprecated Use the {@link #builder()} instead + */ + @Deprecated public TemplatePathBuildItem(String path, Path fullPath, String content) { + this(Objects.requireNonNull(path), Objects.requireNonNull(content), Objects.requireNonNull(fullPath), null); + } + + private TemplatePathBuildItem(String path, String content, Path fullPath, String extensionInfo) { this.path = path; - this.fullPath = fullPath; this.content = content; + this.fullPath = fullPath; + this.extensionInfo = extensionInfo; } /** - * Uses the {@code /} path separator. + * The path relative to the template root. The {@code /} is used as a path separator. + *

+ * The path must be unique, i.e. if there are multiple templates with the same path then the template analysis fails during + * build. * * @return the path relative to the template root */ @@ -31,14 +62,30 @@ public String getPath() { } /** - * Uses the system-dependent path separator. + * The full path of the template which uses the system-dependent path separator. * - * @return the full path of the template + * @return the full path, or {@code null} for templates that are not backed by a file */ public Path getFullPath() { return fullPath; } + /** + * + * @return the content of the template + */ + public String getContent() { + return content; + } + + /** + * + * @return the extension info + */ + public String getExtensionInfo() { + return extensionInfo; + } + /** * * @return {@code true} if it represents a user tag, {@code false} otherwise @@ -47,12 +94,87 @@ public boolean isTag() { return path.startsWith(TAGS); } + /** + * + * @return {@code true} if it does not represent a tag, {@code false} otherwise + */ public boolean isRegular() { return !isTag(); } - public String getContent() { - return content; + /** + * + * @return {@code true} if it's backed by a file + */ + public boolean isFileBased() { + return fullPath != null; + } + + public String getSourceInfo() { + return isFileBased() ? getFullPath().toString() : extensionInfo; + } + + public static class Builder { + + private String path; + private String content; + private Path fullPath; + private String extensionInfo; + + /** + * Set the path relative to the template root. The {@code /} is used as a path separator. + *

+ * The path must be unique, i.e. if there are multiple templates with the same path then the template analysis fails + * during build. + * + * @param path + * @return self + */ + public Builder path(String path) { + this.path = Objects.requireNonNull(path); + return this; + } + + /** + * Set the content of the template. + * + * @param content + * @return self + */ + public Builder content(String content) { + this.content = Objects.requireNonNull(content); + return this; + } + + /** + * Set the full path of the template for templates that are backed by a file. + * + * @param fullPath + * @return self + */ + public Builder fullPath(Path fullPath) { + this.fullPath = Objects.requireNonNull(fullPath); + return this; + } + + /** + * Set the extension info for templates that are not backed by a file. + * + * @param info + * @return self + */ + public Builder extensionInfo(String info) { + this.extensionInfo = info; + return this; + } + + public TemplatePathBuildItem build() { + if (fullPath == null && extensionInfo == null) { + throw new IllegalStateException("Templates that are not backed by a file must provide extension info"); + } + return new TemplatePathBuildItem(path, content, fullPath, extensionInfo); + } + } } diff --git a/extensions/qute/deployment/src/test/java/io/quarkus/qute/deployment/builditemtemplate/AdditionalTemplatePathDuplicatesTest.java b/extensions/qute/deployment/src/test/java/io/quarkus/qute/deployment/builditemtemplate/AdditionalTemplatePathDuplicatesTest.java new file mode 100644 index 0000000000000..4f2ac577bfa09 --- /dev/null +++ b/extensions/qute/deployment/src/test/java/io/quarkus/qute/deployment/builditemtemplate/AdditionalTemplatePathDuplicatesTest.java @@ -0,0 +1,60 @@ +package io.quarkus.qute.deployment.builditemtemplate; + +import static org.junit.jupiter.api.Assertions.fail; + +import java.util.function.Consumer; + +import org.jboss.shrinkwrap.api.asset.StringAsset; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import io.quarkus.builder.BuildChainBuilder; +import io.quarkus.builder.BuildContext; +import io.quarkus.builder.BuildStep; +import io.quarkus.qute.deployment.TemplatePathBuildItem; +import io.quarkus.test.QuarkusUnitTest; + +public class AdditionalTemplatePathDuplicatesTest { + + @RegisterExtension + static final QuarkusUnitTest config = new QuarkusUnitTest() + .withApplicationRoot(root -> root + .addAsResource(new StringAsset("Hi {name}!"), "templates/hi.txt")) + .addBuildChainCustomizer(buildCustomizer()) + .setExpectedException(IllegalStateException.class, true); + + static Consumer buildCustomizer() { + return new Consumer() { + @Override + public void accept(BuildChainBuilder builder) { + builder.addBuildStep(new BuildStep() { + @Override + public void execute(BuildContext context) { + context.produce(TemplatePathBuildItem.builder() + .path("hi.txt") + .extensionInfo("test-ext") + .content("Hello {name}!").build()); + } + }).produces(TemplatePathBuildItem.class) + .build(); + + builder.addBuildStep(new BuildStep() { + @Override + public void execute(BuildContext context) { + context.produce(TemplatePathBuildItem.builder() + .path("hi.txt") + .extensionInfo("test-ext") + .content("Hello {name}!").build()); + } + }).produces(TemplatePathBuildItem.class) + .build(); + } + }; + } + + @Test + public void test() { + fail(); + } + +} diff --git a/extensions/qute/deployment/src/test/java/io/quarkus/qute/deployment/builditemtemplate/AdditionalTemplatePathTest.java b/extensions/qute/deployment/src/test/java/io/quarkus/qute/deployment/builditemtemplate/AdditionalTemplatePathTest.java new file mode 100644 index 0000000000000..24e4306532a16 --- /dev/null +++ b/extensions/qute/deployment/src/test/java/io/quarkus/qute/deployment/builditemtemplate/AdditionalTemplatePathTest.java @@ -0,0 +1,59 @@ +package io.quarkus.qute.deployment.builditemtemplate; + +import static org.junit.jupiter.api.Assertions.assertEquals; + +import java.util.function.Consumer; + +import jakarta.inject.Inject; + +import org.jboss.shrinkwrap.api.asset.StringAsset; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import io.quarkus.builder.BuildChainBuilder; +import io.quarkus.builder.BuildContext; +import io.quarkus.builder.BuildStep; +import io.quarkus.qute.Engine; +import io.quarkus.qute.deployment.TemplatePathBuildItem; +import io.quarkus.test.QuarkusUnitTest; + +public class AdditionalTemplatePathTest { + + @RegisterExtension + static final QuarkusUnitTest config = new QuarkusUnitTest() + .withApplicationRoot(root -> root + .addAsResource(new StringAsset("Hi {name}!"), "templates/hi.txt") + .addAsResource(new StringAsset("And... {#include foo/hello /}"), "templates/include.txt")) + .addBuildChainCustomizer(buildCustomizer()); + + static Consumer buildCustomizer() { + return new Consumer() { + @Override + public void accept(BuildChainBuilder builder) { + builder.addBuildStep(new BuildStep() { + @Override + public void execute(BuildContext context) { + context.produce(TemplatePathBuildItem.builder() + .path("foo/hello.txt") + .extensionInfo("test-ext") + .content("Hello {name}!").build()); + } + }).produces(TemplatePathBuildItem.class) + .build(); + + } + }; + } + + @Inject + Engine engine; + + @Test + public void testTemplate() { + assertEquals("Hi M!", engine.getTemplate("hi").data("name", "M").render()); + assertEquals("Hello M!", engine.getTemplate("foo/hello.txt").data("name", "M").render()); + assertEquals("Hello M!", engine.getTemplate("foo/hello").data("name", "M").render()); + assertEquals("And... Hello M!", engine.getTemplate("include").data("name", "M").render()); + } + +} diff --git a/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/EngineProducer.java b/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/EngineProducer.java index 48a5360c3e92e..5e513a98d04c5 100644 --- a/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/EngineProducer.java +++ b/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/EngineProducer.java @@ -3,6 +3,7 @@ import java.io.IOException; import java.io.InputStreamReader; import java.io.Reader; +import java.io.StringReader; import java.lang.reflect.InvocationTargetException; import java.net.URL; import java.nio.charset.Charset; @@ -78,6 +79,7 @@ public class EngineProducer { private final List tags; private final List suffixes; private final Set templateRoots; + private final Map templateContents; private final Pattern templatePathExclude; private final Locale defaultLocale; private final Charset defaultCharset; @@ -91,6 +93,7 @@ public EngineProducer(QuteContext context, QuteConfig config, QuteRuntimeConfig this.contentTypes = contentTypes; this.suffixes = config.suffixes; this.templateRoots = context.getTemplateRoots(); + this.templateContents = Map.copyOf(context.getTemplateContents()); this.tags = context.getTags(); this.templatePathExclude = config.templatePathExclude; this.defaultLocale = locales.defaultLocale; @@ -334,10 +337,11 @@ private Optional locate(String path) { if (templatePathExclude.matcher(path).matches()) { return Optional.empty(); } + // First try to locate file-based templates for (String templateRoot : templateRoots) { URL resource = null; String templatePath = templateRoot + path; - LOGGER.debugf("Locate template for %s", templatePath); + LOGGER.debugf("Locate template file for %s", templatePath); resource = locatePath(templatePath); if (resource == null) { // Try path with suffixes @@ -357,6 +361,25 @@ private Optional locate(String path) { return Optional.of(new ResourceTemplateLocation(resource, createVariant(templatePath))); } } + // Then try the template contents + LOGGER.debugf("Locate template contents for %s", path); + String content = templateContents.get(path); + if (path == null) { + // Try path with suffixes + for (String suffix : suffixes) { + String pathWithSuffix = path + "." + suffix; + if (templatePathExclude.matcher(pathWithSuffix).matches()) { + continue; + } + content = templateContents.get(pathWithSuffix); + if (content != null) { + break; + } + } + } + if (content != null) { + return Optional.of(new ContentTemplateLocation(content, createVariant(path))); + } return Optional.empty(); } @@ -439,7 +462,7 @@ static class ResourceTemplateLocation implements TemplateLocation { private final URL resource; private final Optional variant; - public ResourceTemplateLocation(URL resource, Variant variant) { + ResourceTemplateLocation(URL resource, Variant variant) { this.resource = resource; this.variant = Optional.ofNullable(variant); } @@ -467,4 +490,26 @@ public Optional getVariant() { } + static class ContentTemplateLocation implements TemplateLocation { + + private final String content; + private final Optional variant; + + ContentTemplateLocation(String content, Variant variant) { + this.content = content; + this.variant = Optional.ofNullable(variant); + } + + @Override + public Reader read() { + return new StringReader(content); + } + + @Override + public Optional getVariant() { + return variant; + } + + } + } diff --git a/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/QuteRecorder.java b/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/QuteRecorder.java index 0fff3270c5735..2e53fe166580d 100644 --- a/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/QuteRecorder.java +++ b/extensions/qute/runtime/src/main/java/io/quarkus/qute/runtime/QuteRecorder.java @@ -12,7 +12,7 @@ public class QuteRecorder { public Supplier createContext(List resolverClasses, List templatePaths, List tags, Map> variants, - List templateGlobalProviderClasses, Set templateRoots) { + List templateGlobalProviderClasses, Set templateRoots, Map templateContents) { return new Supplier() { @Override @@ -48,6 +48,11 @@ public List getTemplateGlobalProviderClasses() { public Set getTemplateRoots() { return templateRoots; } + + @Override + public Map getTemplateContents() { + return templateContents; + } }; } }; @@ -67,6 +72,8 @@ public interface QuteContext { Set getTemplateRoots(); + Map getTemplateContents(); + } }