| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044 | 
							- using MediaBrowser.Common.Configuration;
 
- using MediaBrowser.Controller.Entities;
 
- using MediaBrowser.Controller.LiveTv;
 
- using MediaBrowser.Controller.Persistence;
 
- using MediaBrowser.Model.Entities;
 
- using MediaBrowser.Model.Logging;
 
- using MediaBrowser.Model.Querying;
 
- using MediaBrowser.Model.Serialization;
 
- using System;
 
- using System.Collections.Generic;
 
- using System.Data;
 
- using System.Globalization;
 
- using System.IO;
 
- using System.Linq;
 
- using System.Threading;
 
- using System.Threading.Tasks;
 
- namespace MediaBrowser.Server.Implementations.Persistence
 
- {
 
-     /// <summary>
 
-     /// Class SQLiteItemRepository
 
-     /// </summary>
 
-     public class SqliteItemRepository : IItemRepository
 
-     {
 
-         private IDbConnection _connection;
 
-         private readonly ILogger _logger;
 
-         private readonly TypeMapper _typeMapper = new TypeMapper();
 
-         /// <summary>
 
-         /// Gets the name of the repository
 
-         /// </summary>
 
-         /// <value>The name.</value>
 
-         public string Name
 
-         {
 
-             get
 
-             {
 
-                 return "SQLite";
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Gets the json serializer.
 
-         /// </summary>
 
-         /// <value>The json serializer.</value>
 
-         private readonly IJsonSerializer _jsonSerializer;
 
-         /// <summary>
 
-         /// The _app paths
 
-         /// </summary>
 
-         private readonly IApplicationPaths _appPaths;
 
-         /// <summary>
 
-         /// The _save item command
 
-         /// </summary>
 
-         private IDbCommand _saveItemCommand;
 
-         private readonly string _criticReviewsPath;
 
-         private SqliteChapterRepository _chapterRepository;
 
-         private SqliteMediaStreamsRepository _mediaStreamsRepository;
 
-         private IDbCommand _deleteChildrenCommand;
 
-         private IDbCommand _saveChildrenCommand;
 
-         private IDbCommand _deleteItemCommand;
 
-         /// <summary>
 
-         /// Initializes a new instance of the <see cref="SqliteItemRepository"/> class.
 
-         /// </summary>
 
-         /// <param name="appPaths">The app paths.</param>
 
-         /// <param name="jsonSerializer">The json serializer.</param>
 
-         /// <param name="logManager">The log manager.</param>
 
-         /// <exception cref="System.ArgumentNullException">
 
-         /// appPaths
 
-         /// or
 
-         /// jsonSerializer
 
-         /// </exception>
 
-         public SqliteItemRepository(IApplicationPaths appPaths, IJsonSerializer jsonSerializer, ILogManager logManager)
 
-         {
 
-             if (appPaths == null)
 
-             {
 
-                 throw new ArgumentNullException("appPaths");
 
-             }
 
-             if (jsonSerializer == null)
 
-             {
 
-                 throw new ArgumentNullException("jsonSerializer");
 
-             }
 
-             _appPaths = appPaths;
 
-             _jsonSerializer = jsonSerializer;
 
-             _criticReviewsPath = Path.Combine(_appPaths.DataPath, "critic-reviews");
 
-             _logger = logManager.GetLogger(GetType().Name);
 
-             var chapterDbFile = Path.Combine(_appPaths.DataPath, "chapters.db");
 
-             var chapterConnection = SqliteExtensions.ConnectToDb(chapterDbFile, _logger).Result;
 
-             _chapterRepository = new SqliteChapterRepository(chapterConnection, logManager);
 
-             var mediaStreamsDbFile = Path.Combine(_appPaths.DataPath, "mediainfo.db");
 
-             var mediaStreamsConnection = SqliteExtensions.ConnectToDb(mediaStreamsDbFile, _logger).Result;
 
-             _mediaStreamsRepository = new SqliteMediaStreamsRepository(mediaStreamsConnection, logManager);
 
-         }
 
-         /// <summary>
 
-         /// Opens the connection to the database
 
-         /// </summary>
 
-         /// <returns>Task.</returns>
 
-         public async Task Initialize()
 
-         {
 
-             var dbFile = Path.Combine(_appPaths.DataPath, "library.db");
 
-             _connection = await SqliteExtensions.ConnectToDb(dbFile, _logger).ConfigureAwait(false);
 
-             string[] queries = {
 
-                                 "create table if not exists TypedBaseItems (guid GUID primary key, type TEXT, data BLOB)",
 
-                                 "create index if not exists idx_TypedBaseItems on TypedBaseItems(guid)",
 
-                                 "create table if not exists ChildrenIds (ParentId GUID, ItemId GUID, PRIMARY KEY (ParentId, ItemId))",
 
-                                 "create index if not exists idx_ChildrenIds on ChildrenIds(ParentId,ItemId)",
 
-                                 //pragmas
 
-                                 "pragma temp_store = memory",
 
-                                 "pragma shrink_memory"
 
-                                };
 
-             _connection.RunQueries(queries, _logger);
 
-             _connection.AddColumn(_logger, "TypedBaseItems", "StartDate", "DATETIME");
 
-             _connection.AddColumn(_logger, "TypedBaseItems", "EndDate", "DATETIME");
 
-             _connection.AddColumn(_logger, "TypedBaseItems", "ChannelId", "Text");
 
-             _connection.AddColumn(_logger, "TypedBaseItems", "IsMovie", "BIT");
 
-             _connection.AddColumn(_logger, "TypedBaseItems", "IsSports", "BIT");
 
-             _connection.AddColumn(_logger, "TypedBaseItems", "IsKids", "BIT");
 
-             PrepareStatements();
 
-             _mediaStreamsRepository.Initialize();
 
-             _chapterRepository.Initialize();
 
-         }
 
-         /// <summary>
 
-         /// The _write lock
 
-         /// </summary>
 
-         private readonly SemaphoreSlim _writeLock = new SemaphoreSlim(1, 1);
 
-         /// <summary>
 
-         /// Prepares the statements.
 
-         /// </summary>
 
-         private void PrepareStatements()
 
-         {
 
-             _saveItemCommand = _connection.CreateCommand();
 
-             _saveItemCommand.CommandText = "replace into TypedBaseItems (guid, type, data, StartDate, EndDate, ChannelId, IsKids, IsMovie, IsSports) values (@1, @2, @3, @4, @5, @6, @7, @8, @9)";
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@1");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@2");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@3");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@4");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@5");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@6");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@7");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@8");
 
-             _saveItemCommand.Parameters.Add(_saveItemCommand, "@9");
 
-             _deleteChildrenCommand = _connection.CreateCommand();
 
-             _deleteChildrenCommand.CommandText = "delete from ChildrenIds where ParentId=@ParentId";
 
-             _deleteChildrenCommand.Parameters.Add(_deleteChildrenCommand, "@ParentId");
 
-             _deleteItemCommand = _connection.CreateCommand();
 
-             _deleteItemCommand.CommandText = "delete from TypedBaseItems where guid=@Id";
 
-             _deleteItemCommand.Parameters.Add(_deleteItemCommand, "@Id");
 
-             _saveChildrenCommand = _connection.CreateCommand();
 
-             _saveChildrenCommand.CommandText = "replace into ChildrenIds (ParentId, ItemId) values (@ParentId, @ItemId)";
 
-             _saveChildrenCommand.Parameters.Add(_saveChildrenCommand, "@ParentId");
 
-             _saveChildrenCommand.Parameters.Add(_saveChildrenCommand, "@ItemId");
 
-         }
 
