mirror of
https://github.com/scm-manager/scm-manager.git
synced 2026-02-08 07:39:15 +01:00
Invalidation of caches and search index
In the general admin settings, the user can find two button to either invalidate the cache or rebuild the search index. The endpoints are defined in the InvalidationResource class in scm-webapp. Co-authored-by: René Pfeuffer<rene.pfeuffer@cloudogu.com>
This commit is contained in:
@@ -135,6 +135,10 @@ public class IndexDtoGenerator extends HalAppenderMapper {
|
||||
}
|
||||
if (ConfigurationPermissions.list().isPermitted()) {
|
||||
builder.single(link("config", resourceLinks.config().self()));
|
||||
if (ConfigurationPermissions.write(configuration.getId()).isPermitted()) {
|
||||
builder.single(link("invalidateCaches", resourceLinks.invalidationLinks().caches()));
|
||||
builder.single(link("invalidateSearchIndex", resourceLinks.invalidationLinks().searchIndex()));
|
||||
}
|
||||
if (!Strings.isNullOrEmpty(configuration.getReleaseFeedUrl())) {
|
||||
builder.single(link("updateInfo", resourceLinks.adminInfo().updateInfo()));
|
||||
}
|
||||
|
||||
@@ -0,0 +1,102 @@
|
||||
/*
|
||||
* MIT License
|
||||
*
|
||||
* Copyright (c) 2020-present Cloudogu GmbH and Contributors
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*/
|
||||
|
||||
package sonia.scm.api.v2.resources;
|
||||
|
||||
import io.swagger.v3.oas.annotations.OpenAPIDefinition;
|
||||
import io.swagger.v3.oas.annotations.Operation;
|
||||
import io.swagger.v3.oas.annotations.media.Content;
|
||||
import io.swagger.v3.oas.annotations.media.Schema;
|
||||
import io.swagger.v3.oas.annotations.responses.ApiResponse;
|
||||
import io.swagger.v3.oas.annotations.tags.Tag;
|
||||
import sonia.scm.cache.CacheManager;
|
||||
import sonia.scm.config.ConfigurationPermissions;
|
||||
import sonia.scm.search.IndexRebuilder;
|
||||
import sonia.scm.web.VndMediaType;
|
||||
|
||||
import javax.inject.Inject;
|
||||
import javax.ws.rs.POST;
|
||||
import javax.ws.rs.Path;
|
||||
|
||||
@OpenAPIDefinition(tags = {
|
||||
@Tag(name = "Invalidations", description = "Invalidations of different resources like caches and search index")
|
||||
})
|
||||
@Path("v2/invalidations")
|
||||
public class InvalidationResource {
|
||||
|
||||
private final CacheManager cacheManager;
|
||||
private final IndexRebuilder indexRebuilder;
|
||||
|
||||
@Inject
|
||||
public InvalidationResource(CacheManager cacheManager, IndexRebuilder indexRebuilder) {
|
||||
this.cacheManager = cacheManager;
|
||||
this.indexRebuilder = indexRebuilder;
|
||||
}
|
||||
|
||||
@POST
|
||||
@Path("/caches")
|
||||
@Operation(
|
||||
summary = "Invalidates the caches of every store",
|
||||
description = "Deletes every cached object of every store from the cache",
|
||||
tags = "Invalidations"
|
||||
)
|
||||
@ApiResponse(responseCode = "204", description = "Invalidated cache successfully")
|
||||
@ApiResponse(responseCode = "401", description = "not authenticated / invalid credentials")
|
||||
@ApiResponse(responseCode = "403", description = "not authorized, the current user does not have the \"configuration:write:global\" privilege")
|
||||
@ApiResponse(
|
||||
responseCode = "500",
|
||||
description = "internal server error",
|
||||
content = @Content(
|
||||
mediaType = VndMediaType.ERROR_TYPE,
|
||||
schema = @Schema(implementation = ErrorDto.class)
|
||||
)
|
||||
)
|
||||
public void invalidateCaches() {
|
||||
ConfigurationPermissions.write("global").check();
|
||||
cacheManager.clearAllCaches();
|
||||
}
|
||||
|
||||
@POST
|
||||
@Path("/search-index")
|
||||
@Operation(
|
||||
summary = "Invalidates the search index",
|
||||
description = "Invalidates the search index, by completely recreating it",
|
||||
tags = "Invalidations"
|
||||
)
|
||||
@ApiResponse(responseCode = "204", description = "Invalidated search index successfully")
|
||||
@ApiResponse(responseCode = "401", description = "not authenticated / invalid credentials")
|
||||
@ApiResponse(responseCode = "403", description = "not authorized, the current user does not have the \"configuration:write:global\" privilege")
|
||||
@ApiResponse(
|
||||
responseCode = "500",
|
||||
description = "internal server error",
|
||||
content = @Content(
|
||||
mediaType = VndMediaType.ERROR_TYPE,
|
||||
schema = @Schema(implementation = ErrorDto.class)
|
||||
)
|
||||
)
|
||||
public void invalidateSearchIndex() {
|
||||
ConfigurationPermissions.write("global").check();
|
||||
indexRebuilder.rebuildAll();
|
||||
}
|
||||
}
|
||||
@@ -347,6 +347,27 @@ class ResourceLinks {
|
||||
}
|
||||
}
|
||||
|
||||
InvalidationLinks invalidationLinks() {
|
||||
return new InvalidationLinks(accessScmPathInfoStore().get());
|
||||
}
|
||||
|
||||
static class InvalidationLinks {
|
||||
private final LinkBuilder invalidationLinkBuilder;
|
||||
|
||||
InvalidationLinks(ScmPathInfo pathInfo) {
|
||||
this.invalidationLinkBuilder = new LinkBuilder(pathInfo, InvalidationResource.class);
|
||||
}
|
||||
|
||||
String caches() {
|
||||
return invalidationLinkBuilder.method("invalidateCaches").parameters().href();
|
||||
}
|
||||
|
||||
|
||||
String searchIndex() {
|
||||
return invalidationLinkBuilder.method("invalidateSearchIndex").parameters().href();
|
||||
}
|
||||
}
|
||||
|
||||
AdminInfoLinks adminInfo() {
|
||||
return new AdminInfoLinks(accessScmPathInfoStore().get());
|
||||
}
|
||||
|
||||
@@ -79,6 +79,13 @@ public class GuavaCacheManager implements CacheManager, org.apache.shiro.cache.C
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
public void clearAllCaches() {
|
||||
for(GuavaCache<?, ?> cache : caches.values()) {
|
||||
cache.clear();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void close() throws IOException {
|
||||
LOG.info("close guava cache manager");
|
||||
|
||||
@@ -0,0 +1,46 @@
|
||||
/*
|
||||
* MIT License
|
||||
*
|
||||
* Copyright (c) 2020-present Cloudogu GmbH and Contributors
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*/
|
||||
|
||||
package sonia.scm.search;
|
||||
|
||||
import javax.inject.Inject;
|
||||
import java.util.Set;
|
||||
|
||||
public class IndexRebuilder {
|
||||
|
||||
private final SearchEngine searchEngine;
|
||||
private final Set<Indexer> indexers;
|
||||
|
||||
@Inject
|
||||
public IndexRebuilder(SearchEngine searchEngine, Set<Indexer> indexers) {
|
||||
this.searchEngine = searchEngine;
|
||||
this.indexers = indexers;
|
||||
}
|
||||
|
||||
public void rebuildAll() {
|
||||
for (Indexer indexer : indexers) {
|
||||
searchEngine.forType(indexer.getType()).update(indexer.getReIndexAllTask());
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user