2024-06-27 01:47:44 +02:00
|
|
|
using MediaBrowser.Model.Tasks;
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
using MediaBrowser.Controller;
|
|
|
|
using MediaBrowser.Controller.Library;
|
|
|
|
using MediaBrowser.Controller.Playlists;
|
|
|
|
using MediaBrowser.Controller.Providers;
|
|
|
|
using MediaBrowser.Model.IO;
|
2024-06-29 18:29:40 +02:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Jellyfin.Data.Entities;
|
|
|
|
using Jellyfin.Data.Enums;
|
|
|
|
using MediaBrowser.Controller;
|
|
|
|
using MediaBrowser.Controller.Entities;
|
|
|
|
using MediaBrowser.Controller.Library;
|
|
|
|
using MediaBrowser.Controller.Playlists;
|
|
|
|
using MediaBrowser.Controller.Providers;
|
|
|
|
using MediaBrowser.Model.IO;
|
|
|
|
using MediaBrowser.Model.Playlists;
|
|
|
|
using MediaBrowser.Model.Tasks;
|
|
|
|
using Microsoft.Extensions.Logging;
|
2024-06-27 01:47:44 +02:00
|
|
|
|
2024-10-24 23:53:21 +02:00
|
|
|
using Jellyfin.Plugin.SmartPlaylist.Lisp;
|
|
|
|
using Jellyfin.Plugin.SmartPlaylist.Lisp.Compiler;
|
|
|
|
using Lisp_Object = Jellyfin.Plugin.SmartPlaylist.Lisp.Compiler.Object;
|
|
|
|
using Lisp_Boolean = Jellyfin.Plugin.SmartPlaylist.Lisp.Compiler.Boolean;
|
|
|
|
|
|
|
|
|
2024-06-27 01:47:44 +02:00
|
|
|
namespace Jellyfin.Plugin.SmartPlaylist.ScheduledTasks {
|
|
|
|
public class GeneratePlaylist : IScheduledTask {
|
2024-10-24 23:53:21 +02:00
|
|
|
|
|
|
|
public static readonly BaseItemKind[] AvailableFilterItems = {
|
|
|
|
BaseItemKind.Audio
|
|
|
|
};
|
|
|
|
|
2024-06-27 01:47:44 +02:00
|
|
|
private readonly ILogger _logger;
|
2024-06-29 18:29:40 +02:00
|
|
|
private readonly ILibraryManager _libraryManager;
|
|
|
|
private readonly IUserManager _userManager;
|
2024-10-25 02:18:13 +02:00
|
|
|
private readonly IProviderManager _providerManager;
|
|
|
|
private readonly IFileSystem _fileSystem;
|
2024-10-24 23:53:21 +02:00
|
|
|
private readonly IPlaylistManager _playlistManager;
|
|
|
|
|
|
|
|
private readonly IStore _store;
|
2024-06-29 18:29:40 +02:00
|
|
|
|
2024-06-27 01:47:44 +02:00
|
|
|
public GeneratePlaylist(
|
2024-06-29 18:29:40 +02:00
|
|
|
ILogger<Plugin> logger,
|
|
|
|
ILibraryManager libraryManager,
|
2024-10-24 23:53:21 +02:00
|
|
|
IUserManager userManager,
|
2024-10-25 02:18:13 +02:00
|
|
|
IProviderManager providerManager,
|
|
|
|
IFileSystem fileSystem,
|
2024-10-24 23:53:21 +02:00
|
|
|
IPlaylistManager playlistManager,
|
|
|
|
IServerApplicationPaths serverApplicationPaths
|
2024-06-27 01:47:44 +02:00
|
|
|
) {
|
|
|
|
_logger = logger;
|
2024-06-29 18:29:40 +02:00
|
|
|
_libraryManager = libraryManager;
|
|
|
|
_userManager = userManager;
|
2024-10-25 02:18:13 +02:00
|
|
|
_providerManager = providerManager;
|
|
|
|
_fileSystem = fileSystem;
|
2024-10-24 23:53:21 +02:00
|
|
|
_playlistManager = playlistManager;
|
|
|
|
|
2024-10-25 02:18:13 +02:00
|
|
|
_store = new Store(new SmartPlaylistFileSystem(serverApplicationPaths));
|
2024-06-27 01:47:44 +02:00
|
|
|
}
|
2024-06-29 18:29:40 +02:00
|
|
|
|
2024-06-27 01:47:44 +02:00
|
|
|
public string Category => "Library";
|
|
|
|
public string Name => "(re)generate Smart Playlists";
|
|
|
|
public string Description => "Generate or regenerate all Smart Playlists";
|
|
|
|
public string Key => nameof(GeneratePlaylist);
|
2024-06-29 18:29:40 +02:00
|
|
|
|
2024-06-27 01:47:44 +02:00
|
|
|
public IEnumerable<TaskTriggerInfo> GetDefaultTriggers() {
|
|
|
|
return new[] {
|
|
|
|
new TaskTriggerInfo {
|
2024-10-24 23:53:21 +02:00
|
|
|
IntervalTicks = TimeSpan.FromHours(24).Ticks,
|
2024-06-27 01:47:44 +02:00
|
|
|
Type = TaskTriggerInfo.TriggerInterval,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2024-06-29 18:29:40 +02:00
|
|
|
|
|
|
|
private void GetUsers() {
|
|
|
|
foreach (var user in _userManager.Users) {
|
|
|
|
_logger.LogInformation("User {0}", user);
|
|
|
|
var query = new InternalItemsQuery(user) {
|
2024-10-24 23:53:21 +02:00
|
|
|
IncludeItemTypes = AvailableFilterItems,
|
2024-06-29 18:29:40 +02:00
|
|
|
Recursive = true,
|
|
|
|
};
|
|
|
|
foreach (BaseItem item in _libraryManager.GetItemsResult(query).Items) {
|
|
|
|
_logger.LogInformation("Item {0}", item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-25 02:18:13 +02:00
|
|
|
private SmartPlaylistId CreateNewPlaylist(SmartPlaylistDto dto) {
|
2024-10-24 23:53:21 +02:00
|
|
|
var req = new PlaylistCreationRequest {
|
|
|
|
Name = dto.Name,
|
2024-10-25 02:18:13 +02:00
|
|
|
UserId = dto.User
|
2024-10-24 23:53:21 +02:00
|
|
|
};
|
2024-10-25 02:18:13 +02:00
|
|
|
var playlistGuid = Guid.Parse(_playlistManager.CreatePlaylist(req).Result.Id);
|
|
|
|
return playlistGuid;
|
2024-10-24 23:53:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private IEnumerable<Guid> FilterPlaylistItems(IEnumerable<BaseItem> items, User user, SmartPlaylistDto smartPlaylist) {
|
|
|
|
List<Guid> results = new List<Guid>();
|
|
|
|
Expression expression = new Parser(StringTokenStream.generate(smartPlaylist.Program)).parse();
|
|
|
|
Executor executor = new Executor();
|
2024-10-25 02:18:13 +02:00
|
|
|
executor.environment["user"] = new Lisp_Object(user);
|
2024-10-24 23:53:21 +02:00
|
|
|
foreach (var i in items) {
|
|
|
|
executor.environment["item"] = new Lisp_Object(i);
|
|
|
|
var r = executor.eval(expression);
|
2024-10-25 02:18:13 +02:00
|
|
|
_logger.LogDebug("Item {0} evaluated to {1}", i, r.ToString());
|
2024-10-24 23:53:21 +02:00
|
|
|
if (r is Lisp_Boolean r_bool) {
|
2024-10-25 02:18:13 +02:00
|
|
|
if (r_bool.value) {
|
|
|
|
_logger.LogDebug("Added "{0}" to Smart Playlist {1}", i, smartPlaylist.Name);
|
|
|
|
results.Add(i.Id);
|
|
|
|
}
|
2024-10-24 23:53:21 +02:00
|
|
|
} else {
|
|
|
|
_logger.LogInformation("Program did not return a boolean, returned {0}", r.ToString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
private IEnumerable<BaseItem> GetAllUserMedia(User user) {
|
|
|
|
var req = new InternalItemsQuery(user) {
|
|
|
|
IncludeItemTypes = new[] {BaseItemKind.Audio},
|
|
|
|
Recursive = true,
|
|
|
|
};
|
|
|
|
return _libraryManager.GetItemsResult(req).Items;
|
|
|
|
}
|
|
|
|
|
2024-06-27 01:47:44 +02:00
|
|
|
public async Task ExecuteAsync(IProgress<double> progress, CancellationToken cancellationToken) {
|
2024-10-24 23:53:21 +02:00
|
|
|
_logger.LogInformation("Started regenerate Smart Playlists");
|
|
|
|
foreach (SmartPlaylistDto dto in await _store.GetAllSmartPlaylistsAsync()) {
|
|
|
|
var user = _userManager.GetUserById(dto.User);
|
|
|
|
List<Playlist> playlists = _playlistManager.GetPlaylists(user.Id).Where(x => x.Id == dto.Id).ToList();
|
|
|
|
if ((dto.Id == null) || !playlists.Any()) {
|
|
|
|
_logger.LogInformation("Generating new smart playlist (dto.Id = {0}, playlists.Any() = {1})", dto.Id, playlists.Any());
|
2024-10-25 02:18:13 +02:00
|
|
|
_store.DeleteSmartPlaylist(dto);
|
|
|
|
dto.Id = CreateNewPlaylist(dto);
|
2024-10-24 23:53:21 +02:00
|
|
|
await _store.SaveSmartPlaylistAsync(dto);
|
|
|
|
playlists = _playlistManager.GetPlaylists(user.Id).Where(x => x.Id == dto.Id).ToList();
|
|
|
|
}
|
|
|
|
var insertItems = FilterPlaylistItems(GetAllUserMedia(user), user, dto);
|
|
|
|
Playlist playlist = playlists.First();
|
|
|
|
await ClearPlaylist(dto, playlist, user);
|
|
|
|
await _playlistManager.AddItemToPlaylistAsync(playlist.Id, insertItems.ToArray(), user.Id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private async Task ClearPlaylist(SmartPlaylistDto smartPlaylist, Playlist playlist, User user) {
|
2024-10-25 02:18:13 +02:00
|
|
|
// fuck if I know
|
|
|
|
if (_libraryManager.GetItemById(playlist.Id) is not Playlist playlist_new) {
|
|
|
|
throw new ArgumentException("");
|
|
|
|
}
|
|
|
|
var existingItems = playlist_new.GetManageableItems().ToList();
|
|
|
|
await _playlistManager.RemoveItemFromPlaylistAsync(playlist.Id.ToString(), existingItems.Select(x => x.Item1.Id));
|
2024-06-27 01:47:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|