jellyfin-server/Emby.Dlna/ContentDirectory/ControlHandler.cs

1737 lines
68 KiB
C#
Raw Normal View History

using System;
2020-09-13 13:31:12 +00:00
using System.Collections;
2016-10-29 22:22:20 +00:00
using System.Collections.Generic;
using System.Globalization;
2020-01-28 12:19:25 +00:00
using System.IO;
2016-10-29 22:22:20 +00:00
using System.Linq;
using System.Text;
using System.Threading;
using System.Xml;
2020-09-13 13:31:12 +00:00
using Emby.Dlna.Configuration;
2019-01-13 19:16:19 +00:00
using Emby.Dlna.Didl;
using Emby.Dlna.Service;
2020-05-20 17:07:53 +00:00
using Jellyfin.Data.Entities;
2020-07-01 01:44:41 +00:00
using Jellyfin.Data.Enums;
2019-01-13 19:16:19 +00:00
using MediaBrowser.Common.Extensions;
using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Drawing;
2017-05-21 07:25:49 +00:00
using MediaBrowser.Controller.Dto;
2019-01-13 19:16:19 +00:00
using MediaBrowser.Controller.Entities;
2016-12-20 07:31:30 +00:00
using MediaBrowser.Controller.Entities.Audio;
2019-01-13 19:16:19 +00:00
using MediaBrowser.Controller.Entities.Movies;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.LiveTv;
2016-10-29 22:22:20 +00:00
using MediaBrowser.Controller.MediaEncoding;
2017-07-22 23:00:48 +00:00
using MediaBrowser.Controller.Playlists;
using MediaBrowser.Controller.TV;
2019-01-13 19:16:19 +00:00
using MediaBrowser.Model.Dlna;
using MediaBrowser.Model.Entities;
2016-10-29 22:22:20 +00:00
using MediaBrowser.Model.Globalization;
2019-01-13 19:16:19 +00:00
using MediaBrowser.Model.Querying;
using Microsoft.Extensions.Logging;
2020-05-20 17:07:53 +00:00
using Book = MediaBrowser.Controller.Entities.Book;
using Episode = MediaBrowser.Controller.Entities.TV.Episode;
using Genre = MediaBrowser.Controller.Entities.Genre;
using Movie = MediaBrowser.Controller.Entities.Movies.Movie;
using MusicAlbum = MediaBrowser.Controller.Entities.Audio.MusicAlbum;
using Series = MediaBrowser.Controller.Entities.TV.Series;
2016-10-29 22:22:20 +00:00
2016-10-29 22:34:54 +00:00
namespace Emby.Dlna.ContentDirectory
2016-10-29 22:22:20 +00:00
{
2020-09-13 13:31:12 +00:00
/// <summary>
/// Defines the <see cref="ControlHandler" />.
/// </summary>
2016-10-29 22:22:20 +00:00
public class ControlHandler : BaseControlHandler
{
2020-08-20 19:04:57 +00:00
private const string NsDc = "http://purl.org/dc/elements/1.1/";
private const string NsDidl = "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/";
private const string NsDlna = "urn:schemas-dlna-org:metadata-1-0/";
private const string NsUpnp = "urn:schemas-upnp-org:metadata-1-0/upnp/";
2016-10-29 22:22:20 +00:00
private readonly ILibraryManager _libraryManager;
private readonly IUserDataManager _userDataManager;
private readonly IServerConfigurationManager _config;
2020-05-20 17:07:53 +00:00
private readonly User _user;
2016-10-29 22:22:20 +00:00
private readonly IUserViewManager _userViewManager;
private readonly ITVSeriesManager _tvSeriesManager;
2016-10-29 22:22:20 +00:00
private readonly int _systemUpdateId;
private readonly DidlBuilder _didlBuilder;
private readonly DeviceProfile _profile;
2020-09-13 13:31:12 +00:00
/// <summary>
/// Initializes a new instance of the <see cref="ControlHandler"/> class.
/// </summary>
/// <param name="logger">The <see cref="ILogger"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="libraryManager">The <see cref="ILibraryManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="profile">The <see cref="DeviceProfile"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="serverAddress">The server address to use in this instance> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="accessToken">The <see cref="string"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="imageProcessor">The <see cref="IImageProcessor"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="userDataManager">The <see cref="IUserDataManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="user">The <see cref="User"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="systemUpdateId">The system id for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="config">The <see cref="IServerConfigurationManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="localization">The <see cref="ILocalizationManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="mediaSourceManager">The <see cref="IMediaSourceManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="userViewManager">The <see cref="IUserViewManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="mediaEncoder">The <see cref="IMediaEncoder"/> for use with the <see cref="ControlHandler"/> instance.</param>
/// <param name="tvSeriesManager">The <see cref="ITVSeriesManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
public ControlHandler(
ILogger logger,
ILibraryManager libraryManager,
DeviceProfile profile,
string serverAddress,
string accessToken,
IImageProcessor imageProcessor,
IUserDataManager userDataManager,
2020-05-20 17:07:53 +00:00
User user,
2020-01-28 20:54:51 +00:00
int systemUpdateId,
IServerConfigurationManager config,
ILocalizationManager localization,
IMediaSourceManager mediaSourceManager,
IUserViewManager userViewManager,
IMediaEncoder mediaEncoder,
ITVSeriesManager tvSeriesManager)
: base(config, logger)
2016-10-29 22:22:20 +00:00
{
_libraryManager = libraryManager;
_userDataManager = userDataManager;
_user = user;
_systemUpdateId = systemUpdateId;
_userViewManager = userViewManager;
_tvSeriesManager = tvSeriesManager;
2016-10-29 22:22:20 +00:00
_profile = profile;
_config = config;
2020-04-02 14:49:58 +00:00
_didlBuilder = new DidlBuilder(
profile,
user,
imageProcessor,
serverAddress,
accessToken,
userDataManager,
localization,
mediaSourceManager,
Logger,
mediaEncoder,
libraryManager);
2016-10-29 22:22:20 +00:00
}
/// <inheritdoc />
protected override void WriteResult(string methodName, IDictionary<string, string> methodParams, XmlWriter xmlWriter)
2016-10-29 22:22:20 +00:00
{
2020-09-13 13:31:12 +00:00
if (xmlWriter == null)
{
throw new ArgumentNullException(nameof(xmlWriter));
}
if (methodParams == null)
{
throw new ArgumentNullException(nameof(methodParams));
}
const string DeviceId = "test";
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "GetSearchCapabilities", StringComparison.OrdinalIgnoreCase))
{
HandleGetSearchCapabilities(xmlWriter);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "GetSortCapabilities", StringComparison.OrdinalIgnoreCase))
{
HandleGetSortCapabilities(xmlWriter);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "GetSortExtensionCapabilities", StringComparison.OrdinalIgnoreCase))
{
HandleGetSortExtensionCapabilities(xmlWriter);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "GetSystemUpdateID", StringComparison.OrdinalIgnoreCase))
{
HandleGetSystemUpdateID(xmlWriter);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "Browse", StringComparison.OrdinalIgnoreCase))
{
HandleBrowse(xmlWriter, methodParams, DeviceId);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "X_GetFeatureList", StringComparison.OrdinalIgnoreCase))
{
HandleXGetFeatureList(xmlWriter);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "GetFeatureList", StringComparison.OrdinalIgnoreCase))
{
HandleGetFeatureList(xmlWriter);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "X_SetBookmark", StringComparison.OrdinalIgnoreCase))
{
HandleXSetBookmark(methodParams);
return;
}
2016-10-29 22:22:20 +00:00
if (string.Equals(methodName, "Search", StringComparison.OrdinalIgnoreCase))
{
HandleSearch(xmlWriter, methodParams, DeviceId);
return;
}
2016-10-29 22:22:20 +00:00
2016-12-14 20:58:55 +00:00
if (string.Equals(methodName, "X_BrowseByLetter", StringComparison.OrdinalIgnoreCase))
{
HandleXBrowseByLetter(xmlWriter, methodParams, DeviceId);
return;
}
2016-12-14 20:58:55 +00:00
2016-10-29 22:22:20 +00:00
throw new ResourceNotFoundException("Unexpected control request name: " + methodName);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Adds a "XSetBookmark" element to the xml document.
/// </summary>
/// <param name="sparams">The <see cref="IDictionary"/>.</param>
private void HandleXSetBookmark(IDictionary<string, string> sparams)
2016-10-29 22:22:20 +00:00
{
var id = sparams["ObjectID"];
2020-04-02 14:49:58 +00:00
var serverItem = GetItemFromObjectId(id);
2016-10-29 22:22:20 +00:00
var item = serverItem.Item;
2020-01-26 16:37:47 +00:00
var newbookmark = int.Parse(sparams["PosSecond"], CultureInfo.InvariantCulture);
2016-10-29 22:22:20 +00:00
var userdata = _userDataManager.GetUserData(_user, item);
2016-10-29 22:22:20 +00:00
userdata.PlaybackPositionTicks = TimeSpan.FromSeconds(newbookmark).Ticks;
2020-08-20 19:04:57 +00:00
_userDataManager.SaveUserData(
_user,
item,
userdata,
UserDataSaveReason.TogglePlayed,
2016-10-29 22:22:20 +00:00
CancellationToken.None);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Adds the "SearchCaps" element to the xml document.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
private static void HandleGetSearchCapabilities(XmlWriter xmlWriter)
2016-10-29 22:22:20 +00:00
{
xmlWriter.WriteElementString(
"SearchCaps",
"res@resolution,res@size,res@duration,dc:title,dc:creator,upnp:actor,upnp:artist,upnp:genre,upnp:album,dc:date,upnp:class,@id,@refID,@protocolInfo,upnp:author,dc:description,pv:avKeywords");
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Adds the "SortCaps" element to the xml document.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
private static void HandleGetSortCapabilities(XmlWriter xmlWriter)
2016-10-29 22:22:20 +00:00
{
xmlWriter.WriteElementString(
"SortCaps",
"res@duration,res@size,res@bitrate,dc:date,dc:title,dc:size,upnp:album,upnp:artist,upnp:albumArtist,upnp:episodeNumber,upnp:genre,upnp:originalTrackNumber,upnp:rating");
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Adds the "SortExtensionCaps" element to the xml document.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
private static void HandleGetSortExtensionCapabilities(XmlWriter xmlWriter)
2016-10-29 22:22:20 +00:00
{
xmlWriter.WriteElementString(
"SortExtensionCaps",
"res@duration,res@size,res@bitrate,dc:date,dc:title,dc:size,upnp:album,upnp:artist,upnp:albumArtist,upnp:episodeNumber,upnp:genre,upnp:originalTrackNumber,upnp:rating");
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Adds the "Id" element to the xml document.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
private void HandleGetSystemUpdateID(XmlWriter xmlWriter)
2016-10-29 22:22:20 +00:00
{
xmlWriter.WriteElementString("Id", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Adds the "FeatureList" element to the xml document.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
private static void HandleGetFeatureList(XmlWriter xmlWriter)
2016-10-29 22:22:20 +00:00
{
xmlWriter.WriteElementString("FeatureList", WriteFeatureListXml());
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Adds the "FeatureList" element to the xml document.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
private static void HandleXGetFeatureList(XmlWriter xmlWriter)
=> HandleGetFeatureList(xmlWriter);
2016-10-29 22:22:20 +00:00
2020-09-13 13:31:12 +00:00
/// <summary>
/// Builds a static feature list.
/// </summary>
/// <returns>The xml feature list.</returns>
private static string WriteFeatureListXml()
2016-10-29 22:22:20 +00:00
{
// TODO: clean this up
2016-10-29 22:22:20 +00:00
var builder = new StringBuilder();
builder.Append("<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
builder.Append("<Features xmlns=\"urn:schemas-upnp-org:av:avs\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xsi:schemaLocation=\"urn:schemas-upnp-org:av:avs http://www.upnp.org/schemas/av/avs.xsd\">");
builder.Append("<Feature name=\"samsung.com_BASICVIEW\" version=\"1\">");
builder.Append("<container id=\"I\" type=\"object.item.imageItem\"/>");
builder.Append("<container id=\"A\" type=\"object.item.audioItem\"/>");
builder.Append("<container id=\"V\" type=\"object.item.videoItem\"/>");
builder.Append("</Feature>");
builder.Append("</Features>");
return builder.ToString();
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the value in the key of the dictionary, or defaultValue if it doesn't exist.
/// </summary>
/// <param name="sparams">The <see cref="IDictionary"/>.</param>
/// <param name="key">The key.</param>
/// <param name="defaultValue">The defaultValue.</param>
/// <returns>The <see cref="string"/>.</returns>
public static string GetValueOrDefault(IDictionary<string, string> sparams, string key, string defaultValue)
2016-12-04 21:30:38 +00:00
{
2020-09-13 13:31:12 +00:00
if (sparams != null && sparams.TryGetValue(key, out string val))
2016-12-04 21:30:38 +00:00
{
return val;
}
return defaultValue;
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Builds the "Browse" xml response.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
/// <param name="sparams">The <see cref="IDictionary"/>.</param>
/// <param name="deviceId">The device Id to use.</param>
private void HandleBrowse(XmlWriter xmlWriter, IDictionary<string, string> sparams, string deviceId)
2016-10-29 22:22:20 +00:00
{
var id = sparams["ObjectID"];
var flag = sparams["BrowseFlag"];
2016-12-04 21:30:38 +00:00
var filter = new Filter(GetValueOrDefault(sparams, "Filter", "*"));
2020-08-20 15:01:04 +00:00
var sortCriteria = new SortCriteria(GetValueOrDefault(sparams, "SortCriteria", string.Empty));
2016-10-29 22:22:20 +00:00
var provided = 0;
// Default to null instead of 0
// Upnp inspector sends 0 as requestedCount when it wants everything
int? requestedCount = null;
int? start = 0;
if (sparams.ContainsKey("RequestedCount") && int.TryParse(sparams["RequestedCount"], out var requestedVal) && requestedVal > 0)
2016-10-29 22:22:20 +00:00
{
requestedCount = requestedVal;
}
if (sparams.ContainsKey("StartingIndex") && int.TryParse(sparams["StartingIndex"], out var startVal) && startVal > 0)
2016-10-29 22:22:20 +00:00
{
start = startVal;
}
int totalCount;
2020-01-28 12:19:25 +00:00
using (StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8))
2016-10-29 22:22:20 +00:00
{
2020-01-29 16:30:27 +00:00
var settings = new XmlWriterSettings()
{
Encoding = Encoding.UTF8,
CloseOutput = false,
OmitXmlDeclaration = true,
ConformanceLevel = ConformanceLevel.Fragment
};
2016-10-29 22:22:20 +00:00
2020-01-29 16:30:27 +00:00
using (var writer = XmlWriter.Create(builder, settings))
2016-10-29 22:22:20 +00:00
{
2020-08-20 19:04:57 +00:00
writer.WriteStartElement(string.Empty, "DIDL-Lite", NsDidl);
2016-10-29 22:22:20 +00:00
2020-08-20 19:04:57 +00:00
writer.WriteAttributeString("xmlns", "dc", null, NsDc);
writer.WriteAttributeString("xmlns", "dlna", null, NsDlna);
writer.WriteAttributeString("xmlns", "upnp", null, NsUpnp);
2016-10-29 22:22:20 +00:00
2020-01-29 16:30:27 +00:00
DidlBuilder.WriteXmlRootAttributes(_profile, writer);
2020-01-28 12:19:25 +00:00
2020-04-02 14:49:58 +00:00
var serverItem = GetItemFromObjectId(id);
2020-01-29 16:30:27 +00:00
var item = serverItem.Item;
2016-10-29 22:22:20 +00:00
2020-01-29 16:30:27 +00:00
if (string.Equals(flag, "BrowseMetadata", StringComparison.Ordinal))
2020-01-28 12:19:25 +00:00
{
2020-01-29 16:30:27 +00:00
totalCount = 1;
2020-01-28 12:19:25 +00:00
2020-01-29 16:30:27 +00:00
if (item.IsDisplayedAsFolder || serverItem.StubType.HasValue)
2020-01-28 12:19:25 +00:00
{
2020-01-29 16:30:27 +00:00
var childrenResult = GetUserItems(item, serverItem.StubType, _user, sortCriteria, start, requestedCount);
2020-01-28 12:19:25 +00:00
2020-01-29 16:30:27 +00:00
_didlBuilder.WriteFolderElement(writer, item, serverItem.StubType, null, childrenResult.TotalRecordCount, filter, id);
2020-01-28 12:19:25 +00:00
}
else
{
2020-04-02 14:49:58 +00:00
_didlBuilder.WriteItemElement(writer, item, _user, null, null, deviceId, filter);
2020-01-28 12:19:25 +00:00
}
2020-01-29 16:30:27 +00:00
provided++;
2020-01-28 12:19:25 +00:00
}
2020-01-29 16:30:27 +00:00
else
{
var childrenResult = GetUserItems(item, serverItem.StubType, _user, sortCriteria, start, requestedCount);
totalCount = childrenResult.TotalRecordCount;
provided = childrenResult.Items.Count;
2020-01-28 12:19:25 +00:00
2020-01-29 16:30:27 +00:00
foreach (var i in childrenResult.Items)
{
var childItem = i.Item;
var displayStubType = i.StubType;
if (childItem.IsDisplayedAsFolder || displayStubType.HasValue)
{
var childCount = GetUserItems(childItem, displayStubType, _user, sortCriteria, null, 0)
.TotalRecordCount;
_didlBuilder.WriteFolderElement(writer, childItem, displayStubType, item, childCount, filter);
}
else
{
2020-04-02 14:49:58 +00:00
_didlBuilder.WriteItemElement(writer, childItem, _user, item, serverItem.StubType, deviceId, filter);
2020-01-29 16:30:27 +00:00
}
}
}
writer.WriteFullEndElement();
}
2020-01-28 12:19:25 +00:00
xmlWriter.WriteElementString("Result", builder.ToString());
}
2016-10-29 22:22:20 +00:00
2020-01-26 16:37:47 +00:00
xmlWriter.WriteElementString("NumberReturned", provided.ToString(CultureInfo.InvariantCulture));
xmlWriter.WriteElementString("TotalMatches", totalCount.ToString(CultureInfo.InvariantCulture));
xmlWriter.WriteElementString("UpdateID", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Builds the response to the "X_BrowseByLetter request.
/// </summary>
/// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
/// <param name="sparams">The <see cref="IDictionary"/>.</param>
/// <param name="deviceId">The device id.</param>
private void HandleXBrowseByLetter(XmlWriter xmlWriter, IDictionary<string, string> sparams, string deviceId)
2016-12-14 20:58:55 +00:00
{
// TODO: Implement this method
HandleSearch(xmlWriter, sparams, deviceId);
2016-12-14 20:58:55 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Builds a response to the "Search" request.
/// </summary>
/// <param name="xmlWriter">The xmlWriter<see cref="XmlWriter"/>.</param>
/// <param name="sparams">The sparams<see cref="IDictionary"/>.</param>
/// <param name="deviceId">The deviceId<see cref="string"/>.</param>
private void HandleSearch(XmlWriter xmlWriter, IDictionary<string, string> sparams, string deviceId)
2016-10-29 22:22:20 +00:00
{
2020-08-20 15:01:04 +00:00
var searchCriteria = new SearchCriteria(GetValueOrDefault(sparams, "SearchCriteria", string.Empty));
var sortCriteria = new SortCriteria(GetValueOrDefault(sparams, "SortCriteria", string.Empty));
2016-12-04 21:30:38 +00:00
var filter = new Filter(GetValueOrDefault(sparams, "Filter", "*"));
2016-10-29 22:22:20 +00:00
// sort example: dc:title, dc:date
// Default to null instead of 0
// Upnp inspector sends 0 as requestedCount when it wants everything
int? requestedCount = null;
int? start = 0;
if (sparams.ContainsKey("RequestedCount") && int.TryParse(sparams["RequestedCount"], out var requestedVal) && requestedVal > 0)
2016-10-29 22:22:20 +00:00
{
requestedCount = requestedVal;
}
if (sparams.ContainsKey("StartingIndex") && int.TryParse(sparams["StartingIndex"], out var startVal) && startVal > 0)
2016-10-29 22:22:20 +00:00
{
start = startVal;
}
2020-01-28 12:19:25 +00:00
QueryResult<BaseItem> childrenResult;
2016-10-29 22:22:20 +00:00
2020-01-28 12:19:25 +00:00
using (StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8))
{
2020-01-29 16:30:27 +00:00
var settings = new XmlWriterSettings()
{
Encoding = Encoding.UTF8,
CloseOutput = false,
OmitXmlDeclaration = true,
ConformanceLevel = ConformanceLevel.Fragment
};
2016-10-29 22:22:20 +00:00
2020-01-29 16:30:27 +00:00
using (var writer = XmlWriter.Create(builder, settings))
{
2020-08-20 19:04:57 +00:00
writer.WriteStartElement(string.Empty, "DIDL-Lite", NsDidl);
2016-10-29 22:22:20 +00:00
2020-08-20 19:04:57 +00:00
writer.WriteAttributeString("xmlns", "dc", null, NsDc);
writer.WriteAttributeString("xmlns", "dlna", null, NsDlna);
writer.WriteAttributeString("xmlns", "upnp", null, NsUpnp);
2016-10-29 22:22:20 +00:00
2020-01-29 16:30:27 +00:00
DidlBuilder.WriteXmlRootAttributes(_profile, writer);
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
var serverItem = GetItemFromObjectId(sparams["ContainerID"]);
2016-10-29 22:22:20 +00:00
2020-01-29 16:30:27 +00:00
var item = serverItem.Item;
2016-11-04 08:31:05 +00:00
2020-01-29 16:30:27 +00:00
childrenResult = GetChildrenSorted(item, _user, searchCriteria, sortCriteria, start, requestedCount);
2018-09-12 17:26:21 +00:00
2020-01-29 16:30:27 +00:00
var dlnaOptions = _config.GetDlnaConfiguration();
2016-11-04 08:31:05 +00:00
2020-01-29 16:30:27 +00:00
foreach (var i in childrenResult.Items)
2020-01-28 12:19:25 +00:00
{
2020-01-29 16:30:27 +00:00
if (i.IsDisplayedAsFolder)
{
var childCount = GetChildrenSorted(i, _user, searchCriteria, sortCriteria, null, 0)
.TotalRecordCount;
_didlBuilder.WriteFolderElement(writer, i, null, item, childCount, filter);
}
else
{
2020-04-02 14:49:58 +00:00
_didlBuilder.WriteItemElement(writer, i, _user, item, serverItem.StubType, deviceId, filter);
2020-01-29 16:30:27 +00:00
}
2020-01-28 12:19:25 +00:00
}
2016-10-29 22:22:20 +00:00
2020-01-29 16:30:27 +00:00
writer.WriteFullEndElement();
}
2020-01-28 12:19:25 +00:00
xmlWriter.WriteElementString("Result", builder.ToString());
}
2016-10-29 22:22:20 +00:00
2020-01-26 16:37:47 +00:00
xmlWriter.WriteElementString("NumberReturned", childrenResult.Items.Count.ToString(CultureInfo.InvariantCulture));
xmlWriter.WriteElementString("TotalMatches", childrenResult.TotalRecordCount.ToString(CultureInfo.InvariantCulture));
xmlWriter.WriteElementString("UpdateID", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the child items meeting the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="search">The <see cref="SearchCriteria"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{BaseItem}"/>.</returns>
private static QueryResult<BaseItem> GetChildrenSorted(BaseItem item, User user, SearchCriteria search, SortCriteria sort, int? startIndex, int? limit)
2016-10-29 22:22:20 +00:00
{
var folder = (Folder)item;
2020-01-26 16:37:47 +00:00
var sortOrders = folder.IsPreSorted
? Array.Empty<(string, SortOrder)>()
: new[] { (ItemSortBy.SortName, sort.SortOrder) };
2016-10-29 22:22:20 +00:00
string[] mediaTypes = Array.Empty<string>();
2016-10-29 22:22:20 +00:00
bool? isFolder = null;
if (search.SearchType == SearchType.Audio)
{
mediaTypes = new[] { MediaType.Audio };
2016-10-29 22:22:20 +00:00
isFolder = false;
}
else if (search.SearchType == SearchType.Video)
{
mediaTypes = new[] { MediaType.Video };
2016-10-29 22:22:20 +00:00
isFolder = false;
}
else if (search.SearchType == SearchType.Image)
{
mediaTypes = new[] { MediaType.Photo };
2016-10-29 22:22:20 +00:00
isFolder = false;
}
else if (search.SearchType == SearchType.Playlist)
{
2020-06-14 09:11:11 +00:00
// items = items.OfType<Playlist>();
2016-10-29 22:22:20 +00:00
isFolder = true;
}
else if (search.SearchType == SearchType.MusicAlbum)
{
2020-06-14 09:11:11 +00:00
// items = items.OfType<MusicAlbum>();
2016-10-29 22:22:20 +00:00
isFolder = true;
}
return folder.GetItems(new InternalItemsQuery
{
Limit = limit,
StartIndex = startIndex,
2019-10-20 14:08:40 +00:00
OrderBy = sortOrders,
2016-10-29 22:22:20 +00:00
User = user,
Recursive = true,
IsMissing = false,
2020-10-17 14:19:57 +00:00
ExcludeItemTypes = new[] { nameof(Book) },
2016-10-29 22:22:20 +00:00
IsFolder = isFolder,
MediaTypes = mediaTypes,
2017-05-21 07:25:49 +00:00
DtoOptions = GetDtoOptions()
2016-10-29 22:22:20 +00:00
});
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns a new DtoOptions object.
/// </summary>
/// <returns>The <see cref="DtoOptions"/>.</returns>
private static DtoOptions GetDtoOptions()
2017-05-21 07:25:49 +00:00
{
return new DtoOptions(true);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the User items meeting the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="stubType">The <see cref="StubType"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetUserItems(BaseItem item, StubType? stubType, User user, SortCriteria sort, int? startIndex, int? limit)
2016-10-29 22:22:20 +00:00
{
2016-12-20 19:59:25 +00:00
if (item is MusicGenre)
{
2018-09-12 17:26:21 +00:00
return GetMusicGenreItems(item, Guid.Empty, user, sort, startIndex, limit);
2016-12-20 19:59:25 +00:00
}
if (item is MusicArtist)
{
2018-09-12 17:26:21 +00:00
return GetMusicArtistItems(item, Guid.Empty, user, sort, startIndex, limit);
2016-12-20 19:59:25 +00:00
}
2017-08-19 19:43:35 +00:00
if (item is Genre)
{
2018-09-12 17:26:21 +00:00
return GetGenreItems(item, Guid.Empty, user, sort, startIndex, limit);
2017-08-19 19:43:35 +00:00
}
2019-01-26 17:21:41 +00:00
if ((!stubType.HasValue || stubType.Value != StubType.Folder)
&& item is IHasCollectionType collectionFolder)
2017-07-22 23:00:48 +00:00
{
2019-01-26 17:21:41 +00:00
if (string.Equals(CollectionType.Music, collectionFolder.CollectionType, StringComparison.OrdinalIgnoreCase))
2017-11-20 17:38:36 +00:00
{
return GetMusicFolders(item, user, stubType, sort, startIndex, limit);
}
2019-01-26 17:21:41 +00:00
else if (string.Equals(CollectionType.Movies, collectionFolder.CollectionType, StringComparison.OrdinalIgnoreCase))
2017-11-20 17:38:36 +00:00
{
return GetMovieFolders(item, user, stubType, sort, startIndex, limit);
}
2019-01-26 17:21:41 +00:00
else if (string.Equals(CollectionType.TvShows, collectionFolder.CollectionType, StringComparison.OrdinalIgnoreCase))
2017-11-20 17:38:36 +00:00
{
return GetTvFolders(item, user, stubType, sort, startIndex, limit);
}
2019-01-26 17:21:41 +00:00
else if (string.Equals(CollectionType.Folders, collectionFolder.CollectionType, StringComparison.OrdinalIgnoreCase))
2017-11-20 17:38:36 +00:00
{
2020-04-02 14:49:58 +00:00
return GetFolders(user, startIndex, limit);
2017-11-20 17:38:36 +00:00
}
2019-01-26 17:21:41 +00:00
else if (string.Equals(CollectionType.LiveTv, collectionFolder.CollectionType, StringComparison.OrdinalIgnoreCase))
2018-09-12 17:26:21 +00:00
{
2020-04-02 14:49:58 +00:00
return GetLiveTvChannels(user, sort, startIndex, limit);
2018-09-12 17:26:21 +00:00
}
}
2017-07-22 23:00:48 +00:00
2016-10-29 22:22:20 +00:00
if (stubType.HasValue)
{
2017-11-20 17:38:36 +00:00
if (stubType.Value != StubType.Folder)
{
return ApplyPaging(new QueryResult<ServerItem>(), startIndex, limit);
}
2016-10-29 22:22:20 +00:00
}
2016-12-20 07:31:30 +00:00
var folder = (Folder)item;
2018-09-12 17:26:21 +00:00
var query = new InternalItemsQuery(user)
2016-10-29 22:22:20 +00:00
{
Limit = limit,
StartIndex = startIndex,
2017-08-13 02:09:07 +00:00
IsVirtualItem = false,
2020-10-17 14:19:57 +00:00
ExcludeItemTypes = new[] { nameof(Book) },
2017-05-21 07:25:49 +00:00
IsPlaceHolder = false,
DtoOptions = GetDtoOptions()
2016-12-20 07:31:30 +00:00
};
SetSorting(query, sort, folder.IsPreSorted);
2017-05-26 06:48:54 +00:00
var queryResult = folder.GetItems(query);
2016-10-29 22:22:20 +00:00
2016-12-20 07:31:30 +00:00
return ToResult(queryResult);
}
2016-10-29 22:22:20 +00:00
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the Live Tv Channels meeting the criteria.
/// </summary>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetLiveTvChannels(User user, SortCriteria sort, int? startIndex, int? limit)
2018-09-12 17:26:21 +00:00
{
var query = new InternalItemsQuery(user)
{
StartIndex = startIndex,
Limit = limit,
};
2020-10-17 14:19:57 +00:00
query.IncludeItemTypes = new[] { nameof(LiveTvChannel) };
2018-09-12 17:26:21 +00:00
SetSorting(query, sort, false);
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the music folders meeting the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="stubType">The <see cref="StubType"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
2017-07-22 23:00:48 +00:00
{
var query = new InternalItemsQuery(user)
{
StartIndex = startIndex,
Limit = limit
};
SetSorting(query, sort, false);
if (stubType.HasValue && stubType.Value == StubType.Latest)
{
return GetMusicLatest(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Playlists)
{
2020-04-02 14:49:58 +00:00
return GetMusicPlaylists(user, query);
2017-07-22 23:00:48 +00:00
}
if (stubType.HasValue && stubType.Value == StubType.Albums)
{
return GetMusicAlbums(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Artists)
{
return GetMusicArtists(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.AlbumArtists)
{
return GetMusicAlbumArtists(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.FavoriteAlbums)
{
return GetFavoriteAlbums(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.FavoriteArtists)
{
return GetFavoriteArtists(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.FavoriteSongs)
{
return GetFavoriteSongs(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Songs)
{
return GetMusicSongs(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Genres)
{
return GetMusicGenres(item, user, query);
}
2020-09-13 13:31:12 +00:00
var list = new List<ServerItem>
2017-07-22 23:00:48 +00:00
{
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Latest
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Playlists
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Albums
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.AlbumArtists
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Artists
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Songs
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Genres
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.FavoriteArtists
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.FavoriteAlbums
},
2017-07-22 23:00:48 +00:00
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.FavoriteSongs
}
};
2017-07-22 23:00:48 +00:00
return new QueryResult<ServerItem>
{
Items = list,
2017-07-22 23:00:48 +00:00
TotalRecordCount = list.Count
};
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the movie folders meeting the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="stubType">The <see cref="StubType"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMovieFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
{
var query = new InternalItemsQuery(user)
{
StartIndex = startIndex,
Limit = limit
};
SetSorting(query, sort, false);
if (stubType.HasValue && stubType.Value == StubType.ContinueWatching)
{
return GetMovieContinueWatching(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Latest)
{
return GetMovieLatest(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Movies)
{
return GetMovieMovies(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Collections)
{
2020-04-02 14:49:58 +00:00
return GetMovieCollections(user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Favorites)
{
return GetMovieFavorites(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Genres)
{
return GetGenres(item, user, query);
}
2020-05-13 02:10:35 +00:00
var array = new[]
{
2020-04-02 14:49:58 +00:00
new ServerItem(item)
{
StubType = StubType.ContinueWatching
},
new ServerItem(item)
{
StubType = StubType.Latest
},
new ServerItem(item)
{
StubType = StubType.Movies
},
new ServerItem(item)
{
StubType = StubType.Collections
},
new ServerItem(item)
{
StubType = StubType.Favorites
},
new ServerItem(item)
{
StubType = StubType.Genres
}
};
return new QueryResult<ServerItem>
{
2020-04-02 14:49:58 +00:00
Items = array,
TotalRecordCount = array.Length
};
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the folders meeting the criteria.
/// </summary>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetFolders(User user, int? startIndex, int? limit)
2017-11-20 17:38:36 +00:00
{
2018-09-12 17:26:21 +00:00
var folders = _libraryManager.GetUserRootFolder().GetChildren(user, true)
2017-11-20 17:38:36 +00:00
.OrderBy(i => i.SortName)
.Select(i => new ServerItem(i)
{
StubType = StubType.Folder
})
.ToArray();
2020-08-20 19:04:57 +00:00
return ApplyPaging(
new QueryResult<ServerItem>
{
Items = folders,
TotalRecordCount = folders.Length
},
startIndex,
limit);
2017-11-20 17:38:36 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the TV folders meeting the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="stubType">The <see cref="StubType"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetTvFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
{
var query = new InternalItemsQuery(user)
{
StartIndex = startIndex,
Limit = limit
};
SetSorting(query, sort, false);
if (stubType.HasValue && stubType.Value == StubType.ContinueWatching)
{
return GetMovieContinueWatching(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.NextUp)
{
2020-04-02 14:49:58 +00:00
return GetNextUp(item, query);
}
if (stubType.HasValue && stubType.Value == StubType.Latest)
{
return GetTvLatest(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Series)
{
return GetSeries(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.FavoriteSeries)
{
return GetFavoriteSeries(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.FavoriteEpisodes)
{
return GetFavoriteEpisodes(item, user, query);
}
if (stubType.HasValue && stubType.Value == StubType.Genres)
{
return GetGenres(item, user, query);
}
2020-09-13 13:31:12 +00:00
var list = new List<ServerItem>
{
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.ContinueWatching
},
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.NextUp
},
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Latest
},
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Series
},
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.FavoriteSeries
},
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.FavoriteEpisodes
},
2020-09-13 13:31:12 +00:00
new ServerItem(item)
{
StubType = StubType.Genres
}
};
return new QueryResult<ServerItem>
{
Items = list,
TotalRecordCount = list.Count
};
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the Movies that are part watched that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMovieContinueWatching(BaseItem parent, User user, InternalItemsQuery query)
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
2019-10-20 14:08:40 +00:00
query.OrderBy = new[]
{
2019-10-20 14:08:40 +00:00
(ItemSortBy.DatePlayed, SortOrder.Descending),
(ItemSortBy.SortName, SortOrder.Ascending)
};
query.IsResumable = true;
query.Limit = 10;
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the series meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetSeries(BaseItem parent, User user, InternalItemsQuery query)
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
2020-10-17 14:19:57 +00:00
query.IncludeItemTypes = new[] { nameof(Series) };
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the Movie folders meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMovieMovies(BaseItem parent, User user, InternalItemsQuery query)
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
2020-10-17 14:19:57 +00:00
query.IncludeItemTypes = new[] { nameof(Movie) };
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the Movie collections meeting the criteria.
/// </summary>
/// <param name="user">The see cref="User"/>.</param>
/// <param name="query">The see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMovieCollections(User user, InternalItemsQuery query)
{
query.Recursive = true;
2020-06-14 09:11:11 +00:00
// query.Parent = parent;
query.SetUser(user);
2020-10-17 14:19:57 +00:00
query.IncludeItemTypes = new[] { nameof(BoxSet) };
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the Music albums meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicAlbums(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
2020-10-17 14:19:57 +00:00
query.IncludeItemTypes = new[] { nameof(MusicAlbum) };
2017-07-22 23:00:48 +00:00
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the Music songs meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicSongs(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
2020-10-17 14:19:57 +00:00
query.IncludeItemTypes = new[] { nameof(Audio) };
2017-07-22 23:00:48 +00:00
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the songs tagged as favourite that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetFavoriteSongs(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
query.IsFavorite = true;
2020-10-17 14:19:57 +00:00
query.IncludeItemTypes = new[] { nameof(Audio) };
2017-07-22 23:00:48 +00:00
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the series tagged as favourite that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetFavoriteSeries(BaseItem parent, User user, InternalItemsQuery query)
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
query.IsFavorite = true;
2020-10-17 14:27:31 +00:00
query.IncludeItemTypes = new[] { nameof(Series) };
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the episodes tagged as favourite that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetFavoriteEpisodes(BaseItem parent, User user, InternalItemsQuery query)
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
query.IsFavorite = true;
2020-10-17 14:27:31 +00:00
query.IncludeItemTypes = new[] { nameof(Episode) };
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the movies tagged as favourite that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMovieFavorites(BaseItem parent, User user, InternalItemsQuery query)
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
query.IsFavorite = true;
2020-10-17 14:27:31 +00:00
query.IncludeItemTypes = new[] { nameof(Movie) };
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// /// Returns the albums tagged as favourite that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetFavoriteAlbums(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
query.Recursive = true;
query.Parent = parent;
query.SetUser(user);
query.IsFavorite = true;
2020-10-17 14:27:31 +00:00
query.IncludeItemTypes = new[] { nameof(MusicAlbum) };
2017-07-22 23:00:48 +00:00
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the genres meeting the criteria.
/// The GetGenres.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetGenres(BaseItem parent, User user, InternalItemsQuery query)
{
var genresResult = _libraryManager.GetGenres(new InternalItemsQuery(user)
{
2018-09-12 17:26:21 +00:00
AncestorIds = new[] { parent.Id },
StartIndex = query.StartIndex,
Limit = query.Limit
});
var result = new QueryResult<BaseItem>
{
TotalRecordCount = genresResult.TotalRecordCount,
2018-12-28 15:48:26 +00:00
Items = genresResult.Items.Select(i => i.Item1).ToArray()
};
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the music genres meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicGenres(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
var genresResult = _libraryManager.GetMusicGenres(new InternalItemsQuery(user)
{
2018-09-12 17:26:21 +00:00
AncestorIds = new[] { parent.Id },
2017-07-22 23:00:48 +00:00
StartIndex = query.StartIndex,
Limit = query.Limit
});
var result = new QueryResult<BaseItem>
{
TotalRecordCount = genresResult.TotalRecordCount,
2018-12-28 15:48:26 +00:00
Items = genresResult.Items.Select(i => i.Item1).ToArray()
2017-07-22 23:00:48 +00:00
};
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the music albums by artist that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicAlbumArtists(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
var artists = _libraryManager.GetAlbumArtists(new InternalItemsQuery(user)
{
2018-09-12 17:26:21 +00:00
AncestorIds = new[] { parent.Id },
2017-07-22 23:00:48 +00:00
StartIndex = query.StartIndex,
Limit = query.Limit
});
var result = new QueryResult<BaseItem>
{
TotalRecordCount = artists.TotalRecordCount,
2018-12-28 15:48:26 +00:00
Items = artists.Items.Select(i => i.Item1).ToArray()
2017-07-22 23:00:48 +00:00
};
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the music artists meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicArtists(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
var artists = _libraryManager.GetArtists(new InternalItemsQuery(user)
{
2018-09-12 17:26:21 +00:00
AncestorIds = new[] { parent.Id },
2017-07-22 23:00:48 +00:00
StartIndex = query.StartIndex,
Limit = query.Limit
});
var result = new QueryResult<BaseItem>
{
TotalRecordCount = artists.TotalRecordCount,
2018-12-28 15:48:26 +00:00
Items = artists.Items.Select(i => i.Item1).ToArray()
2017-07-22 23:00:48 +00:00
};
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the artists tagged as favourite that meet the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetFavoriteArtists(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
var artists = _libraryManager.GetArtists(new InternalItemsQuery(user)
{
2018-09-12 17:26:21 +00:00
AncestorIds = new[] { parent.Id },
2017-07-22 23:00:48 +00:00
StartIndex = query.StartIndex,
Limit = query.Limit,
IsFavorite = true
});
var result = new QueryResult<BaseItem>
{
TotalRecordCount = artists.TotalRecordCount,
2018-12-28 15:48:26 +00:00
Items = artists.Items.Select(i => i.Item1).ToArray()
2017-07-22 23:00:48 +00:00
};
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the music playlists meeting the criteria.
/// </summary>
/// <param name="user">The user<see cref="User"/>.</param>
/// <param name="query">The query<see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicPlaylists(User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
query.Parent = null;
2020-05-13 02:10:35 +00:00
query.IncludeItemTypes = new[] { nameof(Playlist) };
2017-07-22 23:00:48 +00:00
query.SetUser(user);
query.Recursive = true;
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the latest music meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicLatest(BaseItem parent, User user, InternalItemsQuery query)
2017-07-22 23:00:48 +00:00
{
2019-10-20 14:08:40 +00:00
query.OrderBy = Array.Empty<(string, SortOrder)>();
2017-07-22 23:00:48 +00:00
2020-08-20 19:04:57 +00:00
var items = _userViewManager.GetLatestItems(
new LatestItemsQuery
{
UserId = user.Id,
Limit = 50,
IncludeItemTypes = new[] { nameof(Audio) },
ParentId = parent?.Id ?? Guid.Empty,
GroupItems = true
},
query.DtoOptions).Select(i => i.Item1 ?? i.Item2.FirstOrDefault()).Where(i => i != null).ToArray();
2017-07-22 23:00:48 +00:00
return ToResult(items);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the next up item meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-04-02 14:49:58 +00:00
private QueryResult<ServerItem> GetNextUp(BaseItem parent, InternalItemsQuery query)
{
2019-10-20 14:08:40 +00:00
query.OrderBy = Array.Empty<(string, SortOrder)>();
2020-08-20 19:04:57 +00:00
var result = _tvSeriesManager.GetNextUp(
new NextUpQuery
{
Limit = query.Limit,
StartIndex = query.StartIndex,
UserId = query.User.Id
},
new[] { parent },
query.DtoOptions);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the latest tv meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetTvLatest(BaseItem parent, User user, InternalItemsQuery query)
{
2019-10-20 14:08:40 +00:00
query.OrderBy = Array.Empty<(string, SortOrder)>();
2020-08-20 19:04:57 +00:00
var items = _userViewManager.GetLatestItems(
new LatestItemsQuery
{
UserId = user.Id,
Limit = 50,
2020-10-17 14:27:31 +00:00
IncludeItemTypes = new[] { nameof(Episode) },
2020-08-20 19:04:57 +00:00
ParentId = parent == null ? Guid.Empty : parent.Id,
GroupItems = false
},
query.DtoOptions).Select(i => i.Item1 ?? i.Item2.FirstOrDefault()).Where(i => i != null).ToArray();
return ToResult(items);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the latest movies meeting the criteria.
/// </summary>
/// <param name="parent">The <see cref="BaseItem"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMovieLatest(BaseItem parent, User user, InternalItemsQuery query)
{
2019-10-20 14:08:40 +00:00
query.OrderBy = Array.Empty<(string, SortOrder)>();
2020-05-13 02:10:35 +00:00
var items = _userViewManager.GetLatestItems(
new LatestItemsQuery
2020-08-20 19:04:57 +00:00
{
UserId = user.Id,
Limit = 50,
IncludeItemTypes = new[] { nameof(Movie) },
ParentId = parent?.Id ?? Guid.Empty,
GroupItems = true
},
query.DtoOptions).Select(i => i.Item1 ?? i.Item2.FirstOrDefault()).Where(i => i != null).ToArray();
return ToResult(items);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns music artist items that meet the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="parentId">The <see cref="Guid"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicArtistItems(BaseItem item, Guid parentId, User user, SortCriteria sort, int? startIndex, int? limit)
2016-12-20 19:59:25 +00:00
{
var query = new InternalItemsQuery(user)
{
Recursive = true,
ParentId = parentId,
2018-09-12 17:26:21 +00:00
ArtistIds = new[] { item.Id },
2020-10-17 14:27:31 +00:00
IncludeItemTypes = new[] { nameof(MusicAlbum) },
2016-12-20 19:59:25 +00:00
Limit = limit,
2017-05-21 07:25:49 +00:00
StartIndex = startIndex,
DtoOptions = GetDtoOptions()
2016-12-20 19:59:25 +00:00
};
SetSorting(query, sort, false);
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the genre items meeting the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="parentId">The <see cref="Guid"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetGenreItems(BaseItem item, Guid parentId, User user, SortCriteria sort, int? startIndex, int? limit)
2017-08-19 19:43:35 +00:00
{
var query = new InternalItemsQuery(user)
{
Recursive = true,
ParentId = parentId,
2018-09-12 17:26:21 +00:00
GenreIds = new[] { item.Id },
2020-05-13 02:10:35 +00:00
IncludeItemTypes = new[]
{
nameof(Movie),
nameof(Series)
},
2017-08-19 19:43:35 +00:00
Limit = limit,
StartIndex = startIndex,
DtoOptions = GetDtoOptions()
};
SetSorting(query, sort, false);
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Returns the music genre items meeting the criteria.
/// </summary>
/// <param name="item">The <see cref="BaseItem"/>.</param>
/// <param name="parentId">The <see cref="Guid"/>.</param>
/// <param name="user">The <see cref="User"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
2020-05-20 17:07:53 +00:00
private QueryResult<ServerItem> GetMusicGenreItems(BaseItem item, Guid parentId, User user, SortCriteria sort, int? startIndex, int? limit)
2016-12-20 07:31:30 +00:00
{
var query = new InternalItemsQuery(user)
{
Recursive = true,
ParentId = parentId,
2018-09-12 17:26:21 +00:00
GenreIds = new[] { item.Id },
2020-10-17 14:27:31 +00:00
IncludeItemTypes = new[] { nameof(MusicAlbum) },
2016-12-20 07:31:30 +00:00
Limit = limit,
2017-05-21 07:25:49 +00:00
StartIndex = startIndex,
DtoOptions = GetDtoOptions()
2016-12-20 07:31:30 +00:00
};
SetSorting(query, sort, false);
var result = _libraryManager.GetItemsResult(query);
return ToResult(result);
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Converts a <see cref="BaseItem"/> array into a <see cref="QueryResult{ServerItem}"/>.
/// </summary>
/// <param name="result">An array of <see cref="BaseItem"/>.</param>
/// <returns>A <see cref="QueryResult{ServerItem}"/>.</returns>
private static QueryResult<ServerItem> ToResult(BaseItem[] result)
2017-07-22 23:00:48 +00:00
{
var serverItems = result
.Select(i => new ServerItem(i))
2018-12-28 15:48:26 +00:00
.ToArray();
2017-07-22 23:00:48 +00:00
return new QueryResult<ServerItem>
{
2018-09-12 17:26:21 +00:00
TotalRecordCount = result.Length,
2017-07-22 23:00:48 +00:00
Items = serverItems
};
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Converts a <see cref="QueryResult{BaseItem}"/> to a <see cref="QueryResult{ServerItem}"/>.
/// </summary>
/// <param name="result">A <see cref="QueryResult{BaseItem}"/>.</param>
/// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
private static QueryResult<ServerItem> ToResult(QueryResult<BaseItem> result)
2016-12-20 07:31:30 +00:00
{
var serverItems = result
2016-10-29 22:22:20 +00:00
.Items
2016-12-20 19:59:25 +00:00
.Select(i => new ServerItem(i))
2018-09-12 17:26:21 +00:00
.ToArray();
2016-10-29 22:22:20 +00:00
return new QueryResult<ServerItem>
{
2016-12-20 07:31:30 +00:00
TotalRecordCount = result.TotalRecordCount,
2016-10-29 22:22:20 +00:00
Items = serverItems
};
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Sets the sorting method on a query.
/// </summary>
/// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
/// <param name="sort">The <see cref="SortCriteria"/>.</param>
/// <param name="isPreSorted">True if pre-sorted.</param>
private static void SetSorting(InternalItemsQuery query, SortCriteria sort, bool isPreSorted)
2016-12-20 07:31:30 +00:00
{
2019-10-20 14:08:40 +00:00
if (isPreSorted)
2016-12-20 07:31:30 +00:00
{
2019-10-20 14:08:40 +00:00
query.OrderBy = Array.Empty<(string, SortOrder)>();
}
2019-12-06 10:58:45 +00:00
else
2019-10-20 14:08:40 +00:00
{
query.OrderBy = new[] { (ItemSortBy.SortName, sort.SortOrder) };
2016-12-20 07:31:30 +00:00
}
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Apply paging to a query.
/// </summary>
/// <param name="result">The <see cref="QueryResult{ServerItem}"/>.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The maximum number to return.</param>
/// <returns>A <see cref="QueryResult{ServerItem}"/>.</returns>
private static QueryResult<ServerItem> ApplyPaging(QueryResult<ServerItem> result, int? startIndex, int? limit)
2016-10-29 22:22:20 +00:00
{
result.Items = result.Items.Skip(startIndex ?? 0).Take(limit ?? int.MaxValue).ToArray();
return result;
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Retreives the ServerItem id.
/// </summary>
/// <param name="id">The id<see cref="string"/>.</param>
/// <returns>The <see cref="ServerItem"/>.</returns>
2020-04-02 14:49:58 +00:00
private ServerItem GetItemFromObjectId(string id)
2016-10-29 22:22:20 +00:00
{
return DidlBuilder.IsIdRoot(id)
2018-09-12 17:26:21 +00:00
? new ServerItem(_libraryManager.GetUserRootFolder())
2020-04-02 14:49:58 +00:00
: ParseItemId(id);
2016-10-29 22:22:20 +00:00
}
2020-09-13 13:31:12 +00:00
/// <summary>
/// Parses the item id into a <see cref="ServerItem"/>.
/// </summary>
/// <param name="id">The <see cref="string"/>.</param>
/// <returns>The corresponding <see cref="ServerItem"/>.</returns>
2020-04-02 14:49:58 +00:00
private ServerItem ParseItemId(string id)
2016-10-29 22:22:20 +00:00
{
StubType? stubType = null;
// After using PlayTo, MediaMonkey sends a request to the server trying to get item info
const string ParamsSrch = "Params=";
var paramsIndex = id.IndexOf(ParamsSrch, StringComparison.OrdinalIgnoreCase);
2016-10-29 22:22:20 +00:00
if (paramsIndex != -1)
{
id = id.Substring(paramsIndex + ParamsSrch.Length);
2016-10-29 22:22:20 +00:00
var parts = id.Split(';');
id = parts[23];
}
var enumNames = Enum.GetNames(typeof(StubType));
foreach (var name in enumNames)
2016-10-29 22:22:20 +00:00
{
if (id.StartsWith(name + "_", StringComparison.OrdinalIgnoreCase))
{
2020-11-06 15:15:30 +00:00
stubType = Enum.Parse<StubType>(name, true);
id = id.Split('_', 2)[1];
break;
}
2017-07-22 23:00:48 +00:00
}
2016-10-29 22:22:20 +00:00
if (Guid.TryParse(id, out var itemId))
2016-10-29 22:22:20 +00:00
{
var item = _libraryManager.GetItemById(itemId);
2016-12-20 19:59:25 +00:00
return new ServerItem(item)
2016-10-29 22:22:20 +00:00
{
StubType = stubType
};
}
2020-08-31 20:20:19 +00:00
Logger.LogError("Error parsing item Id: {Id}. Returning user root folder.", id);
2016-10-29 22:22:20 +00:00
2018-09-12 17:26:21 +00:00
return new ServerItem(_libraryManager.GetUserRootFolder());
2016-10-29 22:22:20 +00:00
}
}
}