mirror of
				https://github.com/zoriya/Kyoo.git
				synced 2025-10-25 15:52:36 -04:00 
			
		
		
		
	Adding a field property with auto loading & removing non-loaded fields from the json serialization. Reworking all json serialization
This commit is contained in:
		
							parent
							
								
									97ab1affa0
								
							
						
					
					
						commit
						faf453e64a
					
				| @ -69,10 +69,10 @@ namespace Kyoo.Controllers | |||||||
| 			where T : class, IResource | 			where T : class, IResource | ||||||
| 			where T2 : class, new(); | 			where T2 : class, new(); | ||||||
| 
 | 
 | ||||||
| 		Task<T> Load<T>(T obj, string memberName) | 		Task<T> Load<T>([NotNull] T obj, string memberName) | ||||||
| 			where T : class, IResource; | 			where T : class, IResource; | ||||||
| 
 | 
 | ||||||
| 		Task Load(IResource obj, string memberName); | 		Task Load([NotNull] IResource obj, string memberName); | ||||||
| 
 | 
 | ||||||
| 		// Library Items relations | 		// Library Items relations | ||||||
| 		Task<ICollection<LibraryItem>> GetItemsFromLibrary(int id, | 		Task<ICollection<LibraryItem>> GetItemsFromLibrary(int id, | ||||||
|  | |||||||
| @ -3,5 +3,5 @@ using System; | |||||||
| namespace Kyoo.Models.Attributes | namespace Kyoo.Models.Attributes | ||||||
| { | { | ||||||
| 	[AttributeUsage(AttributeTargets.Class)] | 	[AttributeUsage(AttributeTargets.Class)] | ||||||
| 	public class ComposedSlug : Attribute { } | 	public class ComposedSlugAttribute : Attribute { } | ||||||
| } | } | ||||||
| @ -1,7 +0,0 @@ | |||||||
| using System; |  | ||||||
| 
 |  | ||||||
| namespace Kyoo.Models.Attributes |  | ||||||
| { |  | ||||||
| 	[AttributeUsage(AttributeTargets.Property, Inherited = false)] |  | ||||||
| 	public class EditableRelation : Attribute { } |  | ||||||
| } |  | ||||||
| @ -8,7 +8,4 @@ namespace Kyoo.Models.Attributes | |||||||
| 	{ | 	{ | ||||||
| 		void OnMerge(object merged); | 		void OnMerge(object merged); | ||||||
| 	} | 	} | ||||||
| 	 |  | ||||||
| 	public class JsonReadOnly : Attribute { } |  | ||||||
| 	public class JsonIgnore : JsonReadOnly { } |  | ||||||
| } | } | ||||||
							
								
								
									
										10
									
								
								Kyoo.Common/Models/Attributes/RelationAttributes.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								Kyoo.Common/Models/Attributes/RelationAttributes.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | using System; | ||||||
|  | 
 | ||||||
|  | namespace Kyoo.Models.Attributes | ||||||
|  | { | ||||||
|  | 	[AttributeUsage(AttributeTargets.Property, Inherited = false)] | ||||||
|  | 	public class EditableRelationAttribute : Attribute { } | ||||||
|  | 	 | ||||||
|  | 	[AttributeUsage(AttributeTargets.Property)] | ||||||
|  | 	public class LoadableRelationAttribute : Attribute { } | ||||||
|  | } | ||||||
							
								
								
									
										10
									
								
								Kyoo.Common/Models/Attributes/SerializeAttribute.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								Kyoo.Common/Models/Attributes/SerializeAttribute.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | using System; | ||||||
|  | 
 | ||||||
