Skip to content

Commit 6e83d3f

Browse files
authored
"Cacheability" option for critical REST resources (#3335)
* Closes #3329. This implements a new optional `cacheable` parameter for these REST endpoints: - `/rest/items` - `/rest/things` - `/rest/rules` When this parameter is set, a flat list of all elements excluding non-cacheable fields (e.g. "state", "transformedState", "stateDescription", "commandDescription" for items, "statusInfo", "firmwareStatus", "properties" for things, "status" for rules) will be retrieved along with a `Last-Modified` HTTP response header. When unknown, the Last-Modified header will be set to the date of the request. Also only when this parameter is set, and a `If-Modified-Since` header is found in the request, that header will be compared to the last known modified date for the corresponding cacheable list. The last modified date will be reset when any change is made on the elements of the underlying registry. If the `If-Modified-Since` date is equal or more recent than the last modified date, then a 304 Not Modified response with no content will be served instead of the usual 200 OK, informing the client that its cache is still valid at the provided date. All other request parameters will be ignored except for "metadata" in the `/rest/items` endpoint. When a metadata selector is set, the resulting item list will be considered like a completely different resource, i.e. it will have its own last modified date. Regarding metadata, the approach to invalidating last modified dates is very conservative: when any metadata is changed, all cacheable lists of items will have their last modified date reset even if the change was in a metadata namespace that wasn't requested. This also implements the abovedescribed behavior for the `/rest/ui/components/{namespace}` endpoint, but no `cacheable` parameter is necessary. The last modified date is tracked by namespace. Signed-off-by: Yannick Schaus <[email protected]>
1 parent 885a854 commit 6e83d3f

File tree

5 files changed

+322
-23
lines changed

5 files changed

+322
-23
lines changed

bundles/org.openhab.core.automation.rest/src/main/java/org/openhab/core/automation/rest/internal/RuleResource.java

+66-3
Original file line numberDiff line numberDiff line change
@@ -17,9 +17,11 @@
1717
import java.io.IOException;
1818
import java.net.URLEncoder;
1919
import java.nio.charset.StandardCharsets;
20+
import java.time.Instant;
2021
import java.time.ZonedDateTime;
2122
import java.time.temporal.ChronoUnit;
2223
import java.util.Collection;
24+
import java.util.Date;
2325
import java.util.List;
2426
import java.util.Map;
2527
import java.util.function.Predicate;
@@ -29,15 +31,18 @@
2931
import javax.annotation.security.RolesAllowed;
3032
import javax.ws.rs.Consumes;
3133
import javax.ws.rs.DELETE;
34+
import javax.ws.rs.DefaultValue;
3235
import javax.ws.rs.GET;
3336
import javax.ws.rs.POST;
3437
import javax.ws.rs.PUT;
3538
import javax.ws.rs.Path;
3639
import javax.ws.rs.PathParam;
3740
import javax.ws.rs.Produces;
3841
import javax.ws.rs.QueryParam;
42+
import javax.ws.rs.core.CacheControl;
3943
import javax.ws.rs.core.Context;
4044
import javax.ws.rs.core.MediaType;
45+
import javax.ws.rs.core.Request;
4146
import javax.ws.rs.core.Response;
4247
import javax.ws.rs.core.Response.Status;
4348
import javax.ws.rs.core.SecurityContext;
@@ -69,6 +74,7 @@
6974
import org.openhab.core.automation.rest.internal.dto.EnrichedRuleDTOMapper;
7075
import org.openhab.core.automation.util.ModuleBuilder;
7176
import org.openhab.core.automation.util.RuleBuilder;
77+
import org.openhab.core.common.registry.RegistryChangeListener;
7278
import org.openhab.core.config.core.ConfigUtil;
7379
import org.openhab.core.config.core.Configuration;
7480
import org.openhab.core.events.Event;
@@ -80,6 +86,7 @@
8086
import org.openhab.core.library.types.DateTimeType;
8187
import org.osgi.service.component.annotations.Activate;
8288
import org.osgi.service.component.annotations.Component;
89+
import org.osgi.service.component.annotations.Deactivate;
8390
import org.osgi.service.component.annotations.Reference;
8491
import org.osgi.service.jaxrs.whiteboard.JaxrsWhiteboardConstants;
8592
import org.osgi.service.jaxrs.whiteboard.propertytypes.JSONRequired;
@@ -128,8 +135,10 @@ public class RuleResource implements RESTResource {
128135
private final RuleManager ruleManager;
129136
private final RuleRegistry ruleRegistry;
130137
private final ManagedRuleProvider managedRuleProvider;
138+
private final ResetLastModifiedChangeListener resetLastModifiedChangeListener = new ResetLastModifiedChangeListener();
131139

132140
private @Context @NonNullByDefault({}) UriInfo uriInfo;
141+
private @Nullable Date cacheableListLastModified = null;
133142

134143
@Activate
135144
public RuleResource( //
@@ -141,20 +150,52 @@ public RuleResource( //
141150
this.ruleManager = ruleManager;
142151
this.ruleRegistry = ruleRegistry;
143152
this.managedRuleProvider = managedRuleProvider;
153+
154+
this.ruleRegistry.addRegistryChangeListener(resetLastModifiedChangeListener);
155+
}
156+
157+
@Deactivate
158+
void deactivate() {
159+
this.ruleRegistry.removeRegistryChangeListener(resetLastModifiedChangeListener);
144160
}
145161

146162
@GET
147163
@RolesAllowed({ Role.USER, Role.ADMIN })
148164
@Produces(MediaType.APPLICATION_JSON)
149165
@Operation(operationId = "getRules", summary = "Get available rules, optionally filtered by tags and/or prefix.", responses = {
150166
@ApiResponse(responseCode = "200", description = "OK", content = @Content(array = @ArraySchema(schema = @Schema(implementation = EnrichedRuleDTO.class)))) })
151-
public Response get(@Context SecurityContext securityContext, @QueryParam("prefix") final @Nullable String prefix,
152-
@QueryParam("tags") final @Nullable List<String> tags,
153-
@QueryParam("summary") @Parameter(description = "summary fields only") @Nullable Boolean summary) {
167+
public Response get(@Context SecurityContext securityContext, @Context Request request,
168+
@QueryParam("prefix") final @Nullable String prefix, @QueryParam("tags") final @Nullable List<String> tags,
169+
@QueryParam("summary") @Parameter(description = "summary fields only") @Nullable Boolean summary,
170+
@DefaultValue("false") @QueryParam("staticDataOnly") @Parameter(description = "provides a cacheable list of values not expected to change regularly and honors the If-Modified-Since header, all other parameters are ignored") boolean staticDataOnly) {
171+
154172
if ((summary == null || !summary) && !securityContext.isUserInRole(Role.ADMIN)) {
155173
// users may only access the summary
156174
return JSONResponse.createErrorResponse(Status.UNAUTHORIZED, "Authentication required");
157175
}
176+
177+
if (staticDataOnly) {
178+
if (cacheableListLastModified != null) {
179+
Response.ResponseBuilder responseBuilder = request.evaluatePreconditions(cacheableListLastModified);
180+
if (responseBuilder != null) {
181+
// send 304 Not Modified
182+
return responseBuilder.build();
183+
}
184+
} else {
185+
cacheableListLastModified = Date.from(Instant.now().truncatedTo(ChronoUnit.SECONDS));
186+
}
187+
188+
Stream<EnrichedRuleDTO> rules = ruleRegistry.stream()
189+
.map(rule -> EnrichedRuleDTOMapper.map(rule, ruleManager, managedRuleProvider));
190+
191+
CacheControl cc = new CacheControl();
192+
cc.setMustRevalidate(true);
193+
cc.setPrivate(true);
194+
rules = dtoMapper.limitToFields(rules, "uid,templateUID,name,visibility,description,tags,editable");
195+
return Response.ok(new Stream2JSONInputStream(rules)).lastModified(cacheableListLastModified)
196+
.cacheControl(cc).build();
197+
}
198+
158199
// match all
159200
Predicate<Rule> p = r -> true;
160201

@@ -567,4 +608,26 @@ public Response setModuleConfigParam(@PathParam("ruleUID") @Parameter(descriptio
567608
return null;
568609
}
569610
}
611+
612+
private void resetStaticListLastModified() {
613+
cacheableListLastModified = null;
614+
}
615+
616+
private class ResetLastModifiedChangeListener implements RegistryChangeListener<Rule> {
617+
618+
@Override
619+
public void added(Rule element) {
620+
resetStaticListLastModified();
621+
}
622+
623+
@Override
624+
public void removed(Rule element) {
625+
resetStaticListLastModified();
626+
}
627+
628+
@Override
629+
public void updated(Rule oldElement, Rule element) {
630+
resetStaticListLastModified();
631+
}
632+
}
570633
}

bundles/org.openhab.core.io.rest.core/src/main/java/org/openhab/core/io/rest/core/internal/item/ItemResource.java

+96-1
Original file line numberDiff line numberDiff line change
@@ -12,9 +12,12 @@
1212
*/
1313
package org.openhab.core.io.rest.core.internal.item;
1414

15+
import java.time.Instant;
16+
import java.time.temporal.ChronoUnit;
1517
import java.util.ArrayList;
1618
import java.util.Arrays;
1719
import java.util.Collection;
20+
import java.util.Date;
1821
import java.util.HashMap;
1922
import java.util.HashSet;
2023
import java.util.List;
@@ -40,9 +43,11 @@
4043
import javax.ws.rs.Produces;
4144
import javax.ws.rs.QueryParam;
4245
import javax.ws.rs.WebApplicationException;
46+
import javax.ws.rs.core.CacheControl;
4347
import javax.ws.rs.core.Context;
4448
import javax.ws.rs.core.HttpHeaders;
4549
import javax.ws.rs.core.MediaType;
50+
import javax.ws.rs.core.Request;
4651
import javax.ws.rs.core.Response;
4752
import javax.ws.rs.core.Response.ResponseBuilder;
4853
import javax.ws.rs.core.Response.Status;
@@ -52,6 +57,7 @@
5257
import org.eclipse.jdt.annotation.NonNullByDefault;
5358
import org.eclipse.jdt.annotation.Nullable;
5459
import org.openhab.core.auth.Role;
60+
import org.openhab.core.common.registry.RegistryChangeListener;
5561
import org.openhab.core.events.EventPublisher;
5662
import org.openhab.core.io.rest.DTOMapper;
5763
import org.openhab.core.io.rest.JSONResponse;
@@ -68,6 +74,7 @@
6874
import org.openhab.core.items.ItemBuilderFactory;
6975
import org.openhab.core.items.ItemNotFoundException;
7076
import org.openhab.core.items.ItemRegistry;
77+
import org.openhab.core.items.ItemRegistryChangeListener;
7178
import org.openhab.core.items.ManagedItemProvider;
7279
import org.openhab.core.items.Metadata;
7380
import org.openhab.core.items.MetadataKey;
@@ -88,6 +95,7 @@
8895
import org.openhab.core.types.TypeParser;
8996
import org.osgi.service.component.annotations.Activate;
9097
import org.osgi.service.component.annotations.Component;
98+
import org.osgi.service.component.annotations.Deactivate;
9199
import org.osgi.service.component.annotations.Reference;
92100
import org.osgi.service.jaxrs.whiteboard.JaxrsWhiteboardConstants;
93101
import org.osgi.service.jaxrs.whiteboard.propertytypes.JSONRequired;
@@ -174,6 +182,10 @@ private static void respectForwarded(final UriBuilder uriBuilder, final @Context
174182
private final ManagedItemProvider managedItemProvider;
175183
private final MetadataRegistry metadataRegistry;
176184
private final MetadataSelectorMatcher metadataSelectorMatcher;
185+
private final ItemRegistryChangeListener resetLastModifiedItemChangeListener = new ResetLastModifiedItemChangeListener();
186+
private final RegistryChangeListener<Metadata> resetLastModifiedMetadataChangeListener = new ResetLastModifiedMetadataChangeListener();
187+
188+
private Map<@Nullable String, Date> cacheableListsLastModified = new HashMap<>();
177189

178190
@Activate
179191
public ItemResource(//
@@ -193,6 +205,15 @@ public ItemResource(//
193205
this.managedItemProvider = managedItemProvider;
194206
this.metadataRegistry = metadataRegistry;
195207
this.metadataSelectorMatcher = metadataSelectorMatcher;
208+
209+
this.itemRegistry.addRegistryChangeListener(resetLastModifiedItemChangeListener);
210+
this.metadataRegistry.addRegistryChangeListener(resetLastModifiedMetadataChangeListener);
211+
}
212+
213+
@Deactivate
214+
void deactivate() {
215+
this.itemRegistry.removeRegistryChangeListener(resetLastModifiedItemChangeListener);
216+
this.metadataRegistry.removeRegistryChangeListener(resetLastModifiedMetadataChangeListener);
196217
}
197218

198219
private UriBuilder uriBuilder(final UriInfo uriInfo, final HttpHeaders httpHeaders) {
@@ -207,17 +228,47 @@ private UriBuilder uriBuilder(final UriInfo uriInfo, final HttpHeaders httpHeade
207228
@Operation(operationId = "getItems", summary = "Get all available items.", responses = {
208229
@ApiResponse(responseCode = "200", description = "OK", content = @Content(array = @ArraySchema(schema = @Schema(implementation = EnrichedItemDTO.class)))) })
209230
public Response getItems(final @Context UriInfo uriInfo, final @Context HttpHeaders httpHeaders,
231+
@Context Request request,
210232
@HeaderParam(HttpHeaders.ACCEPT_LANGUAGE) @Parameter(description = "language") @Nullable String language,
211233
@QueryParam("type") @Parameter(description = "item type filter") @Nullable String type,
212234
@QueryParam("tags") @Parameter(description = "item tag filter") @Nullable String tags,
213235
@DefaultValue(".*") @QueryParam("metadata") @Parameter(description = "metadata selector - a comma separated list or a regular expression (returns all if no value given)") @Nullable String namespaceSelector,
214236
@DefaultValue("false") @QueryParam("recursive") @Parameter(description = "get member items recursively") boolean recursive,
215-
@QueryParam("fields") @Parameter(description = "limit output to the given fields (comma separated)") @Nullable String fields) {
237+
@QueryParam("fields") @Parameter(description = "limit output to the given fields (comma separated)") @Nullable String fields,
238+
@DefaultValue("false") @QueryParam("staticDataOnly") @Parameter(description = "provides a cacheable list of values not expected to change regularly and checks the If-Modified-Since header, all other parameters are ignored except \"metadata\"") boolean staticDataOnly) {
216239
final Locale locale = localeService.getLocale(language);
217240
final Set<String> namespaces = splitAndFilterNamespaces(namespaceSelector, locale);
218241

219242
final UriBuilder uriBuilder = uriBuilder(uriInfo, httpHeaders);
220243

244+
if (staticDataOnly) {
245+
Date lastModifiedDate = Date.from(Instant.now());
246+
if (cacheableListsLastModified.containsKey(namespaceSelector)) {
247+
lastModifiedDate = cacheableListsLastModified.get(namespaceSelector);
248+
Response.ResponseBuilder responseBuilder = request.evaluatePreconditions(lastModifiedDate);
249+
if (responseBuilder != null) {
250+
// send 304 Not Modified
251+
return responseBuilder.build();
252+
}
253+
} else {
254+
lastModifiedDate = Date.from(Instant.now().truncatedTo(ChronoUnit.SECONDS));
255+
cacheableListsLastModified.put(namespaceSelector, lastModifiedDate);
256+
}
257+
258+
Stream<EnrichedItemDTO> itemStream = getItems(null, null).stream() //
259+
.map(item -> EnrichedItemDTOMapper.map(item, false, null, uriBuilder, locale)) //
260+
.peek(dto -> addMetadata(dto, namespaces, null)) //
261+
.peek(dto -> dto.editable = isEditable(dto.name));
262+
itemStream = dtoMapper.limitToFields(itemStream,
263+
"name,label,type,groupType,function,category,editable,groupNames,link,tags,metadata");
264+
265+
CacheControl cc = new CacheControl();
266+
cc.setMustRevalidate(true);
267+
cc.setPrivate(true);
268+
return Response.ok(new Stream2JSONInputStream(itemStream)).lastModified(lastModifiedDate).cacheControl(cc)
269+
.build();
270+
}
271+
221272
Stream<EnrichedItemDTO> itemStream = getItems(type, tags).stream() //
222273
.map(item -> EnrichedItemDTOMapper.map(item, recursive, null, uriBuilder, locale)) //
223274
.peek(dto -> addMetadata(dto, namespaces, null)) //
@@ -935,4 +986,48 @@ private void addMetadata(EnrichedItemDTO dto, Set<String> namespaces, @Nullable
935986
private boolean isEditable(String itemName) {
936987
return managedItemProvider.get(itemName) != null;
937988
}
989+
990+
private void resetCacheableListsLastModified() {
991+
this.cacheableListsLastModified.clear();
992+
}
993+
994+
private class ResetLastModifiedItemChangeListener implements ItemRegistryChangeListener {
995+
@Override
996+
public void added(Item element) {
997+
resetCacheableListsLastModified();
998+
}
999+
1000+
@Override
1001+
public void allItemsChanged(Collection<String> oldItemNames) {
1002+
resetCacheableListsLastModified();
1003+
}
1004+
1005+
@Override
1006+
public void removed(Item element) {
1007+
resetCacheableListsLastModified();
1008+
}
1009+
1010+
@Override
1011+
public void updated(Item oldElement, Item element) {
1012+
resetCacheableListsLastModified();
1013+
}
1014+
}
1015+
1016+
private class ResetLastModifiedMetadataChangeListener implements RegistryChangeListener<Metadata> {
1017+
1018+
@Override
1019+
public void added(Metadata element) {
1020+
resetCacheableListsLastModified();
1021+
}
1022+
1023+
@Override
1024+
public void removed(Metadata element) {
1025+
resetCacheableListsLastModified();
1026+
}
1027+
1028+
@Override
1029+
public void updated(Metadata oldElement, Metadata element) {
1030+
resetCacheableListsLastModified();
1031+
}
1032+
}
9381033
}

0 commit comments

Comments
 (0)