* Properly cache regexes in chat sanitization/accents Wow I wonder if `new Regex()` has a cost to it *looks at server profile*. * Avoid lag caused by Tippy command completions CompletionHelper.PrototypeIDs explicitly says *not* to use it with EntityPrototype. Unsurprisingly, reporting a completion result for every entity prototype in the game is a *bad idea*. * Add active count metrics to some high-load systems Mover & NPCs I suspect the thing that caused the Leviathan round to shit itself on performance is NPC spam in space or something. So let's verify that. * Enable parallel processing on pow3r again Originally disabled due to a theory of it causing bugs, it was re-enabled on Vulture, and I'm not aware of it having caused any issues there. * Replace hashset with bitflags for AtmosMonitor alert types. Allocating these hashsets was like 20% of the CPU of atmos, somehow. * Cache HashSet used for space movement collider checks Turns out this was a ton of server allocations. Huh.
486 lines
19 KiB
C#
486 lines
19 KiB
C#
using Content.Server.Atmos.EntitySystems;
|
|
using Content.Server.Atmos.Monitor.Components;
|
|
using Content.Server.Atmos.Piping.Components;
|
|
using Content.Server.Atmos.Piping.EntitySystems;
|
|
using Content.Server.DeviceNetwork;
|
|
using Content.Server.DeviceNetwork.Systems;
|
|
using Content.Server.NodeContainer;
|
|
using Content.Server.NodeContainer.EntitySystems;
|
|
using Content.Server.NodeContainer.Nodes;
|
|
using Content.Server.Power.Components;
|
|
using Content.Server.Power.EntitySystems;
|
|
using Content.Shared.Administration.Logs;
|
|
using Content.Shared.Atmos;
|
|
using Content.Shared.Atmos.Monitor;
|
|
using Content.Shared.Atmos.Piping.Components;
|
|
using Content.Shared.Database;
|
|
using Content.Shared.DeviceNetwork;
|
|
using Content.Shared.DeviceNetwork.Events;
|
|
using Content.Shared.Power;
|
|
using Content.Shared.Tag;
|
|
using Robust.Shared.Prototypes;
|
|
|
|
namespace Content.Server.Atmos.Monitor.Systems;
|
|
|
|
// AtmosMonitorSystem. Grabs all the AtmosAlarmables connected
|
|
// to it via local APC net, and starts sending updates of the
|
|
// current atmosphere. Monitors fire (which always triggers as
|
|
// a danger), and atmos (which triggers based on set thresholds).
|
|
public sealed class AtmosMonitorSystem : EntitySystem
|
|
{
|
|
[Dependency] private readonly ISharedAdminLogManager _adminLogger = default!;
|
|
[Dependency] private readonly AtmosphereSystem _atmosphereSystem = default!;
|
|
[Dependency] private readonly AtmosDeviceSystem _atmosDeviceSystem = default!;
|
|
[Dependency] private readonly DeviceNetworkSystem _deviceNetSystem = default!;
|
|
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
|
[Dependency] private readonly NodeContainerSystem _nodeContainerSystem = default!;
|
|
|
|
// Commands
|
|
public const string AtmosMonitorSetThresholdCmd = "atmos_monitor_set_threshold";
|
|
public const string AtmosMonitorSetAllThresholdsCmd = "atmos_monitor_set_all_thresholds";
|
|
|
|
// Packet data
|
|
public const string AtmosMonitorThresholdData = "atmos_monitor_threshold_data";
|
|
public const string AtmosMonitorAllThresholdData = "atmos_monitor_all_threshold_data";
|
|
public const string AtmosMonitorThresholdDataType = "atmos_monitor_threshold_type";
|
|
|
|
public const string AtmosMonitorThresholdGasType = "atmos_monitor_threshold_gas";
|
|
|
|
public override void Initialize()
|
|
{
|
|
SubscribeLocalEvent<AtmosMonitorComponent, ComponentStartup>(OnAtmosMonitorStartup);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, MapInitEvent>(OnMapInit);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, AtmosDeviceUpdateEvent>(OnAtmosUpdate);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, TileFireEvent>(OnFireEvent);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, PowerChangedEvent>(OnPowerChangedEvent);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, BeforePacketSentEvent>(BeforePacketRecv);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, DeviceNetworkPacketEvent>(OnPacketRecv);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, AtmosDeviceDisabledEvent>(OnAtmosDeviceLeaveAtmosphere);
|
|
SubscribeLocalEvent<AtmosMonitorComponent, AtmosDeviceEnabledEvent>(OnAtmosDeviceEnterAtmosphere);
|
|
}
|
|
|
|
private void OnAtmosDeviceLeaveAtmosphere(EntityUid uid, AtmosMonitorComponent atmosMonitor, ref AtmosDeviceDisabledEvent args)
|
|
{
|
|
atmosMonitor.TileGas = null;
|
|
}
|
|
|
|
private void OnAtmosDeviceEnterAtmosphere(EntityUid uid, AtmosMonitorComponent atmosMonitor, ref AtmosDeviceEnabledEvent args)
|
|
{
|
|
if (atmosMonitor.MonitorsPipeNet && _nodeContainerSystem.TryGetNode<PipeNode>(uid, atmosMonitor.NodeNameMonitoredPipe, out var pipeNode))
|
|
{
|
|
atmosMonitor.TileGas = pipeNode.Air;
|
|
return;
|
|
}
|
|
|
|
atmosMonitor.TileGas = _atmosphereSystem.GetContainingMixture(uid, true);
|
|
}
|
|
|
|
private void OnMapInit(EntityUid uid, AtmosMonitorComponent component, MapInitEvent args)
|
|
{
|
|
if (component.TemperatureThresholdId != null)
|
|
{
|
|
var proto = _prototypeManager.Index<AtmosAlarmThresholdPrototype>(component.TemperatureThresholdId);
|
|
component.TemperatureThreshold ??= new(proto);
|
|
}
|
|
|
|
if (component.PressureThresholdId != null)
|
|
{
|
|
var proto = _prototypeManager.Index<AtmosAlarmThresholdPrototype>(component.PressureThresholdId);
|
|
component.PressureThreshold ??= new(proto);
|
|
}
|
|
|
|
if (component.GasThresholdPrototypes == null)
|
|
return;
|
|
|
|
component.GasThresholds ??= new();
|
|
foreach (var (gas, id) in component.GasThresholdPrototypes)
|
|
{
|
|
var proto = _prototypeManager.Index<AtmosAlarmThresholdPrototype>(id);
|
|
component.GasThresholds.TryAdd(gas, new(proto));
|
|
}
|
|
}
|
|
|
|
private void OnAtmosMonitorStartup(EntityUid uid, AtmosMonitorComponent component, ComponentStartup args)
|
|
{
|
|
if (!HasComp<ApcPowerReceiverComponent>(uid)
|
|
&& TryComp<AtmosDeviceComponent>(uid, out var atmosDeviceComponent))
|
|
{
|
|
_atmosDeviceSystem.LeaveAtmosphere((uid, atmosDeviceComponent));
|
|
}
|
|
}
|
|
|
|
private void BeforePacketRecv(EntityUid uid, AtmosMonitorComponent component, BeforePacketSentEvent args)
|
|
{
|
|
if (!component.NetEnabled) args.Cancel();
|
|
}
|
|
|
|
private void OnPacketRecv(EntityUid uid, AtmosMonitorComponent component, DeviceNetworkPacketEvent args)
|
|
{
|
|
// sync the internal 'last alarm state' from
|
|
// the other alarms, so that we can calculate
|
|
// the highest network alarm state at any time
|
|
if (!args.Data.TryGetValue(DeviceNetworkConstants.Command, out string? cmd))
|
|
{
|
|
return;
|
|
}
|
|
|
|
switch (cmd)
|
|
{
|
|
case AtmosDeviceNetworkSystem.RegisterDevice:
|
|
component.RegisteredDevices.Add(args.SenderAddress);
|
|
break;
|
|
case AtmosDeviceNetworkSystem.DeregisterDevice:
|
|
component.RegisteredDevices.Remove(args.SenderAddress);
|
|
break;
|
|
case AtmosAlarmableSystem.ResetAll:
|
|
Reset(uid);
|
|
// Don't clear alarm states here.
|
|
break;
|
|
case AtmosMonitorSetThresholdCmd:
|
|
if (args.Data.TryGetValue(AtmosMonitorThresholdData, out AtmosAlarmThreshold? thresholdData)
|
|
&& args.Data.TryGetValue(AtmosMonitorThresholdDataType, out AtmosMonitorThresholdType? thresholdType))
|
|
{
|
|
args.Data.TryGetValue(AtmosMonitorThresholdGasType, out Gas? gas);
|
|
SetThreshold(uid, thresholdType.Value, thresholdData, gas);
|
|
}
|
|
break;
|
|
case AtmosMonitorSetAllThresholdsCmd:
|
|
if (args.Data.TryGetValue(AtmosMonitorAllThresholdData, out AtmosSensorData? allThresholdData))
|
|
{
|
|
SetAllThresholds(uid, allThresholdData);
|
|
}
|
|
break;
|
|
case AtmosDeviceNetworkSystem.SyncData:
|
|
var payload = new NetworkPayload();
|
|
payload.Add(DeviceNetworkConstants.Command, AtmosDeviceNetworkSystem.SyncData);
|
|
if (component.TileGas != null)
|
|
{
|
|
var gases = new Dictionary<Gas, float>();
|
|
foreach (var gas in Enum.GetValues<Gas>())
|
|
{
|
|
gases.Add(gas, component.TileGas.GetMoles(gas));
|
|
}
|
|
|
|
payload.Add(AtmosDeviceNetworkSystem.SyncData, new AtmosSensorData(
|
|
component.TileGas.Pressure,
|
|
component.TileGas.Temperature,
|
|
component.TileGas.TotalMoles,
|
|
component.LastAlarmState,
|
|
gases,
|
|
component.PressureThreshold ?? new(),
|
|
component.TemperatureThreshold ?? new(),
|
|
component.GasThresholds ?? new()
|
|
));
|
|
}
|
|
|
|
_deviceNetSystem.QueuePacket(uid, args.SenderAddress, payload);
|
|
Alert(uid, component.LastAlarmState);
|
|
break;
|
|
}
|
|
}
|
|
|
|
private void OnPowerChangedEvent(Entity<AtmosMonitorComponent> ent, ref PowerChangedEvent args)
|
|
{
|
|
if (TryComp<AtmosDeviceComponent>(ent, out var atmosDeviceComponent))
|
|
{
|
|
if (!args.Powered)
|
|
{
|
|
_atmosDeviceSystem.LeaveAtmosphere((ent, atmosDeviceComponent));
|
|
}
|
|
else
|
|
{
|
|
_atmosDeviceSystem.JoinAtmosphere((ent, atmosDeviceComponent));
|
|
Alert(ent, ent.Comp.LastAlarmState);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void OnFireEvent(EntityUid uid, AtmosMonitorComponent component, ref TileFireEvent args)
|
|
{
|
|
if (!this.IsPowered(uid, EntityManager))
|
|
return;
|
|
|
|
// if we're monitoring for atmos fire, then we make it similar to a smoke detector
|
|
// and just outright trigger a danger event
|
|
//
|
|
// somebody else can reset it :sunglasses:
|
|
if (component.MonitorFire
|
|
&& component.LastAlarmState != AtmosAlarmType.Danger)
|
|
{
|
|
component.TrippedThresholds |= AtmosMonitorThresholdTypeFlags.Temperature;
|
|
Alert(uid, AtmosAlarmType.Danger, null, component); // technically???
|
|
}
|
|
|
|
// only monitor state elevation so that stuff gets alarmed quicker during a fire,
|
|
// let the atmos update loop handle when temperature starts to reach different
|
|
// thresholds and different states than normal -> warning -> danger
|
|
if (component.TemperatureThreshold != null
|
|
&& component.TemperatureThreshold.CheckThreshold(args.Temperature, out var temperatureState)
|
|
&& temperatureState > component.LastAlarmState)
|
|
{
|
|
component.TrippedThresholds |= AtmosMonitorThresholdTypeFlags.Temperature;
|
|
Alert(uid, AtmosAlarmType.Danger, null, component);
|
|
}
|
|
}
|
|
|
|
private void OnAtmosUpdate(EntityUid uid, AtmosMonitorComponent component, ref AtmosDeviceUpdateEvent args)
|
|
{
|
|
if (!this.IsPowered(uid, EntityManager))
|
|
return;
|
|
|
|
if (args.Grid == null)
|
|
return;
|
|
|
|
// if we're not monitoring atmos, don't bother
|
|
if (component.TemperatureThreshold == null
|
|
&& component.PressureThreshold == null
|
|
&& component.GasThresholds == null)
|
|
return;
|
|
|
|
// If monitoring a pipe network, get its most recent gas mixture
|
|
if (component.MonitorsPipeNet && _nodeContainerSystem.TryGetNode<PipeNode>(uid, component.NodeNameMonitoredPipe, out var pipeNode))
|
|
component.TileGas = pipeNode.Air;
|
|
|
|
UpdateState(uid, component.TileGas, component);
|
|
}
|
|
|
|
// Update checks the current air if it exceeds thresholds of
|
|
// any kind.
|
|
//
|
|
// If any threshold exceeds the other, that threshold
|
|
// immediately replaces the current recorded state.
|
|
//
|
|
// If the threshold does not match the current state
|
|
// of the monitor, it is set in the Alert call.
|
|
private void UpdateState(EntityUid uid, GasMixture? air, AtmosMonitorComponent? monitor = null)
|
|
{
|
|
if (air == null) return;
|
|
|
|
if (!Resolve(uid, ref monitor)) return;
|
|
|
|
var state = AtmosAlarmType.Normal;
|
|
var alarmTypes = monitor.TrippedThresholds;
|
|
|
|
if (monitor.TemperatureThreshold != null
|
|
&& monitor.TemperatureThreshold.CheckThreshold(air.Temperature, out var temperatureState))
|
|
{
|
|
if (temperatureState > state)
|
|
{
|
|
state = temperatureState;
|
|
alarmTypes |= AtmosMonitorThresholdTypeFlags.Temperature;
|
|
}
|
|
else if (temperatureState == AtmosAlarmType.Normal)
|
|
{
|
|
alarmTypes &= ~AtmosMonitorThresholdTypeFlags.Temperature;
|
|
}
|
|
}
|
|
|
|
if (monitor.PressureThreshold != null
|
|
&& monitor.PressureThreshold.CheckThreshold(air.Pressure, out var pressureState)
|
|
)
|
|
{
|
|
if (pressureState > state)
|
|
{
|
|
state = pressureState;
|
|
alarmTypes |= AtmosMonitorThresholdTypeFlags.Pressure;
|
|
}
|
|
else if (pressureState == AtmosAlarmType.Normal)
|
|
{
|
|
alarmTypes &= ~AtmosMonitorThresholdTypeFlags.Pressure;
|
|
}
|
|
}
|
|
|
|
if (monitor.GasThresholds != null)
|
|
{
|
|
var tripped = false;
|
|
foreach (var (gas, threshold) in monitor.GasThresholds)
|
|
{
|
|
var gasRatio = air.GetMoles(gas) / air.TotalMoles;
|
|
if (threshold.CheckThreshold(gasRatio, out var gasState)
|
|
&& gasState > state)
|
|
{
|
|
state = gasState;
|
|
tripped = true;
|
|
}
|
|
}
|
|
|
|
if (tripped)
|
|
{
|
|
alarmTypes |= AtmosMonitorThresholdTypeFlags.Gas;
|
|
}
|
|
else
|
|
{
|
|
alarmTypes &= ~AtmosMonitorThresholdTypeFlags.Gas;
|
|
}
|
|
}
|
|
|
|
// if the state of the current air doesn't match the last alarm state,
|
|
// we update the state
|
|
if (state != monitor.LastAlarmState || alarmTypes != monitor.TrippedThresholds)
|
|
{
|
|
Alert(uid, state, alarmTypes, monitor);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Alerts the network that the state of a monitor has changed.
|
|
/// </summary>
|
|
/// <param name="state">The alarm state to set this monitor to.</param>
|
|
/// <param name="alarms">The alarms that caused this alarm state.</param>
|
|
public void Alert(EntityUid uid, AtmosAlarmType state, AtmosMonitorThresholdTypeFlags? alarms = null, AtmosMonitorComponent? monitor = null)
|
|
{
|
|
if (!Resolve(uid, ref monitor))
|
|
return;
|
|
|
|
monitor.LastAlarmState = state;
|
|
monitor.TrippedThresholds = alarms ?? monitor.TrippedThresholds;
|
|
|
|
BroadcastAlertPacket((uid, monitor));
|
|
|
|
// TODO: Central system that grabs *all* alarms from wired network
|
|
}
|
|
|
|
/// <summary>
|
|
/// Resets a single monitor's alarm.
|
|
/// </summary>
|
|
private void Reset(EntityUid uid)
|
|
{
|
|
Alert(uid, AtmosAlarmType.Normal);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Broadcasts an alert packet to all devices on the network,
|
|
/// which consists of the current alarm types,
|
|
/// the highest alarm currently cached by this monitor,
|
|
/// and the current alarm state of the monitor (so other
|
|
/// alarms can sync to it).
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Alarmables use the highest alarm to ensure that a monitor's
|
|
/// state doesn't override if the alarm is lower. The state
|
|
/// is synced between monitors the moment a monitor sends out an alarm,
|
|
/// or if it is explicitly synced (see ResetAll/Sync).
|
|
/// </remarks>
|
|
private void BroadcastAlertPacket(Entity<AtmosMonitorComponent> ent, TagComponent? tags = null)
|
|
{
|
|
var (owner, monitor) = ent;
|
|
if (!monitor.NetEnabled)
|
|
return;
|
|
|
|
if (!Resolve(owner, ref tags, false))
|
|
{
|
|
return;
|
|
}
|
|
|
|
var payload = new NetworkPayload
|
|
{
|
|
[DeviceNetworkConstants.Command] = AtmosAlarmableSystem.AlertCmd,
|
|
[DeviceNetworkConstants.CmdSetState] = monitor.LastAlarmState,
|
|
[AtmosAlarmableSystem.AlertSource] = tags.Tags,
|
|
[AtmosAlarmableSystem.AlertTypes] = monitor.TrippedThresholds
|
|
};
|
|
|
|
foreach (var addr in monitor.RegisteredDevices)
|
|
{
|
|
_deviceNetSystem.QueuePacket(owner, addr, payload);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Set a monitor's threshold.
|
|
/// </summary>
|
|
/// <param name="type">The type of threshold to change.</param>
|
|
/// <param name="threshold">Threshold data.</param>
|
|
/// <param name="gas">Gas, if applicable.</param>
|
|
public void SetThreshold(EntityUid uid, AtmosMonitorThresholdType type, AtmosAlarmThreshold threshold, Gas? gas = null, AtmosMonitorComponent? monitor = null)
|
|
{
|
|
if (!Resolve(uid, ref monitor))
|
|
return;
|
|
|
|
// Used for logging after the switch statement
|
|
string logPrefix = "";
|
|
string logValueSuffix = "";
|
|
AtmosAlarmThreshold? logPreviousThreshold = null;
|
|
|
|
switch (type)
|
|
{
|
|
case AtmosMonitorThresholdType.Pressure:
|
|
logPrefix = "pressure";
|
|
logValueSuffix = "kPa";
|
|
logPreviousThreshold = monitor.PressureThreshold;
|
|
|
|
monitor.PressureThreshold = threshold;
|
|
break;
|
|
case AtmosMonitorThresholdType.Temperature:
|
|
logPrefix = "temperature";
|
|
logValueSuffix = "K";
|
|
logPreviousThreshold = monitor.TemperatureThreshold;
|
|
|
|
monitor.TemperatureThreshold = threshold;
|
|
break;
|
|
case AtmosMonitorThresholdType.Gas:
|
|
if (gas == null || monitor.GasThresholds == null)
|
|
return;
|
|
|
|
logPrefix = ((Gas) gas).ToString();
|
|
logValueSuffix = "kPa";
|
|
monitor.GasThresholds.TryGetValue((Gas) gas, out logPreviousThreshold);
|
|
|
|
monitor.GasThresholds[(Gas) gas] = threshold;
|
|
break;
|
|
}
|
|
|
|
// Admin log each change separately rather than logging the whole state
|
|
if (logPreviousThreshold != null)
|
|
{
|
|
if (threshold.Ignore != logPreviousThreshold.Ignore)
|
|
{
|
|
string enabled = threshold.Ignore ? "disabled" : "enabled";
|
|
_adminLogger.Add(
|
|
LogType.AtmosDeviceSetting,
|
|
LogImpact.Medium,
|
|
$"{ToPrettyString(uid)} {logPrefix} thresholds {enabled}"
|
|
);
|
|
}
|
|
|
|
foreach (var change in threshold.GetChanges(logPreviousThreshold))
|
|
{
|
|
if (change.Current.Enabled != change.Previous?.Enabled)
|
|
{
|
|
string enabled = change.Current.Enabled ? "enabled" : "disabled";
|
|
_adminLogger.Add(
|
|
LogType.AtmosDeviceSetting,
|
|
LogImpact.Medium,
|
|
$"{ToPrettyString(uid)} {logPrefix} {change.Type} {enabled}"
|
|
);
|
|
}
|
|
|
|
if (change.Current.Value != change.Previous?.Value)
|
|
{
|
|
_adminLogger.Add(
|
|
LogType.AtmosDeviceSetting,
|
|
LogImpact.Medium,
|
|
$"{ToPrettyString(uid)} {logPrefix} {change.Type} changed from {change.Previous?.Value} {logValueSuffix} to {change.Current.Value} {logValueSuffix}"
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets all of a monitor's thresholds at once according to the incoming
|
|
/// AtmosSensorData object's thresholds.
|
|
/// </summary>
|
|
/// <param name="uid">The entity's uid</param>
|
|
/// <param name="allThresholdData">An AtmosSensorData object from which the thresholds will be loaded.</param>
|
|
public void SetAllThresholds(EntityUid uid, AtmosSensorData allThresholdData)
|
|
{
|
|
SetThreshold(uid, AtmosMonitorThresholdType.Temperature, allThresholdData.TemperatureThreshold);
|
|
SetThreshold(uid, AtmosMonitorThresholdType.Pressure, allThresholdData.PressureThreshold);
|
|
foreach (var gas in Enum.GetValues<Gas>())
|
|
{
|
|
SetThreshold(uid, AtmosMonitorThresholdType.Gas, allThresholdData.GasThresholds[gas], gas);
|
|
}
|
|
}
|
|
}
|