Some refactoring
Some checks are pending
/ build (push) Waiting to run

Signed-off-by: Minecon724 <git@m724.eu>
This commit is contained in:
Minecon724 2025-03-08 10:54:17 +01:00
parent 053ce27f3c
commit 5d98a8fd60
Signed by untrusted user who does not match committer: Minecon724
GPG key ID: 3CCC4D267742C8E8
17 changed files with 188 additions and 106 deletions

View file

@ -2,10 +2,11 @@ blog-software(config, template, content) = blog website
## Usage ## Usage
1. [Download the program from here](/Minecon724/blog-software-java/releases) 1. [Download the program from here](/Minecon724/blog-software-java/releases) \
**Mini** or **Full**? **Full** contains dependencies, which use native libraries. Choose **Full**, if unsure.
2. Run the program: 2. Run the program:
```shell ```shell
java -jar blog-0.0.1-shaded.jar -s example_workdir java -jar blog-0.0.2-standalone-full.jar -s example_workdir
``` ```
For tips on how to create your own project (workdir), see [Project format](#Project format) below. For tips on how to create your own project (workdir), see [Project format](#Project format) below.
@ -25,11 +26,11 @@ There's an ["Example workdir"](/Minecon724/blog-software-java/src/branch/master/
Basically: Basically:
- `assets/` - contains static assets - `assets/` - contains static assets
- `posts/` - contains posts. Post format: - `articles/` - contains articles. Post format:
- Header / metadata: - Header / metadata:
- `title A title` - post title - `title A title` - article title
- `summary This is a post with a title` - post summary - `summary This is a article with a title` - article summary
- `live` - is the post live (not draft), doesn't need an argument - `live` - is the article live (not draft), doesn't need an argument
- Custom properties, which are Strings - Custom properties, which are Strings
- ` ` - Empty line separates header from content - ` ` - Empty line separates header from content
- Post content in HTML. Generally not sanitized, but depends on template. - Post content in HTML. Generally not sanitized, but depends on template.
@ -43,5 +44,5 @@ Basically:
https://pebbletemplates.io is used https://pebbletemplates.io is used
- `static/` - contains static assets - `static/` - contains static assets
- `article_template.html` - post template - `article_template.html` - article template
- `index_template.html` - index.html template - `index_template.html` - index.html template

View file

@ -1,6 +1,9 @@
name: my blog name: my blog
baseUrl: https://example.com/blog baseUrl: https://example.com/blog
# Whether to apply Pebble templating to posts. Disabled by default, not recommended.
# templateArticles: true
coolProperty: 1231 coolProperty: 1231
coolerProperty: coolerProperty:
isMap: true isMap: true

View file

@ -36,7 +36,7 @@
</header> </header>
<div> <div>
{{ article.htmlContent | raw }} {{ content | raw }}
</div> </div>
</article> </article>
</body> </body>

View file

@ -10,7 +10,7 @@
<body> <body>
<h1>{{ site.name }} - {{ site.custom.coolProperty }}</h1> <h1>{{ site.name }} - {{ site.custom.coolProperty }}</h1>
{% for article in articles %} {% for article in articles %}
<a href="post/{{ article.slug }}.html" class="post-short"> <a href="article/{{ article.slug }}.html" class="article-short">
<article> <article>
<header> <header>
<p class="title">{{ article.title }}</p> <p class="title">{{ article.title }}</p>

32
pom.xml
View file

@ -72,6 +72,7 @@
<groupId>com.github.luben</groupId> <groupId>com.github.luben</groupId>
<artifactId>zstd-jni</artifactId> <artifactId>zstd-jni</artifactId>
<version>1.5.7-1</version> <!-- Released Feb 20, 2025 --> <version>1.5.7-1</version> <!-- Released Feb 20, 2025 -->
<optional>true</optional>
</dependency> </dependency>
@ -113,6 +114,7 @@
<version>3.6.0</version> <!-- Released May 31, 2024 --> <version>3.6.0</version> <!-- Released May 31, 2024 -->
<executions> <executions>
<execution> <execution>
<id>shade-mini</id>
<phase>package</phase> <phase>package</phase>
<goals> <goals>
<goal>shade</goal> <goal>shade</goal>
@ -121,6 +123,36 @@
<minimizeJar>true</minimizeJar> <minimizeJar>true</minimizeJar>
<createDependencyReducedPom>false</createDependencyReducedPom> <createDependencyReducedPom>false</createDependencyReducedPom>
<shadedArtifactAttached>true</shadedArtifactAttached> <shadedArtifactAttached>true</shadedArtifactAttached>
<shadedClassifierName>standalone-mini</shadedClassifierName>
<artifactSet>
<excludes>
<exclude>com.github.luben:zstd-jni</exclude>
</excludes>
</artifactSet>
<filters>
<filter>
<artifact>*:*</artifact>
<excludes>
<exclude>META-INF/*.SF</exclude>
<exclude>META-INF/*.DSA</exclude>
<exclude>META-INF/*.RSA</exclude>
</excludes>
</filter>
</filters>
</configuration>
</execution>
<execution>
<id>shade-full</id>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<minimizeJar>true</minimizeJar>
<createDependencyReducedPom>false</createDependencyReducedPom>
<shadedArtifactAttached>true</shadedArtifactAttached>
<shadedClassifierName>standalone-full</shadedClassifierName>
<filters> <filters>
<filter> <filter>
<artifact>*:*</artifact> <artifact>*:*</artifact>

View file

@ -7,8 +7,9 @@
package eu.m724.blog; package eu.m724.blog;
import eu.m724.blog.compress.FileCompressor; import eu.m724.blog.compress.FileCompressor;
import eu.m724.blog.compress.NoSuchAlgorithmException;
import eu.m724.blog.object.Article;
import eu.m724.blog.object.Feed; import eu.m724.blog.object.Feed;
import eu.m724.blog.object.Post;
import eu.m724.blog.object.RenderOptions; import eu.m724.blog.object.RenderOptions;
import eu.m724.blog.object.Site; import eu.m724.blog.object.Site;
import eu.m724.blog.template.TemplateRenderer; import eu.m724.blog.template.TemplateRenderer;
@ -28,7 +29,7 @@ import java.util.stream.Collectors;
/** /**
* The {@code BlogBuilder} class facilitates building a static blog by managing templates, * The {@code BlogBuilder} class facilitates building a static blog by managing templates,
* assets, posts, and rendering output files. It uses a Git repository as the * assets, articles, and rendering output files. It uses a Git repository as the
* source for the blog's content and configuration. * source for the blog's content and configuration.
*/ */
public class BlogBuilder { public class BlogBuilder {
@ -141,14 +142,14 @@ public class BlogBuilder {
if (template == null) if (template == null)
this.template = new TemplateRenderer(site, templateDirectory, fileHashes); this.template = new TemplateRenderer(site, templateDirectory, fileHashes);
LOGGER.debug("Rendering posts..."); LOGGER.debug("Rendering articles...");
var posts = renderPosts(); var articles = renderArticles();
LOGGER.debug("Rendering meta..."); LOGGER.debug("Rendering meta...");
posts.sort(Comparator.comparing(Post::createdAt).reversed()); articles.sort(Comparator.comparing(Article::createdAt).reversed());
Files.writeString(outputDirectory.resolve("index.html"), template.renderIndex(posts)); Files.writeString(outputDirectory.resolve("index.html"), template.renderIndex(articles));
Files.writeString(outputDirectory.resolve("posts.rss"), Feed.generateRss(site, posts)); Files.writeString(outputDirectory.resolve("articles.rss"), Feed.generateRss(site, articles));
if (!renderOptions.compress().isEmpty()) { if (!renderOptions.compress().isEmpty()) {
LOGGER.debug("Compressing..."); LOGGER.debug("Compressing...");
@ -166,43 +167,44 @@ public class BlogBuilder {
} }
} }
private List<Post> renderPosts() throws IOException { private List<Article> renderArticles() throws IOException {
Files.createDirectory(outputDirectory.resolve("post")); Files.createDirectory(outputDirectory.resolve("article"));
var postDirectory = workingDirectory.resolve("posts"); var articleDirectory = workingDirectory.resolve("articles");
var posts = new ArrayList<Post>(); var articles = new ArrayList<Article>();
try (var stream = Files.walk(postDirectory)) { try (var stream = Files.walk(articleDirectory)) {
for (var path : stream.collect(Collectors.toSet())) { for (var path : stream.collect(Collectors.toSet())) {
if (!Files.isRegularFile(path)) if (!Files.isRegularFile(path))
continue; // directory is created below continue; // directory is created below
if (!path.toString().endsWith(".html")) { if (!path.toString().endsWith(".html")) {
LOGGER.warn("Post {}: unsupported file type", path.getFileName()); // TODO print file type too
LOGGER.warn("[Article {}] Unsupported file type", path.getFileName());
continue; continue;
} }
path = postDirectory.relativize(path); path = articleDirectory.relativize(path);
var post = Post.fromFile(git, path); var article = Article.fromFile(git, path);
if (post.draft() && !renderDrafts) { if (article.draft() && !renderDrafts) {
LOGGER.info("Post {}: draft, ignoring", path.getFileName()); LOGGER.info("[Article {}] Draft. Ignoring, because you didn't specify to render drafts.", path.getFileName());
continue; continue;
} }
var render = template.renderPost(post); var render = template.renderArticle(article);
var outFile = outputDirectory.resolve("post").resolve(path); var outFile = outputDirectory.resolve("article").resolve(path);
try { try {
Files.createDirectory(outFile.getParent()); Files.createDirectory(outFile.getParent());
} catch (FileAlreadyExistsException ignored) { } } catch (FileAlreadyExistsException ignored) { }
Files.writeString(outFile, render); Files.writeString(outFile, render);
posts.add(post); articles.add(article);
} }
} }
return posts; return articles;
} }
private Map<String, String> copyStaticAssets() throws IOException { private Map<String, String> copyStaticAssets() throws IOException {
@ -232,9 +234,17 @@ public class BlogBuilder {
} }
private void compressOutput() throws IOException { private void compressOutput() throws IOException {
var compressors = renderOptions.compress().stream() var compressors = new ArrayList<FileCompressor>();
.map(FileCompressor::new)
.toList(); for (var algorithm : renderOptions.compress()) {
try {
var compressor = new FileCompressor(algorithm);
compressors.add(compressor);
} catch (NoSuchAlgorithmException e) {
LOGGER.error("No such compression algorithm, ignoring: {}", e.getAlgorithm());
}
}
Set<Path> tree; Set<Path> tree;
try (var walk = Files.walk(outputDirectory)) { try (var walk = Files.walk(outputDirectory)) {
@ -246,7 +256,7 @@ public class BlogBuilder {
try { try {
compressor.compress(path); compressor.compress(path);
} catch (CompressorException e) { } catch (CompressorException e) {
LOGGER.error("Error compressing {} to {}: {}", path, compressor.getAlgorithm(), e.getMessage()); LOGGER.error("Error compressing \"{}\" to \"{}\": {}", path, compressor.getAlgorithm(), e.getMessage());
} }
} }
} }

View file

@ -51,7 +51,9 @@ public class Main {
var end = System.nanoTime(); var end = System.nanoTime();
LOGGER.info("Exported to {} in {} ms", outputDirectory.toAbsolutePath(), "%.2f".formatted((end - start) / 1000000.0)); LOGGER.info("Exported to {} in {} ms", outputDirectory.toAbsolutePath(), "%.2f".formatted((end - start) / 1000000.0));
builder.startServer(openBrowser); if (startServer) {
builder.startServer(openBrowser);
}
} }
private static CommandLine getCommandLine(String[] args) { private static CommandLine getCommandLine(String[] args) {

View file

@ -58,7 +58,6 @@ public class Server {
* @throws IOException if an I/O error occurs during server initialization * @throws IOException if an I/O error occurs during server initialization
*/ */
public void start() throws IOException { public void start() throws IOException {
System.out.println(contextPath);
var server = HttpServer.create(listenAddress, 0); var server = HttpServer.create(listenAddress, 0);
server.createContext(contextPath, SimpleFileServer.createFileHandler(sitePath.toAbsolutePath())); server.createContext(contextPath, SimpleFileServer.createFileHandler(sitePath.toAbsolutePath()));
server.start(); server.start();

View file

@ -10,13 +10,29 @@ import org.apache.commons.compress.compressors.CompressorException;
import org.apache.commons.compress.compressors.CompressorStreamFactory; import org.apache.commons.compress.compressors.CompressorStreamFactory;
import java.io.IOException; import java.io.IOException;
import java.io.OutputStream;
import java.nio.file.*; import java.nio.file.*;
public class FileCompressor { public class FileCompressor {
private final String algorithm; private final String algorithm;
public FileCompressor(String algorithm) { /**
* Constructs a {@link FileCompressor} instance using the provided algorithm.
*
* @param algorithm The algorithm, for valid names see {@link CompressorStreamFactory}
* @throws NoSuchAlgorithmException If that algorithm is unavailable or the name is invalid.
*/
public FileCompressor(String algorithm) throws NoSuchAlgorithmException {
this.algorithm = algorithm; this.algorithm = algorithm;
try {
var os = new CompressorStreamFactory().createCompressorOutputStream(algorithm, OutputStream.nullOutputStream());
os.close();
} catch (NoClassDefFoundError | CompressorException e) {
throw new NoSuchAlgorithmException(algorithm, e);
} catch (IOException e) {
throw new RuntimeException("Unexpected IOException closing test output stream", e);
}
} }
public void compress(Path source) throws IOException, CompressorException { public void compress(Path source) throws IOException, CompressorException {

View file

@ -0,0 +1,21 @@
/*
* Copyright (c) 2025 blog-software-java developers
* blog-software-java is licensed under the GNU General Public License. See the LICENSE.md file
* in the project root for the full license text.
*/
package eu.m724.blog.compress;
public class NoSuchAlgorithmException extends Exception {
private final String algorithm;
public NoSuchAlgorithmException(String algorithm, Throwable cause) {
super("Algorithm unavailable:" + algorithm, cause);
this.algorithm = algorithm;
}
public String getAlgorithm() {
return algorithm;
}
}

View file

@ -19,7 +19,7 @@ import java.util.HashMap;
import java.util.Map; import java.util.Map;
/** /**
* The {@code Post} class represents a blog post with various attributes including metadata and content. * The {@code Article} class represents a blog post with various attributes including metadata and content.
* *
* @param slug A unique identifier for the post derived from the file name. * @param slug A unique identifier for the post derived from the file name.
* @param title The title of the post. * @param title The title of the post.
@ -33,7 +33,7 @@ import java.util.Map;
* @param custom A map of custom properties or metadata associated with the post. * @param custom A map of custom properties or metadata associated with the post.
* @param rawContent The raw content of the post, which <em>currently</em> is usually HTML. * @param rawContent The raw content of the post, which <em>currently</em> is usually HTML.
*/ */
public record Post( public record Article(
String slug, String slug,
String title, String title,
String summary, String summary,
@ -48,24 +48,24 @@ public record Post(
Map<String, String> custom, Map<String, String> custom,
String rawContent String rawContent
) { ) {
private static final Logger LOGGER = LoggerFactory.getLogger(Post.class); private static final Logger LOGGER = LoggerFactory.getLogger(Article.class);
/** /**
* Creates a {@link Post} instance by reading and parsing the content of a post file. * Creates a {@link Article} instance by reading and parsing the content of an article file.
* <p> * <p>
* The method extracts metadata properties, content, and versioning information * The method extracts metadata properties, content, and versioning information
* based on the Git history of the file. * based on the Git history of the file.
* *
* @param git the Git repository used to retrieve versioning and commit information * @param git the Git repository used to retrieve versioning and commit information
* @param path the relative path to the file within the "posts" directory * @param path the relative path to the file within the "articles" directory
* @return a {@link Post} object populated with data extracted from the specified file * @return a {@link Article} object populated with data extracted from the specified file
* @throws IOException if an error occurs during file reading * @throws IOException if an error occurs during file reading
*/ */
public static Post fromFile(Git git, Path path) throws IOException { public static Article fromFile(Git git, Path path) throws IOException {
/* read properties before filtering */ /* read properties before filtering */
var slug = path.getFileName().toString().split("\\.")[0]; var slug = path.getFileName().toString().split("\\.")[0];
path = Path.of("posts").resolve(path); path = Path.of("articles").resolve(path);
var lines = Files.readAllLines(git.getRepository().getDirectory().toPath().getParent().resolve(path)); var lines = Files.readAllLines(git.getRepository().getDirectory().toPath().getParent().resolve(path));
var properties = new HashMap<String, String>(); var properties = new HashMap<String, String>();
@ -81,7 +81,7 @@ public record Post(
break; break;
if (properties.putIfAbsent(key, data) != null) if (properties.putIfAbsent(key, data) != null)
LOGGER.warn("[Post {}] Ignoring duplicate property: {}", slug, key); LOGGER.warn("[Article {}] Ignoring duplicate property: {}", slug, key);
} }
var content = String.join("\n", lines).strip(); var content = String.join("\n", lines).strip();
@ -132,18 +132,9 @@ public record Post(
} }
} catch (GitAPIException e) { } catch (GitAPIException e) {
draft = true; draft = true;
LOGGER.warn("[Post {}] Draft because of a Git exception: {}", slug, e.getMessage()); LOGGER.warn("[Article {}] Draft because of a Git exception: {}", slug, e.getMessage());
} }
return new Post(slug, title, summary, draft, revisions, createdBy, createdAt, modifiedBy, modifiedAt, custom, content); return new Article(slug, title, summary, draft, revisions, createdBy, createdAt, modifiedBy, modifiedAt, custom, content);
}
/**
* Retrieves the raw HTML content associated with the post.
*
* @return the raw HTML content as a string
*/
public String htmlContent() {
return rawContent;
} }
} }

View file

@ -16,21 +16,21 @@ public class Feed {
* Generates an RSS feed XML string for a given website and its list of blog posts. * Generates an RSS feed XML string for a given website and its list of blog posts.
* *
* @param site the {@code Site} object representing the website for which the RSS feed is generated * @param site the {@code Site} object representing the website for which the RSS feed is generated
* @param posts the list of {@code Post} objects representing the blog posts to include in the RSS feed * @param articles the list of {@link Article} objects representing the blog posts to include in the RSS feed
* @return a {@code String} containing the formatted RSS feed in XML * @return a {@code String} containing the formatted RSS feed in XML
*/ */
public static String generateRss(Site site, List<Post> posts) { public static String generateRss(Site site, List<Article> articles) {
var content = new StringBuilder("<?xml version=\"1.0\" encoding=\"UTF-8\" ?>"); var content = new StringBuilder("<?xml version=\"1.0\" encoding=\"UTF-8\" ?>");
content.append("<rss version=\"2.0\"><channel>"); content.append("<rss version=\"2.0\"><channel>");
content.append("<title>%s</title>".formatted(site.name())); content.append("<title>%s</title>".formatted(site.name()));
content.append("<link>%s</link>".formatted(site.baseUrl())); content.append("<link>%s</link>".formatted(site.baseUrl()));
for (var post : posts) { for (var article : articles) {
content.append("<item>"); content.append("<item>");
content.append("<title>%s</title>".formatted(post.title())); content.append("<title>%s</title>".formatted(article.title()));
content.append("<link>%s/post/%s.html</link>".formatted(site.baseUrl(), post.slug())); content.append("<link>%s/article/%s.html</link>".formatted(site.baseUrl(), article.slug()));
content.append("<description>%s</description>".formatted(post.summary())); content.append("<description>%s</description>".formatted(article.summary()));
content.append("<pubDate>%s</pubDate>".formatted(post.createdAt().format(formatter))); content.append("<pubDate>%s</pubDate>".formatted(article.createdAt().format(formatter)));
content.append("</item>"); content.append("</item>");
} }

View file

@ -12,7 +12,6 @@ import org.snakeyaml.engine.v2.api.LoadSettings;
import java.io.IOException; import java.io.IOException;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
import java.util.HashMap;
import java.util.Map; import java.util.Map;
/** /**
@ -21,6 +20,7 @@ import java.util.Map;
* @param name the name of the site * @param name the name of the site
* @param baseUrl the base URL of the site * @param baseUrl the base URL of the site
* @param directory The directory that the site is installed into. Like "https://example.com/blog" turns to "/blog" * @param directory The directory that the site is installed into. Like "https://example.com/blog" turns to "/blog"
* @param templateArticles whether to parse posts with Pebble templating
* @param custom a map of additional custom properties * @param custom a map of additional custom properties
*/ */
public record Site( public record Site(
@ -29,6 +29,8 @@ public record Site(
String directory, String directory,
boolean templateArticles,
Map<String, Object> custom Map<String, Object> custom
) { ) {
/** /**
@ -43,33 +45,21 @@ public record Site(
var load = new Load(LoadSettings.builder().build()); var load = new Load(LoadSettings.builder().build());
var yaml = (Map<String, Object>) load.loadFromInputStream(Files.newInputStream(path)); var yaml = (Map<String, Object>) load.loadFromInputStream(Files.newInputStream(path));
String name = null; String name = (String) yaml.get("name");
String baseUrl = null; String baseUrl = (String) yaml.getOrDefault("baseUrl", "/");
var custom = new HashMap<String, Object>(); var templateArticles = (boolean) yaml.getOrDefault("templateArticles", false);
for (var key : yaml.keySet()) { String directory = "/";
var value = yaml.get(key); if (baseUrl != null) {
var temp = baseUrl.substring(baseUrl.indexOf(':') + 3);
switch (key) { var slashIndex = temp.indexOf('/');
case "name": if (slashIndex != -1) {
name = (String) value; directory += temp.substring(slashIndex + 1) + "/";
break;
case "baseUrl":
baseUrl = (String) value;
break;
default:
custom.put(key, value);
} }
} }
String directory = null;
if (baseUrl != null) {
var temp = baseUrl.substring(baseUrl.indexOf(':') + 3);
directory = temp.substring(temp.indexOf('/'));
}
return new Site( return new Site(
name, baseUrl, directory, custom name, baseUrl, directory, templateArticles, yaml
); );
} }
} }

View file

@ -43,7 +43,7 @@ public class TemplateExtension extends AbstractExtension {
path = CacheBuster.insertHashInPath(path, hash); path = CacheBuster.insertHashInPath(path, hash);
} }
return site.directory() + "/" + path; return site.directory() + path;
} }
}, },
"asset", new Function() { "asset", new Function() {
@ -61,10 +61,15 @@ public class TemplateExtension extends AbstractExtension {
path = CacheBuster.insertHashInPath(path, hash); path = CacheBuster.insertHashInPath(path, hash);
} }
return site.directory() + "/" + path; return site.directory() + path;
} }
} }
// TODO make url_for that supports relative and absolute paths // TODO make url_for that supports relative and absolute paths
); );
} }
@Override
public Map<String, Object> getGlobalVariables() {
return Map.of("site", site);
}
} }

View file

@ -6,7 +6,7 @@
package eu.m724.blog.template; package eu.m724.blog.template;
import eu.m724.blog.object.Post; import eu.m724.blog.object.Article;
import eu.m724.blog.object.Site; import eu.m724.blog.object.Site;
import io.pebbletemplates.pebble.PebbleEngine; import io.pebbletemplates.pebble.PebbleEngine;
import io.pebbletemplates.pebble.loader.FileLoader; import io.pebbletemplates.pebble.loader.FileLoader;
@ -24,6 +24,8 @@ import java.util.Map;
*/ */
public class TemplateRenderer { public class TemplateRenderer {
private final Site site; private final Site site;
private final PebbleEngine pebbleEngine;
private final PebbleTemplate indexTemplate, articleTemplate; private final PebbleTemplate indexTemplate, articleTemplate;
/** /**
@ -38,7 +40,7 @@ public class TemplateRenderer {
loader.setPrefix(templateDirectory.toString()); loader.setPrefix(templateDirectory.toString());
loader.setSuffix(".html"); loader.setSuffix(".html");
var pebbleEngine = new PebbleEngine.Builder() this.pebbleEngine = new PebbleEngine.Builder()
.loader(loader) .loader(loader)
.extension(new TemplateExtension(site, fileHashes)) .extension(new TemplateExtension(site, fileHashes))
.build(); .build();
@ -51,37 +53,47 @@ public class TemplateRenderer {
/** /**
* Renders the index page using this template. * Renders the index page using this template.
* *
* @param posts the {@link Post}s to be included in the index page * @param articles the {@link Article}s to be included in the index page
* @return the rendered index HTML page as a string * @return the rendered index HTML page as a string
* @throws IOException if an error occurs during the template evaluation * @throws IOException if an error occurs during the template evaluation
*/ */
public String renderIndex(List<Post> posts) throws IOException { public String renderIndex(List<Article> articles) throws IOException {
Map<String, Object> context = Map.of( var context = Map.of(
"site", site, "articles", articles
"articles", posts
); );
var writer = new StringWriter(); return renderTemplate(indexTemplate, context);
indexTemplate.evaluate(writer, context);
return writer.toString();
} }
/** /**
* Renders the content of a post using this template. * Renders the content of a post using this template.
* *
* @param post the {@link Post} to be rendered * @param article the {@link Article} to be rendered
* @return the rendered post HTML page as a string * @return the rendered post HTML page as a string
* @throws IOException if an error occurs during template evaluation * @throws IOException if an error occurs during template evaluation
*/ */
public String renderPost(Post post) throws IOException { public String renderArticle(Article article) throws IOException {
Map<String, Object> context = Map.of( String content = article.rawContent();
"site", site,
"article", post if (site.templateArticles()) {
var context = Map.of(
"article", article
);
content = renderTemplate(pebbleEngine.getLiteralTemplate(content), context);
}
var context = Map.of(
"article", article,
"content", content
); );
return renderTemplate(articleTemplate, context);
}
private String renderTemplate(PebbleTemplate template, Map<String, ?> context) throws IOException {
var writer = new StringWriter(); var writer = new StringWriter();
articleTemplate.evaluate(writer, context); template.evaluate(writer, (Map<String, Object>) context);
return writer.toString(); return writer.toString();
} }