Skip to content

Commit

Permalink
[BREAKING] Improve logging
Browse files Browse the repository at this point in the history
  • Loading branch information
Apollo3zehn committed Jul 1, 2024
1 parent 3b3de6e commit 11a11dc
Show file tree
Hide file tree
Showing 9 changed files with 62 additions and 72 deletions.
2 changes: 1 addition & 1 deletion .vscode/launch.json
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
"type": "coreclr",
"request": "launch",
"preLaunchTask": "build",
"program": "${workspaceFolder}/artifacts/bin/SampleServerClientTcp/SampleServerClientTcp.dll",
"program": "${workspaceFolder}/artifacts/bin/SampleServerClientTcp/debug/SampleServerClientTcp.dll",
"args": [],
"cwd": "${workspaceFolder}/sample/SampleServerClientTcp",
"console": "externalTerminal",
Expand Down
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
## vNext - xxxx-xx-xx

### Breaking changes
- The constructors of `ModbusTcpClient` and `ModbusRtuClient` have been unified so that both accept an `ILogger`.

## v5.2.0 - 2024-04-23

### Features
Expand Down
34 changes: 3 additions & 31 deletions sample/SampleServerClientTcp/Program.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
using Microsoft.Extensions.Logging;
using System.Net;
using Microsoft.Extensions.Logging;

namespace FluentModbus.SampleMaster
{
Expand All @@ -14,7 +15,6 @@ static async Task Main(string[] args)
});

var serverLogger = loggerFactory.CreateLogger("Server");
var clientLogger = loggerFactory.CreateLogger("Client");

/* create Modbus TCP server */
using var server = new ModbusTcpServer(serverLogger)
Expand All @@ -29,12 +29,9 @@ static async Task Main(string[] args)
// the variable 'registerAddresses' contains a list of modified register addresses
};

/* create Modbus TCP client */
var client = new ModbusTcpClient();

/* run Modbus TCP server */
var cts = new CancellationTokenSource();
server.Start();
server.Start(new IPEndPoint(IPAddress.Parse("0.0.0.0"), 1999));
serverLogger.LogInformation("Server started.");

var task_server = Task.Run(async () =>
Expand All @@ -52,31 +49,6 @@ static async Task Main(string[] args)
}
}, cts.Token);

/* run Modbus TCP client */
var task_client = Task.Run(() =>
{
client.Connect();

try
{
DoClientWork(client, clientLogger);
}
catch (Exception ex)
{
clientLogger.LogError(ex.Message);
}

client.Disconnect();

Console.WriteLine("Tests finished. Press any key to continue.");
Console.ReadKey(intercept: true);
});

// wait for client task to finish
await task_client;

// stop server
cts.Cancel();
await task_server;

server.Stop();
Expand Down
11 changes: 9 additions & 2 deletions src/FluentModbus/Server/ModbusRtuRequestHandler.cs
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
using Microsoft.Extensions.Logging;

namespace FluentModbus
{
internal class ModbusRtuRequestHandler : ModbusRequestHandler, IDisposable
Expand All @@ -6,12 +8,15 @@ internal class ModbusRtuRequestHandler : ModbusRequestHandler, IDisposable

private IModbusRtuSerialPort _serialPort;

private readonly ILogger _logger;

#endregion

#region Constructors

public ModbusRtuRequestHandler(IModbusRtuSerialPort serialPort, ModbusRtuServer rtuServer) : base(rtuServer, 256)
public ModbusRtuRequestHandler(IModbusRtuSerialPort serialPort, ModbusRtuServer rtuServer, ILogger logger) : base(rtuServer, 256)
{
_logger = logger;
_serialPort = serialPort;
_serialPort.Open();

Expand Down Expand Up @@ -47,8 +52,10 @@ internal override async Task ReceiveRequestAsync()
WriteResponse();
}
}
catch
catch (Exception ex)
{
_logger.LogDebug(ex, "The connection will be closed");

CancelToken();
}
}
Expand Down
18 changes: 13 additions & 5 deletions src/FluentModbus/Server/ModbusRtuServer.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
using System.IO.Ports;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Logging.Abstractions;