-         /// <summary>
 
-         /// Save a standard item in the repo
 
-         /// </summary>
 
-         /// <param name="item">The item.</param>
 
-         /// <param name="cancellationToken">The cancellation token.</param>
 
-         /// <returns>Task.</returns>
 
-         /// <exception cref="System.ArgumentNullException">item</exception>
 
-         public Task SaveItem(BaseItem item, CancellationToken cancellationToken)
 
-         {
 
-             if (item == null)
 
-             {
 
-                 throw new ArgumentNullException("item");
 
-             }
 
-             return SaveItems(new[] { item }, cancellationToken);
 
-         }
 
-         /// <summary>
 
-         /// Saves the items.
 
-         /// </summary>
 
-         /// <param name="items">The items.</param>
 
-         /// <param name="cancellationToken">The cancellation token.</param>
 
-         /// <returns>Task.</returns>
 
-         /// <exception cref="System.ArgumentNullException">
 
-         /// items
 
-         /// or
 
-         /// cancellationToken
 
-         /// </exception>
 
-         public async Task SaveItems(IEnumerable<BaseItem> items, CancellationToken cancellationToken)
 
-         {
 
-             if (items == null)
 
-             {
 
-                 throw new ArgumentNullException("items");
 
-             }
 
-             cancellationToken.ThrowIfCancellationRequested();
 
-             CheckDisposed();
 
-             await _writeLock.WaitAsync(cancellationToken).ConfigureAwait(false);
 
-             IDbTransaction transaction = null;
 
-             try
 
-             {
 
-                 transaction = _connection.BeginTransaction();
 
-                 foreach (var item in items)
 
-                 {
 
-                     cancellationToken.ThrowIfCancellationRequested();
 
-                     _saveItemCommand.GetParameter(0).Value = item.Id;
 
-                     _saveItemCommand.GetParameter(1).Value = item.GetType().FullName;
 
-                     _saveItemCommand.GetParameter(2).Value = _jsonSerializer.SerializeToBytes(item);
 
-                     var hasStartDate = item as IHasStartDate;
 
-                     if (hasStartDate != null)
 
-                     {
 
-                         _saveItemCommand.GetParameter(3).Value = hasStartDate.StartDate;
 
-                     }
 
-                     else
 
-                     {
 
-                         _saveItemCommand.GetParameter(3).Value = null;
 
-                     }
 
-                     _saveItemCommand.GetParameter(4).Value = item.EndDate;
 
-                     _saveItemCommand.GetParameter(5).Value = item.ChannelId;
 
-                     var hasProgramAttributes = item as IHasProgramAttributes;
 
-                     if (hasProgramAttributes != null)
 
-                     {
 
-                         _saveItemCommand.GetParameter(6).Value = hasProgramAttributes.IsKids;
 
-                         _saveItemCommand.GetParameter(7).Value = hasProgramAttributes.IsMovie;
 
-                         _saveItemCommand.GetParameter(8).Value = hasProgramAttributes.IsSports;
 
-                     }
 
-                     else
 
-                     {
 
-                         _saveItemCommand.GetParameter(6).Value = null;
 
-                         _saveItemCommand.GetParameter(7).Value = null;
 
-                         _saveItemCommand.GetParameter(8).Value = null;
 
-                     }
 
-                     _saveItemCommand.Transaction = transaction;
 
-                     _saveItemCommand.ExecuteNonQuery();
 
-                 }
 
-                 transaction.Commit();
 
-             }
 
-             catch (OperationCanceledException)
 
-             {
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Rollback();
 
-                 }
 
-                 throw;
 
-             }
 
