2022-03-13 22:47:30 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Collections.ObjectModel;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Reactive.Disposables;
|
|
|
|
using System.Reactive.Subjects;
|
|
|
|
using DynamicData;
|
|
|
|
using DynamicData.Kernel;
|
2022-05-17 05:26:59 +00:00
|
|
|
using Microsoft.Extensions.Logging;
|
2022-03-13 22:47:30 +00:00
|
|
|
using ReactiveUI;
|
|
|
|
using Wabbajack.RateLimiter;
|
|
|
|
|
|
|
|
namespace Wabbajack.Models;
|
|
|
|
|
|
|
|
public class ResourceMonitor : IDisposable
|
|
|
|
{
|
2023-12-30 20:34:36 +00:00
|
|
|
private readonly TimeSpan _pollInterval = TimeSpan.FromMilliseconds(1000);
|
2022-03-13 22:47:30 +00:00
|
|
|
|
|
|
|
private readonly IResource[] _resources;
|
2022-10-08 03:43:44 +00:00
|
|
|
|
2022-03-13 22:47:30 +00:00
|
|
|
private readonly Subject<(string Name, long Througput)[]> _updates = new ();
|
|
|
|
private (string Name, long Throughput)[] _prev;
|
|
|
|
public IObservable<(string Name, long Throughput)[]> Updates => _updates;
|
|
|
|
|
|
|
|
|
|
|
|
private readonly SourceCache<CPUDisplayVM, ulong> _tasks = new(x => x.ID);
|
|
|
|
public readonly ReadOnlyObservableCollection<CPUDisplayVM> _tasksFiltered;
|
|
|
|
private readonly CompositeDisposable _compositeDisposable;
|
2022-05-17 05:26:59 +00:00
|
|
|
private readonly ILogger<ResourceMonitor> _logger;
|
2022-03-13 22:47:30 +00:00
|
|
|
public ReadOnlyObservableCollection<CPUDisplayVM> Tasks => _tasksFiltered;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2022-05-17 05:26:59 +00:00
|
|
|
public ResourceMonitor(ILogger<ResourceMonitor> logger, IEnumerable<IResource> resources)
|
2022-03-13 22:47:30 +00:00
|
|
|
{
|
2022-05-17 05:26:59 +00:00
|
|
|
_logger = logger;
|
2022-03-13 22:47:30 +00:00
|
|
|
_compositeDisposable = new CompositeDisposable();
|
|
|
|
_resources = resources.ToArray();
|
|
|
|
_prev = _resources.Select(x => (x.Name, (long)0)).ToArray();
|
|
|
|
|
2022-10-08 03:43:44 +00:00
|
|
|
RxApp.MainThreadScheduler.ScheduleRecurringAction(_pollInterval, Elapsed)
|
2022-03-13 22:47:30 +00:00
|
|
|
.DisposeWith(_compositeDisposable);
|
|
|
|
|
|
|
|
_tasks.Connect()
|
2022-08-09 11:54:21 +00:00
|
|
|
.Filter(x => x.IsWorking)
|
2022-03-13 22:47:30 +00:00
|
|
|
.Bind(out _tasksFiltered)
|
|
|
|
.Subscribe()
|
|
|
|
.DisposeWith(_compositeDisposable);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void Elapsed()
|
|
|
|
{
|
|
|
|
var current = _resources.Select(x => (x.Name, x.StatusReport.Transferred)).ToArray();
|
|
|
|
var diff = _prev.Zip(current)
|
2022-10-08 03:43:44 +00:00
|
|
|
.Select(t => (t.First.Name, (long)((t.Second.Transferred - t.First.Throughput) / _pollInterval.TotalSeconds)))
|
2022-03-13 22:47:30 +00:00
|
|
|
.ToArray();
|
|
|
|
_prev = current;
|
|
|
|
_updates.OnNext(diff);
|
|
|
|
|
|
|
|
_tasks.Edit(l =>
|
|
|
|
{
|
|
|
|
var used = new HashSet<ulong>();
|
|
|
|
foreach (var resource in _resources)
|
|
|
|
{
|
2022-08-09 11:54:21 +00:00
|
|
|
foreach (var job in resource.Jobs.Where(j => j.Current > 0))
|
2022-03-13 22:47:30 +00:00
|
|
|
{
|
|
|
|
used.Add(job.ID);
|
|
|
|
var tsk = l.Lookup(job.ID);
|
|
|
|
// Update
|
|
|
|
if (tsk != Optional<CPUDisplayVM>.None)
|
|
|
|
{
|
|
|
|
var t = tsk.Value;
|
|
|
|
t.Msg = job.Description;
|
|
|
|
t.ProgressPercent = job.Size == 0 ? Percent.Zero : Percent.FactoryPutInRange(job.Current, (long)job.Size);
|
2022-05-17 05:26:59 +00:00
|
|
|
t.IsWorking = job.Current > 0;
|
2022-03-13 22:47:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var vm = new CPUDisplayVM
|
|
|
|
{
|
|
|
|
ID = job.ID,
|
|
|
|
StartTime = DateTime.Now,
|
|
|
|
Msg = job.Description,
|
2022-05-17 05:26:59 +00:00
|
|
|
ProgressPercent = job.Size == 0 ? Percent.Zero : Percent.FactoryPutInRange(job.Current, (long) job.Size),
|
|
|
|
IsWorking = job.Current > 0,
|
2022-03-13 22:47:30 +00:00
|
|
|
};
|
|
|
|
l.AddOrUpdate(vm);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete
|
|
|
|
foreach (var itm in l.Items.Where(v => !used.Contains(v.ID)))
|
|
|
|
l.Remove(itm);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
_compositeDisposable.Dispose();
|
|
|
|
}
|
|
|
|
}
|