namespace FluentModbus
{
Expand All @@ -18,37 +20,43 @@ public class ModbusRtuServer : ModbusServer
/// <summary>
/// Creates a Modbus RTU server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
/// <param name="isAsynchronous">Enables or disables the asynchronous operation, where each client request is processed immediately using a locking mechanism. Use synchronuous operation to avoid locks in the hosting application. See the <see href="https://github.com/Apollo3zehn/FluentModbus">documentation</see> for more details.</param>
/// <param name="unitIdentifier">The unique Modbus RTU unit identifier (1..247).</param>
public ModbusRtuServer(byte unitIdentifier) : this(unitIdentifier, true)
public ModbusRtuServer(byte unitIdentifier, bool isAsynchronous = true)
: this(logger: NullLogger.Instance, unitIdentifier, isAsynchronous)
{
//
}

/// <summary>
/// Creates a Modbus RTU server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
/// <param name="logger">The logger to use.</param>
/// <param name="isAsynchronous">Enables or disables the asynchronous operation, where each client request is processed immediately using a locking mechanism. Use synchronuous operation to avoid locks in the hosting application. See the <see href="https://github.com/Apollo3zehn/FluentModbus">documentation</see> for more details.</param>
/// <param name="unitIdentifier">The unique Modbus RTU unit identifier (1..247).</param>
public ModbusRtuServer(byte unitIdentifier, bool isAsynchronous) : base(isAsynchronous)
public ModbusRtuServer(ILogger logger, byte unitIdentifier, bool isAsynchronous = true) : base(isAsynchronous, logger)
{
AddUnit(unitIdentifier);
}

/// <summary>
/// Creates a multi-unit Modbus RTU server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
/// <param name="isAsynchronous">Enables or disables the asynchronous operation, where each client request is processed immediately using a locking mechanism. Use synchronuous operation to avoid locks in the hosting application. See the <see href="https://github.com/Apollo3zehn/FluentModbus">documentation</see> for more details.</param>
/// <param name="unitIdentifiers">The unique Modbus RTU unit identifiers (1..247).</param>
public ModbusRtuServer(IEnumerable<byte> unitIdentifiers) : this(unitIdentifiers, true)
public ModbusRtuServer(IEnumerable<byte> unitIdentifiers, bool isAsynchronous = true)
: this(logger: NullLogger.Instance, unitIdentifiers, isAsynchronous)
{
//
}

/// <summary>
/// Creates a multi-unit Modbus RTU server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
/// <param name="logger">The logger to use.</param>
/// <param name="isAsynchronous">Enables or disables the asynchronous operation, where each client request is processed immediately using a locking mechanism. Use synchronuous operation to avoid locks in the hosting application. See the <see href="https://github.com/Apollo3zehn/FluentModbus">documentation</see> for more details.</param>
/// <param name="unitIdentifiers">The unique Modbus RTU unit identifiers (1..247).</param>
public ModbusRtuServer(IEnumerable<byte> unitIdentifiers, bool isAsynchronous) : base(isAsynchronous)
public ModbusRtuServer(ILogger logger, IEnumerable<byte> unitIdentifiers, bool isAsynchronous = true) : base(isAsynchronous, logger)
{
foreach (var unitIdentifier in unitIdentifiers)
{
Expand Down Expand Up @@ -147,7 +155,7 @@ public void Start(IModbusRtuSerialPort serialPort)
base.StopProcessing();
base.StartProcessing();

RequestHandler = new ModbusRtuRequestHandler(serialPort, this);
RequestHandler = new ModbusRtuRequestHandler(serialPort, this, Logger);

// remove clients asynchronously
/* https://stackoverflow.com/questions/2782802/can-net-task-instances-go-out-of-scope-during-run */
Expand Down
10 changes: 9 additions & 1 deletion src/FluentModbus/Server/ModbusServer.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
using Microsoft.Extensions.Logging;

[assembly: InternalsVisibleTo("FluentModbus.Tests")]

Expand Down Expand Up @@ -79,10 +80,12 @@ public abstract class ModbusServer : IDisposable
/// Initializes a new instance of the <see cref="ModbusServer"/>.
/// </summary>
/// <param name="isAsynchronous">A boolean which indicates if the server responds to client requests asynchronously (immediately) or synchronously (regularly at fixed events).</param>
protected ModbusServer(bool isAsynchronous)
/// <param name="logger">The logger to use.</param>
protected ModbusServer(bool isAsynchronous, ILogger logger)
{
Lock = this;
IsAsynchronous = isAsynchronous;
Logger = logger;

MaxInputRegisterAddress = ushort.MaxValue;
MaxHoldingRegisterAddress = ushort.MaxValue;
Expand Down Expand Up @@ -153,6 +156,11 @@ protected ModbusServer(bool isAsynchronous)
/// </summary>
public bool AlwaysRaiseChangedEvent { get; set; } = false;

/// <summary>
/// Gets the logger.
/// </summary>
protected ILogger Logger { get; }

internal bool IsSingleZeroUnitMode => UnitIdentifiers.Count == 1 && UnitIdentifiers[0] == 0;

private protected CancellationTokenSource CTS { get; private set; } = new CancellationTokenSource();
Expand Down
10 changes: 8 additions & 2 deletions src/FluentModbus/Server/ModbusTcpRequestHandler.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
using System.Net;
using System.Net.Sockets;
using Microsoft.Extensions.Logging;

namespace FluentModbus
{
Expand All @@ -14,13 +15,16 @@ internal class ModbusTcpRequestHandler : ModbusRequestHandler, IDisposable
private ushort _protocolIdentifier;
private ushort _bytesFollowing;

private readonly ILogger _logger;

#endregion

#region Constructors

public ModbusTcpRequestHandler(TcpClient tcpClient, ModbusTcpServer tcpServer)
public ModbusTcpRequestHandler(TcpClient tcpClient, ModbusTcpServer tcpServer, ILogger logger)
: base(tcpServer, 260)
{
_logger = logger;
_tcpClient = tcpClient;
_networkStream = tcpClient.GetStream();

Expand Down Expand Up @@ -59,8 +63,10 @@ internal override async Task ReceiveRequestAsync()
WriteResponse();
}
}
catch
catch (Exception ex)
{
_logger.LogDebug(ex, "The connection will be closed");

CancelToken();
}
}
Expand Down
42 changes: 13 additions & 29 deletions src/FluentModbus/Server/ModbusTcpServer.cs
Original file line number Diff line number Diff line change
Expand Up @@ -19,40 +19,22 @@ public class ModbusTcpServer : ModbusServer

#region Constructors

/// <summary>
/// Creates a Modbus TCP server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
public ModbusTcpServer() : this(NullLogger.Instance, true)
{
//
}

/// <summary>
/// Creates a Modbus TCP server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
/// <param name="logger">A logger instance to provide runtime information.</param>
public ModbusTcpServer(ILogger logger) : this(logger, true)
{
//
}

/// <summary>
/// Creates a Modbus TCP server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
/// <param name="isAsynchronous">Enables or disables the asynchronous operation, where each client request is processed immediately using a locking mechanism. Use synchronuous operation to avoid locks in the hosting application. See the <see href="https://github.com/Apollo3zehn/FluentModbus">documentation</see> for more details.</param>
public ModbusTcpServer(bool isAsynchronous) : this(NullLogger.Instance, isAsynchronous)
public ModbusTcpServer(bool isAsynchronous = true) : base(isAsynchronous, logger: NullLogger.Instance)
{
//
AddUnit(0);
}

/// <summary>
/// Creates a Modbus TCP server with support for holding registers (read and write, 16 bit), input registers (read-only, 16 bit), coils (read and write, 1 bit) and discete inputs (read-only, 1 bit).
/// </summary>
/// <param name="logger">A logger instance to provide runtime information.</param>
/// <param name="logger">The logger to use.</param>
/// <param name="isAsynchronous">Enables or disables the asynchronous operation, where each client request is processed immediately using a locking mechanism. Use synchronuous operation to avoid locks in the hosting application. See the <see href="https://github.com/Apollo3zehn/FluentModbus">documentation</see> for more details.</param>
public ModbusTcpServer(ILogger logger, bool isAsynchronous) : base(isAsynchronous)
public ModbusTcpServer(ILogger logger, bool isAsynchronous = true) : base(isAsynchronous, logger)
{
Logger = logger;
AddUnit(0);
}

Expand All @@ -63,7 +45,7 @@ public ModbusTcpServer(ILogger logger, bool isAsynchronous) : base(isAsynchronou
/// <summary>
/// Gets or sets the timeout for each client connection. When the client does not send any request within the specified period of time, the connection will be reset. Default is 1 minute.
/// </summary>
public TimeSpan ConnectionTimeout { get; set; } = ModbusTcpServer.DefaultConnectionTimeout;
public TimeSpan ConnectionTimeout { get; set; } = DefaultConnectionTimeout;

/// <summary>
/// Gets or sets the maximum number of concurrent client connections. A value of zero means there is no limit.
Expand All @@ -79,8 +61,6 @@ public ModbusTcpServer(ILogger logger, bool isAsynchronous) : base(isAsynchronou

internal List<ModbusTcpRequestHandler> RequestHandlers { get; private set; } = new List<ModbusTcpRequestHandler>();

private ILogger Logger { get; }

#endregion

#region Methods
Expand Down Expand Up @@ -133,7 +113,7 @@ public void Start(ITcpClientProvider tcpClientProvider, bool leaveOpen = false)
// There are no default timeouts (SendTimeout and ReceiveTimeout = 0),
// use ConnectionTimeout instead.
var tcpClient = await _tcpClientProvider.AcceptTcpClientAsync();
var requestHandler = new ModbusTcpRequestHandler(tcpClient, this);
var requestHandler = new ModbusTcpRequestHandler(tcpClient, this, Logger);

lock (Lock)
{
Expand Down Expand Up @@ -170,12 +150,16 @@ public void Start(ITcpClientProvider tcpClientProvider, bool leaveOpen = false)
// This should be the only cause but since "ReceiveRequestAsync" is never
// awaited, the actual cause may be different.
requestHandler.CancellationToken.IsCancellationRequested ||
// or there was not request received within the specified timeout
// or there was no request received within the specified timeout
requestHandler.LastRequest.Elapsed > ConnectionTimeout)
{
try
{
Logger.LogInformation($"Connection {requestHandler.DisplayName} timed out.");
if (requestHandler.CancellationToken.IsCancellationRequested)
Logger.LogInformation($"Connection {requestHandler.DisplayName} was closed due to an error");

else
Logger.LogInformation($"Connection {requestHandler.DisplayName} timed out");

// remove request handler
RequestHandlers.Remove(requestHandler);
Expand Down Expand Up @@ -207,7 +191,7 @@ public void Start(TcpClient tcpClient)

RequestHandlers = new List<ModbusTcpRequestHandler>()
{
new ModbusTcpRequestHandler(tcpClient, this)
new ModbusTcpRequestHandler(tcpClient, this, Logger)
};
}

Expand Down
2 changes: 1 addition & 1 deletion version.json
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
{
"version": "5.2.0",
"version": "vNext",
"suffix": ""
}

0 comments on commit 11a11dc

Please sign in to comment.