mirror of
				https://github.com/jellyfin/jellyfin.git
				synced 2025-10-31 10:37:22 -04:00 
			
		
		
		
	
						commit
						16dac0f55b
					
				| @ -10,6 +10,7 @@ using System.Collections.Generic; | |||||||
| using System.Globalization; | using System.Globalization; | ||||||
| using System.IO; | using System.IO; | ||||||
| using System.Linq; | using System.Linq; | ||||||
|  | using MediaBrowser.Model.Querying; | ||||||
| 
 | 
 | ||||||
| namespace MediaBrowser.Api | namespace MediaBrowser.Api | ||||||
| { | { | ||||||
| @ -187,18 +188,40 @@ namespace MediaBrowser.Api | |||||||
|         /// <returns>System.Object.</returns> |         /// <returns>System.Object.</returns> | ||||||
|         public object Get(GetSimilarGames request) |         public object Get(GetSimilarGames request) | ||||||
|         { |         { | ||||||
|             var dtoOptions = GetDtoOptions(request); |             var result = GetSimilarItemsResult(request); | ||||||
| 
 |  | ||||||
|             var result = SimilarItemsHelper.GetSimilarItemsResult(dtoOptions, _userManager, |  | ||||||
|                 _itemRepo, |  | ||||||
|                 _libraryManager, |  | ||||||
|                 _userDataRepository, |  | ||||||
|                 _dtoService, |  | ||||||
|                 Logger, |  | ||||||
|                 request, new[] { typeof(Game) }, |  | ||||||
|                 SimilarItemsHelper.GetSimiliarityScore); |  | ||||||
| 
 | 
 | ||||||
|             return ToOptimizedSerializedResultUsingCache(result); |             return ToOptimizedSerializedResultUsingCache(result); | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         private QueryResult<BaseItemDto> GetSimilarItemsResult(BaseGetSimilarItemsFromItem request) | ||||||
|  |         { | ||||||
|  |             var user = !string.IsNullOrWhiteSpace(request.UserId) ? _userManager.GetUserById(request.UserId) : null; | ||||||
|  | 
 | ||||||
|  |             var item = string.IsNullOrEmpty(request.Id) ? | ||||||
|  |                 (!string.IsNullOrWhiteSpace(request.UserId) ? user.RootFolder : | ||||||
|  |                 _libraryManager.RootFolder) : _libraryManager.GetItemById(request.Id); | ||||||
|  | 
 | ||||||
|  |             var itemsResult = _libraryManager.GetItemList(new InternalItemsQuery(user) | ||||||
|  |             { | ||||||
|  |                 Limit = request.Limit, | ||||||
|  |                 IncludeItemTypes = new[] | ||||||
|  |                 { | ||||||
|  |                         typeof(Game).Name | ||||||
|  |                 }, | ||||||
|  |                 SimilarTo = item | ||||||
|  | 
 | ||||||
|  |             }).ToList(); | ||||||
|  | 
 | ||||||
|  |             var dtoOptions = GetDtoOptions(request); | ||||||
|  | 
 | ||||||
|  |             var result = new QueryResult<BaseItemDto> | ||||||
|  |             { | ||||||
|  |                 Items = _dtoService.GetBaseItemDtos(itemsResult, dtoOptions, user).ToArray(), | ||||||
|  | 
 | ||||||
|  |                 TotalRecordCount = itemsResult.Count | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             return result; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -111,18 +111,16 @@ namespace MediaBrowser.Api.Movies | |||||||
|         /// </summary> |         /// </summary> | ||||||
|         /// <param name="request">The request.</param> |         /// <param name="request">The request.</param> | ||||||
|         /// <returns>System.Object.</returns> |         /// <returns>System.Object.</returns> | ||||||
|         public async Task<object> Get(GetSimilarMovies request) |         public object Get(GetSimilarMovies request) | ||||||
|         { |         { | ||||||
|             var result = await GetSimilarItemsResult( |             var result = GetSimilarItemsResult(request); | ||||||
|                 request, SimilarItemsHelper.GetSimiliarityScore).ConfigureAwait(false); |  | ||||||
| 
 | 
 | ||||||
|             return ToOptimizedSerializedResultUsingCache(result); |             return ToOptimizedSerializedResultUsingCache(result); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public async Task<object> Get(GetSimilarTrailers request) |         public object Get(GetSimilarTrailers request) | ||||||
|         { |         { | ||||||
|             var result = await GetSimilarItemsResult( |             var result = GetSimilarItemsResult(request); | ||||||
|                 request, SimilarItemsHelper.GetSimiliarityScore).ConfigureAwait(false); |  | ||||||
| 
 | 
 | ||||||
|             return ToOptimizedSerializedResultUsingCache(result); |             return ToOptimizedSerializedResultUsingCache(result); | ||||||
|         } |         } | ||||||
| @ -131,42 +129,16 @@ namespace MediaBrowser.Api.Movies | |||||||
|         { |         { | ||||||
|             var user = _userManager.GetUserById(request.UserId); |             var user = _userManager.GetUserById(request.UserId); | ||||||
| 
 | 
 | ||||||
|             var query = new InternalItemsQuery(user) |  | ||||||
|             { |  | ||||||
|                 IncludeItemTypes = new[] |  | ||||||
|                 { |  | ||||||
|                     typeof(Movie).Name, |  | ||||||
|                     typeof(Trailer).Name, |  | ||||||
|                     //typeof(LiveTvProgram).Name |  | ||||||
|                 }, |  | ||||||
|                 // IsMovie = true |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             var parentIds = string.IsNullOrWhiteSpace(request.ParentId) ? new string[] { } : new[] { request.ParentId }; |  | ||||||
|             var movies = _libraryManager.GetItemList(query, parentIds) |  | ||||||
|                 .OrderBy(i => (int)i.SourceType); |  | ||||||
| 
 |  | ||||||
|             var listEligibleForSuggestion = new List<BaseItem>(); |  | ||||||
| 
 |  | ||||||
|             var list = movies.ToList(); |  | ||||||
| 
 |  | ||||||
|             listEligibleForSuggestion.AddRange(list); |  | ||||||
| 
 |  | ||||||
|             listEligibleForSuggestion = listEligibleForSuggestion |  | ||||||
|                 .DistinctBy(i => i.Name, StringComparer.OrdinalIgnoreCase) |  | ||||||
|                 .DistinctBy(i => i.GetProviderId(MetadataProviders.Imdb) ?? Guid.NewGuid().ToString(), StringComparer.OrdinalIgnoreCase) |  | ||||||
|                 .ToList(); |  | ||||||
| 
 |  | ||||||
|             var dtoOptions = GetDtoOptions(request); |             var dtoOptions = GetDtoOptions(request); | ||||||
| 
 | 
 | ||||||
|             dtoOptions.Fields = request.GetItemFields().ToList(); |             dtoOptions.Fields = request.GetItemFields().ToList(); | ||||||
| 
 | 
 | ||||||
|             var result = GetRecommendationCategories(user, request.ParentId, listEligibleForSuggestion, request.CategoryLimit, request.ItemLimit, dtoOptions); |             var result = GetRecommendationCategories(user, request.ParentId, request.CategoryLimit, request.ItemLimit, dtoOptions); | ||||||
| 
 | 
 | ||||||
|             return ToOptimizedResult(result); |             return ToOptimizedResult(result); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private async Task<ItemsResult> GetSimilarItemsResult(BaseGetSimilarItemsFromItem request, Func<BaseItem, List<PersonInfo>, List<PersonInfo>, BaseItem, int> getSimilarityScore) |         private QueryResult<BaseItemDto> GetSimilarItemsResult(BaseGetSimilarItemsFromItem request) | ||||||
|         { |         { | ||||||
|             var user = !string.IsNullOrWhiteSpace(request.UserId) ? _userManager.GetUserById(request.UserId) : null; |             var user = !string.IsNullOrWhiteSpace(request.UserId) ? _userManager.GetUserById(request.UserId) : null; | ||||||
| 
 | 
 | ||||||
| @ -174,57 +146,32 @@ namespace MediaBrowser.Api.Movies | |||||||
|                 (!string.IsNullOrWhiteSpace(request.UserId) ? user.RootFolder : |                 (!string.IsNullOrWhiteSpace(request.UserId) ? user.RootFolder : | ||||||
|                 _libraryManager.RootFolder) : _libraryManager.GetItemById(request.Id); |                 _libraryManager.RootFolder) : _libraryManager.GetItemById(request.Id); | ||||||
| 
 | 
 | ||||||
|             var query = new InternalItemsQuery(user) |             var itemsResult = _libraryManager.GetItemList(new InternalItemsQuery(user) | ||||||
|             { |             { | ||||||
|  |                 Limit = request.Limit, | ||||||
|                 IncludeItemTypes = new[] |                 IncludeItemTypes = new[] | ||||||
|                 { |                 { | ||||||
|                     typeof(Movie).Name, |                         typeof(Movie).Name, | ||||||
|                     typeof(Trailer).Name, |                         typeof(Trailer).Name, | ||||||
|                     //typeof(LiveTvProgram).Name |                         typeof(LiveTvProgram).Name | ||||||
|                 }, |                 }, | ||||||
|                 //IsMovie = true |                 IsMovie = true, | ||||||
|             }; |                 SimilarTo = item | ||||||
| 
 |             }).ToList(); | ||||||
|             var list = _libraryManager.GetItemList(query) |  | ||||||
|                 .OrderBy(i => (int)i.SourceType) |  | ||||||
|                 .DistinctBy(i => i.GetProviderId(MetadataProviders.Imdb) ?? Guid.NewGuid().ToString("N")) |  | ||||||
|                 .ToList(); |  | ||||||
| 
 |  | ||||||
|             if (item is Video) |  | ||||||
|             { |  | ||||||
|                 var imdbId = item.GetProviderId(MetadataProviders.Imdb); |  | ||||||
| 
 |  | ||||||
|                 // Use imdb id to try to filter duplicates of the same item |  | ||||||
|                 if (!string.IsNullOrWhiteSpace(imdbId)) |  | ||||||
|                 { |  | ||||||
|                     list = list |  | ||||||
|                         .Where(i => !string.Equals(imdbId, i.GetProviderId(MetadataProviders.Imdb), StringComparison.OrdinalIgnoreCase)) |  | ||||||
|                         .ToList(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var items = SimilarItemsHelper.GetSimilaritems(item, _libraryManager, list, getSimilarityScore).ToList(); |  | ||||||
| 
 |  | ||||||
|             IEnumerable<BaseItem> returnItems = items; |  | ||||||
| 
 |  | ||||||
|             if (request.Limit.HasValue) |  | ||||||
|             { |  | ||||||
|                 returnItems = returnItems.Take(request.Limit.Value); |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|             var dtoOptions = GetDtoOptions(request); |             var dtoOptions = GetDtoOptions(request); | ||||||
| 
 | 
 | ||||||
|             var result = new ItemsResult |             var result = new QueryResult<BaseItemDto> | ||||||
|             { |             { | ||||||
|                 Items = _dtoService.GetBaseItemDtos(returnItems, dtoOptions, user).ToArray(), |                 Items = _dtoService.GetBaseItemDtos(itemsResult, dtoOptions, user).ToArray(), | ||||||
| 
 | 
 | ||||||
|                 TotalRecordCount = items.Count |                 TotalRecordCount = itemsResult.Count | ||||||
|             }; |             }; | ||||||
| 
 | 
 | ||||||
|             return result; |             return result; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private IEnumerable<RecommendationDto> GetRecommendationCategories(User user, string parentId, List<BaseItem> allMovies, int categoryLimit, int itemLimit, DtoOptions dtoOptions) |         private IEnumerable<RecommendationDto> GetRecommendationCategories(User user, string parentId, int categoryLimit, int itemLimit, DtoOptions dtoOptions) | ||||||
|         { |         { | ||||||
|             var categories = new List<RecommendationDto>(); |             var categories = new List<RecommendationDto>(); | ||||||
| 
 | 
 | ||||||
| @ -260,7 +207,7 @@ namespace MediaBrowser.Api.Movies | |||||||
|                 IsFavoriteOrLiked = true, |                 IsFavoriteOrLiked = true, | ||||||
|                 ExcludeItemIds = recentlyPlayedMovies.Select(i => i.Id.ToString("N")).ToArray() |                 ExcludeItemIds = recentlyPlayedMovies.Select(i => i.Id.ToString("N")).ToArray() | ||||||
| 
 | 
 | ||||||
|             }, parentIds); |             }, parentIds).ToList(); | ||||||
| 
 | 
 | ||||||
|             var mostRecentMovies = recentlyPlayedMovies.Take(6).ToList(); |             var mostRecentMovies = recentlyPlayedMovies.Take(6).ToList(); | ||||||
|             // Get recently played directors |             // Get recently played directors | ||||||
| @ -273,8 +220,8 @@ namespace MediaBrowser.Api.Movies | |||||||
|                 .OrderBy(i => Guid.NewGuid()) |                 .OrderBy(i => Guid.NewGuid()) | ||||||
|                 .ToList(); |                 .ToList(); | ||||||
| 
 | 
 | ||||||
|             var similarToRecentlyPlayed = GetSimilarTo(user, allMovies, recentlyPlayedMovies.Take(7).OrderBy(i => Guid.NewGuid()), itemLimit, dtoOptions, RecommendationType.SimilarToRecentlyPlayed).GetEnumerator(); |             var similarToRecentlyPlayed = GetSimilarTo(user, recentlyPlayedMovies, itemLimit, dtoOptions, RecommendationType.SimilarToRecentlyPlayed).GetEnumerator(); | ||||||
|             var similarToLiked = GetSimilarTo(user, allMovies, likedMovies, itemLimit, dtoOptions, RecommendationType.SimilarToLikedItem).GetEnumerator(); |             var similarToLiked = GetSimilarTo(user, likedMovies, itemLimit, dtoOptions, RecommendationType.SimilarToLikedItem).GetEnumerator(); | ||||||
| 
 | 
 | ||||||
|             var hasDirectorFromRecentlyPlayed = GetWithDirector(user, recentDirectors, itemLimit, dtoOptions, RecommendationType.HasDirectorFromRecentlyPlayed).GetEnumerator(); |             var hasDirectorFromRecentlyPlayed = GetWithDirector(user, recentDirectors, itemLimit, dtoOptions, RecommendationType.HasDirectorFromRecentlyPlayed).GetEnumerator(); | ||||||
|             var hasActorFromRecentlyPlayed = GetWithActor(user, recentActors, itemLimit, dtoOptions, RecommendationType.HasActorFromRecentlyPlayed).GetEnumerator(); |             var hasActorFromRecentlyPlayed = GetWithActor(user, recentActors, itemLimit, dtoOptions, RecommendationType.HasActorFromRecentlyPlayed).GetEnumerator(); | ||||||
| @ -389,14 +336,23 @@ namespace MediaBrowser.Api.Movies | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private IEnumerable<RecommendationDto> GetSimilarTo(User user, List<BaseItem> allMovies, IEnumerable<BaseItem> baselineItems, int itemLimit, DtoOptions dtoOptions, RecommendationType type) |         private IEnumerable<RecommendationDto> GetSimilarTo(User user, List<BaseItem> baselineItems, int itemLimit, DtoOptions dtoOptions, RecommendationType type) | ||||||
|         { |         { | ||||||
|             foreach (var item in baselineItems) |             foreach (var item in baselineItems) | ||||||
|             { |             { | ||||||
|                 var similar = SimilarItemsHelper |                 var similar = _libraryManager.GetItemList(new InternalItemsQuery(user) | ||||||
|                     .GetSimilaritems(item, _libraryManager, allMovies, SimilarItemsHelper.GetSimiliarityScore) |                 { | ||||||
|                     .Take(itemLimit) |                     Limit = itemLimit, | ||||||
|                     .ToList(); |                     IncludeItemTypes = new[] | ||||||
|  |                     { | ||||||
|  |                         typeof(Movie).Name, | ||||||
|  |                         typeof(Trailer).Name, | ||||||
|  |                         typeof(LiveTvProgram).Name | ||||||
|  |                     }, | ||||||
|  |                     IsMovie = true, | ||||||
|  |                     SimilarTo = item | ||||||
|  | 
 | ||||||
|  |                 }).ToList(); | ||||||
| 
 | 
 | ||||||
|                 if (similar.Count > 0) |                 if (similar.Count > 0) | ||||||
|                 { |                 { | ||||||
|  | |||||||
| @ -12,6 +12,7 @@ using ServiceStack; | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Linq; | using System.Linq; | ||||||
|  | using MediaBrowser.Model.Dto; | ||||||
| 
 | 
 | ||||||
| namespace MediaBrowser.Api | namespace MediaBrowser.Api | ||||||
| { | { | ||||||
| @ -273,20 +274,42 @@ namespace MediaBrowser.Api | |||||||
|         /// <returns>System.Object.</returns> |         /// <returns>System.Object.</returns> | ||||||
|         public object Get(GetSimilarShows request) |         public object Get(GetSimilarShows request) | ||||||
|         { |         { | ||||||
|             var dtoOptions = GetDtoOptions(request); |             var result = GetSimilarItemsResult(request); | ||||||
| 
 |  | ||||||
|             var result = SimilarItemsHelper.GetSimilarItemsResult(dtoOptions, _userManager, |  | ||||||
|                 _itemRepo, |  | ||||||
|                 _libraryManager, |  | ||||||
|                 _userDataManager, |  | ||||||
|                 _dtoService, |  | ||||||
|                 Logger, |  | ||||||
|                 request, new[] { typeof(Series) }, |  | ||||||
|                 SimilarItemsHelper.GetSimiliarityScore); |  | ||||||
| 
 | 
 | ||||||
|             return ToOptimizedSerializedResultUsingCache(result); |             return ToOptimizedSerializedResultUsingCache(result); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         private QueryResult<BaseItemDto> GetSimilarItemsResult(BaseGetSimilarItemsFromItem request) | ||||||
|  |         { | ||||||
|  |             var user = !string.IsNullOrWhiteSpace(request.UserId) ? _userManager.GetUserById(request.UserId) : null; | ||||||
|  | 
 | ||||||
|  |             var item = string.IsNullOrEmpty(request.Id) ? | ||||||
|  |                 (!string.IsNullOrWhiteSpace(request.UserId) ? user.RootFolder : | ||||||
|  |                 _libraryManager.RootFolder) : _libraryManager.GetItemById(request.Id); | ||||||
|  | 
 | ||||||
|  |             var itemsResult = _libraryManager.GetItemList(new InternalItemsQuery(user) | ||||||
|  |             { | ||||||
|  |                 Limit = request.Limit, | ||||||
|  |                 IncludeItemTypes = new[] | ||||||
|  |                 { | ||||||
|  |                         typeof(Series).Name | ||||||
|  |                 }, | ||||||
|  |                 SimilarTo = item | ||||||
|  | 
 | ||||||
|  |             }).ToList(); | ||||||
|  | 
 | ||||||
|  |             var dtoOptions = GetDtoOptions(request); | ||||||
|  | 
 | ||||||
|  |             var result = new QueryResult<BaseItemDto> | ||||||
|  |             { | ||||||
|  |                 Items = _dtoService.GetBaseItemDtos(itemsResult, dtoOptions, user).ToArray(), | ||||||
|  | 
 | ||||||
|  |                 TotalRecordCount = itemsResult.Count | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             return result; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         public object Get(GetUpcomingEpisodes request) |         public object Get(GetUpcomingEpisodes request) | ||||||
|         { |         { | ||||||
|             var user = _userManager.GetUserById(request.UserId); |             var user = _userManager.GetUserById(request.UserId); | ||||||
|  | |||||||
| @ -793,11 +793,6 @@ namespace MediaBrowser.Controller.Entities | |||||||
|                     Logger.Debug("Query requires post-filtering due to ItemSortBy.Metascore"); |                     Logger.Debug("Query requires post-filtering due to ItemSortBy.Metascore"); | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (query.SortBy.Contains(ItemSortBy.OfficialRating, StringComparer.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     Logger.Debug("Query requires post-filtering due to ItemSortBy.OfficialRating"); |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|                 if (query.SortBy.Contains(ItemSortBy.Players, StringComparer.OrdinalIgnoreCase)) |                 if (query.SortBy.Contains(ItemSortBy.Players, StringComparer.OrdinalIgnoreCase)) | ||||||
|                 { |                 { | ||||||
|                     Logger.Debug("Query requires post-filtering due to ItemSortBy.Players"); |                     Logger.Debug("Query requires post-filtering due to ItemSortBy.Players"); | ||||||
| @ -813,11 +808,6 @@ namespace MediaBrowser.Controller.Entities | |||||||
|                     Logger.Debug("Query requires post-filtering due to ItemSortBy.SeriesSortName"); |                     Logger.Debug("Query requires post-filtering due to ItemSortBy.SeriesSortName"); | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (query.SortBy.Contains(ItemSortBy.Studio, StringComparer.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     Logger.Debug("Query requires post-filtering due to ItemSortBy.Studio"); |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|                 if (query.SortBy.Contains(ItemSortBy.VideoBitRate, StringComparer.OrdinalIgnoreCase)) |                 if (query.SortBy.Contains(ItemSortBy.VideoBitRate, StringComparer.OrdinalIgnoreCase)) | ||||||
|                 { |                 { | ||||||
|                     Logger.Debug("Query requires post-filtering due to ItemSortBy.VideoBitRate"); |                     Logger.Debug("Query requires post-filtering due to ItemSortBy.VideoBitRate"); | ||||||
| @ -962,12 +952,6 @@ namespace MediaBrowser.Controller.Entities | |||||||
|                 return true; |                 return true; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             if (query.OfficialRatings.Length > 0) |  | ||||||
|             { |  | ||||||
|                 Logger.Debug("Query requires post-filtering due to OfficialRatings"); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (query.IsMissing.HasValue) |             if (query.IsMissing.HasValue) | ||||||
|             { |             { | ||||||
|                 Logger.Debug("Query requires post-filtering due to IsMissing"); |                 Logger.Debug("Query requires post-filtering due to IsMissing"); | ||||||
|  | |||||||
| @ -19,6 +19,8 @@ namespace MediaBrowser.Controller.Entities | |||||||
| 
 | 
 | ||||||
|         public User User { get; set; } |         public User User { get; set; } | ||||||
| 
 | 
 | ||||||
|  |         public BaseItem SimilarTo { get; set; } | ||||||
|  | 
 | ||||||
|         public bool? IsFolder { get; set; } |         public bool? IsFolder { get; set; } | ||||||
|         public bool? IsFavorite { get; set; } |         public bool? IsFavorite { get; set; } | ||||||
|         public bool? IsFavoriteOrLiked { get; set; } |         public bool? IsFavoriteOrLiked { get; set; } | ||||||
|  | |||||||
| @ -11,13 +11,25 @@ namespace MediaBrowser.Controller.Entities.TV | |||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Class Episode |     /// Class Episode | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class Episode : Video, IHasLookupInfo<EpisodeInfo>, IHasSeries |     public class Episode : Video, IHasTrailers, IHasLookupInfo<EpisodeInfo>, IHasSeries | ||||||
|     { |     { | ||||||
|         /// <summary> | 
 | ||||||
|         /// Gets the season in which it aired. |         public Episode() | ||||||
|         /// </summary> |         { | ||||||
|         /// <value>The aired season.</value> |             RemoteTrailers = new List<MediaUrl>(); | ||||||
|         public int? AirsBeforeSeasonNumber { get; set; } |             LocalTrailerIds = new List<Guid>(); | ||||||
|  |             RemoteTrailerIds = new List<Guid>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public List<Guid> LocalTrailerIds { get; set; } | ||||||
|  |         public List<Guid> RemoteTrailerIds { get; set; } | ||||||
|  |         public List<MediaUrl> RemoteTrailers { get; set; } | ||||||
|  | 
 | ||||||
|  |     /// <summary> | ||||||
|  |     /// Gets the season in which it aired. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <value>The aired season.</value> | ||||||
|  |     public int? AirsBeforeSeasonNumber { get; set; } | ||||||
|         public int? AirsAfterSeasonNumber { get; set; } |         public int? AirsAfterSeasonNumber { get; set; } | ||||||
|         public int? AirsBeforeEpisodeNumber { get; set; } |         public int? AirsBeforeEpisodeNumber { get; set; } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -148,6 +148,13 @@ namespace MediaBrowser.Providers.Movies | |||||||
| 
 | 
 | ||||||
|         public bool Supports(IHasProviderIds item) |         public bool Supports(IHasProviderIds item) | ||||||
|         { |         { | ||||||
|  |             // Supports images for tv movies | ||||||
|  |             var tvProgram = item as LiveTvProgram; | ||||||
|  |             if (tvProgram != null && tvProgram.IsMovie) | ||||||
|  |             { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             return item is Movie || item is MusicVideo || item is Series || item is Episode || item is Trailer; |             return item is Movie || item is MusicVideo || item is Series || item is Episode || item is Trailer; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -63,16 +63,8 @@ namespace MediaBrowser.Server.Implementations.Intros | |||||||
|                 ? null |                 ? null | ||||||
|                 : _localization.GetRatingLevel(item.OfficialRating); |                 : _localization.GetRatingLevel(item.OfficialRating); | ||||||
| 
 | 
 | ||||||
|             var random = new Random(Environment.TickCount + Guid.NewGuid().GetHashCode()); |  | ||||||
| 
 |  | ||||||
|             var candidates = new List<ItemWithTrailer>(); |             var candidates = new List<ItemWithTrailer>(); | ||||||
| 
 | 
 | ||||||
|             var itemPeople = _libraryManager.GetPeople(item); |  | ||||||
|             var allPeople = _libraryManager.GetPeople(new InternalPeopleQuery |  | ||||||
|             { |  | ||||||
|                 AppearsInItemId = item.Id |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             var trailerTypes = new List<TrailerType>(); |             var trailerTypes = new List<TrailerType>(); | ||||||
| 
 | 
 | ||||||
|             if (config.EnableIntrosFromMoviesInLibrary) |             if (config.EnableIntrosFromMoviesInLibrary) | ||||||
| @ -105,26 +97,25 @@ namespace MediaBrowser.Server.Implementations.Intros | |||||||
|                 var trailerResult = _libraryManager.GetItemList(new InternalItemsQuery |                 var trailerResult = _libraryManager.GetItemList(new InternalItemsQuery | ||||||
|                 { |                 { | ||||||
|                     IncludeItemTypes = new[] { typeof(Trailer).Name }, |                     IncludeItemTypes = new[] { typeof(Trailer).Name }, | ||||||
|                     TrailerTypes = trailerTypes.ToArray() |                     TrailerTypes = trailerTypes.ToArray(), | ||||||
|  |                     SimilarTo = item, | ||||||
|  |                     IsPlayed = config.EnableIntrosForWatchedContent ? (bool?) null : false, | ||||||
|  |                     MaxParentalRating = config.EnableIntrosParentalControl ? ratingLevel : null, | ||||||
|  |                     Limit = config.TrailerLimit | ||||||
|                 }); |                 }); | ||||||
| 
 | 
 | ||||||
|                 candidates.AddRange(trailerResult.Select(i => new ItemWithTrailer |                 candidates.AddRange(trailerResult.Select(i => new ItemWithTrailer | ||||||
|                 { |                 { | ||||||
|                     Item = i, |                     Item = i, | ||||||
|                     Type = i.SourceType == SourceType.Channel ? ItemWithTrailerType.ChannelTrailer : ItemWithTrailerType.ItemWithTrailer, |                     Type = i.SourceType == SourceType.Channel ? ItemWithTrailerType.ChannelTrailer : ItemWithTrailerType.ItemWithTrailer, | ||||||
|                     User = user, |  | ||||||
|                     WatchingItem = item, |  | ||||||
|                     WatchingItemPeople = itemPeople, |  | ||||||
|                     AllPeople = allPeople, |  | ||||||
|                     Random = random, |  | ||||||
|                     LibraryManager = _libraryManager |                     LibraryManager = _libraryManager | ||||||
|                 })); |                 })); | ||||||
|             }  |             }  | ||||||
| 
 | 
 | ||||||
|             return GetResult(item, candidates, config, ratingLevel); |             return GetResult(item, candidates, config); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private IEnumerable<IntroInfo> GetResult(BaseItem item, IEnumerable<ItemWithTrailer> candidates, CinemaModeConfiguration config, int? ratingLevel) |         private IEnumerable<IntroInfo> GetResult(BaseItem item, IEnumerable<ItemWithTrailer> candidates, CinemaModeConfiguration config) | ||||||
|         { |         { | ||||||
|             var customIntros = !string.IsNullOrWhiteSpace(config.CustomIntroPath) ? |             var customIntros = !string.IsNullOrWhiteSpace(config.CustomIntroPath) ? | ||||||
|                 GetCustomIntros(config) : |                 GetCustomIntros(config) : | ||||||
| @ -134,48 +125,12 @@ namespace MediaBrowser.Server.Implementations.Intros | |||||||
|                 GetMediaInfoIntros(config, item) : |                 GetMediaInfoIntros(config, item) : | ||||||
|                 new List<IntroInfo>(); |                 new List<IntroInfo>(); | ||||||
| 
 | 
 | ||||||
|             var trailerLimit = config.TrailerLimit; |  | ||||||
| 
 |  | ||||||
|             // Avoid implicitly captured closure |             // Avoid implicitly captured closure | ||||||
|             return candidates.Where(i => |             return candidates.Select(i => i.IntroInfo) | ||||||
|             { |  | ||||||
|                 if (config.EnableIntrosParentalControl && !FilterByParentalRating(ratingLevel, i.Item)) |  | ||||||
|                 { |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 if (!config.EnableIntrosForWatchedContent && i.IsPlayed) |  | ||||||
|                 { |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
|                 return !IsDuplicate(item, i.Item); |  | ||||||
|             }) |  | ||||||
|                 .OrderByDescending(i => i.Score) |  | ||||||
|                 .ThenBy(i => Guid.NewGuid()) |  | ||||||
|                 .ThenByDescending(i => i.IsPlayed ? 0 : 1) |  | ||||||
|                 .Select(i => i.IntroInfo) |  | ||||||
|                 .Take(trailerLimit) |  | ||||||
|                 .Concat(customIntros.Take(1)) |                 .Concat(customIntros.Take(1)) | ||||||
|                 .Concat(mediaInfoIntros); |                 .Concat(mediaInfoIntros); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private bool IsDuplicate(BaseItem playingContent, BaseItem test) |  | ||||||
|         { |  | ||||||
|             var id = playingContent.GetProviderId(MetadataProviders.Imdb); |  | ||||||
|             if (!string.IsNullOrWhiteSpace(id) && string.Equals(id, test.GetProviderId(MetadataProviders.Imdb), StringComparison.OrdinalIgnoreCase)) |  | ||||||
|             { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             id = playingContent.GetProviderId(MetadataProviders.Tmdb); |  | ||||||
|             if (!string.IsNullOrWhiteSpace(id) && string.Equals(id, test.GetProviderId(MetadataProviders.Tmdb), StringComparison.OrdinalIgnoreCase)) |  | ||||||
|             { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private CinemaModeConfiguration GetOptions() |         private CinemaModeConfiguration GetOptions() | ||||||
|         { |         { | ||||||
|             return _serverConfig.GetConfiguration<CinemaModeConfiguration>("cinemamode"); |             return _serverConfig.GetConfiguration<CinemaModeConfiguration>("cinemamode"); | ||||||
| @ -346,96 +301,6 @@ namespace MediaBrowser.Server.Implementations.Intros | |||||||
|             return list.Distinct(StringComparer.OrdinalIgnoreCase); |             return list.Distinct(StringComparer.OrdinalIgnoreCase); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private bool FilterByParentalRating(int? ratingLevel, BaseItem item) |  | ||||||
|         { |  | ||||||
|             // Only content rated same or lower |  | ||||||
|             if (ratingLevel.HasValue) |  | ||||||
|             { |  | ||||||
|                 var level = string.IsNullOrWhiteSpace(item.OfficialRating) |  | ||||||
|                     ? (int?)null |  | ||||||
|                     : _localization.GetRatingLevel(item.OfficialRating); |  | ||||||
| 
 |  | ||||||
|                 return level.HasValue && level.Value <= ratingLevel.Value; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         internal static int GetSimiliarityScore(BaseItem item1, List<PersonInfo> item1People, List<PersonInfo> allPeople, BaseItem item2, Random random, ILibraryManager libraryManager) |  | ||||||
|         { |  | ||||||
|             var points = 0; |  | ||||||
| 
 |  | ||||||
|             if (!string.IsNullOrEmpty(item1.OfficialRating) && string.Equals(item1.OfficialRating, item2.OfficialRating, StringComparison.OrdinalIgnoreCase)) |  | ||||||
|             { |  | ||||||
|                 points += 10; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             // Find common genres |  | ||||||
|             points += item1.Genres.Where(i => item2.Genres.Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 10); |  | ||||||
| 
 |  | ||||||
|             // Find common tags |  | ||||||
|             points += GetTags(item1).Where(i => GetTags(item2).Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 10); |  | ||||||
| 
 |  | ||||||
|             // Find common keywords |  | ||||||
|             points += GetKeywords(item1).Where(i => GetKeywords(item2).Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 10); |  | ||||||
| 
 |  | ||||||
|             // Find common studios |  | ||||||
|             points += item1.Studios.Where(i => item2.Studios.Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 5); |  | ||||||
| 
 |  | ||||||
|             var item2PeopleNames = allPeople.Where(i => i.ItemId == item2.Id) |  | ||||||
|                 .Select(i => i.Name) |  | ||||||
|                 .Where(i => !string.IsNullOrWhiteSpace(i)) |  | ||||||
|                 .DistinctNames() |  | ||||||
|                 .ToDictionary(i => i, StringComparer.OrdinalIgnoreCase); |  | ||||||
| 
 |  | ||||||
|             points += item1People.Where(i => item2PeopleNames.ContainsKey(i.Name)).Sum(i => |  | ||||||
|             { |  | ||||||
|                 if (string.Equals(i.Type, PersonType.Director, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Director, StringComparison.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     return 5; |  | ||||||
|                 } |  | ||||||
|                 if (string.Equals(i.Type, PersonType.Actor, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Actor, StringComparison.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     return 3; |  | ||||||
|                 } |  | ||||||
|                 if (string.Equals(i.Type, PersonType.Composer, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Composer, StringComparison.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     return 3; |  | ||||||
|                 } |  | ||||||
|                 if (string.Equals(i.Type, PersonType.GuestStar, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.GuestStar, StringComparison.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     return 3; |  | ||||||
|                 } |  | ||||||
|                 if (string.Equals(i.Type, PersonType.Writer, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Writer, StringComparison.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     return 2; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 return 1; |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             // Add some randomization so that you're not always seeing the same ones for a given movie |  | ||||||
|             points += random.Next(0, 50); |  | ||||||
| 
 |  | ||||||
|             return points; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static IEnumerable<string> GetTags(BaseItem item) |  | ||||||
|         { |  | ||||||
|             var hasTags = item as IHasTags; |  | ||||||
|             if (hasTags != null) |  | ||||||
|             { |  | ||||||
|                 return hasTags.Tags; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return new List<string>(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static IEnumerable<string> GetKeywords(BaseItem item) |  | ||||||
|         { |  | ||||||
|             return item.Keywords; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public IEnumerable<string> GetAllIntroFiles() |         public IEnumerable<string> GetAllIntroFiles() | ||||||
|         { |         { | ||||||
|             return GetCustomIntroFiles(GetOptions(), true, true); |             return GetCustomIntroFiles(GetOptions(), true, true); | ||||||
| @ -455,39 +320,8 @@ namespace MediaBrowser.Server.Implementations.Intros | |||||||
|         { |         { | ||||||
|             internal BaseItem Item; |             internal BaseItem Item; | ||||||
|             internal ItemWithTrailerType Type; |             internal ItemWithTrailerType Type; | ||||||
|             internal User User; |  | ||||||
|             internal BaseItem WatchingItem; |  | ||||||
|             internal List<PersonInfo> WatchingItemPeople; |  | ||||||
|             internal List<PersonInfo> AllPeople; |  | ||||||
|             internal Random Random; |  | ||||||
|             internal ILibraryManager LibraryManager; |             internal ILibraryManager LibraryManager; | ||||||
| 
 | 
 | ||||||
|             private bool? _isPlayed; |  | ||||||
|             public bool IsPlayed |  | ||||||
|             { |  | ||||||
|                 get |  | ||||||
|                 { |  | ||||||
|                     if (!_isPlayed.HasValue) |  | ||||||
|                     { |  | ||||||
|                         _isPlayed = Item.IsPlayed(User); |  | ||||||
|                     } |  | ||||||
|                     return _isPlayed.Value; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             private int? _score; |  | ||||||
|             public int Score |  | ||||||
|             { |  | ||||||
|                 get |  | ||||||
|                 { |  | ||||||
|                     if (!_score.HasValue) |  | ||||||
|                     { |  | ||||||
|                         _score = GetSimiliarityScore(WatchingItem, WatchingItemPeople, AllPeople, Item, Random, LibraryManager); |  | ||||||
|                     } |  | ||||||
|                     return _score.Value; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             public IntroInfo IntroInfo |             public IntroInfo IntroInfo | ||||||
|             { |             { | ||||||
|                 get |                 get | ||||||
|  | |||||||
| @ -2309,7 +2309,7 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
| 
 | 
 | ||||||
|         public IEnumerable<Video> FindTrailers(BaseItem owner, List<FileSystemMetadata> fileSystemChildren, IDirectoryService directoryService) |         public IEnumerable<Video> FindTrailers(BaseItem owner, List<FileSystemMetadata> fileSystemChildren, IDirectoryService directoryService) | ||||||
|         { |         { | ||||||
|             var files = fileSystemChildren.Where(i => i.IsDirectory) |             var files = owner.IsInMixedFolder ? new List<FileSystemMetadata>() : fileSystemChildren.Where(i => i.IsDirectory) | ||||||
|                 .Where(i => string.Equals(i.Name, BaseItem.TrailerFolderName, StringComparison.OrdinalIgnoreCase)) |                 .Where(i => string.Equals(i.Name, BaseItem.TrailerFolderName, StringComparison.OrdinalIgnoreCase)) | ||||||
|                 .SelectMany(i => _fileSystem.GetFiles(i.FullName, false)) |                 .SelectMany(i => _fileSystem.GetFiles(i.FullName, false)) | ||||||
|                 .ToList(); |                 .ToList(); | ||||||
|  | |||||||
| @ -46,11 +46,12 @@ | |||||||
|       <HintPath>..\packages\CommonIO.1.0.0.9\lib\net45\CommonIO.dll</HintPath> |       <HintPath>..\packages\CommonIO.1.0.0.9\lib\net45\CommonIO.dll</HintPath> | ||||||
|     </Reference> |     </Reference> | ||||||
|     <Reference Include="Emby.XmlTv, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL"> |     <Reference Include="Emby.XmlTv, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL"> | ||||||
|       <SpecificVersion>False</SpecificVersion> |       <HintPath>..\packages\Emby.XmlTv.1.0.0.50\lib\net45\Emby.XmlTv.dll</HintPath> | ||||||
|       <HintPath>..\packages\Emby.XmlTv.1.0.0.48\lib\net45\Emby.XmlTv.dll</HintPath> |       <Private>True</Private> | ||||||
|     </Reference> |     </Reference> | ||||||
|     <Reference Include="INIFileParser"> |     <Reference Include="INIFileParser, Version=2.3.0.0, Culture=neutral, PublicKeyToken=79af7b307b65cf3c, processorArchitecture=MSIL"> | ||||||
|       <HintPath>..\packages\ini-parser.2.2.4\lib\net20\INIFileParser.dll</HintPath> |       <HintPath>..\packages\ini-parser.2.3.0\lib\net20\INIFileParser.dll</HintPath> | ||||||
|  |       <Private>True</Private> | ||||||
|     </Reference> |     </Reference> | ||||||
|     <Reference Include="Interfaces.IO"> |     <Reference Include="Interfaces.IO"> | ||||||
|       <HintPath>..\packages\Interfaces.IO.1.0.0.5\lib\portable-net45+sl4+wp71+win8+wpa81\Interfaces.IO.dll</HintPath> |       <HintPath>..\packages\Interfaces.IO.1.0.0.5\lib\portable-net45+sl4+wp71+win8+wpa81\Interfaces.IO.dll</HintPath> | ||||||
|  | |||||||
| @ -6,5 +6,6 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|     public interface IDbConnector |     public interface IDbConnector | ||||||
|     { |     { | ||||||
|         Task<IDbConnection> Connect(string dbPath); |         Task<IDbConnection> Connect(string dbPath); | ||||||
|  |         void BindSimilarityScoreFunction(IDbConnection connection); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -0,0 +1,175 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Data; | ||||||
|  | using System.Data.SQLite; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Text; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Persistence | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class SQLiteExtensions | ||||||
|  |     /// </summary> | ||||||
|  |     public static class SqliteExtensions | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Connects to db. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="dbPath">The db path.</param> | ||||||
|  |         /// <param name="logger">The logger.</param> | ||||||
|  |         /// <returns>Task{IDbConnection}.</returns> | ||||||
|  |         /// <exception cref="System.ArgumentNullException">dbPath</exception> | ||||||
|  |         public static async Task<IDbConnection> ConnectToDb(string dbPath, ILogger logger) | ||||||
|  |         { | ||||||
|  |             if (string.IsNullOrEmpty(dbPath)) | ||||||
|  |             { | ||||||
|  |                 throw new ArgumentNullException("dbPath"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             logger.Info("Sqlite {0} opening {1}", SQLiteConnection.SQLiteVersion, dbPath); | ||||||
|  | 
 | ||||||
|  |             var connectionstr = new SQLiteConnectionStringBuilder | ||||||
|  |             { | ||||||
|  |                 PageSize = 4096, | ||||||
|  |                 CacheSize = 2000, | ||||||
|  |                 SyncMode = SynchronizationModes.Normal, | ||||||
|  |                 DataSource = dbPath, | ||||||
|  |                 JournalMode = SQLiteJournalModeEnum.Wal | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             var connection = new SQLiteConnection(connectionstr.ConnectionString); | ||||||
|  | 
 | ||||||
|  |             await connection.OpenAsync().ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             return connection; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static void BindGetSimilarityScore(IDbConnection connection, ILogger logger) | ||||||
|  |         { | ||||||
|  |             var sqlConnection = (SQLiteConnection) connection; | ||||||
|  |             SimiliarToFunction.Logger = logger; | ||||||
|  |             sqlConnection.BindFunction(new SimiliarToFunction()); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static void BindFunction(this SQLiteConnection connection, SQLiteFunction function) | ||||||
|  |         { | ||||||
|  |             var attributes = function.GetType().GetCustomAttributes(typeof(SQLiteFunctionAttribute), true).Cast<SQLiteFunctionAttribute>().ToArray(); | ||||||
|  |             if (attributes.Length == 0) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidOperationException("SQLiteFunction doesn't have SQLiteFunctionAttribute"); | ||||||
|  |             } | ||||||
|  |             connection.BindFunction(attributes[0], function); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     [SQLiteFunction(Name = "GetSimilarityScore", Arguments = 12, FuncType = FunctionType.Scalar)] | ||||||
|  |     public class SimiliarToFunction : SQLiteFunction | ||||||
|  |     { | ||||||
|  |         internal static ILogger Logger; | ||||||
|  | 
 | ||||||
|  |         public override object Invoke(object[] args) | ||||||
|  |         { | ||||||
|  |             var score = 0; | ||||||
|  | 
 | ||||||
|  |             var inputOfficialRating = args[0] as string; | ||||||
|  |             var rowOfficialRating = args[1] as string; | ||||||
|  |             if (!string.IsNullOrWhiteSpace(inputOfficialRating) && string.Equals(inputOfficialRating, rowOfficialRating)) | ||||||
|  |             { | ||||||
|  |                 score += 10; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             long? inputYear = args[2] == null ? (long?)null : (long)args[2]; | ||||||
|  |             long? rowYear = args[3] == null ? (long?)null : (long)args[3]; | ||||||
|  | 
 | ||||||
|  |             if (inputYear.HasValue && rowYear.HasValue) | ||||||
|  |             { | ||||||
|  |                 var diff = Math.Abs(inputYear.Value - rowYear.Value); | ||||||
|  | 
 | ||||||
|  |                 // Add if they came out within the same decade | ||||||
|  |                 if (diff < 10) | ||||||
|  |                 { | ||||||
|  |                     score += 2; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // And more if within five years | ||||||
|  |                 if (diff < 5) | ||||||
|  |                 { | ||||||
|  |                     score += 2; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // genres | ||||||
|  |             score += GetListScore(args, 4, 5); | ||||||
|  | 
 | ||||||
|  |             // tags | ||||||
|  |             score += GetListScore(args, 6, 7); | ||||||
|  | 
 | ||||||
|  |             // keywords | ||||||
|  |             score += GetListScore(args, 8, 9); | ||||||
|  | 
 | ||||||
|  |             // studios | ||||||
|  |             score += GetListScore(args, 10, 11, 3); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |             // TODO: People | ||||||
|  |     //        var item2PeopleNames = allPeople.Where(i => i.ItemId == item2.Id) | ||||||
|  |     //.Select(i => i.Name) | ||||||
|  |     //.Where(i => !string.IsNullOrWhiteSpace(i)) | ||||||
|  |     //.DistinctNames() | ||||||
|  |     //.ToDictionary(i => i, StringComparer.OrdinalIgnoreCase); | ||||||
|  | 
 | ||||||
|  |     //        points += item1People.Where(i => item2PeopleNames.ContainsKey(i.Name)).Sum(i => | ||||||
|  |     //        { | ||||||
|  |     //            if (string.Equals(i.Type, PersonType.Director, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Director, StringComparison.OrdinalIgnoreCase)) | ||||||
|  |     //            { | ||||||
|  |     //                return 5; | ||||||
|  |     //            } | ||||||
|  |     //            if (string.Equals(i.Type, PersonType.Actor, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Actor, StringComparison.OrdinalIgnoreCase)) | ||||||
|  |     //            { | ||||||
|  |     //                return 3; | ||||||
|  |     //            } | ||||||
|  |     //            if (string.Equals(i.Type, PersonType.Composer, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Composer, StringComparison.OrdinalIgnoreCase)) | ||||||
|  |     //            { | ||||||
|  |     //                return 3; | ||||||
|  |     //            } | ||||||
|  |     //            if (string.Equals(i.Type, PersonType.GuestStar, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.GuestStar, StringComparison.OrdinalIgnoreCase)) | ||||||
|  |     //            { | ||||||
|  |     //                return 3; | ||||||
|  |     //            } | ||||||
|  |     //            if (string.Equals(i.Type, PersonType.Writer, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Writer, StringComparison.OrdinalIgnoreCase)) | ||||||
|  |     //            { | ||||||
|  |     //                return 2; | ||||||
|  |     //            } | ||||||
|  | 
 | ||||||
|  |     //            return 1; | ||||||
|  |     //        }); | ||||||
|  | 
 | ||||||
|  |     //        return points; | ||||||
|  | 
 | ||||||
|  |             //Logger.Debug("Returning score {0}", score); | ||||||
|  |             return score; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private int GetListScore(object[] args, int index1, int index2, int value = 10) | ||||||
|  |         { | ||||||
|  |             var score = 0; | ||||||
|  | 
 | ||||||
|  |             var inputGenres = args[index1] as string; | ||||||
|  |             var rowGenres = args[index2] as string; | ||||||
|  |             var inputGenreList = string.IsNullOrWhiteSpace(inputGenres) ? new string[] { } : inputGenres.Split(new[] { '|' }, StringSplitOptions.RemoveEmptyEntries); | ||||||
|  |             var rowGenresList = string.IsNullOrWhiteSpace(rowGenres) ? new string[] { } : rowGenres.Split(new[] { '|' }, StringSplitOptions.RemoveEmptyEntries); | ||||||
|  | 
 | ||||||
|  |             foreach (var genre in inputGenreList) | ||||||
|  |             { | ||||||
|  |                 if (rowGenresList.Contains(genre, StringComparer.OrdinalIgnoreCase)) | ||||||
|  |                 { | ||||||
|  |                     score += value; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return score; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -15,6 +15,7 @@ using System.Globalization; | |||||||
| using System.IO; | using System.IO; | ||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Runtime.Serialization; | using System.Runtime.Serialization; | ||||||
|  | using System.Text; | ||||||
| using System.Threading; | using System.Threading; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using MediaBrowser.Common.Extensions; | using MediaBrowser.Common.Extensions; | ||||||
| @ -258,6 +259,8 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|             new MediaStreamColumns(_connection, Logger).AddColumns(); |             new MediaStreamColumns(_connection, Logger).AddColumns(); | ||||||
| 
 | 
 | ||||||
|             DataExtensions.Attach(_connection, Path.Combine(_config.ApplicationPaths.DataPath, "userdata_v2.db"), "UserDataDb"); |             DataExtensions.Attach(_connection, Path.Combine(_config.ApplicationPaths.DataPath, "userdata_v2.db"), "UserDataDb"); | ||||||
|  | 
 | ||||||
|  |             dbConnector.BindSimilarityScoreFunction(_connection); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private readonly string[] _retriveItemColumns = |         private readonly string[] _retriveItemColumns = | ||||||
| @ -1523,6 +1526,11 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|                 return false; |                 return false; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |             if (query.SimilarTo != null) | ||||||
|  |             { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             if (query.SortBy != null && query.SortBy.Length > 0) |             if (query.SortBy != null && query.SortBy.Length > 0) | ||||||
|             { |             { | ||||||
|                 if (query.SortBy.Contains(ItemSortBy.IsFavoriteOrLiked, StringComparer.OrdinalIgnoreCase)) |                 if (query.SortBy.Contains(ItemSortBy.IsFavoriteOrLiked, StringComparer.OrdinalIgnoreCase)) | ||||||
| @ -1575,7 +1583,7 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private string[] GetFinalColumnsToSelect(InternalItemsQuery query, string[] startColumns) |         private string[] GetFinalColumnsToSelect(InternalItemsQuery query, string[] startColumns, IDbCommand cmd) | ||||||
|         { |         { | ||||||
|             var list = startColumns.ToList(); |             var list = startColumns.ToList(); | ||||||
| 
 | 
 | ||||||
| @ -1590,6 +1598,45 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|                 list.Add("UserDataDb.UserData.rating"); |                 list.Add("UserDataDb.UserData.rating"); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |             if (query.SimilarTo != null) | ||||||
|  |             { | ||||||
|  |                 var item = query.SimilarTo; | ||||||
|  | 
 | ||||||
|  |                 var builder = new StringBuilder(); | ||||||
|  |                 builder.Append("GetSimilarityScore("); | ||||||
|  | 
 | ||||||
|  |                 builder.Append("@ItemOfficialRating,"); | ||||||
|  |                 builder.Append("OfficialRating,"); | ||||||
|  | 
 | ||||||
|  |                 builder.Append("@ItemProductionYear,"); | ||||||
|  |                 builder.Append("ProductionYear,"); | ||||||
|  | 
 | ||||||
|  |                 builder.Append("@ItemGenres,"); | ||||||
|  |                 builder.Append("Genres,"); | ||||||
|  | 
 | ||||||
|  |                 builder.Append("@ItemTags,"); | ||||||
|  |                 builder.Append("Tags,"); | ||||||
|  | 
 | ||||||
|  |                 builder.Append("@ItemKeywords,"); | ||||||
|  |                 builder.Append("(select group_concat((Select Value from ItemValues where ItemId=Guid and Type=5), '|')),"); | ||||||
|  | 
 | ||||||
|  |                 builder.Append("@ItemStudios,"); | ||||||
|  |                 builder.Append("Studios"); | ||||||
|  |                 builder.Append(") as SimilarityScore"); | ||||||
|  | 
 | ||||||
|  |                 list.Add(builder.ToString()); | ||||||
|  |                 cmd.Parameters.Add(cmd, "@ItemOfficialRating", DbType.String).Value = item.OfficialRating; | ||||||
|  |                 cmd.Parameters.Add(cmd, "@ItemProductionYear", DbType.Int32).Value = item.ProductionYear ?? -1; | ||||||
|  |                 cmd.Parameters.Add(cmd, "@ItemGenres", DbType.String).Value = string.Join("|", item.Genres.ToArray()); | ||||||
|  |                 cmd.Parameters.Add(cmd, "@ItemTags", DbType.String).Value = string.Join("|", item.Tags.ToArray()); | ||||||
|  |                 cmd.Parameters.Add(cmd, "@ItemKeywords", DbType.String).Value = string.Join("|", item.Keywords.ToArray()); | ||||||
|  |                 cmd.Parameters.Add(cmd, "@ItemStudios", DbType.String).Value = string.Join("|", item.Studios.ToArray()); | ||||||
|  | 
 | ||||||
|  |                 var excludeIds = query.ExcludeItemIds.ToList(); | ||||||
|  |                 excludeIds.Add(item.Id.ToString("N")); | ||||||
|  |                 query.ExcludeItemIds = excludeIds.ToArray(); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             return list.ToArray(); |             return list.ToArray(); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @ -1616,7 +1663,7 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
| 
 | 
 | ||||||
|             using (var cmd = _connection.CreateCommand()) |             using (var cmd = _connection.CreateCommand()) | ||||||
|             { |             { | ||||||
|                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, _retriveItemColumns)) + " from TypedBaseItems"; |                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, _retriveItemColumns, cmd)) + " from TypedBaseItems"; | ||||||
|                 cmd.CommandText += GetJoinUserDataText(query); |                 cmd.CommandText += GetJoinUserDataText(query); | ||||||
| 
 | 
 | ||||||
|                 if (EnableJoinUserData(query)) |                 if (EnableJoinUserData(query)) | ||||||
| @ -1706,7 +1753,7 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
| 
 | 
 | ||||||
|             using (var cmd = _connection.CreateCommand()) |             using (var cmd = _connection.CreateCommand()) | ||||||
|             { |             { | ||||||
|                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, _retriveItemColumns)) + " from TypedBaseItems"; |                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, _retriveItemColumns, cmd)) + " from TypedBaseItems"; | ||||||
|                 cmd.CommandText += GetJoinUserDataText(query); |                 cmd.CommandText += GetJoinUserDataText(query); | ||||||
| 
 | 
 | ||||||
|                 if (EnableJoinUserData(query)) |                 if (EnableJoinUserData(query)) | ||||||
| @ -1789,6 +1836,22 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
| 
 | 
 | ||||||
|         private string GetOrderByText(InternalItemsQuery query) |         private string GetOrderByText(InternalItemsQuery query) | ||||||
|         { |         { | ||||||
|  |             if (query.SimilarTo != null) | ||||||
|  |             { | ||||||
|  |                 if (query.SortBy == null || query.SortBy.Length == 0) | ||||||
|  |                 { | ||||||
|  |                     if (query.User != null) | ||||||
|  |                     { | ||||||
|  |                         query.SortBy = new[] { "SimilarityScore", "IsUnplayed", "Random" }; | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         query.SortBy = new[] { "SimilarityScore", "Random" }; | ||||||
|  |                     } | ||||||
|  |                     query.SortOrder = SortOrder.Descending; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             if (query.SortBy == null || query.SortBy.Length == 0) |             if (query.SortBy == null || query.SortBy.Length == 0) | ||||||
|             { |             { | ||||||
|                 return string.Empty; |                 return string.Empty; | ||||||
| @ -1862,6 +1925,14 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|             { |             { | ||||||
|                 return new Tuple<string, bool>("(select value from itemvalues where ItemId=Guid and Type=1 LIMIT 1)", false); |                 return new Tuple<string, bool>("(select value from itemvalues where ItemId=Guid and Type=1 LIMIT 1)", false); | ||||||
|             } |             } | ||||||
|  |             if (string.Equals(name, ItemSortBy.OfficialRating, StringComparison.OrdinalIgnoreCase)) | ||||||
|  |             { | ||||||
|  |                 return new Tuple<string, bool>("ParentalRatingValue", false); | ||||||
|  |             } | ||||||
|  |             if (string.Equals(name, ItemSortBy.Studio, StringComparison.OrdinalIgnoreCase)) | ||||||
|  |             { | ||||||
|  |                 return new Tuple<string, bool>("(select value from itemvalues where ItemId=Guid and Type=3 LIMIT 1)", false); | ||||||
|  |             } | ||||||
| 
 | 
 | ||||||
|             return new Tuple<string, bool>(name, false); |             return new Tuple<string, bool>(name, false); | ||||||
|         } |         } | ||||||
| @ -1879,7 +1950,7 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
| 
 | 
 | ||||||
|             using (var cmd = _connection.CreateCommand()) |             using (var cmd = _connection.CreateCommand()) | ||||||
|             { |             { | ||||||
|                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, new[] { "guid" })) + " from TypedBaseItems"; |                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, new[] { "guid" }, cmd)) + " from TypedBaseItems"; | ||||||
|                 cmd.CommandText += GetJoinUserDataText(query); |                 cmd.CommandText += GetJoinUserDataText(query); | ||||||
| 
 | 
 | ||||||
|                 if (EnableJoinUserData(query)) |                 if (EnableJoinUserData(query)) | ||||||
| @ -2022,7 +2093,7 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
| 
 | 
 | ||||||
|             using (var cmd = _connection.CreateCommand()) |             using (var cmd = _connection.CreateCommand()) | ||||||
|             { |             { | ||||||
|                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, new[] { "guid" })) + " from TypedBaseItems"; |                 cmd.CommandText = "select " + string.Join(",", GetFinalColumnsToSelect(query, new[] { "guid" }, cmd)) + " from TypedBaseItems"; | ||||||
| 
 | 
 | ||||||
|                 var whereClauses = GetWhereClauses(query, cmd); |                 var whereClauses = GetWhereClauses(query, cmd); | ||||||
|                 cmd.CommandText += GetJoinUserDataText(query); |                 cmd.CommandText += GetJoinUserDataText(query); | ||||||
| @ -2148,24 +2219,7 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     if (query.IsMovie.Value) |                     whereClauses.Add("(IsMovie is null OR IsMovie=@IsMovie)"); | ||||||
|                     { |  | ||||||
|                         var typeClauses = new List<string>(); |  | ||||||
|                         var typeIndex = 0; |  | ||||||
|                         foreach (var type in alternateTypes) |  | ||||||
|                         { |  | ||||||
|                             var paramName = "@AlternateType" + typeIndex.ToString(CultureInfo.InvariantCulture); |  | ||||||
|                             typeClauses.Add("Type=" + paramName); |  | ||||||
|                             cmd.Parameters.Add(cmd, paramName, DbType.String).Value = type; |  | ||||||
|                             typeIndex++; |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         whereClauses.Add("(IsMovie=@IsMovie OR " + string.Join(" OR ", typeClauses.ToArray()) + ")"); |  | ||||||
|                     } |  | ||||||
|                     else |  | ||||||
|                     { |  | ||||||
|                         whereClauses.Add("(IsMovie is null OR IsMovie=@IsMovie)"); |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|                 cmd.Parameters.Add(cmd, "@IsMovie", DbType.Boolean).Value = query.IsMovie; |                 cmd.Parameters.Add(cmd, "@IsMovie", DbType.Boolean).Value = query.IsMovie; | ||||||
|             } |             } | ||||||
| @ -2539,6 +2593,20 @@ namespace MediaBrowser.Server.Implementations.Persistence | |||||||
|                 whereClauses.Add(clause); |                 whereClauses.Add(clause); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |             if (query.OfficialRatings.Length > 0) | ||||||
|  |             { | ||||||
|  |                 var clauses = new List<string>(); | ||||||
|  |                 var index = 0; | ||||||
|  |                 foreach (var item in query.OfficialRatings) | ||||||
|  |                 { | ||||||
|  |                     clauses.Add("OfficialRating=@OfficialRating" + index); | ||||||
|  |                     cmd.Parameters.Add(cmd, "@OfficialRating" + index, DbType.String).Value = item; | ||||||
|  |                     index++; | ||||||
|  |                 } | ||||||
|  |                 var clause = "(" + string.Join(" OR ", clauses.ToArray()) + ")"; | ||||||
|  |                 whereClauses.Add(clause); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             if (query.MinParentalRating.HasValue) |             if (query.MinParentalRating.HasValue) | ||||||
|             { |             { | ||||||
|                 whereClauses.Add("InheritedParentalRatingValue<=@MinParentalRating"); |                 whereClauses.Add("InheritedParentalRatingValue<=@MinParentalRating"); | ||||||
|  | |||||||
| @ -1,8 +1,8 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <packages> | <packages> | ||||||
|   <package id="CommonIO" version="1.0.0.9" targetFramework="net45" /> |   <package id="CommonIO" version="1.0.0.9" targetFramework="net45" /> | ||||||
|   <package id="Emby.XmlTv" version="1.0.0.48" targetFramework="net45" /> |   <package id="Emby.XmlTv" version="1.0.0.50" targetFramework="net45" /> | ||||||
|   <package id="ini-parser" version="2.2.4" targetFramework="net45" /> |   <package id="ini-parser" version="2.3.0" targetFramework="net45" /> | ||||||
|   <package id="Interfaces.IO" version="1.0.0.5" targetFramework="net45" /> |   <package id="Interfaces.IO" version="1.0.0.5" targetFramework="net45" /> | ||||||
|   <package id="MediaBrowser.Naming" version="1.0.0.50" targetFramework="net45" /> |   <package id="MediaBrowser.Naming" version="1.0.0.50" targetFramework="net45" /> | ||||||
|   <package id="Mono.Nat" version="1.2.24.0" targetFramework="net45" /> |   <package id="Mono.Nat" version="1.2.24.0" targetFramework="net45" /> | ||||||
|  | |||||||
| @ -82,11 +82,14 @@ | |||||||
|     </Reference> |     </Reference> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|  |     <Compile Include="..\MediaBrowser.Server.Implementations\Persistence\SqliteExtensions.cs"> | ||||||
|  |       <Link>Native\SqliteExtensions.cs</Link> | ||||||
|  |     </Compile> | ||||||
|     <Compile Include="..\SharedVersion.cs"> |     <Compile Include="..\SharedVersion.cs"> | ||||||
|       <Link>Properties\SharedVersion.cs</Link> |       <Link>Properties\SharedVersion.cs</Link> | ||||||
|     </Compile> |     </Compile> | ||||||
|     <Compile Include="Native\BaseMonoApp.cs" /> |     <Compile Include="Native\BaseMonoApp.cs" /> | ||||||
|     <Compile Include="Native\SqliteExtensions.cs" /> |     <Compile Include="Native\DbConnector.cs" /> | ||||||
|     <Compile Include="Networking\CertificateGenerator.cs" /> |     <Compile Include="Networking\CertificateGenerator.cs" /> | ||||||
|     <Compile Include="Program.cs" /> |     <Compile Include="Program.cs" /> | ||||||
|     <Compile Include="Properties\AssemblyInfo.cs" /> |     <Compile Include="Properties\AssemblyInfo.cs" /> | ||||||
|  | |||||||
							
								
								
									
										29
									
								
								MediaBrowser.Server.Mono/Native/DbConnector.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								MediaBrowser.Server.Mono/Native/DbConnector.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,29 @@ | |||||||
|  | using System; | ||||||
|  | using System.Data; | ||||||
|  | using System.Data.SQLite; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using MediaBrowser.Server.Implementations.Persistence; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Mono.Native | ||||||
|  | { | ||||||
|  |     public class DbConnector : IDbConnector | ||||||
|  |     { | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public DbConnector(ILogger logger) | ||||||
|  |         { | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void BindSimilarityScoreFunction(IDbConnection connection) | ||||||
|  |         { | ||||||
|  |             SqliteExtensions.BindGetSimilarityScore(connection, _logger); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public Task<IDbConnection> Connect(string dbPath) | ||||||
|  |         { | ||||||
|  |             return SqliteExtensions.ConnectToDb(dbPath, _logger); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -1,62 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Data; |  | ||||||
| using System.Data.SQLite; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using MediaBrowser.Model.Logging; |  | ||||||
| using MediaBrowser.Server.Implementations.Persistence; |  | ||||||
| 
 |  | ||||||
| namespace MediaBrowser.Server.Mono.Native |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Class SQLiteExtensions |  | ||||||
|     /// </summary> |  | ||||||
|     static class SqliteExtensions |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Connects to db. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="dbPath">The db path.</param> |  | ||||||
|         /// <param name="logger">The logger.</param> |  | ||||||
|         /// <returns>Task{IDbConnection}.</returns> |  | ||||||
|         /// <exception cref="System.ArgumentNullException">dbPath</exception> |  | ||||||
|         public static async Task<IDbConnection> ConnectToDb(string dbPath, ILogger logger) |  | ||||||
|         { |  | ||||||
|             if (string.IsNullOrEmpty(dbPath)) |  | ||||||
|             { |  | ||||||
|                 throw new ArgumentNullException("dbPath"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             logger.Info("Sqlite {0} opening {1}", SQLiteConnection.SQLiteVersion, dbPath); |  | ||||||
| 
 |  | ||||||
|             var connectionstr = new SQLiteConnectionStringBuilder |  | ||||||
|             { |  | ||||||
|                 PageSize = 4096, |  | ||||||
|                 CacheSize = 2000, |  | ||||||
|                 SyncMode = SynchronizationModes.Normal, |  | ||||||
|                 DataSource = dbPath, |  | ||||||
|                 JournalMode = SQLiteJournalModeEnum.Wal |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             var connection = new SQLiteConnection(connectionstr.ConnectionString); |  | ||||||
| 
 |  | ||||||
|             await connection.OpenAsync().ConfigureAwait(false); |  | ||||||
| 
 |  | ||||||
|             return connection; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public class DbConnector : IDbConnector |  | ||||||
|     { |  | ||||||
|         private readonly ILogger _logger; |  | ||||||
| 
 |  | ||||||
|         public DbConnector(ILogger logger) |  | ||||||
|         { |  | ||||||
|             _logger = logger; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public Task<IDbConnection> Connect(string dbPath) |  | ||||||
|         { |  | ||||||
|             return SqliteExtensions.ConnectToDb(dbPath, _logger); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -97,6 +97,9 @@ | |||||||
|     <Reference Include="System.Xml" /> |     <Reference Include="System.Xml" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|  |     <Compile Include="..\MediaBrowser.Server.Implementations\Persistence\SqliteExtensions.cs"> | ||||||
|  |       <Link>Native\SqliteExtensions.cs</Link> | ||||||
|  |     </Compile> | ||||||
|     <Compile Include="..\SharedVersion.cs"> |     <Compile Include="..\SharedVersion.cs"> | ||||||
|       <Link>Properties\SharedVersion.cs</Link> |       <Link>Properties\SharedVersion.cs</Link> | ||||||
|     </Compile> |     </Compile> | ||||||
| @ -114,7 +117,7 @@ | |||||||
|     </Compile> |     </Compile> | ||||||
|     <Compile Include="MainStartup.cs" /> |     <Compile Include="MainStartup.cs" /> | ||||||
|     <Compile Include="Native\LnkShortcutHandler.cs" /> |     <Compile Include="Native\LnkShortcutHandler.cs" /> | ||||||
|     <Compile Include="Native\SqliteExtensions.cs" /> |     <Compile Include="Native\DbConnector.cs" /> | ||||||
|     <Compile Include="Native\Standby.cs" /> |     <Compile Include="Native\Standby.cs" /> | ||||||
|     <Compile Include="Native\ServerAuthorization.cs" /> |     <Compile Include="Native\ServerAuthorization.cs" /> | ||||||
|     <Compile Include="Native\WindowsApp.cs" /> |     <Compile Include="Native\WindowsApp.cs" /> | ||||||
|  | |||||||
							
								
								
									
										38
									
								
								MediaBrowser.ServerApplication/Native/DbConnector.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								MediaBrowser.ServerApplication/Native/DbConnector.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,38 @@ | |||||||
|  | using System; | ||||||
|  | using System.Data; | ||||||
|  | using System.Data.SQLite; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using MediaBrowser.Server.Implementations.Persistence; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.ServerApplication.Native | ||||||
|  | { | ||||||
|  |     public class DbConnector : IDbConnector | ||||||
|  |     { | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public DbConnector(ILogger logger) | ||||||
|  |         { | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void BindSimilarityScoreFunction(IDbConnection connection) | ||||||
|  |         { | ||||||
|  |             SqliteExtensions.BindGetSimilarityScore(connection, _logger); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public async Task<IDbConnection> Connect(string dbPath) | ||||||
|  |         { | ||||||
|  |             try | ||||||
|  |             { | ||||||
|  |                 return await SqliteExtensions.ConnectToDb(dbPath, _logger).ConfigureAwait(false); | ||||||
|  |             } | ||||||
|  |             catch (Exception ex) | ||||||
|  |             { | ||||||
|  |                 _logger.ErrorException("Error opening database {0}", ex, dbPath); | ||||||
|  | 
 | ||||||
|  |                 throw; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -1,71 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Data; |  | ||||||
| using System.Data.SQLite; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using MediaBrowser.Model.Logging; |  | ||||||
| using MediaBrowser.Server.Implementations.Persistence; |  | ||||||
| 
 |  | ||||||
| namespace MediaBrowser.ServerApplication.Native |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Class SQLiteExtensions |  | ||||||
|     /// </summary> |  | ||||||
|     static class SqliteExtensions |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Connects to db. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="dbPath">The db path.</param> |  | ||||||
|         /// <param name="logger">The logger.</param> |  | ||||||
|         /// <returns>Task{IDbConnection}.</returns> |  | ||||||
|         /// <exception cref="System.ArgumentNullException">dbPath</exception> |  | ||||||
|         public static async Task<IDbConnection> ConnectToDb(string dbPath, ILogger logger) |  | ||||||
|         { |  | ||||||
|             if (string.IsNullOrEmpty(dbPath)) |  | ||||||
|             { |  | ||||||
|                 throw new ArgumentNullException("dbPath"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             logger.Info("Sqlite {0} opening {1}", SQLiteConnection.SQLiteVersion, dbPath); |  | ||||||
| 
 |  | ||||||
|             var connectionstr = new SQLiteConnectionStringBuilder |  | ||||||
|             { |  | ||||||
|                 PageSize = 4096, |  | ||||||
|                 CacheSize = 2000, |  | ||||||
|                 SyncMode = SynchronizationModes.Normal, |  | ||||||
|                 DataSource = dbPath, |  | ||||||
|                 JournalMode = SQLiteJournalModeEnum.Wal |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             var connection = new SQLiteConnection(connectionstr.ConnectionString); |  | ||||||
| 
 |  | ||||||
|             await connection.OpenAsync().ConfigureAwait(false); |  | ||||||
| 
 |  | ||||||
|             return connection; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public class DbConnector : IDbConnector |  | ||||||
|     { |  | ||||||
|         private readonly ILogger _logger; |  | ||||||
| 
 |  | ||||||
|         public DbConnector(ILogger logger) |  | ||||||
|         { |  | ||||||
|             _logger = logger; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public async Task<IDbConnection> Connect(string dbPath) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return await SqliteExtensions.ConnectToDb(dbPath, _logger).ConfigureAwait(false); |  | ||||||
|             } |  | ||||||
|             catch (Exception ex) |  | ||||||
|             { |  | ||||||
|                 _logger.ErrorException("Error opening database {0}", ex, dbPath); |  | ||||||
| 
 |  | ||||||
|                 throw; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user