mirror of
https://github.com/jellyfin/jellyfin.git
synced 2024-11-18 03:19:06 -07:00
471 lines
18 KiB
C#
471 lines
18 KiB
C#
using MediaBrowser.Controller;
|
|
using MediaBrowser.Controller.Notifications;
|
|
using MediaBrowser.Model.Logging;
|
|
using MediaBrowser.Model.Notifications;
|
|
using MediaBrowser.Server.Implementations.Persistence;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Data;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace MediaBrowser.Server.Implementations.Notifications
|
|
{
|
|
public class SqliteNotificationsRepository : BaseSqliteRepository, INotificationsRepository
|
|
{
|
|
public SqliteNotificationsRepository(ILogManager logManager, IServerApplicationPaths appPaths, IDbConnector dbConnector) : base(logManager, dbConnector)
|
|
{
|
|
DbFilePath = Path.Combine(appPaths.DataPath, "notifications.db");
|
|
}
|
|
|
|
public event EventHandler<NotificationUpdateEventArgs> NotificationAdded;
|
|
public event EventHandler<NotificationReadEventArgs> NotificationsMarkedRead;
|
|
////public event EventHandler<NotificationUpdateEventArgs> NotificationUpdated;
|
|
|
|
public async Task Initialize()
|
|
{
|
|
using (var connection = await CreateConnection().ConfigureAwait(false))
|
|
{
|
|
string[] queries = {
|
|
|
|
"create table if not exists Notifications (Id GUID NOT NULL, UserId GUID NOT NULL, Date DATETIME NOT NULL, Name TEXT NOT NULL, Description TEXT, Url TEXT, Level TEXT NOT NULL, IsRead BOOLEAN NOT NULL, Category TEXT NOT NULL, RelatedId TEXT, PRIMARY KEY (Id, UserId))",
|
|
"create index if not exists idx_Notifications1 on Notifications(Id)",
|
|
"create index if not exists idx_Notifications2 on Notifications(UserId)"
|
|
};
|
|
|
|
connection.RunQueries(queries, Logger);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the notifications.
|
|
/// </summary>
|
|
/// <param name="query">The query.</param>
|
|
/// <returns>NotificationResult.</returns>
|
|
public NotificationResult GetNotifications(NotificationQuery query)
|
|
{
|
|
var result = new NotificationResult();
|
|
|
|
using (var connection = CreateConnection(true).Result)
|
|
{
|
|
using (var cmd = connection.CreateCommand())
|
|
{
|
|
var clauses = new List<string>();
|
|
|
|
if (query.IsRead.HasValue)
|
|
{
|
|
clauses.Add("IsRead=@IsRead");
|
|
cmd.Parameters.Add(cmd, "@IsRead", DbType.Boolean).Value = query.IsRead.Value;
|
|
}
|
|
|
|
clauses.Add("UserId=@UserId");
|
|
cmd.Parameters.Add(cmd, "@UserId", DbType.Guid).Value = new Guid(query.UserId);
|
|
|
|
var whereClause = " where " + string.Join(" And ", clauses.ToArray());
|
|
|
|
cmd.CommandText = string.Format("select count(Id) from Notifications{0};select Id,UserId,Date,Name,Description,Url,Level,IsRead,Category,RelatedId from Notifications{0} order by IsRead asc, Date desc", whereClause);
|
|
|
|
using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess))
|
|
{
|
|
if (reader.Read())
|
|
{
|
|
result.TotalRecordCount = reader.GetInt32(0);
|
|
}
|
|
|
|
if (reader.NextResult())
|
|
{
|
|
var notifications = GetNotifications(reader);
|
|
|
|
if (query.StartIndex.HasValue)
|
|
{
|
|
notifications = notifications.Skip(query.StartIndex.Value);
|
|
}
|
|
|
|
if (query.Limit.HasValue)
|
|
{
|
|
notifications = notifications.Take(query.Limit.Value);
|
|
}
|
|
|
|
result.Notifications = notifications.ToArray();
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
|
|
public NotificationsSummary GetNotificationsSummary(string userId)
|
|
{
|
|
var result = new NotificationsSummary();
|
|
|
|
using (var connection = CreateConnection(true).Result)
|
|
{
|
|
using (var cmd = connection.CreateCommand())
|
|
{
|
|
cmd.CommandText = "select Level from Notifications where UserId=@UserId and IsRead=@IsRead";
|
|
|
|
cmd.Parameters.Add(cmd, "@UserId", DbType.Guid).Value = new Guid(userId);
|
|
cmd.Parameters.Add(cmd, "@IsRead", DbType.Boolean).Value = false;
|
|
|
|
using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess))
|
|
{
|
|
var levels = new List<NotificationLevel>();
|
|
|
|
while (reader.Read())
|
|
{
|
|
levels.Add(GetLevel(reader, 0));
|
|
}
|
|
|
|
result.UnreadCount = levels.Count;
|
|
|
|
if (levels.Count > 0)
|
|
{
|
|
result.MaxUnreadNotificationLevel = levels.Max();
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the notifications.
|
|
/// </summary>
|
|
/// <param name="reader">The reader.</param>
|
|
/// <returns>IEnumerable{Notification}.</returns>
|
|
private IEnumerable<Notification> GetNotifications(IDataReader reader)
|
|
{
|
|
var list = new List<Notification>();
|
|
|
|
while (reader.Read())
|
|
{
|
|
list.Add(GetNotification(reader));
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
private Notification GetNotification(IDataReader reader)
|
|
{
|
|
var notification = new Notification
|
|
{
|
|
Id = reader.GetGuid(0).ToString("N"),
|
|
UserId = reader.GetGuid(1).ToString("N"),
|
|
Date = reader.GetDateTime(2).ToUniversalTime(),
|
|
Name = reader.GetString(3)
|
|
};
|
|
|
|
if (!reader.IsDBNull(4))
|
|
{
|
|
notification.Description = reader.GetString(4);
|
|
}
|
|
|
|
if (!reader.IsDBNull(5))
|
|
{
|
|
notification.Url = reader.GetString(5);
|
|
}
|
|
|
|
notification.Level = GetLevel(reader, 6);
|
|
notification.IsRead = reader.GetBoolean(7);
|
|
|
|
return notification;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the level.
|
|
/// </summary>
|
|
/// <param name="reader">The reader.</param>
|
|
/// <param name="index">The index.</param>
|
|
/// <returns>NotificationLevel.</returns>
|
|
private NotificationLevel GetLevel(IDataReader reader, int index)
|
|
{
|
|
NotificationLevel level;
|
|
|
|
var val = reader.GetString(index);
|
|
|
|
Enum.TryParse(val, true, out level);
|
|
|
|
return level;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds the notification.
|
|
/// </summary>
|
|
/// <param name="notification">The notification.</param>
|
|
/// <param name="cancellationToken">The cancellation token.</param>
|
|
/// <returns>Task.</returns>
|
|
public async Task AddNotification(Notification notification, CancellationToken cancellationToken)
|
|
{
|
|
await ReplaceNotification(notification, cancellationToken).ConfigureAwait(false);
|
|
|
|
if (NotificationAdded != null)
|
|
{
|
|
try
|
|
{
|
|
NotificationAdded(this, new NotificationUpdateEventArgs
|
|
{
|
|
Notification = notification
|
|
});
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logger.ErrorException("Error in NotificationAdded event handler", ex);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Replaces the notification.
|
|
/// </summary>
|
|
/// <param name="notification">The notification.</param>
|
|
/// <param name="cancellationToken">The cancellation token.</param>
|
|
/// <returns>Task.</returns>
|
|
private async Task ReplaceNotification(Notification notification, CancellationToken cancellationToken)
|
|
{
|
|
if (string.IsNullOrEmpty(notification.Id))
|
|
{
|
|
notification.Id = Guid.NewGuid().ToString("N");
|
|
}
|
|
if (string.IsNullOrEmpty(notification.UserId))
|
|
{
|
|
throw new ArgumentException("The notification must have a user id");
|
|
}
|
|
|
|
cancellationToken.ThrowIfCancellationRequested();
|
|
|
|
using (var connection = await CreateConnection().ConfigureAwait(false))
|
|
{
|
|
using (var replaceNotificationCommand = connection.CreateCommand())
|
|
{
|
|
replaceNotificationCommand.CommandText = "replace into Notifications (Id, UserId, Date, Name, Description, Url, Level, IsRead, Category, RelatedId) values (@Id, @UserId, @Date, @Name, @Description, @Url, @Level, @IsRead, @Category, @RelatedId)";
|
|
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@Id");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@UserId");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@Date");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@Name");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@Description");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@Url");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@Level");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@IsRead");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@Category");
|
|
replaceNotificationCommand.Parameters.Add(replaceNotificationCommand, "@RelatedId");
|
|
|
|
IDbTransaction transaction = null;
|
|
|
|
try
|
|
{
|
|
transaction = connection.BeginTransaction();
|
|
|
|
replaceNotificationCommand.GetParameter(0).Value = new Guid(notification.Id);
|
|
replaceNotificationCommand.GetParameter(1).Value = new Guid(notification.UserId);
|
|
replaceNotificationCommand.GetParameter(2).Value = notification.Date.ToUniversalTime();
|
|
replaceNotificationCommand.GetParameter(3).Value = notification.Name;
|
|
replaceNotificationCommand.GetParameter(4).Value = notification.Description;
|
|
replaceNotificationCommand.GetParameter(5).Value = notification.Url;
|
|
replaceNotificationCommand.GetParameter(6).Value = notification.Level.ToString();
|
|
replaceNotificationCommand.GetParameter(7).Value = notification.IsRead;
|
|
replaceNotificationCommand.GetParameter(8).Value = string.Empty;
|
|
replaceNotificationCommand.GetParameter(9).Value = string.Empty;
|
|
|
|
replaceNotificationCommand.Transaction = transaction;
|
|
|
|
replaceNotificationCommand.ExecuteNonQuery();
|
|
|
|
transaction.Commit();
|
|
}
|
|
catch (OperationCanceledException)
|
|
{
|
|
if (transaction != null)
|
|
{
|
|
transaction.Rollback();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Logger.ErrorException("Failed to save notification:", e);
|
|
|
|
if (transaction != null)
|
|
{
|
|
transaction.Rollback();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
finally
|
|
{
|
|
if (transaction != null)
|
|
{
|
|
transaction.Dispose();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Marks the read.
|
|
/// </summary>
|
|
/// <param name="notificationIdList">The notification id list.</param>
|
|
/// <param name="userId">The user id.</param>
|
|
/// <param name="isRead">if set to <c>true</c> [is read].</param>
|
|
/// <param name="cancellationToken">The cancellation token.</param>
|
|
/// <returns>Task.</returns>
|
|
public async Task MarkRead(IEnumerable<string> notificationIdList, string userId, bool isRead, CancellationToken cancellationToken)
|
|
{
|
|
var list = notificationIdList.ToList();
|
|
var idArray = list.Select(i => new Guid(i)).ToArray();
|
|
|
|
await MarkReadInternal(idArray, userId, isRead, cancellationToken).ConfigureAwait(false);
|
|
|
|
if (NotificationsMarkedRead != null)
|
|
{
|
|
try
|
|
{
|
|
NotificationsMarkedRead(this, new NotificationReadEventArgs
|
|
{
|
|
IdList = list.ToArray(),
|
|
IsRead = isRead,
|
|
UserId = userId
|
|
});
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logger.ErrorException("Error in NotificationsMarkedRead event handler", ex);
|
|
}
|
|
}
|
|
}
|
|
|
|
public async Task MarkAllRead(string userId, bool isRead, CancellationToken cancellationToken)
|
|
{
|
|
cancellationToken.ThrowIfCancellationRequested();
|
|
|
|
using (var connection = await CreateConnection().ConfigureAwait(false))
|
|
{
|
|
using (var markAllReadCommand = connection.CreateCommand())
|
|
{
|
|
markAllReadCommand.CommandText = "update Notifications set IsRead=@IsRead where UserId=@UserId";
|
|
|
|
markAllReadCommand.Parameters.Add(markAllReadCommand, "@UserId");
|
|
markAllReadCommand.Parameters.Add(markAllReadCommand, "@IsRead");
|
|
|
|
IDbTransaction transaction = null;
|
|
|
|
try
|
|
{
|
|
cancellationToken.ThrowIfCancellationRequested();
|
|
|
|
transaction = connection.BeginTransaction();
|
|
|
|
markAllReadCommand.GetParameter(0).Value = new Guid(userId);
|
|
markAllReadCommand.GetParameter(1).Value = isRead;
|
|
|
|
markAllReadCommand.ExecuteNonQuery();
|
|
|
|
transaction.Commit();
|
|
}
|
|
catch (OperationCanceledException)
|
|
{
|
|
if (transaction != null)
|
|
{
|
|
transaction.Rollback();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Logger.ErrorException("Failed to save notification:", e);
|
|
|
|
if (transaction != null)
|
|
{
|
|
transaction.Rollback();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
finally
|
|
{
|
|
if (transaction != null)
|
|
{
|
|
transaction.Dispose();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private async Task MarkReadInternal(IEnumerable<Guid> notificationIdList, string userId, bool isRead, CancellationToken cancellationToken)
|
|
{
|
|
cancellationToken.ThrowIfCancellationRequested();
|
|
|
|
using (var connection = await CreateConnection().ConfigureAwait(false))
|
|
{
|
|
using (var markReadCommand = connection.CreateCommand())
|
|
{
|
|
markReadCommand.CommandText = "update Notifications set IsRead=@IsRead where Id=@Id and UserId=@UserId";
|
|
|
|
markReadCommand.Parameters.Add(markReadCommand, "@UserId");
|
|
markReadCommand.Parameters.Add(markReadCommand, "@IsRead");
|
|
markReadCommand.Parameters.Add(markReadCommand, "@Id");
|
|
|
|
IDbTransaction transaction = null;
|
|
|
|
try
|
|
{
|
|
cancellationToken.ThrowIfCancellationRequested();
|
|
|
|
transaction = connection.BeginTransaction();
|
|
|
|
markReadCommand.GetParameter(0).Value = new Guid(userId);
|
|
markReadCommand.GetParameter(1).Value = isRead;
|
|
|
|
foreach (var id in notificationIdList)
|
|
{
|
|
markReadCommand.GetParameter(2).Value = id;
|
|
|
|
markReadCommand.Transaction = transaction;
|
|
|
|
markReadCommand.ExecuteNonQuery();
|
|
}
|
|
|
|
transaction.Commit();
|
|
}
|
|
catch (OperationCanceledException)
|
|
{
|
|
if (transaction != null)
|
|
{
|
|
transaction.Rollback();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Logger.ErrorException("Failed to save notification:", e);
|
|
|
|
if (transaction != null)
|
|
{
|
|
transaction.Rollback();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
finally
|
|
{
|
|
if (transaction != null)
|
|
{
|
|
transaction.Dispose();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|