From 0562f6bdfd05326a35e66e16afc84c99d7ed6f67 Mon Sep 17 00:00:00 2001 From: Reinhard Pointner Date: Sat, 12 Mar 2016 10:01:11 +0000 Subject: [PATCH] Optimize TheMovieDB caching and default to using HTTPS instead of HTTP --- source/net/filebot/CachedResource.java | 11 ++--- source/net/filebot/web/TMDbClient.java | 57 ++++++++++++------------ test/net/filebot/web/TMDbClientTest.java | 2 +- 3 files changed, 35 insertions(+), 35 deletions(-) diff --git a/source/net/filebot/CachedResource.java b/source/net/filebot/CachedResource.java index b9f834de..dd76fb5c 100644 --- a/source/net/filebot/CachedResource.java +++ b/source/net/filebot/CachedResource.java @@ -12,6 +12,9 @@ import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.concurrent.Callable; +import java.util.function.BiConsumer; +import java.util.function.Consumer; +import java.util.function.Function; import net.filebot.util.JsonUtilities; import net.filebot.web.WebRequest; @@ -177,13 +180,11 @@ public class CachedResource implements Resource { }; } - public static Fetch fetchIfNoneMatch(Cache etagStorage) { + public static Fetch fetchIfNoneMatch(Function etagRetrieve, BiConsumer etagStore) { return (url, lastModified) -> { // record ETag response header Map> responseHeaders = new HashMap>(); - - String etagKey = url.toString(); - Object etagValue = etagStorage.get(etagKey); + Object etagValue = etagRetrieve.apply(url); try { debug.fine(WebRequest.log(url, lastModified, etagValue)); @@ -197,7 +198,7 @@ public class CachedResource implements Resource { } finally { WebRequest.getETag(responseHeaders).ifPresent(etag -> { debug.finest(format("Store ETag: %s", etag)); - etagStorage.put(etagKey, etag); + etagStore.accept(url, etag); }); } }; diff --git a/source/net/filebot/web/TMDbClient.java b/source/net/filebot/web/TMDbClient.java index c399ebca..49605ea6 100644 --- a/source/net/filebot/web/TMDbClient.java +++ b/source/net/filebot/web/TMDbClient.java @@ -34,6 +34,7 @@ import javax.swing.Icon; import net.filebot.Cache; import net.filebot.CacheType; +import net.filebot.CachedResource.Fetch; import net.filebot.Language; import net.filebot.ResourceManager; import net.filebot.web.TMDbClient.MovieInfo.MovieProperty; @@ -112,7 +113,7 @@ public class TMDbClient implements MovieIdentificationService { if (extendedInfo) { try { - Object titles = request("movie/" + id + "/alternative_titles", null, null, REQUEST_LIMIT); + Object titles = request("movie/" + id + "/alternative_titles", emptyMap(), Locale.ENGLISH, REQUEST_LIMIT); streamJsonObjects(titles, "titles").map(n -> { return getString(n, "title"); }).filter(t -> t != null && t.length() >= 3).forEach(alternativeTitles::add); @@ -167,7 +168,7 @@ public class TMDbClient implements MovieIdentificationService { } public MovieInfo getMovieInfo(String id, Locale locale, boolean extendedInfo) throws Exception { - Object response = request("movie/" + id, extendedInfo ? singletonMap("append_to_response", "alternative_titles,releases,casts,trailers") : null, locale, REQUEST_LIMIT); + Object response = request("movie/" + id, extendedInfo ? singletonMap("append_to_response", "alternative_titles,releases,casts,trailers") : emptyMap(), locale, REQUEST_LIMIT); Map fields = getEnumMap(response, MovieProperty.class); @@ -268,10 +269,10 @@ public class TMDbClient implements MovieIdentificationService { public List getArtwork(String id) throws Exception { // http://api.themoviedb.org/3/movie/11/images - Object config = request("configuration", null, null, REQUEST_LIMIT); + Object config = request("configuration", emptyMap(), Locale.ENGLISH, REQUEST_LIMIT); String baseUrl = getString(getMap(config, "images"), "base_url"); - Object images = request("movie/" + id + "/images", null, null, REQUEST_LIMIT); + Object images = request("movie/" + id + "/images", emptyMap(), Locale.ENGLISH, REQUEST_LIMIT); return Stream.of("backdrops", "posters").flatMap(section -> { Stream artwork = streamJsonObjects(images, section).map(it -> { @@ -292,39 +293,37 @@ public class TMDbClient implements MovieIdentificationService { public Object request(String resource, Map parameters, Locale locale, final FloodLimit limit) throws Exception { // default parameters - LinkedHashMap data = new LinkedHashMap(); - if (parameters != null) { - data.putAll(parameters); - } + String key = parameters.isEmpty() ? resource : resource + '?' + encodeParameters(parameters, true); - if (locale != null && locale.getLanguage().length() > 0) { - String code = locale.getLanguage(); + Cache etagStorage = Cache.getCache(getName() + "_" + locale + "_etag", CacheType.Monthly); + Fetch fetchIfNoneMatch = fetchIfNoneMatch(url -> etagStorage.get(key), (url, etag) -> etagStorage.put(key, etag)); - // require 2-letter language code - if (code.length() != 2) { - Language lang = Language.getLanguage(locale); - if (lang != null) { - code = lang.getISO2(); - } - } - data.put("language", code); - } - data.put("api_key", apikey); - - Cache cache = Cache.getCache(getName(), CacheType.Monthly); - Cache etagStorage = Cache.getCache("etag", CacheType.Monthly); - String key = resource + '?' + encodeParameters(data, true); - - Object json = cache.json(key, s -> getResource(s)).fetch(withPermit(fetchIfNoneMatch(etagStorage), r -> limit.acquirePermit())).expire(Cache.ONE_WEEK).get(); + Cache cache = Cache.getCache(getName() + "_" + locale, CacheType.Monthly); + Object json = cache.json(key, s -> getResource(s, locale)).fetch(withPermit(fetchIfNoneMatch, r -> limit.acquirePermit())).expire(Cache.ONE_WEEK).get(); if (asMap(json).isEmpty()) { - throw new FileNotFoundException(String.format("Resource is empty: %s => %s", json, getResource(key))); + throw new FileNotFoundException(String.format("Resource is empty: %s => %s", json, getResource(key, locale))); } return json; } - public URL getResource(String file) throws Exception { - return new URL("http", host, "/" + version + "/" + file); + protected URL getResource(String file, Locale locale) throws Exception { + return new URL("https", host, "/" + version + "/" + file + (file.lastIndexOf('?') < 0 ? '?' : '&') + "language=" + getLanguageCode(locale) + "&api_key=" + apikey); + } + + protected String getLanguageCode(Locale locale) { + // require 2-letter language code + String language = locale.getLanguage(); + if (language.length() == 2) { + return language; + } + + Language lang = Language.getLanguage(locale); + if (lang != null) { + return lang.getISO2(); + } + + throw new IllegalArgumentException("Illegal language code: " + language); } public static class MovieInfo implements Serializable { diff --git a/test/net/filebot/web/TMDbClientTest.java b/test/net/filebot/web/TMDbClientTest.java index 56415002..ac90e2ed 100644 --- a/test/net/filebot/web/TMDbClientTest.java +++ b/test/net/filebot/web/TMDbClientTest.java @@ -108,7 +108,7 @@ public class TMDbClientTest { public void etag() throws Exception { Cache cache = Cache.getCache("test", CacheType.Persistent); Cache etagStorage = Cache.getCache("etag", CacheType.Persistent); - CachedResource resource = cache.bytes("http://devel.squid-cache.org/old_projects.html#etag", URL::new).fetch(fetchIfNoneMatch(etagStorage)).expire(Duration.ZERO); + CachedResource resource = cache.bytes("http://devel.squid-cache.org/old_projects.html#etag", URL::new).fetch(fetchIfNoneMatch(etagStorage::get, etagStorage::put)).expire(Duration.ZERO); assertArrayEquals(resource.get(), resource.get()); }