mirror of
				https://github.com/zoriya/Kyoo.git
				synced 2025-10-25 15:52:36 -04:00 
			
		
		
		
	Adding slugs setters
This commit is contained in:
		
							parent
							
								
									af39793b7c
								
							
						
					
					
						commit
						32f12ae833
					
				| @ -1,5 +1,6 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using System.Text.RegularExpressions; | ||||||
| using JetBrains.Annotations; | using JetBrains.Annotations; | ||||||
| using Kyoo.Controllers; | using Kyoo.Controllers; | ||||||
| using Kyoo.Models.Attributes; | using Kyoo.Models.Attributes; | ||||||
| @ -16,7 +17,34 @@ namespace Kyoo.Models | |||||||
| 		public int ID { get; set; } | 		public int ID { get; set; } | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| 		public string Slug => GetSlug(ShowSlug, SeasonNumber, EpisodeNumber, AbsoluteNumber); | 		public string Slug | ||||||
|  | 		{ | ||||||
|  | 			get => GetSlug(ShowSlug, SeasonNumber, EpisodeNumber, AbsoluteNumber); | ||||||
|  | 			set | ||||||
|  | 			{ | ||||||
|  | 				Match match = Regex.Match(value, @"(?<show>.*)-s(?<season>\d*)e(?<episode>\d*)"); | ||||||
|  | 
 | ||||||
|  | 				if (match.Success) | ||||||
|  | 				{ | ||||||
|  | 					ShowSlug = match.Groups["show"].Value; | ||||||
|  | 					SeasonNumber = int.Parse(match.Groups["season"].Value); | ||||||
|  | 					EpisodeNumber = int.Parse(match.Groups["episode"].Value); | ||||||
|  | 				} | ||||||
|  | 				else | ||||||
|  | 				{ | ||||||
|  | 					match = Regex.Match(value, @"(?<show>.*)-(?<absolute>\d*)"); | ||||||
|  | 					if (match.Success) | ||||||
|  | 					{ | ||||||
|  | 						ShowSlug = match.Groups["Show"].Value; | ||||||
|  | 						AbsoluteNumber = int.Parse(match.Groups["absolute"].Value); | ||||||
|  | 					} | ||||||
|  | 					else | ||||||
|  | 						ShowSlug = value; | ||||||
|  | 					SeasonNumber = -1; | ||||||
|  | 					EpisodeNumber = -1; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 		 | 		 | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// The slug of the Show that contain this episode. If this is not set, this episode is ill-formed. | 		/// The slug of the Show that contain this episode. If this is not set, this episode is ill-formed. | ||||||
|  | |||||||
| @ -1,5 +1,6 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using System.Text.RegularExpressions; | ||||||
| using Kyoo.Controllers; | using Kyoo.Controllers; | ||||||
| using Kyoo.Models.Attributes; | using Kyoo.Models.Attributes; | ||||||
| 
 | 
 | ||||||
| @ -14,7 +15,19 @@ namespace Kyoo.Models | |||||||
| 		public int ID  { get; set; } | 		public int ID  { get; set; } | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| 		public string Slug => $"{ShowSlug}-s{SeasonNumber}"; | 		public string Slug | ||||||
|  | 		{ | ||||||
|  | 			get => $"{ShowSlug}-s{SeasonNumber}"; | ||||||
|  | 			set | ||||||
|  | 			{ | ||||||
|  | 				Match match = Regex.Match(value, @"(?<show>.*)-s(?<season>\d*)"); | ||||||
|  | 			 | ||||||
|  | 				if (!match.Success) | ||||||
|  | 					throw new ArgumentException("Invalid season slug. Format: {showSlug}-s{seasonNumber}"); | ||||||
|  | 				ShowSlug = match.Groups["show"].Value; | ||||||
|  | 				SeasonNumber = int.Parse(match.Groups["season"].Value); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 		 | 		 | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// The slug of the Show that contain this episode. If this is not set, this season is ill-formed. | 		/// The slug of the Show that contain this episode. If this is not set, this season is ill-formed. | ||||||
|  | |||||||
| @ -1,5 +1,7 @@ | |||||||
| using System.Globalization; | using System; | ||||||
|  | using System.Globalization; | ||||||
| using System.Linq; | using System.Linq; | ||||||
|  | using System.Text.RegularExpressions; | ||||||
| using Kyoo.Models.Attributes; | using Kyoo.Models.Attributes; | ||||||
| 
 | 
 | ||||||
| namespace Kyoo.Models | namespace Kyoo.Models | ||||||
| @ -44,10 +46,36 @@ namespace Kyoo.Models | |||||||
| 					"subrip" => ".srt", | 					"subrip" => ".srt", | ||||||
| 					{} x => $".{x}" | 					{} x => $".{x}" | ||||||
| 				}; | 				}; | ||||||
| 				return $"{Episode.Slug}.{type}{Language}{index}{(IsForced ? "-forced" : "")}{codec}"; | 				return $"{EpisodeSlug}.{type}{Language}{index}{(IsForced ? "-forced" : "")}{codec}"; | ||||||
|  | 			} | ||||||
|  | 			set | ||||||
|  | 			{ | ||||||
|  | 				Match match = Regex.Match(value, @"(?<show>.*)-s(?<season>\d+)e(?<episode>\d+)"  | ||||||
|  | 				                                 + @"(\.(?<type>\w*))?\.(?<language>.{0,3})(?<forced>-forced)?(\..\w)?"); | ||||||
|  | 
 | ||||||
|  | 				if (!match.Success) | ||||||
|  | 				{ | ||||||
|  | 					match = Regex.Match(value, @"(?<show>.*)\.(?<language>.{0,3})(?<forced>-forced)?(\..\w)?"); | ||||||
|  | 					if (!match.Success) | ||||||
|  | 						throw new ArgumentException("Invalid track slug. " + | ||||||
|  | 						                            "Format: {episodeSlug}.{language}[-forced][.{extension}]"); | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				EpisodeSlug = Episode.GetSlug(match.Groups["show"].Value,  | ||||||
|  | 					match.Groups["season"].Success ? int.Parse(match.Groups["season"].Value) : -1, | ||||||
|  | 					match.Groups["episode"].Success ? int.Parse(match.Groups["episode"].Value) : -1); | ||||||
|  | 				Language = match.Groups["language"].Value; | ||||||
|  | 				IsForced = match.Groups["forced"].Success; | ||||||
|  | 				if (match.Groups["type"].Success) | ||||||
|  | 					Type = Enum.Parse<StreamType>(match.Groups["type"].Value, true); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
|  | 		/// <summary> | ||||||
|  |         /// The slug of the episode that contain this track. If this is not set, this track is ill-formed. | ||||||
|  |         /// </summary> | ||||||
|  |         [SerializeIgnore] public string EpisodeSlug { private get; set; } | ||||||
|  | 		 | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// The title of the stream. | 		/// The title of the stream. | ||||||
| 		/// </summary> | 		/// </summary> | ||||||
|  | |||||||
| @ -12,8 +12,8 @@ | |||||||
| 	</PropertyGroup> | 	</PropertyGroup> | ||||||
| 
 | 
 | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
| 	  <PackageReference Include="Microsoft.AspNetCore.Mvc.NewtonsoftJson" Version="5.0.6" /> | 	  <PackageReference Include="Microsoft.AspNetCore.Mvc.NewtonsoftJson" Version="5.0.7" /> | ||||||
| 	  <PackageReference Include="Microsoft.EntityFrameworkCore" Version="5.0.6" /> | 	  <PackageReference Include="Microsoft.EntityFrameworkCore" Version="5.0.7" /> | ||||||
| 	  <PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" Version="5.0.0" /> | 	  <PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" Version="5.0.0" /> | ||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -18,7 +18,7 @@ | |||||||
| <!--	</PropertyGroup>--> | <!--	</PropertyGroup>--> | ||||||
| 
 | 
 | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
| 		<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="5.0.6"> | 		<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="5.0.7"> | ||||||
| 			<PrivateAssets>all</PrivateAssets> | 			<PrivateAssets>all</PrivateAssets> | ||||||
| 			<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets> | 			<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets> | ||||||
| 		</PackageReference> | 		</PackageReference> | ||||||
|  | |||||||
| @ -19,11 +19,11 @@ | |||||||
| <!--	</PropertyGroup>--> | <!--	</PropertyGroup>--> | ||||||
| 
 | 
 | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
| 		<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="5.0.6"> | 		<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="5.0.7"> | ||||||
| 			<PrivateAssets>all</PrivateAssets> | 			<PrivateAssets>all</PrivateAssets> | ||||||
| 			<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets> | 			<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets> | ||||||
| 		</PackageReference> | 		</PackageReference> | ||||||
| 		<PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite" Version="5.0.6" /> | 		<PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite" Version="5.0.7" /> | ||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
| 
 | 
 | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
|  | |||||||
| @ -82,10 +82,6 @@ namespace Kyoo.SqLite | |||||||
| 		/// <param name="modelBuilder">The database's model builder.</param> | 		/// <param name="modelBuilder">The database's model builder.</param> | ||||||
| 		protected override void OnModelCreating(ModelBuilder modelBuilder) | 		protected override void OnModelCreating(ModelBuilder modelBuilder) | ||||||
| 		{ | 		{ | ||||||
| 			// modelBuilder.HasPostgresEnum<Status>(); |  | ||||||
| 			// modelBuilder.HasPostgresEnum<ItemType>(); |  | ||||||
| 			// modelBuilder.HasPostgresEnum<StreamType>(); |  | ||||||
| 
 |  | ||||||
| 			ValueConverter<string[], string> arrayConvertor = new( | 			ValueConverter<string[], string> arrayConvertor = new( | ||||||
| 				x => string.Join(";", x), | 				x => string.Join(";", x), | ||||||
| 				x => x.Split(';', StringSplitOptions.None)); | 				x => x.Split(';', StringSplitOptions.None)); | ||||||
| @ -112,7 +108,6 @@ namespace Kyoo.SqLite | |||||||
| 			modelBuilder.Entity<User>() | 			modelBuilder.Entity<User>() | ||||||
| 				.Property(x => x.ExtraData) | 				.Property(x => x.ExtraData) | ||||||
| 				.HasConversion(jsonConvertor); | 				.HasConversion(jsonConvertor); | ||||||
| 			 |  | ||||||
| 			base.OnModelCreating(modelBuilder); | 			base.OnModelCreating(modelBuilder); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -127,7 +122,7 @@ namespace Kyoo.SqLite | |||||||
| 		public override Expression<Func<T, bool>> Like<T>(Expression<Func<T, string>> query, string format) | 		public override Expression<Func<T, bool>> Like<T>(Expression<Func<T, string>> query, string format) | ||||||
| 		{ | 		{ | ||||||
| 			MethodInfo iLike = MethodOfUtils.MethodOf<string, string, bool>(EF.Functions.Like); | 			MethodInfo iLike = MethodOfUtils.MethodOf<string, string, bool>(EF.Functions.Like); | ||||||
| 			MethodCallExpression call = Expression.Call(iLike, query.Body, Expression.Constant(format)); | 			MethodCallExpression call = Expression.Call(iLike, Expression.Constant(EF.Functions), query.Body, Expression.Constant(format)); | ||||||
| 
 | 
 | ||||||
| 			return Expression.Lambda<Func<T, bool>>(call, query.Parameters); | 			return Expression.Lambda<Func<T, bool>>(call, query.Parameters); | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -25,13 +25,13 @@ namespace Kyoo.Tests | |||||||
| 			PeopleRepository people = new(_database, provider,  | 			PeopleRepository people = new(_database, provider,  | ||||||
| 				new Lazy<IShowRepository>(() => LibraryManager.ShowRepository)); | 				new Lazy<IShowRepository>(() => LibraryManager.ShowRepository)); | ||||||
| 			ShowRepository show = new(_database, studio, people, genre, provider); | 			ShowRepository show = new(_database, studio, people, genre, provider); | ||||||
| 			SeasonRepository season = new(_database, provider, show); | 			SeasonRepository season = new(_database, provider); | ||||||
| 			LibraryItemRepository libraryItem = new(_database,  | 			LibraryItemRepository libraryItem = new(_database,  | ||||||
| 				new Lazy<ILibraryRepository>(() => LibraryManager.LibraryRepository), | 				new Lazy<ILibraryRepository>(() => LibraryManager.LibraryRepository), | ||||||
| 				new Lazy<IShowRepository>(() => LibraryManager.ShowRepository), | 				new Lazy<IShowRepository>(() => LibraryManager.ShowRepository), | ||||||
| 				new Lazy<ICollectionRepository>(() => LibraryManager.CollectionRepository)); | 				new Lazy<ICollectionRepository>(() => LibraryManager.CollectionRepository)); | ||||||
| 			TrackRepository track = new(_database); | 			TrackRepository track = new(_database); | ||||||
| 			EpisodeRepository episode = new(_database, provider, show, track); | 			EpisodeRepository episode = new(_database, provider, track); | ||||||
| 
 | 
 | ||||||
| 			LibraryManager = new LibraryManager(new IBaseRepository[] { | 			LibraryManager = new LibraryManager(new IBaseRepository[] { | ||||||
| 				provider, | 				provider, | ||||||
|  | |||||||
							
								
								
									
										15
									
								
								Kyoo.Tests/Library/SpecificTests/SeasonTests.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								Kyoo.Tests/Library/SpecificTests/SeasonTests.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | |||||||
|  | using Kyoo.Controllers; | ||||||
|  | using Kyoo.Models; | ||||||
|  | 
 | ||||||
|  | namespace Kyoo.Tests.SpecificTests | ||||||
|  | { | ||||||
|  | 	public class SeasonTests : RepositoryTests<Season> | ||||||
|  | 	{ | ||||||
|  | 		private readonly ISeasonRepository _repository; | ||||||
|  | 
 | ||||||
|  | 		public SeasonTests() | ||||||
|  | 		{ | ||||||
|  | 			_repository = Repositories.LibraryManager.SeasonRepository; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -1,4 +1,5 @@ | |||||||
| using System; | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Kyoo.Controllers; | using Kyoo.Controllers; | ||||||
| @ -233,5 +234,23 @@ namespace Kyoo.Tests.SpecificTests | |||||||
| 			Show reference = TestSample.Get<Show>(); | 			Show reference = TestSample.Get<Show>(); | ||||||
| 			Assert.Equal(reference.Slug, await _repository.GetSlug(reference.ID)); | 			Assert.Equal(reference.Slug, await _repository.GetSlug(reference.ID)); | ||||||
| 		} | 		} | ||||||
|  | 		 | ||||||
|  | 		[Theory] | ||||||
|  | 		[InlineData("test")] | ||||||
|  | 		[InlineData("super")] | ||||||
|  | 		[InlineData("title")] | ||||||
|  | 		[InlineData("TiTlE")] | ||||||
|  | 		[InlineData("SuPeR")] | ||||||
|  | 		public async Task SearchTest(string query) | ||||||
|  | 		{ | ||||||
|  | 			Show value = new() | ||||||
|  | 			{ | ||||||
|  | 				Slug = "super-test", | ||||||
|  | 				Title = "This is a test title²" | ||||||
|  | 			}; | ||||||
|  | 			await _repository.Create(value); | ||||||
|  | 			ICollection<Show> ret = await _repository.Search(query); | ||||||
|  | 			KAssert.DeepEqual(value, ret.First()); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -45,6 +45,7 @@ namespace Kyoo.Tests | |||||||
| 					SeasonNumber = 1, | 					SeasonNumber = 1, | ||||||
| 					Title = "Season 1", | 					Title = "Season 1", | ||||||
| 					Overview = "The first season", | 					Overview = "The first season", | ||||||
|  | 					Show = Get<Show>(), | ||||||
| 					StartDate = new DateTime(2020, 06, 05), | 					StartDate = new DateTime(2020, 06, 05), | ||||||
| 					EndDate =  new DateTime(2020, 07, 05), | 					EndDate =  new DateTime(2020, 07, 05), | ||||||
| 					Poster = "poster" | 					Poster = "poster" | ||||||
|  | |||||||
| @ -2,7 +2,6 @@ using System; | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Linq.Expressions; | using System.Linq.Expressions; | ||||||
| using System.Text.RegularExpressions; |  | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Kyoo.Models; | using Kyoo.Models; | ||||||
| using Kyoo.Models.Exceptions; | using Kyoo.Models.Exceptions; | ||||||
| @ -16,7 +15,7 @@ namespace Kyoo.Controllers | |||||||
| 	public class EpisodeRepository : LocalRepository<Episode>, IEpisodeRepository | 	public class EpisodeRepository : LocalRepository<Episode>, IEpisodeRepository | ||||||
| 	{ | 	{ | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// The databse handle | 		/// The database handle | ||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		private readonly DatabaseContext _database; | 		private readonly DatabaseContext _database; | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| @ -24,10 +23,6 @@ namespace Kyoo.Controllers | |||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		private readonly IProviderRepository _providers; | 		private readonly IProviderRepository _providers; | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// A show repository to get show's slug from their ID and keep the slug in each episode. |  | ||||||
| 		/// </summary> |  | ||||||
| 		private readonly IShowRepository _shows; |  | ||||||
| 		/// <summary> |  | ||||||
| 		/// A track repository to handle creation and deletion of tracks related to the current episode. | 		/// A track repository to handle creation and deletion of tracks related to the current episode. | ||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		private readonly ITrackRepository _tracks; | 		private readonly ITrackRepository _tracks; | ||||||
| @ -41,66 +36,31 @@ namespace Kyoo.Controllers | |||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		/// <param name="database">The database handle to use.</param> | 		/// <param name="database">The database handle to use.</param> | ||||||
| 		/// <param name="providers">A provider repository</param> | 		/// <param name="providers">A provider repository</param> | ||||||
| 		/// <param name="shows">A show repository</param> |  | ||||||
| 		/// <param name="tracks">A track repository</param> | 		/// <param name="tracks">A track repository</param> | ||||||
| 		public EpisodeRepository(DatabaseContext database, | 		public EpisodeRepository(DatabaseContext database, | ||||||
| 			IProviderRepository providers, | 			IProviderRepository providers, | ||||||
| 			IShowRepository shows, |  | ||||||
| 			ITrackRepository tracks)  | 			ITrackRepository tracks)  | ||||||
| 			: base(database) | 			: base(database) | ||||||
| 		{ | 		{ | ||||||
| 			_database = database; | 			_database = database; | ||||||
| 			_providers = providers; | 			_providers = providers; | ||||||
| 			_shows = shows; |  | ||||||
| 			_tracks = tracks; | 			_tracks = tracks; | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| 
 |  | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| 		public override async Task<Episode> GetOrDefault(int id) | 		public Task<Episode> GetOrDefault(int showID, int seasonNumber, int episodeNumber) | ||||||
| 		{ | 		{ | ||||||
| 			Episode ret = await base.GetOrDefault(id); | 			return _database.Episodes.FirstOrDefaultAsync(x => x.ShowID == showID  | ||||||
| 			if (ret != null) | 			                                                   && x.SeasonNumber == seasonNumber  | ||||||
| 				ret.ShowSlug = await _shows.GetSlug(ret.ShowID); | 			                                                   && x.EpisodeNumber == episodeNumber); | ||||||
| 			return ret; |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| 		public override async Task<Episode> GetOrDefault(string slug) | 		public Task<Episode> GetOrDefault(string showSlug, int seasonNumber, int episodeNumber) | ||||||
| 		{ | 		{ | ||||||
| 			Match match = Regex.Match(slug, @"(?<show>.*)-s(?<season>\d*)e(?<episode>\d*)"); | 			return _database.Episodes.FirstOrDefaultAsync(x => x.Show.Slug == showSlug  | ||||||
| 
 | 			                                                   && x.SeasonNumber == seasonNumber  | ||||||
| 			if (match.Success) | 			                                                   && x.EpisodeNumber == episodeNumber); | ||||||
| 			{ |  | ||||||
| 				return await GetOrDefault(match.Groups["show"].Value, |  | ||||||
| 					int.Parse(match.Groups["season"].Value), |  | ||||||
| 					int.Parse(match.Groups["episode"].Value)); |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			Episode episode = await _database.Episodes.FirstOrDefaultAsync(x => x.Show.Slug == slug); |  | ||||||
| 			if (episode != null) |  | ||||||
| 				episode.ShowSlug = slug; |  | ||||||
| 			return episode; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		/// <inheritdoc /> |  | ||||||
| 		public override async Task<Episode> GetOrDefault(Expression<Func<Episode, bool>> where) |  | ||||||
| 		{ |  | ||||||
| 			Episode ret = await base.GetOrDefault(where); |  | ||||||
| 			if (ret != null) |  | ||||||
| 				ret.ShowSlug = await _shows.GetSlug(ret.ShowID); |  | ||||||
| 			return ret; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		/// <inheritdoc /> |  | ||||||
| 		public async Task<Episode> GetOrDefault(string showSlug, int seasonNumber, int episodeNumber) |  | ||||||
| 		{ |  | ||||||
| 			Episode ret = await _database.Episodes.FirstOrDefaultAsync(x => x.Show.Slug == showSlug  |  | ||||||
| 			                                                                && x.SeasonNumber == seasonNumber  |  | ||||||
| 			                                                                && x.EpisodeNumber == episodeNumber); |  | ||||||
| 			if (ret != null) |  | ||||||
| 				ret.ShowSlug = showSlug; |  | ||||||
| 			return ret; |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| @ -122,59 +82,28 @@ namespace Kyoo.Controllers | |||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| 		public async Task<Episode> GetOrDefault(int showID, int seasonNumber, int episodeNumber) | 		public Task<Episode> GetAbsolute(int showID, int absoluteNumber) | ||||||
| 		{ | 		{ | ||||||
| 			Episode ret = await _database.Episodes.FirstOrDefaultAsync(x => x.ShowID == showID  | 			return _database.Episodes.FirstOrDefaultAsync(x => x.ShowID == showID  | ||||||
| 			                                                                && x.SeasonNumber == seasonNumber  | 			                                                   && x.AbsoluteNumber == absoluteNumber); | ||||||
| 			                                                                && x.EpisodeNumber == episodeNumber); |  | ||||||
| 			if (ret != null) |  | ||||||
| 				ret.ShowSlug = await _shows.GetSlug(showID); |  | ||||||
| 			return ret; |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| 		public async Task<Episode> GetAbsolute(int showID, int absoluteNumber) | 		public Task<Episode> GetAbsolute(string showSlug, int absoluteNumber) | ||||||
| 		{ | 		{ | ||||||
| 			Episode ret = await _database.Episodes.FirstOrDefaultAsync(x => x.ShowID == showID  | 			return _database.Episodes.FirstOrDefaultAsync(x => x.Show.Slug == showSlug  | ||||||
| 			                                                                && x.AbsoluteNumber == absoluteNumber); | 			                                                   && x.AbsoluteNumber == absoluteNumber); | ||||||
| 			if (ret != null) |  | ||||||
| 				ret.ShowSlug = await _shows.GetSlug(showID); |  | ||||||
| 			return ret; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		/// <inheritdoc /> |  | ||||||
| 		public async Task<Episode> GetAbsolute(string showSlug, int absoluteNumber) |  | ||||||
| 		{ |  | ||||||
| 			Episode ret = await _database.Episodes.FirstOrDefaultAsync(x => x.Show.Slug == showSlug  |  | ||||||
| 			                                                                && x.AbsoluteNumber == absoluteNumber); |  | ||||||
| 			if (ret != null) |  | ||||||
| 				ret.ShowSlug = showSlug; |  | ||||||
| 			return ret; |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
| 		public override async Task<ICollection<Episode>> Search(string query) | 		public override async Task<ICollection<Episode>> Search(string query) | ||||||
| 		{ | 		{ | ||||||
| 			List<Episode> episodes = await _database.Episodes | 			return await _database.Episodes | ||||||
| 				.Where(x => x.EpisodeNumber != -1) | 				.Where(x => x.EpisodeNumber != -1) | ||||||
| 				.Where(_database.Like<Episode>(x => x.Title, $"%{query}%")) | 				.Where(_database.Like<Episode>(x => x.Title, $"%{query}%")) | ||||||
| 				.OrderBy(DefaultSort) | 				.OrderBy(DefaultSort) | ||||||
| 				.Take(20) | 				.Take(20) | ||||||
| 				.ToListAsync(); | 				.ToListAsync(); | ||||||
| 			foreach (Episode episode in episodes) |  | ||||||
| 				episode.ShowSlug = await _shows.GetSlug(episode.ShowID); |  | ||||||
| 			return episodes; |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		/// <inheritdoc /> |  | ||||||
| 		public override async Task<ICollection<Episode>> GetAll(Expression<Func<Episode, bool>> where = null,  |  | ||||||
| 			Sort<Episode> sort = default,  |  | ||||||
| 			Pagination limit = default) |  | ||||||
| 		{ |  | ||||||
| 			ICollection<Episode> episodes = await base.GetAll(where, sort, limit); |  | ||||||
| 			foreach (Episode episode in episodes) |  | ||||||
| 				episode.ShowSlug = await _shows.GetSlug(episode.ShowID); |  | ||||||
| 			return episodes; |  | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| 		/// <inheritdoc /> | 		/// <inheritdoc /> | ||||||
|  | |||||||
| @ -2,7 +2,6 @@ using System; | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Linq.Expressions; | using System.Linq.Expressions; | ||||||
| using System.Text.RegularExpressions; |  | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Kyoo.Models; | using Kyoo.Models; | ||||||
| using Kyoo.Models.Exceptions; | using Kyoo.Models.Exceptions; | ||||||
| @ -23,57 +22,22 @@ namespace Kyoo.Controllers | |||||||
| 		/// A provider repository to handle externalID creation and deletion | 		/// A provider repository to handle externalID creation and deletion | ||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		private readonly IProviderRepository _providers; | 		private readonly IProviderRepository _providers; | ||||||
| 		/// <summary> |  | ||||||
| 		/// A show repository to get show's slug from their ID and keep the slug in each episode. |  | ||||||
| 		/// </summary> |  | ||||||
| 		private readonly IShowRepository _shows; |  | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc/> | 		/// <inheritdoc/> | ||||||
| 		protected override Expression<Func<Season, object>> DefaultSort => x => x.SeasonNumber; | 		protected override Expression<Func<Season, object>> DefaultSort => x => x.SeasonNumber; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// Create a new <see cref="SeasonRepository"/> using the provided handle, a provider & a show repository and | 		/// Create a new <see cref="SeasonRepository"/>. | ||||||
| 		/// a service provider to lazilly request an episode repository. |  | ||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		/// <param name="database">The database handle that will be used</param> | 		/// <param name="database">The database handle that will be used</param> | ||||||
| 		/// <param name="providers">A provider repository</param> | 		/// <param name="providers">A provider repository</param> | ||||||
| 		/// <param name="shows">A show repository</param> |  | ||||||
| 		public SeasonRepository(DatabaseContext database, | 		public SeasonRepository(DatabaseContext database, | ||||||
| 			IProviderRepository providers, | 			IProviderRepository providers) | ||||||
| 			IShowRepository shows) |  | ||||||
| 			: base(database) | 			: base(database) | ||||||
| 		{ | 		{ | ||||||
| 			_database = database; | 			_database = database; | ||||||
| 			_providers = providers; | 			_providers = providers; | ||||||
| 			_shows = shows; |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 
 |  | ||||||
| 		/// <inheritdoc/> |  | ||||||
| 		public override async Task<Season> Get(int id) |  | ||||||
| 		{ |  | ||||||
| 			Season ret = await base.Get(id); |  | ||||||
| 			ret.ShowSlug = await _shows.GetSlug(ret.ShowID); |  | ||||||
| 			return ret; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		/// <inheritdoc/> |  | ||||||
| 		public override async Task<Season> Get(Expression<Func<Season, bool>> where) |  | ||||||
| 		{ |  | ||||||
| 			Season ret = await base.Get(where); |  | ||||||
| 			ret.ShowSlug = await _shows.GetSlug(ret.ShowID); |  | ||||||
| 			return ret; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		/// <inheritdoc/> |  | ||||||
| 		public override Task<Season> Get(string slug) |  | ||||||
| 		{ |  | ||||||
| 			Match match = Regex.Match(slug, @"(?<show>.*)-s(?<season>\d*)"); |  | ||||||
| 			 |  | ||||||
| 			if (!match.Success) |  | ||||||
| 				throw new ArgumentException("Invalid season slug. Format: {showSlug}-s{seasonNumber}"); |  | ||||||
| 			return Get(match.Groups["show"].Value, int.Parse(match.Groups["season"].Value)); |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc/> | 		/// <inheritdoc/> | ||||||
| @ -82,7 +46,6 @@ namespace Kyoo.Controllers | |||||||
| 			Season ret = await GetOrDefault(showID, seasonNumber); | 			Season ret = await GetOrDefault(showID, seasonNumber); | ||||||
| 			if (ret == null) | 			if (ret == null) | ||||||
| 				throw new ItemNotFoundException($"No season {seasonNumber} found for the show {showID}"); | 				throw new ItemNotFoundException($"No season {seasonNumber} found for the show {showID}"); | ||||||
| 			ret.ShowSlug = await _shows.GetSlug(showID); |  | ||||||
| 			return ret; | 			return ret; | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| @ -92,7 +55,6 @@ namespace Kyoo.Controllers | |||||||
| 			Season ret = await GetOrDefault(showSlug, seasonNumber); | 			Season ret = await GetOrDefault(showSlug, seasonNumber); | ||||||
| 			if (ret == null) | 			if (ret == null) | ||||||
| 				throw new ItemNotFoundException($"No season {seasonNumber} found for the show {showSlug}"); | 				throw new ItemNotFoundException($"No season {seasonNumber} found for the show {showSlug}"); | ||||||
| 			ret.ShowSlug = showSlug; |  | ||||||
| 			return ret; | 			return ret; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -113,25 +75,11 @@ namespace Kyoo.Controllers | |||||||
| 		/// <inheritdoc/> | 		/// <inheritdoc/> | ||||||
| 		public override async Task<ICollection<Season>> Search(string query) | 		public override async Task<ICollection<Season>> Search(string query) | ||||||
| 		{ | 		{ | ||||||
| 			List<Season> seasons = await _database.Seasons | 			return await _database.Seasons | ||||||
| 				.Where(_database.Like<Season>(x => x.Title, $"%{query}%")) | 				.Where(_database.Like<Season>(x => x.Title, $"%{query}%")) | ||||||
| 				.OrderBy(DefaultSort) | 				.OrderBy(DefaultSort) | ||||||
| 				.Take(20) | 				.Take(20) | ||||||
| 				.ToListAsync(); | 				.ToListAsync(); | ||||||
| 			foreach (Season season in seasons) |  | ||||||
| 				season.ShowSlug = await _shows.GetSlug(season.ShowID); |  | ||||||
| 			return seasons; |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		/// <inheritdoc/> |  | ||||||
| 		public override async Task<ICollection<Season>> GetAll(Expression<Func<Season, bool>> where = null, |  | ||||||
| 			Sort<Season> sort = default,  |  | ||||||
| 			Pagination limit = default) |  | ||||||
| 		{ |  | ||||||
| 			ICollection<Season> seasons = await base.GetAll(where, sort, limit); |  | ||||||
| 			foreach (Season season in seasons) |  | ||||||
| 				season.ShowSlug = await _shows.GetSlug(season.ShowID); |  | ||||||
| 			return seasons; |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/// <inheritdoc/> | 		/// <inheritdoc/> | ||||||
|  | |||||||
| @ -16,7 +16,7 @@ namespace Kyoo.Controllers | |||||||
| 	public class TrackRepository : LocalRepository<Track>, ITrackRepository | 	public class TrackRepository : LocalRepository<Track>, ITrackRepository | ||||||
| 	{ | 	{ | ||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// The databse handle | 		/// The database handle | ||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		private readonly DatabaseContext _database; | 		private readonly DatabaseContext _database; | ||||||
| 		 | 		 | ||||||
| @ -27,7 +27,7 @@ namespace Kyoo.Controllers | |||||||
| 		/// <summary> | 		/// <summary> | ||||||
| 		/// Create a new <see cref="TrackRepository"/>. | 		/// Create a new <see cref="TrackRepository"/>. | ||||||
| 		/// </summary> | 		/// </summary> | ||||||
| 		/// <param name="database">The datatabse handle</param> | 		/// <param name="database">The database handle</param> | ||||||
| 		public TrackRepository(DatabaseContext database)  | 		public TrackRepository(DatabaseContext database)  | ||||||
| 			: base(database) | 			: base(database) | ||||||
| 		{ | 		{ | ||||||
|  | |||||||
| @ -35,9 +35,9 @@ | |||||||
| 		<ProjectReference Include="../Kyoo.Common/Kyoo.Common.csproj" /> | 		<ProjectReference Include="../Kyoo.Common/Kyoo.Common.csproj" /> | ||||||
| 		<ProjectReference Include="../Kyoo.CommonAPI/Kyoo.CommonAPI.csproj" /> | 		<ProjectReference Include="../Kyoo.CommonAPI/Kyoo.CommonAPI.csproj" /> | ||||||
| 		<PackageReference Include="Microsoft.AspNet.WebApi.Client" Version="5.2.7" /> | 		<PackageReference Include="Microsoft.AspNet.WebApi.Client" Version="5.2.7" /> | ||||||
| 		<PackageReference Include="Microsoft.AspNetCore.Mvc.NewtonsoftJson" Version="5.0.6" /> | 		<PackageReference Include="Microsoft.AspNetCore.Mvc.NewtonsoftJson" Version="5.0.7" /> | ||||||
| 		<PackageReference Include="Microsoft.AspNetCore.SpaServices" Version="3.1.15" /> | 		<PackageReference Include="Microsoft.AspNetCore.SpaServices" Version="3.1.16" /> | ||||||
| 		<PackageReference Include="Microsoft.AspNetCore.SpaServices.Extensions" Version="5.0.6" /> | 		<PackageReference Include="Microsoft.AspNetCore.SpaServices.Extensions" Version="5.0.7" /> | ||||||
| 		<PackageReference Include="Newtonsoft.Json" Version="13.0.1" /> | 		<PackageReference Include="Newtonsoft.Json" Version="13.0.1" /> | ||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
| 	 | 	 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user