Skip to content

Commit 2a8602d

Browse files
Adds support for local theme files to be specified via theme (#331).
1 parent 2ed0cca commit 2a8602d

File tree

11 files changed

+262
-58
lines changed

11 files changed

+262
-58
lines changed

changelog.md

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
- structurizr-dsl: Adds an `!elements` keyword that can be used to find a set of elements via an expression.
1212
- structurizr-dsl: Adds a `!relationships` keyword that can be used to find a set of relationships via an expression.
1313
- structurizr-dsl: Adds a DSL wrapper around the `structurizr-component` component finder.
14+
- structurizr-dsl: Adds support for local theme files to be specified via `theme` (https://github.com/structurizr/java/issues/331).
1415

1516
## 2.2.0 (2nd July 2024)
1617

structurizr-client/src/main/java/com/structurizr/view/ThemeUtils.java

Lines changed: 73 additions & 29 deletions
Original file line numberDiff line numberDiff line change
@@ -6,17 +6,22 @@
66
import com.fasterxml.jackson.databind.SerializationFeature;
77
import com.structurizr.Workspace;
88
import com.structurizr.io.WorkspaceWriterException;
9+
import com.structurizr.model.Relationship;
10+
import com.structurizr.util.ImageUtils;
911
import com.structurizr.util.StringUtils;
12+
import com.structurizr.util.Url;
1013
import org.apache.hc.client5.http.classic.methods.HttpGet;
1114
import org.apache.hc.client5.http.config.ConnectionConfig;
1215
import org.apache.hc.client5.http.impl.classic.CloseableHttpClient;
1316
import org.apache.hc.client5.http.impl.classic.CloseableHttpResponse;
1417
import org.apache.hc.client5.http.impl.classic.HttpClientBuilder;
18+
import org.apache.hc.client5.http.impl.classic.HttpClients;
1519
import org.apache.hc.client5.http.impl.io.BasicHttpClientConnectionManager;
1620
import org.apache.hc.core5.http.io.entity.EntityUtils;
1721

1822
import java.io.*;
1923
import java.nio.charset.StandardCharsets;
24+
import java.nio.file.Files;
2025
import java.util.concurrent.TimeUnit;
2126

2227
/**
@@ -65,7 +70,7 @@ public static String toJson(Workspace workspace) throws Exception {
6570
}
6671

6772
/**
68-
* Loads (and inlines) the element and relationship styles from the themes defined in the workspace, into the workspace itself.
73+
* Loads the element and relationship styles from the themes defined in the workspace, into the workspace itself.
6974
* This implementation simply copies the styles from all themes into the workspace.
7075
* This uses a default timeout value of 10000ms.
7176
*
@@ -77,40 +82,19 @@ public static void loadThemes(Workspace workspace) throws Exception {
7782
}
7883

7984
/**
80-
* Loads (and inlines) the element and relationship styles from the themes defined in the workspace, into the workspace itself.
85+
* Loads the element and relationship styles from the themes defined in the workspace, into the workspace itself.
8186
* This implementation simply copies the styles from all themes into the workspace.
8287
*
8388
* @param workspace a Workspace object
8489
* @param timeoutInMilliseconds the timeout in milliseconds
8590
* @throws Exception if something goes wrong
8691
*/
8792
public static void loadThemes(Workspace workspace, int timeoutInMilliseconds) throws Exception {
88-
for (String url : workspace.getViews().getConfiguration().getThemes()) {
89-
ConnectionConfig connectionConfig = ConnectionConfig.custom()
90-
.setConnectTimeout(timeoutInMilliseconds, TimeUnit.MILLISECONDS)
91-
.setSocketTimeout(timeoutInMilliseconds, TimeUnit.MILLISECONDS)
92-
.build();
93-
94-
BasicHttpClientConnectionManager cm = new BasicHttpClientConnectionManager();
95-
cm.setConnectionConfig(connectionConfig);
96-
97-
CloseableHttpClient httpClient = HttpClientBuilder.create()
98-
.useSystemProperties()
99-
.setConnectionManager(cm)
100-
.build();
101-
102-
HttpGet httpGet = new HttpGet(url);
103-
104-
CloseableHttpResponse response = httpClient.execute(httpGet);
105-
if (response.getCode() == HTTP_OK_STATUS) {
106-
String json = EntityUtils.toString(response.getEntity());
107-
108-
ObjectMapper objectMapper = new ObjectMapper();
109-
objectMapper.enable(DeserializationFeature.ACCEPT_EMPTY_STRING_AS_NULL_OBJECT);
110-
objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
111-
112-
Theme theme = objectMapper.readValue(json, Theme.class);
113-
String baseUrl = url.substring(0, url.lastIndexOf('/') + 1);
93+
for (String themeLocation : workspace.getViews().getConfiguration().getThemes()) {
94+
if (Url.isUrl(themeLocation)) {
95+
String json = loadFrom(themeLocation, timeoutInMilliseconds);
96+
Theme theme = fromJson(json);
97+
String baseUrl = themeLocation.substring(0, themeLocation.lastIndexOf('/') + 1);
11498

11599
for (ElementStyle elementStyle : theme.getElements()) {
116100
String icon = elementStyle.getIcon();
@@ -128,9 +112,69 @@ public static void loadThemes(Workspace workspace, int timeoutInMilliseconds) th
128112

129113
workspace.getViews().getConfiguration().getStyles().addStylesFromTheme(theme);
130114
}
115+
}
116+
}
131117

132-
httpClient.close();
118+
/**
119+
* Inlines the element and relationship styles from the specified file, adding the styles into the workspace
120+
* and overriding any properties already set.
121+
*
122+
* @param workspace the Workspace to load the theme into
123+
* @param file a File object representing a theme (a JSON file)
124+
* @throws Exception if something goes wrong
125+
*/
126+
public static void inlineTheme(Workspace workspace, File file) throws Exception {
127+
String json = Files.readString(file.toPath());
128+
Theme theme = fromJson(json);
129+
130+
for (ElementStyle elementStyle : theme.getElements()) {
131+
String icon = elementStyle.getIcon();
132+
if (!StringUtils.isNullOrEmpty(icon)) {
133+
if (icon.startsWith("http")) {
134+
// okay, image served over HTTP
135+
} else if (icon.startsWith("data:image")) {
136+
// also okay, data URI
137+
} else {
138+
// convert the relative icon filename into a data URI
139+
elementStyle.setIcon(ImageUtils.getImageAsDataUri(new File(file.getParentFile(), icon)));
140+
}
141+
}
133142
}
143+
144+
workspace.getViews().getConfiguration().getStyles().inlineTheme(theme);
145+
}
146+
147+
private static String loadFrom(String url, int timeoutInMilliseconds) throws Exception {
148+
ConnectionConfig connectionConfig = ConnectionConfig.custom()
149+
.setConnectTimeout(timeoutInMilliseconds, TimeUnit.MILLISECONDS)
150+
.setSocketTimeout(timeoutInMilliseconds, TimeUnit.MILLISECONDS)
151+
.build();
152+
153+
BasicHttpClientConnectionManager cm = new BasicHttpClientConnectionManager();
154+
cm.setConnectionConfig(connectionConfig);
155+
156+
try (CloseableHttpClient httpClient = HttpClientBuilder.create()
157+
.useSystemProperties()
158+
.setConnectionManager(cm)
159+
.build()) {
160+
161+
HttpGet httpGet = new HttpGet(url);
162+
163+
CloseableHttpResponse response = httpClient.execute(httpGet);
164+
if (response.getCode() == HTTP_OK_STATUS) {
165+
return EntityUtils.toString(response.getEntity());
166+
}
167+
}
168+
169+
return "";
170+
}
171+
172+
private static Theme fromJson(String json) throws Exception {
173+
ObjectMapper objectMapper = new ObjectMapper();
174+
objectMapper.enable(DeserializationFeature.ACCEPT_EMPTY_STRING_AS_NULL_OBJECT);
175+
objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
176+
177+
return objectMapper.readValue(json, Theme.class);
134178
}
135179

136180
private static void write(Workspace workspace, Writer writer) throws Exception {
Lines changed: 35 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
package com.structurizr.util;
2+
3+
import com.structurizr.Workspace;
4+
import com.structurizr.view.ThemeUtils;
5+
import org.junit.jupiter.api.Test;
6+
7+
import java.io.File;
8+
9+
import static org.junit.jupiter.api.Assertions.assertEquals;
10+
11+
public class ThemeUtilsTests {
12+
13+
@Test
14+
void inlineTheme() throws Exception {
15+
File themeFile = new File("src/test/resources/theme.json");
16+
17+
try {
18+
Workspace theme = new Workspace("Theme", "");
19+
theme.getViews().getConfiguration().getStyles().addElementStyle("Tag").background("#ff0000").icon("logo.png");
20+
theme.getViews().getConfiguration().getStyles().addRelationshipStyle("Tag").color("#00ff00");
21+
ThemeUtils.toJson(theme, themeFile);
22+
} catch (Exception e) {
23+
throw new RuntimeException(e);
24+
}
25+
26+
Workspace workspace = new Workspace("Name", "Description");
27+
ThemeUtils.inlineTheme(workspace, themeFile);
28+
29+
assertEquals(0, workspace.getViews().getConfiguration().getThemes().length);
30+
assertEquals("#ff0000", workspace.getViews().getConfiguration().getStyles().getElementStyle("Tag").getBackground());
31+
assertEquals("", workspace.getViews().getConfiguration().getStyles().getElementStyle("Tag").getIcon());
32+
assertEquals("#00ff00", workspace.getViews().getConfiguration().getStyles().getRelationshipStyle("Tag").getColor());
33+
}
34+
35+
}
9.04 KB
Loading
Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
{
2+
"name" : "Theme",
3+
"elements" : [ {
4+
"tag" : "Tag",
5+
"background" : "#ff0000",
6+
"icon" : "logo.png"
7+
} ],
8+
"relationships" : [ {
9+
"tag" : "Tag",
10+
"color" : "#00ff00"
11+
} ]
12+
}

structurizr-core/src/main/java/com/structurizr/view/Styles.java

Lines changed: 26 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -309,4 +309,30 @@ public void addStylesFromTheme(Theme theme) {
309309
}
310310
}
311311

312+
/**
313+
* Inlines the element and relationship styles from the specified theme, adding the styles into the workspace
314+
* and overriding any properties already set.
315+
*
316+
* @param theme a Theme object
317+
*/
318+
public void inlineTheme(Theme theme) {
319+
for (ElementStyle elementStyle : theme.getElements()) {
320+
ElementStyle es = getElementStyle(elementStyle.getTag());
321+
if (es == null) {
322+
es = addElementStyle(elementStyle.getTag());
323+
}
324+
325+
es.copyFrom(elementStyle);
326+
}
327+
328+
for (RelationshipStyle relationshipStyle : theme.getRelationships()) {
329+
RelationshipStyle rs = getRelationshipStyle(relationshipStyle.getTag());
330+
if (rs == null) {
331+
rs = addRelationshipStyle(relationshipStyle.getTag());
332+
}
333+
334+
rs.copyFrom(relationshipStyle);
335+
}
336+
}
337+
312338
}

0 commit comments

Comments
 (0)