-             catch (Exception e)
 
-             {
 
-                 _logger.ErrorException("Failed to save items:", e);
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Rollback();
 
-                 }
 
-                 throw;
 
-             }
 
-             finally
 
-             {
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Dispose();
 
-                 }
 
-                 _writeLock.Release();
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Internal retrieve from items or users table
 
-         /// </summary>
 
-         /// <param name="id">The id.</param>
 
-         /// <returns>BaseItem.</returns>
 
-         /// <exception cref="System.ArgumentNullException">id</exception>
 
-         /// <exception cref="System.ArgumentException"></exception>
 
-         public BaseItem RetrieveItem(Guid id)
 
-         {
 
-             if (id == Guid.Empty)
 
-             {
 
-                 throw new ArgumentNullException("id");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select type,data from TypedBaseItems where guid = @guid";
 
-                 cmd.Parameters.Add(cmd, "@guid", DbType.Guid).Value = id;
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess | CommandBehavior.SingleResult | CommandBehavior.SingleRow))
 
-                 {
 
-                     if (reader.Read())
 
-                     {
 
-                         return GetItem(reader);
 
-                     }
 
-                 }
 
-                 return null;
 
-             }
 
-         }
 
-         private BaseItem GetItem(IDataReader reader)
 
-         {
 
-             var typeString = reader.GetString(0);
 
-             var type = _typeMapper.GetType(typeString);
 
-             if (type == null)
 
-             {
 
-                 _logger.Debug("Unknown type {0}", typeString);
 
-                 return null;
 
-             }
 
-             using (var stream = reader.GetMemoryStream(1))
 
-             {
 
-                 return _jsonSerializer.DeserializeFromStream(stream, type) as BaseItem;
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Gets the critic reviews.
 
-         /// </summary>
 
-         /// <param name="itemId">The item id.</param>
 
-         /// <returns>Task{IEnumerable{ItemReview}}.</returns>
 
-         public IEnumerable<ItemReview> GetCriticReviews(Guid itemId)
 
-         {
 
-             try
 
-             {
 
-                 var path = Path.Combine(_criticReviewsPath, itemId + ".json");
 
-                 return _jsonSerializer.DeserializeFromFile<List<ItemReview>>(path);
 
-             }
 
-             catch (DirectoryNotFoundException)
 
-             {
 
-                 return new List<ItemReview>();
 
-             }
 
-             catch (FileNotFoundException)
 
-             {
 
-                 return new List<ItemReview>();
 
-             }
 
-         }
 
-         private readonly Task _cachedTask = Task.FromResult(true);
 
-         /// <summary>
 
-         /// Saves the critic reviews.
 
-         /// </summary>
 
-         /// <param name="itemId">The item id.</param>
 
-         /// <param name="criticReviews">The critic reviews.</param>
 
-         /// <returns>Task.</returns>
 
-         public Task SaveCriticReviews(Guid itemId, IEnumerable<ItemReview> criticReviews)
 
-         {
 
-             Directory.CreateDirectory(_criticReviewsPath);
 
-             var path = Path.Combine(_criticReviewsPath, itemId + ".json");
 
-             _jsonSerializer.SerializeToFile(criticReviews.ToList(), path);
 
-             return _cachedTask;
 
-         }
 
-         /// <summary>
 
-         /// Gets chapters for an item
 
-         /// </summary>
 
-         /// <param name="id">The id.</param>
 
-         /// <returns>IEnumerable{ChapterInfo}.</returns>
 
-         /// <exception cref="System.ArgumentNullException">id</exception>
 
-         public IEnumerable<ChapterInfo> GetChapters(Guid id)
 
-         {
 
-             CheckDisposed();
 
-             return _chapterRepository.GetChapters(id);
 
-         }
 
-         /// <summary>
 
-         /// Gets a single chapter for an item
 
-         /// </summary>
 
-         /// <param name="id">The id.</param>
 
-         /// <param name="index">The index.</param>
 
-         /// <returns>ChapterInfo.</returns>
 
-         /// <exception cref="System.ArgumentNullException">id</exception>
 
-         public ChapterInfo GetChapter(Guid id, int index)
 
-         {
 
-             CheckDisposed();
 
-             return _chapterRepository.GetChapter(id, index);
 
-         }
 
-         /// <summary>
 
-         /// Saves the chapters.
 
-         /// </summary>
 
-         /// <param name="id">The id.</param>
 
-         /// <param name="chapters">The chapters.</param>
 
-         /// <param name="cancellationToken">The cancellation token.</param>
 
-         /// <returns>Task.</returns>
 
-         /// <exception cref="System.ArgumentNullException">
 
-         /// id
 
-         /// or
 
-         /// chapters
 
-         /// or
 
-         /// cancellationToken
 
-         /// </exception>
 
-         public Task SaveChapters(Guid id, IEnumerable<ChapterInfo> chapters, CancellationToken cancellationToken)
 
-         {
 
-             CheckDisposed();
 
-             return _chapterRepository.SaveChapters(id, chapters, cancellationToken);
 
-         }
 
-         /// <summary>
 
-         /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.
 
-         /// </summary>
 
-         public void Dispose()
 
-         {
 
-             Dispose(true);
 
-             GC.SuppressFinalize(this);
 
-         }
 
-         private readonly object _disposeLock = new object();
 
-         private bool _disposed;
 
-         private void CheckDisposed()
 
-         {
 
-             if (_disposed)
 
-             {
 
-                 throw new ObjectDisposedException(GetType().Name + " has been disposed and cannot be accessed.");
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Releases unmanaged and - optionally - managed resources.
 
-         /// </summary>
 
-         /// <param name="dispose"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
 
-         protected virtual void Dispose(bool dispose)
 
-         {
 
-             if (dispose)
 
-             {
 
-                 _disposed = true;
 
-                 try
 
-                 {
 
-                     lock (_disposeLock)
 
-                     {
 
-                         _writeLock.Wait();
 
-                         if (_connection != null)
 
-                         {
 
-                             if (_connection.IsOpen())
 
-                             {
 
-                                 _connection.Close();
 
-                             }
 
-                             _connection.Dispose();
 
-                             _connection = null;
 
-                         }
 
-                         if (_chapterRepository != null)
 
-                         {
 
-                             _chapterRepository.Dispose();
 
-                             _chapterRepository = null;
 
-                         }
 
-                         if (_mediaStreamsRepository != null)
 
-                         {
 
-                             _mediaStreamsRepository.Dispose();
 
-                             _mediaStreamsRepository = null;
 
-                         }
 
-                     }
 
-                 }
 
-                 catch (Exception ex)
 
-                 {
 
-                     _logger.ErrorException("Error disposing database", ex);
 
-                 }
 
-             }
 
-         }
 
-         public IEnumerable<Guid> GetChildren(Guid parentId)
 
-         {
 
-             if (parentId == Guid.Empty)
 
-             {
 
-                 throw new ArgumentNullException("parentId");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select ItemId from ChildrenIds where ParentId = @ParentId";
 
-                 cmd.Parameters.Add(cmd, "@ParentId", DbType.Guid).Value = parentId;
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess | CommandBehavior.SingleResult))
 
-                 {
 
-                     while (reader.Read())
 
-                     {
 
-                         yield return reader.GetGuid(0);
 
-                     }
 
-                 }
 
-             }
 
-         }
 
-         public IEnumerable<BaseItem> GetChildrenItems(Guid parentId)
 
-         {
 
-             if (parentId == Guid.Empty)
 
-             {
 
-                 throw new ArgumentNullException("parentId");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select type,data from TypedBaseItems where guid in (select ItemId from ChildrenIds where ParentId = @ParentId)";
 
-                 cmd.Parameters.Add(cmd, "@ParentId", DbType.Guid).Value = parentId;
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess | CommandBehavior.SingleResult))
 
-                 {
 
-                     while (reader.Read())
 
-                     {
 
-                         var item = GetItem(reader);
 
-                         if (item != null)
 
-                         {
 
-                             yield return item;
 
-                         }
 
-                     }
 
-                 }
 
-             }
 
-         }
 
