I discovered a strange behavior when I want to use a Font, that is included in the springboot fat jar. When running tests on my local machine, which loads a Font from the /resources
directory, it works perfectly. But if I build an app with maven and run it from the terminal, then I will receive:
java.io.IOException: Problem reading font data.
at java.awt.Font.createFont0(Font.java:1000)
at java.awt.Font.createFont(Font.java:877)
I tried to find a solution and did the following:
- using different Fonts (sometimes with another error about a damaged table or sth.)
- built various versions of docker openjdk containers to test behavior
- build a ubuntu container, installed openjdk
- used a debian image, installed fc-cache and made the Font available via
/usr/share/fonts/truetype/europlate
fc-cache -f -v
- created a temp-File, just to be sure, that there is no access problem from within the jar.
- ran the application from my terminal, where loading the font fails, too
- using the font works on Editors, and even when the app runs from the IDE (no test)
The method:
public Font getFont() throws IOException, FontFormatException {
File f = File.createTempFile("dang", "tmp");
assert f != null;
f.delete();
ClassLoader classLoader = getClass().getClassLoader();
Font font = Font.createFont(Font.TRUETYPE_FONT, classLoader.getSystemResourceAsStream("EuroPlate.ttf"));
font.deriveFont(105f);
System.out.println(font.getFontName());
return font;
}
EDIT: IDE runs application -> works
Terminal runs application -> fails
Stacktrace:
java.io.IOException: Problem reading font data.
at java.desktop/java.awt.Font.createFont0(Font.java:1183)
at java.desktop/java.awt.Font.createFont(Font.java:1052)
at components.NumberPlateUtility.getFont(NumberPlateUtility.java:81)
at components.NumberPlateUtility.completeImage(NumberPlateUtility.java:173)
at main.NumberplateClientCommands.one(NumberplateClientCommands.java:63)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:566)
at org.springframework.util.ReflectionUtils.invokeMethod(ReflectionUtils.java:223)
at org.springframework.shell.Shell.evaluate(Shell.java:169)
at org.springframework.shell.Shell.run(Shell.java:134)
at org.springframework.shell.jline.InteractiveShellApplicationRunner.run(InteractiveShellApplicationRunner.java:84)
at org.springframework.boot.SpringApplication.callRunner(SpringApplication.java:783)
at org.springframework.boot.SpringApplication.callRunners(SpringApplication.java:773)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:315)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1242)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1230)
at main.Main.main(Main.java:20)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:566)
at org.springframework.boot.loader.MainMethodRunner.run(MainMethodRunner.java:48)
at org.springframework.boot.loader.Launcher.launch(Launcher.java:87)
at org.springframework.boot.loader.Launcher.launch(Launcher.java:50)
at org.springframework.boot.loader.JarLauncher.main(JarLauncher.java:51)
The repository: https://github.com/Semo/numberplate_generator.git