Signed-off-by: Minecon724 <git@m724.eu>
This commit is contained in:
parent
121a2c6915
commit
fff71d1140
9 changed files with 201 additions and 35 deletions
|
@ -1 +1 @@
|
||||||
this is a static asset
|
Hello, world!
|
|
@ -1,5 +1,12 @@
|
||||||
# Render options here
|
# Render options here
|
||||||
|
|
||||||
|
# Pre-compress files to serve with web server software
|
||||||
compress:
|
compress:
|
||||||
- gz
|
- gz
|
||||||
- zstd
|
- zstd
|
||||||
|
|
||||||
|
# Add .hash. to static assets provided by template
|
||||||
|
remapTemplateStatic: true
|
||||||
|
|
||||||
|
# Add .hash. to site static assets
|
||||||
|
remapAssets: false
|
||||||
|
|
|
@ -25,5 +25,10 @@
|
||||||
<li>{{ e }}</li>
|
<li>{{ e }}</li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
|
<a href="{{ asset('hello.txt') }}">This is an asset that says:</a>
|
||||||
|
<blockquote>Hello, world!</blockquote>
|
||||||
|
|
||||||
|
<a href="{{ asset('another.txt') }}">This is another asset that says things about assets.</a>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
|
@ -17,10 +17,8 @@ import java.io.IOException;
|
||||||
import java.nio.file.FileAlreadyExistsException;
|
import java.nio.file.FileAlreadyExistsException;
|
||||||
import java.nio.file.Files;
|
import java.nio.file.Files;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.util.ArrayList;
|
import java.util.*;
|
||||||
import java.util.Comparator;
|
import java.util.concurrent.ThreadLocalRandom;
|
||||||
import java.util.List;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -132,12 +130,11 @@ public class BlogBuilder {
|
||||||
if (renderOptions == null)
|
if (renderOptions == null)
|
||||||
this.renderOptions = RenderOptions.fromConfig(workingDirectory.resolve("render.yml"));
|
this.renderOptions = RenderOptions.fromConfig(workingDirectory.resolve("render.yml"));
|
||||||
|
|
||||||
if (template == null)
|
LOGGER.debug("Copying static assets...");
|
||||||
this.template = new TemplateRenderer(site, templateDirectory);
|
var fileHashes = copyStaticAssets();
|
||||||
|
|
||||||
LOGGER.debug("Copying assets...");
|
if (template == null)
|
||||||
copyTree(workingDirectory.resolve("assets"), outputDirectory.resolve("assets"));
|
this.template = new TemplateRenderer(site, templateDirectory, fileHashes);
|
||||||
copyTree(templateDirectory.resolve("static"), outputDirectory.resolve("static"));
|
|
||||||
|
|
||||||
LOGGER.debug("Rendering posts...");
|
LOGGER.debug("Rendering posts...");
|
||||||
var posts = renderPosts();
|
var posts = renderPosts();
|
||||||
|
@ -203,6 +200,41 @@ public class BlogBuilder {
|
||||||
return posts;
|
return posts;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Map<String, String> copyStaticAssets() throws IOException {
|
||||||
|
var fileHashes = new HashMap<String, String>();
|
||||||
|
|
||||||
|
if (renderOptions.remapAssets()) {
|
||||||
|
var remapper = StaticCacheRemapper.fromGitRepository(workingDirectory);
|
||||||
|
if (remapper == null) {
|
||||||
|
LOGGER.warn("Site should be a Git directory"); // TODO like it isn't?
|
||||||
|
remapper = new StaticCacheRemapper(Integer.toHexString(ThreadLocalRandom.current().nextInt()));
|
||||||
|
}
|
||||||
|
|
||||||
|
var assetHashes = remapper.copyTree(workingDirectory.resolve("assets"), outputDirectory.resolve("assets"));
|
||||||
|
assetHashes.forEach((k, v) -> {
|
||||||
|
fileHashes.put("assets/" + k, v); // TODO this seems like a hack
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
copyTree(workingDirectory.resolve("assets"), outputDirectory.resolve("assets"));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (renderOptions.remapTemplateStatic()) {
|
||||||
|
var remapper = StaticCacheRemapper.fromGitRepository(templateDirectory);
|
||||||
|
if (remapper == null) {
|
||||||
|
LOGGER.warn("Template should be a Git directory");
|
||||||
|
remapper = new StaticCacheRemapper(Integer.toHexString(ThreadLocalRandom.current().nextInt()));
|
||||||
|
}
|
||||||
|
var templateStaticHashes = remapper.copyTree(templateDirectory.resolve("static"), outputDirectory.resolve("static"));
|
||||||
|
templateStaticHashes.forEach((k, v) -> {
|
||||||
|
fileHashes.put("static/" + k, v); // TODO this seems like a hack
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
copyTree(templateDirectory.resolve("static"), outputDirectory.resolve("static"));
|
||||||
|
}
|
||||||
|
|
||||||
|
return fileHashes;
|
||||||
|
}
|
||||||
|
|
||||||
private void compressOutput() throws IOException {
|
private void compressOutput() throws IOException {
|
||||||
var compressors = renderOptions.compress().stream()
|
var compressors = renderOptions.compress().stream()
|
||||||
.map(FileCompressor::new)
|
.map(FileCompressor::new)
|
||||||
|
|
112
src/main/java/eu/m724/blog/StaticCacheRemapper.java
Normal file
112
src/main/java/eu/m724/blog/StaticCacheRemapper.java
Normal file
|
@ -0,0 +1,112 @@
|
||||||
|
package eu.m724.blog;
|
||||||
|
|
||||||
|
import org.eclipse.jgit.api.Git;
|
||||||
|
import org.eclipse.jgit.api.errors.GitAPIException;
|
||||||
|
import org.eclipse.jgit.lib.RepositoryBuilder;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
public class StaticCacheRemapper {
|
||||||
|
private final String revision;
|
||||||
|
|
||||||
|
public StaticCacheRemapper(String revision) {
|
||||||
|
this.revision = revision;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getRevision() {
|
||||||
|
return revision;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a {@link StaticCacheRemapper} from a static files folder
|
||||||
|
* @param path the {@link Path} that points to the folder with the static files
|
||||||
|
* @return null if no Git repository at {@code path}
|
||||||
|
*/
|
||||||
|
public static StaticCacheRemapper fromGitRepository(Path path) {
|
||||||
|
path = path.toAbsolutePath();
|
||||||
|
|
||||||
|
var builder = new RepositoryBuilder()
|
||||||
|
.findGitDir(path.toFile());
|
||||||
|
|
||||||
|
if (builder.getGitDir() != null) {
|
||||||
|
var relativePath = builder.getGitDir().toPath().getParent().relativize(path);
|
||||||
|
|
||||||
|
try (
|
||||||
|
var repository = builder.build();
|
||||||
|
var git = new Git(repository)
|
||||||
|
) {
|
||||||
|
var log = git.log().addPath(relativePath.toString()).call();
|
||||||
|
var commit = log.iterator().next();
|
||||||
|
|
||||||
|
if (commit != null) {
|
||||||
|
var commitIdShort = commit.getId().getName().substring(0, 10); // TODO maybe less than 10 is ok
|
||||||
|
return new StaticCacheRemapper(commitIdShort);
|
||||||
|
}
|
||||||
|
} catch (GitAPIException | IOException e) {
|
||||||
|
// TODO do something about it
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, String> copyTree(Path srcDir, Path destDir) throws IOException {
|
||||||
|
var map = new HashMap<String, String>();
|
||||||
|
|
||||||
|
try (var walk = Files.walk(srcDir)) {
|
||||||
|
for (var src : walk.collect(Collectors.toSet())) {
|
||||||
|
var rel = srcDir.relativize(src);
|
||||||
|
var dest = destDir.resolve(rel);
|
||||||
|
|
||||||
|
if (Files.isRegularFile(src)) {
|
||||||
|
var parent = dest.getParent();
|
||||||
|
|
||||||
|
if (!Files.isDirectory(parent)) {
|
||||||
|
Files.createDirectories(parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
var fileName = dest.getFileName().toString();
|
||||||
|
fileName = insertHashInPath(fileName, revision);
|
||||||
|
dest = dest.resolveSibling(fileName);
|
||||||
|
|
||||||
|
Files.copy(src, dest);
|
||||||
|
map.put(rel.toString(), revision);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inserts a hash string before the file extension in a path.
|
||||||
|
* Example: insertHashInPath("a/path/like.this", "abc123") returns "a/path/like.abc123.this"
|
||||||
|
*
|
||||||
|
* @param path The original file path
|
||||||
|
* @param hash The hash to insert before the file extension
|
||||||
|
* @return The path with the hash inserted before the extension
|
||||||
|
*/
|
||||||
|
public static String insertHashInPath(String path, String hash) {
|
||||||
|
if (path == null || path.isEmpty() || hash == null) {
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
int lastDotIndex = path.lastIndexOf('.');
|
||||||
|
|
||||||
|
// If there's no extension, just append the hash
|
||||||
|
if (lastDotIndex == -1) {
|
||||||
|
return path + "." + hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Insert the hash before the extension
|
||||||
|
String basePath = path.substring(0, lastDotIndex);
|
||||||
|
String extension = path.substring(lastDotIndex + 1);
|
||||||
|
|
||||||
|
return basePath + "." + hash + "." + extension;
|
||||||
|
}
|
||||||
|
}
|
|
@ -126,7 +126,7 @@ public record Post(
|
||||||
}
|
}
|
||||||
} catch (GitAPIException e) {
|
} catch (GitAPIException e) {
|
||||||
draft = true;
|
draft = true;
|
||||||
LOGGER.warn("[Post {}] Draft because of a Git exception: {}\n", slug, e.getMessage());
|
LOGGER.warn("[Post {}] 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 Post(slug, title, summary, draft, revisions, createdBy, createdAt, modifiedBy, modifiedAt, custom, content);
|
||||||
|
|
|
@ -13,10 +13,11 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
public record RenderOptions(
|
public record RenderOptions(
|
||||||
List<String> compress // TODO rename?
|
List<String> compress, // TODO rename?
|
||||||
) {
|
|
||||||
private static final Logger LOGGER = LoggerFactory.getLogger(RenderOptions.class);
|
|
||||||
|
|
||||||
|
boolean remapTemplateStatic,
|
||||||
|
boolean remapAssets
|
||||||
|
) {
|
||||||
/**
|
/**
|
||||||
* Creates a {@link Site} object by reading and parsing the configuration file at the specified path.<br>
|
* Creates a {@link Site} object by reading and parsing the configuration file at the specified path.<br>
|
||||||
* The configuration file must be a JSON file.
|
* The configuration file must be a JSON file.
|
||||||
|
@ -29,22 +30,13 @@ public record RenderOptions(
|
||||||
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));
|
||||||
|
|
||||||
List<String> compress = new ArrayList<>();
|
List<String> compress = (List<String>) yaml.getOrDefault("compress", new ArrayList<>());
|
||||||
|
boolean remapTemplateStatic = (boolean) yaml.getOrDefault("remapTemplateStatic", true);
|
||||||
for (var key : yaml.keySet()) {
|
// assets are not remapped by default, because they might be hotlinked
|
||||||
var value = yaml.get(key);
|
boolean remapAssets = (boolean) yaml.getOrDefault("remapAssets", false);
|
||||||
|
|
||||||
switch (key) {
|
|
||||||
case "compress":
|
|
||||||
compress = (List<String>) value;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
LOGGER.warn("Ignoring unrecognized render option: {}", key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return new RenderOptions(
|
return new RenderOptions(
|
||||||
compress
|
compress, remapTemplateStatic, remapAssets
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
package eu.m724.blog.template;
|
package eu.m724.blog.template;
|
||||||
|
|
||||||
|
import eu.m724.blog.StaticCacheRemapper;
|
||||||
import eu.m724.blog.data.Site;
|
import eu.m724.blog.data.Site;
|
||||||
import io.pebbletemplates.pebble.extension.AbstractExtension;
|
import io.pebbletemplates.pebble.extension.AbstractExtension;
|
||||||
import io.pebbletemplates.pebble.extension.Function;
|
import io.pebbletemplates.pebble.extension.Function;
|
||||||
|
@ -11,9 +12,11 @@ import java.util.Map;
|
||||||
|
|
||||||
public class TemplateExtension extends AbstractExtension {
|
public class TemplateExtension extends AbstractExtension {
|
||||||
private final Site site;
|
private final Site site;
|
||||||
|
private final Map<String, String> fileHashes;
|
||||||
|
|
||||||
public TemplateExtension(Site site) {
|
public TemplateExtension(Site site, Map<String, String> fileHashes) {
|
||||||
this.site = site;
|
this.site = site;
|
||||||
|
this.fileHashes = fileHashes;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -27,7 +30,14 @@ public class TemplateExtension extends AbstractExtension {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object execute(Map<String, Object> args, PebbleTemplate self, EvaluationContext context, int lineNumber) {
|
public Object execute(Map<String, Object> args, PebbleTemplate self, EvaluationContext context, int lineNumber) {
|
||||||
return site.directory() + "/static/" + args.get("path");
|
var path = "static/" + args.get("path");
|
||||||
|
var hash = fileHashes.get(path);
|
||||||
|
|
||||||
|
if (hash != null) {
|
||||||
|
path = StaticCacheRemapper.insertHashInPath(path, hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
return site.directory() + "/" + path;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"asset", new Function() {
|
"asset", new Function() {
|
||||||
|
@ -38,7 +48,14 @@ public class TemplateExtension extends AbstractExtension {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object execute(Map<String, Object> args, PebbleTemplate self, EvaluationContext context, int lineNumber) {
|
public Object execute(Map<String, Object> args, PebbleTemplate self, EvaluationContext context, int lineNumber) {
|
||||||
return site.directory() + "/assets/" + args.get("path");
|
var path = "assets/" + args.get("path");
|
||||||
|
var hash = fileHashes.get(path);
|
||||||
|
|
||||||
|
if (hash != null) {
|
||||||
|
path = StaticCacheRemapper.insertHashInPath(path, hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
return site.directory() + "/" + path;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// TODO make url_for that supports relative and absolute paths
|
// TODO make url_for that supports relative and absolute paths
|
||||||
|
|
|
@ -24,16 +24,17 @@ public class TemplateRenderer {
|
||||||
* Constructs a TemplateRenderer instance for rendering templates from the specified directory.
|
* Constructs a TemplateRenderer instance for rendering templates from the specified directory.
|
||||||
|
|
||||||
* @param site the {@link Site} this renderer renders
|
* @param site the {@link Site} this renderer renders
|
||||||
* @param templateDirectory the root directory containing the template files
|
* @param templateDirectory the root directory containing the template file
|
||||||
|
* @param fileHashes file hashes. currently only applies to assets
|
||||||
*/
|
*/
|
||||||
public TemplateRenderer(Site site, Path templateDirectory) {
|
public TemplateRenderer(Site site, Path templateDirectory, Map<String, String> fileHashes) {
|
||||||
var loader = new FileLoader();
|
var loader = new FileLoader();
|
||||||
loader.setPrefix(templateDirectory.toString());
|
loader.setPrefix(templateDirectory.toString());
|
||||||
loader.setSuffix(".html");
|
loader.setSuffix(".html");
|
||||||
|
|
||||||
var pebbleEngine = new PebbleEngine.Builder()
|
var pebbleEngine = new PebbleEngine.Builder()
|
||||||
.loader(loader)
|
.loader(loader)
|
||||||
.extension(new TemplateExtension(site))
|
.extension(new TemplateExtension(site, fileHashes))
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
this.site = site;
|
this.site = site;
|
||||||
|
|
Loading…
Add table
Reference in a new issue