diff --git a/back/src/Kyoo.Abstractions/Models/LibraryItem.cs b/back/src/Kyoo.Abstractions/Models/LibraryItem.cs
index a57fb191..b60378af 100644
--- a/back/src/Kyoo.Abstractions/Models/LibraryItem.cs
+++ b/back/src/Kyoo.Abstractions/Models/LibraryItem.cs
@@ -94,6 +94,11 @@ namespace Kyoo.Abstractions.Models
///
public Status Status { get; set; }
+ ///
+ /// How well this item is rated? (from 0 to 100).
+ ///
+ public int Rating { get; set; }
+
///
/// The date this show started airing. It can be null if this is unknown.
///
diff --git a/back/src/Kyoo.Abstractions/Models/News.cs b/back/src/Kyoo.Abstractions/Models/News.cs
index bb5351d8..8216068a 100644
--- a/back/src/Kyoo.Abstractions/Models/News.cs
+++ b/back/src/Kyoo.Abstractions/Models/News.cs
@@ -90,6 +90,11 @@ namespace Kyoo.Abstractions.Models
///
public Status? Status { get; set; }
+ ///
+ /// How well this item is rated? (from 0 to 100).
+ ///
+ public int Rating { get; set; }
+
///
/// The date this movie aired.
///
diff --git a/back/src/Kyoo.Abstractions/Models/Resources/Movie.cs b/back/src/Kyoo.Abstractions/Models/Resources/Movie.cs
index e4489389..7ff7c4b2 100644
--- a/back/src/Kyoo.Abstractions/Models/Resources/Movie.cs
+++ b/back/src/Kyoo.Abstractions/Models/Resources/Movie.cs
@@ -19,7 +19,6 @@
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
-using Kyoo.Abstractions.Controllers;
using Kyoo.Abstractions.Models.Attributes;
using Kyoo.Utils;
using Newtonsoft.Json;
@@ -78,6 +77,11 @@ namespace Kyoo.Abstractions.Models
///
public Status Status { get; set; }
+ ///
+ /// How well this item is rated? (from 0 to 100).
+ ///
+ public int Rating { get; set; }
+
///
/// The date this movie aired.
///
diff --git a/back/src/Kyoo.Abstractions/Models/Resources/Show.cs b/back/src/Kyoo.Abstractions/Models/Resources/Show.cs
index 17b60ee3..7e35c482 100644
--- a/back/src/Kyoo.Abstractions/Models/Resources/Show.cs
+++ b/back/src/Kyoo.Abstractions/Models/Resources/Show.cs
@@ -72,6 +72,11 @@ namespace Kyoo.Abstractions.Models
///
public Status Status { get; set; }
+ ///
+ /// How well this item is rated? (from 0 to 100).
+ ///
+ public int Rating { get; set; }
+
///
/// The date this show started airing. It can be null if this is unknown.
///
diff --git a/back/src/Kyoo.Meilisearch/MeilisearchModule.cs b/back/src/Kyoo.Meilisearch/MeilisearchModule.cs
index 72a0937c..9f983d39 100644
--- a/back/src/Kyoo.Meilisearch/MeilisearchModule.cs
+++ b/back/src/Kyoo.Meilisearch/MeilisearchModule.cs
@@ -60,12 +60,23 @@ namespace Kyoo.Meiliseach
{
CamelCase.ConvertName(nameof(LibraryItem.AirDate)),
CamelCase.ConvertName(nameof(LibraryItem.AddedDate)),
+ CamelCase.ConvertName(nameof(LibraryItem.Rating)),
},
DisplayedAttributes = new[]
{
CamelCase.ConvertName(nameof(LibraryItem.Id)),
CamelCase.ConvertName(nameof(LibraryItem.Kind)),
},
+ RankingRules = new[]
+ {
+ "words",
+ "typo",
+ "proximity",
+ "attribute",
+ "sort",
+ "exactness",
+ $"{CamelCase.ConvertName(nameof(LibraryItem.Rating))}:desc",
+ }
// TODO: Add stopwords
// TODO: Extend default ranking to add ratings.
}
diff --git a/back/src/Kyoo.Postgresql/Migrations/20231031212819_rating.Designer.cs b/back/src/Kyoo.Postgresql/Migrations/20231031212819_rating.Designer.cs
new file mode 100644
index 00000000..fbda3775
--- /dev/null
+++ b/back/src/Kyoo.Postgresql/Migrations/20231031212819_rating.Designer.cs
@@ -0,0 +1,1863 @@
+//
+using System;
+using System.Collections.Generic;
+using Kyoo.Abstractions.Models;
+using Kyoo.Postgresql;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
+
+#nullable disable
+
+namespace Kyoo.Postgresql.Migrations
+{
+ [DbContext(typeof(PostgresContext))]
+ [Migration("20231031212819_rating")]
+ partial class Rating
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "7.0.12")
+ .HasAnnotation("Relational:MaxIdentifierLength", 63);
+
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "genre", new[] { "action", "adventure", "animation", "comedy", "crime", "documentary", "drama", "family", "fantasy", "history", "horror", "music", "mystery", "romance", "science_fiction", "thriller", "war", "western" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "item_kind", new[] { "show", "movie", "collection" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "news_kind", new[] { "episode", "movie" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "status", new[] { "unknown", "finished", "airing", "planned" });
+ NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Collection", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Overview")
+ .HasColumnType("text")
+ .HasColumnName("overview");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.HasKey("Id")
+ .HasName("pk_collections");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_collections_slug");
+
+ b.ToTable("collections", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Episode", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AbsoluteNumber")
+ .HasColumnType("integer")
+ .HasColumnName("absolute_number");
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property("EpisodeNumber")
+ .HasColumnType("integer")
+ .HasColumnName("episode_number");
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Name")
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Overview")
+ .HasColumnType("text")
+ .HasColumnName("overview");
+
+ b.Property("Path")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("path");
+
+ b.Property("ReleaseDate")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("release_date");
+
+ b.Property("SeasonId")
+ .HasColumnType("integer")
+ .HasColumnName("season_id");
+
+ b.Property("SeasonNumber")
+ .HasColumnType("integer")
+ .HasColumnName("season_number");
+
+ b.Property("ShowId")
+ .HasColumnType("integer")
+ .HasColumnName("show_id");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.HasKey("Id")
+ .HasName("pk_episodes");
+
+ b.HasIndex("SeasonId")
+ .HasDatabaseName("ix_episodes_season_id");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_episodes_slug");
+
+ b.HasIndex("ShowId", "SeasonNumber", "EpisodeNumber", "AbsoluteNumber")
+ .IsUnique()
+ .HasDatabaseName("ix_episodes_show_id_season_number_episode_number_absolute_numb");
+
+ b.ToTable("episodes", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.LibraryItem", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property("AirDate")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("air_date");
+
+ b.Property("Aliases")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("aliases");
+
+ b.Property("EndAir")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("end_air");
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Genres")
+ .IsRequired()
+ .HasColumnType("genre[]")
+ .HasColumnName("genres");
+
+ b.Property("Kind")
+ .HasColumnType("item_kind")
+ .HasColumnName("kind");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Overview")
+ .HasColumnType("text")
+ .HasColumnName("overview");
+
+ b.Property("Path")
+ .HasColumnType("text")
+ .HasColumnName("path");
+
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.Property("StartAir")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("start_air");
+
+ b.Property("Status")
+ .HasColumnType("status")
+ .HasColumnName("status");
+
+ b.Property("Tagline")
+ .HasColumnType("text")
+ .HasColumnName("tagline");
+
+ b.Property("Tags")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("tags");
+
+ b.Property("Trailer")
+ .HasColumnType("text")
+ .HasColumnName("trailer");
+
+ b.HasKey("Id")
+ .HasName("pk_library_items");
+
+ b.ToTable("library_items", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Movie", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property("AirDate")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("air_date");
+
+ b.Property("Aliases")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("aliases");
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Genres")
+ .IsRequired()
+ .HasColumnType("genre[]")
+ .HasColumnName("genres");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Overview")
+ .HasColumnType("text")
+ .HasColumnName("overview");
+
+ b.Property("Path")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("path");
+
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.Property("Status")
+ .HasColumnType("status")
+ .HasColumnName("status");
+
+ b.Property("StudioID")
+ .HasColumnType("integer")
+ .HasColumnName("studio_id");
+
+ b.Property("Tagline")
+ .HasColumnType("text")
+ .HasColumnName("tagline");
+
+ b.Property("Tags")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("tags");
+
+ b.Property("Trailer")
+ .HasColumnType("text")
+ .HasColumnName("trailer");
+
+ b.HasKey("Id")
+ .HasName("pk_movies");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_movies_slug");
+
+ b.HasIndex("StudioID")
+ .HasDatabaseName("ix_movies_studio_id");
+
+ b.ToTable("movies", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.News", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AbsoluteNumber")
+ .HasColumnType("integer")
+ .HasColumnName("absolute_number");
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property("AirDate")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("air_date");
+
+ b.Property("Aliases")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("aliases");
+
+ b.Property("EpisodeNumber")
+ .HasColumnType("integer")
+ .HasColumnName("episode_number");
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Genres")
+ .IsRequired()
+ .HasColumnType("genre[]")
+ .HasColumnName("genres");
+
+ b.Property("Kind")
+ .HasColumnType("news_kind")
+ .HasColumnName("kind");
+
+ b.Property("Name")
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Overview")
+ .HasColumnType("text")
+ .HasColumnName("overview");
+
+ b.Property("Path")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("path");
+
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
+ b.Property("SeasonNumber")
+ .HasColumnType("integer")
+ .HasColumnName("season_number");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.Property("Status")
+ .HasColumnType("status")
+ .HasColumnName("status");
+
+ b.Property("Tagline")
+ .HasColumnType("text")
+ .HasColumnName("tagline");
+
+ b.Property("Tags")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("tags");
+
+ b.Property("Trailer")
+ .HasColumnType("text")
+ .HasColumnName("trailer");
+
+ b.HasKey("Id")
+ .HasName("pk_news");
+
+ b.ToTable("news", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.People", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.HasKey("Id")
+ .HasName("pk_people");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_people_slug");
+
+ b.ToTable("people", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.PeopleRole", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("MovieID")
+ .HasColumnType("integer")
+ .HasColumnName("movie_id");
+
+ b.Property("PeopleID")
+ .HasColumnType("integer")
+ .HasColumnName("people_id");
+
+ b.Property("Role")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("role");
+
+ b.Property("ShowID")
+ .HasColumnType("integer")
+ .HasColumnName("show_id");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("type");
+
+ b.HasKey("Id")
+ .HasName("pk_people_roles");
+
+ b.HasIndex("MovieID")
+ .HasDatabaseName("ix_people_roles_movie_id");
+
+ b.HasIndex("PeopleID")
+ .HasDatabaseName("ix_people_roles_people_id");
+
+ b.HasIndex("ShowID")
+ .HasDatabaseName("ix_people_roles_show_id");
+
+ b.ToTable("people_roles", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Season", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property("EndDate")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("end_date");
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Name")
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Overview")
+ .HasColumnType("text")
+ .HasColumnName("overview");
+
+ b.Property("SeasonNumber")
+ .HasColumnType("integer")
+ .HasColumnName("season_number");
+
+ b.Property("ShowId")
+ .HasColumnType("integer")
+ .HasColumnName("show_id");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.Property("StartDate")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("start_date");
+
+ b.HasKey("Id")
+ .HasName("pk_seasons");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_seasons_slug");
+
+ b.HasIndex("ShowId", "SeasonNumber")
+ .IsUnique()
+ .HasDatabaseName("ix_seasons_show_id_season_number");
+
+ b.ToTable("seasons", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Show", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property>("Aliases")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("aliases");
+
+ b.Property("EndAir")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("end_air");
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property>("Genres")
+ .IsRequired()
+ .HasColumnType("genre[]")
+ .HasColumnName("genres");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Overview")
+ .HasColumnType("text")
+ .HasColumnName("overview");
+
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.Property("StartAir")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("start_air");
+
+ b.Property("Status")
+ .HasColumnType("status")
+ .HasColumnName("status");
+
+ b.Property("StudioId")
+ .HasColumnType("integer")
+ .HasColumnName("studio_id");
+
+ b.Property("Tagline")
+ .HasColumnType("text")
+ .HasColumnName("tagline");
+
+ b.Property>("Tags")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("tags");
+
+ b.Property("Trailer")
+ .HasColumnType("text")
+ .HasColumnName("trailer");
+
+ b.HasKey("Id")
+ .HasName("pk_shows");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_shows_slug");
+
+ b.HasIndex("StudioId")
+ .HasDatabaseName("ix_shows_studio_id");
+
+ b.ToTable("shows", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Studio", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("ExternalId")
+ .IsRequired()
+ .HasColumnType("json")
+ .HasColumnName("external_id");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("name");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.HasKey("Id")
+ .HasName("pk_studios");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_studios_slug");
+
+ b.ToTable("studios", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.User", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property("Id"));
+
+ b.Property("AddedDate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("added_date")
+ .HasDefaultValueSql("now() at time zone 'utc'");
+
+ b.Property("Email")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("email");
+
+ b.Property("Password")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("password");
+
+ b.Property("Permissions")
+ .IsRequired()
+ .HasColumnType("text[]")
+ .HasColumnName("permissions");
+
+ b.Property("Slug")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("slug");
+
+ b.Property("Username")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("username");
+
+ b.HasKey("Id")
+ .HasName("pk_users");
+
+ b.HasIndex("Slug")
+ .IsUnique()
+ .HasDatabaseName("ix_users_slug");
+
+ b.ToTable("users", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.WatchedEpisode", b =>
+ {
+ b.Property("UserID")
+ .HasColumnType("integer")
+ .HasColumnName("user_id");
+
+ b.Property("EpisodeID")
+ .HasColumnType("integer")
+ .HasColumnName("episode_id");
+
+ b.Property("WatchedPercentage")
+ .HasColumnType("integer")
+ .HasColumnName("watched_percentage");
+
+ b.HasKey("UserID", "EpisodeID")
+ .HasName("pk_watched_episode");
+
+ b.HasIndex("EpisodeID")
+ .HasDatabaseName("ix_watched_episode_episode_id");
+
+ b.ToTable("watched_episode", (string)null);
+ });
+
+ modelBuilder.Entity("ShowUser", b =>
+ {
+ b.Property("UsersId")
+ .HasColumnType("integer")
+ .HasColumnName("users_id");
+
+ b.Property("WatchedId")
+ .HasColumnType("integer")
+ .HasColumnName("watched_id");
+
+ b.HasKey("UsersId", "WatchedId")
+ .HasName("pk_link_user_show");
+
+ b.HasIndex("WatchedId")
+ .HasDatabaseName("ix_link_user_show_watched_id");
+
+ b.ToTable("link_user_show", (string)null);
+ });
+
+ modelBuilder.Entity("link_collection_movie", b =>
+ {
+ b.Property("collection_id")
+ .HasColumnType("integer")
+ .HasColumnName("collection_id");
+
+ b.Property("movie_id")
+ .HasColumnType("integer")
+ .HasColumnName("movie_id");
+
+ b.HasKey("collection_id", "movie_id")
+ .HasName("pk_link_collection_movie");
+
+ b.HasIndex("movie_id")
+ .HasDatabaseName("ix_link_collection_movie_movie_id");
+
+ b.ToTable("link_collection_movie", (string)null);
+ });
+
+ modelBuilder.Entity("link_collection_show", b =>
+ {
+ b.Property("collection_id")
+ .HasColumnType("integer")
+ .HasColumnName("collection_id");
+
+ b.Property("show_id")
+ .HasColumnType("integer")
+ .HasColumnName("show_id");
+
+ b.HasKey("collection_id", "show_id")
+ .HasName("pk_link_collection_show");
+
+ b.HasIndex("show_id")
+ .HasDatabaseName("ix_link_collection_show_show_id");
+
+ b.ToTable("link_collection_show", (string)null);
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Collection", b =>
+ {
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("CollectionId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("CollectionId");
+
+ b1.ToTable("collections");
+
+ b1.WithOwner()
+ .HasForeignKey("CollectionId")
+ .HasConstraintName("fk_collections_collections_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("CollectionId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("CollectionId");
+
+ b1.ToTable("collections");
+
+ b1.WithOwner()
+ .HasForeignKey("CollectionId")
+ .HasConstraintName("fk_collections_collections_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("CollectionId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("CollectionId");
+
+ b1.ToTable("collections");
+
+ b1.WithOwner()
+ .HasForeignKey("CollectionId")
+ .HasConstraintName("fk_collections_collections_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Episode", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Season", "Season")
+ .WithMany("Episodes")
+ .HasForeignKey("SeasonId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .HasConstraintName("fk_episodes_seasons_season_id");
+
+ b.HasOne("Kyoo.Abstractions.Models.Show", "Show")
+ .WithMany("Episodes")
+ .HasForeignKey("ShowId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_episodes_shows_show_id");
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("EpisodeId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("EpisodeId");
+
+ b1.ToTable("episodes");
+
+ b1.WithOwner()
+ .HasForeignKey("EpisodeId")
+ .HasConstraintName("fk_episodes_episodes_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("EpisodeId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("EpisodeId");
+
+ b1.ToTable("episodes");
+
+ b1.WithOwner()
+ .HasForeignKey("EpisodeId")
+ .HasConstraintName("fk_episodes_episodes_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("EpisodeId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("EpisodeId");
+
+ b1.ToTable("episodes");
+
+ b1.WithOwner()
+ .HasForeignKey("EpisodeId")
+ .HasConstraintName("fk_episodes_episodes_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Season");
+
+ b.Navigation("Show");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.LibraryItem", b =>
+ {
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("LibraryItemId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("LibraryItemId");
+
+ b1.ToTable("library_items");
+
+ b1.WithOwner()
+ .HasForeignKey("LibraryItemId")
+ .HasConstraintName("fk_library_items_library_items_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("LibraryItemId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("LibraryItemId");
+
+ b1.ToTable("library_items");
+
+ b1.WithOwner()
+ .HasForeignKey("LibraryItemId")
+ .HasConstraintName("fk_library_items_library_items_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("LibraryItemId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("LibraryItemId");
+
+ b1.ToTable("library_items");
+
+ b1.WithOwner()
+ .HasForeignKey("LibraryItemId")
+ .HasConstraintName("fk_library_items_library_items_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Movie", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Studio", "Studio")
+ .WithMany("Movies")
+ .HasForeignKey("StudioID")
+ .OnDelete(DeleteBehavior.SetNull)
+ .HasConstraintName("fk_movies_studios_studio_id");
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("MovieId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("MovieId");
+
+ b1.ToTable("movies");
+
+ b1.WithOwner()
+ .HasForeignKey("MovieId")
+ .HasConstraintName("fk_movies_movies_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("MovieId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("MovieId");
+
+ b1.ToTable("movies");
+
+ b1.WithOwner()
+ .HasForeignKey("MovieId")
+ .HasConstraintName("fk_movies_movies_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("MovieId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("MovieId");
+
+ b1.ToTable("movies");
+
+ b1.WithOwner()
+ .HasForeignKey("MovieId")
+ .HasConstraintName("fk_movies_movies_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Studio");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.News", b =>
+ {
+ b.OwnsOne("Kyoo.Abstractions.Models.News+ShowInfo", "Show", b1 =>
+ {
+ b1.Property("NewsId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Id")
+ .HasColumnType("integer")
+ .HasColumnName("show_id");
+
+ b1.Property("Name")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("show_name");
+
+ b1.Property("Slug")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("show_slug");
+
+ b1.HasKey("NewsId");
+
+ b1.ToTable("news");
+
+ b1.WithOwner()
+ .HasForeignKey("NewsId")
+ .HasConstraintName("fk_news_news_id");
+
+ b1.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b2 =>
+ {
+ b2.Property("ShowInfoNewsId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b2.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("show_logo_blurhash");
+
+ b2.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("show_logo_source");
+
+ b2.HasKey("ShowInfoNewsId");
+
+ b2.ToTable("news");
+
+ b2.WithOwner()
+ .HasForeignKey("ShowInfoNewsId")
+ .HasConstraintName("fk_news_news_id");
+ });
+
+ b1.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b2 =>
+ {
+ b2.Property("ShowInfoNewsId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b2.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("show_poster_blurhash");
+
+ b2.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("show_poster_source");
+
+ b2.HasKey("ShowInfoNewsId");
+
+ b2.ToTable("news");
+
+ b2.WithOwner()
+ .HasForeignKey("ShowInfoNewsId")
+ .HasConstraintName("fk_news_news_id");
+ });
+
+ b1.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b2 =>
+ {
+ b2.Property("ShowInfoNewsId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b2.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("show_thumbnail_blurhash");
+
+ b2.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("show_thumbnail_source");
+
+ b2.HasKey("ShowInfoNewsId");
+
+ b2.ToTable("news");
+
+ b2.WithOwner()
+ .HasForeignKey("ShowInfoNewsId")
+ .HasConstraintName("fk_news_news_id");
+ });
+
+ b1.Navigation("Logo");
+
+ b1.Navigation("Poster");
+
+ b1.Navigation("Thumbnail");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("NewsId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("NewsId");
+
+ b1.ToTable("news");
+
+ b1.WithOwner()
+ .HasForeignKey("NewsId")
+ .HasConstraintName("fk_news_news_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("NewsId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("NewsId");
+
+ b1.ToTable("news");
+
+ b1.WithOwner()
+ .HasForeignKey("NewsId")
+ .HasConstraintName("fk_news_news_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("NewsId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("NewsId");
+
+ b1.ToTable("news");
+
+ b1.WithOwner()
+ .HasForeignKey("NewsId")
+ .HasConstraintName("fk_news_news_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Show");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.People", b =>
+ {
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("PeopleId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("PeopleId");
+
+ b1.ToTable("people");
+
+ b1.WithOwner()
+ .HasForeignKey("PeopleId")
+ .HasConstraintName("fk_people_people_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("PeopleId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("PeopleId");
+
+ b1.ToTable("people");
+
+ b1.WithOwner()
+ .HasForeignKey("PeopleId")
+ .HasConstraintName("fk_people_people_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("PeopleId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("PeopleId");
+
+ b1.ToTable("people");
+
+ b1.WithOwner()
+ .HasForeignKey("PeopleId")
+ .HasConstraintName("fk_people_people_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.PeopleRole", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Movie", "Movie")
+ .WithMany("People")
+ .HasForeignKey("MovieID")
+ .HasConstraintName("fk_people_roles_movies_movie_id");
+
+ b.HasOne("Kyoo.Abstractions.Models.People", "People")
+ .WithMany("Roles")
+ .HasForeignKey("PeopleID")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_people_roles_people_people_id");
+
+ b.HasOne("Kyoo.Abstractions.Models.Show", "Show")
+ .WithMany("People")
+ .HasForeignKey("ShowID")
+ .HasConstraintName("fk_people_roles_shows_show_id");
+
+ b.Navigation("Movie");
+
+ b.Navigation("People");
+
+ b.Navigation("Show");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Season", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Show", "Show")
+ .WithMany("Seasons")
+ .HasForeignKey("ShowId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_seasons_shows_show_id");
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("SeasonId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("SeasonId");
+
+ b1.ToTable("seasons");
+
+ b1.WithOwner()
+ .HasForeignKey("SeasonId")
+ .HasConstraintName("fk_seasons_seasons_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("SeasonId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("SeasonId");
+
+ b1.ToTable("seasons");
+
+ b1.WithOwner()
+ .HasForeignKey("SeasonId")
+ .HasConstraintName("fk_seasons_seasons_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("SeasonId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("SeasonId");
+
+ b1.ToTable("seasons");
+
+ b1.WithOwner()
+ .HasForeignKey("SeasonId")
+ .HasConstraintName("fk_seasons_seasons_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Show");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Show", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Studio", "Studio")
+ .WithMany("Shows")
+ .HasForeignKey("StudioId")
+ .OnDelete(DeleteBehavior.SetNull)
+ .HasConstraintName("fk_shows_studios_studio_id");
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("ShowId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("ShowId");
+
+ b1.ToTable("shows");
+
+ b1.WithOwner()
+ .HasForeignKey("ShowId")
+ .HasConstraintName("fk_shows_shows_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Poster", b1 =>
+ {
+ b1.Property("ShowId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("poster_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("poster_source");
+
+ b1.HasKey("ShowId");
+
+ b1.ToTable("shows");
+
+ b1.WithOwner()
+ .HasForeignKey("ShowId")
+ .HasConstraintName("fk_shows_shows_id");
+ });
+
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Thumbnail", b1 =>
+ {
+ b1.Property("ShowId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("thumbnail_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("thumbnail_source");
+
+ b1.HasKey("ShowId");
+
+ b1.ToTable("shows");
+
+ b1.WithOwner()
+ .HasForeignKey("ShowId")
+ .HasConstraintName("fk_shows_shows_id");
+ });
+
+ b.Navigation("Logo");
+
+ b.Navigation("Poster");
+
+ b.Navigation("Studio");
+
+ b.Navigation("Thumbnail");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.User", b =>
+ {
+ b.OwnsOne("Kyoo.Abstractions.Models.Image", "Logo", b1 =>
+ {
+ b1.Property("UserId")
+ .HasColumnType("integer")
+ .HasColumnName("id");
+
+ b1.Property("Blurhash")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)")
+ .HasColumnName("logo_blurhash");
+
+ b1.Property("Source")
+ .IsRequired()
+ .HasColumnType("text")
+ .HasColumnName("logo_source");
+
+ b1.HasKey("UserId");
+
+ b1.ToTable("users");
+
+ b1.WithOwner()
+ .HasForeignKey("UserId")
+ .HasConstraintName("fk_users_users_id");
+ });
+
+ b.Navigation("Logo");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.WatchedEpisode", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Episode", "Episode")
+ .WithMany()
+ .HasForeignKey("EpisodeID")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_watched_episode_episodes_episode_id");
+
+ b.HasOne("Kyoo.Abstractions.Models.User", null)
+ .WithMany("CurrentlyWatching")
+ .HasForeignKey("UserID")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_watched_episode_users_user_id");
+
+ b.Navigation("Episode");
+ });
+
+ modelBuilder.Entity("ShowUser", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.User", null)
+ .WithMany()
+ .HasForeignKey("UsersId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_link_user_show_users_users_id");
+
+ b.HasOne("Kyoo.Abstractions.Models.Show", null)
+ .WithMany()
+ .HasForeignKey("WatchedId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_link_user_show_shows_watched_id");
+ });
+
+ modelBuilder.Entity("link_collection_movie", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Collection", null)
+ .WithMany()
+ .HasForeignKey("collection_id")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_link_collection_movie_collections_collection_id");
+
+ b.HasOne("Kyoo.Abstractions.Models.Movie", null)
+ .WithMany()
+ .HasForeignKey("movie_id")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_link_collection_movie_movies_movie_id");
+ });
+
+ modelBuilder.Entity("link_collection_show", b =>
+ {
+ b.HasOne("Kyoo.Abstractions.Models.Collection", null)
+ .WithMany()
+ .HasForeignKey("collection_id")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_link_collection_show_collections_collection_id");
+
+ b.HasOne("Kyoo.Abstractions.Models.Show", null)
+ .WithMany()
+ .HasForeignKey("show_id")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_link_collection_show_shows_show_id");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Movie", b =>
+ {
+ b.Navigation("People");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.People", b =>
+ {
+ b.Navigation("Roles");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Season", b =>
+ {
+ b.Navigation("Episodes");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Show", b =>
+ {
+ b.Navigation("Episodes");
+
+ b.Navigation("People");
+
+ b.Navigation("Seasons");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.Studio", b =>
+ {
+ b.Navigation("Movies");
+
+ b.Navigation("Shows");
+ });
+
+ modelBuilder.Entity("Kyoo.Abstractions.Models.User", b =>
+ {
+ b.Navigation("CurrentlyWatching");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}
diff --git a/back/src/Kyoo.Postgresql/Migrations/20231031212819_rating.cs b/back/src/Kyoo.Postgresql/Migrations/20231031212819_rating.cs
new file mode 100644
index 00000000..75e9d5f9
--- /dev/null
+++ b/back/src/Kyoo.Postgresql/Migrations/20231031212819_rating.cs
@@ -0,0 +1,120 @@
+// Kyoo - A portable and vast media library solution.
+// Copyright (c) Kyoo.
+//
+// See AUTHORS.md and LICENSE file in the project root for full license information.
+//
+// Kyoo is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// any later version.
+//
+// Kyoo is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with Kyoo. If not, see .
+
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace Kyoo.Postgresql.Migrations
+{
+ ///
+ public partial class Rating : Items
+ {
+ public static void CreateItemView(MigrationBuilder migrationBuilder)
+ {
+ // language=PostgreSQL
+ migrationBuilder.Sql(@"
+ CREATE VIEW library_items AS
+ SELECT
+ s.id, s.slug, s.name, s.tagline, s.aliases, s.overview, s.tags, s.genres, s.status,
+ s.start_air, s.end_air, s.poster_source, s.poster_blurhash, s.thumbnail_source, s.thumbnail_blurhash,
+ s.logo_source, s.logo_blurhash, s.trailer, s.external_id, s.start_air AS air_date, NULL as path,
+ 'show'::item_kind AS kind, s.added_date, s.rating
+ FROM shows AS s
+ UNION ALL
+ SELECT
+ -m.id, m.slug, m.name, m.tagline, m.aliases, m.overview, m.tags, m.genres, m.status,
+ m.air_date as start_air, m.air_date as end_air, m.poster_source, m.poster_blurhash, m.thumbnail_source,
+ m.thumbnail_blurhash, m.logo_source, m.logo_blurhash, m.trailer, m.external_id, m.air_date, m.path,
+ 'movie'::item_kind AS kind, m.added_date, m.rating
+ FROM movies AS m
+ UNION ALL
+ SELECT
+ c.id + 10000 AS id, c.slug, c.name, NULL as tagline, NULL as alises, c.overview, NULL AS tags, NULL AS genres, 'unknown'::status AS status,
+ NULL AS start_air, NULL AS end_air, c.poster_source, c.poster_blurhash, c.thumbnail_source,
+ c.thumbnail_blurhash, c.logo_source, c.logo_blurhash, NULL as trailer, c.external_id, NULL AS air_date, NULL as path,
+ 'collection'::item_kind AS kind, c.added_date, NULL as rating
+ FROM collections AS c
+ ");
+
+ // language=PostgreSQL
+ migrationBuilder.Sql(@"
+ CREATE VIEW news AS
+ SELECT
+ e.id, e.slug, e.name, NULL AS tagline, '{}' AS aliases, e.path, e.overview, '{}' AS tags, '{}' AS genres,
+ NULL AS status, e.release_date AS air_date, e.poster_source, e.poster_blurhash, e.thumbnail_source, e.thumbnail_blurhash,
+ e.logo_source,e.logo_blurhash, NULL AS trailer, e.external_id, e.season_number, e.episode_number, e.absolute_number,
+ 'episode'::news_kind AS kind, e.added_date, s.id AS show_id, s.slug AS show_slug, s.name AS show_name,
+ s.poster_source AS show_poster_source, s.poster_blurhash AS show_poster_blurhash, s.thumbnail_source AS show_thumbnail_source,
+ s.thumbnail_blurhash AS show_thumbnail_blurhash, s.logo_source AS show_logo_source, s.logo_blurhash AS show_logo_blurhash,
+ NULL as rating
+ FROM episodes AS e
+ LEFT JOIN shows AS s ON e.show_id = s.id
+ UNION ALL
+ SELECT
+ -m.id, m.slug, m.name, m.tagline, m.aliases, m.path, m.overview, m.tags, m.genres,
+ m.status, m.air_date, m.poster_source, m.poster_blurhash, m.thumbnail_source, m.thumbnail_blurhash,
+ m.logo_source, m.logo_blurhash, m.trailer, m.external_id, NULL AS season_number, NULL AS episode_number, NULL as absolute_number,
+ 'movie'::news_kind AS kind, m.added_date, NULL AS show_id, NULL AS show_slug, NULL AS show_name,
+ NULL AS show_poster_source, NULL AS show_poster_blurhash, NULL AS show_thumbnail_source, NULL AS show_thumbnail_blurhash,
+ NULL AS show_logo_source, NULL AS show_logo_blurhash, m.rating
+ FROM movies AS m
+ ");
+ }
+
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ base.Down(migrationBuilder);
+ // language=PostgreSQL
+ migrationBuilder.Sql(@"DROP VIEW news");
+
+ migrationBuilder.AddColumn(
+ name: "rating",
+ table: "shows",
+ type: "integer",
+ nullable: false);
+
+ migrationBuilder.AddColumn(
+ name: "rating",
+ table: "movies",
+ type: "integer",
+ nullable: false);
+
+ CreateItemView(migrationBuilder);
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ base.Down(migrationBuilder);
+ // language=PostgreSQL
+ migrationBuilder.Sql(@"DROP VIEW news");
+
+ migrationBuilder.DropColumn(
+ name: "rating",
+ table: "shows");
+
+ migrationBuilder.DropColumn(
+ name: "rating",
+ table: "movies");
+
+ AddedDate.CreateItemView(migrationBuilder);
+ }
+ }
+}
diff --git a/back/src/Kyoo.Postgresql/Migrations/PostgresContextModelSnapshot.cs b/back/src/Kyoo.Postgresql/Migrations/PostgresContextModelSnapshot.cs
index accbb6cb..c8961480 100644
--- a/back/src/Kyoo.Postgresql/Migrations/PostgresContextModelSnapshot.cs
+++ b/back/src/Kyoo.Postgresql/Migrations/PostgresContextModelSnapshot.cs
@@ -1,4 +1,4 @@
-//
+//
using System;
using System.Collections.Generic;
using Kyoo.Abstractions.Models;
@@ -208,6 +208,11 @@ namespace Kyoo.Postgresql.Migrations
.HasColumnType("text")
.HasColumnName("path");
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
b.Property("Slug")
.IsRequired()
.HasMaxLength(256)
@@ -289,6 +294,11 @@ namespace Kyoo.Postgresql.Migrations
.HasColumnType("text")
.HasColumnName("path");
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
b.Property("Slug")
.IsRequired()
.HasMaxLength(256)
@@ -388,6 +398,11 @@ namespace Kyoo.Postgresql.Migrations
.HasColumnType("text")
.HasColumnName("path");
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
b.Property("SeasonNumber")
.HasColumnType("integer")
.HasColumnName("season_number");
@@ -609,6 +624,11 @@ namespace Kyoo.Postgresql.Migrations
.HasColumnType("text")
.HasColumnName("overview");
+ b.Property("Rating")
+ .IsRequired()
+ .HasColumnType("integer")
+ .HasColumnName("rating");
+
b.Property("Slug")
.IsRequired()
.HasMaxLength(256)
diff --git a/front/packages/ui/src/browse/types.ts b/front/packages/ui/src/browse/types.ts
index 55e33f3b..f44181e0 100644
--- a/front/packages/ui/src/browse/types.ts
+++ b/front/packages/ui/src/browse/types.ts
@@ -23,12 +23,14 @@ export enum SortBy {
StartAir = "startAir",
EndAir = "endAir",
AddedDate = "addedDate",
+ Ratings = "rating",
}
export enum SearchSort {
Relevance = "relevance",
AirDate = "airDate",
AddedDate = "addedDate",
+ Ratings = "rating",
}
export enum SortOrd {
diff --git a/front/translations/en.json b/front/translations/en.json
index 3ed9f7de..dfd3a370 100644
--- a/front/translations/en.json
+++ b/front/translations/en.json
@@ -28,7 +28,8 @@
"airDate": "Air Date",
"startAir": "Start air",
"endAir": "End air",
- "addedDate": "Added date"
+ "addedDate": "Added date",
+ "rating": "Ratings"
},
"sortord": {
"asc": "asc",
diff --git a/front/translations/fr.json b/front/translations/fr.json
index 47b85d2c..cf1ac44f 100644
--- a/front/translations/fr.json
+++ b/front/translations/fr.json
@@ -23,10 +23,13 @@
"sortby": "Trier par {{key}}",
"sortby-tt": "Trier par",
"sortkey": {
+ "relevance": "Pertinence",
"name": "Nom",
+ "airDate": "Date de sortie",
"startAir": "Date de sortie",
"endAir": "Date de fin de sortie",
- "addedDate": "Date d'ajout"
+ "addedDate": "Date d'ajout",
+ "rating": "Notes"
},
"sortord": {
"asc": "asc",
diff --git a/scanner/providers/implementations/themoviedatabase.py b/scanner/providers/implementations/themoviedatabase.py
index f507ca83..43352005 100644
--- a/scanner/providers/implementations/themoviedatabase.py
+++ b/scanner/providers/implementations/themoviedatabase.py
@@ -138,6 +138,7 @@ class TheMovieDatabase(Provider):
status=MovieStatus.FINISHED
if movie["status"] == "Released"
else MovieStatus.PLANNED,
+ rating=round(float(movie["vote_average"]) * 10),
studios=[self.to_studio(x) for x in movie["production_companies"]],
genres=[
self.genre_map[x["id"]]
@@ -232,6 +233,7 @@ class TheMovieDatabase(Provider):
else ShowStatus.AIRING
if show["in_production"]
else ShowStatus.FINISHED,
+ rating=round(float(show["vote_average"]) * 10),
studios=[self.to_studio(x) for x in show["production_companies"]],
genres=[
self.genre_map[x["id"]]
diff --git a/scanner/providers/types/movie.py b/scanner/providers/types/movie.py
index 11117993..76afc089 100644
--- a/scanner/providers/types/movie.py
+++ b/scanner/providers/types/movie.py
@@ -34,6 +34,7 @@ class Movie:
aliases: list[str] = field(default_factory=list)
air_date: Optional[date | int] = None
status: Status = Status.UNKNOWN
+ rating: int = None
path: Optional[str] = None
studios: list[Studio] = field(default_factory=list)
genres: list[Genre] = field(default_factory=list)
diff --git a/scanner/providers/types/show.py b/scanner/providers/types/show.py
index 1a2da3ab..65a37a9c 100644
--- a/scanner/providers/types/show.py
+++ b/scanner/providers/types/show.py
@@ -37,6 +37,7 @@ class Show:
start_air: Optional[date | int]
end_air: Optional[date | int]
status: Status
+ rating: int
studios: list[Studio]
genres: list[Genre]
seasons: list[Season]