attempt to gracefully reconnect, do not send notification for player on connect, do not check other players on framework update thread, delay palette+ sending data
This commit is contained in:
@@ -15,6 +15,7 @@ public class CacheCreationService : MediatorSubscriberBase, IDisposable
|
|||||||
private readonly CharacterData _lastCreatedData = new();
|
private readonly CharacterData _lastCreatedData = new();
|
||||||
private readonly CancellationTokenSource _cts = new();
|
private readonly CancellationTokenSource _cts = new();
|
||||||
private readonly List<GameObjectHandler> _playerRelatedObjects = new();
|
private readonly List<GameObjectHandler> _playerRelatedObjects = new();
|
||||||
|
private CancellationTokenSource _palettePlusCts = new();
|
||||||
|
|
||||||
public unsafe CacheCreationService(MareMediator mediator, CharacterDataFactory characterDataFactory, DalamudUtil dalamudUtil) : base(mediator)
|
public unsafe CacheCreationService(MareMediator mediator, CharacterDataFactory characterDataFactory, DalamudUtil dalamudUtil) : base(mediator)
|
||||||
{
|
{
|
||||||
@@ -45,7 +46,17 @@ public class CacheCreationService : MediatorSubscriberBase, IDisposable
|
|||||||
if (!string.Equals(msg.Data, _lastCreatedData.PalettePlusPalette, StringComparison.Ordinal))
|
if (!string.Equals(msg.Data, _lastCreatedData.PalettePlusPalette, StringComparison.Ordinal))
|
||||||
{
|
{
|
||||||
_lastCreatedData.PalettePlusPalette = msg.Data ?? string.Empty;
|
_lastCreatedData.PalettePlusPalette = msg.Data ?? string.Empty;
|
||||||
Mediator.Publish(new CharacterDataCreatedMessage(_lastCreatedData));
|
|
||||||
|
_palettePlusCts?.Cancel();
|
||||||
|
_palettePlusCts?.Dispose();
|
||||||
|
_palettePlusCts = new();
|
||||||
|
var token = _palettePlusCts.Token;
|
||||||
|
|
||||||
|
Task.Run(async () =>
|
||||||
|
{
|
||||||
|
await Task.Delay(TimeSpan.FromSeconds(1), token).ConfigureAwait(false);
|
||||||
|
Mediator.Publish(new CharacterDataCreatedMessage(_lastCreatedData));
|
||||||
|
}, token);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ public class CachedPlayer : MediatorSubscriberBase, IDisposable
|
|||||||
private readonly IpcManager _ipcManager;
|
private readonly IpcManager _ipcManager;
|
||||||
private readonly FileCacheManager _fileDbManager;
|
private readonly FileCacheManager _fileDbManager;
|
||||||
private API.Data.CharacterData _cachedData = new();
|
private API.Data.CharacterData _cachedData = new();
|
||||||
private GameObjectHandler? _currentCharacterEquipment;
|
private GameObjectHandler? _currentOtherChara;
|
||||||
private CancellationTokenSource? _downloadCancellationTokenSource = new();
|
private CancellationTokenSource? _downloadCancellationTokenSource = new();
|
||||||
private bool _isVisible;
|
private bool _isVisible;
|
||||||
|
|
||||||
@@ -209,7 +209,7 @@ public class CachedPlayer : MediatorSubscriberBase, IDisposable
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_currentCharacterEquipment?.CheckAndUpdateObject() ?? false)
|
if (_currentOtherChara?.CheckAndUpdateObject() ?? false)
|
||||||
{
|
{
|
||||||
OnPlayerChanged();
|
OnPlayerChanged();
|
||||||
}
|
}
|
||||||
@@ -228,7 +228,7 @@ public class CachedPlayer : MediatorSubscriberBase, IDisposable
|
|||||||
Logger.Debug("Disposing " + PlayerName + " (" + OnlineUser + ")");
|
Logger.Debug("Disposing " + PlayerName + " (" + OnlineUser + ")");
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
_currentCharacterEquipment?.Dispose();
|
_currentOtherChara?.Dispose();
|
||||||
Logger.Verbose("Restoring state for " + PlayerName);
|
Logger.Verbose("Restoring state for " + PlayerName);
|
||||||
_ipcManager.PenumbraRemoveTemporaryCollection(PlayerName);
|
_ipcManager.PenumbraRemoveTemporaryCollection(PlayerName);
|
||||||
_downloadCancellationTokenSource?.Cancel();
|
_downloadCancellationTokenSource?.Cancel();
|
||||||
@@ -267,7 +267,7 @@ public class CachedPlayer : MediatorSubscriberBase, IDisposable
|
|||||||
|
|
||||||
Mediator.Subscribe<PenumbraRedrawMessage>(this, (msg) => IpcManagerOnPenumbraRedrawEvent(((PenumbraRedrawMessage)msg)));
|
Mediator.Subscribe<PenumbraRedrawMessage>(this, (msg) => IpcManagerOnPenumbraRedrawEvent(((PenumbraRedrawMessage)msg)));
|
||||||
_originalGlamourerData = _ipcManager.GlamourerGetCharacterCustomization(PlayerCharacter);
|
_originalGlamourerData = _ipcManager.GlamourerGetCharacterCustomization(PlayerCharacter);
|
||||||
_currentCharacterEquipment = new GameObjectHandler(Mediator, ObjectKind.Player, () => _dalamudUtil.GetPlayerCharacterFromObjectTableByName(PlayerName)?.Address ?? IntPtr.Zero, false);
|
_currentOtherChara = new GameObjectHandler(Mediator, ObjectKind.Player, () => _dalamudUtil.GetPlayerCharacterFromObjectTableByName(PlayerName)?.Address ?? IntPtr.Zero, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override string ToString()
|
public override string ToString()
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ public class PairManager : MediatorSubscriberBase, IDisposable
|
|||||||
_configurationService = configurationService;
|
_configurationService = configurationService;
|
||||||
Mediator.Subscribe<ZoneSwitchStartMessage>(this, (_) => DalamudUtilOnZoneSwitched());
|
Mediator.Subscribe<ZoneSwitchStartMessage>(this, (_) => DalamudUtilOnZoneSwitched());
|
||||||
Mediator.Subscribe<DelayedFrameworkUpdateMessage>(this, (_) => DalamudUtilOnDelayedFrameworkUpdate());
|
Mediator.Subscribe<DelayedFrameworkUpdateMessage>(this, (_) => DalamudUtilOnDelayedFrameworkUpdate());
|
||||||
|
Mediator.Subscribe<DisconnectedMessage>(this, (_) => ClearPairs());
|
||||||
_directPairsInternal = DirectPairsLazy();
|
_directPairsInternal = DirectPairsLazy();
|
||||||
_groupPairsInternal = GroupPairsLazy();
|
_groupPairsInternal = GroupPairsLazy();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -34,22 +34,25 @@ public class GameObjectHandler : MediatorSubscriberBase
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public GameObjectHandler(MareMediator mediator, ObjectKind objectKind, Func<IntPtr> getAddress, bool sendUpdates = true) : base(mediator)
|
public GameObjectHandler(MareMediator mediator, ObjectKind objectKind, Func<IntPtr> getAddress, bool watchedPlayer = true) : base(mediator)
|
||||||
{
|
{
|
||||||
_mediator = mediator;
|
_mediator = mediator;
|
||||||
ObjectKind = objectKind;
|
ObjectKind = objectKind;
|
||||||
this._getAddress = getAddress;
|
this._getAddress = getAddress;
|
||||||
_sendUpdates = sendUpdates;
|
_sendUpdates = watchedPlayer;
|
||||||
_name = string.Empty;
|
_name = string.Empty;
|
||||||
|
|
||||||
Mediator.Subscribe<TransientResourceChangedMessage>(this, (msg) =>
|
if (watchedPlayer)
|
||||||
{
|
{
|
||||||
var actualMsg = (TransientResourceChangedMessage)msg;
|
Mediator.Subscribe<TransientResourceChangedMessage>(this, (msg) =>
|
||||||
if (actualMsg.Address != Address || !sendUpdates) return;
|
{
|
||||||
Mediator.Publish(new CreateCacheForObjectMessage(this));
|
var actualMsg = (TransientResourceChangedMessage)msg;
|
||||||
});
|
if (actualMsg.Address != Address) return;
|
||||||
|
Mediator.Publish(new CreateCacheForObjectMessage(this));
|
||||||
|
});
|
||||||
|
|
||||||
Mediator.Subscribe<FrameworkUpdateMessage>(this, (_) => CheckAndUpdateObject());
|
Mediator.Subscribe<FrameworkUpdateMessage>(this, (_) => CheckAndUpdateObject());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public byte[] EquipSlotData { get; set; } = new byte[40];
|
public byte[] EquipSlotData { get; set; } = new byte[40];
|
||||||
|
|||||||
@@ -294,7 +294,7 @@ public partial class ApiController : MediatorSubscriberBase, IDisposable, IMareH
|
|||||||
|
|
||||||
foreach (var entry in await UserGetOnlinePairs().ConfigureAwait(false))
|
foreach (var entry in await UserGetOnlinePairs().ConfigureAwait(false))
|
||||||
{
|
{
|
||||||
_pairManager.MarkPairOnline(entry, this, false);
|
_pairManager.MarkPairOnline(entry, this, sendNotif: false);
|
||||||
}
|
}
|
||||||
|
|
||||||
_healthCheckTokenSource?.Cancel();
|
_healthCheckTokenSource?.Cancel();
|
||||||
@@ -323,7 +323,7 @@ public partial class ApiController : MediatorSubscriberBase, IDisposable, IMareH
|
|||||||
options.Headers.Add("Authorization", "Bearer " + _serverManager.GetToken());
|
options.Headers.Add("Authorization", "Bearer " + _serverManager.GetToken());
|
||||||
options.Transports = HttpTransportType.WebSockets | HttpTransportType.ServerSentEvents | HttpTransportType.LongPolling;
|
options.Transports = HttpTransportType.WebSockets | HttpTransportType.ServerSentEvents | HttpTransportType.LongPolling;
|
||||||
})
|
})
|
||||||
.WithAutomaticReconnect(new ForeverRetryPolicy())
|
.WithAutomaticReconnect(new ForeverRetryPolicy(Mediator))
|
||||||
.ConfigureLogging(a =>
|
.ConfigureLogging(a =>
|
||||||
{
|
{
|
||||||
a.ClearProviders().AddProvider(new DalamudLoggingProvider());
|
a.ClearProviders().AddProvider(new DalamudLoggingProvider());
|
||||||
@@ -347,15 +347,10 @@ public partial class ApiController : MediatorSubscriberBase, IDisposable, IMareH
|
|||||||
|
|
||||||
private Task MareHubOnReconnecting(Exception? arg)
|
private Task MareHubOnReconnecting(Exception? arg)
|
||||||
{
|
{
|
||||||
_connectionDto = null;
|
|
||||||
_healthCheckTokenSource?.Cancel();
|
_healthCheckTokenSource?.Cancel();
|
||||||
ServerState = ServerState.Reconnecting;
|
ServerState = ServerState.Reconnecting;
|
||||||
Mediator.Publish(new NotificationMessage("Connection lost", "Connection lost to " + _serverManager.CurrentServer!.ServerName, NotificationType.Error, 5000));
|
Mediator.Publish(new NotificationMessage("Connection lost", "Connection lost to " + _serverManager.CurrentServer!.ServerName, NotificationType.Warning, 5000));
|
||||||
Logger.Warn("Connection closed... Reconnecting");
|
Logger.Warn("Connection closed... Reconnecting", arg);
|
||||||
Logger.Warn(arg?.Message ?? string.Empty);
|
|
||||||
Logger.Warn(arg?.StackTrace ?? string.Empty);
|
|
||||||
Mediator.Publish(new DisconnectedMessage());
|
|
||||||
_pairManager.ClearPairs();
|
|
||||||
return Task.CompletedTask;
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,11 +1,35 @@
|
|||||||
using Microsoft.AspNetCore.SignalR.Client;
|
using MareSynchronos.Mediator;
|
||||||
|
using Microsoft.AspNetCore.SignalR.Client;
|
||||||
|
|
||||||
namespace MareSynchronos.WebAPI.Utils;
|
namespace MareSynchronos.WebAPI.Utils;
|
||||||
|
|
||||||
public class ForeverRetryPolicy : IRetryPolicy
|
public class ForeverRetryPolicy : IRetryPolicy
|
||||||
{
|
{
|
||||||
|
private readonly MareMediator _mediator;
|
||||||
|
private bool _sentDisconnected = false;
|
||||||
|
|
||||||
|
public ForeverRetryPolicy(MareMediator mediator)
|
||||||
|
{
|
||||||
|
_mediator = mediator;
|
||||||
|
}
|
||||||
|
|
||||||
public TimeSpan? NextRetryDelay(RetryContext retryContext)
|
public TimeSpan? NextRetryDelay(RetryContext retryContext)
|
||||||
{
|
{
|
||||||
return TimeSpan.FromSeconds(new Random().Next(10, 20));
|
TimeSpan timeToWait = TimeSpan.FromSeconds(new Random().Next(10, 20));
|
||||||
|
if (retryContext.PreviousRetryCount == 0)
|
||||||
|
{
|
||||||
|
_sentDisconnected = false;
|
||||||
|
timeToWait = TimeSpan.FromSeconds(1);
|
||||||
|
}
|
||||||
|
else if (retryContext.PreviousRetryCount == 1) timeToWait = TimeSpan.FromSeconds(2);
|
||||||
|
else if (retryContext.PreviousRetryCount == 2) timeToWait = TimeSpan.FromSeconds(3);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (!_sentDisconnected)
|
||||||
|
_mediator.Publish(new DisconnectedMessage());
|
||||||
|
_sentDisconnected = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return timeToWait;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user