using System.ComponentModel.DataAnnotations;
namespace ZB.MOM.WW.OtOpcUa.Driver.FOCAS;
///
/// FOCAS driver configuration. One instance supports N CNC devices. Per plan decision #144
/// each device gets its own (DriverInstanceId, HostAddress) bulkhead key at the
/// Phase 6.1 resilience layer.
///
public sealed class FocasDriverOptions
{
/// Gets the list of configured CNC devices.
public IReadOnlyList Devices { get; init; } = [];
/// Gets the list of FOCAS tag definitions.
public IReadOnlyList Tags { get; init; } = [];
/// Gets the probe options.
public FocasProbeOptions Probe { get; init; } = new();
/// Gets the timeout duration for operations.
public TimeSpan Timeout { get; init; } = TimeSpan.FromSeconds(2);
/// Gets the alarm projection options.
public FocasAlarmProjectionOptions AlarmProjection { get; init; } = new();
/// Gets the handle recycle options.
public FocasHandleRecycleOptions HandleRecycle { get; init; } = new();
/// Gets the fixed tree options.
public FocasFixedTreeOptions FixedTree { get; init; } = new();
///
/// Timeout for the AdminUI Test Connect probe, in seconds. The AdminUI clamps to a
/// 60s server-side maximum; this default is what the form pre-fills for new instances.
///
[Display(Name = "Probe timeout (seconds)", Description = "Connection test timeout. Default 10s.", GroupName = "Diagnostics")]
[Range(1, 60)]
public int ProbeTimeoutSeconds { get; init; } = 10;
}
///
/// Fixed-node tree exposed by FOCAS per docs/v2/driver-specs.md §7 —
/// Identity/, Axes/{name}/, etc. populated from
/// cnc_sysinfo / cnc_rdaxisname / cnc_rddynamic2. Disabled by
/// default so existing configs that only use user-authored tags don't grow new
/// nodes on upgrade.
///
public sealed class FocasFixedTreeOptions
{
/// Gets or sets a value indicating whether the fixed-node tree is enabled for every configured device.
public bool Enabled { get; init; } = false;
///
/// Poll cadence for cnc_rddynamic2. Each tick calls the API once per
/// configured axis + publishes OnDataChange for the axis subtree. Real CNCs
/// serve ~100ms loops comfortably; the default is conservative.
///
public TimeSpan PollInterval { get; init; } = TimeSpan.FromMilliseconds(250);
///
/// Poll cadence for program + operation-mode info. Slower than the axis
/// poll because program / mode transitions happen on operator timescales.
/// Zero / negative disables the program poll entirely.
///
public TimeSpan ProgramPollInterval { get; init; } = TimeSpan.FromSeconds(1);
///
/// Poll cadence for timers (power-on / operating / cutting / cycle).
/// These change at human timescales — default is 30s. Zero / negative
/// disables the timer poll entirely.
///
public TimeSpan TimerPollInterval { get; init; } = TimeSpan.FromSeconds(30);
}
///
/// Proactive session-recycle cadence. Fanuc CNCs have a finite FWLIB handle pool
/// (~5–10 concurrent connections) and certain series have documented handle-leak bugs
/// that manifest after long uptime. When is true the
/// driver closes + reopens each device's session on the cadence,
/// forcing FWLIB to release its handle slot back to the pool. Reads / writes during
/// recycle wait for the reconnect rather than failing — worst case an operator sees a
/// brief read latency spike once per cadence.
///
///
/// Disabled by default because a healthy CNC + driver doesn't need it. Enable when
/// field experience shows handle exhaustion against a specific series / firmware.
/// Typical tuning: 30 min for sites running multiple OtOpcUa instances against the
/// same CNC (they share the pool); 6 h for a single-client deployment.
///
public sealed class FocasHandleRecycleOptions
{
/// Gets or sets a value indicating whether handle recycling is enabled.
public bool Enabled { get; init; } = false;
/// Gets or sets the interval for handle recycle operations.
public TimeSpan Interval { get; init; } = TimeSpan.FromHours(1);
}
///
/// Controls the CNC active-alarm polling projection that surfaces FOCAS alarms via
/// IAlarmSource. Disabled by default — operators opt in by setting
/// in appsettings.json.
///
public sealed class FocasAlarmProjectionOptions
{
/// Gets or sets a value indicating whether alarm projection is enabled.
public bool Enabled { get; init; } = false;
/// Poll cadence. One cnc_rdalmmsg2 call per device per tick.
public TimeSpan PollInterval { get; init; } = TimeSpan.FromSeconds(2);
}
///
/// One CNC the driver talks to. enables per-series
/// address validation at FocasDriver.InitializeAsync; leave as
/// to skip validation (legacy behaviour).
///
///
/// FOCAS TCP endpoint in focas://{ip}[:{port}] form, e.g.
/// focas://10.20.30.40:8193. The port defaults to 8193 when omitted.
/// Parsed by FocasHostAddress.TryParse at FocasDriver.InitializeAsync.
///
///
/// Optional human-readable label shown in the OPC UA address-space folder for this
/// device. Defaults to when null.
///
///
/// CNC controller series. Used by to gate which
/// FOCAS addresses are valid. Leave as (the
/// default) to skip range validation and preserve legacy behaviour.
///
///
/// Axis positions returned by cnc_rddynamic2 are scaled integers. The driver
/// divides AbsolutePosition / MachinePosition / RelativePosition / DistanceToGo by
/// 10^PositionDecimalPlaces at the publish seam so they surface in engineering
/// units on the Float64 axis nodes. Default 0 (no scaling) is byte-identical to
/// legacy behaviour. Auto-fetching this via cnc_getfigure is deferred (wire-gated),
/// so it is config-supplied. Negative values are clamped to 0 (no scaling).
///
public sealed record FocasDeviceOptions(
string HostAddress,
string? DeviceName = null,
FocasCncSeries Series = FocasCncSeries.Unknown,
int PositionDecimalPlaces = 0)
{
///
/// Axis-position decimal places, clamped to a non-negative value so the
/// 10^PositionDecimalPlaces divide at the publish seam can never misbehave.
///
public int PositionDecimalPlaces { get; init; } =
PositionDecimalPlaces < 0 ? 0 : PositionDecimalPlaces;
}
///
/// One FOCAS-backed OPC UA variable. is the canonical FOCAS
/// address string that parses via FocasAddress.TryParse —
/// X0.0 / R100 / PARAM:1815/0 / MACRO:500.
///
///
/// Whether the tag is declared writable in config. Reflected at the
/// DiscoverAsync address-space advertisement seam and gates the driver's
/// write path — FOCAS supports PMC/data writes (see FocasPmcBitRmwTests /
/// FocasReadWriteTests). Defaults to true.
///
///
/// Whether repeated writes of the same value are safe. Threaded through to
/// DriverAttributeInfo.WriteIdempotent by DiscoverAsync so OPC UA
/// clients can optimise write coalescing for idempotent tags. Defaults to false.
///
public sealed record FocasTagDefinition(
string Name,
string DeviceHostAddress,
string Address,
FocasDataType DataType,
bool Writable = true,
bool WriteIdempotent = false);
///
/// Controls periodic connectivity probing. One cnc_rdcncstat call per
/// configured device per tick; transitions fire OnHostStatusChanged.
/// Enabled by default so the driver surfaces a fast host-state transition when
/// a CNC goes offline between data reads.
///
public sealed class FocasProbeOptions
{
/// Gets or sets a value indicating whether probing is enabled.
public bool Enabled { get; init; } = true;
/// Gets or sets the probe interval.
public TimeSpan Interval { get; init; } = TimeSpan.FromSeconds(5);
/// Gets or sets the probe timeout.
public TimeSpan Timeout { get; init; } = TimeSpan.FromSeconds(2);
}