-         public IEnumerable<BaseItem> GetItemsOfType(Type type)
 
-         {
 
-             if (type == null)
 
-             {
 
-                 throw new ArgumentNullException("type");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select type,data from TypedBaseItems where type = @type";
 
-                 cmd.Parameters.Add(cmd, "@type", DbType.String).Value = type.FullName;
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess | CommandBehavior.SingleResult))
 
-                 {
 
-                     while (reader.Read())
 
-                     {
 
-                         var item = GetItem(reader);
 
-                         if (item != null)
 
-                         {
 
-                             yield return item;
 
-                         }
 
-                     }
 
-                 }
 
-             }
 
-         }
 
-         public QueryResult<BaseItem> GetItems(InternalItemsQuery query)
 
-         {
 
-             if (query == null)
 
-             {
 
-                 throw new ArgumentNullException("query");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select type,data from TypedBaseItems";
 
-                 var whereClauses = GetWhereClauses(query, cmd, false);
 
-                 var whereTextWithoutPaging = whereClauses.Count == 0 ?
 
-                     string.Empty :
 
-                     " where " + string.Join(" AND ", whereClauses.ToArray());
 
-                 whereClauses = GetWhereClauses(query, cmd, true);
 
-                 var whereText = whereClauses.Count == 0 ?
 
-                     string.Empty :
 
-                     " where " + string.Join(" AND ", whereClauses.ToArray());
 
-                 cmd.CommandText += whereText;
 
-                 if (query.Limit.HasValue)
 
-                 {
 
-                     cmd.CommandText += " LIMIT " + query.Limit.Value.ToString(CultureInfo.InvariantCulture);
 
-                 }
 
-                 cmd.CommandText += "; select count (guid) from TypedBaseItems" + whereTextWithoutPaging;
 
-                 var list = new List<BaseItem>();
 
-                 var count = 0;
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess))
 
