From 4bcb9954a46aac13b81223850e48b02e458f42a6 Mon Sep 17 00:00:00 2001 From: Geoff Bourne Date: Tue, 15 Oct 2024 18:29:59 -0500 Subject: [PATCH] cf: gracefully handle corrupted API cache index (#483) --- .../me/itzg/helpers/cache/ApiCachingImpl.java | 40 ++++++++++++++++++- 1 file changed, 38 insertions(+), 2 deletions(-) diff --git a/src/main/java/me/itzg/helpers/cache/ApiCachingImpl.java b/src/main/java/me/itzg/helpers/cache/ApiCachingImpl.java index d3674d02..ae991111 100644 --- a/src/main/java/me/itzg/helpers/cache/ApiCachingImpl.java +++ b/src/main/java/me/itzg/helpers/cache/ApiCachingImpl.java @@ -2,8 +2,11 @@ import com.fasterxml.jackson.databind.ObjectMapper; import java.io.IOException; +import java.nio.file.FileVisitResult; import java.nio.file.Files; import java.nio.file.Path; +import java.nio.file.SimpleFileVisitor; +import java.nio.file.attribute.BasicFileAttributes; import java.time.Duration; import java.time.Instant; import java.time.temporal.TemporalAmount; @@ -74,17 +77,50 @@ private Path resolveContentFile(String operation, String filename) { return cacheNamespaceDir.resolve(operation).resolve(filename); } - private CacheIndex loadCacheIndex() throws IOException { + private CacheIndex loadCacheIndex() { final Path cacheIndexPath = cacheNamespaceDir.resolve(CACHE_INDEX_FILENAME); if (Files.exists(cacheIndexPath)) { log.debug("Loading cache index from {}", cacheIndexPath); - return objectMapper.readValue(cacheIndexPath.toFile(), CacheIndex.class); + try { + return objectMapper.readValue(cacheIndexPath.toFile(), CacheIndex.class); + } catch (IOException e) { + log.warn("Failed to load API cache index from {}", cacheIndexPath, e); + wipeCacheDirectory(); + return new CacheIndex(); + } } else { return new CacheIndex(); } } + private void wipeCacheDirectory() { + if (!Files.exists(cacheNamespaceDir)) { + log.debug("Skipping wipe of non-existent cache directory {}", cacheNamespaceDir); + return; + } + + log.debug("Wiping cache directory {}", cacheNamespaceDir); + try { + Files.walkFileTree(cacheNamespaceDir, new SimpleFileVisitor() { + @Override + public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) { + if (file.getFileName().toString().endsWith(".json")) { + try { + log.debug("Wiping cache file {}", file); + Files.delete(file); + } catch (IOException e) { + log.warn("Failed to delete cache file {}", file, e); + } + } + return FileVisitResult.CONTINUE; + } + }); + } catch (IOException e) { + log.warn("Unexpected failure while wiping cache directory", e); + } + } + @Override public Mono cache(String operation, Class returnType, Mono resolver, Object... keys) { final String keysKey = Stream.of(keys)