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