-                 {
 
-                     while (reader.Read())
 
-                     {
 
-                         list.Add(GetItem(reader));
 
-                     }
 
-                     if (reader.NextResult() && reader.Read())
 
-                     {
 
-                         count = reader.GetInt32(0);
 
-                     }
 
-                 }
 
-                 return new QueryResult<BaseItem>()
 
-                 {
 
-                     Items = list.ToArray(),
 
-                     TotalRecordCount = count
 
-                 };
 
-             }
 
-         }
 
-         public List<Guid> GetItemIdsList(InternalItemsQuery query)
 
-         {
 
-             if (query == null)
 
-             {
 
-                 throw new ArgumentNullException("query");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select guid from TypedBaseItems";
 
-                 var whereClauses = GetWhereClauses(query, cmd, false);
 
-                 whereClauses = GetWhereClauses(query, cmd, true);
 
-                 var whereText = whereClauses.Count == 0 ?
 
-                     string.Empty :
 
-                     " where " + string.Join(" AND ", whereClauses.ToArray());
 
-                 cmd.CommandText += whereText;
 
-                 if (query.Limit.HasValue)
 
-                 {
 
-                     cmd.CommandText += " LIMIT " + query.Limit.Value.ToString(CultureInfo.InvariantCulture);
 
-                 }
 
-                 var list = new List<Guid>();
 
-                 _logger.Debug(cmd.CommandText);
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess))
 
