mirror of
https://github.com/immense/Remotely.git
synced 2025-10-26 11:27:15 +00:00
115 lines
4.4 KiB
C#
115 lines
4.4 KiB
C#
using Microsoft.AspNetCore.Hosting;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.Extensions.Caching.Memory;
|
|
using Microsoft.Extensions.Logging;
|
|
using Microsoft.Extensions.Primitives;
|
|
using Remotely.Server.Services;
|
|
using Remotely.Shared.Enums;
|
|
using System;
|
|
using System.IO;
|
|
using System.Net;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace Remotely.Server.API
|
|
{
|
|
[Route("api/[controller]")]
|
|
[ApiController]
|
|
public class AgentUpdateController : ControllerBase
|
|
{
|
|
private static readonly MemoryCache _downloadingAgents = new MemoryCache(new MemoryCacheOptions()
|
|
{ ExpirationScanFrequency = TimeSpan.FromSeconds(10) });
|
|
|
|
|
|
public AgentUpdateController(IWebHostEnvironment hostingEnv,
|
|
IDataService dataService,
|
|
IApplicationConfig appConfig)
|
|
{
|
|
HostEnv = hostingEnv;
|
|
DataService = dataService;
|
|
AppConfig = appConfig;
|
|
}
|
|
|
|
private IDataService DataService { get; }
|
|
public IApplicationConfig AppConfig { get; }
|
|
private IWebHostEnvironment HostEnv { get; }
|
|
|
|
|
|
[HttpGet("[action]/{downloadId}")]
|
|
public ActionResult ClearDownload(string downloadId)
|
|
{
|
|
DataService.WriteEvent($"Clearing download ID {downloadId}.", EventType.Debug, null);
|
|
_downloadingAgents.Remove(downloadId);
|
|
return Ok();
|
|
}
|
|
|
|
[HttpGet("[action]/{platform}/{downloadId}")]
|
|
public async Task<ActionResult> DownloadPackage(string platform, string downloadId)
|
|
{
|
|
try
|
|
{
|
|
var startWait = DateTimeOffset.Now;
|
|
|
|
while (_downloadingAgents.Count >= AppConfig.MaxConcurrentUpdates)
|
|
{
|
|
await Task.Delay(new Random().Next(100, 10000));
|
|
|
|
// A get operation is necessary to evaluate item eviction.
|
|
_downloadingAgents.TryGetValue(string.Empty, out _);
|
|
}
|
|
|
|
var entryExpirationTime = TimeSpan.FromMinutes(3);
|
|
var tokenExpirationTime = entryExpirationTime.Add(TimeSpan.FromSeconds(15));
|
|
|
|
var expirationToken = new CancellationChangeToken(
|
|
new CancellationTokenSource(tokenExpirationTime).Token);
|
|
|
|
var cacheOptions = new MemoryCacheEntryOptions()
|
|
.SetAbsoluteExpiration(entryExpirationTime)
|
|
.AddExpirationToken(expirationToken);
|
|
|
|
_downloadingAgents.Set(downloadId, string.Empty, cacheOptions);
|
|
|
|
var waitTime = DateTimeOffset.Now - startWait;
|
|
DataService.WriteEvent($"Download started after wait time of {waitTime}. " +
|
|
$"ID: {downloadId}. " +
|
|
$"IP: {Request?.HttpContext?.Connection?.RemoteIpAddress}. " +
|
|
$"Current Downloads: {_downloadingAgents.Count}. Max Allowed: {AppConfig.MaxConcurrentUpdates}", EventType.Debug, null);
|
|
|
|
|
|
string filePath;
|
|
|
|
switch (platform.ToLower())
|
|
{
|
|
case "win-x64":
|
|
filePath = Path.Combine(HostEnv.WebRootPath, "Downloads", "Remotely-Win10-x64.zip");
|
|
break;
|
|
case "win-x86":
|
|
filePath = Path.Combine(HostEnv.WebRootPath, "Downloads", "Remotely-Win10-x86.zip");
|
|
break;
|
|
case "linux":
|
|
filePath = Path.Combine(HostEnv.WebRootPath, "Downloads", "Remotely-Linux.zip");
|
|
break;
|
|
default:
|
|
DataService.WriteEvent($"Unknown platform requested in { nameof(AgentUpdateController)}. " +
|
|
$"Platform: {platform}. " +
|
|
$"IP: {Request?.HttpContext?.Connection?.RemoteIpAddress}.",
|
|
EventType.Warning,
|
|
null);
|
|
return BadRequest();
|
|
}
|
|
|
|
var fileStream = System.IO.File.OpenRead(filePath);
|
|
|
|
return File(fileStream, "application/octet-stream", "RemotelyUpdate.zip");
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_downloadingAgents.Remove(downloadId);
|
|
DataService.WriteEvent(ex, null);
|
|
return StatusCode((int)HttpStatusCode.InternalServerError);
|
|
}
|
|
}
|
|
}
|
|
}
|