using System.Net.Mime; using System.Threading.Tasks; using Jellyfin.Api.Attributes; using Jellyfin.Api.Constants; using Jellyfin.Api.Models.ClientLogDtos; using MediaBrowser.Controller.ClientEvent; using MediaBrowser.Controller.Configuration; using MediaBrowser.Controller.Net; using MediaBrowser.Model.ClientLog; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; namespace Jellyfin.Api.Controllers { /// /// Client log controller. /// [Authorize(Policy = Policies.DefaultAuthorization)] public class ClientLogController : BaseJellyfinApiController { private const int MaxDocumentSize = 1_000_000; private readonly IClientEventLogger _clientEventLogger; private readonly IAuthorizationContext _authorizationContext; private readonly IServerConfigurationManager _serverConfigurationManager; /// /// Initializes a new instance of the class. /// /// Instance of the interface. /// Instance of the interface. /// Instance of the interface. public ClientLogController( IClientEventLogger clientEventLogger, IAuthorizationContext authorizationContext, IServerConfigurationManager serverConfigurationManager) { _clientEventLogger = clientEventLogger; _authorizationContext = authorizationContext; _serverConfigurationManager = serverConfigurationManager; } /// /// Post event from client. /// /// The client log dto. /// Event logged. /// Submission status. [HttpPost] [ProducesResponseType(StatusCodes.Status204NoContent)] public ActionResult LogEvent([FromBody] ClientLogEventDto clientLogEventDto) { if (!_serverConfigurationManager.Configuration.AllowClientLogUpload) { return Forbid(); } Log(clientLogEventDto); return NoContent(); } /// /// Bulk post events from client. /// /// The list of client log dtos. /// All events logged. /// Submission status. [HttpPost("Bulk")] [ProducesResponseType(StatusCodes.Status204NoContent)] public ActionResult LogEvents([FromBody] ClientLogEventDto[] clientLogEventDtos) { if (!_serverConfigurationManager.Configuration.AllowClientLogUpload) { return Forbid(); } foreach (var dto in clientLogEventDtos) { Log(dto); } return NoContent(); } /// /// Upload a document. /// /// Submission status. [HttpPost("Document")] [ProducesResponseType(StatusCodes.Status204NoContent)] [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 authorizationInfo = await _authorizationContext.GetAuthorizationInfo(Request) .ConfigureAwait(false); await _clientEventLogger.WriteDocumentAsync(authorizationInfo, Request.Body) .ConfigureAwait(false); return NoContent(); } private void Log(ClientLogEventDto dto) { _clientEventLogger.Log(new ClientLogEvent( dto.Timestamp, dto.Level, dto.UserId, dto.ClientName, dto.ClientVersion, dto.DeviceId, dto.Message)); } } }