-                 {
 
-                     while (reader.Read())
 
-                     {
 
-                         list.Add(reader.GetGuid(0));
 
-                     }
 
-                 }
 
-                 return list;
 
-             }
 
-         }
 
-         public QueryResult<Guid> GetItemIds(InternalItemsQuery query)
 
-         {
 
-             if (query == null)
 
-             {
 
-                 throw new ArgumentNullException("query");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select guid from TypedBaseItems";
 
-                 var whereClauses = GetWhereClauses(query, cmd, false);
 
-                 var whereTextWithoutPaging = whereClauses.Count == 0 ?
 
-                     string.Empty :
 
-                     " where " + string.Join(" AND ", whereClauses.ToArray());
 
-                 whereClauses = GetWhereClauses(query, cmd, true);
 
-                 var whereText = whereClauses.Count == 0 ?
 
-                     string.Empty :
 
-                     " where " + string.Join(" AND ", whereClauses.ToArray());
 
-                 cmd.CommandText += whereText;
 
-                 if (query.Limit.HasValue)
 
-                 {
 
-                     cmd.CommandText += " LIMIT " + query.Limit.Value.ToString(CultureInfo.InvariantCulture);
 
-                 }
 
-                 cmd.CommandText += "; select count (guid) from TypedBaseItems" + whereTextWithoutPaging;
 
-                 var list = new List<Guid>();
 
-                 var count = 0;
 
-                 _logger.Debug(cmd.CommandText);
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess))
 
-                 {
 
-                     while (reader.Read())
 
-                     {
 
-                         list.Add(reader.GetGuid(0));
 
-                     }
 
-                     if (reader.NextResult() && reader.Read())
 
-                     {
 
-                         count = reader.GetInt32(0);
 
-                     }
 
-                 }
 
-                 return new QueryResult<Guid>()
 
-                 {
 
-                     Items = list.ToArray(),
 
-                     TotalRecordCount = count
 
-                 };
 
-             }
 
-         }
 
-         private List<string> GetWhereClauses(InternalItemsQuery query, IDbCommand cmd, bool addPaging)
 
