jellyfin/Emby.Server.Implementations/Playlists/PlaylistManager.cs

568 lines
21 KiB
C#
Raw Normal View History

#nullable disable
#pragma warning disable CS1591
2014-08-01 22:34:45 -04:00
using System;
using System.Collections.Generic;
using System.Globalization;
2014-08-01 22:34:45 -04:00
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
2020-05-20 13:07:53 -04:00
using Jellyfin.Data.Entities;
using Jellyfin.Data.Enums;
using Jellyfin.Extensions;
2017-05-21 03:25:49 -04:00
using MediaBrowser.Controller.Dto;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Extensions;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Playlists;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
2016-10-25 15:02:04 -04:00
using MediaBrowser.Model.IO;
using MediaBrowser.Model.Playlists;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Logging;
2018-09-12 13:26:21 -04:00
using PlaylistsNET.Content;
using PlaylistsNET.Models;
2020-05-20 13:07:53 -04:00
using Genre = MediaBrowser.Controller.Entities.Genre;
using MusicAlbum = MediaBrowser.Controller.Entities.Audio.MusicAlbum;
2014-08-01 22:34:45 -04:00
namespace Emby.Server.Implementations.Playlists
2014-08-01 22:34:45 -04:00
{
public class PlaylistManager : IPlaylistManager
{
private readonly ILibraryManager _libraryManager;
private readonly IFileSystem _fileSystem;
private readonly ILibraryMonitor _iLibraryMonitor;
2020-06-05 20:15:56 -04:00
private readonly ILogger<PlaylistManager> _logger;
2014-08-01 22:34:45 -04:00
private readonly IUserManager _userManager;
2015-07-24 11:20:11 -04:00
private readonly IProviderManager _providerManager;
private readonly IConfiguration _appConfig;
2014-08-01 22:34:45 -04:00
public PlaylistManager(
ILibraryManager libraryManager,
IFileSystem fileSystem,
ILibraryMonitor iLibraryMonitor,
ILogger<PlaylistManager> logger,
IUserManager userManager,
IProviderManager providerManager,
IConfiguration appConfig)
2014-08-01 22:34:45 -04:00
{
_libraryManager = libraryManager;
_fileSystem = fileSystem;
_iLibraryMonitor = iLibraryMonitor;
_logger = logger;
2014-08-01 22:34:45 -04:00
_userManager = userManager;
2015-07-24 11:20:11 -04:00
_providerManager = providerManager;
_appConfig = appConfig;
2014-08-01 22:34:45 -04:00
}
2018-09-12 13:26:21 -04:00
public IEnumerable<Playlist> GetPlaylists(Guid userId)
2014-08-01 22:34:45 -04:00
{
2014-09-14 11:10:51 -04:00
var user = _userManager.GetUserById(userId);
2014-08-01 22:34:45 -04:00
return GetPlaylistsFolder(userId).GetChildren(user, true).OfType<Playlist>();
}
2014-08-21 11:55:35 -04:00
public async Task<PlaylistCreationResult> CreatePlaylist(PlaylistCreationRequest options)
2014-08-01 22:34:45 -04:00
{
var name = options.Name;
2019-12-13 22:51:27 -05:00
var folderName = _fileSystem.GetValidFilename(name);
var parentFolder = GetPlaylistsFolder(options.UserId);
2022-12-05 09:00:20 -05:00
if (parentFolder is null)
2014-08-01 22:34:45 -04:00
{
2021-12-24 16:18:24 -05:00
throw new ArgumentException(nameof(parentFolder));
2014-08-01 22:34:45 -04:00
}
if (options.MediaType is null || options.MediaType == MediaType.Unknown)
{
foreach (var itemId in options.ItemIdList)
{
var item = _libraryManager.GetItemById(itemId);
2022-12-05 09:00:20 -05:00
if (item is null)
{
throw new ArgumentException("No item exists with the supplied Id");
}
2023-10-14 13:01:03 -04:00
if (item.MediaType != MediaType.Unknown)
{
options.MediaType = item.MediaType;
}
else if (item is MusicArtist || item is MusicAlbum || item is MusicGenre)
{
options.MediaType = MediaType.Audio;
}
else if (item is Genre)
{
options.MediaType = MediaType.Video;
}
else
{
2019-09-10 16:37:53 -04:00
if (item is Folder folder)
{
2015-01-25 01:34:50 -05:00
options.MediaType = folder.GetRecursiveChildren(i => !i.IsFolder && i.SupportsAddingToPlaylist)
.Select(i => i.MediaType)
.FirstOrDefault(i => i != MediaType.Unknown);
}
}
2023-10-14 13:01:03 -04:00
if (options.MediaType is not null && options.MediaType != MediaType.Unknown)
{
break;
}
}
}
if (options.MediaType is null || options.MediaType == MediaType.Unknown)
{
options.MediaType = MediaType.Audio;
}
2014-09-14 11:10:51 -04:00
var user = _userManager.GetUserById(options.UserId);
2014-08-01 22:34:45 -04:00
var path = Path.Combine(parentFolder.Path, folderName);
path = GetTargetPath(path);
2014-08-01 22:34:45 -04:00
_iLibraryMonitor.ReportFileSystemChangeBeginning(path);
try
{
Directory.CreateDirectory(path);
var playlist = new Playlist
2014-08-01 22:34:45 -04:00
{
Name = name,
2018-09-12 13:26:21 -04:00
Path = path,
2023-03-10 11:46:59 -05:00
OwnerUserId = options.UserId,
Shares = options.Shares ?? Array.Empty<Share>()
2014-08-01 22:34:45 -04:00
};
playlist.SetMediaType(options.MediaType);
parentFolder.AddChild(playlist);
2014-08-01 22:34:45 -04:00
2019-09-10 16:37:53 -04:00
await playlist.RefreshMetadata(new MetadataRefreshOptions(new DirectoryService(_fileSystem)) { ForceSave = true }, CancellationToken.None)
2014-08-01 22:34:45 -04:00
.ConfigureAwait(false);
if (options.ItemIdList.Count > 0)
2014-08-01 22:34:45 -04:00
{
2020-08-21 16:19:16 -04:00
await AddToPlaylistInternal(playlist.Id, options.ItemIdList, user, new DtoOptions(false)
2017-05-21 03:25:49 -04:00
{
EnableImages = true
2020-08-21 16:19:16 -04:00
}).ConfigureAwait(false);
2014-08-01 22:34:45 -04:00
}
return new PlaylistCreationResult(playlist.Id.ToString("N", CultureInfo.InvariantCulture));
2014-08-01 22:34:45 -04:00
}
finally
{
// Refresh handled internally
_iLibraryMonitor.ReportFileSystemChangeComplete(path, false);
}
}
private string GetTargetPath(string path)
{
while (Directory.Exists(path))
{
path += "1";
}
return path;
}
private List<BaseItem> GetPlaylistItems(IEnumerable<Guid> itemIds, MediaType playlistMediaType, User user, DtoOptions options)
{
2022-12-05 09:01:13 -05:00
var items = itemIds.Select(i => _libraryManager.GetItemById(i)).Where(i => i is not null);
2017-05-21 03:25:49 -04:00
return Playlist.GetPlaylistItems(playlistMediaType, items, user, options);
2014-08-21 22:24:38 -04:00
}
public Task AddToPlaylistAsync(Guid playlistId, IReadOnlyCollection<Guid> itemIds, Guid userId)
2014-08-21 22:24:38 -04:00
{
var user = userId.IsEmpty() ? null : _userManager.GetUserById(userId);
2014-08-21 22:24:38 -04:00
2020-08-21 16:01:19 -04:00
return AddToPlaylistInternal(playlistId, itemIds, user, new DtoOptions(false)
2017-05-21 03:25:49 -04:00
{
EnableImages = true
});
}
private async Task AddToPlaylistInternal(Guid playlistId, IReadOnlyCollection<Guid> newItemIds, User user, DtoOptions options)
2014-08-01 22:34:45 -04:00
{
// Retrieve the existing playlist
var playlist = _libraryManager.GetItemById(playlistId) as Playlist
?? throw new ArgumentException("No Playlist exists with Id " + playlistId);
2014-08-01 22:34:45 -04:00
// Retrieve all the items to be added to the playlist
var newItems = GetPlaylistItems(newItemIds, playlist.MediaType, user, options)
.Where(i => i.SupportsAddingToPlaylist);
// Filter out duplicate items, if necessary
if (!_appConfig.DoPlaylistsAllowDuplicates())
{
var existingIds = playlist.LinkedChildren.Select(c => c.ItemId).ToHashSet();
newItems = newItems
.Where(i => !existingIds.Contains(i.Id))
.Distinct();
}
2014-08-01 22:34:45 -04:00
// Create a list of the new linked children to add to the playlist
var childrenToAdd = newItems
2021-04-17 06:19:09 -04:00
.Select(LinkedChild.Create)
.ToList();
// Log duplicates that have been ignored, if any
int numDuplicates = newItemIds.Count - childrenToAdd.Count;
if (numDuplicates > 0)
2014-08-06 22:51:09 -04:00
{
_logger.LogWarning("Ignored adding {DuplicateCount} duplicate items to playlist {PlaylistName}.", numDuplicates, playlist.Name);
2014-08-01 22:34:45 -04:00
}
// Do nothing else if there are no items to add to the playlist
if (childrenToAdd.Count == 0)
{
return;
}
// Create a new array with the updated playlist items
var newLinkedChildren = new LinkedChild[playlist.LinkedChildren.Length + childrenToAdd.Count];
playlist.LinkedChildren.CopyTo(newLinkedChildren, 0);
childrenToAdd.CopyTo(newLinkedChildren, playlist.LinkedChildren.Length);
// Update the playlist in the repository
playlist.LinkedChildren = newLinkedChildren;
2020-08-21 16:01:19 -04:00
await playlist.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
2015-07-24 11:20:11 -04:00
// Update the playlist on disk
2018-09-12 13:26:21 -04:00
if (playlist.IsFile)
{
SavePlaylistFile(playlist);
}
// Refresh playlist metadata
2019-09-10 16:37:53 -04:00
_providerManager.QueueRefresh(
playlist.Id,
2020-03-03 08:40:07 -05:00
new MetadataRefreshOptions(new DirectoryService(_fileSystem))
{
ForceSave = true
},
2019-09-10 16:37:53 -04:00
RefreshPriority.High);
2014-08-01 22:34:45 -04:00
}
2020-08-21 16:01:19 -04:00
public async Task RemoveFromPlaylistAsync(string playlistId, IEnumerable<string> entryIds)
2014-08-01 22:34:45 -04:00
{
2021-08-28 18:32:50 -04:00
if (_libraryManager.GetItemById(playlistId) is not Playlist playlist)
2014-08-06 00:18:13 -04:00
{
throw new ArgumentException("No Playlist exists with the supplied Id");
}
2014-08-11 19:41:11 -04:00
var children = playlist.GetManageableItems().ToList();
2014-08-06 00:18:13 -04:00
var idList = entryIds.ToList();
2014-08-11 19:41:11 -04:00
var removals = children.Where(i => idList.Contains(i.Item1.Id));
2014-08-06 00:18:13 -04:00
playlist.LinkedChildren = children.Except(removals)
2014-08-11 19:41:11 -04:00
.Select(i => i.Item1)
2017-08-10 14:01:31 -04:00
.ToArray();
2014-08-06 00:18:13 -04:00
2020-08-21 16:01:19 -04:00
await playlist.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
2015-07-24 11:20:11 -04:00
2018-09-12 13:26:21 -04:00
if (playlist.IsFile)
{
SavePlaylistFile(playlist);
}
2019-12-13 22:51:27 -05:00
_providerManager.QueueRefresh(
playlist.Id,
new MetadataRefreshOptions(new DirectoryService(_fileSystem))
{
ForceSave = true
},
RefreshPriority.High);
2014-08-01 22:34:45 -04:00
}
2020-08-21 16:01:19 -04:00
public async Task MoveItemAsync(string playlistId, string entryId, int newIndex)
{
2021-08-28 18:32:50 -04:00
if (_libraryManager.GetItemById(playlistId) is not Playlist playlist)
{
throw new ArgumentException("No Playlist exists with the supplied Id");
}
var children = playlist.GetManageableItems().ToList();
var oldIndex = children.FindIndex(i => string.Equals(entryId, i.Item1.Id, StringComparison.OrdinalIgnoreCase));
2015-10-15 11:51:00 -04:00
if (oldIndex == newIndex)
{
return;
}
var item = playlist.LinkedChildren[oldIndex];
2017-08-10 14:01:31 -04:00
var newList = playlist.LinkedChildren.ToList();
2016-07-16 14:02:39 -04:00
2017-08-10 14:01:31 -04:00
newList.Remove(item);
if (newIndex >= newList.Count)
2016-07-16 14:02:39 -04:00
{
2017-08-10 14:01:31 -04:00
newList.Add(item);
2016-07-16 14:02:39 -04:00
}
else
{
2017-08-10 14:01:31 -04:00
newList.Insert(newIndex, item);
2016-07-16 14:02:39 -04:00
}
2018-12-28 10:48:26 -05:00
playlist.LinkedChildren = newList.ToArray();
2017-08-10 14:01:31 -04:00
2020-08-21 16:01:19 -04:00
await playlist.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
2018-09-12 13:26:21 -04:00
if (playlist.IsFile)
{
SavePlaylistFile(playlist);
}
}
/// <inheritdoc />
public void SavePlaylistFile(Playlist item)
2018-09-12 13:26:21 -04:00
{
2019-12-13 22:51:27 -05:00
// this is probably best done as a metadata provider
// saving a file over itself will require some work to prevent this from happening when not needed
2018-09-12 13:26:21 -04:00
var playlistPath = item.Path;
var extension = Path.GetExtension(playlistPath.AsSpan());
2018-09-12 13:26:21 -04:00
if (extension.Equals(".wpl", StringComparison.OrdinalIgnoreCase))
2018-09-12 13:26:21 -04:00
{
var playlist = new WplPlaylist();
foreach (var child in item.GetLinkedChildren())
{
var entry = new WplPlaylistEntry()
{
Path = NormalizeItemPath(playlistPath, child.Path),
TrackTitle = child.Name,
AlbumTitle = child.Album
};
2020-07-24 10:37:54 -04:00
if (child is IHasAlbumArtist hasAlbumArtist)
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
2018-09-12 13:26:21 -04:00
}
2020-07-24 10:37:54 -04:00
if (child is IHasArtist hasArtist)
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
entry.TrackArtist = hasArtist.Artists.Count > 0 ? hasArtist.Artists[0] : null;
2018-09-12 13:26:21 -04:00
}
if (child.RunTimeTicks.HasValue)
{
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
}
2019-12-13 22:51:27 -05:00
2018-09-12 13:26:21 -04:00
playlist.PlaylistEntries.Add(entry);
}
string text = new WplContent().ToText(playlist);
File.WriteAllText(playlistPath, text);
2018-09-12 13:26:21 -04:00
}
else if (extension.Equals(".zpl", StringComparison.OrdinalIgnoreCase))
2018-09-12 13:26:21 -04:00
{
var playlist = new ZplPlaylist();
foreach (var child in item.GetLinkedChildren())
{
var entry = new ZplPlaylistEntry()
{
Path = NormalizeItemPath(playlistPath, child.Path),
TrackTitle = child.Name,
AlbumTitle = child.Album
};
2020-07-24 10:37:54 -04:00
if (child is IHasAlbumArtist hasAlbumArtist)
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
2018-09-12 13:26:21 -04:00
}
2020-07-24 10:37:54 -04:00
if (child is IHasArtist hasArtist)
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
entry.TrackArtist = hasArtist.Artists.Count > 0 ? hasArtist.Artists[0] : null;
2018-09-12 13:26:21 -04:00
}
if (child.RunTimeTicks.HasValue)
{
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
}
2020-06-15 17:43:52 -04:00
2018-09-12 13:26:21 -04:00
playlist.PlaylistEntries.Add(entry);
}
string text = new ZplContent().ToText(playlist);
File.WriteAllText(playlistPath, text);
2018-09-12 13:26:21 -04:00
}
else if (extension.Equals(".m3u", StringComparison.OrdinalIgnoreCase))
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
var playlist = new M3uPlaylist
{
IsExtended = true
};
2018-09-12 13:26:21 -04:00
foreach (var child in item.GetLinkedChildren())
{
var entry = new M3uPlaylistEntry()
{
Path = NormalizeItemPath(playlistPath, child.Path),
Title = child.Name,
Album = child.Album
};
2020-07-24 10:37:54 -04:00
if (child is IHasAlbumArtist hasAlbumArtist)
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
2018-09-12 13:26:21 -04:00
}
if (child.RunTimeTicks.HasValue)
{
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
}
2019-12-13 22:51:27 -05:00
2018-09-12 13:26:21 -04:00
playlist.PlaylistEntries.Add(entry);
}
string text = new M3uContent().ToText(playlist);
File.WriteAllText(playlistPath, text);
2018-09-12 13:26:21 -04:00
}
else if (extension.Equals(".m3u8", StringComparison.OrdinalIgnoreCase))
2018-09-12 13:26:21 -04:00
{
var playlist = new M3uPlaylist();
playlist.IsExtended = true;
foreach (var child in item.GetLinkedChildren())
{
var entry = new M3uPlaylistEntry()
{
Path = NormalizeItemPath(playlistPath, child.Path),
Title = child.Name,
Album = child.Album
};
2020-07-24 10:37:54 -04:00
if (child is IHasAlbumArtist hasAlbumArtist)
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
2018-09-12 13:26:21 -04:00
}
if (child.RunTimeTicks.HasValue)
{
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
}
2019-12-13 22:51:27 -05:00
2018-09-12 13:26:21 -04:00
playlist.PlaylistEntries.Add(entry);
}
2020-06-15 10:34:24 -04:00
string text = new M3uContent().ToText(playlist);
File.WriteAllText(playlistPath, text);
2018-09-12 13:26:21 -04:00
}
else if (extension.Equals(".pls", StringComparison.OrdinalIgnoreCase))
2018-09-12 13:26:21 -04:00
{
var playlist = new PlsPlaylist();
foreach (var child in item.GetLinkedChildren())
{
var entry = new PlsPlaylistEntry()
{
Path = NormalizeItemPath(playlistPath, child.Path),
Title = child.Name
};
if (child.RunTimeTicks.HasValue)
{
entry.Length = TimeSpan.FromTicks(child.RunTimeTicks.Value);
}
2019-12-13 22:51:27 -05:00
2018-09-12 13:26:21 -04:00
playlist.PlaylistEntries.Add(entry);
}
string text = new PlsContent().ToText(playlist);
File.WriteAllText(playlistPath, text);
2018-09-12 13:26:21 -04:00
}
}
private string NormalizeItemPath(string playlistPath, string itemPath)
{
return MakeRelativePath(Path.GetDirectoryName(playlistPath), itemPath);
2018-09-12 13:26:21 -04:00
}
private static string MakeRelativePath(string folderPath, string fileAbsolutePath)
2018-09-12 13:26:21 -04:00
{
ArgumentException.ThrowIfNullOrEmpty(folderPath);
ArgumentException.ThrowIfNullOrEmpty(fileAbsolutePath);
2018-09-12 13:26:21 -04:00
2019-12-13 22:51:27 -05:00
if (!folderPath.EndsWith(Path.DirectorySeparatorChar))
2018-09-12 13:26:21 -04:00
{
2020-07-24 10:37:54 -04:00
folderPath += Path.DirectorySeparatorChar;
2018-09-12 13:26:21 -04:00
}
2019-01-13 15:37:13 -05:00
var folderUri = new Uri(folderPath);
var fileAbsoluteUri = new Uri(fileAbsolutePath);
2018-09-12 13:26:21 -04:00
2019-12-13 22:51:27 -05:00
// path can't be made relative
if (folderUri.Scheme != fileAbsoluteUri.Scheme)
{
return fileAbsolutePath;
}
2018-09-12 13:26:21 -04:00
2019-01-13 15:37:13 -05:00
var relativeUri = folderUri.MakeRelativeUri(fileAbsoluteUri);
string relativePath = Uri.UnescapeDataString(relativeUri.ToString());
2018-09-12 13:26:21 -04:00
2021-11-15 09:57:07 -05:00
if (fileAbsoluteUri.Scheme.Equals("file", StringComparison.OrdinalIgnoreCase))
2018-09-12 13:26:21 -04:00
{
relativePath = relativePath.Replace(Path.AltDirectorySeparatorChar, Path.DirectorySeparatorChar);
}
return relativePath;
}
public Folder GetPlaylistsFolder()
{
return GetPlaylistsFolder(Guid.Empty);
}
2018-09-12 13:26:21 -04:00
public Folder GetPlaylistsFolder(Guid userId)
2014-08-01 22:34:45 -04:00
{
2020-07-24 10:37:54 -04:00
const string TypeName = "PlaylistsFolder";
2020-07-24 10:37:54 -04:00
return _libraryManager.RootFolder.Children.OfType<Folder>().FirstOrDefault(i => string.Equals(i.GetType().Name, TypeName, StringComparison.Ordinal)) ??
_libraryManager.GetUserRootFolder().Children.OfType<Folder>().FirstOrDefault(i => string.Equals(i.GetType().Name, TypeName, StringComparison.Ordinal));
2014-08-01 22:34:45 -04:00
}
2023-03-10 11:46:59 -05:00
/// <inheritdoc />
2023-03-12 14:42:18 -04:00
public async Task RemovePlaylistsAsync(Guid userId)
2023-03-10 11:46:59 -05:00
{
var playlists = GetPlaylists(userId);
foreach (var playlist in playlists)
{
// Update owner if shared
var rankedShares = playlist.Shares.OrderByDescending(x => x.CanEdit).ToArray();
if (rankedShares.Length > 0 && Guid.TryParse(rankedShares[0].UserId, out var guid))
{
playlist.OwnerUserId = guid;
playlist.Shares = rankedShares.Skip(1).ToArray();
await playlist.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
if (playlist.IsFile)
{
SavePlaylistFile(playlist);
}
}
else if (!playlist.OpenAccess)
2023-03-10 11:46:59 -05:00
{
// Remove playlist if not shared
_libraryManager.DeleteItem(
playlist,
new DeleteOptions
{
DeleteFileLocation = false,
DeleteFromExternalProvider = false
},
playlist.GetParent(),
false);
}
}
}
2014-08-01 22:34:45 -04:00
}
}