From c10b8d8e2f39ef5d5fc3433e32800d4cbab995f6 Mon Sep 17 00:00:00 2001
From: shayaantx <5449086+shayaantx@users.noreply.github.com>
Date: Thu, 22 Dec 2022 21:09:01 -0500
Subject: [PATCH] Add sonarr v4 support + update dependencies + refactor api
urls across (#86)
Arr apis
---
README.md | 2 +
docker/entrypoint.sh | 1 +
pom.xml | 4 +-
sample.properties | 2 +
src/main/java/com/botdarr/Config.java | 6 +
src/main/java/com/botdarr/api/Api.java | 35 +---
.../com/botdarr/api/ArrRequestBuilder.java | 54 ++++++
.../com/botdarr/api/CacheContentStrategy.java | 15 +-
.../com/botdarr/api/DownloadsStrategy.java | 30 +--
.../com/botdarr/api/lidarr/LidarrApi.java | 52 +++---
.../com/botdarr/api/lidarr/LidarrUrls.java | 10 +
.../com/botdarr/api/radarr/RadarrApi.java | 62 +++----
.../com/botdarr/api/radarr/RadarrUrls.java | 14 ++
.../com/botdarr/api/sonarr/SonarrApi.java | 172 ++++++++++++++----
.../com/botdarr/api/sonarr/SonarrCache.java | 18 ++
.../api/sonarr/SonarrDownloadActivity.java | 62 +++++++
.../api/sonarr/SonarrEpisodeInformation.java | 31 ++++
.../com/botdarr/api/sonarr/SonarrProfile.java | 6 +-
.../api/sonarr/SonarrProfileQualityItem.java | 21 +++
.../com/botdarr/api/sonarr/SonarrQueue.java | 34 +++-
...ueEpisode.java => SonarrQueueEpisode.java} | 11 +-
.../com/botdarr/api/sonarr/SonarrUrls.java | 27 ++-
.../discord/DiscordResponseBuilder.java | 47 ++---
.../clients/matrix/MatrixResponseBuilder.java | 40 ++--
.../clients/slack/SlackResponseBuilder.java | 44 +++--
.../telegram/TelegramResponseBuilder.java | 51 +++---
.../responses/ShowDownloadResponse.java | 8 +-
.../botdarr/connections/ConnectionHelper.java | 16 +-
.../botdarr/connections/RequestBuilder.java | 74 ++++++++
src/main/resources/version.txt | 2 +-
.../botdarr/api/DownloadsStrategyTests.java | 30 +--
.../botdarr/api/radarr/RadarrApiTests.java | 6 +-
32 files changed, 672 insertions(+), 315 deletions(-)
create mode 100644 src/main/java/com/botdarr/api/ArrRequestBuilder.java
create mode 100644 src/main/java/com/botdarr/api/sonarr/SonarrDownloadActivity.java
create mode 100644 src/main/java/com/botdarr/api/sonarr/SonarrEpisodeInformation.java
rename src/main/java/com/botdarr/api/sonarr/{SonarQueueEpisode.java => SonarrQueueEpisode.java} (84%)
create mode 100644 src/main/java/com/botdarr/connections/RequestBuilder.java
diff --git a/README.md b/README.md
index ea45eae..846a6db 100644
--- a/README.md
+++ b/README.md
@@ -12,6 +12,7 @@ Made this simple multi chat-client bot to access radarr, sonarr, and lidarr with
## Currently, Supported API's
- [x] Radarr (v4)
+- [x] Sonarr (v4)
- [x] Sonarr (v3)
- [x] Lidarr (v1)
- [x] Lidarr (v0)
@@ -166,6 +167,7 @@ botdarr:
| SONARR_DEFAULT_PROFILE | The sonarr quality profile (should be already configured in sonarr) | yes - if you use sonarr |
| SONARR_PATH | Where your sonarr shows should go (if you add/update them) | yes - if you use sonarr |
| SONARR_URL_BASE | Only populate this if you use a custom sonarr url base (which is configurable in Sonarr->Settings->General->URL Base) don't include prefix/suffix slashes | no |
+| SONARR_V4 | Whether you are using sonarr v4 or not | no | no |
#### Lidarr
| Environment Variable | Description | Required | Default |
diff --git a/docker/entrypoint.sh b/docker/entrypoint.sh
index da16c1d..d0f6bf7 100644
--- a/docker/entrypoint.sh
+++ b/docker/entrypoint.sh
@@ -39,6 +39,7 @@ if [ ! -e "$propertiesFile" ]; then
[[ ! -z "${SONARR_DEFAULT_PROFILE}" ]] && addConfiguration "sonarr-default-profile" "${SONARR_DEFAULT_PROFILE}" "${propertiesFile}"
[[ ! -z "${SONARR_PATH}" ]] && addConfiguration "sonarr-path" "${SONARR_PATH}" "${propertiesFile}"
[[ ! -z "${SONARR_URL_BASE}" ]] && addConfiguration "sonarr-url-base" "${SONARR_URL_BASE}" "${propertiesFile}"
+ [[ ! -z "${SONARR_V4}" ]] && addConfiguration "sonarr-v4" "${SONARR_V4}" "${propertiesFile}"
[[ ! -z "${LIDARR_URL}" ]] && addConfiguration "lidarr-url" "${LIDARR_URL}" "${propertiesFile}"
[[ ! -z "${LIDARR_TOKEN}" ]] && addConfiguration "lidarr-token" "${LIDARR_TOKEN}" "${propertiesFile}"
diff --git a/pom.xml b/pom.xml
index 2656dde..1c11ddd 100644
--- a/pom.xml
+++ b/pom.xml
@@ -46,7 +46,7 @@
com.google.code.gson
gson
- 2.8.5
+ 2.8.9
org.apache.logging.log4j
@@ -66,7 +66,7 @@
com.fasterxml.jackson.core
jackson-databind
- 2.13.1
+ 2.13.4.1
com.google.guava
diff --git a/sample.properties b/sample.properties
index dbc7a6c..cb3377f 100644
--- a/sample.properties
+++ b/sample.properties
@@ -53,6 +53,8 @@ sonarr-path=
# Only populate this if you use a custom sonarr url base (which is configurable in Sonarr->Settings->General->URL Base)
# don't include prefix/suffix slashes
sonarr-url-base=
+# Whether your sonarr instance is v4 or not
+sonarr-v4
lidarr-url=
lidarr-token=
diff --git a/src/main/java/com/botdarr/Config.java b/src/main/java/com/botdarr/Config.java
index a62e778..9dc8eb7 100644
--- a/src/main/java/com/botdarr/Config.java
+++ b/src/main/java/com/botdarr/Config.java
@@ -327,6 +327,11 @@ public static final class Constants {
*/
public static final String SONARR_URL_BASE = "sonarr-url-base";
+ /**
+ * Whether to turn on sonarr v4 support or not
+ */
+ public static final String SONARR_V4 = "sonarr-v4";
+
/**
* The url to your lidarr instance
*/
@@ -416,6 +421,7 @@ public static final class Constants {
* Config for the log level
*/
public static final String LOG_LEVEL = "log-level";
+ public static final int VALUE_MAX_LENGTH = 1024;
}
private static String propertiesPath = "config/properties";
diff --git a/src/main/java/com/botdarr/api/Api.java b/src/main/java/com/botdarr/api/Api.java
index 3103978..b04c9cc 100644
--- a/src/main/java/com/botdarr/api/Api.java
+++ b/src/main/java/com/botdarr/api/Api.java
@@ -1,30 +1,12 @@
package com.botdarr.api;
-import com.botdarr.Config;
-import com.botdarr.clients.ChatClient;
import com.botdarr.clients.ChatClientResponse;
-import com.botdarr.clients.ChatClientResponseBuilder;
-import com.botdarr.clients.discord.DiscordResponse;
import com.botdarr.commands.responses.CommandResponse;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
-import org.apache.logging.log4j.util.Strings;
-
-import java.io.UnsupportedEncodingException;
-import java.net.URLEncoder;
import java.util.List;
public interface Api {
- /**
- * The url base for this api (can be null/empty)
- */
- String getUrlBase();
-
- /**
- * Get this api's url endpoint
- */
- String getApiUrl(String path);
-
/**
* Gets all the in-progress downloads
*/
@@ -35,24 +17,9 @@ public interface Api {
*/
void cacheData();
- /**
- * Gets the auth token for this api
- */
- String getApiToken();
-
- default String getApiUrl(String apiUrlKey, String apiTokenKey, String path) {
- try {
- String urlBase = Strings.isBlank(getUrlBase()) ? "" : "/" + getUrlBase();
- return Config.getProperty(apiUrlKey) + urlBase + "/api/" + path + "?apikey=" + URLEncoder.encode(Config.getProperty(apiTokenKey), "UTF-8");
- } catch (UnsupportedEncodingException e) {
- LOGGER.error("Error encoding api token", e);
- throw new RuntimeException("Error calculating the api url", e);
- }
- }
-
default List subList(List responses, int max) {
return responses.subList(0, responses.size() > max ? max - 1 : responses.size());
}
- static final Logger LOGGER = LogManager.getLogger();
+ Logger LOGGER = LogManager.getLogger();
}
diff --git a/src/main/java/com/botdarr/api/ArrRequestBuilder.java b/src/main/java/com/botdarr/api/ArrRequestBuilder.java
new file mode 100644
index 0000000..3aa28dd
--- /dev/null
+++ b/src/main/java/com/botdarr/api/ArrRequestBuilder.java
@@ -0,0 +1,54 @@
+package com.botdarr.api;
+
+import com.botdarr.Config;
+import com.botdarr.connections.RequestBuilder;
+import org.apache.logging.log4j.util.Strings;
+import java.util.HashMap;
+import java.util.Map;
+
+public class ArrRequestBuilder {
+ private final String urlConfigName;
+ private final String urlBaseConfigName;
+ private final String tokenName;
+ private final RequestBuilder requestBuilder = new RequestBuilder();
+ public ArrRequestBuilder(String urlConfigName, String urlBaseConfigName, String tokenName) {
+ this.urlConfigName = urlConfigName;
+ this.urlBaseConfigName = urlBaseConfigName;
+ this.tokenName = tokenName;
+ }
+
+ private RequestBuilder tokens(RequestBuilder requestBuilder) {
+ //TODO: which *arr's actually need the api key in path and headers? (this is likely a side effect from one
+ //of the apis requiring one or the other and me being lazy)
+ return requestBuilder.param("apiKey", Config.getProperty(this.tokenName))
+ .headers("X-Api-Key", Config.getProperty(this.tokenName));
+ }
+
+ private String getRadarrHost() {
+ String host = Config.getProperty(this.urlConfigName);
+ String rawUrlBase = Config.getProperty(this.urlBaseConfigName);
+ String urlBase = Strings.isBlank(rawUrlBase) ? "" : "/" + rawUrlBase;
+ return host + urlBase + getApiSuffix();
+ }
+
+ public String getApiSuffix() {
+ return "/api/";
+ }
+
+ public RequestBuilder buildGet(String path) {
+ return buildGet(path, new HashMap<>());
+ }
+
+ public RequestBuilder buildGet(String path, Map params) {
+ for (Map.Entry entry : params.entrySet()) {
+ this.requestBuilder.param(entry.getKey(), entry.getValue());
+ }
+ RequestBuilder requestBuilder = this.requestBuilder.host(getRadarrHost() + path);
+ return this.tokens(requestBuilder);
+ }
+
+ public RequestBuilder buildPost(String path, String postBody) {
+ RequestBuilder requestBuilder = this.requestBuilder.host(getRadarrHost() + path).post(postBody);
+ return this.tokens(requestBuilder);
+ }
+}
\ No newline at end of file
diff --git a/src/main/java/com/botdarr/api/CacheContentStrategy.java b/src/main/java/com/botdarr/api/CacheContentStrategy.java
index e60eaaa..229d551 100644
--- a/src/main/java/com/botdarr/api/CacheContentStrategy.java
+++ b/src/main/java/com/botdarr/api/CacheContentStrategy.java
@@ -1,6 +1,7 @@
package com.botdarr.api;
import com.botdarr.connections.ConnectionHelper;
+import com.botdarr.connections.RequestBuilder;
import com.google.gson.JsonArray;
import com.google.gson.JsonElement;
import com.google.gson.JsonParser;
@@ -11,9 +12,8 @@
import java.util.List;
public abstract class CacheContentStrategy {
- public CacheContentStrategy(Api api, String url) {
- this.api = api;
- this.url = url;
+ public CacheContentStrategy(RequestBuilder requestBuilder) {
+ this.requestBuilder = requestBuilder;
}
public abstract void deleteFromCache(List itemsAddedUpdated);
@@ -21,7 +21,9 @@ public CacheContentStrategy(Api api, String url) {
public void cacheData() {
List itemsAddedUpdated = new ArrayList<>();
- ConnectionHelper.makeGetRequest(this.api, this.url, new ConnectionHelper.SimpleEntityResponseHandler>() {
+ ConnectionHelper.makeGetRequest(
+ this.requestBuilder,
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) throws Exception {
JsonParser parser = new JsonParser();
@@ -33,10 +35,9 @@ public List onSuccess(String response) throws Exception {
}
});
deleteFromCache(itemsAddedUpdated);
- LOGGER.debug("Finished caching content data for api " + api.getClass().getName());
+ LOGGER.debug("Finished caching content data for host " + this.requestBuilder.getHost());
}
- private final Api api;
- private final String url;
+ private final RequestBuilder requestBuilder;
private static final Logger LOGGER = LogManager.getLogger();
}
diff --git a/src/main/java/com/botdarr/api/DownloadsStrategy.java b/src/main/java/com/botdarr/api/DownloadsStrategy.java
index 2db384a..2bbb8ac 100644
--- a/src/main/java/com/botdarr/api/DownloadsStrategy.java
+++ b/src/main/java/com/botdarr/api/DownloadsStrategy.java
@@ -3,6 +3,7 @@
import com.botdarr.commands.responses.CommandResponse;
import com.botdarr.commands.responses.InfoResponse;
import com.botdarr.connections.ConnectionHelper;
+import com.botdarr.connections.RequestBuilder;
import com.botdarr.utilities.ListUtils;
import com.google.gson.JsonArray;
import com.google.gson.JsonElement;
@@ -16,13 +17,8 @@
import java.util.List;
public abstract class DownloadsStrategy {
- public DownloadsStrategy(Api api,
- String url) {
- this.api = api;
- this.url = url;
- }
-
public abstract CommandResponse getResponse(JsonElement rawElement);
+ public abstract List getContentDownloads();
public List downloads() {
if (MAX_DOWNLOADS_TO_SHOW <= 0) {
@@ -31,27 +27,9 @@ public List downloads() {
return getContentDownloads();
}
- public List getContentDownloads() {
- return ConnectionHelper.makeGetRequest(this.api, this.url, new ConnectionHelper.SimpleMessageEmbedResponseHandler() {
-
- @Override
- public List onConnectException(HttpHostConnectException e) {
- String message = "Error trying to connect to " + DownloadsStrategy.this.api.getApiUrl(DownloadsStrategy.this.url);
- LOGGER.error(message);
- return Collections.emptyList();
- }
-
- @Override
- public List onSuccess(String response) {
- return parseContent(response);
- }
- });
- }
-
public List parseContent(String response) {
List chatClientResponses = new ArrayList<>();
- JsonParser parser = new JsonParser();
- JsonArray json = parser.parse(response).getAsJsonArray();
+ JsonArray json = JsonParser.parseString(response).getAsJsonArray();
boolean tooManyDownloads = json.size() >= MAX_DOWNLOADS_TO_SHOW;
for (int i = 0; i < json.size(); i++) {
CommandResponse chatClientResponse = getResponse(json.get(i));
@@ -67,8 +45,6 @@ public List parseContent(String response) {
return chatClientResponses;
}
- private final Api api;
- private final String url;
private final int MAX_DOWNLOADS_TO_SHOW = new ApiRequests().getMaxDownloadsToShow();
private static Logger LOGGER = LogManager.getLogger();
}
diff --git a/src/main/java/com/botdarr/api/lidarr/LidarrApi.java b/src/main/java/com/botdarr/api/lidarr/LidarrApi.java
index 3dcb4ba..f67d3a5 100644
--- a/src/main/java/com/botdarr/api/lidarr/LidarrApi.java
+++ b/src/main/java/com/botdarr/api/lidarr/LidarrApi.java
@@ -2,39 +2,27 @@
import com.botdarr.Config;
import com.botdarr.api.*;
-import com.botdarr.api.sonarr.SonarrUrls;
import com.botdarr.commands.CommandContext;
import com.botdarr.commands.responses.*;
import com.botdarr.connections.ConnectionHelper;
+import com.botdarr.connections.RequestBuilder;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.gson.*;
import org.apache.commons.io.IOUtils;
import org.apache.http.client.methods.CloseableHttpResponse;
-import org.apache.http.client.methods.HttpPost;
-import org.apache.http.entity.StringEntity;
+import org.apache.http.client.methods.HttpRequestBase;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.logging.log4j.LogManager;
import java.io.IOException;
-import java.net.URLEncoder;
-import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Collections;
+import java.util.HashMap;
import java.util.List;
public class LidarrApi implements Api {
- @Override
- public String getUrlBase() {
- return Config.getProperty(Config.Constants.LIDARR_URL_BASE);
- }
-
- @Override
- public String getApiUrl(String path) {
- return getApiUrl(Config.Constants.LIDARR_URL, Config.Constants.LIDARR_TOKEN, path);
- }
-
@Override
public List downloads() {
return getDownloadsStrategy().downloads();
@@ -51,7 +39,9 @@ public void deleteFromCache(List profilesAddUpdated) {
@Override
public List getProfiles() {
- return ConnectionHelper.makeGetRequest(LidarrApi.this, LidarrUrls.PROFILE, new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new LidarrUrls.LidarrRequestBuilder().buildGet(LidarrUrls.PROFILE),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
List lidarrQualityProfiles = new ArrayList<>();
@@ -81,7 +71,9 @@ public void deleteFromCache(List profilesAddUpdated) {
@Override
public List getProfiles() {
- return ConnectionHelper.makeGetRequest(LidarrApi.this, LidarrUrls.METADATA_PROFILE, new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new LidarrUrls.LidarrRequestBuilder().buildGet(LidarrUrls.METADATA_PROFILE),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
List lidarrMetadataProfiles = new ArrayList<>();
@@ -102,7 +94,7 @@ public void addProfile(LidarrMetadataProfile profile) {
}
}.cacheData();
- new CacheContentStrategy(this, LidarrUrls.ALL_ARTISTS) {
+ new CacheContentStrategy(new LidarrUrls.LidarrRequestBuilder().buildGet(LidarrUrls.ALL_ARTISTS)) {
@Override
public void deleteFromCache(List itemsToRetain) {
@@ -120,11 +112,6 @@ public String addToCache(JsonElement cacheItem) {
//TODO: add album cache
}
- @Override
- public String getApiToken() {
- return Config.Constants.LIDARR_TOKEN;
- }
-
public CommandResponse addArtistWithId(String id, String artistName) {
return getArtistAddStrategy().addWithSearchId(artistName, id);
}
@@ -204,7 +191,8 @@ public CommandResponse getResponse(LidarrArtist item) {
}
private DownloadsStrategy getDownloadsStrategy() {
- return new DownloadsStrategy(this, LidarrUrls.DOWNLOAD_BASE) {
+ RequestBuilder requestBuilder = new LidarrUrls.LidarrRequestBuilder().buildGet(LidarrUrls.DOWNLOAD_BASE);
+ return new DownloadsStrategy() {
@Override
public CommandResponse getResponse(JsonElement rawElement) {
LidarrQueueRecord lidarrQueueRecord = new Gson().fromJson(rawElement, LidarrQueueRecord.class);
@@ -212,11 +200,12 @@ public CommandResponse getResponse(JsonElement rawElement) {
}
@Override
public List getContentDownloads() {
- return ConnectionHelper.makeGetRequest(LidarrApi.this, LidarrUrls.DOWNLOAD_BASE, new ConnectionHelper.SimpleMessageEmbedResponseHandler() {
+ return ConnectionHelper.makeGetRequest(
+ requestBuilder,
+ new ConnectionHelper.SimpleCommandResponseHandler() {
@Override
public List onSuccess(String response) {
- JsonParser parser = new JsonParser();
- JsonObject json = parser.parse(response).getAsJsonObject();
+ JsonObject json = JsonParser.parseString(response).getAsJsonObject();
return parseContent(json.get("records").toString());
}
});
@@ -225,7 +214,10 @@ public List onSuccess(String response) {
}
private List lookupArtists(String search) throws Exception {
- return ConnectionHelper.makeGetRequest(this, LidarrUrls.LOOKUP_ARTISTS, "&term=" + URLEncoder.encode(search, "UTF-8"),
+ return ConnectionHelper.makeGetRequest(
+ new LidarrUrls.LidarrRequestBuilder().buildGet(LidarrUrls.LOOKUP_ARTISTS, new HashMap() {{
+ put("term", search);
+ }}),
new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
@@ -259,14 +251,12 @@ private CommandResponse addArtist(LidarrArtist lidarrArtist) {
lidarrArtist.setMetadataProfileId(lidarrMetadataProfile.getId());
lidarrArtist.setQualityProfileId(lidarrQualityProfile.getId());
try (CloseableHttpClient client = HttpClientBuilder.create().build()) {
- HttpPost post = new HttpPost(getApiUrl(SonarrUrls.ARTIST_BASE));
- post.addHeader("content-type", "application/json");
ObjectMapper mapper = new ObjectMapper();
//lidarr for some reason doesn't support raw unicode characters in json parsing (since they should be allowed), so we escape them here
mapper.getFactory().configure(JsonGenerator.Feature.ESCAPE_NON_ASCII, true);
String json = mapper.writeValueAsString(lidarrArtist);
- post.setEntity(new StringEntity(json, Charset.forName("UTF-8")));
+ HttpRequestBase post = new LidarrUrls.LidarrRequestBuilder().buildPost(LidarrUrls.ARTIST_BASE, json).build();
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("Client request=" + post);
diff --git a/src/main/java/com/botdarr/api/lidarr/LidarrUrls.java b/src/main/java/com/botdarr/api/lidarr/LidarrUrls.java
index 5c36eb3..a623b62 100644
--- a/src/main/java/com/botdarr/api/lidarr/LidarrUrls.java
+++ b/src/main/java/com/botdarr/api/lidarr/LidarrUrls.java
@@ -1,6 +1,15 @@
package com.botdarr.api.lidarr;
+import com.botdarr.Config;
+import com.botdarr.api.ArrRequestBuilder;
+
public class LidarrUrls {
+ public static class LidarrRequestBuilder extends ArrRequestBuilder {
+ public LidarrRequestBuilder() {
+ super(Config.Constants.LIDARR_URL, Config.Constants.LIDARR_URL_BASE, Config.Constants.LIDARR_TOKEN);
+ }
+ }
+
/**
* The base endpoint prefix for lidarr v1 api, https://github.com/lidarr/Lidarr/wiki/API
*/
@@ -30,4 +39,5 @@ public class LidarrUrls {
* TODO: doc
*/
public static final String METADATA_PROFILE = VERSION + "metadataprofile";
+ public static final String ARTIST_BASE = "v1/artist";
}
diff --git a/src/main/java/com/botdarr/api/radarr/RadarrApi.java b/src/main/java/com/botdarr/api/radarr/RadarrApi.java
index ea56ef8..7d71147 100644
--- a/src/main/java/com/botdarr/api/radarr/RadarrApi.java
+++ b/src/main/java/com/botdarr/api/radarr/RadarrApi.java
@@ -8,28 +8,16 @@
import com.google.gson.*;
import org.apache.commons.io.IOUtils;
import org.apache.http.client.methods.CloseableHttpResponse;
-import org.apache.http.client.methods.HttpPost;
-import org.apache.http.entity.StringEntity;
+import org.apache.http.client.methods.HttpRequestBase;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.logging.log4j.LogManager;
import java.io.IOException;
-import java.net.URLEncoder;
import java.nio.charset.Charset;
import java.util.*;
public class RadarrApi implements Api {
- @Override
- public String getUrlBase() {
- return Config.getProperty(Config.Constants.RADARR_URL_BASE);
- }
-
- @Override
- public String getApiUrl(String path) {
- return getApiUrl(Config.Constants.RADARR_URL, Config.Constants.RADARR_TOKEN, "v3/" + path);
- }
-
public List lookup(String search, boolean findNew) {
return new LookupStrategy(ContentType.MOVIE) {
@@ -96,7 +84,9 @@ public void deleteFromCache(List profilesAddUpdated) {
@Override
public List getProfiles() {
- return ConnectionHelper.makeGetRequest(RadarrApi.this, RadarrUrls.PROFILE_BASE, new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new RadarrUrls.RadarrV3RequestBuilder().buildGet(RadarrUrls.PROFILE_BASE),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
List radarrProfiles = new ArrayList<>();
@@ -117,7 +107,7 @@ public void addProfile(RadarrProfile profile) {
}
}.cacheData();
- new CacheContentStrategy(this, RadarrUrls.MOVIE_BASE) {
+ new CacheContentStrategy(new RadarrUrls.RadarrV3RequestBuilder().buildGet(RadarrUrls.MOVIE_BASE)) {
@Override
public void deleteFromCache(List itemsAddedUpdated) {
RADARR_CACHE.removeDeletedMovies(itemsAddedUpdated);
@@ -132,13 +122,12 @@ public Long addToCache(JsonElement cacheItem) {
}.cacheData();
}
- @Override
- public String getApiToken() {
- return Config.Constants.RADARR_TOKEN;
- }
-
public List discover() {
- return ConnectionHelper.makeGetRequest(this, RadarrUrls.DISCOVER_MOVIES, "&includeRecommendations=true", new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new RadarrUrls.RadarrV3RequestBuilder().buildGet(RadarrUrls.DISCOVER_MOVIES, new HashMap() {{
+ put("includeRecommendations", "true");
+ }}),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) throws Exception {
List recommendedMovies = new ArrayList<>();
@@ -163,7 +152,7 @@ public List onSuccess(String response) throws Exception {
}
private DownloadsStrategy getDownloadsStrategy() {
- return new DownloadsStrategy(this, RadarrUrls.DOWNLOAD_BASE) {
+ return new DownloadsStrategy() {
@Override
public CommandResponse getResponse(JsonElement rawElement) {
RadarrQueue radarrQueue = new Gson().fromJson(rawElement, RadarrQueue.class);
@@ -186,16 +175,14 @@ public CommandResponse getResponse(JsonElement rawElement) {
@Override
public List getContentDownloads() {
return ConnectionHelper.makeGetRequest(
- RadarrApi.this,
- RadarrUrls.DOWNLOAD_BASE,
- new ConnectionHelper.SimpleMessageEmbedResponseHandler() {
+ new RadarrUrls.RadarrV3RequestBuilder().buildGet(RadarrUrls.DOWNLOAD_BASE),
+ new ConnectionHelper.SimpleCommandResponseHandler() {
@Override
public List onSuccess(String response) {
if (response == null || response.isEmpty() || response.equals("{}")) {
return new ArrayList<>();
}
- JsonParser parser = new JsonParser();
- JsonObject json = parser.parse(response).getAsJsonObject();
+ JsonObject json = JsonParser.parseString(response).getAsJsonObject();
return parseContent(json.get("records").toString());
}
});
@@ -251,15 +238,12 @@ private CommandResponse addMovie(RadarrMovie radarrMovie) {
radarrMovie.setQualityProfileId((int) radarrProfile.getId());
try (CloseableHttpClient client = HttpClientBuilder.create().build()) {
- HttpPost post = new HttpPost(getApiUrl(RadarrUrls.MOVIE_BASE));
-
- post.addHeader("content-type", "application/json");
String json = new Gson().toJson(radarrMovie, RadarrMovie.class);
- post.setEntity(new StringEntity(json, Charset.forName("UTF-8")));
+ HttpRequestBase post = new RadarrUrls.RadarrV3RequestBuilder().buildPost(RadarrUrls.MOVIE_BASE, json).build();
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("Client request=" + post);
- LOGGER.debug("Client data=" + (json));
+ LOGGER.debug("Client data=" + json);
}
String username = CommandContext.getConfig().getUsername();
@@ -292,8 +276,11 @@ private CommandResponse addMovie(RadarrMovie radarrMovie) {
}
private List lookupMovies(String search) throws Exception {
- return ConnectionHelper.makeGetRequest(this, RadarrUrls.MOVIE_LOOKUP, "&term=" + URLEncoder.encode(search, "UTF-8"),
- new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new RadarrUrls.RadarrV3RequestBuilder().buildGet(RadarrUrls.MOVIE_LOOKUP, new HashMap() {{
+ put("term", search);
+ }}),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
List movies = new ArrayList<>();
@@ -308,8 +295,11 @@ public List onSuccess(String response) {
}
private List lookupMoviesById(String tmdbid) throws Exception {
- return ConnectionHelper.makeGetRequest(this, RadarrUrls.MOVIE_LOOKUP_TMDB, "&tmdbId=" + URLEncoder.encode(tmdbid, "UTF-8"),
- new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new RadarrUrls.RadarrV3RequestBuilder().buildGet(RadarrUrls.MOVIE_LOOKUP_TMDB, new HashMap() {{
+ put("tmdbId", tmdbid);
+ }}),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
List movies = new ArrayList<>();
diff --git a/src/main/java/com/botdarr/api/radarr/RadarrUrls.java b/src/main/java/com/botdarr/api/radarr/RadarrUrls.java
index 3f3578e..97293b1 100644
--- a/src/main/java/com/botdarr/api/radarr/RadarrUrls.java
+++ b/src/main/java/com/botdarr/api/radarr/RadarrUrls.java
@@ -1,6 +1,20 @@
package com.botdarr.api.radarr;
+import com.botdarr.Config;
+import com.botdarr.api.ArrRequestBuilder;
+
public class RadarrUrls {
+ public static class RadarrV3RequestBuilder extends ArrRequestBuilder {
+ public RadarrV3RequestBuilder() {
+ super(Config.Constants.RADARR_URL, Config.Constants.RADARR_URL_BASE, Config.Constants.RADARR_TOKEN);
+ }
+
+ @Override
+ public String getApiSuffix() {
+ return "/api/v3/";
+ }
+ }
+
/**
* The base download(s) url for get, put, delete requests (which each do different things in radarr)
* See https://github.com/Radarr/Radarr/wiki/API:Queue
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrApi.java b/src/main/java/com/botdarr/api/sonarr/SonarrApi.java
index 6405db7..6356c86 100644
--- a/src/main/java/com/botdarr/api/sonarr/SonarrApi.java
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrApi.java
@@ -7,28 +7,17 @@
import com.botdarr.connections.ConnectionHelper;
import com.google.gson.*;
import org.apache.http.client.methods.CloseableHttpResponse;
-import org.apache.http.client.methods.HttpPost;
-import org.apache.http.entity.StringEntity;
+import org.apache.http.client.methods.HttpRequestBase;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.logging.log4j.LogManager;
import java.io.IOException;
-import java.net.URLEncoder;
-import java.nio.charset.Charset;
import java.util.*;
-public class SonarrApi implements Api {
- @Override
- public String getUrlBase() {
- return Config.getProperty(Config.Constants.SONARR_URL_BASE);
- }
-
- @Override
- public String getApiUrl(String path) {
- return getApiUrl(Config.Constants.SONARR_URL, Config.Constants.SONARR_TOKEN, path);
- }
+import static com.botdarr.Config.Constants.VALUE_MAX_LENGTH;
+public class SonarrApi implements Api {
@Override
public List downloads() {
return getDownloadsStrategy().downloads();
@@ -95,7 +84,9 @@ public void deleteFromCache(List profilesAddUpdated) {
@Override
public List getProfiles() {
- return ConnectionHelper.makeGetRequest(SonarrApi.this, SonarrUrls.PROFILE, new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new SonarrUrls.SonarrRequestBuilder().buildGet(SonarrUrls.PROFILE),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
List sonarrProfiles = new ArrayList<>();
@@ -116,7 +107,7 @@ public void addProfile(SonarrProfile profile) {
}
}.cacheData();
- new CacheContentStrategy(this, SonarrUrls.SERIES_BASE) {
+ new CacheContentStrategy(new SonarrUrls.SonarrRequestBuilder().buildGet(SonarrUrls.SERIES_BASE)) {
@Override
public void deleteFromCache(List itemsAddedUpdated) {
SONARR_CACHE.removeDeletedShows(itemsAddedUpdated);
@@ -131,11 +122,6 @@ public Long addToCache(JsonElement cacheItem) {
}.cacheData();
}
- @Override
- public String getApiToken() {
- return Config.Constants.SONARR_TOKEN;
- }
-
private AddStrategy getAddStrategy() {
return new AddStrategy(ContentType.SHOW) {
@Override
@@ -171,17 +157,77 @@ public CommandResponse getResponse(SonarrShow item) {
};
}
+ private SonarrEpisodeInformation getEpisode(long seriesId, long episodeId) {
+ return ConnectionHelper.makeGetRequest(
+ new SonarrUrls.SonarrRequestBuilder().buildGet(SonarrUrls.EPISODES_LOOKUP, new HashMap() {{
+ put("seriesId", String.valueOf(seriesId));
+ put("episodeIds", String.valueOf(episodeId));
+ }}),
+ new ConnectionHelper.SimpleEntityResponseHandler() {
+ @Override
+ public SonarrEpisodeInformation onSuccess(String response) {
+ if (response == null || response.isEmpty() || response.equals("{}")) {
+ return null;
+ }
+ JsonArray array = JsonParser.parseString(response).getAsJsonArray();
+ for (int i = 0; i < array.size(); i++) {
+ SonarrQueueEpisode episode = new Gson().fromJson(array.get(i), SonarrQueueEpisode.class);
+ if (episode.getId() != episodeId) {
+ continue;
+ }
+ return new SonarrEpisodeInformation(episode.getSeasonNumber(), episode.getEpisodeNumber(), episode.getTitle(), episode.getOverview());
+ }
+ return null;
+ }
+ }
+ );
+ }
+
private DownloadsStrategy getDownloadsStrategy() {
- return new DownloadsStrategy(this, SonarrUrls.DOWNLOAD_BASE) {
- @Override
- public CommandResponse getResponse(JsonElement rawElement) {
- SonarrQueue showQueue = new Gson().fromJson(rawElement, SonarrQueue.class);
- SonarQueueEpisode episode = showQueue.getEpisode();
- if (episode == null) {
- //something is wrong with the download, skip
- LOGGER.error("Series " + showQueue.getSonarrQueueShow().getTitle() + " missing episode info for id " + showQueue.getId());
+ return new DownloadsStrategy() {
+ private ShowDownloadResponse getDownloadBasedApiV3Queue(SonarrQueue showQueue) {
+ SonarrShow sonarrShow = SONARR_CACHE.getExistingShowFromSonarrId(showQueue.getSeriesId());
+ if (sonarrShow == null) {
+ LOGGER.warn("Could not load sonarr show from cache for id " + showQueue.getSeriesId());
return null;
}
+ if (isPathBlacklisted(sonarrShow)) {
+ LOGGER.warn("The following show is blacklisted: " + sonarrShow.getTitle() + " from being displayed in downloads");
+ return null;
+ }
+ SonarrEpisodeInformation episodeInformation = SONARR_CACHE.getEpisode(showQueue.getSeriesId(), showQueue.getEpisodeId());
+ if (episodeInformation == null) {
+ episodeInformation = getEpisode(showQueue.getSeriesId(), showQueue.getEpisodeId());
+ if (episodeInformation == null) {
+ // if episode is still null, we can't display any download data
+ LOGGER.error("Couldn't find download data in sonarr cache or sonarr api for " + showQueue.getSeriesId() + ", episode " + showQueue.getEpisodeId());
+ return null;
+ }
+ // cache information for faster lookups since download data is displayed periodically
+ SONARR_CACHE.addEpisode(showQueue.getSeriesId(), showQueue.getEpisodeId(), episodeInformation);
+ }
+ List statusMessages = new ArrayList<>();
+ for (SonarrQueueStatusMessages sonarrQueueStatusMessages : showQueue.getStatusMessages()) {
+ statusMessages.add(sonarrQueueStatusMessages.getTitle());
+ }
+ String overview = episodeInformation.getOverview();
+ if (overview.length() > VALUE_MAX_LENGTH) {
+ overview = overview.substring(0, VALUE_MAX_LENGTH);
+ }
+ return new ShowDownloadResponse(new SonarrDownloadActivity(
+ sonarrShow.getTitle() + ": " + episodeInformation.getTitle(),
+ episodeInformation.getSeasonNumber(),
+ episodeInformation.getEpisodeNumber(),
+ showQueue.getQuality().getQuality().getName(),
+ showQueue.getStatus(),
+ showQueue.getTimeleft(),
+ overview,
+ statusMessages.toArray(new String[]{})
+ ));
+ }
+
+ private ShowDownloadResponse getDownloadBasedApiQueue(SonarrQueue showQueue) {
+ SonarrQueueEpisode episode = showQueue.getEpisode();
SonarrShow sonarrShow = SONARR_CACHE.getExistingShowFromSonarrId(showQueue.getEpisode().getSeriesId());
if (sonarrShow == null) {
LOGGER.warn("Could not load sonarr show from cache for id " + showQueue.getEpisode().getSeriesId() + " title=" + showQueue.getSonarrQueueShow().getTitle());
@@ -191,7 +237,54 @@ public CommandResponse getResponse(JsonElement rawElement) {
LOGGER.warn("The following show is blacklisted: " + sonarrShow.getTitle() + " from being displayed in downloads");
return null;
}
- return new ShowDownloadResponse(showQueue);
+ List statusMessages = new ArrayList<>();
+ for (SonarrQueueStatusMessages sonarrQueueStatusMessages : showQueue.getStatusMessages()) {
+ statusMessages.add(sonarrQueueStatusMessages.getTitle());
+ }
+ String overview = episode.getOverview();
+ if (overview.length() > VALUE_MAX_LENGTH) {
+ overview = overview.substring(0, VALUE_MAX_LENGTH);
+ }
+ return new ShowDownloadResponse(new SonarrDownloadActivity(
+ sonarrShow.getTitle() + ": " + episode.getTitle(),
+ episode.getSeasonNumber(),
+ episode.getEpisodeNumber(),
+ showQueue.getQuality().getQuality().getName(),
+ showQueue.getStatus(),
+ showQueue.getTimeleft(),
+ overview,
+ statusMessages.toArray(new String[]{})
+ )
+ );
+ }
+
+ @Override
+ public CommandResponse getResponse(JsonElement rawElement) {
+ SonarrQueue showQueue = new Gson().fromJson(rawElement, SonarrQueue.class);
+ SonarrQueueEpisode episode = showQueue.getEpisode();
+ // api/queue vs /api/v3/queue differ in results which makes the episode object not exist,
+ // so we support both cause I have no idea why I originally used api/queue and if its still in use and will stay in use...
+ //TODO: eventually get rid of the api/queue one when you're sure its not anything of value
+ if (episode == null) {
+ return getDownloadBasedApiV3Queue(showQueue);
+ }
+ return getDownloadBasedApiQueue(showQueue);
+ }
+
+ @Override
+ public List getContentDownloads() {
+ return ConnectionHelper.makeGetRequest(
+ new SonarrUrls.SonarrRequestBuilder().buildGet(SonarrUrls.DOWNLOAD_BASE),
+ new ConnectionHelper.SimpleCommandResponseHandler() {
+ @Override
+ public List onSuccess(String response) {
+ if (response == null || response.isEmpty() || response.equals("{}")) {
+ return new ArrayList<>();
+ }
+ JsonObject json = JsonParser.parseString(response).getAsJsonObject();
+ return parseContent(json.get("records").toString());
+ }
+ });
}
};
}
@@ -219,12 +312,11 @@ private CommandResponse addShow(SonarrShow sonarrShow) {
return new ErrorResponse("Could not add show, user " + username + " has exceeded max show requests for " + requestThreshold.getReadableName());
}
try (CloseableHttpClient client = HttpClientBuilder.create().build()) {
- HttpPost post = new HttpPost(getApiUrl(SonarrUrls.SERIES_BASE));
+ String json = new GsonBuilder().addSerializationExclusionStrategy(excludeUnnecessaryFields).create().toJson(sonarrShow, SonarrShow.class);
+ HttpRequestBase post = new SonarrUrls.SonarrRequestBuilder().buildPost(SonarrUrls.SERIES_BASE, json).build();
+ //TODO: why isn't the content type json
post.addHeader("content-type", "application/x-www-form-urlencoded");
- post.setEntity(
- new StringEntity(
- new GsonBuilder().addSerializationExclusionStrategy(excludeUnnecessaryFields).create().toJson(sonarrShow, SonarrShow.class), Charset.forName("UTF-8")));
try (CloseableHttpResponse response = client.execute(post)) {
int statusCode = response.getStatusLine().getStatusCode();
@@ -244,16 +336,20 @@ private CommandResponse addShow(SonarrShow sonarrShow) {
}
private List lookupShows(String search) throws Exception {
- return ConnectionHelper.makeGetRequest(this, SonarrUrls.LOOKUP_SERIES, "&term=" + URLEncoder.encode(search, "UTF-8"), new ConnectionHelper.SimpleEntityResponseHandler>() {
+ return ConnectionHelper.makeGetRequest(
+ new SonarrUrls.SonarrRequestBuilder().buildGet(SonarrUrls.LOOKUP_SERIES, new HashMap() {{
+ put("term", search);
+ }}),
+ new ConnectionHelper.SimpleEntityResponseHandler>() {
@Override
public List onSuccess(String response) {
- List movies = new ArrayList<>();
+ List shows = new ArrayList<>();
JsonParser parser = new JsonParser();
JsonArray json = parser.parse(response).getAsJsonArray();
for (int i = 0; i < json.size(); i++) {
- movies.add(new Gson().fromJson(json.get(i), SonarrShow.class));
+ shows.add(new Gson().fromJson(json.get(i), SonarrShow.class));
}
- return movies;
+ return shows;
}
});
}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrCache.java b/src/main/java/com/botdarr/api/sonarr/SonarrCache.java
index cf44e2f..16860cc 100644
--- a/src/main/java/com/botdarr/api/sonarr/SonarrCache.java
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrCache.java
@@ -12,6 +12,22 @@ public SonarrShow getExistingShowFromSonarrId(long sonarrId) {
return existingSonarrIdsToShows.get(sonarrId);
}
+ public SonarrEpisodeInformation getEpisode(long seriesId, long episodeNumber) {
+ if (existingSeriesToEpisodes.get(seriesId) == null) {
+ return null;
+ }
+ return existingSeriesToEpisodes.get(seriesId).get(episodeNumber);
+ }
+
+ public void addEpisode(long seriesId, long episodeNumber, SonarrEpisodeInformation sonarrEpisodeInformation) {
+ Map episodeInformationMap = existingSeriesToEpisodes.get(seriesId);
+ if (episodeInformationMap == null) {
+ episodeInformationMap = new ConcurrentHashMap<>();
+ existingSeriesToEpisodes.put(seriesId, episodeInformationMap);
+ }
+ episodeInformationMap.put(episodeNumber, sonarrEpisodeInformation);
+ }
+
public boolean doesShowExist(String title) {
return existingShowTitlesToSonarrId.containsKey(title.toLowerCase());
}
@@ -51,10 +67,12 @@ public void removeDeletedShows(List addUpdatedTvdbShowIds) {
existingShowTitlesToSonarrId.keySet().retainAll(existingShowTitles);
existingTvdbIdsToShows.keySet().retainAll(addUpdatedTvdbShowIds);
existingSonarrIdsToShows.keySet().retainAll(existingShowIds);
+ existingSeriesToEpisodes.keySet().retainAll(existingShowIds);
}
private Map existingProfiles = new ConcurrentHashMap<>();
private Map existingShowTitlesToSonarrId = new ConcurrentHashMap<>();
private Map existingTvdbIdsToShows = new ConcurrentHashMap<>();
private Map existingSonarrIdsToShows = new ConcurrentHashMap<>();
+ private Map> existingSeriesToEpisodes = new ConcurrentHashMap<>();
}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrDownloadActivity.java b/src/main/java/com/botdarr/api/sonarr/SonarrDownloadActivity.java
new file mode 100644
index 0000000..9b70fb3
--- /dev/null
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrDownloadActivity.java
@@ -0,0 +1,62 @@
+package com.botdarr.api.sonarr;
+
+public class SonarrDownloadActivity {
+ public SonarrDownloadActivity(String title,
+ long seasonNumber,
+ long episodeNumber,
+ String qualityProfileName,
+ String status,
+ String timeleft,
+ String overview,
+ String[] statusMessages) {
+ this.title = title;
+ this.seasonNumber = seasonNumber;
+ this.episodeNumber = episodeNumber;
+ this.qualityProfileName = qualityProfileName;
+ this.status = status;
+ this.timeleft = timeleft;
+ this.statusMessages = statusMessages;
+ this.overview = overview;
+ }
+
+ public String getTitle() {
+ return title;
+ }
+
+ public long getSeasonNumber() {
+ return seasonNumber;
+ }
+
+ public long getEpisodeNumber() {
+ return episodeNumber;
+ }
+
+ public String getQualityProfileName() {
+ return qualityProfileName;
+ }
+
+ public String getStatus() {
+ return status;
+ }
+
+ public String getTimeleft() {
+ return timeleft;
+ }
+
+ public String[] getStatusMessages() {
+ return statusMessages;
+ }
+
+ public String getOverview() {
+ return overview;
+ }
+
+ private final String title;
+ private final String overview;
+ private final long seasonNumber;
+ private final long episodeNumber;
+ private final String qualityProfileName;
+ private final String status;
+ private final String timeleft;
+ private final String[] statusMessages;
+}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrEpisodeInformation.java b/src/main/java/com/botdarr/api/sonarr/SonarrEpisodeInformation.java
new file mode 100644
index 0000000..dc89c8e
--- /dev/null
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrEpisodeInformation.java
@@ -0,0 +1,31 @@
+package com.botdarr.api.sonarr;
+
+public class SonarrEpisodeInformation {
+ public SonarrEpisodeInformation(long seasonNumber, long episodeNumber, String title, String overview) {
+ this.seasonNumber = seasonNumber;
+ this.episodeNumber = episodeNumber;
+ this.title = title;
+ this.overview = overview;
+ }
+
+ public long getSeasonNumber() {
+ return seasonNumber;
+ }
+
+ public long getEpisodeNumber() {
+ return episodeNumber;
+ }
+
+ public String getTitle() {
+ return title;
+ }
+
+ public String getOverview() {
+ return overview;
+ }
+
+ private final long seasonNumber;
+ private final long episodeNumber;
+ private final String title;
+ private final String overview;
+}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrProfile.java b/src/main/java/com/botdarr/api/sonarr/SonarrProfile.java
index 0c1302e..fdca442 100644
--- a/src/main/java/com/botdarr/api/sonarr/SonarrProfile.java
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrProfile.java
@@ -14,7 +14,7 @@ public String getName() {
return name;
}
- public SonarrProfileCutoff getCutoff() {
+ public Integer getCutoff() {
return cutoff;
}
@@ -22,7 +22,7 @@ public void setName(String name) {
this.name = name;
}
- public void setCutoff(SonarrProfileCutoff cutoff) {
+ public void setCutoff(Integer cutoff) {
this.cutoff = cutoff;
}
@@ -43,7 +43,7 @@ public void setId(long id) {
}
private String name;
- private SonarrProfileCutoff cutoff;
+ private Integer cutoff;
private List items;
private long id;
}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrProfileQualityItem.java b/src/main/java/com/botdarr/api/sonarr/SonarrProfileQualityItem.java
index 97e84b8..4fc5748 100644
--- a/src/main/java/com/botdarr/api/sonarr/SonarrProfileQualityItem.java
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrProfileQualityItem.java
@@ -1,5 +1,8 @@
package com.botdarr.api.sonarr;
+import java.util.ArrayList;
+import java.util.List;
+
public class SonarrProfileQualityItem {
public SonarrProfileQuality getQuality() {
return quality;
@@ -17,6 +20,24 @@ public void setAllowed(boolean allowed) {
this.allowed = allowed;
}
+ public List getItems() {
+ return items;
+ }
+
+ public void setItems(List items) {
+ this.items = items;
+ }
+
+ public int getId() {
+ return id;
+ }
+
+ public void setId(int id) {
+ this.id = id;
+ }
+
+ private int id;
+ private List items = new ArrayList<>();
private SonarrProfileQuality quality;
private boolean allowed;
}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrQueue.java b/src/main/java/com/botdarr/api/sonarr/SonarrQueue.java
index 4f6bd03..a99a1bc 100644
--- a/src/main/java/com/botdarr/api/sonarr/SonarrQueue.java
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrQueue.java
@@ -46,23 +46,41 @@ public SonarrQueueShow getSonarrQueueShow() {
return series;
}
- public void setRadarrQueueMovie(SonarrQueueShow radarrQueueMovie) {
- this.series = radarrQueueMovie;
+ public void setSeries(SonarrQueueShow series) {
+ this.series = series;
}
- public SonarQueueEpisode getEpisode() {
+ public SonarrQueueEpisode getEpisode() {
return episode;
}
- public void setEpisode(SonarQueueEpisode episode) {
+ public void setEpisode(SonarrQueueEpisode episode) {
this.episode = episode;
}
- private String status;
- private String timeleft;
+ public long getSeriesId() {
+ return seriesId;
+ }
+
+ public void setSeriesId(long seriesId) {
+ this.seriesId = seriesId;
+ }
+
+ public long getEpisodeId() {
+ return episodeId;
+ }
+
+ public void setEpisodeId(long episodeId) {
+ this.episodeId = episodeId;
+ }
+
private SonarrProfileQualityItem quality;
- private long id;
private SonarrQueueStatusMessages[] statusMessages;
private SonarrQueueShow series;
- private SonarQueueEpisode episode;
+ private SonarrQueueEpisode episode;
+ private String status;
+ private String timeleft;
+ private long seriesId;
+ private long episodeId;
+ private long id;
}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarQueueEpisode.java b/src/main/java/com/botdarr/api/sonarr/SonarrQueueEpisode.java
similarity index 84%
rename from src/main/java/com/botdarr/api/sonarr/SonarQueueEpisode.java
rename to src/main/java/com/botdarr/api/sonarr/SonarrQueueEpisode.java
index 268361b..ae40a55 100644
--- a/src/main/java/com/botdarr/api/sonarr/SonarQueueEpisode.java
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrQueueEpisode.java
@@ -1,6 +1,6 @@
package com.botdarr.api.sonarr;
-public class SonarQueueEpisode {
+public class SonarrQueueEpisode {
public int getSeasonNumber() {
return seasonNumber;
}
@@ -37,9 +37,18 @@ public long getSeriesId() {
return seriesId;
}
+ public long getId() {
+ return id;
+ }
+
+ public void setId(long id) {
+ this.id = id;
+ }
+
private int seasonNumber;
private int episodeNumber;
private String title;
private String overview;
private long seriesId;
+ private long id;
}
diff --git a/src/main/java/com/botdarr/api/sonarr/SonarrUrls.java b/src/main/java/com/botdarr/api/sonarr/SonarrUrls.java
index 3812e06..129f709 100644
--- a/src/main/java/com/botdarr/api/sonarr/SonarrUrls.java
+++ b/src/main/java/com/botdarr/api/sonarr/SonarrUrls.java
@@ -1,9 +1,32 @@
package com.botdarr.api.sonarr;
+import com.botdarr.Config;
+import com.botdarr.api.ArrRequestBuilder;
+import org.apache.logging.log4j.util.Strings;
+
public class SonarrUrls {
+ public static class SonarrRequestBuilder extends ArrRequestBuilder {
+ public SonarrRequestBuilder() {
+ super(Config.Constants.SONARR_URL, Config.Constants.SONARR_URL_BASE, Config.Constants.SONARR_TOKEN);
+ }
+
+ //TODO: do i even need this?
+ private boolean isV4Enabled() {
+ String isSonarrV4 = Config.getProperty(Config.Constants.SONARR_V4);
+ return !Strings.isEmpty(isSonarrV4) && Boolean.parseBoolean(isSonarrV4);
+ }
+
+ @Override
+ public String getApiSuffix() {
+ if (isV4Enabled()) {
+ return "/api/v3/";
+ }
+ return super.getApiSuffix();
+ }
+ }
public static final String DOWNLOAD_BASE = "queue";
public static final String LOOKUP_SERIES = "series/lookup";
public static final String SERIES_BASE = "series";
- public static final String PROFILE = "profile";
- public static final String ARTIST_BASE = "v1/artist";
+ public static final String PROFILE = "qualityprofile";
+ public static final String EPISODES_LOOKUP = "episode";
}
diff --git a/src/main/java/com/botdarr/clients/discord/DiscordResponseBuilder.java b/src/main/java/com/botdarr/clients/discord/DiscordResponseBuilder.java
index b73b596..9724a87 100644
--- a/src/main/java/com/botdarr/clients/discord/DiscordResponseBuilder.java
+++ b/src/main/java/com/botdarr/clients/discord/DiscordResponseBuilder.java
@@ -31,7 +31,6 @@
import static com.botdarr.api.sonarr.SonarrApi.SHOW_LOOKUP_FIELD;
import static com.botdarr.commands.StatusCommand.STATUS_COMMAND;
import static com.botdarr.commands.StatusCommand.STATUS_COMMAND_DESCRIPTION;
-import static net.dv8tion.jda.api.entities.MessageEmbed.VALUE_MAX_LENGTH;
public class DiscordResponseBuilder implements ChatClientResponseBuilder {
@Override
@@ -107,23 +106,15 @@ public DiscordResponse build(MusicArtistResponse musicArtistResponse) {
@Override
public DiscordResponse build(ShowDownloadResponse showDownloadResponse) {
EmbedBuilder embedBuilder = new EmbedBuilder();
- SonarrQueue showQueue = showDownloadResponse.getShowQueue();
- SonarQueueEpisode episode = showQueue.getEpisode();
- embedBuilder.setTitle(showQueue.getSonarrQueueShow().getTitle());
- embedBuilder.addField("Season/Episode", "S" + episode.getSeasonNumber() + "E" + episode.getEpisodeNumber(), true);
- embedBuilder.addField("Quality", showQueue.getQuality().getQuality().getName(), true);
- embedBuilder.addField("Status", showQueue.getStatus(), true);
- embedBuilder.addField("Time Left", showQueue.getTimeleft() == null ? "unknown" : showQueue.getTimeleft(), true);
- String overview = episode.getTitle() + ": " + episode.getOverview();
- if (overview.length() > VALUE_MAX_LENGTH) {
- overview = overview.substring(0, VALUE_MAX_LENGTH);
- }
- embedBuilder.addField("Overview", overview, false);
- if (showQueue.getStatusMessages() != null) {
- for (SonarrQueueStatusMessages statusMessage : showQueue.getStatusMessages()) {
- for (String message : statusMessage.getMessages()) {
- embedBuilder.addField("Download message", message, true);
- }
+ embedBuilder.setTitle(showDownloadResponse.getShowQueue().getTitle());
+ embedBuilder.addField("Season/Episode", "S" + showDownloadResponse.getShowQueue().getSeasonNumber() + "E" + showDownloadResponse.getShowQueue().getEpisodeNumber(), true);
+ embedBuilder.addField("Quality", showDownloadResponse.getShowQueue().getQualityProfileName(), true);
+ embedBuilder.addField("Status", showDownloadResponse.getShowQueue().getStatus(), true);
+ embedBuilder.addField("Time Left", showDownloadResponse.getShowQueue().getTimeleft() == null ? "unknown" : showDownloadResponse.getShowQueue().getTimeleft(), true);
+ embedBuilder.addField("Overview", showDownloadResponse.getShowQueue().getOverview(), true);
+ if (showDownloadResponse.getShowQueue().getStatusMessages() != null) {
+ for (String statusMessage : showDownloadResponse.getShowQueue().getStatusMessages()) {
+ embedBuilder.addField("Download message", statusMessage, true);
}
}
return new DiscordResponse(embedBuilder.build());
@@ -164,21 +155,31 @@ public DiscordResponse build(MusicArtistDownloadResponse musicArtistDownloadResp
return new DiscordResponse(embedBuilder.build());
}
+ private void addQualityItem(EmbedBuilder embedBuilder, SonarrProfileQualityItem sonarrProfileQualityItem) {
+ embedBuilder.addField(
+ "Quality",
+ "name=" + sonarrProfileQualityItem.getQuality().getName() + ", resolution=" + sonarrProfileQualityItem.getQuality().getResolution(),
+ true);
+ }
+
@Override
public DiscordResponse build(ShowProfileResponse showProfileResponse) {
EmbedBuilder embedBuilder = new EmbedBuilder();
embedBuilder.setTitle("Profile");
SonarrProfile sonarrProfile = showProfileResponse.getShowProfile();
embedBuilder.addField("Name", sonarrProfile.getName(), false);
- embedBuilder.addField("Cutoff", sonarrProfile.getCutoff().getName(), false);
+ embedBuilder.addField("Cutoff", "" + sonarrProfile.getCutoff(), false);
embedBuilder.addBlankField(false);
for (int k = 0; k < sonarrProfile.getItems().size(); k++) {
SonarrProfileQualityItem sonarrProfileQualityItem = sonarrProfile.getItems().get(k);
if (sonarrProfileQualityItem.isAllowed()) {
- embedBuilder.addField(
- "Quality",
- "name=" + sonarrProfileQualityItem.getQuality().getName() + ", resolution=" + sonarrProfileQualityItem.getQuality().getResolution(),
- true);
+ if (sonarrProfileQualityItem.getQuality() == null) {
+ for(SonarrProfileQualityItem qualityItem : sonarrProfileQualityItem.getItems()) {
+ addQualityItem(embedBuilder, qualityItem);
+ }
+ } else {
+ addQualityItem(embedBuilder, sonarrProfileQualityItem);
+ }
}
}
return new DiscordResponse(embedBuilder.build());
diff --git a/src/main/java/com/botdarr/clients/matrix/MatrixResponseBuilder.java b/src/main/java/com/botdarr/clients/matrix/MatrixResponseBuilder.java
index 83c003a..407d1c2 100644
--- a/src/main/java/com/botdarr/clients/matrix/MatrixResponseBuilder.java
+++ b/src/main/java/com/botdarr/clients/matrix/MatrixResponseBuilder.java
@@ -104,24 +104,16 @@ public MatrixResponse build(MovieResponse movieResponse) {
@Override
public MatrixResponse build(ShowDownloadResponse showDownloadResponse) {
- SonarrQueue sonarrQueue = showDownloadResponse.getShowQueue();
MatrixResponse matrixResponse = new MatrixResponse();
- SonarQueueEpisode episode = sonarrQueue.getEpisode();
- matrixResponse.addContent("Title - " + sonarrQueue.getSonarrQueueShow().getTitle());
- matrixResponse.addContent("Season/Episode - " + "S" + episode.getSeasonNumber() + "E" + episode.getEpisodeNumber());
- matrixResponse.addContent("Quality - " + sonarrQueue.getQuality().getQuality().getName());
- matrixResponse.addContent("Status - " + sonarrQueue.getStatus());
- matrixResponse.addContent("Time Left - " + (sonarrQueue.getTimeleft() == null ? "unknown" : sonarrQueue.getTimeleft()) + "");
- String overview = episode.getTitle() + ": " + episode.getOverview();
- if (overview.length() > 1024) {
- overview = overview.substring(0, 1024);
- }
- matrixResponse.addContent("Overview - " + overview);
- if (sonarrQueue.getStatusMessages() != null) {
- for (SonarrQueueStatusMessages statusMessage : sonarrQueue.getStatusMessages()) {
- for (String message : statusMessage.getMessages()) {
- matrixResponse.addContent("Download message - " + message);
- }
+ matrixResponse.addContent("Title - " + showDownloadResponse.getShowQueue().getTitle());
+ matrixResponse.addContent("Season/Episode - " + "S" + showDownloadResponse.getShowQueue().getSeasonNumber() + "E" + showDownloadResponse.getShowQueue().getEpisodeNumber());
+ matrixResponse.addContent("Quality - " + showDownloadResponse.getShowQueue().getQualityProfileName());
+ matrixResponse.addContent("Status - " + showDownloadResponse.getShowQueue().getStatus());
+ matrixResponse.addContent("Time Left - " + (showDownloadResponse.getShowQueue().getTimeleft() == null ? "unknown" : showDownloadResponse.getShowQueue().getTimeleft()) + "");
+ matrixResponse.addContent("Overview - " + showDownloadResponse.getShowQueue().getOverview());
+ if (showDownloadResponse.getShowQueue().getStatusMessages() != null) {
+ for (String statusMessage : showDownloadResponse.getShowQueue().getStatusMessages()) {
+ matrixResponse.addContent("Download message - " + statusMessage);
}
}
return matrixResponse;
@@ -183,17 +175,27 @@ public MatrixResponse build(SuccessResponse successResponse) {
return matrixResponse;
}
+ private String addQualityItem(SonarrProfileQualityItem sonarrProfileQualityItem) {
+ return "Quality - Name: " + sonarrProfileQualityItem.getQuality().getName() + ", Resolution: " + sonarrProfileQualityItem.getQuality().getResolution();
+ }
+
@Override
public MatrixResponse build(ShowProfileResponse showProfileResponse) {
SonarrProfile sonarrProfile = showProfileResponse.getShowProfile();
MatrixResponse matrixResponse = new MatrixResponse();
matrixResponse.addContent("Profile");
matrixResponse.addContent("Name - " + sonarrProfile.getName());
- matrixResponse.addContent("Cutoff - " + sonarrProfile.getCutoff().getName());
+ matrixResponse.addContent("Cutoff - " + sonarrProfile.getCutoff());
for (int k = 0; k < sonarrProfile.getItems().size(); k++) {
SonarrProfileQualityItem sonarrProfileQualityItem = sonarrProfile.getItems().get(k);
if (sonarrProfileQualityItem.isAllowed()) {
- matrixResponse.addContent("Quality - Name: " + sonarrProfileQualityItem.getQuality().getName() + ", Resolution: " + sonarrProfileQualityItem.getQuality().getResolution());
+ if (sonarrProfileQualityItem.getQuality() == null) {
+ for(SonarrProfileQualityItem qualityItem : sonarrProfileQualityItem.getItems()) {
+ matrixResponse.addContent(addQualityItem(qualityItem));
+ }
+ } else {
+ matrixResponse.addContent(addQualityItem(sonarrProfileQualityItem));
+ }
}
}
return matrixResponse;
diff --git a/src/main/java/com/botdarr/clients/slack/SlackResponseBuilder.java b/src/main/java/com/botdarr/clients/slack/SlackResponseBuilder.java
index d1b3776..efcb8d4 100644
--- a/src/main/java/com/botdarr/clients/slack/SlackResponseBuilder.java
+++ b/src/main/java/com/botdarr/clients/slack/SlackResponseBuilder.java
@@ -25,7 +25,6 @@
import static com.botdarr.api.sonarr.SonarrApi.ADD_SHOW_COMMAND_FIELD_PREFIX;
import static com.botdarr.commands.StatusCommand.STATUS_COMMAND;
import static com.botdarr.commands.StatusCommand.STATUS_COMMAND_DESCRIPTION;
-import static net.dv8tion.jda.api.entities.MessageEmbed.VALUE_MAX_LENGTH;
public class SlackResponseBuilder implements ChatClientResponseBuilder {
@Override
@@ -146,17 +145,16 @@ public SlackResponse build(MusicArtistResponse musicArtistResponse) {
@Override
public SlackResponse build(ShowDownloadResponse showDownloadResponse) {
- SonarrQueue showQueue = showDownloadResponse.getShowQueue();
- SonarQueueEpisode episode = showQueue.getEpisode();
+ SonarrDownloadActivity showQueue = showDownloadResponse.getShowQueue();
SlackResponse slackResponse = new SlackResponse();
slackResponse.addBlock(SectionBlock.builder()
- .text(MarkdownTextObject.builder().text("*Title* - " + showQueue.getSonarrQueueShow().getTitle()).build())
+ .text(MarkdownTextObject.builder().text("*Title* - " + showQueue.getTitle()).build())
.build());
slackResponse.addBlock(SectionBlock.builder()
- .text(MarkdownTextObject.builder().text("Season/Episode - " + "S" + episode.getSeasonNumber() + "E" + episode.getEpisodeNumber()).build())
+ .text(MarkdownTextObject.builder().text("Season/Episode - " + "S" + showQueue.getSeasonNumber() + "E" + showQueue.getEpisodeNumber()).build())
.build());
slackResponse.addBlock(SectionBlock.builder()
- .text(MarkdownTextObject.builder().text("Quality - " + showQueue.getQuality().getQuality().getName()).build())
+ .text(MarkdownTextObject.builder().text("Quality - " + showQueue.getQualityProfileName()).build())
.build());
slackResponse.addBlock(SectionBlock.builder()
.text(MarkdownTextObject.builder().text("Status - " + showQueue.getStatus()).build())
@@ -164,19 +162,13 @@ public SlackResponse build(ShowDownloadResponse showDownloadResponse) {
slackResponse.addBlock(SectionBlock.builder()
.text(MarkdownTextObject.builder().text("Time Left - *" + (showQueue.getTimeleft() == null ? "unknown" : showQueue.getTimeleft()) + "*").build())
.build());
- String overview = episode.getTitle() + ": " + episode.getOverview();
- if (overview.length() > VALUE_MAX_LENGTH) {
- overview = overview.substring(0, VALUE_MAX_LENGTH);
- }
slackResponse.addBlock(SectionBlock.builder()
- .text(MarkdownTextObject.builder().text("Overview - " + overview).build())
- .build());
+ .text(MarkdownTextObject.builder().text("Overview - " + showQueue.getOverview()).build())
+ .build());
if (showQueue.getStatusMessages() != null) {
List contextBlockElements = new ArrayList<>();
- for (SonarrQueueStatusMessages statusMessage : showQueue.getStatusMessages()) {
- for (String message : statusMessage.getMessages()) {
- contextBlockElements.add(PlainTextObject.builder().text(message).build());
- }
+ for (String statusMessage : showQueue.getStatusMessages()) {
+ contextBlockElements.add(PlainTextObject.builder().text(statusMessage).build());
}
if (showQueue.getStatusMessages() != null && showQueue.getStatusMessages().length > 0) {
slackResponse.addBlock(ContextBlock.builder()
@@ -275,6 +267,10 @@ public SlackResponse build(SuccessResponse successResponse) {
return slackResponse;
}
+ private String addQualityItem(SonarrProfileQualityItem sonarrProfileQualityItem) {
+ return "Quality - name=" + sonarrProfileQualityItem.getQuality().getName() + ", resolution=" + sonarrProfileQualityItem.getQuality().getResolution();
+ }
+
@Override
public SlackResponse build(ShowProfileResponse showProfileResponse) {
SonarrProfile sonarrProfile = showProfileResponse.getShowProfile();
@@ -286,16 +282,24 @@ public SlackResponse build(ShowProfileResponse showProfileResponse) {
.text(MarkdownTextObject.builder().text("Name - " + sonarrProfile.getName()).build())
.build());
slackResponse.addBlock(SectionBlock.builder()
- .text(MarkdownTextObject.builder().text("Cutoff - " + sonarrProfile.getCutoff().getName()).build())
+ .text(MarkdownTextObject.builder().text("Cutoff - " + sonarrProfile.getCutoff()).build())
.build());
List contextBlockElements = new ArrayList<>();
for (int k = 0; k < sonarrProfile.getItems().size(); k++) {
SonarrProfileQualityItem sonarrProfileQualityItem = sonarrProfile.getItems().get(k);
if (sonarrProfileQualityItem.isAllowed()) {
- contextBlockElements.add(PlainTextObject.builder()
- .text("Quality - name=" + sonarrProfileQualityItem.getQuality().getName() + ", resolution=" + sonarrProfileQualityItem.getQuality().getResolution())
- .build());
+ if (sonarrProfileQualityItem.getQuality() == null) {
+ for(SonarrProfileQualityItem qualityItem : sonarrProfileQualityItem.getItems()) {
+ contextBlockElements.add(PlainTextObject.builder()
+ .text(addQualityItem(qualityItem))
+ .build());
+ }
+ } else {
+ contextBlockElements.add(PlainTextObject.builder()
+ .text(addQualityItem(sonarrProfileQualityItem))
+ .build());
+ }
}
}
slackResponse.addBlock(ContextBlock.builder()
diff --git a/src/main/java/com/botdarr/clients/telegram/TelegramResponseBuilder.java b/src/main/java/com/botdarr/clients/telegram/TelegramResponseBuilder.java
index 06b3da8..8a54a45 100644
--- a/src/main/java/com/botdarr/clients/telegram/TelegramResponseBuilder.java
+++ b/src/main/java/com/botdarr/clients/telegram/TelegramResponseBuilder.java
@@ -24,7 +24,6 @@
import static com.botdarr.commands.StatusCommand.STATUS_COMMAND;
import static com.botdarr.commands.StatusCommand.STATUS_COMMAND_DESCRIPTION;
import static j2html.TagCreator.*;
-import static net.dv8tion.jda.api.entities.MessageEmbed.VALUE_MAX_LENGTH;
import java.io.IOException;
import java.util.ArrayList;
@@ -101,28 +100,19 @@ public TelegramResponse build(MusicArtistResponse musicArtistResponse) {
@Override
public TelegramResponse build(ShowDownloadResponse showDownloadResponse) {
- SonarrQueue showQueue = showDownloadResponse.getShowQueue();
- SonarQueueEpisode episode = showQueue.getEpisode();
-
List domContents = new ArrayList<>();
- domContents.add(b(showQueue.getSonarrQueueShow().getTitle()));
- String queueDetails = "Season/Episode - " + "S" + episode.getSeasonNumber() + "E" + episode.getEpisodeNumber() + "\n" +
- "Quality - " + showQueue.getQuality().getQuality().getName() + "\n" +
- "Status - " + showQueue.getStatus() + "\n" +
- "Time Left - " + (showQueue.getTimeleft() == null ? "unknown" : showQueue.getTimeleft()) + "\n";
+ domContents.add(b(showDownloadResponse.getShowQueue().getTitle()));
+ String queueDetails = "Season/Episode - " + "S" + showDownloadResponse.getShowQueue().getSeasonNumber() + "E" + showDownloadResponse.getShowQueue().getEpisodeNumber() + "\n" +
+ "Quality - " + showDownloadResponse.getShowQueue().getQualityProfileName() + "\n" +
+ "Status - " + showDownloadResponse.getShowQueue().getStatus() + "\n" +
+ "Time Left - " + (showDownloadResponse.getShowQueue().getTimeleft() == null ? "unknown" : showDownloadResponse.getShowQueue().getTimeleft()) + "\n" +
+ "Overview - " + showDownloadResponse.getShowQueue().getOverview() + "\n";
domContents.add(code(queueDetails));
- String overview = episode.getTitle() + ": " + episode.getOverview();
- if (overview.length() > VALUE_MAX_LENGTH) {
- overview = overview.substring(0, VALUE_MAX_LENGTH);
- }
- domContents.add(b("Overview - " + overview));
- if (showQueue.getStatusMessages() != null) {
+ if (showDownloadResponse.getShowQueue().getStatusMessages() != null) {
StringBuilder statusMessageBuilder = new StringBuilder();
- for (SonarrQueueStatusMessages statusMessage : showQueue.getStatusMessages()) {
- for (String message : statusMessage.getMessages()) {
- statusMessageBuilder.append("Download Message - ").append(message).append("\n");
- }
+ for (String statusMessage : showDownloadResponse.getShowQueue().getStatusMessages()) {
+ statusMessageBuilder.append("Download Message - ").append(statusMessage).append("\n");
}
domContents.add(code(statusMessageBuilder.toString()));
}
@@ -190,19 +180,38 @@ public TelegramResponse build(SuccessResponse successResponse) {
return new TelegramResponse(domContents);
}
+ private StringBuilder appendQualityItem(SonarrProfileQualityItem qualityItem) {
+ StringBuilder qualityItems = new StringBuilder();
+ qualityItems.append("Quality - ")
+ .append("id=")
+ .append(qualityItem.getQuality().getId())
+ .append(", name=")
+ .append(qualityItem.getQuality().getName())
+ .append(", resolution=")
+ .append(qualityItem.getQuality().getResolution())
+ .append("\n");
+ return qualityItems;
+ }
+
@Override
public TelegramResponse build(ShowProfileResponse showProfileResponse) {
SonarrProfile sonarrProfile = showProfileResponse.getShowProfile();
List domContents = new ArrayList<>();
domContents.add(b("Profile"));
domContents.add(text("Name - " + sonarrProfile.getName()));
- domContents.add(text("Cutoff - " + sonarrProfile.getCutoff().getName()));
+ domContents.add(text("Cutoff - " + sonarrProfile.getCutoff()));
if (sonarrProfile.getItems() != null) {
StringBuilder qualityItems = new StringBuilder();
for (int k = 0; k < sonarrProfile.getItems().size(); k++) {
SonarrProfileQualityItem sonarrProfileQualityItem = sonarrProfile.getItems().get(k);
if (sonarrProfileQualityItem.isAllowed()) {
- qualityItems.append("Quality - name=").append(sonarrProfileQualityItem.getQuality().getName()).append(", resolution=").append(sonarrProfileQualityItem.getQuality().getResolution()).append("\n");
+ if (sonarrProfileQualityItem.getQuality() == null) {
+ for(SonarrProfileQualityItem qualityItem : sonarrProfileQualityItem.getItems()) {
+ qualityItems.append(appendQualityItem(qualityItem));
+ }
+ } else {
+ qualityItems.append(appendQualityItem(sonarrProfileQualityItem));
+ }
}
}
domContents.add(code(qualityItems.toString()));
diff --git a/src/main/java/com/botdarr/commands/responses/ShowDownloadResponse.java b/src/main/java/com/botdarr/commands/responses/ShowDownloadResponse.java
index 8ead06b..d8cfebf 100644
--- a/src/main/java/com/botdarr/commands/responses/ShowDownloadResponse.java
+++ b/src/main/java/com/botdarr/commands/responses/ShowDownloadResponse.java
@@ -1,15 +1,15 @@
package com.botdarr.commands.responses;
-import com.botdarr.api.sonarr.SonarrQueue;
+import com.botdarr.api.sonarr.SonarrDownloadActivity;
import com.botdarr.clients.ChatClientResponse;
import com.botdarr.clients.ChatClientResponseBuilder;
public class ShowDownloadResponse implements CommandResponse {
- private final SonarrQueue showQueue;
- public ShowDownloadResponse(SonarrQueue showQueue) {
+ private final SonarrDownloadActivity showQueue;
+ public ShowDownloadResponse(SonarrDownloadActivity showQueue) {
this.showQueue = showQueue;
}
- public SonarrQueue getShowQueue() {
+ public SonarrDownloadActivity getShowQueue() {
return showQueue;
}
diff --git a/src/main/java/com/botdarr/connections/ConnectionHelper.java b/src/main/java/com/botdarr/connections/ConnectionHelper.java
index 08d2edb..d017d30 100644
--- a/src/main/java/com/botdarr/connections/ConnectionHelper.java
+++ b/src/main/java/com/botdarr/connections/ConnectionHelper.java
@@ -1,6 +1,5 @@
package com.botdarr.connections;
-import com.botdarr.api.Api;
import com.botdarr.Config;
import com.botdarr.commands.responses.CommandResponse;
import com.botdarr.commands.responses.ErrorResponse;
@@ -18,17 +17,11 @@
import java.util.List;
public class ConnectionHelper {
- public static T makeGetRequest(Api api, String path, ResponseHandler responseHandler) {
- return makeGetRequest(api, path, "", responseHandler);
- }
-
- public static T makeGetRequest(Api api, String path, String params, ResponseHandler responseHandler) {
+ public static T makeGetRequest(RequestBuilder builder, ResponseHandler responseHandler) {
return makeRequest(new RequestHandler() {
@Override
public HttpRequestBase buildRequest() throws Exception {
- HttpGet get = new HttpGet(api.getApiUrl(path) + params);
- get.setHeader("X-Api-Key", Config.getProperty(api.getApiToken()));
- return get;
+ return builder.build();
}
@Override
@@ -58,6 +51,7 @@ public static T makeRequest(RequestHandler requestHandler, ResponseHandler entry : headers.entrySet()) {
+ post.addHeader(entry.getKey(), entry.getValue());
+ }
+ post.addHeader("content-type", "application/json");
+ post.setEntity(new StringEntity(postBody, StandardCharsets.UTF_8));
+ return post;
+ }
+ HttpGet get = new HttpGet(builder.build());
+ for (Map.Entry entry : headers.entrySet()) {
+ get.addHeader(entry.getKey(), entry.getValue());
+ }
+ return get;
+ } catch (URISyntaxException e) {
+ LOGGER.error("Error trying to build get request", e);
+ }
+ throw new RuntimeException("Could not build get or post request");
+ }
+ private static final Logger LOGGER = LogManager.getLogger();
+}
diff --git a/src/main/resources/version.txt b/src/main/resources/version.txt
index c7ba1e8..4cc0e35 100644
--- a/src/main/resources/version.txt
+++ b/src/main/resources/version.txt
@@ -1 +1 @@
-5.5.0
\ No newline at end of file
+5.6.0
\ No newline at end of file
diff --git a/src/test/java/com/botdarr/api/DownloadsStrategyTests.java b/src/test/java/com/botdarr/api/DownloadsStrategyTests.java
index 833ebdd..dd5191e 100644
--- a/src/test/java/com/botdarr/api/DownloadsStrategyTests.java
+++ b/src/test/java/com/botdarr/api/DownloadsStrategyTests.java
@@ -4,6 +4,7 @@
import com.botdarr.TestCommandResponse;
import com.botdarr.commands.responses.CommandResponse;
import com.botdarr.commands.responses.InfoResponse;
+import com.botdarr.connections.RequestBuilder;
import com.google.gson.JsonElement;
import mockit.*;
import org.apache.commons.lang3.builder.EqualsBuilder;
@@ -36,6 +37,7 @@ public void beforeEachTest() throws Exception {
properties.put("telegram-token", "%H$$54j45i");
properties.put("telegram-private-channels", "channel1:459349");
writeFakePropertiesFile(properties);
+ requestBuilder = new RequestBuilder().host("http://localhost");
}
@Test
@@ -71,27 +73,8 @@ public void parseContent_tooManyDownloads_infoMessageIncluded() {
responses.get(0)));
}
- @Test
- public void getContentDownloads_endpointUnavailable_emptyResponse() {
- DownloadsStrategy downloadsStrategy = new DownloadsStrategy(api, "") {
- @Override
- public CommandResponse getResponse(JsonElement rawElement) {
- return null;
- }
- };
- Deencapsulation.setField(downloadsStrategy, "LOGGER", logger);
- new Expectations(downloadsStrategy) {{
- api.getApiUrl(""); result = "http://localhost"; times = 2;
- api.getApiToken(); result = "token1"; times = 1;
- logger.error("Error trying to connect to http://localhost"); times = 1;
- }};
-
- //confirm even tho we failed to connect we don't return error notifications that could flood chat clients
- Assert.assertTrue(downloadsStrategy.getContentDownloads().isEmpty());
- }
-
private DownloadsStrategy getMockDownloadsStrategy(int maxDownloadsToShow) {
- DownloadsStrategy mockDownloadsStrategy = new MockDownloadsStrategy(api, "");
+ DownloadsStrategy mockDownloadsStrategy = new MockDownloadsStrategy();
Deencapsulation.setField(mockDownloadsStrategy, "MAX_DOWNLOADS_TO_SHOW", maxDownloadsToShow);
return mockDownloadsStrategy;
}
@@ -113,15 +96,10 @@ private void writeFakePropertiesFile(Properties properties) throws Exception {
@Mocked
private Logger logger;
- @Injectable
- private Api api;
+ private RequestBuilder requestBuilder;
private static class MockDownloadsStrategy extends DownloadsStrategy {
- public MockDownloadsStrategy(Api api, String url) {
- super(api, url);
- }
-
@Override
public CommandResponse getResponse(JsonElement rawElement) {
return null;
diff --git a/src/test/java/com/botdarr/api/radarr/RadarrApiTests.java b/src/test/java/com/botdarr/api/radarr/RadarrApiTests.java
index 29d4159..6cc2dd1 100644
--- a/src/test/java/com/botdarr/api/radarr/RadarrApiTests.java
+++ b/src/test/java/com/botdarr/api/radarr/RadarrApiTests.java
@@ -9,6 +9,7 @@
import org.junit.*;
import org.junit.rules.TemporaryFolder;
import org.mockserver.junit.MockServerRule;
+import org.mockserver.model.Header;
import org.mockserver.model.HttpRequest;
import org.mockserver.model.HttpResponse;
import org.mockserver.model.MediaType;
@@ -219,7 +220,10 @@ public void downloads_noDownloadsFound() {
HttpRequest request = HttpRequest.request()
.withMethod("GET")
.withPath("/api/v3/queue")
- .withQueryStringParameter("apiKey", "FSJDkjmf#$Kf3");
+ .withQueryStringParameter("apiKey", "FSJDkjmf#$Kf3")
+ .withHeaders(new ArrayList() {{
+ add(new Header("content-length", "0"));
+ }});
//setup expected response in mock server
mockServerRule.getClient()