mirror of
https://github.com/BililiveRecorder/BililiveRecorder.git
synced 2024-12-26 12:15:42 +08:00
230 lines
9.5 KiB
C#
230 lines
9.5 KiB
C#
using System;
|
|
using System.Collections.ObjectModel;
|
|
using System.ComponentModel;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using BililiveRecorder.Core.Config;
|
|
using BililiveRecorder.Core.Config.V3;
|
|
using BililiveRecorder.Core.Event;
|
|
using BililiveRecorder.Core.SimpleWebhook;
|
|
using Serilog;
|
|
|
|
namespace BililiveRecorder.Core
|
|
{
|
|
internal class Recorder : IRecorder
|
|
{
|
|
private readonly object lockObject = new object();
|
|
private readonly ObservableCollection<IRoom> roomCollection;
|
|
private readonly IRoomFactory roomFactory;
|
|
private readonly ILogger logger;
|
|
private readonly BasicWebhookV1 basicWebhookV1;
|
|
private readonly BasicWebhookV2 basicWebhookV2;
|
|
|
|
private bool disposedValue;
|
|
|
|
public Recorder(IRoomFactory roomFactory, ConfigV3 config, ILogger logger)
|
|
{
|
|
this.roomFactory = roomFactory ?? throw new ArgumentNullException(nameof(roomFactory));
|
|
this.Config = config ?? throw new ArgumentNullException(nameof(config));
|
|
this.logger = logger?.ForContext<Recorder>() ?? throw new ArgumentNullException(nameof(logger));
|
|
this.roomCollection = new ObservableCollection<IRoom>();
|
|
this.Rooms = new ReadOnlyObservableCollection<IRoom>(this.roomCollection);
|
|
|
|
this.basicWebhookV1 = new BasicWebhookV1(config);
|
|
this.basicWebhookV2 = new BasicWebhookV2(config.Global);
|
|
|
|
{
|
|
logger.Debug("Recorder created with {RoomCount} rooms", config.Rooms.Count);
|
|
for (var i = 0; i < config.Rooms.Count; i++)
|
|
{
|
|
var item = config.Rooms[i];
|
|
if (item is not null)
|
|
this.AddRoom(roomConfig: item, initDelayFactor: i);
|
|
}
|
|
|
|
this.SaveConfig();
|
|
}
|
|
}
|
|
|
|
public event EventHandler<AggregatedRoomEventArgs<RecordSessionStartedEventArgs>>? RecordSessionStarted;
|
|
public event EventHandler<AggregatedRoomEventArgs<RecordSessionEndedEventArgs>>? RecordSessionEnded;
|
|
public event EventHandler<AggregatedRoomEventArgs<RecordFileOpeningEventArgs>>? RecordFileOpening;
|
|
public event EventHandler<AggregatedRoomEventArgs<RecordFileClosedEventArgs>>? RecordFileClosed;
|
|
public event EventHandler<AggregatedRoomEventArgs<IOStatsEventArgs>>? IOStats;
|
|
public event EventHandler<AggregatedRoomEventArgs<RecordingStatsEventArgs>>? RecordingStats;
|
|
public event EventHandler<IRoom>? StreamStarted;
|
|
#pragma warning disable CS0067 // The event 'Recorder.PropertyChanged' is never used
|
|
public event PropertyChangedEventHandler? PropertyChanged;
|
|
#pragma warning restore CS0067 // The event 'Recorder.PropertyChanged' is never used
|
|
|
|
public ConfigV3 Config { get; }
|
|
|
|
public ReadOnlyObservableCollection<IRoom> Rooms { get; }
|
|
|
|
public IRoom AddRoom(int roomid) => this.AddRoom(roomid, true);
|
|
|
|
public IRoom AddRoom(int roomid, bool enabled)
|
|
{
|
|
lock (this.lockObject)
|
|
{
|
|
this.logger.Debug("AddRoom {RoomId}, AutoRecord: {AutoRecord}", roomid, enabled);
|
|
var roomConfig = new RoomConfig { RoomId = roomid, AutoRecord = enabled };
|
|
var room = this.AddRoom(roomConfig, 0);
|
|
this.SaveConfig();
|
|
return room;
|
|
}
|
|
}
|
|
|
|
private IRoom AddRoom(RoomConfig roomConfig, int initDelayFactor)
|
|
{
|
|
roomConfig.SetParent(this.Config.Global);
|
|
var room = this.roomFactory.CreateRoom(roomConfig, initDelayFactor);
|
|
|
|
room.RecordSessionStarted += this.Room_RecordSessionStarted;
|
|
room.RecordSessionEnded += this.Room_RecordSessionEnded;
|
|
room.RecordFileOpening += this.Room_RecordFileOpening;
|
|
room.RecordFileClosed += this.Room_RecordFileClosed;
|
|
room.IOStats += this.Room_IOStats;
|
|
room.RecordingStats += this.Room_RecordingStats;
|
|
room.PropertyChanged += this.Room_PropertyChanged;
|
|
|
|
this.roomCollection.Add(room);
|
|
return room;
|
|
}
|
|
|
|
public void RemoveRoom(IRoom room)
|
|
{
|
|
lock (this.lockObject)
|
|
{
|
|
if (this.roomCollection.Remove(room))
|
|
{
|
|
// 如果提前 detach 会导致 FileClosed SessionEnded 收不到
|
|
// 目前没有在各种 event 里再使用 room object
|
|
// 如果以后要使用 IRecorder 上的 event 再重新捋一遍防止出现奇怪 bug
|
|
// 此处不会导致内存泄漏
|
|
|
|
// room.RecordSessionStarted -= this.Room_RecordSessionStarted;
|
|
// room.RecordSessionEnded -= this.Room_RecordSessionEnded;
|
|
// room.RecordFileOpening -= this.Room_RecordFileOpening;
|
|
// room.RecordFileClosed -= this.Room_RecordFileClosed;
|
|
// room.RecordingStats -= this.Room_RecordingStats;
|
|
// room.PropertyChanged -= this.Room_PropertyChanged;
|
|
|
|
this.logger.Debug("RemoveRoom {RoomId}", room.RoomConfig.RoomId);
|
|
room.Dispose();
|
|
this.SaveConfig();
|
|
}
|
|
}
|
|
}
|
|
|
|
public void SaveConfig()
|
|
{
|
|
this.Config.Rooms = this.Rooms.Select(x => x.RoomConfig).ToList();
|
|
ConfigParser.Save(this.Config);
|
|
}
|
|
|
|
#region Events
|
|
|
|
private void Room_IOStats(object? sender, IOStatsEventArgs e)
|
|
{
|
|
if (sender is not IRoom room) return;
|
|
IOStats?.Invoke(this, new AggregatedRoomEventArgs<IOStatsEventArgs>(room, e));
|
|
}
|
|
|
|
private void Room_RecordingStats(object? sender, RecordingStatsEventArgs e)
|
|
{
|
|
if (sender is not IRoom room) return;
|
|
RecordingStats?.Invoke(this, new AggregatedRoomEventArgs<RecordingStatsEventArgs>(room, e));
|
|
}
|
|
|
|
private void Room_RecordFileClosed(object? sender, RecordFileClosedEventArgs e)
|
|
{
|
|
if (sender is not IRoom room) return;
|
|
_ = Task.Run(async () => await this.basicWebhookV2.SendFileClosedAsync(e).ConfigureAwait(false));
|
|
_ = Task.Run(async () => await this.basicWebhookV1.SendAsync(new RecordEndData(e)).ConfigureAwait(false));
|
|
RecordFileClosed?.Invoke(this, new AggregatedRoomEventArgs<RecordFileClosedEventArgs>(room, e));
|
|
}
|
|
|
|
private void Room_RecordFileOpening(object? sender, RecordFileOpeningEventArgs e)
|
|
{
|
|
if (sender is not IRoom room) return;
|
|
_ = Task.Run(async () => await this.basicWebhookV2.SendFileOpeningAsync(e).ConfigureAwait(false));
|
|
RecordFileOpening?.Invoke(this, new AggregatedRoomEventArgs<RecordFileOpeningEventArgs>(room, e));
|
|
}
|
|
|
|
private void Room_RecordSessionStarted(object? sender, RecordSessionStartedEventArgs e)
|
|
{
|
|
if (sender is not IRoom room) return;
|
|
_ = Task.Run(async () => await this.basicWebhookV2.SendSessionStartedAsync(e).ConfigureAwait(false));
|
|
RecordSessionStarted?.Invoke(this, new AggregatedRoomEventArgs<RecordSessionStartedEventArgs>(room, e));
|
|
}
|
|
|
|
private void Room_RecordSessionEnded(object? sender, RecordSessionEndedEventArgs e)
|
|
{
|
|
if (sender is not IRoom room) return;
|
|
_ = Task.Run(async () => await this.basicWebhookV2.SendSessionEndedAsync(e).ConfigureAwait(false));
|
|
RecordSessionEnded?.Invoke(this, new AggregatedRoomEventArgs<RecordSessionEndedEventArgs>(room, e));
|
|
}
|
|
|
|
private void Room_PropertyChanged(object? sender, PropertyChangedEventArgs e)
|
|
{
|
|
if (sender is not IRoom room)
|
|
return;
|
|
|
|
if (e.PropertyName == nameof(IRoom.Streaming))
|
|
{
|
|
if (room.Streaming)
|
|
{
|
|
_ = Task.Run(async () => await this.basicWebhookV2.SendStreamStartedAsync(new StreamStartedEventArgs(room)).ConfigureAwait(false));
|
|
_ = Task.Run(() => StreamStarted?.Invoke(this, room));
|
|
}
|
|
else
|
|
{
|
|
_ = Task.Run(async () => await this.basicWebhookV2.SendStreamEndedAsync(new StreamEndedEventArgs(room)).ConfigureAwait(false));
|
|
}
|
|
}
|
|
// TODO
|
|
// throw new NotImplementedException();
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Dispose
|
|
|
|
protected virtual void Dispose(bool disposing)
|
|
{
|
|
if (!this.disposedValue)
|
|
{
|
|
if (disposing)
|
|
{
|
|
// dispose managed state (managed objects)
|
|
this.logger.Debug("Dispose called");
|
|
this.SaveConfig();
|
|
foreach (var room in this.roomCollection)
|
|
room.Dispose();
|
|
}
|
|
|
|
// free unmanaged resources (unmanaged objects) and override finalizer
|
|
// set large fields to null
|
|
this.disposedValue = true;
|
|
}
|
|
}
|
|
|
|
// override finalizer only if 'Dispose(bool disposing)' has code to free unmanaged resources
|
|
// ~Recorder()
|
|
// {
|
|
// // Do not change this code. Put cleanup code in 'Dispose(bool disposing)' method
|
|
// Dispose(disposing: false);
|
|
// }
|
|
|
|
public void Dispose()
|
|
{
|
|
// Do not change this code. Put cleanup code in 'Dispose(bool disposing)' method
|
|
this.Dispose(disposing: true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|