|  | namespace Kyoo.Models.Attributes | ||||||
|  | { | ||||||
|  | 	[AttributeUsage(AttributeTargets.Property | AttributeTargets.Field)] | ||||||
|  | 	public class SerializeIgnoreAttribute : Attribute {} | ||||||
|  | 	 | ||||||
|  | 	[AttributeUsage(AttributeTargets.Property | AttributeTargets.Field)] | ||||||
|  | 	public class DeserializeIgnoreAttribute : Attribute {} | ||||||
|  | } | ||||||
| @ -4,21 +4,21 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class MetadataID | 	public class MetadataID | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID { get; set; } | 		[SerializeIgnore] public int ID { get; set; } | ||||||
| 		[JsonIgnore] public int ProviderID { get; set; } | 		[SerializeIgnore] public int ProviderID { get; set; } | ||||||
| 		public virtual ProviderID Provider {get; set; } | 		public virtual ProviderID Provider {get; set; } | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] public int? ShowID { get; set; }  | 		[SerializeIgnore] public int? ShowID { get; set; }  | ||||||
| 		[JsonIgnore] public virtual Show Show { get; set; } | 		[SerializeIgnore] public virtual Show Show { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] public int? EpisodeID { get; set; }  | 		[SerializeIgnore] public int? EpisodeID { get; set; }  | ||||||
| 		[JsonIgnore] public virtual Episode Episode { get; set; } | 		[SerializeIgnore] public virtual Episode Episode { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] public int? SeasonID { get; set; }  | 		[SerializeIgnore] public int? SeasonID { get; set; }  | ||||||
| 		[JsonIgnore] public virtual Season Season { get; set; } | 		[SerializeIgnore] public virtual Season Season { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] public int? PeopleID { get; set; }  | 		[SerializeIgnore] public int? PeopleID { get; set; }  | ||||||
| 		[JsonIgnore] public virtual People People { get; set; } | 		[SerializeIgnore] public virtual People People { get; set; } | ||||||
| 		 | 		 | ||||||
| 		public string DataID { get; set; } | 		public string DataID { get; set; } | ||||||
| 		public string Link { get; set; } | 		public string Link { get; set; } | ||||||
|  | |||||||
| @ -8,9 +8,9 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class PeopleRole : IResource | 	public class PeopleRole : IResource | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID { get; set; } | 		[SerializeIgnore] public int ID { get; set; } | ||||||
| 		[JsonIgnore] public int PeopleID { get; set; } | 		[SerializeIgnore] public int PeopleID { get; set; } | ||||||
| 		[JsonIgnore] public virtual People People { get; set; } | 		[SerializeIgnore] public virtual People People { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[ExpressionRewrite(nameof(People) + "." + nameof(Models.People.Slug))] | 		[ExpressionRewrite(nameof(People) + "." + nameof(Models.People.Slug))] | ||||||
| 		public string Slug | 		public string Slug | ||||||
| @ -40,8 +40,8 @@ namespace Kyoo.Models | |||||||
| 			set => People.ExternalIDs = value; | 			set => People.ExternalIDs = value; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		[JsonIgnore] public int ShowID { get; set; } | 		[SerializeIgnore] public int ShowID { get; set; } | ||||||
| 		[JsonIgnore] public virtual Show Show { get; set; } | 		[SerializeIgnore] public virtual Show Show { get; set; } | ||||||
| 		public string Role { get; set; } | 		public string Role { get; set; } | ||||||
| 		public string Type { get; set; } | 		public string Type { get; set; } | ||||||
| 
 | 
 | ||||||
| @ -77,7 +77,7 @@ namespace Kyoo.Models | |||||||
| 		public string Slug { get; set; } | 		public string Slug { get; set; } | ||||||
| 		public string Title { get; set; } | 		public string Title { get; set; } | ||||||
| 		public ICollection<string> Aliases { get; set; } | 		public ICollection<string> Aliases { get; set; } | ||||||
| 		[JsonIgnore] public string Path { get; set; } | 		[SerializeIgnore] public string Path { get; set; } | ||||||
| 		public string Overview { get; set; } | 		public string Overview { get; set; } | ||||||
| 		public Status? Status { get; set; } | 		public Status? Status { get; set; } | ||||||
| 		public string TrailerUrl { get; set; } | 		public string TrailerUrl { get; set; } | ||||||
|  | |||||||
| @ -10,8 +10,8 @@ namespace Kyoo.Models | |||||||
| 		public string Name { get; set; } | 		public string Name { get; set; } | ||||||
| 		public string Poster { get; set; } | 		public string Poster { get; set; } | ||||||
| 		public string Overview { get; set; } | 		public string Overview { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Show> Shows { get; set; } | 		[LoadableRelation] public virtual ICollection<Show> Shows { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Library> Libraries { get; set; } | 		[LoadableRelation] public virtual ICollection<Library> Libraries { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public Collection() { } | 		public Collection() { } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -9,37 +9,29 @@ namespace Kyoo.Models | |||||||
| 	public class Episode : IResource, IOnMerge | 	public class Episode : IResource, IOnMerge | ||||||
| 	{ | 	{ | ||||||
| 		public int ID { get; set; } | 		public int ID { get; set; } | ||||||
|  | 		public string Slug => Show != null ? GetSlug(Show.Slug, SeasonNumber, EpisodeNumber) : ID.ToString(); | ||||||
| 		public int ShowID { get; set; } | 		public int ShowID { get; set; } | ||||||
| 		[JsonIgnore] public virtual Show Show { get; set; } | 		[LoadableRelation] public virtual Show Show { get; set; } | ||||||
| 		public int? SeasonID { get; set; } | 		public int? SeasonID { get; set; } | ||||||
| 		[JsonIgnore] public virtual Season Season { get; set; } | 		[LoadableRelation] public virtual Season Season { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public int SeasonNumber { get; set; } = -1; | 		public int SeasonNumber { get; set; } = -1; | ||||||
| 		public int EpisodeNumber { get; set; } = -1; | 		public int EpisodeNumber { get; set; } = -1; | ||||||
| 		public int AbsoluteNumber { get; set; } = -1; | 		public int AbsoluteNumber { get; set; } = -1; | ||||||
| 		[JsonIgnore] public string Path { get; set; } | 		[SerializeIgnore] public string Path { get; set; } | ||||||
|  | 		public string Thumb => $"/api/episodes/{Slug}/thumb"; | ||||||
| 		public string Title { get; set; } | 		public string Title { get; set; } | ||||||
| 		public string Overview { get; set; } | 		public string Overview { get; set; } | ||||||
| 		public DateTime? ReleaseDate { get; set; } | 		public DateTime? ReleaseDate { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public int Runtime { get; set; } //This runtime variable should be in minutes | 		public int Runtime { get; set; } //This runtime variable should be in minutes | ||||||
| 
 | 
 | ||||||
| 		[JsonIgnore] public string Poster { get; set; } | 		[SerializeIgnore] public string Poster { get; set; } | ||||||
| 		[EditableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | 		[LoadableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | ||||||
| 
 | 
 | ||||||
| 		[JsonIgnore] public virtual ICollection<Track> Tracks { get; set; } | 		[LoadableRelation] public virtual ICollection<Track> Tracks { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public string ShowTitle => Show.Title; | 		public string ShowTitle => Show?.Title; | ||||||
| 		public string Slug => Show != null ? GetSlug(Show.Slug, SeasonNumber, EpisodeNumber) : ID.ToString(); |  | ||||||
| 		public string Thumb |  | ||||||
| 		{ |  | ||||||
| 			get |  | ||||||
| 			{ |  | ||||||
| 				if (Show != null) |  | ||||||
| 					return "thumb/" + Slug; |  | ||||||
| 				return Poster; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 		public Episode() { } | 		public Episode() { } | ||||||
|  | |||||||
| @ -5,11 +5,11 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class Genre : IResource | 	public class Genre : IResource | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID { get; set; } | 		public int ID { get; set; } | ||||||
| 		public string Slug { get; set; } | 		public string Slug { get; set; } | ||||||
| 		public string Name { get; set; } | 		public string Name { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] public virtual ICollection<Show> Shows { get; set; } | 		[LoadableRelation] public virtual ICollection<Show> Shows { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public Genre() {} | 		public Genre() {} | ||||||
| 		 | 		 | ||||||
|  | |||||||
| @ -6,15 +6,15 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class Library : IResource | 	public class Library : IResource | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID { get; set; } | 		public int ID { get; set; } | ||||||
| 		public string Slug { get; set; } | 		public string Slug { get; set; } | ||||||
| 		public string Name { get; set; } | 		public string Name { get; set; } | ||||||
| 		public string[] Paths { get; set; } | 		public string[] Paths { get; set; } | ||||||
| 
 | 
 | ||||||
| 		[EditableRelation] public virtual ICollection<ProviderID> Providers { get; set; } | 		[EditableRelation] public virtual ICollection<ProviderID> Providers { get; set; } | ||||||
| 
 | 
 | ||||||
| 		[JsonIgnore] public virtual ICollection<Show> Shows { get; set; } | 		[LoadableRelation] public virtual ICollection<Show> Shows { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Collection> Collections { get; set; } | 		[LoadableRelation] public virtual ICollection<Collection> Collections { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public Library()  { } | 		public Library()  { } | ||||||
| 		 | 		 | ||||||
|  | |||||||
| @ -10,9 +10,9 @@ namespace Kyoo.Models | |||||||
| 		public string Slug { get; set; } | 		public string Slug { get; set; } | ||||||
| 		public string Name { get; set; } | 		public string Name { get; set; } | ||||||
| 		public string Poster { get; set; } | 		public string Poster { get; set; } | ||||||
| 		[EditableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | 		[EditableRelation] [LoadableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[EditableRelation] [JsonReadOnly] public virtual ICollection<PeopleRole> Roles { get; set; } | 		[EditableRelation] [LoadableRelation] public virtual ICollection<PeopleRole> Roles { get; set; } | ||||||
| 		 | 		 | ||||||
| 		public People() {} | 		public People() {} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class ProviderID : IResource | 	public class ProviderID : IResource | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID { get; set; } | 		public int ID { get; set; } | ||||||
| 		public string Slug { get; set; } | 		public string Slug { get; set; } | ||||||
| 		public string Name { get; set; } | 		public string Name { get; set; } | ||||||
| 		public string Logo { get; set; } | 		public string Logo { get; set; } | ||||||
|  | |||||||
| @ -7,8 +7,8 @@ namespace Kyoo.Models | |||||||
| 	[ComposedSlug] | 	[ComposedSlug] | ||||||
| 	public class Season : IResource | 	public class Season : IResource | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID  { get; set; } | 		public int ID  { get; set; } | ||||||
| 		[JsonIgnore] public int ShowID { get; set; } | 		public int ShowID { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public int SeasonNumber { get; set; } = -1; | 		public int SeasonNumber { get; set; } = -1; | ||||||
| 
 | 
 | ||||||
| @ -17,11 +17,12 @@ namespace Kyoo.Models | |||||||
| 		public string Overview { get; set; } | 		public string Overview { get; set; } | ||||||
| 		public int? Year { get; set; } | 		public int? Year { get; set; } | ||||||
| 
 | 
 | ||||||
| 		[JsonIgnore] public string Poster { get; set; } | 		[SerializeIgnore] public string Poster { get; set; } | ||||||
| 		[EditableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | 		public string Thumb => $"/api/seasons/{Slug}/thumb"; | ||||||
|  | 		[EditableRelation] [LoadableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | ||||||
| 
 | 
 | ||||||
| 		[JsonIgnore] public virtual Show Show { get; set; } | 		[LoadableRelation] public virtual Show Show { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Episode> Episodes { get; set; } | 		[LoadableRelation] public virtual ICollection<Episode> Episodes { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public Season() { } | 		public Season() { } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -10,7 +10,7 @@ namespace Kyoo.Models | |||||||
| 		public string Slug { get; set; } | 		public string Slug { get; set; } | ||||||
| 		public string Title { get; set; } | 		public string Title { get; set; } | ||||||
| 		[EditableRelation] public string[] Aliases { get; set; } | 		[EditableRelation] public string[] Aliases { get; set; } | ||||||
| 		[JsonIgnore] public string Path { get; set; } | 		[SerializeIgnore] public string Path { get; set; } | ||||||
| 		public string Overview { get; set; } | 		public string Overview { get; set; } | ||||||
| 		public Status? Status { get; set; } | 		public Status? Status { get; set; } | ||||||
| 		public string TrailerUrl { get; set; } | 		public string TrailerUrl { get; set; } | ||||||
| @ -24,17 +24,17 @@ namespace Kyoo.Models | |||||||
| 
 | 
 | ||||||
| 		public bool IsMovie { get; set; } | 		public bool IsMovie { get; set; } | ||||||
| 
 | 
 | ||||||
| 		[EditableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | 		[EditableRelation] [LoadableRelation] public virtual ICollection<MetadataID> ExternalIDs { get; set; } | ||||||
| 		 | 		 | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] public int? StudioID { get; set; } | 		public int? StudioID { get; set; } | ||||||
| 		[EditableRelation] [JsonReadOnly] public virtual Studio Studio { get; set; } | 		[LoadableRelation] [EditableRelation] public virtual Studio Studio { get; set; } | ||||||
| 		[EditableRelation] [JsonReadOnly] public virtual ICollection<Genre> Genres { get; set; } | 		[LoadableRelation] [EditableRelation] public virtual ICollection<Genre> Genres { get; set; } | ||||||
| 		[EditableRelation] [JsonReadOnly] public virtual ICollection<PeopleRole> People { get; set; } | 		[LoadableRelation] [EditableRelation] public virtual ICollection<PeopleRole> People { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Season> Seasons { get; set; } | 		[LoadableRelation] public virtual ICollection<Season> Seasons { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Episode> Episodes { get; set; } | 		[LoadableRelation] public virtual ICollection<Episode> Episodes { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Library> Libraries { get; set; } | 		[LoadableRelation] public virtual ICollection<Library> Libraries { get; set; } | ||||||
| 		[JsonIgnore] public virtual ICollection<Collection> Collections { get; set; } | 		[LoadableRelation] public virtual ICollection<Collection> Collections { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public Show() { } | 		public Show() { } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -5,11 +5,11 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class Studio : IResource | 	public class Studio : IResource | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID { get; set; } | 		public int ID { get; set; } | ||||||
| 		public string Slug { get; set; } | 		public string Slug { get; set; } | ||||||
| 		public string Name { get; set; } | 		public string Name { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] public virtual ICollection<Show> Shows { get; set; } | 		[LoadableRelation] public virtual ICollection<Show> Shows { get; set; } | ||||||
| 
 | 
 | ||||||
| 		public Studio() { } | 		public Studio() { } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -25,8 +25,8 @@ namespace Kyoo.Models | |||||||
| 			public string Codec { get; set; } | 			public string Codec { get; set; } | ||||||
| 			[MarshalAs(UnmanagedType.I1)] public bool isDefault; | 			[MarshalAs(UnmanagedType.I1)] public bool isDefault; | ||||||
| 			[MarshalAs(UnmanagedType.I1)] public bool isForced; | 			[MarshalAs(UnmanagedType.I1)] public bool isForced; | ||||||
| 			[JsonIgnore] public string Path { get; set; } | 			[SerializeIgnore] public string Path { get; set; } | ||||||
| 			[JsonIgnore] public StreamType Type { get; set; } | 			[SerializeIgnore] public StreamType Type { get; set; } | ||||||
| 			 | 			 | ||||||
| 			public Stream() {} | 			public Stream() {} | ||||||
| 			 | 			 | ||||||
| @ -56,8 +56,8 @@ namespace Kyoo.Models | |||||||
| 
 | 
 | ||||||
| 	public class Track : Stream, IResource | 	public class Track : Stream, IResource | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public int ID { get; set; } | 		public int ID { get; set; } | ||||||
| 		[JsonIgnore] public int EpisodeID { get; set; } | 		public int EpisodeID { get; set; } | ||||||
| 		public bool IsDefault | 		public bool IsDefault | ||||||
| 		{ | 		{ | ||||||
| 			get => isDefault; | 			get => isDefault; | ||||||
| @ -113,8 +113,8 @@ namespace Kyoo.Models | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		[JsonIgnore] public bool IsExternal { get; set; } | 		public bool IsExternal { get; set; } | ||||||
| 		[JsonIgnore] public virtual Episode Episode { get; set; } | 		[LoadableRelation] public virtual Episode Episode { get; set; } | ||||||
| 		 | 		 | ||||||
| 		public Track() { } | 		public Track() { } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -5,7 +5,6 @@ using System.Linq; | |||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Kyoo.Controllers; | using Kyoo.Controllers; | ||||||
| using Kyoo.Models.Attributes; | using Kyoo.Models.Attributes; | ||||||
| using Kyoo.Models.Watch; |  | ||||||
| using PathIO = System.IO.Path; | using PathIO = System.IO.Path; | ||||||
| 
 | 
 | ||||||
| namespace Kyoo.Models | namespace Kyoo.Models | ||||||
| @ -26,7 +25,7 @@ namespace Kyoo.Models | |||||||
| 	 | 	 | ||||||
| 	public class WatchItem | 	public class WatchItem | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] public readonly int EpisodeID = -1; | 		public readonly int EpisodeID = -1; | ||||||
| 
 | 
 | ||||||
| 		public string ShowTitle; | 		public string ShowTitle; | ||||||
| 		public string ShowSlug; | 		public string ShowSlug; | ||||||
| @ -35,7 +34,7 @@ namespace Kyoo.Models | |||||||
| 		public string Title; | 		public string Title; | ||||||
| 		public string Slug; | 		public string Slug; | ||||||
| 		public DateTime? ReleaseDate; | 		public DateTime? ReleaseDate; | ||||||
| 		[JsonIgnore] public string Path; | 		[SerializeIgnore] public string Path; | ||||||
| 		public Episode PreviousEpisode; | 		public Episode PreviousEpisode; | ||||||
| 		public Episode NextEpisode; | 		public Episode NextEpisode; | ||||||
| 		public bool IsMovie; | 		public bool IsMovie; | ||||||
|  | |||||||
| @ -27,7 +27,6 @@ namespace Kyoo.CommonApi | |||||||
| 
 | 
 | ||||||
| 		[HttpGet("{id:int}")] | 		[HttpGet("{id:int}")] | ||||||
| 		[Authorize(Policy = "Read")] | 		[Authorize(Policy = "Read")] | ||||||
| 		[JsonDetailed] |  | ||||||
| 		public virtual async Task<ActionResult<T>> Get(int id) | 		public virtual async Task<ActionResult<T>> Get(int id) | ||||||
| 		{ | 		{ | ||||||
| 			T resource = await _repository.Get(id); | 			T resource = await _repository.Get(id); | ||||||
| @ -39,7 +38,6 @@ namespace Kyoo.CommonApi | |||||||
| 
 | 
 | ||||||
| 		[HttpGet("{slug}")] | 		[HttpGet("{slug}")] | ||||||
| 		[Authorize(Policy = "Read")] | 		[Authorize(Policy = "Read")] | ||||||
| 		[JsonDetailed] |  | ||||||
| 		public virtual async Task<ActionResult<T>> Get(string slug) | 		public virtual async Task<ActionResult<T>> Get(string slug) | ||||||
| 		{ | 		{ | ||||||
| 			T resource = await _repository.Get(slug); | 			T resource = await _repository.Get(slug); | ||||||
|  | |||||||
| @ -1,13 +1,5 @@ | |||||||
| using System; |  | ||||||
| using System.Buffers; |  | ||||||
| using System.Collections.Generic; |  | ||||||
| using System.Reflection; | using System.Reflection; | ||||||
| using Kyoo.Models; |  | ||||||
| using Kyoo.Models.Attributes; | using Kyoo.Models.Attributes; | ||||||
| using Microsoft.AspNetCore.Mvc; |  | ||||||
| using Microsoft.AspNetCore.Mvc.Filters; |  | ||||||
| using Microsoft.AspNetCore.Mvc.Formatters; |  | ||||||
| using Microsoft.Extensions.DependencyInjection; |  | ||||||
| using Newtonsoft.Json; | using Newtonsoft.Json; | ||||||
| using Newtonsoft.Json.Serialization; | using Newtonsoft.Json.Serialization; | ||||||
| 
 | 
 | ||||||
| @ -19,85 +11,15 @@ namespace Kyoo.Controllers | |||||||
| 		{ | 		{ | ||||||
| 			JsonProperty property = base.CreateProperty(member, memberSerialization); | 			JsonProperty property = base.CreateProperty(member, memberSerialization); | ||||||
| 			 | 			 | ||||||
| 			property.ShouldSerialize = i => member.GetCustomAttribute<JsonReadOnly>(true) == null; | 			// TODO this get called only once and get cached. | ||||||
| 			property.ShouldDeserialize = i => member.GetCustomAttribute<JsonIgnore>(true) == null; | 
 | ||||||
|  | 			// if (member?.GetCustomAttributes<LoadableRelationAttribute>() != null) | ||||||
|  | 			// 	property.NullValueHandling = NullValueHandling.Ignore; | ||||||
|  | 			// if (member?.GetCustomAttributes<SerializeIgnoreAttribute>() != null) | ||||||
|  | 			// 	property.ShouldSerialize = _ => false; | ||||||
|  | 			// if (member?.GetCustomAttributes<DeserializeIgnoreAttribute>() != null) | ||||||
|  | 			// 	property.ShouldDeserialize = _ => false; | ||||||
| 			return property; | 			return property; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	 |  | ||||||
| 	public class JsonPropertySelector : JsonPropertyIgnorer |  | ||||||
| 	{ |  | ||||||
| 		private readonly Dictionary<Type, HashSet<string>> _ignored; |  | ||||||
| 		private readonly Dictionary<Type, HashSet<string>> _forceSerialize; |  | ||||||
| 
 |  | ||||||
| 		public JsonPropertySelector() |  | ||||||
| 		{ |  | ||||||
| 			_ignored = new Dictionary<Type, HashSet<string>>();	 |  | ||||||
| 			_forceSerialize = new Dictionary<Type, HashSet<string>>(); |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		public JsonPropertySelector(Dictionary<Type, HashSet<string>> ignored,  |  | ||||||
| 			Dictionary<Type, HashSet<string>> forceSerialize = null) |  | ||||||
| 		{ |  | ||||||
| 			_ignored = ignored ?? new Dictionary<Type, HashSet<string>>(); |  | ||||||
| 			_forceSerialize = forceSerialize ?? new Dictionary<Type, HashSet<string>>(); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		private bool IsIgnored(Type type, string propertyName) |  | ||||||
| 		{ |  | ||||||
| 			while (type != null) |  | ||||||
| 			{ |  | ||||||
| 				if (_ignored.ContainsKey(type) && _ignored[type].Contains(propertyName)) |  | ||||||
| 					return true; |  | ||||||
| 				type = type.BaseType; |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			return false; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		private bool IsSerializationForced(Type type, string propertyName) |  | ||||||
| 		{ |  | ||||||
| 			while (type != null) |  | ||||||
| 			{ |  | ||||||
| 				if (_forceSerialize.ContainsKey(type) && _forceSerialize[type].Contains(propertyName)) |  | ||||||
| 					return true; |  | ||||||
| 				type = type.BaseType; |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			return false; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		protected override JsonProperty CreateProperty(MemberInfo member, MemberSerialization memberSerialization) |  | ||||||
| 		{ |  | ||||||
| 			JsonProperty property = base.CreateProperty(member, memberSerialization); |  | ||||||
| 
 |  | ||||||
| 			if (IsSerializationForced(property.DeclaringType, property.PropertyName)) |  | ||||||
| 				property.ShouldSerialize = i => true; |  | ||||||
| 			else if (IsIgnored(property.DeclaringType, property.PropertyName)) |  | ||||||
| 				property.ShouldSerialize = i => false; |  | ||||||
| 			return property; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	public class JsonDetailed : ActionFilterAttribute |  | ||||||
| 	{ |  | ||||||
| 		public override void OnActionExecuted(ActionExecutedContext context) |  | ||||||
| 		{ |  | ||||||
| 			if (context.Result is ObjectResult result) |  | ||||||
| 			{ |  | ||||||
| 				result.Formatters.Add(new NewtonsoftJsonOutputFormatter( |  | ||||||
| 					new JsonSerializerSettings |  | ||||||
| 					{ |  | ||||||
| 						ContractResolver = new JsonPropertySelector(null, new Dictionary<Type, HashSet<string>> |  | ||||||
| 						{ |  | ||||||
| 							{typeof(Show), new HashSet<string> {"genres", "studio"}}, |  | ||||||
| 							{typeof(Episode), new HashSet<string> {"tracks"}}, |  | ||||||
| 							{typeof(PeopleRole), new HashSet<string> {"show"}} |  | ||||||
| 						}) |  | ||||||
| 					}, |  | ||||||
| 				context.HttpContext.RequestServices.GetRequiredService<ArrayPool<char>>(),  |  | ||||||
| 					new MvcOptions())); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
| @ -1,5 +1,4 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections; |  | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Linq.Expressions; | using System.Linq.Expressions; | ||||||
| @ -164,7 +163,7 @@ namespace Kyoo.Controllers | |||||||
| 
 | 
 | ||||||
| 				foreach (NavigationEntry navigation in Database.Entry(old).Navigations) | 				foreach (NavigationEntry navigation in Database.Entry(old).Navigations) | ||||||
| 				{ | 				{ | ||||||
| 					if (navigation.Metadata.PropertyInfo.GetCustomAttribute<EditableRelation>() != null) | 					if (navigation.Metadata.PropertyInfo.GetCustomAttribute<EditableRelationAttribute>() != null) | ||||||
| 					{ | 					{ | ||||||
| 						if (resetOld) | 						if (resetOld) | ||||||
| 						{ | 						{ | ||||||
| @ -207,7 +206,7 @@ namespace Kyoo.Controllers | |||||||
| 		{ | 		{ | ||||||
| 			if (string.IsNullOrEmpty(resource.Slug)) | 			if (string.IsNullOrEmpty(resource.Slug)) | ||||||
| 				throw new ArgumentException("Resource can't have null as a slug."); | 				throw new ArgumentException("Resource can't have null as a slug."); | ||||||
| 			if (int.TryParse(resource.Slug, out int _) && typeof(T).GetCustomAttribute<ComposedSlug>() == null) | 			if (int.TryParse(resource.Slug, out int _) && typeof(T).GetCustomAttribute<ComposedSlugAttribute>() == null) | ||||||
| 			{ | 			{ | ||||||
| 				try | 				try | ||||||
| 				{ | 				{ | ||||||
| @ -222,18 +221,6 @@ namespace Kyoo.Controllers | |||||||
| 					throw new ArgumentException("Resources slug can't be number only."); | 					throw new ArgumentException("Resources slug can't be number only."); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			 |  | ||||||
| 			foreach (PropertyInfo property in typeof(T).GetProperties() |  | ||||||
| 				.Where(x => typeof(IEnumerable).IsAssignableFrom(x.PropertyType)  |  | ||||||
| 				            && !typeof(string).IsAssignableFrom(x.PropertyType)  |  | ||||||
| 				            && x.GetCustomAttribute<EditableRelation>() != null)) |  | ||||||
| 			{ |  | ||||||
| 				object value = property.GetValue(resource); |  | ||||||
| 				if (value == null || value is ICollection || Utility.IsOfGenericType(value, typeof(ICollection<>))) |  | ||||||
| 					continue; |  | ||||||
| 				value = Utility.RunGenericMethod<object>(typeof(Enumerable), "ToList", Utility.GetEnumerableType((IEnumerable)value), value); |  | ||||||
| 				property.SetValue(resource, value); |  | ||||||
| 			} |  | ||||||
| 			return Task.CompletedTask; | 			return Task.CompletedTask; | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
|  | |||||||
| @ -1,9 +1,13 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Reflection; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Kyoo.Controllers; | using Kyoo.Controllers; | ||||||
| using Kyoo.Models; | using Kyoo.Models; | ||||||
|  | using Kyoo.Models.Attributes; | ||||||
| using Microsoft.AspNetCore.Mvc; | using Microsoft.AspNetCore.Mvc; | ||||||
|  | using Microsoft.AspNetCore.Mvc.Controllers; | ||||||
| using Microsoft.AspNetCore.Mvc.Filters; | using Microsoft.AspNetCore.Mvc.Filters; | ||||||
| using Microsoft.Extensions.DependencyInjection; | using Microsoft.Extensions.DependencyInjection; | ||||||
| 
 | 
 | ||||||
| @ -20,8 +24,30 @@ namespace Kyoo.CommonApi | |||||||
| 					where.Remove(key); | 					where.Remove(key); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			context.HttpContext.Items["fields"] = context.HttpContext.Request.Query["fields"].ToArray(); | 			string[] fields = context.HttpContext.Request.Query["fields"].ToArray(); | ||||||
| 			// TODO Check if fields are loadable properties of the return type. If not, shorfail the request. | 			 | ||||||
|  | 			context.HttpContext.Items["fields"] = fields; | ||||||
|  | 			if (context.ActionDescriptor is ControllerActionDescriptor descriptor) | ||||||
|  | 			{ | ||||||
|  | 				Type type = descriptor.MethodInfo.ReturnType; | ||||||
|  | 				type = Utility.GetGenericDefinition(type, typeof(Task<>))?.GetGenericArguments()[0] ?? type; | ||||||
|  | 				type = Utility.GetGenericDefinition(type, typeof(ActionResult<>))?.GetGenericArguments()[0] ?? type; | ||||||
|  | 				type = Utility.GetGenericDefinition(type, typeof(Page<>))?.GetGenericArguments()[0] ?? type; | ||||||
|  | 				 | ||||||
|  | 				PropertyInfo[] properties = type.GetProperties() | ||||||
|  | 					.Where(x => x.GetCustomAttribute<LoadableRelationAttribute>() != null) | ||||||
|  | 					.ToArray(); | ||||||
|  | 				foreach (string field in fields) | ||||||
|  | 				{ | ||||||
|  | 					if (properties.Any(y => string.Equals(y.Name,field, StringComparison.InvariantCultureIgnoreCase))) | ||||||
|  | 						continue; | ||||||
|  | 					context.Result = new BadRequestObjectResult(new | ||||||
|  | 					{ | ||||||
|  | 						Error = $"{field} does not exist on {type.Name}." | ||||||
|  | 					}); | ||||||
|  | 					return; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 			base.OnActionExecuting(context); | 			base.OnActionExecuting(context); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -44,7 +70,7 @@ namespace Kyoo.CommonApi | |||||||
| 			 | 			 | ||||||
| 			if (pageType != null) | 			if (pageType != null) | ||||||
| 			{ | 			{ | ||||||
| 				foreach (IResource resource in ((Page<IResource>)result.Value).Items) | 				foreach (IResource resource in ((dynamic)result.Value).Items) | ||||||
| 				{ | 				{ | ||||||
| 					foreach (string field in fields!) | 					foreach (string field in fields!) | ||||||
| 						await library!.Load(resource, field); | 						await library!.Load(resource, field); | ||||||
| @ -53,7 +79,7 @@ namespace Kyoo.CommonApi | |||||||
| 			else if (result.DeclaredType.IsAssignableTo(typeof(IResource))) | 			else if (result.DeclaredType.IsAssignableTo(typeof(IResource))) | ||||||
| 			{ | 			{ | ||||||
| 				foreach (string field in fields!) | 				foreach (string field in fields!) | ||||||
| 					await library!.Load(result.Value as IResource, field); | 					await library!.Load((IResource)result.Value, field); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class CollectionDE : Collection | 	public class CollectionDE : Collection | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] [NotMergable] public virtual ICollection<CollectionLink> Links { get; set; } | 		[SerializeIgnore] [NotMergable] public virtual ICollection<CollectionLink> Links { get; set; } | ||||||
| 		[ExpressionRewrite(nameof(Links), nameof(CollectionLink.Child))] | 		[ExpressionRewrite(nameof(Links), nameof(CollectionLink.Child))] | ||||||
| 		public override ICollection<Show> Shows | 		public override ICollection<Show> Shows | ||||||
| 		{ | 		{ | ||||||
| @ -14,7 +14,7 @@ namespace Kyoo.Models | |||||||
| 			set => Links = value?.Select(x => new CollectionLink(this, x)).ToList(); | 			set => Links = value?.Select(x => new CollectionLink(this, x)).ToList(); | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] [NotMergable] public virtual ICollection<LibraryLink> LibraryLinks { get; set; } | 		[SerializeIgnore] [NotMergable] public virtual ICollection<LibraryLink> LibraryLinks { get; set; } | ||||||
| 		 | 		 | ||||||
| 		[ExpressionRewrite(nameof(LibraryLinks), nameof(GenreLink.Child))] | 		[ExpressionRewrite(nameof(LibraryLinks), nameof(GenreLink.Child))] | ||||||
| 		public override ICollection<Library> Libraries | 		public override ICollection<Library> Libraries | ||||||
|  | |||||||
| @ -6,10 +6,10 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class GenreDE : Genre | 	public class GenreDE : Genre | ||||||
| 	{ | 	{ | ||||||
| 		[JsonIgnore] [NotMergable] public virtual ICollection<GenreLink> Links { get; set; } | 		[SerializeIgnore] [NotMergable] public virtual ICollection<GenreLink> Links { get; set; } | ||||||
| 
 | 
 | ||||||
| 		[ExpressionRewrite(nameof(Links), nameof(GenreLink.Child))] | 		[ExpressionRewrite(nameof(Links), nameof(GenreLink.Child))] | ||||||
| 		[JsonIgnore] [NotMergable] public override ICollection<Show> Shows | 		[SerializeIgnore] [NotMergable] public override ICollection<Show> Shows | ||||||
| 		{ | 		{ | ||||||
| 			get => Links?.Select(x => x.Parent).ToList(); | 			get => Links?.Select(x => x.Parent).ToList(); | ||||||
| 			set => Links = value?.Select(x => new GenreLink(x, this)).ToList(); | 			set => Links = value?.Select(x => new GenreLink(x, this)).ToList(); | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class LibraryDE : Library | 	public class LibraryDE : Library | ||||||
| 	{ | 	{ | ||||||
| 		[EditableRelation] [JsonIgnore] [NotMergable] public virtual ICollection<ProviderLink> ProviderLinks { get; set; } | 		[EditableRelation] [SerializeIgnore] [NotMergable] public virtual ICollection<ProviderLink> ProviderLinks { get; set; } | ||||||
| 		[ExpressionRewrite(nameof(ProviderLinks), nameof(ProviderLink.Child))] | 		[ExpressionRewrite(nameof(ProviderLinks), nameof(ProviderLink.Child))] | ||||||
| 		public override ICollection<ProviderID> Providers | 		public override ICollection<ProviderID> Providers | ||||||
| 		{ | 		{ | ||||||
| @ -14,7 +14,7 @@ namespace Kyoo.Models | |||||||
| 			set => ProviderLinks = value?.Select(x => new ProviderLink(x, this)).ToList(); | 			set => ProviderLinks = value?.Select(x => new ProviderLink(x, this)).ToList(); | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| 		[JsonIgnore] [NotMergable] public virtual ICollection<LibraryLink> Links { get; set; } | 		[SerializeIgnore] [NotMergable] public virtual ICollection<LibraryLink> Links { get; set; } | ||||||
| 		[ExpressionRewrite(nameof(Links), nameof(LibraryLink.Show))] | 		[ExpressionRewrite(nameof(Links), nameof(LibraryLink.Show))] | ||||||
| 		public override ICollection<Show> Shows | 		public override ICollection<Show> Shows | ||||||
| 		{ | 		{ | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ namespace Kyoo.Models | |||||||
| { | { | ||||||
| 	public class ShowDE : Show | 	public class ShowDE : Show | ||||||
| 	{ | 	{ | ||||||
| 		[EditableRelation] [JsonReadOnly] [NotMergable] public virtual ICollection<GenreLink> GenreLinks { get; set; } | 		[EditableRelation] [SerializeIgnore] [NotMergable] public virtual ICollection<GenreLink> GenreLinks { get; set; } | ||||||
| 		[ExpressionRewrite(nameof(GenreLinks), nameof(GenreLink.Child))] | 		[ExpressionRewrite(nameof(GenreLinks), nameof(GenreLink.Child))] | ||||||
| 		public override ICollection<Genre> Genres | 		public override ICollection<Genre> Genres | ||||||
| 		{ | 		{ | ||||||
| @ -14,7 +14,7 @@ namespace Kyoo.Models | |||||||
| 			set => GenreLinks = value?.Select(x => new GenreLink(this, x)).ToList(); | 			set => GenreLinks = value?.Select(x => new GenreLink(this, x)).ToList(); | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| 		[JsonReadOnly] [NotMergable] public virtual ICollection<LibraryLink> LibraryLinks { get; set; } | 		[SerializeIgnore] [NotMergable] public virtual ICollection<LibraryLink> LibraryLinks { get; set; } | ||||||
| 		[ExpressionRewrite(nameof(LibraryLinks), nameof(LibraryLink.Library))] | 		[ExpressionRewrite(nameof(LibraryLinks), nameof(LibraryLink.Library))] | ||||||
| 		public override ICollection<Library> Libraries | 		public override ICollection<Library> Libraries | ||||||
| 		{ | 		{ | ||||||
| @ -22,7 +22,7 @@ namespace Kyoo.Models | |||||||
| 			set => LibraryLinks = value?.Select(x => new LibraryLink(x, this)).ToList(); | 			set => LibraryLinks = value?.Select(x => new LibraryLink(x, this)).ToList(); | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| 		[JsonReadOnly] [NotMergable] public virtual ICollection<CollectionLink> CollectionLinks { get; set; } | 		[SerializeIgnore] [NotMergable] public virtual ICollection<CollectionLink> CollectionLinks { get; set; } | ||||||
| 		[ExpressionRewrite(nameof(CollectionLinks), nameof(CollectionLink.Parent))] | 		[ExpressionRewrite(nameof(CollectionLinks), nameof(CollectionLink.Parent))] | ||||||
| 		public override ICollection<Collection> Collections | 		public override ICollection<Collection> Collections | ||||||
| 		{ | 		{ | ||||||
|  | |||||||
| @ -7,7 +7,6 @@ using System.Linq; | |||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Kyoo.CommonApi; | using Kyoo.CommonApi; | ||||||
| using Kyoo.Controllers; | using Kyoo.Controllers; | ||||||
| using Kyoo.Models.Exceptions; |  | ||||||
| using Microsoft.AspNetCore.Authorization; | using Microsoft.AspNetCore.Authorization; | ||||||
| using Microsoft.Extensions.Configuration; | using Microsoft.Extensions.Configuration; | ||||||
| 
 | 
 | ||||||
| @ -77,8 +76,6 @@ namespace Kyoo.Api | |||||||
| 			[FromQuery] Dictionary<string, string> where, | 			[FromQuery] Dictionary<string, string> where, | ||||||
| 			[FromQuery] int limit = 30) | 			[FromQuery] int limit = 30) | ||||||
| 		{ | 		{ | ||||||
| 			 |  | ||||||
| 
 |  | ||||||
| 			try | 			try | ||||||
| 			{ | 			{ | ||||||
| 				ICollection<Track> resources = await _libraryManager.GetTracks( | 				ICollection<Track> resources = await _libraryManager.GetTracks( | ||||||
| @ -107,8 +104,6 @@ namespace Kyoo.Api | |||||||
| 			[FromQuery] Dictionary<string, string> where, | 			[FromQuery] Dictionary<string, string> where, | ||||||
| 			[FromQuery] int limit = 30) | 			[FromQuery] int limit = 30) | ||||||
| 		{ | 		{ | ||||||
| 			 |  | ||||||
| 
 |  | ||||||
| 			try | 			try | ||||||
| 			{ | 			{ | ||||||
| 				ICollection<Track> resources = await _libraryManager.GetTracks( | 				ICollection<Track> resources = await _libraryManager.GetTracks( | ||||||
| @ -139,8 +134,6 @@ namespace Kyoo.Api | |||||||
| 			[FromQuery] Dictionary<string, string> where, | 			[FromQuery] Dictionary<string, string> where, | ||||||
| 			[FromQuery] int limit = 30) | 			[FromQuery] int limit = 30) | ||||||
| 		{ | 		{ | ||||||
| 			 |  | ||||||
| 
 |  | ||||||
| 			try | 			try | ||||||
| 			{ | 			{ | ||||||
| 				ICollection<Track> resources = await _libraryManager.GetTracks(ApiHelper.ParseWhere<Track>(where, x => x.Episode.Show.Slug == showSlug  | 				ICollection<Track> resources = await _libraryManager.GetTracks(ApiHelper.ParseWhere<Track>(where, x => x.Episode.Show.Slug == showSlug  | ||||||
| @ -159,6 +152,21 @@ namespace Kyoo.Api | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
|  | 		[HttpGet("{id:int}/thumb")] | ||||||
|  | 		[Authorize(Policy="Read")] | ||||||
|  | 		public async Task<IActionResult> GetThumb(int id) | ||||||
|  | 		{ | ||||||
|  | 			string path = (await _libraryManager.GetEpisode(id))?.Path; | ||||||
|  | 			if (path == null) | ||||||
|  | 				return NotFound(); | ||||||
|  | 
 | ||||||
|  | 			string thumb = Path.ChangeExtension(path, "jpg"); | ||||||
|  | 
 | ||||||
|  | 			if (System.IO.File.Exists(thumb)) | ||||||
|  | 				return new PhysicalFileResult(Path.GetFullPath(thumb), "image/jpg"); | ||||||
|  | 			return NotFound(); | ||||||
|  | 		} | ||||||
|  | 		 | ||||||
| 		[HttpGet("{showSlug}-s{seasonNumber:int}e{episodeNumber:int}/thumb")] | 		[HttpGet("{showSlug}-s{seasonNumber:int}e{episodeNumber:int}/thumb")] | ||||||
| 		[Authorize(Policy="Read")] | 		[Authorize(Policy="Read")] | ||||||
| 		public async Task<IActionResult> GetThumb(string showSlug, int seasonNumber, int episodeNumber) | 		public async Task<IActionResult> GetThumb(string showSlug, int seasonNumber, int episodeNumber) | ||||||
|  | |||||||
| @ -29,7 +29,6 @@ namespace Kyoo.Api | |||||||
| 		[HttpGet("{id:int}/role")] | 		[HttpGet("{id:int}/role")] | ||||||
| 		[HttpGet("{id:int}/roles")] | 		[HttpGet("{id:int}/roles")] | ||||||
| 		[Authorize(Policy = "Read")] | 		[Authorize(Policy = "Read")] | ||||||
| 		[JsonDetailed] |  | ||||||
| 		public async Task<ActionResult<Page<ShowRole>>> GetRoles(int id, | 		public async Task<ActionResult<Page<ShowRole>>> GetRoles(int id, | ||||||
| 			[FromQuery] string sortBy, | 			[FromQuery] string sortBy, | ||||||
| 			[FromQuery] int afterID, | 			[FromQuery] int afterID, | ||||||
| @ -58,7 +57,6 @@ namespace Kyoo.Api | |||||||
| 		[HttpGet("{slug}/role")] | 		[HttpGet("{slug}/role")] | ||||||
| 		[HttpGet("{slug}/roles")] | 		[HttpGet("{slug}/roles")] | ||||||
| 		[Authorize(Policy = "Read")] | 		[Authorize(Policy = "Read")] | ||||||
| 		[JsonDetailed] |  | ||||||
| 		public async Task<ActionResult<Page<ShowRole>>> GetRoles(string slug, | 		public async Task<ActionResult<Page<ShowRole>>> GetRoles(string slug, | ||||||
| 			[FromQuery] string sortBy, | 			[FromQuery] string sortBy, | ||||||
| 			[FromQuery] int afterID, | 			[FromQuery] int afterID, | ||||||
| @ -84,12 +82,20 @@ namespace Kyoo.Api | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
|  | 		[HttpGet("{id:int}/poster")] | ||||||
|  | 		[Authorize(Policy="Read")] | ||||||
|  | 		public async Task<IActionResult> GetPeopleIcon(int id) | ||||||
|  | 		{ | ||||||
|  | 			string slug = (await _libraryManager.GetPeople(id)).Slug; | ||||||
|  | 			return GetPeopleIcon(slug); | ||||||
|  | 		} | ||||||
|  | 		 | ||||||
| 		[HttpGet("{slug}/poster")] | 		[HttpGet("{slug}/poster")] | ||||||
| 		[Authorize(Policy="Read")] | 		[Authorize(Policy="Read")] | ||||||
| 		public IActionResult GetPeopleIcon(string slug) | 		public IActionResult GetPeopleIcon(string slug) | ||||||
| 		{ | 		{ | ||||||
| 			string thumbPath = Path.Combine(_peoplePath, slug + ".jpg"); | 			string thumbPath = Path.GetFullPath(Path.Combine(_peoplePath, slug + ".jpg")); | ||||||
| 			if (!System.IO.File.Exists(thumbPath)) | 			if (!thumbPath.StartsWith(_peoplePath) || !System.IO.File.Exists(thumbPath)) | ||||||
| 				return NotFound(); | 				return NotFound(); | ||||||
| 
 | 
 | ||||||
| 			return new PhysicalFileResult(Path.GetFullPath(thumbPath), "image/jpg"); | 			return new PhysicalFileResult(Path.GetFullPath(thumbPath), "image/jpg"); | ||||||
|  | |||||||
| @ -1,5 +1,6 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
|  | using System.IO; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using Kyoo.CommonApi; | using Kyoo.CommonApi; | ||||||
| using Kyoo.Controllers; | using Kyoo.Controllers; | ||||||
| @ -125,5 +126,36 @@ namespace Kyoo.Api | |||||||
| 		{ | 		{ | ||||||
| 			return await _libraryManager.GetShow(showID); | 			return await _libraryManager.GetShow(showID); | ||||||
| 		} | 		} | ||||||
|  | 		 | ||||||
|  | 		[HttpGet("{id:int}/thumb")] | ||||||
|  | 		[Authorize(Policy="Read")] | ||||||
|  | 		public async Task<IActionResult> GetThumb(int id) | ||||||
|  | 		{ | ||||||
|  | 			// TODO remove the next lambda and use a Season.Path (should exit for seasons in a different folder) | ||||||
|  | 			string path = (await _libraryManager.GetShow(x => x.Seasons.Any(y => y.ID == id)))?.Path; | ||||||
|  | 			int seasonNumber = (await _libraryManager.GetSeason(id)).SeasonNumber; | ||||||
|  | 			if (path == null) | ||||||
|  | 				return NotFound(); | ||||||
|  | 
 | ||||||
|  | 			string thumb = Path.Combine(path, $"season-{seasonNumber}.jpg"); | ||||||
|  | 			if (System.IO.File.Exists(thumb)) | ||||||
|  | 				return new PhysicalFileResult(Path.GetFullPath(thumb), "image/jpg"); | ||||||
|  | 			return NotFound(); | ||||||
|  | 		} | ||||||
|  | 		 | ||||||
|  | 		[HttpGet("{showSlug}-s{seasonNumber:int}/thumb")] | ||||||
|  | 		[Authorize(Policy="Read")] | ||||||
|  | 		public async Task<IActionResult> GetThumb(string showSlug, int seasonNumber) | ||||||
|  | 		{ | ||||||
|  | 			// TODO use a season.Path | ||||||
|  | 			string path = (await _libraryManager.GetShow(showSlug))?.Path; | ||||||
|  | 			if (path == null) | ||||||
|  | 				return NotFound(); | ||||||
|  | 
 | ||||||
|  | 			string thumb = Path.Combine(path, $"season-{seasonNumber}.jpg"); | ||||||
|  | 			if (System.IO.File.Exists(thumb)) | ||||||
|  | 				return new PhysicalFileResult(Path.GetFullPath(thumb), "image/jpg"); | ||||||
|  | 			return NotFound(); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user