-         {
 
-             var whereClauses = new List<string>();
 
-             if (query.IsMovie.HasValue)
 
-             {
 
-                 whereClauses.Add("IsMovie=@IsMovie");
 
-                 cmd.Parameters.Add(cmd, "@IsMovie", DbType.Boolean).Value = query.IsMovie;
 
-             }
 
-             if (query.IsKids.HasValue)
 
-             {
 
-                 whereClauses.Add("IsKids=@IsKids");
 
-                 cmd.Parameters.Add(cmd, "@IsKids", DbType.Boolean).Value = query.IsKids;
 
-             }
 
-             if (query.IsSports.HasValue)
 
-             {
 
-                 whereClauses.Add("IsSports=@IsSports");
 
-                 cmd.Parameters.Add(cmd, "@IsSports", DbType.Boolean).Value = query.IsSports;
 
-             }
 
-             var includeTypes = query.IncludeItemTypes.SelectMany(MapIncludeItemTypes).ToArray();
 
-             if (includeTypes.Length == 1)
 
-             {
 
-                 whereClauses.Add("type=@type");
 
-                 cmd.Parameters.Add(cmd, "@type", DbType.String).Value = includeTypes[0];
 
-             }
 
-             if (includeTypes.Length > 1)
 
-             {
 
-                 var inClause = string.Join(",", includeTypes.Select(i => "'" + i + "'").ToArray());
 
-                 whereClauses.Add(string.Format("type in ({0})", inClause));
 
-             }
 
-             if (query.ChannelIds.Length == 1)
 
-             {
 
-                 whereClauses.Add("ChannelId=@ChannelId");
 
-                 cmd.Parameters.Add(cmd, "@ChannelId", DbType.String).Value = query.ChannelIds[0];
 
-             }
 
-             if (query.ChannelIds.Length > 1)
 
-             {
 
-                 var inClause = string.Join(",", query.ChannelIds.Select(i => "'" + i + "'").ToArray());
 
-                 whereClauses.Add(string.Format("ChannelId in ({0})", inClause));
 
-             }
 
-             if (query.MinEndDate.HasValue)
 
-             {
 
-                 whereClauses.Add("EndDate>=@MinEndDate");
 
-                 cmd.Parameters.Add(cmd, "@MinEndDate", DbType.Date).Value = query.MinEndDate.Value;
 
-             }
 
-             if (query.MaxEndDate.HasValue)
 
-             {
 
-                 whereClauses.Add("EndDate<=@MaxEndDate");
 
-                 cmd.Parameters.Add(cmd, "@MaxEndDate", DbType.Date).Value = query.MaxEndDate.Value;
 
-             }
 
-             if (query.MinStartDate.HasValue)
 
-             {
 
-                 whereClauses.Add("StartDate>=@MinStartDate");
 
-                 cmd.Parameters.Add(cmd, "@MinStartDate", DbType.Date).Value = query.MinStartDate.Value;
 
-             }
 
-             if (query.MaxStartDate.HasValue)
 
-             {
 
-                 whereClauses.Add("StartDate<=@MaxStartDate");
 
-                 cmd.Parameters.Add(cmd, "@MaxStartDate", DbType.Date).Value = query.MaxStartDate.Value;
 
-             }
 
-             if (query.IsAiring.HasValue)
 
-             {
 
-                 if (query.IsAiring.Value)
 
-                 {
 
-                     whereClauses.Add("StartDate<=@MaxStartDate");
 
-                     cmd.Parameters.Add(cmd, "@MaxStartDate", DbType.Date).Value = DateTime.UtcNow;
 
-                     whereClauses.Add("EndDate>=@MinEndDate");
 
-                     cmd.Parameters.Add(cmd, "@MinEndDate", DbType.Date).Value = DateTime.UtcNow;
 
-                 }
 
-                 else
 
-                 {
 
-                     whereClauses.Add("(StartDate>@IsAiringDate OR EndDate < @IsAiringDate)");
 
-                     cmd.Parameters.Add(cmd, "@IsAiringDate", DbType.Date).Value = DateTime.UtcNow;
 
-                 }
 
-             }
 
-             if (addPaging)
 
-             {
 
-                 if (query.StartIndex.HasValue && query.StartIndex.Value > 0)
 
-                 {
 
-                     var pagingWhereText = whereClauses.Count == 0 ?
 
-                         string.Empty :
 
-                         " where " + string.Join(" AND ", whereClauses.ToArray());
 
-                     whereClauses.Add(string.Format("Id NOT IN (SELECT Id FROM TypedBaseItems {0} ORDER BY DateCreated DESC LIMIT {1})",
 
-                         pagingWhereText,
 
-                         query.StartIndex.Value.ToString(CultureInfo.InvariantCulture)));
 
-                 }
 
-             }
 
-             return whereClauses;
 
-         }
 
-         // Not crazy about having this all the way down here, but at least it's in one place
 
-         readonly Dictionary<string, string[]> _types = new Dictionary<string, string[]>(StringComparer.OrdinalIgnoreCase)
 
-             {
 
-                 {typeof(LiveTvProgram).Name, new []{typeof(LiveTvProgram).FullName}},
 
-                 {typeof(LiveTvChannel).Name, new []{typeof(LiveTvChannel).FullName}},
 
-                 {typeof(LiveTvVideoRecording).Name, new []{typeof(LiveTvVideoRecording).FullName}},
 
-                 {typeof(LiveTvAudioRecording).Name, new []{typeof(LiveTvAudioRecording).FullName}},
 
-                 {"Recording", new []{typeof(LiveTvAudioRecording).FullName, typeof(LiveTvVideoRecording).FullName}}
 
-             };
 
-         private IEnumerable<string> MapIncludeItemTypes(string value)
 
-         {
 
-             string[] result;
 
-             if (_types.TryGetValue(value, out result))
 
-             {
 
-                 return result;
 
-             }
 
-             return new[] { value };
 
-         }
 
-         public IEnumerable<Guid> GetItemIdsOfType(Type type)
 
