2016-11-18 08:39:20 +00:00
using System ;
using System.Collections.Generic ;
using System.Globalization ;
using System.IO ;
using System.Linq ;
using System.Threading ;
using System.Threading.Tasks ;
using Emby.Server.Implementations.Data ;
using MediaBrowser.Controller ;
using MediaBrowser.Controller.Notifications ;
using MediaBrowser.Model.Logging ;
using MediaBrowser.Model.Notifications ;
using SQLitePCL.pretty ;
namespace Emby.Server.Implementations.Notifications
{
public class SqliteNotificationsRepository : BaseSqliteRepository , INotificationsRepository
{
public SqliteNotificationsRepository ( ILogger logger , IServerApplicationPaths appPaths ) : base ( logger )
{
DbFilePath = Path . Combine ( appPaths . DataPath , "notifications.db" ) ;
}
public event EventHandler < NotificationUpdateEventArgs > NotificationAdded ;
public event EventHandler < NotificationReadEventArgs > NotificationsMarkedRead ;
////public event EventHandler<NotificationUpdateEventArgs> NotificationUpdated;
public void Initialize ( )
{
using ( var connection = CreateConnection ( ) )
{
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 NULL, Url TEXT NULL, Level TEXT NOT NULL, IsRead BOOLEAN NOT NULL, Category TEXT NOT NULL, RelatedId TEXT NULL, 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 ) ;
}
}
/// <summary>
/// Gets the notifications.
/// </summary>
/// <param name="query">The query.</param>
/// <returns>NotificationResult.</returns>
public NotificationResult GetNotifications ( NotificationQuery query )
{
var result = new NotificationResult ( ) ;
2016-11-19 08:40:13 +00:00
using ( WriteLock . Read ( ) )
2016-11-18 08:39:20 +00:00
{
using ( var connection = CreateConnection ( true ) )
{
var clauses = new List < string > ( ) ;
var paramList = new List < object > ( ) ;
if ( query . IsRead . HasValue )
{
clauses . Add ( "IsRead=?" ) ;
paramList . Add ( query . IsRead . Value ) ;
}
clauses . Add ( "UserId=?" ) ;
paramList . Add ( query . UserId . ToGuidParamValue ( ) ) ;
var whereClause = " where " + string . Join ( " And " , clauses . ToArray ( ) ) ;
result . TotalRecordCount = connection . Query ( "select count(Id) from Notifications" + whereClause , paramList . ToArray ( ) ) . SelectScalarInt ( ) . First ( ) ;
var commandText = string . Format ( "select Id,UserId,Date,Name,Description,Url,Level,IsRead,Category,RelatedId from Notifications{0} order by IsRead asc, Date desc" , whereClause ) ;
if ( query . Limit . HasValue | | query . StartIndex . HasValue )
{
var offset = query . StartIndex ? ? 0 ;
if ( query . Limit . HasValue | | offset > 0 )
{
commandText + = " LIMIT " + ( query . Limit ? ? int . MaxValue ) . ToString ( CultureInfo . InvariantCulture ) ;
}
if ( offset > 0 )
{
commandText + = " OFFSET " + offset . ToString ( CultureInfo . InvariantCulture ) ;
}
}
var resultList = new List < Notification > ( ) ;
foreach ( var row in connection . Query ( commandText , paramList . ToArray ( ) ) )
{
resultList . Add ( GetNotification ( row ) ) ;
}
result . Notifications = resultList . ToArray ( ) ;
}
}
return result ;
}
public NotificationsSummary GetNotificationsSummary ( string userId )
{
var result = new NotificationsSummary ( ) ;
2016-11-19 08:40:13 +00:00
using ( WriteLock . Read ( ) )
2016-11-18 08:39:20 +00:00
{
using ( var connection = CreateConnection ( true ) )
{
foreach ( var row in connection . Query ( "select Level from Notifications where UserId=? and IsRead=?" , userId . ToGuidParamValue ( ) , false ) )
{
var levels = new List < NotificationLevel > ( ) ;
levels . Add ( GetLevel ( row , 0 ) ) ;
result . UnreadCount = levels . Count ;
if ( levels . Count > 0 )
{
result . MaxUnreadNotificationLevel = levels . Max ( ) ;
}
}
return result ;
}
}
}
private Notification GetNotification ( IReadOnlyList < IResultSetValue > reader )
{
var notification = new Notification
{
Id = reader [ 0 ] . ReadGuid ( ) . ToString ( "N" ) ,
UserId = reader [ 1 ] . ReadGuid ( ) . ToString ( "N" ) ,
Date = reader [ 2 ] . ReadDateTime ( ) ,
Name = reader [ 3 ] . ToString ( )
} ;
if ( reader [ 4 ] . SQLiteType ! = SQLiteType . Null )
{
notification . Description = reader [ 4 ] . ToString ( ) ;
}
if ( reader [ 5 ] . SQLiteType ! = SQLiteType . Null )
{
notification . Url = reader [ 5 ] . ToString ( ) ;
}
notification . Level = GetLevel ( reader , 6 ) ;
notification . IsRead = reader [ 7 ] . ToBool ( ) ;
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 ( IReadOnlyList < IResultSetValue > reader , int index )
{
NotificationLevel level ;
var val = reader [ index ] . ToString ( ) ;
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 ( ) ;
2016-11-19 08:40:13 +00:00
using ( WriteLock . Write ( ) )
2016-11-18 08:39:20 +00:00
{
using ( var connection = CreateConnection ( ) )
{
connection . RunInTransaction ( conn = >
{
conn . Execute ( "replace into Notifications (Id, UserId, Date, Name, Description, Url, Level, IsRead, Category, RelatedId) values (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)" ,
notification . Id . ToGuidParamValue ( ) ,
notification . UserId . ToGuidParamValue ( ) ,
notification . Date . ToDateTimeParamValue ( ) ,
notification . Name ,
notification . Description ,
notification . Url ,
notification . Level . ToString ( ) ,
notification . IsRead ,
string . Empty ,
string . Empty ) ;
} ) ;
}
}
}
/// <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 ( ) ;
2016-11-19 08:40:13 +00:00
using ( WriteLock . Write ( ) )
2016-11-18 08:39:20 +00:00
{
using ( var connection = CreateConnection ( ) )
{
connection . RunInTransaction ( conn = >
{
2016-11-20 02:42:58 +00:00
conn . Execute ( "update Notifications set IsRead=? where UserId=?" , isRead , userId . ToGuidParamValue ( ) ) ;
2016-11-18 08:39:20 +00:00
} ) ;
}
}
}
private async Task MarkReadInternal ( IEnumerable < Guid > notificationIdList , string userId , bool isRead , CancellationToken cancellationToken )
{
cancellationToken . ThrowIfCancellationRequested ( ) ;
2016-11-19 08:40:13 +00:00
using ( WriteLock . Write ( ) )
2016-11-18 08:39:20 +00:00
{
using ( var connection = CreateConnection ( ) )
{
connection . RunInTransaction ( conn = >
{
var userIdParam = userId . ToGuidParamValue ( ) ;
foreach ( var id in notificationIdList )
{
2016-11-20 02:42:58 +00:00
conn . Execute ( "update Notifications set IsRead=? where UserId=? and Id=?" , isRead , userIdParam , id ) ;
2016-11-18 08:39:20 +00:00
}
} ) ;
}
}
}
}
}