From bc635874287b6ef6c856035028c3c0451caff84e Mon Sep 17 00:00:00 2001 From: Qstick Date: Mon, 28 Nov 2022 18:39:01 -0600 Subject: [PATCH] New: Add support for Simplepush notifications MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Closes #1989 Closes #1990 Co-Authored-By: Timm Schäuble (cherry picked from commit 4c7df31070fbd370b26dbcc07131f21eb88d35fc) --- .../Notifications/Simplepush/Simplepush.cs | 73 +++++++++++++++++++ .../Simplepush/SimplepushProxy.cs | 58 +++++++++++++++ .../Simplepush/SimplepushSettings.cs | 33 +++++++++ 3 files changed, 164 insertions(+) create mode 100644 src/NzbDrone.Core/Notifications/Simplepush/Simplepush.cs create mode 100644 src/NzbDrone.Core/Notifications/Simplepush/SimplepushProxy.cs create mode 100644 src/NzbDrone.Core/Notifications/Simplepush/SimplepushSettings.cs diff --git a/src/NzbDrone.Core/Notifications/Simplepush/Simplepush.cs b/src/NzbDrone.Core/Notifications/Simplepush/Simplepush.cs new file mode 100644 index 000000000..0aaa32f24 --- /dev/null +++ b/src/NzbDrone.Core/Notifications/Simplepush/Simplepush.cs @@ -0,0 +1,73 @@ +using System.Collections.Generic; +using FluentValidation.Results; +using NzbDrone.Common.Extensions; + +namespace NzbDrone.Core.Notifications.Simplepush +{ + public class Simplepush : NotificationBase + { + private readonly ISimplepushProxy _proxy; + + public Simplepush(ISimplepushProxy proxy) + { + _proxy = proxy; + } + + public override string Name => "Simplepush"; + public override string Link => "https://simplepush.io/"; + + public override void OnGrab(GrabMessage grabMessage) + { + _proxy.SendNotification(BOOK_GRABBED_TITLE, grabMessage.Message, Settings); + } + + public override void OnReleaseImport(BookDownloadMessage message) + { + _proxy.SendNotification(BOOK_DOWNLOADED_TITLE, message.Message, Settings); + } + + public override void OnAuthorDelete(AuthorDeleteMessage deleteMessage) + { + _proxy.SendNotification(AUTHOR_DELETED_TITLE, deleteMessage.Message, Settings); + } + + public override void OnBookDelete(BookDeleteMessage deleteMessage) + { + _proxy.SendNotification(BOOK_DELETED_TITLE, deleteMessage.Message, Settings); + } + + public override void OnBookFileDelete(BookFileDeleteMessage deleteMessage) + { + _proxy.SendNotification(BOOK_FILE_DELETED_TITLE, deleteMessage.Message, Settings); + } + + public override void OnHealthIssue(HealthCheck.HealthCheck healthCheck) + { + _proxy.SendNotification(HEALTH_ISSUE_TITLE, healthCheck.Message, Settings); + } + + public override void OnDownloadFailure(DownloadFailedMessage message) + { + _proxy.SendNotification(DOWNLOAD_FAILURE_TITLE, message.Message, Settings); + } + + public override void OnImportFailure(BookDownloadMessage message) + { + _proxy.SendNotification(IMPORT_FAILURE_TITLE, message.Message, Settings); + } + + public override void OnApplicationUpdate(ApplicationUpdateMessage updateMessage) + { + _proxy.SendNotification(APPLICATION_UPDATE_TITLE, updateMessage.Message, Settings); + } + + public override ValidationResult Test() + { + var failures = new List(); + + failures.AddIfNotNull(_proxy.Test(Settings)); + + return new ValidationResult(failures); + } + } +} diff --git a/src/NzbDrone.Core/Notifications/Simplepush/SimplepushProxy.cs b/src/NzbDrone.Core/Notifications/Simplepush/SimplepushProxy.cs new file mode 100644 index 000000000..cc2276cee --- /dev/null +++ b/src/NzbDrone.Core/Notifications/Simplepush/SimplepushProxy.cs @@ -0,0 +1,58 @@ +using System; +using FluentValidation.Results; +using NLog; +using NzbDrone.Common.Http; + +namespace NzbDrone.Core.Notifications.Simplepush +{ + public interface ISimplepushProxy + { + void SendNotification(string title, string message, SimplepushSettings settings); + ValidationFailure Test(SimplepushSettings settings); + } + + public class SimplepushProxy : ISimplepushProxy + { + private const string URL = "https://api.simplepush.io/send"; + private readonly IHttpClient _httpClient; + private readonly Logger _logger; + + public SimplepushProxy(IHttpClient httpClient, Logger logger) + { + _httpClient = httpClient; + _logger = logger; + } + + public void SendNotification(string title, string message, SimplepushSettings settings) + { + var requestBuilder = new HttpRequestBuilder(URL).Post(); + + requestBuilder.AddFormParameter("key", settings.Key) + .AddFormParameter("event", settings.Event) + .AddFormParameter("title", title) + .AddFormParameter("msg", message); + + var request = requestBuilder.Build(); + + _httpClient.Post(request); + } + + public ValidationFailure Test(SimplepushSettings settings) + { + try + { + const string title = "Test Notification"; + const string body = "This is a test message from Readarr"; + + SendNotification(title, body, settings); + } + catch (Exception ex) + { + _logger.Error(ex, "Unable to send test message"); + return new ValidationFailure("ApiKey", "Unable to send test message"); + } + + return null; + } + } +} diff --git a/src/NzbDrone.Core/Notifications/Simplepush/SimplepushSettings.cs b/src/NzbDrone.Core/Notifications/Simplepush/SimplepushSettings.cs new file mode 100644 index 000000000..af8b211ef --- /dev/null +++ b/src/NzbDrone.Core/Notifications/Simplepush/SimplepushSettings.cs @@ -0,0 +1,33 @@ +using FluentValidation; +using NzbDrone.Core.Annotations; +using NzbDrone.Core.ThingiProvider; +using NzbDrone.Core.Validation; + +namespace NzbDrone.Core.Notifications.Simplepush +{ + public class SimplepushSettingsValidator : AbstractValidator + { + public SimplepushSettingsValidator() + { + RuleFor(c => c.Key).NotEmpty(); + } + } + + public class SimplepushSettings : IProviderConfig + { + private static readonly SimplepushSettingsValidator Validator = new SimplepushSettingsValidator(); + + [FieldDefinition(0, Label = "Key", Privacy = PrivacyLevel.ApiKey, HelpLink = "https://simplepush.io/features")] + public string Key { get; set; } + + [FieldDefinition(1, Label = "Event", HelpText = "Customize the behavior of push notifications", HelpLink = "https://simplepush.io/features")] + public string Event { get; set; } + + public bool IsValid => !string.IsNullOrWhiteSpace(Key); + + public NzbDroneValidationResult Validate() + { + return new NzbDroneValidationResult(Validator.Validate(this)); + } + } +}