using System.Net.Mime; using System.Threading.Tasks; using Jellyfin.Api.Attributes; using Jellyfin.Api.Extensions; using Jellyfin.Api.Models.ClientLogDtos; using MediaBrowser.Controller.ClientEvent; using MediaBrowser.Controller.Configuration; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; namespace Jellyfin.Api.Controllers; /// /// Client log controller. /// [Authorize] public class ClientLogController : BaseJellyfinApiController { private const int MaxDocumentSize = 1_000_000; private readonly IClientEventLogger _clientEventLogger; private readonly IServerConfigurationManager _serverConfigurationManager; /// /// Initializes a new instance of the class. /// /// Instance of the interface. /// Instance of the interface. public ClientLogController( IClientEventLogger clientEventLogger, IServerConfigurationManager serverConfigurationManager) { _clientEventLogger = clientEventLogger; _serverConfigurationManager = serverConfigurationManager; } /// /// Upload a document. /// /// Document saved. /// Event logging disabled. /// Upload size too large. /// Create response. [HttpPost("Document")] [ProducesResponseType(typeof(ClientLogDocumentResponseDto), StatusCodes.Status200OK)] [ProducesResponseType(StatusCodes.Status403Forbidden)] [ProducesResponseType(StatusCodes.Status413PayloadTooLarge)] [AcceptsFile(MediaTypeNames.Text.Plain)] [RequestSizeLimit(MaxDocumentSize)] public async Task> LogFile() { if (!_serverConfigurationManager.Configuration.AllowClientLogUpload) { return Forbid(); } if (Request.ContentLength > MaxDocumentSize) { // Manually validate to return proper status code. return StatusCode(StatusCodes.Status413PayloadTooLarge, $"Payload must be less than {MaxDocumentSize:N0} bytes"); } var (clientName, clientVersion) = GetRequestInformation(); var fileName = await _clientEventLogger.WriteDocumentAsync(clientName, clientVersion, Request.Body) .ConfigureAwait(false); return Ok(new ClientLogDocumentResponseDto(fileName)); } private (string ClientName, string ClientVersion) GetRequestInformation() { var clientName = HttpContext.User.GetClient() ?? "unknown-client"; var clientVersion = HttpContext.User.GetIsApiKey() ? "apikey" : HttpContext.User.GetVersion() ?? "unknown-version"; return (clientName, clientVersion); } }