BaseItemRepository.cs 80 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173
  1. #pragma warning disable RS0030 // Do not use banned APIs
  2. // Do not enforce that because EFCore cannot deal with cultures well.
  3. #pragma warning disable CA1304 // Specify CultureInfo
  4. #pragma warning disable CA1311 // Specify a culture or use an invariant version
  5. #pragma warning disable CA1862 // Use the 'StringComparison' method overloads to perform case-insensitive string comparisons
  6. using System;
  7. using System.Collections.Concurrent;
  8. using System.Collections.Generic;
  9. using System.Collections.Immutable;
  10. using System.Globalization;
  11. using System.IO;
  12. using System.Linq;
  13. using System.Linq.Expressions;
  14. using System.Reflection;
  15. using System.Text;
  16. using System.Text.Json;
  17. using System.Threading;
  18. using Jellyfin.Data.Entities;
  19. using Jellyfin.Data.Enums;
  20. using Jellyfin.Extensions;
  21. using Jellyfin.Extensions.Json;
  22. using MediaBrowser.Common;
  23. using MediaBrowser.Controller;
  24. using MediaBrowser.Controller.Channels;
  25. using MediaBrowser.Controller.Configuration;
  26. using MediaBrowser.Controller.Entities;
  27. using MediaBrowser.Controller.Entities.Audio;
  28. using MediaBrowser.Controller.Entities.TV;
  29. using MediaBrowser.Controller.LiveTv;
  30. using MediaBrowser.Controller.Persistence;
  31. using MediaBrowser.Model.Dto;
  32. using MediaBrowser.Model.Entities;
  33. using MediaBrowser.Model.LiveTv;
  34. using MediaBrowser.Model.Querying;
  35. using Microsoft.EntityFrameworkCore;
  36. using Microsoft.Extensions.Logging;
  37. using BaseItemDto = MediaBrowser.Controller.Entities.BaseItem;
  38. using BaseItemEntity = Jellyfin.Data.Entities.BaseItemEntity;
  39. namespace Jellyfin.Server.Implementations.Item;
  40. /*
  41. All queries in this class and all other nullable enabled EFCore repository classes will make liberal use of the null-forgiving operator "!".
  42. This is done as the code isn't actually executed client side, but only the expressions are interpret and the compiler cannot know that.
  43. This is your only warning/message regarding this topic.
  44. */
  45. /// <summary>
  46. /// Handles all storage logic for BaseItems.
  47. /// </summary>
  48. public sealed class BaseItemRepository
  49. : IItemRepository
  50. {
  51. /// <summary>
  52. /// This holds all the types in the running assemblies
  53. /// so that we can de-serialize properly when we don't have strong types.
  54. /// </summary>
  55. private static readonly ConcurrentDictionary<string, Type?> _typeMap = new ConcurrentDictionary<string, Type?>();
  56. private readonly IDbContextFactory<JellyfinDbContext> _dbProvider;
  57. private readonly IServerApplicationHost _appHost;
  58. private readonly IItemTypeLookup _itemTypeLookup;
  59. private readonly IServerConfigurationManager _serverConfigurationManager;
  60. private readonly ILogger<BaseItemRepository> _logger;
  61. private static readonly IReadOnlyList<ItemValueType> _getAllArtistsValueTypes = [ItemValueType.Artist, ItemValueType.AlbumArtist];
  62. private static readonly IReadOnlyList<ItemValueType> _getArtistValueTypes = [ItemValueType.Artist];
  63. private static readonly IReadOnlyList<ItemValueType> _getAlbumArtistValueTypes = [ItemValueType.AlbumArtist];
  64. private static readonly IReadOnlyList<ItemValueType> _getStudiosValueTypes = [ItemValueType.Studios];
  65. private static readonly IReadOnlyList<ItemValueType> _getGenreValueTypes = [ItemValueType.Studios];
  66. /// <summary>
  67. /// Initializes a new instance of the <see cref="BaseItemRepository"/> class.
  68. /// </summary>
  69. /// <param name="dbProvider">The db factory.</param>
  70. /// <param name="appHost">The Application host.</param>
  71. /// <param name="itemTypeLookup">The static type lookup.</param>
  72. /// <param name="serverConfigurationManager">The server Configuration manager.</param>
  73. /// <param name="logger">System logger.</param>
  74. public BaseItemRepository(
  75. IDbContextFactory<JellyfinDbContext> dbProvider,
  76. IServerApplicationHost appHost,
  77. IItemTypeLookup itemTypeLookup,
  78. IServerConfigurationManager serverConfigurationManager,
  79. ILogger<BaseItemRepository> logger)
  80. {
  81. _dbProvider = dbProvider;
  82. _appHost = appHost;
  83. _itemTypeLookup = itemTypeLookup;
  84. _serverConfigurationManager = serverConfigurationManager;
  85. _logger = logger;
  86. }
  87. /// <inheritdoc />
  88. public void DeleteItem(Guid id)
  89. {
  90. ArgumentNullException.ThrowIfNull(id.IsEmpty() ? null : id);
  91. using var context = _dbProvider.CreateDbContext();
  92. using var transaction = context.Database.BeginTransaction();
  93. context.PeopleBaseItemMap.Where(e => e.ItemId == id).ExecuteDelete();
  94. context.Peoples.Where(e => e.BaseItems!.Count == 0).ExecuteDelete();
  95. context.Chapters.Where(e => e.ItemId == id).ExecuteDelete();
  96. context.MediaStreamInfos.Where(e => e.ItemId == id).ExecuteDelete();
  97. context.AncestorIds.Where(e => e.ItemId == id || e.ParentItemId == id).ExecuteDelete();
  98. context.ItemValuesMap.Where(e => e.ItemId == id).ExecuteDelete();
  99. context.ItemValues.Where(e => e.BaseItemsMap!.Count == 0).ExecuteDelete();
  100. context.BaseItemImageInfos.Where(e => e.ItemId == id).ExecuteDelete();
  101. context.BaseItemProviders.Where(e => e.ItemId == id).ExecuteDelete();
  102. context.BaseItems.Where(e => e.Id == id).ExecuteDelete();
  103. context.SaveChanges();
  104. transaction.Commit();
  105. }
  106. /// <inheritdoc />
  107. public void UpdateInheritedValues()
  108. {
  109. using var context = _dbProvider.CreateDbContext();
  110. using var transaction = context.Database.BeginTransaction();
  111. context.ItemValuesMap.Where(e => e.ItemValue.Type == ItemValueType.InheritedTags).ExecuteDelete();
  112. // ItemValue Inheritance is now correctly mapped via AncestorId on demand
  113. context.SaveChanges();
  114. transaction.Commit();
  115. }
  116. /// <inheritdoc />
  117. public IReadOnlyList<Guid> GetItemIdsList(InternalItemsQuery filter)
  118. {
  119. ArgumentNullException.ThrowIfNull(filter);
  120. PrepareFilterQuery(filter);
  121. using var context = _dbProvider.CreateDbContext();
  122. return ApplyQueryFilter(context.BaseItems.AsNoTracking(), context, filter).Select(e => e.Id).ToArray();
  123. }
  124. /// <inheritdoc />
  125. public QueryResult<(BaseItemDto Item, ItemCounts ItemCounts)> GetAllArtists(InternalItemsQuery filter)
  126. {
  127. return GetItemValues(filter, _getAllArtistsValueTypes, _itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicArtist]);
  128. }
  129. /// <inheritdoc />
  130. public QueryResult<(BaseItemDto Item, ItemCounts ItemCounts)> GetArtists(InternalItemsQuery filter)
  131. {
  132. return GetItemValues(filter, _getArtistValueTypes, _itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicArtist]);
  133. }
  134. /// <inheritdoc />
  135. public QueryResult<(BaseItemDto Item, ItemCounts ItemCounts)> GetAlbumArtists(InternalItemsQuery filter)
  136. {
  137. return GetItemValues(filter, _getAlbumArtistValueTypes, _itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicArtist]);
  138. }
  139. /// <inheritdoc />
  140. public QueryResult<(BaseItemDto Item, ItemCounts ItemCounts)> GetStudios(InternalItemsQuery filter)
  141. {
  142. return GetItemValues(filter, _getStudiosValueTypes, _itemTypeLookup.BaseItemKindNames[BaseItemKind.Studio]);
  143. }
  144. /// <inheritdoc />
  145. public QueryResult<(BaseItemDto Item, ItemCounts ItemCounts)> GetGenres(InternalItemsQuery filter)
  146. {
  147. return GetItemValues(filter, _getGenreValueTypes, _itemTypeLookup.BaseItemKindNames[BaseItemKind.Genre]);
  148. }
  149. /// <inheritdoc />
  150. public QueryResult<(BaseItemDto Item, ItemCounts ItemCounts)> GetMusicGenres(InternalItemsQuery filter)
  151. {
  152. return GetItemValues(filter, _getGenreValueTypes, _itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicGenre]);
  153. }
  154. /// <inheritdoc />
  155. public IReadOnlyList<string> GetStudioNames()
  156. {
  157. return GetItemValueNames(_getStudiosValueTypes, [], []);
  158. }
  159. /// <inheritdoc />
  160. public IReadOnlyList<string> GetAllArtistNames()
  161. {
  162. return GetItemValueNames(_getAllArtistsValueTypes, [], []);
  163. }
  164. /// <inheritdoc />
  165. public IReadOnlyList<string> GetMusicGenreNames()
  166. {
  167. return GetItemValueNames(
  168. _getGenreValueTypes,
  169. _itemTypeLookup.MusicGenreTypes,
  170. []);
  171. }
  172. /// <inheritdoc />
  173. public IReadOnlyList<string> GetGenreNames()
  174. {
  175. return GetItemValueNames(
  176. _getGenreValueTypes,
  177. [],
  178. _itemTypeLookup.MusicGenreTypes);
  179. }
  180. /// <inheritdoc />
  181. public QueryResult<BaseItemDto> GetItems(InternalItemsQuery filter)
  182. {
  183. ArgumentNullException.ThrowIfNull(filter);
  184. if (!filter.EnableTotalRecordCount || (!filter.Limit.HasValue && (filter.StartIndex ?? 0) == 0))
  185. {
  186. var returnList = GetItemList(filter);
  187. return new QueryResult<BaseItemDto>(
  188. filter.StartIndex,
  189. returnList.Count,
  190. returnList);
  191. }
  192. PrepareFilterQuery(filter);
  193. var result = new QueryResult<BaseItemDto>();
  194. using var context = _dbProvider.CreateDbContext();
  195. IQueryable<BaseItemEntity> dbQuery = PrepareItemQuery(context, filter);
  196. dbQuery = TranslateQuery(dbQuery, context, filter);
  197. if (filter.EnableTotalRecordCount)
  198. {
  199. result.TotalRecordCount = dbQuery.Count();
  200. }
  201. dbQuery = ApplyGroupingFilter(dbQuery, filter);
  202. dbQuery = ApplyQueryPageing(dbQuery, filter);
  203. result.Items = dbQuery.AsEnumerable().Where(e => e is not null).Select(w => DeserialiseBaseItem(w, filter.SkipDeserialization)).ToArray();
  204. result.StartIndex = filter.StartIndex ?? 0;
  205. return result;
  206. }
  207. /// <inheritdoc />
  208. public IReadOnlyList<BaseItemDto> GetItemList(InternalItemsQuery filter)
  209. {
  210. ArgumentNullException.ThrowIfNull(filter);
  211. PrepareFilterQuery(filter);
  212. using var context = _dbProvider.CreateDbContext();
  213. IQueryable<BaseItemEntity> dbQuery = PrepareItemQuery(context, filter);
  214. dbQuery = TranslateQuery(dbQuery, context, filter);
  215. dbQuery = ApplyGroupingFilter(dbQuery, filter);
  216. dbQuery = ApplyQueryPageing(dbQuery, filter);
  217. return dbQuery.AsEnumerable().Where(e => e is not null).Select(w => DeserialiseBaseItem(w, filter.SkipDeserialization)).ToArray();
  218. }
  219. private IQueryable<BaseItemEntity> ApplyGroupingFilter(IQueryable<BaseItemEntity> dbQuery, InternalItemsQuery filter)
  220. {
  221. // This whole block is needed to filter duplicate entries on request
  222. // for the time beeing it cannot be used because it would destroy the ordering
  223. // this results in "duplicate" responses for queries that try to lookup individual series or multiple versions but
  224. // for that case the invoker has to run a DistinctBy(e => e.PresentationUniqueKey) on their own
  225. // var enableGroupByPresentationUniqueKey = EnableGroupByPresentationUniqueKey(filter);
  226. // if (enableGroupByPresentationUniqueKey && filter.GroupBySeriesPresentationUniqueKey)
  227. // {
  228. // dbQuery = ApplyOrder(dbQuery, filter);
  229. // dbQuery = dbQuery.GroupBy(e => new { e.PresentationUniqueKey, e.SeriesPresentationUniqueKey }).Select(e => e.First());
  230. // }
  231. // else if (enableGroupByPresentationUniqueKey)
  232. // {
  233. // dbQuery = ApplyOrder(dbQuery, filter);
  234. // dbQuery = dbQuery.GroupBy(e => e.PresentationUniqueKey).Select(e => e.First());
  235. // }
  236. // else if (filter.GroupBySeriesPresentationUniqueKey)
  237. // {
  238. // dbQuery = ApplyOrder(dbQuery, filter);
  239. // dbQuery = dbQuery.GroupBy(e => e.SeriesPresentationUniqueKey).Select(e => e.First());
  240. // }
  241. // else
  242. // {
  243. // dbQuery = dbQuery.Distinct();
  244. // dbQuery = ApplyOrder(dbQuery, filter);
  245. // }
  246. dbQuery = dbQuery.Distinct();
  247. dbQuery = ApplyOrder(dbQuery, filter);
  248. return dbQuery;
  249. }
  250. private IQueryable<BaseItemEntity> ApplyQueryPageing(IQueryable<BaseItemEntity> dbQuery, InternalItemsQuery filter)
  251. {
  252. if (filter.Limit.HasValue || filter.StartIndex.HasValue)
  253. {
  254. var offset = filter.StartIndex ?? 0;
  255. if (offset > 0)
  256. {
  257. dbQuery = dbQuery.Skip(offset);
  258. }
  259. if (filter.Limit.HasValue)
  260. {
  261. dbQuery = dbQuery.Take(filter.Limit.Value);
  262. }
  263. }
  264. return dbQuery;
  265. }
  266. private IQueryable<BaseItemEntity> ApplyQueryFilter(IQueryable<BaseItemEntity> dbQuery, JellyfinDbContext context, InternalItemsQuery filter)
  267. {
  268. dbQuery = TranslateQuery(dbQuery, context, filter);
  269. dbQuery = ApplyOrder(dbQuery, filter);
  270. dbQuery = ApplyGroupingFilter(dbQuery, filter);
  271. dbQuery = ApplyQueryPageing(dbQuery, filter);
  272. return dbQuery;
  273. }
  274. private IQueryable<BaseItemEntity> PrepareItemQuery(JellyfinDbContext context, InternalItemsQuery filter)
  275. {
  276. IQueryable<BaseItemEntity> dbQuery = context.BaseItems.AsNoTracking().AsSplitQuery()
  277. .Include(e => e.TrailerTypes)
  278. .Include(e => e.Provider)
  279. .Include(e => e.LockedFields);
  280. if (filter.DtoOptions.EnableImages)
  281. {
  282. dbQuery = dbQuery.Include(e => e.Images);
  283. }
  284. return dbQuery;
  285. }
  286. /// <inheritdoc/>
  287. public int GetCount(InternalItemsQuery filter)
  288. {
  289. ArgumentNullException.ThrowIfNull(filter);
  290. // Hack for right now since we currently don't support filtering out these duplicates within a query
  291. PrepareFilterQuery(filter);
  292. using var context = _dbProvider.CreateDbContext();
  293. var dbQuery = TranslateQuery(context.BaseItems.AsNoTracking(), context, filter);
  294. return dbQuery.Count();
  295. }
  296. #pragma warning disable CA1307 // Specify StringComparison for clarity
  297. /// <summary>
  298. /// Gets the type.
  299. /// </summary>
  300. /// <param name="typeName">Name of the type.</param>
  301. /// <returns>Type.</returns>
  302. /// <exception cref="ArgumentNullException"><c>typeName</c> is null.</exception>
  303. private static Type? GetType(string typeName)
  304. {
  305. ArgumentException.ThrowIfNullOrEmpty(typeName);
  306. // TODO: this isn't great. Refactor later to be both globally handled by a dedicated service not just an static variable and be loaded eagar.
  307. // currently this is done so that plugins may introduce their own type of baseitems as we dont know when we are first called, before or after plugins are loaded
  308. return _typeMap.GetOrAdd(typeName, k => AppDomain.CurrentDomain.GetAssemblies()
  309. .Select(a => a.GetType(k))
  310. .FirstOrDefault(t => t is not null));
  311. }
  312. /// <inheritdoc />
  313. public void SaveImages(BaseItemDto item)
  314. {
  315. ArgumentNullException.ThrowIfNull(item);
  316. var images = item.ImageInfos.Select(e => Map(item.Id, e));
  317. using var context = _dbProvider.CreateDbContext();
  318. using var transaction = context.Database.BeginTransaction();
  319. context.BaseItemImageInfos.Where(e => e.ItemId == item.Id).ExecuteDelete();
  320. context.BaseItemImageInfos.AddRange(images);
  321. context.SaveChanges();
  322. transaction.Commit();
  323. }
  324. /// <inheritdoc />
  325. public void SaveItems(IReadOnlyList<BaseItemDto> items, CancellationToken cancellationToken)
  326. {
  327. UpdateOrInsertItems(items, cancellationToken);
  328. }
  329. /// <inheritdoc cref="IItemRepository"/>
  330. public void UpdateOrInsertItems(IReadOnlyList<BaseItemDto> items, CancellationToken cancellationToken)
  331. {
  332. ArgumentNullException.ThrowIfNull(items);
  333. cancellationToken.ThrowIfCancellationRequested();
  334. var tuples = new List<(BaseItemDto Item, List<Guid>? AncestorIds, BaseItemDto TopParent, IEnumerable<string> UserDataKey, List<string> InheritedTags)>();
  335. foreach (var item in items.GroupBy(e => e.Id).Select(e => e.Last()))
  336. {
  337. var ancestorIds = item.SupportsAncestors ?
  338. item.GetAncestorIds().Distinct().ToList() :
  339. null;
  340. var topParent = item.GetTopParent();
  341. var userdataKey = item.GetUserDataKeys();
  342. var inheritedTags = item.GetInheritedTags();
  343. tuples.Add((item, ancestorIds, topParent, userdataKey, inheritedTags));
  344. }
  345. var localItemValueCache = new Dictionary<(int MagicNumber, string Value), Guid>();
  346. using var context = _dbProvider.CreateDbContext();
  347. using var transaction = context.Database.BeginTransaction();
  348. foreach (var item in tuples)
  349. {
  350. var entity = Map(item.Item);
  351. // TODO: refactor this "inconsistency"
  352. entity.TopParentId = item.TopParent?.Id;
  353. if (!context.BaseItems.Any(e => e.Id == entity.Id))
  354. {
  355. context.BaseItems.Add(entity);
  356. }
  357. else
  358. {
  359. context.BaseItemProviders.Where(e => e.ItemId == entity.Id).ExecuteDelete();
  360. context.BaseItems.Attach(entity).State = EntityState.Modified;
  361. }
  362. context.AncestorIds.Where(e => e.ItemId == entity.Id).ExecuteDelete();
  363. if (item.Item.SupportsAncestors && item.AncestorIds != null)
  364. {
  365. foreach (var ancestorId in item.AncestorIds)
  366. {
  367. if (!context.BaseItems.Any(f => f.Id == ancestorId))
  368. {
  369. continue;
  370. }
  371. context.AncestorIds.Add(new AncestorId()
  372. {
  373. ParentItemId = ancestorId,
  374. ItemId = entity.Id,
  375. Item = null!,
  376. ParentItem = null!
  377. });
  378. }
  379. }
  380. // Never save duplicate itemValues as they are now mapped anyway.
  381. var itemValuesToSave = GetItemValuesToSave(item.Item, item.InheritedTags).DistinctBy(e => (GetCleanValue(e.Value), e.MagicNumber));
  382. context.ItemValuesMap.Where(e => e.ItemId == entity.Id).ExecuteDelete();
  383. foreach (var itemValue in itemValuesToSave)
  384. {
  385. if (!localItemValueCache.TryGetValue(itemValue, out var refValue))
  386. {
  387. refValue = context.ItemValues
  388. .Where(f => f.CleanValue == GetCleanValue(itemValue.Value) && (int)f.Type == itemValue.MagicNumber)
  389. .Select(e => e.ItemValueId)
  390. .FirstOrDefault();
  391. }
  392. if (refValue.IsEmpty())
  393. {
  394. context.ItemValues.Add(new ItemValue()
  395. {
  396. CleanValue = GetCleanValue(itemValue.Value),
  397. Type = (ItemValueType)itemValue.MagicNumber,
  398. ItemValueId = refValue = Guid.NewGuid(),
  399. Value = itemValue.Value
  400. });
  401. localItemValueCache[itemValue] = refValue;
  402. }
  403. context.ItemValuesMap.Add(new ItemValueMap()
  404. {
  405. Item = null!,
  406. ItemId = entity.Id,
  407. ItemValue = null!,
  408. ItemValueId = refValue
  409. });
  410. }
  411. }
  412. context.SaveChanges();
  413. transaction.Commit();
  414. }
  415. /// <inheritdoc />
  416. public BaseItemDto? RetrieveItem(Guid id)
  417. {
  418. if (id.IsEmpty())
  419. {
  420. throw new ArgumentException("Guid can't be empty", nameof(id));
  421. }
  422. using var context = _dbProvider.CreateDbContext();
  423. var item = PrepareItemQuery(context, new()
  424. {
  425. DtoOptions = new()
  426. {
  427. EnableImages = true
  428. }
  429. }).FirstOrDefault(e => e.Id == id);
  430. if (item is null)
  431. {
  432. return null;
  433. }
  434. return DeserialiseBaseItem(item);
  435. }
  436. /// <summary>
  437. /// Maps a Entity to the DTO.
  438. /// </summary>
  439. /// <param name="entity">The entity.</param>
  440. /// <param name="dto">The dto base instance.</param>
  441. /// <param name="appHost">The Application server Host.</param>
  442. /// <returns>The dto to map.</returns>
  443. public static BaseItemDto Map(BaseItemEntity entity, BaseItemDto dto, IServerApplicationHost? appHost)
  444. {
  445. dto.Id = entity.Id;
  446. dto.ParentId = entity.ParentId.GetValueOrDefault();
  447. dto.Path = appHost?.ExpandVirtualPath(entity.Path) ?? entity.Path;
  448. dto.EndDate = entity.EndDate;
  449. dto.CommunityRating = entity.CommunityRating;
  450. dto.CustomRating = entity.CustomRating;
  451. dto.IndexNumber = entity.IndexNumber;
  452. dto.IsLocked = entity.IsLocked;
  453. dto.Name = entity.Name;
  454. dto.OfficialRating = entity.OfficialRating;
  455. dto.Overview = entity.Overview;
  456. dto.ParentIndexNumber = entity.ParentIndexNumber;
  457. dto.PremiereDate = entity.PremiereDate;
  458. dto.ProductionYear = entity.ProductionYear;
  459. dto.SortName = entity.SortName;
  460. dto.ForcedSortName = entity.ForcedSortName;
  461. dto.RunTimeTicks = entity.RunTimeTicks;
  462. dto.PreferredMetadataLanguage = entity.PreferredMetadataLanguage;
  463. dto.PreferredMetadataCountryCode = entity.PreferredMetadataCountryCode;
  464. dto.IsInMixedFolder = entity.IsInMixedFolder;
  465. dto.InheritedParentalRatingValue = entity.InheritedParentalRatingValue;
  466. dto.CriticRating = entity.CriticRating;
  467. dto.PresentationUniqueKey = entity.PresentationUniqueKey;
  468. dto.OriginalTitle = entity.OriginalTitle;
  469. dto.Album = entity.Album;
  470. dto.LUFS = entity.LUFS;
  471. dto.NormalizationGain = entity.NormalizationGain;
  472. dto.IsVirtualItem = entity.IsVirtualItem;
  473. dto.ExternalSeriesId = entity.ExternalSeriesId;
  474. dto.Tagline = entity.Tagline;
  475. dto.TotalBitrate = entity.TotalBitrate;
  476. dto.ExternalId = entity.ExternalId;
  477. dto.Size = entity.Size;
  478. dto.Genres = entity.Genres?.Split('|') ?? [];
  479. dto.DateCreated = entity.DateCreated.GetValueOrDefault();
  480. dto.DateModified = entity.DateModified.GetValueOrDefault();
  481. dto.ChannelId = string.IsNullOrWhiteSpace(entity.ChannelId) ? Guid.Empty : (Guid.TryParse(entity.ChannelId, out var channelId) ? channelId : Guid.Empty);
  482. dto.DateLastRefreshed = entity.DateLastRefreshed.GetValueOrDefault();
  483. dto.DateLastSaved = entity.DateLastSaved.GetValueOrDefault();
  484. dto.OwnerId = string.IsNullOrWhiteSpace(entity.OwnerId) ? Guid.Empty : (Guid.TryParse(entity.OwnerId, out var ownerId) ? ownerId : Guid.Empty);
  485. dto.Width = entity.Width.GetValueOrDefault();
  486. dto.Height = entity.Height.GetValueOrDefault();
  487. if (entity.Provider is not null)
  488. {
  489. dto.ProviderIds = entity.Provider.ToDictionary(e => e.ProviderId, e => e.ProviderValue);
  490. }
  491. if (entity.ExtraType is not null)
  492. {
  493. dto.ExtraType = (ExtraType)entity.ExtraType;
  494. }
  495. if (entity.LockedFields is not null)
  496. {
  497. dto.LockedFields = entity.LockedFields?.Select(e => (MetadataField)e.Id).ToArray() ?? [];
  498. }
  499. if (entity.Audio is not null)
  500. {
  501. dto.Audio = (ProgramAudio)entity.Audio;
  502. }
  503. dto.ExtraIds = string.IsNullOrWhiteSpace(entity.ExtraIds) ? [] : entity.ExtraIds.Split('|').Select(e => Guid.Parse(e)).ToArray();
  504. dto.ProductionLocations = entity.ProductionLocations?.Split('|') ?? [];
  505. dto.Studios = entity.Studios?.Split('|') ?? [];
  506. dto.Tags = entity.Tags?.Split('|') ?? [];
  507. if (dto is IHasProgramAttributes hasProgramAttributes)
  508. {
  509. hasProgramAttributes.IsMovie = entity.IsMovie;
  510. hasProgramAttributes.IsSeries = entity.IsSeries;
  511. hasProgramAttributes.EpisodeTitle = entity.EpisodeTitle;
  512. hasProgramAttributes.IsRepeat = entity.IsRepeat;
  513. }
  514. if (dto is LiveTvChannel liveTvChannel)
  515. {
  516. liveTvChannel.ServiceName = entity.ExternalServiceId;
  517. }
  518. if (dto is Trailer trailer)
  519. {
  520. trailer.TrailerTypes = entity.TrailerTypes?.Select(e => (TrailerType)e.Id).ToArray() ?? [];
  521. }
  522. if (dto is Video video)
  523. {
  524. video.PrimaryVersionId = entity.PrimaryVersionId;
  525. }
  526. if (dto is IHasSeries hasSeriesName)
  527. {
  528. hasSeriesName.SeriesName = entity.SeriesName;
  529. hasSeriesName.SeriesId = entity.SeriesId.GetValueOrDefault();
  530. hasSeriesName.SeriesPresentationUniqueKey = entity.SeriesPresentationUniqueKey;
  531. }
  532. if (dto is Episode episode)
  533. {
  534. episode.SeasonName = entity.SeasonName;
  535. episode.SeasonId = entity.SeasonId.GetValueOrDefault();
  536. }
  537. if (dto is IHasArtist hasArtists)
  538. {
  539. hasArtists.Artists = entity.Artists?.Split('|', StringSplitOptions.RemoveEmptyEntries) ?? [];
  540. }
  541. if (dto is IHasAlbumArtist hasAlbumArtists)
  542. {
  543. hasAlbumArtists.AlbumArtists = entity.AlbumArtists?.Split('|', StringSplitOptions.RemoveEmptyEntries) ?? [];
  544. }
  545. if (dto is LiveTvProgram program)
  546. {
  547. program.ShowId = entity.ShowId;
  548. }
  549. if (entity.Images is not null)
  550. {
  551. dto.ImageInfos = entity.Images.Select(e => Map(e, appHost)).ToArray();
  552. }
  553. // dto.Type = entity.Type;
  554. // dto.Data = entity.Data;
  555. // dto.MediaType = Enum.TryParse<MediaType>(entity.MediaType);
  556. if (dto is IHasStartDate hasStartDate)
  557. {
  558. hasStartDate.StartDate = entity.StartDate;
  559. }
  560. // Fields that are present in the DB but are never actually used
  561. // dto.UnratedType = entity.UnratedType;
  562. // dto.TopParentId = entity.TopParentId;
  563. // dto.CleanName = entity.CleanName;
  564. // dto.UserDataKey = entity.UserDataKey;
  565. if (dto is Folder folder)
  566. {
  567. folder.DateLastMediaAdded = entity.DateLastMediaAdded;
  568. }
  569. return dto;
  570. }
  571. /// <summary>
  572. /// Maps a Entity to the DTO.
  573. /// </summary>
  574. /// <param name="dto">The entity.</param>
  575. /// <returns>The dto to map.</returns>
  576. public BaseItemEntity Map(BaseItemDto dto)
  577. {
  578. var dtoType = dto.GetType();
  579. var entity = new BaseItemEntity()
  580. {
  581. Type = dtoType.ToString(),
  582. Id = dto.Id
  583. };
  584. if (TypeRequiresDeserialization(dtoType))
  585. {
  586. entity.Data = JsonSerializer.Serialize(dto, dtoType, JsonDefaults.Options);
  587. }
  588. entity.ParentId = !dto.ParentId.IsEmpty() ? dto.ParentId : null;
  589. entity.Path = GetPathToSave(dto.Path);
  590. entity.EndDate = dto.EndDate.GetValueOrDefault();
  591. entity.CommunityRating = dto.CommunityRating;
  592. entity.CustomRating = dto.CustomRating;
  593. entity.IndexNumber = dto.IndexNumber;
  594. entity.IsLocked = dto.IsLocked;
  595. entity.Name = dto.Name;
  596. entity.OfficialRating = dto.OfficialRating;
  597. entity.Overview = dto.Overview;
  598. entity.ParentIndexNumber = dto.ParentIndexNumber;
  599. entity.PremiereDate = dto.PremiereDate;
  600. entity.ProductionYear = dto.ProductionYear;
  601. entity.SortName = dto.SortName;
  602. entity.ForcedSortName = dto.ForcedSortName;
  603. entity.RunTimeTicks = dto.RunTimeTicks;
  604. entity.PreferredMetadataLanguage = dto.PreferredMetadataLanguage;
  605. entity.PreferredMetadataCountryCode = dto.PreferredMetadataCountryCode;
  606. entity.IsInMixedFolder = dto.IsInMixedFolder;
  607. entity.InheritedParentalRatingValue = dto.InheritedParentalRatingValue;
  608. entity.CriticRating = dto.CriticRating;
  609. entity.PresentationUniqueKey = dto.PresentationUniqueKey;
  610. entity.OriginalTitle = dto.OriginalTitle;
  611. entity.Album = dto.Album;
  612. entity.LUFS = dto.LUFS;
  613. entity.NormalizationGain = dto.NormalizationGain;
  614. entity.IsVirtualItem = dto.IsVirtualItem;
  615. entity.ExternalSeriesId = dto.ExternalSeriesId;
  616. entity.Tagline = dto.Tagline;
  617. entity.TotalBitrate = dto.TotalBitrate;
  618. entity.ExternalId = dto.ExternalId;
  619. entity.Size = dto.Size;
  620. entity.Genres = string.Join('|', dto.Genres);
  621. entity.DateCreated = dto.DateCreated;
  622. entity.DateModified = dto.DateModified;
  623. entity.ChannelId = dto.ChannelId.ToString();
  624. entity.DateLastRefreshed = dto.DateLastRefreshed;
  625. entity.DateLastSaved = dto.DateLastSaved;
  626. entity.OwnerId = dto.OwnerId.ToString();
  627. entity.Width = dto.Width;
  628. entity.Height = dto.Height;
  629. entity.Provider = dto.ProviderIds.Select(e => new BaseItemProvider()
  630. {
  631. Item = entity,
  632. ProviderId = e.Key,
  633. ProviderValue = e.Value
  634. }).ToList();
  635. if (dto.Audio.HasValue)
  636. {
  637. entity.Audio = (ProgramAudioEntity)dto.Audio;
  638. }
  639. if (dto.ExtraType.HasValue)
  640. {
  641. entity.ExtraType = (BaseItemExtraType)dto.ExtraType;
  642. }
  643. entity.ExtraIds = dto.ExtraIds is not null ? string.Join('|', dto.ExtraIds) : null;
  644. entity.ProductionLocations = dto.ProductionLocations is not null ? string.Join('|', dto.ProductionLocations) : null;
  645. entity.Studios = dto.Studios is not null ? string.Join('|', dto.Studios) : null;
  646. entity.Tags = dto.Tags is not null ? string.Join('|', dto.Tags) : null;
  647. entity.LockedFields = dto.LockedFields is not null ? dto.LockedFields
  648. .Select(e => new BaseItemMetadataField()
  649. {
  650. Id = (int)e,
  651. Item = entity,
  652. ItemId = entity.Id
  653. })
  654. .ToArray() : null;
  655. if (dto is IHasProgramAttributes hasProgramAttributes)
  656. {
  657. entity.IsMovie = hasProgramAttributes.IsMovie;
  658. entity.IsSeries = hasProgramAttributes.IsSeries;
  659. entity.EpisodeTitle = hasProgramAttributes.EpisodeTitle;
  660. entity.IsRepeat = hasProgramAttributes.IsRepeat;
  661. }
  662. if (dto is LiveTvChannel liveTvChannel)
  663. {
  664. entity.ExternalServiceId = liveTvChannel.ServiceName;
  665. }
  666. if (dto is Video video)
  667. {
  668. entity.PrimaryVersionId = video.PrimaryVersionId;
  669. }
  670. if (dto is IHasSeries hasSeriesName)
  671. {
  672. entity.SeriesName = hasSeriesName.SeriesName;
  673. entity.SeriesId = hasSeriesName.SeriesId;
  674. entity.SeriesPresentationUniqueKey = hasSeriesName.SeriesPresentationUniqueKey;
  675. }
  676. if (dto is Episode episode)
  677. {
  678. entity.SeasonName = episode.SeasonName;
  679. entity.SeasonId = episode.SeasonId;
  680. }
  681. if (dto is IHasArtist hasArtists)
  682. {
  683. entity.Artists = hasArtists.Artists is not null ? string.Join('|', hasArtists.Artists) : null;
  684. }
  685. if (dto is IHasAlbumArtist hasAlbumArtists)
  686. {
  687. entity.AlbumArtists = hasAlbumArtists.AlbumArtists is not null ? string.Join('|', hasAlbumArtists.AlbumArtists) : null;
  688. }
  689. if (dto is LiveTvProgram program)
  690. {
  691. entity.ShowId = program.ShowId;
  692. }
  693. if (dto.ImageInfos is not null)
  694. {
  695. entity.Images = dto.ImageInfos.Select(f => Map(dto.Id, f)).ToArray();
  696. }
  697. if (dto is Trailer trailer)
  698. {
  699. entity.TrailerTypes = trailer.TrailerTypes?.Select(e => new BaseItemTrailerType()
  700. {
  701. Id = (int)e,
  702. Item = entity,
  703. ItemId = entity.Id
  704. }).ToArray() ?? [];
  705. }
  706. // dto.Type = entity.Type;
  707. // dto.Data = entity.Data;
  708. entity.MediaType = dto.MediaType.ToString();
  709. if (dto is IHasStartDate hasStartDate)
  710. {
  711. entity.StartDate = hasStartDate.StartDate;
  712. }
  713. // Fields that are present in the DB but are never actually used
  714. // dto.UnratedType = entity.UnratedType;
  715. // dto.TopParentId = entity.TopParentId;
  716. // dto.CleanName = entity.CleanName;
  717. // dto.UserDataKey = entity.UserDataKey;
  718. if (dto is Folder folder)
  719. {
  720. entity.DateLastMediaAdded = folder.DateLastMediaAdded;
  721. entity.IsFolder = folder.IsFolder;
  722. }
  723. return entity;
  724. }
  725. private string[] GetItemValueNames(IReadOnlyList<ItemValueType> itemValueTypes, IReadOnlyList<string> withItemTypes, IReadOnlyList<string> excludeItemTypes)
  726. {
  727. using var context = _dbProvider.CreateDbContext();
  728. var query = context.ItemValuesMap
  729. .AsNoTracking()
  730. .Where(e => itemValueTypes.Any(w => (ItemValueType)w == e.ItemValue.Type));
  731. if (withItemTypes.Count > 0)
  732. {
  733. query = query.Where(e => withItemTypes.Contains(e.Item.Type));
  734. }
  735. if (excludeItemTypes.Count > 0)
  736. {
  737. query = query.Where(e => !excludeItemTypes.Contains(e.Item.Type));
  738. }
  739. // query = query.DistinctBy(e => e.CleanValue);
  740. return query.Select(e => e.ItemValue.CleanValue).ToArray();
  741. }
  742. private static bool TypeRequiresDeserialization(Type type)
  743. {
  744. return type.GetCustomAttribute<RequiresSourceSerialisationAttribute>() == null;
  745. }
  746. private BaseItemDto DeserialiseBaseItem(BaseItemEntity baseItemEntity, bool skipDeserialization = false)
  747. {
  748. ArgumentNullException.ThrowIfNull(baseItemEntity, nameof(baseItemEntity));
  749. if (_serverConfigurationManager?.Configuration is null)
  750. {
  751. throw new InvalidOperationException("Server Configuration manager or configuration is null");
  752. }
  753. var typeToSerialise = GetType(baseItemEntity.Type);
  754. return BaseItemRepository.DeserialiseBaseItem(
  755. baseItemEntity,
  756. _logger,
  757. _appHost,
  758. skipDeserialization || (_serverConfigurationManager.Configuration.SkipDeserializationForBasicTypes && (typeToSerialise == typeof(Channel) || typeToSerialise == typeof(UserRootFolder))));
  759. }
  760. /// <summary>
  761. /// Deserialises a BaseItemEntity and sets all properties.
  762. /// </summary>
  763. /// <param name="baseItemEntity">The DB entity.</param>
  764. /// <param name="logger">Logger.</param>
  765. /// <param name="appHost">The application server Host.</param>
  766. /// <param name="skipDeserialization">If only mapping should be processed.</param>
  767. /// <returns>A mapped BaseItem.</returns>
  768. /// <exception cref="InvalidOperationException">Will be thrown if an invalid serialisation is requested.</exception>
  769. public static BaseItemDto DeserialiseBaseItem(BaseItemEntity baseItemEntity, ILogger logger, IServerApplicationHost? appHost, bool skipDeserialization = false)
  770. {
  771. var type = GetType(baseItemEntity.Type) ?? throw new InvalidOperationException("Cannot deserialise unkown type.");
  772. BaseItemDto? dto = null;
  773. if (TypeRequiresDeserialization(type) && baseItemEntity.Data is not null && !skipDeserialization)
  774. {
  775. try
  776. {
  777. dto = JsonSerializer.Deserialize(baseItemEntity.Data, type, JsonDefaults.Options) as BaseItemDto;
  778. }
  779. catch (JsonException ex)
  780. {
  781. logger.LogError(ex, "Error deserializing item with JSON: {Data}", baseItemEntity.Data);
  782. }
  783. }
  784. if (dto is null)
  785. {
  786. dto = Activator.CreateInstance(type) as BaseItemDto ?? throw new InvalidOperationException("Cannot deserialise unkown type.");
  787. }
  788. return Map(baseItemEntity, dto, appHost);
  789. }
  790. private QueryResult<(BaseItemDto Item, ItemCounts ItemCounts)> GetItemValues(InternalItemsQuery filter, IReadOnlyList<ItemValueType> itemValueTypes, string returnType)
  791. {
  792. ArgumentNullException.ThrowIfNull(filter);
  793. if (!filter.Limit.HasValue)
  794. {
  795. filter.EnableTotalRecordCount = false;
  796. }
  797. using var context = _dbProvider.CreateDbContext();
  798. var innerQuery = new InternalItemsQuery(filter.User)
  799. {
  800. ExcludeItemTypes = filter.ExcludeItemTypes,
  801. IncludeItemTypes = filter.IncludeItemTypes,
  802. MediaTypes = filter.MediaTypes,
  803. AncestorIds = filter.AncestorIds,
  804. ItemIds = filter.ItemIds,
  805. TopParentIds = filter.TopParentIds,
  806. ParentId = filter.ParentId,
  807. IsAiring = filter.IsAiring,
  808. IsMovie = filter.IsMovie,
  809. IsSports = filter.IsSports,
  810. IsKids = filter.IsKids,
  811. IsNews = filter.IsNews,
  812. IsSeries = filter.IsSeries
  813. };
  814. var query = TranslateQuery(context.BaseItems.AsNoTracking(), context, innerQuery);
  815. query = query.Where(e => e.Type == returnType && e.ItemValues!.Any(f => e.CleanName == f.ItemValue.CleanValue && itemValueTypes.Any(w => (ItemValueType)w == f.ItemValue.Type)));
  816. if (filter.OrderBy.Count != 0
  817. || !string.IsNullOrEmpty(filter.SearchTerm))
  818. {
  819. query = ApplyOrder(query, filter);
  820. }
  821. else
  822. {
  823. query = query.OrderBy(e => e.SortName);
  824. }
  825. if (filter.Limit.HasValue || filter.StartIndex.HasValue)
  826. {
  827. var offset = filter.StartIndex ?? 0;
  828. if (offset > 0)
  829. {
  830. query = query.Skip(offset);
  831. }
  832. if (filter.Limit.HasValue)
  833. {
  834. query = query.Take(filter.Limit.Value);
  835. }
  836. }
  837. var result = new QueryResult<(BaseItemDto, ItemCounts)>();
  838. if (filter.EnableTotalRecordCount)
  839. {
  840. result.TotalRecordCount = query.GroupBy(e => e.PresentationUniqueKey).Select(e => e.First()).Count();
  841. }
  842. var seriesTypeName = _itemTypeLookup.BaseItemKindNames[BaseItemKind.Series];
  843. var movieTypeName = _itemTypeLookup.BaseItemKindNames[BaseItemKind.Movie];
  844. var episodeTypeName = _itemTypeLookup.BaseItemKindNames[BaseItemKind.Episode];
  845. var musicAlbumTypeName = _itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicAlbum];
  846. var musicArtistTypeName = _itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicArtist];
  847. var audioTypeName = _itemTypeLookup.BaseItemKindNames[BaseItemKind.Audio];
  848. var trailerTypeName = _itemTypeLookup.BaseItemKindNames[BaseItemKind.Trailer];
  849. var resultQuery = query.Select(e => new
  850. {
  851. item = e,
  852. // TODO: This is bad refactor!
  853. itemCount = new ItemCounts()
  854. {
  855. SeriesCount = e.ItemValues!.Count(f => f.Item.Type == seriesTypeName),
  856. EpisodeCount = e.ItemValues!.Count(f => f.Item.Type == episodeTypeName),
  857. MovieCount = e.ItemValues!.Count(f => f.Item.Type == movieTypeName),
  858. AlbumCount = e.ItemValues!.Count(f => f.Item.Type == musicAlbumTypeName),
  859. ArtistCount = e.ItemValues!.Count(f => f.Item.Type == musicArtistTypeName),
  860. SongCount = e.ItemValues!.Count(f => f.Item.Type == audioTypeName),
  861. TrailerCount = e.ItemValues!.Count(f => f.Item.Type == trailerTypeName),
  862. }
  863. });
  864. result.StartIndex = filter.StartIndex ?? 0;
  865. result.Items = resultQuery.ToArray().Where(e => e is not null).Select(e =>
  866. {
  867. return (DeserialiseBaseItem(e.item, filter.SkipDeserialization), e.itemCount);
  868. }).ToArray();
  869. return result;
  870. }
  871. private static void PrepareFilterQuery(InternalItemsQuery query)
  872. {
  873. if (query.Limit.HasValue && query.EnableGroupByMetadataKey)
  874. {
  875. query.Limit = query.Limit.Value + 4;
  876. }
  877. if (query.IsResumable ?? false)
  878. {
  879. query.IsVirtualItem = false;
  880. }
  881. }
  882. private string GetCleanValue(string value)
  883. {
  884. if (string.IsNullOrWhiteSpace(value))
  885. {
  886. return value;
  887. }
  888. return value.RemoveDiacritics().ToLowerInvariant();
  889. }
  890. private List<(int MagicNumber, string Value)> GetItemValuesToSave(BaseItemDto item, List<string> inheritedTags)
  891. {
  892. var list = new List<(int, string)>();
  893. if (item is IHasArtist hasArtist)
  894. {
  895. list.AddRange(hasArtist.Artists.Select(i => (0, i)));
  896. }
  897. if (item is IHasAlbumArtist hasAlbumArtist)
  898. {
  899. list.AddRange(hasAlbumArtist.AlbumArtists.Select(i => (1, i)));
  900. }
  901. list.AddRange(item.Genres.Select(i => (2, i)));
  902. list.AddRange(item.Studios.Select(i => (3, i)));
  903. list.AddRange(item.Tags.Select(i => (4, i)));
  904. // keywords was 5
  905. list.AddRange(inheritedTags.Select(i => (6, i)));
  906. // Remove all invalid values.
  907. list.RemoveAll(i => string.IsNullOrWhiteSpace(i.Item2));
  908. return list;
  909. }
  910. private static BaseItemImageInfo Map(Guid baseItemId, ItemImageInfo e)
  911. {
  912. return new BaseItemImageInfo()
  913. {
  914. ItemId = baseItemId,
  915. Id = Guid.NewGuid(),
  916. Path = e.Path,
  917. Blurhash = e.BlurHash is null ? null : Encoding.UTF8.GetBytes(e.BlurHash),
  918. DateModified = e.DateModified,
  919. Height = e.Height,
  920. Width = e.Width,
  921. ImageType = (ImageInfoImageType)e.Type,
  922. Item = null!
  923. };
  924. }
  925. private static ItemImageInfo Map(BaseItemImageInfo e, IServerApplicationHost? appHost)
  926. {
  927. return new ItemImageInfo()
  928. {
  929. Path = appHost?.ExpandVirtualPath(e.Path) ?? e.Path,
  930. BlurHash = e.Blurhash is null ? null : Encoding.UTF8.GetString(e.Blurhash),
  931. DateModified = e.DateModified,
  932. Height = e.Height,
  933. Width = e.Width,
  934. Type = (ImageType)e.ImageType
  935. };
  936. }
  937. private string? GetPathToSave(string path)
  938. {
  939. if (path is null)
  940. {
  941. return null;
  942. }
  943. return _appHost.ReverseVirtualPath(path);
  944. }
  945. private List<string> GetItemByNameTypesInQuery(InternalItemsQuery query)
  946. {
  947. var list = new List<string>();
  948. if (IsTypeInQuery(BaseItemKind.Person, query))
  949. {
  950. list.Add(_itemTypeLookup.BaseItemKindNames[BaseItemKind.Person]!);
  951. }
  952. if (IsTypeInQuery(BaseItemKind.Genre, query))
  953. {
  954. list.Add(_itemTypeLookup.BaseItemKindNames[BaseItemKind.Genre]!);
  955. }
  956. if (IsTypeInQuery(BaseItemKind.MusicGenre, query))
  957. {
  958. list.Add(_itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicGenre]!);
  959. }
  960. if (IsTypeInQuery(BaseItemKind.MusicArtist, query))
  961. {
  962. list.Add(_itemTypeLookup.BaseItemKindNames[BaseItemKind.MusicArtist]!);
  963. }
  964. if (IsTypeInQuery(BaseItemKind.Studio, query))
  965. {
  966. list.Add(_itemTypeLookup.BaseItemKindNames[BaseItemKind.Studio]!);
  967. }
  968. return list;
  969. }
  970. private bool IsTypeInQuery(BaseItemKind type, InternalItemsQuery query)
  971. {
  972. if (query.ExcludeItemTypes.Contains(type))
  973. {
  974. return false;
  975. }
  976. return query.IncludeItemTypes.Length == 0 || query.IncludeItemTypes.Contains(type);
  977. }
  978. private Expression<Func<BaseItemEntity, object>> MapOrderByField(ItemSortBy sortBy, InternalItemsQuery query)
  979. {
  980. #pragma warning disable CS8603 // Possible null reference return.
  981. return sortBy switch
  982. {
  983. ItemSortBy.AirTime => e => e.SortName, // TODO
  984. ItemSortBy.Runtime => e => e.RunTimeTicks,
  985. ItemSortBy.Random => e => EF.Functions.Random(),
  986. ItemSortBy.DatePlayed => e => e.UserData!.FirstOrDefault(f => f.UserId == query.User!.Id)!.LastPlayedDate,
  987. ItemSortBy.PlayCount => e => e.UserData!.FirstOrDefault(f => f.UserId == query.User!.Id)!.PlayCount,
  988. ItemSortBy.IsFavoriteOrLiked => e => e.UserData!.FirstOrDefault(f => f.UserId == query.User!.Id)!.IsFavorite,
  989. ItemSortBy.IsFolder => e => e.IsFolder,
  990. ItemSortBy.IsPlayed => e => e.UserData!.FirstOrDefault(f => f.UserId == query.User!.Id)!.Played,
  991. ItemSortBy.IsUnplayed => e => !e.UserData!.FirstOrDefault(f => f.UserId == query.User!.Id)!.Played,
  992. ItemSortBy.DateLastContentAdded => e => e.DateLastMediaAdded,
  993. ItemSortBy.Artist => e => e.ItemValues!.Where(f => f.ItemValue.Type == ItemValueType.Artist).Select(f => f.ItemValue.CleanValue),
  994. ItemSortBy.AlbumArtist => e => e.ItemValues!.Where(f => f.ItemValue.Type == ItemValueType.AlbumArtist).Select(f => f.ItemValue.CleanValue),
  995. ItemSortBy.Studio => e => e.ItemValues!.Where(f => f.ItemValue.Type == ItemValueType.Studios).Select(f => f.ItemValue.CleanValue),
  996. ItemSortBy.OfficialRating => e => e.InheritedParentalRatingValue,
  997. // ItemSortBy.SeriesDatePlayed => "(Select MAX(LastPlayedDate) from TypedBaseItems B" + GetJoinUserDataText(query) + " where Played=1 and B.SeriesPresentationUniqueKey=A.PresentationUniqueKey)",
  998. ItemSortBy.SeriesSortName => e => e.SeriesName,
  999. // ItemSortBy.AiredEpisodeOrder => "AiredEpisodeOrder",
  1000. ItemSortBy.Album => e => e.Album,
  1001. ItemSortBy.DateCreated => e => e.DateCreated,
  1002. ItemSortBy.PremiereDate => e => e.PremiereDate,
  1003. ItemSortBy.StartDate => e => e.StartDate,
  1004. ItemSortBy.Name => e => e.Name,
  1005. ItemSortBy.CommunityRating => e => e.CommunityRating,
  1006. ItemSortBy.ProductionYear => e => e.ProductionYear,
  1007. ItemSortBy.CriticRating => e => e.CriticRating,
  1008. ItemSortBy.VideoBitRate => e => e.TotalBitrate,
  1009. ItemSortBy.ParentIndexNumber => e => e.ParentIndexNumber,
  1010. ItemSortBy.IndexNumber => e => e.IndexNumber,
  1011. _ => e => e.SortName
  1012. };
  1013. #pragma warning restore CS8603 // Possible null reference return.
  1014. }
  1015. private bool EnableGroupByPresentationUniqueKey(InternalItemsQuery query)
  1016. {
  1017. if (!query.GroupByPresentationUniqueKey)
  1018. {
  1019. return false;
  1020. }
  1021. if (query.GroupBySeriesPresentationUniqueKey)
  1022. {
  1023. return false;
  1024. }
  1025. if (!string.IsNullOrWhiteSpace(query.PresentationUniqueKey))
  1026. {
  1027. return false;
  1028. }
  1029. if (query.User is null)
  1030. {
  1031. return false;
  1032. }
  1033. if (query.IncludeItemTypes.Length == 0)
  1034. {
  1035. return true;
  1036. }
  1037. return query.IncludeItemTypes.Contains(BaseItemKind.Episode)
  1038. || query.IncludeItemTypes.Contains(BaseItemKind.Video)
  1039. || query.IncludeItemTypes.Contains(BaseItemKind.Movie)
  1040. || query.IncludeItemTypes.Contains(BaseItemKind.MusicVideo)
  1041. || query.IncludeItemTypes.Contains(BaseItemKind.Series)
  1042. || query.IncludeItemTypes.Contains(BaseItemKind.Season);
  1043. }
  1044. private IQueryable<BaseItemEntity> ApplyOrder(IQueryable<BaseItemEntity> query, InternalItemsQuery filter)
  1045. {
  1046. var orderBy = filter.OrderBy;
  1047. var hasSearch = !string.IsNullOrEmpty(filter.SearchTerm);
  1048. if (hasSearch)
  1049. {
  1050. orderBy = filter.OrderBy = [(ItemSortBy.SortName, SortOrder.Ascending), .. orderBy];
  1051. }
  1052. else if (orderBy.Count == 0)
  1053. {
  1054. return query;
  1055. }
  1056. IOrderedQueryable<BaseItemEntity>? orderedQuery = null;
  1057. var firstOrdering = orderBy.FirstOrDefault();
  1058. if (firstOrdering != default)
  1059. {
  1060. var expression = MapOrderByField(firstOrdering.OrderBy, filter);
  1061. if (firstOrdering.SortOrder == SortOrder.Ascending)
  1062. {
  1063. orderedQuery = query.OrderBy(expression);
  1064. }
  1065. else
  1066. {
  1067. orderedQuery = query.OrderByDescending(expression);
  1068. }
  1069. if (firstOrdering.OrderBy is ItemSortBy.Default or ItemSortBy.SortName)
  1070. {
  1071. if (firstOrdering.SortOrder is SortOrder.Ascending)
  1072. {
  1073. orderedQuery = orderedQuery.ThenBy(e => e.Name);
  1074. }
  1075. else
  1076. {
  1077. orderedQuery = orderedQuery.ThenByDescending(e => e.Name);
  1078. }
  1079. }
  1080. }
  1081. foreach (var item in orderBy.Skip(1))
  1082. {
  1083. var expression = MapOrderByField(item.OrderBy, filter);
  1084. if (item.SortOrder == SortOrder.Ascending)
  1085. {
  1086. orderedQuery = orderedQuery!.ThenBy(expression);
  1087. }
  1088. else
  1089. {
  1090. orderedQuery = orderedQuery!.ThenByDescending(expression);
  1091. }
  1092. }
  1093. return orderedQuery ?? query;
  1094. }
  1095. private IQueryable<BaseItemEntity> TranslateQuery(
  1096. IQueryable<BaseItemEntity> baseQuery,
  1097. JellyfinDbContext context,
  1098. InternalItemsQuery filter)
  1099. {
  1100. var minWidth = filter.MinWidth;
  1101. var maxWidth = filter.MaxWidth;
  1102. var now = DateTime.UtcNow;
  1103. if (filter.IsHD.HasValue)
  1104. {
  1105. const int Threshold = 1200;
  1106. if (filter.IsHD.Value)
  1107. {
  1108. minWidth = Threshold;
  1109. }
  1110. else
  1111. {
  1112. maxWidth = Threshold - 1;
  1113. }
  1114. }
  1115. if (filter.Is4K.HasValue)
  1116. {
  1117. const int Threshold = 3800;
  1118. if (filter.Is4K.Value)
  1119. {
  1120. minWidth = Threshold;
  1121. }
  1122. else
  1123. {
  1124. maxWidth = Threshold - 1;
  1125. }
  1126. }
  1127. if (minWidth.HasValue)
  1128. {
  1129. baseQuery = baseQuery.Where(e => e.Width >= minWidth);
  1130. }
  1131. if (filter.MinHeight.HasValue)
  1132. {
  1133. baseQuery = baseQuery.Where(e => e.Height >= filter.MinHeight);
  1134. }
  1135. if (maxWidth.HasValue)
  1136. {
  1137. baseQuery = baseQuery.Where(e => e.Width >= maxWidth);
  1138. }
  1139. if (filter.MaxHeight.HasValue)
  1140. {
  1141. baseQuery = baseQuery.Where(e => e.Height <= filter.MaxHeight);
  1142. }
  1143. if (filter.IsLocked.HasValue)
  1144. {
  1145. baseQuery = baseQuery.Where(e => e.IsLocked == filter.IsLocked);
  1146. }
  1147. var tags = filter.Tags.ToList();
  1148. var excludeTags = filter.ExcludeTags.ToList();
  1149. if (filter.IsMovie == true)
  1150. {
  1151. if (filter.IncludeItemTypes.Length == 0
  1152. || filter.IncludeItemTypes.Contains(BaseItemKind.Movie)
  1153. || filter.IncludeItemTypes.Contains(BaseItemKind.Trailer))
  1154. {
  1155. baseQuery = baseQuery.Where(e => e.IsMovie);
  1156. }
  1157. }
  1158. else if (filter.IsMovie.HasValue)
  1159. {
  1160. baseQuery = baseQuery.Where(e => e.IsMovie == filter.IsMovie);
  1161. }
  1162. if (filter.IsSeries.HasValue)
  1163. {
  1164. baseQuery = baseQuery.Where(e => e.IsSeries == filter.IsSeries);
  1165. }
  1166. if (filter.IsSports.HasValue)
  1167. {
  1168. if (filter.IsSports.Value)
  1169. {
  1170. tags.Add("Sports");
  1171. }
  1172. else
  1173. {
  1174. excludeTags.Add("Sports");
  1175. }
  1176. }
  1177. if (filter.IsNews.HasValue)
  1178. {
  1179. if (filter.IsNews.Value)
  1180. {
  1181. tags.Add("News");
  1182. }
  1183. else
  1184. {
  1185. excludeTags.Add("News");
  1186. }
  1187. }
  1188. if (filter.IsKids.HasValue)
  1189. {
  1190. if (filter.IsKids.Value)
  1191. {
  1192. tags.Add("Kids");
  1193. }
  1194. else
  1195. {
  1196. excludeTags.Add("Kids");
  1197. }
  1198. }
  1199. if (!string.IsNullOrEmpty(filter.SearchTerm))
  1200. {
  1201. var searchTerm = filter.SearchTerm.ToLower();
  1202. baseQuery = baseQuery.Where(e => e.CleanName!.ToLower().Contains(searchTerm) || (e.OriginalTitle != null && e.OriginalTitle.ToLower().Contains(searchTerm)));
  1203. }
  1204. if (filter.IsFolder.HasValue)
  1205. {
  1206. baseQuery = baseQuery.Where(e => e.IsFolder == filter.IsFolder);
  1207. }
  1208. var includeTypes = filter.IncludeItemTypes;
  1209. // Only specify excluded types if no included types are specified
  1210. if (filter.IncludeItemTypes.Length == 0)
  1211. {
  1212. var excludeTypes = filter.ExcludeItemTypes;
  1213. if (excludeTypes.Length == 1)
  1214. {
  1215. if (_itemTypeLookup.BaseItemKindNames.TryGetValue(excludeTypes[0], out var excludeTypeName))
  1216. {
  1217. baseQuery = baseQuery.Where(e => e.Type != excludeTypeName);
  1218. }
  1219. }
  1220. else if (excludeTypes.Length > 1)
  1221. {
  1222. var excludeTypeName = new List<string>();
  1223. foreach (var excludeType in excludeTypes)
  1224. {
  1225. if (_itemTypeLookup.BaseItemKindNames.TryGetValue(excludeType, out var baseItemKindName))
  1226. {
  1227. excludeTypeName.Add(baseItemKindName!);
  1228. }
  1229. }
  1230. baseQuery = baseQuery.Where(e => !excludeTypeName.Contains(e.Type));
  1231. }
  1232. }
  1233. else if (includeTypes.Length == 1)
  1234. {
  1235. if (_itemTypeLookup.BaseItemKindNames.TryGetValue(includeTypes[0], out var includeTypeName))
  1236. {
  1237. baseQuery = baseQuery.Where(e => e.Type == includeTypeName);
  1238. }
  1239. }
  1240. else if (includeTypes.Length > 1)
  1241. {
  1242. var includeTypeName = new List<string>();
  1243. foreach (var includeType in includeTypes)
  1244. {
  1245. if (_itemTypeLookup.BaseItemKindNames.TryGetValue(includeType, out var baseItemKindName))
  1246. {
  1247. includeTypeName.Add(baseItemKindName!);
  1248. }
  1249. }
  1250. baseQuery = baseQuery.Where(e => includeTypeName.Contains(e.Type));
  1251. }
  1252. if (filter.ChannelIds.Count > 0)
  1253. {
  1254. var channelIds = filter.ChannelIds.Select(e => e.ToString("N", CultureInfo.InvariantCulture)).ToArray();
  1255. baseQuery = baseQuery.Where(e => channelIds.Contains(e.ChannelId));
  1256. }
  1257. if (!filter.ParentId.IsEmpty())
  1258. {
  1259. baseQuery = baseQuery.Where(e => e.ParentId!.Value == filter.ParentId);
  1260. }
  1261. if (!string.IsNullOrWhiteSpace(filter.Path))
  1262. {
  1263. baseQuery = baseQuery.Where(e => e.Path == filter.Path);
  1264. }
  1265. if (!string.IsNullOrWhiteSpace(filter.PresentationUniqueKey))
  1266. {
  1267. baseQuery = baseQuery.Where(e => e.PresentationUniqueKey == filter.PresentationUniqueKey);
  1268. }
  1269. if (filter.MinCommunityRating.HasValue)
  1270. {
  1271. baseQuery = baseQuery.Where(e => e.CommunityRating >= filter.MinCommunityRating);
  1272. }
  1273. if (filter.MinIndexNumber.HasValue)
  1274. {
  1275. baseQuery = baseQuery.Where(e => e.IndexNumber >= filter.MinIndexNumber);
  1276. }
  1277. if (filter.MinParentAndIndexNumber.HasValue)
  1278. {
  1279. baseQuery = baseQuery
  1280. .Where(e => (e.ParentIndexNumber == filter.MinParentAndIndexNumber.Value.ParentIndexNumber && e.IndexNumber >= filter.MinParentAndIndexNumber.Value.IndexNumber) || e.ParentIndexNumber > filter.MinParentAndIndexNumber.Value.ParentIndexNumber);
  1281. }
  1282. if (filter.MinDateCreated.HasValue)
  1283. {
  1284. baseQuery = baseQuery.Where(e => e.DateCreated >= filter.MinDateCreated);
  1285. }
  1286. if (filter.MinDateLastSaved.HasValue)
  1287. {
  1288. baseQuery = baseQuery.Where(e => e.DateLastSaved != null && e.DateLastSaved >= filter.MinDateLastSaved.Value);
  1289. }
  1290. if (filter.MinDateLastSavedForUser.HasValue)
  1291. {
  1292. baseQuery = baseQuery.Where(e => e.DateLastSaved != null && e.DateLastSaved >= filter.MinDateLastSavedForUser.Value);
  1293. }
  1294. if (filter.IndexNumber.HasValue)
  1295. {
  1296. baseQuery = baseQuery.Where(e => e.IndexNumber == filter.IndexNumber.Value);
  1297. }
  1298. if (filter.ParentIndexNumber.HasValue)
  1299. {
  1300. baseQuery = baseQuery.Where(e => e.ParentIndexNumber == filter.ParentIndexNumber.Value);
  1301. }
  1302. if (filter.ParentIndexNumberNotEquals.HasValue)
  1303. {
  1304. baseQuery = baseQuery.Where(e => e.ParentIndexNumber != filter.ParentIndexNumberNotEquals.Value || e.ParentIndexNumber == null);
  1305. }
  1306. var minEndDate = filter.MinEndDate;
  1307. var maxEndDate = filter.MaxEndDate;
  1308. if (filter.HasAired.HasValue)
  1309. {
  1310. if (filter.HasAired.Value)
  1311. {
  1312. maxEndDate = DateTime.UtcNow;
  1313. }
  1314. else
  1315. {
  1316. minEndDate = DateTime.UtcNow;
  1317. }
  1318. }
  1319. if (minEndDate.HasValue)
  1320. {
  1321. baseQuery = baseQuery.Where(e => e.EndDate >= minEndDate);
  1322. }
  1323. if (maxEndDate.HasValue)
  1324. {
  1325. baseQuery = baseQuery.Where(e => e.EndDate <= maxEndDate);
  1326. }
  1327. if (filter.MinStartDate.HasValue)
  1328. {
  1329. baseQuery = baseQuery.Where(e => e.StartDate >= filter.MinStartDate.Value);
  1330. }
  1331. if (filter.MaxStartDate.HasValue)
  1332. {
  1333. baseQuery = baseQuery.Where(e => e.StartDate <= filter.MaxStartDate.Value);
  1334. }
  1335. if (filter.MinPremiereDate.HasValue)
  1336. {
  1337. baseQuery = baseQuery.Where(e => e.PremiereDate <= filter.MinPremiereDate.Value);
  1338. }
  1339. if (filter.MaxPremiereDate.HasValue)
  1340. {
  1341. baseQuery = baseQuery.Where(e => e.PremiereDate <= filter.MaxPremiereDate.Value);
  1342. }
  1343. if (filter.TrailerTypes.Length > 0)
  1344. {
  1345. var trailerTypes = filter.TrailerTypes.Select(e => (int)e).ToArray();
  1346. baseQuery = baseQuery.Where(e => trailerTypes.Any(f => e.TrailerTypes!.Any(w => w.Id == f)));
  1347. }
  1348. if (filter.IsAiring.HasValue)
  1349. {
  1350. if (filter.IsAiring.Value)
  1351. {
  1352. baseQuery = baseQuery.Where(e => e.StartDate <= now && e.EndDate >= now);
  1353. }
  1354. else
  1355. {
  1356. baseQuery = baseQuery.Where(e => e.StartDate > now && e.EndDate < now);
  1357. }
  1358. }
  1359. if (filter.PersonIds.Length > 0)
  1360. {
  1361. baseQuery = baseQuery
  1362. .Where(e =>
  1363. context.PeopleBaseItemMap.Where(w => context.BaseItems.Where(r => filter.PersonIds.Contains(r.Id)).Any(f => f.Name == w.People.Name))
  1364. .Any(f => f.ItemId == e.Id));
  1365. }
  1366. if (!string.IsNullOrWhiteSpace(filter.Person))
  1367. {
  1368. baseQuery = baseQuery.Where(e => e.Peoples!.Any(f => f.People.Name == filter.Person));
  1369. }
  1370. if (!string.IsNullOrWhiteSpace(filter.MinSortName))
  1371. {
  1372. // this does not makes sense.
  1373. // baseQuery = baseQuery.Where(e => e.SortName >= query.MinSortName);
  1374. // whereClauses.Add("SortName>=@MinSortName");
  1375. // statement?.TryBind("@MinSortName", query.MinSortName);
  1376. }
  1377. if (!string.IsNullOrWhiteSpace(filter.ExternalSeriesId))
  1378. {
  1379. baseQuery = baseQuery.Where(e => e.ExternalSeriesId == filter.ExternalSeriesId);
  1380. }
  1381. if (!string.IsNullOrWhiteSpace(filter.ExternalId))
  1382. {
  1383. baseQuery = baseQuery.Where(e => e.ExternalId == filter.ExternalId);
  1384. }
  1385. if (!string.IsNullOrWhiteSpace(filter.Name))
  1386. {
  1387. var cleanName = GetCleanValue(filter.Name);
  1388. baseQuery = baseQuery.Where(e => e.CleanName == cleanName);
  1389. }
  1390. // These are the same, for now
  1391. var nameContains = filter.NameContains;
  1392. if (!string.IsNullOrWhiteSpace(nameContains))
  1393. {
  1394. baseQuery = baseQuery.Where(e =>
  1395. e.CleanName!.Contains(nameContains)
  1396. || e.OriginalTitle!.ToLower().Contains(nameContains!));
  1397. }
  1398. if (!string.IsNullOrWhiteSpace(filter.NameStartsWith))
  1399. {
  1400. baseQuery = baseQuery.Where(e => e.SortName!.StartsWith(filter.NameStartsWith) || e.Name!.StartsWith(filter.NameStartsWith));
  1401. }
  1402. if (!string.IsNullOrWhiteSpace(filter.NameStartsWithOrGreater))
  1403. {
  1404. // i hate this
  1405. baseQuery = baseQuery.Where(e => e.SortName!.FirstOrDefault() > filter.NameStartsWithOrGreater[0] || e.Name!.FirstOrDefault() > filter.NameStartsWithOrGreater[0]);
  1406. }
  1407. if (!string.IsNullOrWhiteSpace(filter.NameLessThan))
  1408. {
  1409. // i hate this
  1410. baseQuery = baseQuery.Where(e => e.SortName!.FirstOrDefault() < filter.NameLessThan[0] || e.Name!.FirstOrDefault() < filter.NameLessThan[0]);
  1411. }
  1412. if (filter.ImageTypes.Length > 0)
  1413. {
  1414. var imgTypes = filter.ImageTypes.Select(e => (ImageInfoImageType)e).ToArray();
  1415. baseQuery = baseQuery.Where(e => imgTypes.Any(f => e.Images!.Any(w => w.ImageType == f)));
  1416. }
  1417. if (filter.IsLiked.HasValue)
  1418. {
  1419. baseQuery = baseQuery
  1420. .Where(e => e.UserData!.FirstOrDefault(f => f.UserId == filter.User!.Id)!.Rating >= UserItemData.MinLikeValue);
  1421. }
  1422. if (filter.IsFavoriteOrLiked.HasValue)
  1423. {
  1424. baseQuery = baseQuery
  1425. .Where(e => e.UserData!.FirstOrDefault(f => f.UserId == filter.User!.Id)!.IsFavorite == filter.IsFavoriteOrLiked);
  1426. }
  1427. if (filter.IsFavorite.HasValue)
  1428. {
  1429. baseQuery = baseQuery
  1430. .Where(e => e.UserData!.FirstOrDefault(f => f.UserId == filter.User!.Id)!.IsFavorite == filter.IsFavorite);
  1431. }
  1432. if (filter.IsPlayed.HasValue)
  1433. {
  1434. // We should probably figure this out for all folders, but for right now, this is the only place where we need it
  1435. if (filter.IncludeItemTypes.Length == 1 && filter.IncludeItemTypes[0] == BaseItemKind.Series)
  1436. {
  1437. baseQuery = baseQuery.Where(e => context.BaseItems
  1438. .Where(e => e.IsFolder == false && e.IsVirtualItem == false)
  1439. .Where(f => f.UserData!.FirstOrDefault(e => e.UserId == filter.User!.Id && e.Played)!.Played)
  1440. .Any(f => f.SeriesPresentationUniqueKey == e.PresentationUniqueKey) == filter.IsPlayed);
  1441. }
  1442. else
  1443. {
  1444. baseQuery = baseQuery
  1445. .Select(e => new
  1446. {
  1447. IsPlayed = e.UserData!.Where(f => f.UserId == filter.User!.Id).Select(f => (bool?)f.Played).FirstOrDefault() ?? false,
  1448. Item = e
  1449. })
  1450. .Where(e => e.IsPlayed == filter.IsPlayed)
  1451. .Select(f => f.Item);
  1452. }
  1453. }
  1454. if (filter.IsResumable.HasValue)
  1455. {
  1456. if (filter.IsResumable.Value)
  1457. {
  1458. baseQuery = baseQuery
  1459. .Where(e => e.UserData!.FirstOrDefault(f => f.UserId == filter.User!.Id)!.PlaybackPositionTicks > 0);
  1460. }
  1461. else
  1462. {
  1463. baseQuery = baseQuery
  1464. .Where(e => e.UserData!.FirstOrDefault(f => f.UserId == filter.User!.Id)!.PlaybackPositionTicks == 0);
  1465. }
  1466. }
  1467. if (filter.ArtistIds.Length > 0)
  1468. {
  1469. baseQuery = baseQuery
  1470. .Where(e => e.ItemValues!.Any(f => f.ItemValue.Type <= ItemValueType.Artist && filter.ArtistIds.Contains(f.ItemId)));
  1471. }
  1472. if (filter.AlbumArtistIds.Length > 0)
  1473. {
  1474. baseQuery = baseQuery
  1475. .Where(e => e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Artist && filter.AlbumArtistIds.Contains(f.ItemId)));
  1476. }
  1477. if (filter.ContributingArtistIds.Length > 0)
  1478. {
  1479. baseQuery = baseQuery
  1480. .Where(e => e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Artist && filter.ContributingArtistIds.Contains(f.ItemId)));
  1481. }
  1482. if (filter.AlbumIds.Length > 0)
  1483. {
  1484. baseQuery = baseQuery.Where(e => context.BaseItems.Where(f => filter.AlbumIds.Contains(f.Id)).Any(f => f.Name == e.Album));
  1485. }
  1486. if (filter.ExcludeArtistIds.Length > 0)
  1487. {
  1488. baseQuery = baseQuery
  1489. .Where(e => !e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Artist && filter.ExcludeArtistIds.Contains(f.ItemId)));
  1490. }
  1491. if (filter.GenreIds.Count > 0)
  1492. {
  1493. baseQuery = baseQuery
  1494. .Where(e => e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Genre && filter.GenreIds.Contains(f.ItemId)));
  1495. }
  1496. if (filter.Genres.Count > 0)
  1497. {
  1498. var cleanGenres = filter.Genres.Select(e => GetCleanValue(e)).ToArray();
  1499. baseQuery = baseQuery
  1500. .Where(e => e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Genre && cleanGenres.Contains(f.ItemValue.CleanValue)));
  1501. }
  1502. if (tags.Count > 0)
  1503. {
  1504. var cleanValues = tags.Select(e => GetCleanValue(e)).ToArray();
  1505. baseQuery = baseQuery
  1506. .Where(e => e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Tags && cleanValues.Contains(f.ItemValue.CleanValue)));
  1507. }
  1508. if (excludeTags.Count > 0)
  1509. {
  1510. var cleanValues = excludeTags.Select(e => GetCleanValue(e)).ToArray();
  1511. baseQuery = baseQuery
  1512. .Where(e => !e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Tags && cleanValues.Contains(f.ItemValue.CleanValue)));
  1513. }
  1514. if (filter.StudioIds.Length > 0)
  1515. {
  1516. baseQuery = baseQuery
  1517. .Where(e => e.ItemValues!.Any(f => f.ItemValue.Type == ItemValueType.Studios && filter.StudioIds.Contains(f.ItemId)));
  1518. }
  1519. if (filter.OfficialRatings.Length > 0)
  1520. {
  1521. baseQuery = baseQuery
  1522. .Where(e => filter.OfficialRatings.Contains(e.OfficialRating));
  1523. }
  1524. if (filter.HasParentalRating ?? false)
  1525. {
  1526. if (filter.MinParentalRating.HasValue)
  1527. {
  1528. baseQuery = baseQuery
  1529. .Where(e => e.InheritedParentalRatingValue >= filter.MinParentalRating.Value);
  1530. }
  1531. if (filter.MaxParentalRating.HasValue)
  1532. {
  1533. baseQuery = baseQuery
  1534. .Where(e => e.InheritedParentalRatingValue < filter.MaxParentalRating.Value);
  1535. }
  1536. }
  1537. else if (filter.BlockUnratedItems.Length > 0)
  1538. {
  1539. var unratedItems = filter.BlockUnratedItems.Select(f => f.ToString()).ToArray();
  1540. if (filter.MinParentalRating.HasValue)
  1541. {
  1542. if (filter.MaxParentalRating.HasValue)
  1543. {
  1544. baseQuery = baseQuery
  1545. .Where(e => (e.InheritedParentalRatingValue == null && !unratedItems.Contains(e.UnratedType))
  1546. || (e.InheritedParentalRatingValue >= filter.MinParentalRating && e.InheritedParentalRatingValue <= filter.MaxParentalRating));
  1547. }
  1548. else
  1549. {
  1550. baseQuery = baseQuery
  1551. .Where(e => (e.InheritedParentalRatingValue == null && !unratedItems.Contains(e.UnratedType))
  1552. || e.InheritedParentalRatingValue >= filter.MinParentalRating);
  1553. }
  1554. }
  1555. else
  1556. {
  1557. baseQuery = baseQuery
  1558. .Where(e => e.InheritedParentalRatingValue != null && !unratedItems.Contains(e.UnratedType));
  1559. }
  1560. }
  1561. else if (filter.MinParentalRating.HasValue)
  1562. {
  1563. if (filter.MaxParentalRating.HasValue)
  1564. {
  1565. baseQuery = baseQuery
  1566. .Where(e => e.InheritedParentalRatingValue != null && e.InheritedParentalRatingValue >= filter.MinParentalRating.Value && e.InheritedParentalRatingValue <= filter.MaxParentalRating.Value);
  1567. }
  1568. else
  1569. {
  1570. baseQuery = baseQuery
  1571. .Where(e => e.InheritedParentalRatingValue != null && e.InheritedParentalRatingValue >= filter.MinParentalRating.Value);
  1572. }
  1573. }
  1574. else if (filter.MaxParentalRating.HasValue)
  1575. {
  1576. baseQuery = baseQuery
  1577. .Where(e => e.InheritedParentalRatingValue != null && e.InheritedParentalRatingValue >= filter.MaxParentalRating.Value);
  1578. }
  1579. else if (!filter.HasParentalRating ?? false)
  1580. {
  1581. baseQuery = baseQuery
  1582. .Where(e => e.InheritedParentalRatingValue == null);
  1583. }
  1584. if (filter.HasOfficialRating.HasValue)
  1585. {
  1586. if (filter.HasOfficialRating.Value)
  1587. {
  1588. baseQuery = baseQuery
  1589. .Where(e => e.OfficialRating != null && e.OfficialRating != string.Empty);
  1590. }
  1591. else
  1592. {
  1593. baseQuery = baseQuery
  1594. .Where(e => e.OfficialRating == null || e.OfficialRating == string.Empty);
  1595. }
  1596. }
  1597. if (filter.HasOverview.HasValue)
  1598. {
  1599. if (filter.HasOverview.Value)
  1600. {
  1601. baseQuery = baseQuery
  1602. .Where(e => e.Overview != null && e.Overview != string.Empty);
  1603. }
  1604. else
  1605. {
  1606. baseQuery = baseQuery
  1607. .Where(e => e.Overview == null || e.Overview == string.Empty);
  1608. }
  1609. }
  1610. if (filter.HasOwnerId.HasValue)
  1611. {
  1612. if (filter.HasOwnerId.Value)
  1613. {
  1614. baseQuery = baseQuery
  1615. .Where(e => e.OwnerId != null);
  1616. }
  1617. else
  1618. {
  1619. baseQuery = baseQuery
  1620. .Where(e => e.OwnerId == null);
  1621. }
  1622. }
  1623. if (!string.IsNullOrWhiteSpace(filter.HasNoAudioTrackWithLanguage))
  1624. {
  1625. baseQuery = baseQuery
  1626. .Where(e => !e.MediaStreams!.Any(f => f.StreamType == MediaStreamTypeEntity.Audio && f.Language == filter.HasNoAudioTrackWithLanguage));
  1627. }
  1628. if (!string.IsNullOrWhiteSpace(filter.HasNoInternalSubtitleTrackWithLanguage))
  1629. {
  1630. baseQuery = baseQuery
  1631. .Where(e => !e.MediaStreams!.Any(f => f.StreamType == MediaStreamTypeEntity.Subtitle && !f.IsExternal && f.Language == filter.HasNoInternalSubtitleTrackWithLanguage));
  1632. }
  1633. if (!string.IsNullOrWhiteSpace(filter.HasNoExternalSubtitleTrackWithLanguage))
  1634. {
  1635. baseQuery = baseQuery
  1636. .Where(e => !e.MediaStreams!.Any(f => f.StreamType == MediaStreamTypeEntity.Subtitle && f.IsExternal && f.Language == filter.HasNoExternalSubtitleTrackWithLanguage));
  1637. }
  1638. if (!string.IsNullOrWhiteSpace(filter.HasNoSubtitleTrackWithLanguage))
  1639. {
  1640. baseQuery = baseQuery
  1641. .Where(e => !e.MediaStreams!.Any(f => f.StreamType == MediaStreamTypeEntity.Subtitle && f.Language == filter.HasNoSubtitleTrackWithLanguage));
  1642. }
  1643. if (filter.HasSubtitles.HasValue)
  1644. {
  1645. baseQuery = baseQuery
  1646. .Where(e => e.MediaStreams!.Any(f => f.StreamType == MediaStreamTypeEntity.Subtitle) == filter.HasSubtitles.Value);
  1647. }
  1648. if (filter.HasChapterImages.HasValue)
  1649. {
  1650. baseQuery = baseQuery
  1651. .Where(e => e.Chapters!.Any(f => f.ImagePath != null) == filter.HasChapterImages.Value);
  1652. }
  1653. if (filter.HasDeadParentId.HasValue && filter.HasDeadParentId.Value)
  1654. {
  1655. baseQuery = baseQuery
  1656. .Where(e => e.ParentId.HasValue && !context.BaseItems.Any(f => f.Id == e.ParentId.Value));
  1657. }
  1658. if (filter.IsDeadArtist.HasValue && filter.IsDeadArtist.Value)
  1659. {
  1660. baseQuery = baseQuery
  1661. .Where(e => e.ItemValues!.Count(f => f.ItemValue.Type == ItemValueType.Artist || f.ItemValue.Type == ItemValueType.AlbumArtist) == 1);
  1662. }
  1663. if (filter.IsDeadStudio.HasValue && filter.IsDeadStudio.Value)
  1664. {
  1665. baseQuery = baseQuery
  1666. .Where(e => e.ItemValues!.Count(f => f.ItemValue.Type == ItemValueType.Studios) == 1);
  1667. }
  1668. if (filter.IsDeadPerson.HasValue && filter.IsDeadPerson.Value)
  1669. {
  1670. baseQuery = baseQuery
  1671. .Where(e => !context.Peoples.Any(f => f.Name == e.Name));
  1672. }
  1673. if (filter.Years.Length == 1)
  1674. {
  1675. baseQuery = baseQuery
  1676. .Where(e => e.ProductionYear == filter.Years[0]);
  1677. }
  1678. else if (filter.Years.Length > 1)
  1679. {
  1680. baseQuery = baseQuery
  1681. .Where(e => filter.Years.Any(f => f == e.ProductionYear));
  1682. }
  1683. var isVirtualItem = filter.IsVirtualItem ?? filter.IsMissing;
  1684. if (isVirtualItem.HasValue)
  1685. {
  1686. baseQuery = baseQuery
  1687. .Where(e => e.IsVirtualItem == isVirtualItem.Value);
  1688. }
  1689. if (filter.IsSpecialSeason.HasValue)
  1690. {
  1691. if (filter.IsSpecialSeason.Value)
  1692. {
  1693. baseQuery = baseQuery
  1694. .Where(e => e.IndexNumber == 0);
  1695. }
  1696. else
  1697. {
  1698. baseQuery = baseQuery
  1699. .Where(e => e.IndexNumber != 0);
  1700. }
  1701. }
  1702. if (filter.IsUnaired.HasValue)
  1703. {
  1704. if (filter.IsUnaired.Value)
  1705. {
  1706. baseQuery = baseQuery
  1707. .Where(e => e.PremiereDate >= now);
  1708. }
  1709. else
  1710. {
  1711. baseQuery = baseQuery
  1712. .Where(e => e.PremiereDate < now);
  1713. }
  1714. }
  1715. if (filter.MediaTypes.Length > 0)
  1716. {
  1717. var mediaTypes = filter.MediaTypes.Select(f => f.ToString()).ToArray();
  1718. baseQuery = baseQuery
  1719. .Where(e => mediaTypes.Contains(e.MediaType));
  1720. }
  1721. if (filter.ItemIds.Length > 0)
  1722. {
  1723. baseQuery = baseQuery
  1724. .Where(e => filter.ItemIds.Contains(e.Id));
  1725. }
  1726. if (filter.ExcludeItemIds.Length > 0)
  1727. {
  1728. baseQuery = baseQuery
  1729. .Where(e => !filter.ItemIds.Contains(e.Id));
  1730. }
  1731. if (filter.ExcludeProviderIds is not null && filter.ExcludeProviderIds.Count > 0)
  1732. {
  1733. baseQuery = baseQuery.Where(e => !e.Provider!.All(f => !filter.ExcludeProviderIds.All(w => f.ProviderId == w.Key && f.ProviderValue == w.Value)));
  1734. }
  1735. if (filter.HasAnyProviderId is not null && filter.HasAnyProviderId.Count > 0)
  1736. {
  1737. baseQuery = baseQuery.Where(e => e.Provider!.Any(f => !filter.HasAnyProviderId.Any(w => f.ProviderId == w.Key && f.ProviderValue == w.Value)));
  1738. }
  1739. if (filter.HasImdbId.HasValue)
  1740. {
  1741. baseQuery = baseQuery.Where(e => e.Provider!.Any(f => f.ProviderId == "imdb"));
  1742. }
  1743. if (filter.HasTmdbId.HasValue)
  1744. {
  1745. baseQuery = baseQuery.Where(e => e.Provider!.Any(f => f.ProviderId == "tmdb"));
  1746. }
  1747. if (filter.HasTvdbId.HasValue)
  1748. {
  1749. baseQuery = baseQuery.Where(e => e.Provider!.Any(f => f.ProviderId == "tvdb"));
  1750. }
  1751. var queryTopParentIds = filter.TopParentIds;
  1752. if (queryTopParentIds.Length > 0)
  1753. {
  1754. var includedItemByNameTypes = GetItemByNameTypesInQuery(filter);
  1755. var enableItemsByName = (filter.IncludeItemsByName ?? false) && includedItemByNameTypes.Count > 0;
  1756. if (enableItemsByName && includedItemByNameTypes.Count > 0)
  1757. {
  1758. baseQuery = baseQuery.Where(e => includedItemByNameTypes.Contains(e.Type) || queryTopParentIds.Any(w => w == e.TopParentId!.Value));
  1759. }
  1760. else
  1761. {
  1762. baseQuery = baseQuery.Where(e => queryTopParentIds.Contains(e.TopParentId!.Value));
  1763. }
  1764. }
  1765. if (filter.AncestorIds.Length > 0)
  1766. {
  1767. baseQuery = baseQuery.Where(e => e.Children!.Any(f => filter.AncestorIds.Contains(f.ParentItemId)));
  1768. }
  1769. if (!string.IsNullOrWhiteSpace(filter.AncestorWithPresentationUniqueKey))
  1770. {
  1771. baseQuery = baseQuery
  1772. .Where(e => context.BaseItems.Where(f => f.PresentationUniqueKey == filter.AncestorWithPresentationUniqueKey).Any(f => f.ParentAncestors!.Any(w => w.ItemId == f.Id)));
  1773. }
  1774. if (!string.IsNullOrWhiteSpace(filter.SeriesPresentationUniqueKey))
  1775. {
  1776. baseQuery = baseQuery
  1777. .Where(e => e.SeriesPresentationUniqueKey == filter.SeriesPresentationUniqueKey);
  1778. }
  1779. if (filter.ExcludeInheritedTags.Length > 0)
  1780. {
  1781. baseQuery = baseQuery
  1782. .Where(e => !e.ItemValues!.Where(w => w.ItemValue.Type == ItemValueType.InheritedTags)
  1783. .Any(f => filter.ExcludeInheritedTags.Contains(f.ItemValue.CleanValue)));
  1784. }
  1785. if (filter.IncludeInheritedTags.Length > 0)
  1786. {
  1787. // Episodes do not store inherit tags from their parents in the database, and the tag may be still required by the client.
  1788. // In addtion to the tags for the episodes themselves, we need to manually query its parent (the season)'s tags as well.
  1789. if (includeTypes.Length == 1 && includeTypes.FirstOrDefault() is BaseItemKind.Episode)
  1790. {
  1791. baseQuery = baseQuery
  1792. .Where(e => e.ItemValues!.Where(f => f.ItemValue.Type == ItemValueType.InheritedTags)
  1793. .Any(f => filter.IncludeInheritedTags.Contains(f.ItemValue.CleanValue))
  1794. ||
  1795. (e.ParentId.HasValue && context.ItemValuesMap.Where(w => w.ItemId == e.ParentId.Value)!.Where(w => w.ItemValue.Type == ItemValueType.InheritedTags)
  1796. .Any(f => filter.IncludeInheritedTags.Contains(f.ItemValue.CleanValue))));
  1797. }
  1798. // A playlist should be accessible to its owner regardless of allowed tags.
  1799. else if (includeTypes.Length == 1 && includeTypes.FirstOrDefault() is BaseItemKind.Playlist)
  1800. {
  1801. baseQuery = baseQuery
  1802. .Where(e =>
  1803. e.ParentAncestors!
  1804. .Any(f =>
  1805. f.ParentItem.ItemValues!.Any(w => w.ItemValue.Type == ItemValueType.Tags && filter.IncludeInheritedTags.Contains(w.ItemValue.CleanValue))
  1806. || e.Data!.Contains($"OwnerUserId\":\"{filter.User!.Id:N}\"")));
  1807. // d ^^ this is stupid it hate this.
  1808. }
  1809. else
  1810. {
  1811. baseQuery = baseQuery
  1812. .Where(e => e.ParentAncestors!.Any(f => f.ParentItem.ItemValues!.Any(w => w.ItemValue.Type == ItemValueType.Tags && filter.IncludeInheritedTags.Contains(w.ItemValue.CleanValue))));
  1813. }
  1814. }
  1815. if (filter.SeriesStatuses.Length > 0)
  1816. {
  1817. var seriesStatus = filter.SeriesStatuses.Select(e => e.ToString()).ToArray();
  1818. baseQuery = baseQuery
  1819. .Where(e => seriesStatus.Any(f => e.Data!.Contains(f)));
  1820. }
  1821. if (filter.BoxSetLibraryFolders.Length > 0)
  1822. {
  1823. var boxsetFolders = filter.BoxSetLibraryFolders.Select(e => e.ToString("N", CultureInfo.InvariantCulture)).ToArray();
  1824. baseQuery = baseQuery
  1825. .Where(e => boxsetFolders.Any(f => e.Data!.Contains(f)));
  1826. }
  1827. if (filter.VideoTypes.Length > 0)
  1828. {
  1829. var videoTypeBs = filter.VideoTypes.Select(e => $"\"VideoType\":\"{e}\"");
  1830. baseQuery = baseQuery
  1831. .Where(e => videoTypeBs.Any(f => e.Data!.Contains(f)));
  1832. }
  1833. if (filter.Is3D.HasValue)
  1834. {
  1835. if (filter.Is3D.Value)
  1836. {
  1837. baseQuery = baseQuery
  1838. .Where(e => e.Data!.Contains("Video3DFormat"));
  1839. }
  1840. else
  1841. {
  1842. baseQuery = baseQuery
  1843. .Where(e => !e.Data!.Contains("Video3DFormat"));
  1844. }
  1845. }
  1846. if (filter.IsPlaceHolder.HasValue)
  1847. {
  1848. if (filter.IsPlaceHolder.Value)
  1849. {
  1850. baseQuery = baseQuery
  1851. .Where(e => e.Data!.Contains("IsPlaceHolder\":true"));
  1852. }
  1853. else
  1854. {
  1855. baseQuery = baseQuery
  1856. .Where(e => !e.Data!.Contains("IsPlaceHolder\":true"));
  1857. }
  1858. }
  1859. if (filter.HasSpecialFeature.HasValue)
  1860. {
  1861. if (filter.HasSpecialFeature.Value)
  1862. {
  1863. baseQuery = baseQuery
  1864. .Where(e => e.ExtraIds != null);
  1865. }
  1866. else
  1867. {
  1868. baseQuery = baseQuery
  1869. .Where(e => e.ExtraIds == null);
  1870. }
  1871. }
  1872. if (filter.HasTrailer.HasValue || filter.HasThemeSong.HasValue || filter.HasThemeVideo.HasValue)
  1873. {
  1874. if (filter.HasTrailer.GetValueOrDefault() || filter.HasThemeSong.GetValueOrDefault() || filter.HasThemeVideo.GetValueOrDefault())
  1875. {
  1876. baseQuery = baseQuery
  1877. .Where(e => e.ExtraIds != null);
  1878. }
  1879. else
  1880. {
  1881. baseQuery = baseQuery
  1882. .Where(e => e.ExtraIds == null);
  1883. }
  1884. }
  1885. return baseQuery;
  1886. }
  1887. }