| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127 | using System.IO;using System.Threading.Tasks;using Jellyfin.Server.Implementations.SystemBackupService;using MediaBrowser.Common.Api;using MediaBrowser.Common.Configuration;using MediaBrowser.Controller.SystemBackupService;using Microsoft.AspNetCore.Authentication.OAuth.Claims;using Microsoft.AspNetCore.Authorization;using Microsoft.AspNetCore.Http;using Microsoft.AspNetCore.Mvc;using Microsoft.AspNetCore.Mvc.ModelBinding;namespace Jellyfin.Api.Controllers;/// <summary>/// The backup controller./// </summary>[Authorize(Policy = Policies.RequiresElevation)]public class BackupController : BaseJellyfinApiController{    private readonly IBackupService _backupService;    private readonly IApplicationPaths _applicationPaths;    /// <summary>    /// Initializes a new instance of the <see cref="BackupController"/> class.    /// </summary>    /// <param name="backupService">Instance of the <see cref="IBackupService"/> interface.</param>    /// <param name="applicationPaths">Instance of the <see cref="IApplicationPaths"/> interface.</param>    public BackupController(IBackupService backupService, IApplicationPaths applicationPaths)    {        _backupService = backupService;        _applicationPaths = applicationPaths;    }    /// <summary>    /// Creates a new Backup.    /// </summary>    /// <param name="backupOptions">The backup options.</param>    /// <response code="200">Backup created.</response>    /// <response code="403">User does not have permission to retrieve information.</response>    /// <returns>The created backup manifest.</returns>    [HttpPost("Create")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    public async Task<ActionResult<BackupManifestDto>> CreateBackup([FromBody] BackupOptionsDto backupOptions)    {        return Ok(await _backupService.CreateBackupAsync(backupOptions ?? new()).ConfigureAwait(false));    }    /// <summary>    /// Restores to a backup by restarting the server and applying the backup.    /// </summary>    /// <param name="archiveRestoreDto">The data to start a restore process.</param>    /// <response code="204">Backup restore started.</response>    /// <response code="403">User does not have permission to retrieve information.</response>    /// <returns>No-Content.</returns>    [HttpPost("Restore")]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    public IActionResult StartRestoreBackup([FromBody, BindRequired] BackupRestoreRequestDto archiveRestoreDto)    {        var archivePath = SanitizePath(archiveRestoreDto.ArchiveFileName);        if (!System.IO.File.Exists(archivePath))        {            return NotFound();        }        _backupService.ScheduleRestoreAndRestartServer(archivePath);        return NoContent();    }    /// <summary>    /// Gets a list of all currently present backups in the backup directory.    /// </summary>    /// <response code="200">Backups available.</response>    /// <response code="403">User does not have permission to retrieve information.</response>    /// <returns>The list of backups.</returns>    [HttpGet]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    public async Task<ActionResult<BackupManifestDto[]>> ListBackups()    {        return Ok(await _backupService.EnumerateBackups().ConfigureAwait(false));    }    /// <summary>    /// Gets the descriptor from an existing archive is present.    /// </summary>    /// <param name="path">The data to start a restore process.</param>    /// <response code="200">Backup archive manifest.</response>    /// <response code="204">Not a valid jellyfin Archive.</response>    /// <response code="404">Not a valid path.</response>    /// <response code="403">User does not have permission to retrieve information.</response>    /// <returns>The backup manifest.</returns>    [HttpGet("Manifest")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    public async Task<ActionResult<BackupManifestDto>> GetBackup([BindRequired] string path)    {        var backupPath = SanitizePath(path);        if (!System.IO.File.Exists(backupPath))        {            return NotFound();        }        var manifest = await _backupService.GetBackupManifest(backupPath).ConfigureAwait(false);        if (manifest is null)        {            return NoContent();        }        return Ok(manifest);    }    [NonAction]    private string SanitizePath(string path)    {        // sanitize path        var archiveRestorePath = Path.GetFileName(Path.GetFullPath(path));        var archivePath = Path.Combine(_applicationPaths.BackupPath, archiveRestorePath);        return archivePath;    }}
 |