| | 1 | | using System.ComponentModel.DataAnnotations; |
| | 2 | | using System.Linq; |
| | 3 | | using System.Threading.Tasks; |
| | 4 | | using Jellyfin.Api.Constants; |
| | 5 | | using Jellyfin.Api.Models.StartupDtos; |
| | 6 | | using MediaBrowser.Common.Api; |
| | 7 | | using MediaBrowser.Common.Net; |
| | 8 | | using MediaBrowser.Controller.Configuration; |
| | 9 | | using MediaBrowser.Controller.Library; |
| | 10 | | using Microsoft.AspNetCore.Authorization; |
| | 11 | | using Microsoft.AspNetCore.Http; |
| | 12 | | using Microsoft.AspNetCore.Mvc; |
| | 13 | |
|
| | 14 | | namespace Jellyfin.Api.Controllers; |
| | 15 | |
|
| | 16 | | /// <summary> |
| | 17 | | /// The startup wizard controller. |
| | 18 | | /// </summary> |
| | 19 | | [Authorize(Policy = Policies.FirstTimeSetupOrElevated)] |
| | 20 | | public class StartupController : BaseJellyfinApiController |
| | 21 | | { |
| | 22 | | private readonly IServerConfigurationManager _config; |
| | 23 | | private readonly IUserManager _userManager; |
| | 24 | |
|
| | 25 | | /// <summary> |
| | 26 | | /// Initializes a new instance of the <see cref="StartupController" /> class. |
| | 27 | | /// </summary> |
| | 28 | | /// <param name="config">The server configuration manager.</param> |
| | 29 | | /// <param name="userManager">The user manager.</param> |
| 36 | 30 | | public StartupController(IServerConfigurationManager config, IUserManager userManager) |
| | 31 | | { |
| 36 | 32 | | _config = config; |
| 36 | 33 | | _userManager = userManager; |
| 36 | 34 | | } |
| | 35 | |
|
| | 36 | | /// <summary> |
| | 37 | | /// Completes the startup wizard. |
| | 38 | | /// </summary> |
| | 39 | | /// <response code="204">Startup wizard completed.</response> |
| | 40 | | /// <returns>A <see cref="NoContentResult"/> indicating success.</returns> |
| | 41 | | [HttpPost("Complete")] |
| | 42 | | [ProducesResponseType(StatusCodes.Status204NoContent)] |
| | 43 | | public ActionResult CompleteWizard() |
| | 44 | | { |
| 16 | 45 | | _config.Configuration.IsStartupWizardCompleted = true; |
| 16 | 46 | | _config.SaveConfiguration(); |
| 16 | 47 | | return NoContent(); |
| | 48 | | } |
| | 49 | |
|
| | 50 | | /// <summary> |
| | 51 | | /// Gets the initial startup wizard configuration. |
| | 52 | | /// </summary> |
| | 53 | | /// <response code="200">Initial startup wizard configuration retrieved.</response> |
| | 54 | | /// <returns>An <see cref="OkResult"/> containing the initial startup wizard configuration.</returns> |
| | 55 | | [HttpGet("Configuration")] |
| | 56 | | [ProducesResponseType(StatusCodes.Status200OK)] |
| | 57 | | public ActionResult<StartupConfigurationDto> GetStartupConfiguration() |
| | 58 | | { |
| 1 | 59 | | return new StartupConfigurationDto |
| 1 | 60 | | { |
| 1 | 61 | | UICulture = _config.Configuration.UICulture, |
| 1 | 62 | | MetadataCountryCode = _config.Configuration.MetadataCountryCode, |
| 1 | 63 | | PreferredMetadataLanguage = _config.Configuration.PreferredMetadataLanguage |
| 1 | 64 | | }; |
| | 65 | | } |
| | 66 | |
|
| | 67 | | /// <summary> |
| | 68 | | /// Sets the initial startup wizard configuration. |
| | 69 | | /// </summary> |
| | 70 | | /// <param name="startupConfiguration">The updated startup configuration.</param> |
| | 71 | | /// <response code="204">Configuration saved.</response> |
| | 72 | | /// <returns>A <see cref="NoContentResult"/> indicating success.</returns> |
| | 73 | | [HttpPost("Configuration")] |
| | 74 | | [ProducesResponseType(StatusCodes.Status204NoContent)] |
| | 75 | | public ActionResult UpdateInitialConfiguration([FromBody, Required] StartupConfigurationDto startupConfiguration) |
| | 76 | | { |
| 1 | 77 | | _config.Configuration.UICulture = startupConfiguration.UICulture ?? string.Empty; |
| 1 | 78 | | _config.Configuration.MetadataCountryCode = startupConfiguration.MetadataCountryCode ?? string.Empty; |
| 1 | 79 | | _config.Configuration.PreferredMetadataLanguage = startupConfiguration.PreferredMetadataLanguage ?? string.Empty |
| 1 | 80 | | _config.SaveConfiguration(); |
| 1 | 81 | | return NoContent(); |
| | 82 | | } |
| | 83 | |
|
| | 84 | | /// <summary> |
| | 85 | | /// Sets remote access and UPnP. |
| | 86 | | /// </summary> |
| | 87 | | /// <param name="startupRemoteAccessDto">The startup remote access dto.</param> |
| | 88 | | /// <response code="204">Configuration saved.</response> |
| | 89 | | /// <returns>A <see cref="NoContentResult"/> indicating success.</returns> |
| | 90 | | [HttpPost("RemoteAccess")] |
| | 91 | | [ProducesResponseType(StatusCodes.Status204NoContent)] |
| | 92 | | public ActionResult SetRemoteAccess([FromBody, Required] StartupRemoteAccessDto startupRemoteAccessDto) |
| | 93 | | { |
| 0 | 94 | | NetworkConfiguration settings = _config.GetNetworkConfiguration(); |
| 0 | 95 | | settings.EnableRemoteAccess = startupRemoteAccessDto.EnableRemoteAccess; |
| 0 | 96 | | _config.SaveConfiguration(NetworkConfigurationStore.StoreKey, settings); |
| 0 | 97 | | return NoContent(); |
| | 98 | | } |
| | 99 | |
|
| | 100 | | /// <summary> |
| | 101 | | /// Gets the first user. |
| | 102 | | /// </summary> |
| | 103 | | /// <response code="200">Initial user retrieved.</response> |
| | 104 | | /// <returns>The first user.</returns> |
| | 105 | | [HttpGet("User")] |
| | 106 | | [HttpGet("FirstUser", Name = "GetFirstUser_2")] |
| | 107 | | [ProducesResponseType(StatusCodes.Status200OK)] |
| | 108 | | public async Task<StartupUserDto> GetFirstUser() |
| | 109 | | { |
| | 110 | | // TODO: Remove this method when startup wizard no longer requires an existing user. |
| | 111 | | await _userManager.InitializeAsync().ConfigureAwait(false); |
| | 112 | | var user = _userManager.Users.First(); |
| | 113 | | return new StartupUserDto |
| | 114 | | { |
| | 115 | | Name = user.Username, |
| | 116 | | Password = user.Password |
| | 117 | | }; |
| | 118 | | } |
| | 119 | |
|
| | 120 | | /// <summary> |
| | 121 | | /// Sets the user name and password. |
| | 122 | | /// </summary> |
| | 123 | | /// <param name="startupUserDto">The DTO containing username and password.</param> |
| | 124 | | /// <response code="204">Updated user name and password.</response> |
| | 125 | | /// <returns> |
| | 126 | | /// A <see cref="Task" /> that represents the asynchronous update operation. |
| | 127 | | /// The task result contains a <see cref="NoContentResult"/> indicating success. |
| | 128 | | /// </returns> |
| | 129 | | [HttpPost("User")] |
| | 130 | | [ProducesResponseType(StatusCodes.Status204NoContent)] |
| | 131 | | public async Task<ActionResult> UpdateStartupUser([FromBody] StartupUserDto startupUserDto) |
| | 132 | | { |
| | 133 | | var user = _userManager.Users.First(); |
| | 134 | | if (string.IsNullOrWhiteSpace(startupUserDto.Password)) |
| | 135 | | { |
| | 136 | | return BadRequest("Password must not be empty"); |
| | 137 | | } |
| | 138 | |
|
| | 139 | | if (startupUserDto.Name is not null) |
| | 140 | | { |
| | 141 | | user.Username = startupUserDto.Name; |
| | 142 | | } |
| | 143 | |
|
| | 144 | | await _userManager.UpdateUserAsync(user).ConfigureAwait(false); |
| | 145 | |
|
| | 146 | | if (!string.IsNullOrEmpty(startupUserDto.Password)) |
| | 147 | | { |
| | 148 | | await _userManager.ChangePassword(user, startupUserDto.Password).ConfigureAwait(false); |
| | 149 | | } |
| | 150 | |
|
| | 151 | | return NoContent(); |
| | 152 | | } |
| | 153 | | } |