From 424fa1f82c670bcd8c88b509cf47aa9bad5b3e93 Mon Sep 17 00:00:00 2001 From: Luis Cantero Date: Fri, 26 Jul 2024 22:27:09 +0200 Subject: [PATCH 1/6] Improve catch --- src/PlcServer.cs | 65 ++++++++++++++++++++++++------------------------ 1 file changed, 32 insertions(+), 33 deletions(-) diff --git a/src/PlcServer.cs b/src/PlcServer.cs index 198bb5f5..1ac4550f 100644 --- a/src/PlcServer.cs +++ b/src/PlcServer.cs @@ -182,6 +182,13 @@ public override ResponseHeader Publish( out StatusCodeCollection results, out DiagnosticInfoCollection diagnosticInfos) { + subscriptionId = default; + availableSequenceNumbers = default; + moreNotifications = default; + notificationMessage = default; + results = default; + diagnosticInfos = default; + try { OperationContext context = ValidateRequest(requestHeader, RequestType.Publish); @@ -216,48 +223,40 @@ public override ResponseHeader Publish( return responseHeader; } - catch (ServiceResultException ex) + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadNoSubscription) { MetricsHelper.RecordTotalErrors(nameof(Publish)); - subscriptionId = default; - availableSequenceNumbers = default; - moreNotifications = default; - notificationMessage = default; - results = default; - diagnosticInfos = default; - - if (ex.StatusCode == StatusCodes.BadNoSubscription) - { - _logger.LogDebug( - "Failed to publish: {StatusCode}", - StatusCodes.BadNoSubscription.ToString()); - - return new ResponseHeader { ServiceResult = StatusCodes.BadNoSubscription }; - } + _logger.LogDebug( + ex, + "Failed to publish: {StatusCode}", + StatusCodes.BadNoSubscription.ToString()); - if (ex.StatusCode == StatusCodes.BadSessionIdInvalid) - { - _logger.LogDebug( - "Failed to publish: {StatusCode}", - StatusCodes.BadSessionIdInvalid.ToString()); + return new ResponseHeader { ServiceResult = StatusCodes.BadNoSubscription }; + } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadSessionIdInvalid) + { + MetricsHelper.RecordTotalErrors(nameof(Publish)); - return new ResponseHeader { ServiceResult = StatusCodes.BadSessionIdInvalid }; - } + _logger.LogDebug( + ex, + "Failed to publish: {StatusCode}", + StatusCodes.BadSessionIdInvalid.ToString()); - if (ex.StatusCode == StatusCodes.BadSecureChannelIdInvalid) - { - _logger.LogDebug( - "Failed to publish: {StatusCode}", - StatusCodes.BadSecureChannelIdInvalid.ToString()); + return new ResponseHeader { ServiceResult = StatusCodes.BadSessionIdInvalid }; + } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadSecureChannelIdInvalid) + { + MetricsHelper.RecordTotalErrors(nameof(Publish)); - return new ResponseHeader { ServiceResult = StatusCodes.BadSecureChannelIdInvalid }; - } + _logger.LogDebug( + ex, + "Failed to publish: {StatusCode}", + StatusCodes.BadSecureChannelIdInvalid.ToString()); - _logger.LogError(ex, "Error publishing"); - throw; + return new ResponseHeader { ServiceResult = StatusCodes.BadSecureChannelIdInvalid }; } - catch(Exception ex) + catch (Exception ex) { MetricsHelper.RecordTotalErrors(nameof(Publish)); From 96a2b2874e70283344ef0c2a0f430ccfc37023e8 Mon Sep 17 00:00:00 2001 From: Luis Cantero Date: Mon, 29 Jul 2024 00:42:28 +0200 Subject: [PATCH 2/6] Log mem and monitored items --- src/PlcServer.cs | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/src/PlcServer.cs b/src/PlcServer.cs index 1ac4550f..5fae367a 100644 --- a/src/PlcServer.cs +++ b/src/PlcServer.cs @@ -16,6 +16,7 @@ namespace OpcPlc; using System.Collections.Immutable; using System.Diagnostics; using System.IO; +using System.Linq; using System.Reflection; using System.Threading; @@ -50,10 +51,14 @@ public PlcServer(OpcPlcConfiguration config, PlcSimulation plcSimulation, TimeSe (state) => { try { + var curProc = Process.GetCurrentProcess(); + _logger.LogInformation( - "Open sessions: {Sessions}, open subscriptions: {Subscriptions}", + "Open sessions: {Sessions}, open subscriptions: {Subscriptions}, monitored items: {MonitoredItems}, working set {WorkingSet:N0} MB", ServerInternal.SessionManager.GetSessions().Count, - ServerInternal.SubscriptionManager.GetSubscriptions().Count); + ServerInternal.SubscriptionManager.GetSubscriptions().Count, + ServerInternal.SubscriptionManager.GetSubscriptions().Sum(s => s.MonitoredItemCount), + curProc.WorkingSet64 / 1024 / 1024); } catch { From 01958e1e05d1a80ed73904d33f6bd7fedbdc37d9 Mon Sep 17 00:00:00 2001 From: Luis Cantero Date: Mon, 29 Jul 2024 10:07:45 +0200 Subject: [PATCH 3/6] Log thread info --- src/PlcServer.cs | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/src/PlcServer.cs b/src/PlcServer.cs index 5fae367a..89b12bcb 100644 --- a/src/PlcServer.cs +++ b/src/PlcServer.cs @@ -53,12 +53,23 @@ public PlcServer(OpcPlcConfiguration config, PlcSimulation plcSimulation, TimeSe { var curProc = Process.GetCurrentProcess(); + ThreadPool.GetAvailableThreads(out int availWorkerThreads, out int availCompletionPortThreads); + _logger.LogInformation( - "Open sessions: {Sessions}, open subscriptions: {Subscriptions}, monitored items: {MonitoredItems}, working set {WorkingSet:N0} MB", + "\n\t# Open sessions: {Sessions}\n" + + "\t# Open subscriptions: {Subscriptions}\n" + + "\t# Monitored items: {MonitoredItems:N0}\n" + + "\t# Working set {WorkingSet:N0} MB\n" + + "\t# Available worker threads: {AvailWorkerThreads:N0}\n" + + "\t# Available completion port threads: {AvailCompletionPortThreads:N0}\n" + + "\t# Thread count: {ThreadCount:N0}", ServerInternal.SessionManager.GetSessions().Count, ServerInternal.SubscriptionManager.GetSubscriptions().Count, ServerInternal.SubscriptionManager.GetSubscriptions().Sum(s => s.MonitoredItemCount), - curProc.WorkingSet64 / 1024 / 1024); + curProc.WorkingSet64 / 1024 / 1024, + availWorkerThreads, + availCompletionPortThreads, + curProc.Threads.Count); } catch { From ac42f040a873911e1588cf9710c8daf87779ed1a Mon Sep 17 00:00:00 2001 From: Luis Cantero Date: Mon, 29 Jul 2024 10:14:06 +0200 Subject: [PATCH 4/6] Log min threads --- src/OpcPlcServer.cs | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/src/OpcPlcServer.cs b/src/OpcPlcServer.cs index 7ac3d014..e78a5ad2 100644 --- a/src/OpcPlcServer.cs +++ b/src/OpcPlcServer.cs @@ -22,6 +22,9 @@ namespace OpcPlc; public class OpcPlcServer { + private const int DefaultMinThreads = 20; + private const int DefaultCompletionPortThreads = 20; + private string[] _args; private CancellationTokenSource _cancellationTokenSource; private ImmutableList _pluginNodes; @@ -87,9 +90,16 @@ public async Task StartAsync(string[] args, CancellationToken cancellationToken LogLogo(); - Logger.LogInformation("Current directory: {currentDirectory}", Directory.GetCurrentDirectory()); - Logger.LogInformation("Log file: {logFileName}", Path.GetFullPath(Config.LogFileName)); - Logger.LogInformation("Log level: {logLevel}", Config.LogLevelCli); + ThreadPool.SetMinThreads(DefaultMinThreads, DefaultCompletionPortThreads); + ThreadPool.GetMinThreads(out int minWorkerThreads, out int minCompletionPortThreads); + Logger.LogInformation( + "Min worker threads: {MinWorkerThreads}, min completion port threads: {MinCompletionPortThreads}", + minWorkerThreads, + minCompletionPortThreads); + + Logger.LogInformation("Current directory: {CurrentDirectory}", Directory.GetCurrentDirectory()); + Logger.LogInformation("Log file: {LogFileName}", Path.GetFullPath(Config.LogFileName)); + Logger.LogInformation("Log level: {LogLevel}", Config.LogLevelCli); // Show OPC PLC version. var fileVersion = FileVersionInfo.GetVersionInfo(Assembly.GetExecutingAssembly().Location); From 178ddf62fc443da6b77aa2b750cb5956c7d9c056 Mon Sep 17 00:00:00 2001 From: Luis Cantero Date: Mon, 29 Jul 2024 10:14:22 +0200 Subject: [PATCH 5/6] Update version --- version.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/version.json b/version.json index d8b46454..6977793d 100644 --- a/version.json +++ b/version.json @@ -1,6 +1,6 @@ { "$schema": "https://raw.githubusercontent.com/AArnott/Nerdbank.GitVersioning/master/src/NerdBank.GitVersioning/version.schema.json", - "version": "2.12.19", + "version": "2.12.20", "versionHeightOffset": -1, "publicReleaseRefSpec": [ "^refs/heads/main$", From 8c1d9c52c24917c8e205c03e8a8abf232e3fd2d4 Mon Sep 17 00:00:00 2001 From: Luis Cantero Date: Mon, 29 Jul 2024 10:35:39 +0200 Subject: [PATCH 6/6] Improve err handling --- src/PlcServer.cs | 74 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 74 insertions(+) diff --git a/src/PlcServer.cs b/src/PlcServer.cs index 89b12bcb..e8d72b00 100644 --- a/src/PlcServer.cs +++ b/src/PlcServer.cs @@ -164,6 +164,9 @@ public override ResponseHeader CreateMonitoredItems( out MonitoredItemCreateResultCollection results, out DiagnosticInfoCollection diagnosticInfos) { + results = default; + diagnosticInfos = default; + try { OperationContext context = ValidateRequest(requestHeader, RequestType.CreateMonitoredItems); @@ -180,9 +183,43 @@ public override ResponseHeader CreateMonitoredItems( return responseHeader; } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadNoSubscription) + { + MetricsHelper.RecordTotalErrors(nameof(CreateMonitoredItems)); + + _logger.LogDebug( + ex, + "Failed creating monitored items: {StatusCode}", + StatusCodes.BadNoSubscription.ToString()); + + return new ResponseHeader { ServiceResult = StatusCodes.BadNoSubscription }; + } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadSessionIdInvalid) + { + MetricsHelper.RecordTotalErrors(nameof(CreateMonitoredItems)); + + _logger.LogDebug( + ex, + "Failed creating monitored items: {StatusCode}", + StatusCodes.BadSessionIdInvalid.ToString()); + + return new ResponseHeader { ServiceResult = StatusCodes.BadSessionIdInvalid }; + } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadSecureChannelIdInvalid) + { + MetricsHelper.RecordTotalErrors(nameof(CreateMonitoredItems)); + + _logger.LogDebug( + ex, + "Failed creating monitored items: {StatusCode}", + StatusCodes.BadSecureChannelIdInvalid.ToString()); + + return new ResponseHeader { ServiceResult = StatusCodes.BadSecureChannelIdInvalid }; + } catch (Exception ex) { MetricsHelper.RecordTotalErrors(nameof(CreateMonitoredItems)); + _logger.LogError(ex, "Error creating monitored items"); throw; } @@ -289,6 +326,9 @@ public override ResponseHeader Read( out DataValueCollection results, out DiagnosticInfoCollection diagnosticInfos) { + results = default; + diagnosticInfos = default; + try { var responseHeader = base.Read(requestHeader, maxAge, timestampsToReturn, nodesToRead, out results, out diagnosticInfos); @@ -297,9 +337,43 @@ public override ResponseHeader Read( return responseHeader; } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadNoSubscription) + { + MetricsHelper.RecordTotalErrors(nameof(Read)); + + _logger.LogDebug( + ex, + "Failed to read: {StatusCode}", + StatusCodes.BadNoSubscription.ToString()); + + return new ResponseHeader { ServiceResult = StatusCodes.BadNoSubscription }; + } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadSessionIdInvalid) + { + MetricsHelper.RecordTotalErrors(nameof(Read)); + + _logger.LogDebug( + ex, + "Failed to read: {StatusCode}", + StatusCodes.BadSessionIdInvalid.ToString()); + + return new ResponseHeader { ServiceResult = StatusCodes.BadSessionIdInvalid }; + } + catch (ServiceResultException ex) when (ex.StatusCode == StatusCodes.BadSecureChannelIdInvalid) + { + MetricsHelper.RecordTotalErrors(nameof(Read)); + + _logger.LogDebug( + ex, + "Failed to read: {StatusCode}", + StatusCodes.BadSecureChannelIdInvalid.ToString()); + + return new ResponseHeader { ServiceResult = StatusCodes.BadSecureChannelIdInvalid }; + } catch (Exception ex) { MetricsHelper.RecordTotalErrors(nameof(Read)); + _logger.LogError(ex, "Error reading"); throw; }