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:
rootdarkarchon
2023-02-04 02:04:26 +01:00
parent 680c5f4712
commit fc3ad1f7f8
6 changed files with 58 additions and 24 deletions

View File

@@ -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);
} }
} }

View File

@@ -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()

View File

@@ -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();
} }

View File

@@ -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];

View File

@@ -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;
} }

View File

@@ -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;
} }
} }