-         {
 
-             if (type == null)
 
-             {
 
-                 throw new ArgumentNullException("type");
 
-             }
 
-             CheckDisposed();
 
-             using (var cmd = _connection.CreateCommand())
 
-             {
 
-                 cmd.CommandText = "select guid from TypedBaseItems where type = @type";
 
-                 cmd.Parameters.Add(cmd, "@type", DbType.String).Value = type.FullName;
 
-                 using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess | CommandBehavior.SingleResult))
 
-                 {
 
-                     while (reader.Read())
 
-                     {
 
-                         yield return reader.GetGuid(0);
 
-                     }
 
-                 }
 
-             }
 
-         }
 
-         public async Task DeleteItem(Guid id, CancellationToken cancellationToken)
 
-         {
 
-             if (id == Guid.Empty)
 
-             {
 
-                 throw new ArgumentNullException("id");
 
-             }
 
-             CheckDisposed();
 
-             await _writeLock.WaitAsync(cancellationToken).ConfigureAwait(false);
 
-             IDbTransaction transaction = null;
 
-             try
 
-             {
 
-                 transaction = _connection.BeginTransaction();
 
-                 // First delete children
 
-                 _deleteChildrenCommand.GetParameter(0).Value = id;
 
-                 _deleteChildrenCommand.Transaction = transaction;
 
-                 _deleteChildrenCommand.ExecuteNonQuery();
 
-                 // Delete the item
 
-                 _deleteItemCommand.GetParameter(0).Value = id;
 
-                 _deleteItemCommand.Transaction = transaction;
 
-                 _deleteItemCommand.ExecuteNonQuery();
 
-                 transaction.Commit();
 
-             }
 
-             catch (OperationCanceledException)
 
-             {
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Rollback();
 
-                 }
 
-                 throw;
 
-             }
 
-             catch (Exception e)
 
-             {
 
-                 _logger.ErrorException("Failed to save children:", e);
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Rollback();
 
-                 }
 
-                 throw;
 
-             }
 
-             finally
 
-             {
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Dispose();
 
-                 }
 
-                 _writeLock.Release();
 
-             }
 
-         }
 
-         public async Task SaveChildren(Guid parentId, IEnumerable<Guid> children, CancellationToken cancellationToken)
 
-         {
 
-             if (parentId == Guid.Empty)
 
-             {
 
-                 throw new ArgumentNullException("parentId");
 
-             }
 
-             if (children == null)
 
-             {
 
-                 throw new ArgumentNullException("children");
 
-             }
 
-             CheckDisposed();
 
-             await _writeLock.WaitAsync(cancellationToken).ConfigureAwait(false);
 
-             IDbTransaction transaction = null;
 
-             try
 
-             {
 
-                 transaction = _connection.BeginTransaction();
 
-                 // First delete 
 
-                 _deleteChildrenCommand.GetParameter(0).Value = parentId;
 
-                 _deleteChildrenCommand.Transaction = transaction;
 
-                 _deleteChildrenCommand.ExecuteNonQuery();
 
-                 foreach (var id in children)
 
-                 {
 
-                     cancellationToken.ThrowIfCancellationRequested();
 
-                     _saveChildrenCommand.GetParameter(0).Value = parentId;
 
-                     _saveChildrenCommand.GetParameter(1).Value = id;
 
-                     _saveChildrenCommand.Transaction = transaction;
 
-                     _saveChildrenCommand.ExecuteNonQuery();
 
-                 }
 
-                 transaction.Commit();
 
-             }
 
-             catch (OperationCanceledException)
 
-             {
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Rollback();
 
-                 }
 
-                 throw;
 
-             }
 
-             catch (Exception e)
 
-             {
 
-                 _logger.ErrorException("Failed to save children:", e);
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Rollback();
 
-                 }
 
-                 throw;
 
-             }
 
-             finally
 
-             {
 
-                 if (transaction != null)
 
-                 {
 
-                     transaction.Dispose();
 
-                 }
 
-                 _writeLock.Release();
 
-             }
 
-         }
 
-         public IEnumerable<MediaStream> GetMediaStreams(MediaStreamQuery query)
 
-         {
 
-             CheckDisposed();
 
-             return _mediaStreamsRepository.GetMediaStreams(query);
 
-         }
 
-         public Task SaveMediaStreams(Guid id, IEnumerable<MediaStream> streams, CancellationToken cancellationToken)
 
-         {
 
-             CheckDisposed();
 
-             return _mediaStreamsRepository.SaveMediaStreams(id, streams, cancellationToken);
 
-         }
 
-     }
 
- }
 
 
  |