summary refs log tree commit diff
path: root/MxApiExtensions/Controllers/Other/MediaProxyController.cs
diff options
context:
space:
mode:
authorTheArcaneBrony <myrainbowdash949@gmail.com>2023-11-05 17:59:38 +0100
committerTheArcaneBrony <myrainbowdash949@gmail.com>2023-11-05 17:59:38 +0100
commit2abb132234546e61bb0aff3897dc49e72ea84f5d (patch)
treec885c03d35e7a0a6b8fc21bd0b259216c61c877c /MxApiExtensions/Controllers/Other/MediaProxyController.cs
parentUpdate (diff)
downloadMxApiExtensions-2abb132234546e61bb0aff3897dc49e72ea84f5d.tar.xz
Working sync proxy
Diffstat (limited to 'MxApiExtensions/Controllers/Other/MediaProxyController.cs')
-rw-r--r--MxApiExtensions/Controllers/Other/MediaProxyController.cs78
1 files changed, 78 insertions, 0 deletions
diff --git a/MxApiExtensions/Controllers/Other/MediaProxyController.cs b/MxApiExtensions/Controllers/Other/MediaProxyController.cs
new file mode 100644

index 0000000..03b68ba --- /dev/null +++ b/MxApiExtensions/Controllers/Other/MediaProxyController.cs
@@ -0,0 +1,78 @@ +using System.Net.Http.Headers; +using LibMatrix.Homeservers; +using LibMatrix.Services; +using Microsoft.AspNetCore.Mvc; +using MxApiExtensions.Classes.LibMatrix; +using MxApiExtensions.Services; + +namespace MxApiExtensions.Controllers; + +[ApiController] +[Route("/")] +public class MediaProxyController(ILogger<GenericController> logger, MxApiExtensionsConfiguration config, AuthenticationService authenticationService, + AuthenticatedHomeserverProviderService authenticatedHomeserverProviderService, HomeserverProviderService hsProvider) + : ControllerBase { + private class MediaCacheEntry { + public DateTime LastRequested { get; set; } = DateTime.Now; + public byte[] Data { get; set; } + public string ContentType { get; set; } + public long Size => Data.LongCount(); + } + + private static Dictionary<string, MediaCacheEntry> _mediaCache = new(); + private static SemaphoreSlim _semaphore = new(1, 1); + + [HttpGet("/_matrix/media/{_}/download/{serverName}/{mediaId}")] + public async Task Proxy(string? _, string serverName, string mediaId) { + try { + logger.LogInformation("Proxying media: {}{}", serverName, mediaId); + + await _semaphore.WaitAsync(); + MediaCacheEntry entry; + if (!_mediaCache.ContainsKey($"{serverName}/{mediaId}")) { + _mediaCache.Add($"{serverName}/{mediaId}", entry = new()); + List<RemoteHomeserver> FeasibleHomeservers = new(); + { + var a = await authenticatedHomeserverProviderService.TryGetRemoteHomeserver(); + if(a is not null) + FeasibleHomeservers.Add(a); + } + + FeasibleHomeservers.Add(await hsProvider.GetRemoteHomeserver(serverName)); + + foreach (var homeserver in FeasibleHomeservers) { + var resp = await homeserver.ClientHttpClient.GetAsync($"{Request.Path}"); + if(!resp.IsSuccessStatusCode) continue; + entry.ContentType = resp.Content.Headers.ContentType?.ToString() ?? "application/json"; + entry.Data = await resp.Content.ReadAsByteArrayAsync(); + break; + } + } + else entry = _mediaCache[$"{serverName}/{mediaId}"]; + _semaphore.Release(); + + Response.StatusCode = 200; + Response.ContentType = entry.ContentType; + await Response.StartAsync(); + await Response.Body.WriteAsync(entry.Data, 0, entry.Data.Length); + await Response.Body.FlushAsync(); + await Response.CompleteAsync(); + } + catch (MxApiMatrixException e) { + logger.LogError(e, "Matrix error"); + Response.StatusCode = StatusCodes.Status500InternalServerError; + Response.ContentType = "application/json"; + + await Response.WriteAsync(e.GetAsJson()); + await Response.CompleteAsync(); + } + catch (Exception e) { + logger.LogError(e, "Unhandled error"); + Response.StatusCode = StatusCodes.Status500InternalServerError; + Response.ContentType = "text/plain"; + + await Response.WriteAsync(e.ToString()); + await Response.CompleteAsync(); + } + } +}