diff --git a/.github/workflows/sonarcloud.yml b/.github/workflows/sonarcloud.yml
index e784069..aca8b48 100644
--- a/.github/workflows/sonarcloud.yml
+++ b/.github/workflows/sonarcloud.yml
@@ -70,13 +70,13 @@ jobs:
run: dotnet restore NetSdrClient.sln
- name: Build
run: dotnet build NetSdrClient.sln -c Release --no-restore
- #- name: Tests with coverage (OpenCover)
- # run: |
- # dotnet test NetSdrClientAppTests/NetSdrClientAppTests.csproj -c Release --no-build `
- # /p:CollectCoverage=true `
- # /p:CoverletOutput=TestResults/coverage.xml `
- # /p:CoverletOutputFormat=opencover
- # shell: pwsh
+ - name: Tests with coverage (OpenCover)
+ run: |
+ dotnet test NetSdrClientAppTests/NetSdrClientAppTests.csproj -c Release --no-build `
+ /p:CollectCoverage=true `
+ /p:CoverletOutput=TestResults/coverage.xml `
+ /p:CoverletOutputFormat=opencover
+ shell: pwsh
# 3) END: SonarScanner
- name: SonarScanner End
run: dotnet sonarscanner end /d:sonar.token="${{ secrets.SONAR_TOKEN }}"
diff --git a/EchoTspServer/EchoServer.csproj b/EchoTcpServer/EchoServer.csproj
similarity index 100%
rename from EchoTspServer/EchoServer.csproj
rename to EchoTcpServer/EchoServer.csproj
diff --git a/EchoTcpServer/Program.cs b/EchoTcpServer/Program.cs
new file mode 100644
index 0000000..82721a2
--- /dev/null
+++ b/EchoTcpServer/Program.cs
@@ -0,0 +1,172 @@
+using System;
+using System.Net;
+using System.Net.Sockets;
+using System.Text;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace EchoServer
+{
+ public class EchoServer
+ {
+ private readonly int _port;
+ private TcpListener _listener;
+ private CancellationTokenSource _cancellationTokenSource;
+
+ //constuctor
+ public EchoServer(int port)
+ {
+ _port = port;
+ _cancellationTokenSource = new CancellationTokenSource();
+ }
+
+ public async Task StartAsync()
+ {
+ _listener = new TcpListener(IPAddress.Any, _port);
+ _listener.Start();
+ Console.WriteLine($"Server started on port {_port}.");
+
+ while (!_cancellationTokenSource.Token.IsCancellationRequested)
+ {
+ try
+ {
+ TcpClient client = await _listener.AcceptTcpClientAsync();
+ Console.WriteLine("Client connected.");
+
+ _ = Task.Run(() => HandleClientAsync(client, _cancellationTokenSource.Token));
+ }
+ catch (ObjectDisposedException)
+ {
+ // Listener has been closed
+ break;
+ }
+ }
+
+ Console.WriteLine("Server shutdown.");
+ }
+
+ private async Task HandleClientAsync(TcpClient client, CancellationToken token)
+ {
+ using (NetworkStream stream = client.GetStream())
+ {
+ try
+ {
+ byte[] buffer = new byte[8192];
+ int bytesRead;
+
+ while (!token.IsCancellationRequested && (bytesRead = await stream.ReadAsync(buffer, 0, buffer.Length, token)) > 0)
+ {
+ // Echo back the received message
+ await stream.WriteAsync(buffer, 0, bytesRead, token);
+ Console.WriteLine($"Echoed {bytesRead} bytes to the client.");
+ }
+ }
+ catch (Exception ex) when (!(ex is OperationCanceledException))
+ {
+ Console.WriteLine($"Error: {ex.Message}");
+ }
+ finally
+ {
+ client.Close();
+ Console.WriteLine("Client disconnected.");
+ }
+ }
+ }
+
+ public void Stop()
+ {
+ _cancellationTokenSource.Cancel();
+ _listener.Stop();
+ _cancellationTokenSource.Dispose();
+ Console.WriteLine("Server stopped.");
+ }
+
+ public static async Task Main(string[] args)
+ {
+ EchoServer server = new EchoServer(5000);
+
+ // Start the server in a separate task
+ _ = Task.Run(() => server.StartAsync());
+
+ string host = "127.0.0.1"; // Target IP
+ int port = 60000; // Target Port
+ int intervalMilliseconds = 5000; // Send every 3 seconds
+
+ using (var sender = new UdpTimedSender(host, port))
+ {
+ Console.WriteLine("Press any key to stop sending...");
+ sender.StartSending(intervalMilliseconds);
+
+ Console.WriteLine("Press 'q' to quit...");
+ while (Console.ReadKey(intercept: true).Key != ConsoleKey.Q)
+ {
+ // Just wait until 'q' is pressed
+ }
+
+ sender.StopSending();
+ server.Stop();
+ Console.WriteLine("Sender stopped.");
+ }
+ }
+ }
+
+
+ public class UdpTimedSender : IDisposable
+ {
+ private readonly string _host;
+ private readonly int _port;
+ private readonly UdpClient _udpClient;
+ private Timer _timer;
+
+ public UdpTimedSender(string host, int port)
+ {
+ _host = host;
+ _port = port;
+ _udpClient = new UdpClient();
+ }
+
+ public void StartSending(int intervalMilliseconds)
+ {
+ if (_timer != null)
+ throw new InvalidOperationException("Sender is already running.");
+
+ _timer = new Timer(SendMessageCallback, null, 0, intervalMilliseconds);
+ }
+
+ ushort i = 0;
+
+ private void SendMessageCallback(object state)
+ {
+ try
+ {
+ //dummy data
+ Random rnd = new Random();
+ byte[] samples = new byte[1024];
+ rnd.NextBytes(samples);
+ i++;
+
+ byte[] msg = (new byte[] { 0x04, 0x84 }).Concat(BitConverter.GetBytes(i)).Concat(samples).ToArray();
+ var endpoint = new IPEndPoint(IPAddress.Parse(_host), _port);
+
+ _udpClient.Send(msg, msg.Length, endpoint);
+ Console.WriteLine($"Message sent to {_host}:{_port} ");
+ }
+ catch (Exception ex)
+ {
+ Console.WriteLine($"Error sending message: {ex.Message}");
+ }
+ }
+
+ public void StopSending()
+ {
+ _timer?.Dispose();
+ _timer = null;
+ }
+
+ public void Dispose()
+ {
+ StopSending();
+ _udpClient.Dispose();
+ }
+ }
+}
\ No newline at end of file
diff --git a/EchoTspServer/Program.cs b/EchoTspServer/Program.cs
deleted file mode 100644
index 5966c57..0000000
--- a/EchoTspServer/Program.cs
+++ /dev/null
@@ -1,173 +0,0 @@
-using System;
-using System.Net;
-using System.Net.Sockets;
-using System.Text;
-using System.Threading;
-using System.Threading.Tasks;
-
-///
-/// This program was designed for test purposes only
-/// Not for a review
-///
-public class EchoServer
-{
- private readonly int _port;
- private TcpListener _listener;
- private CancellationTokenSource _cancellationTokenSource;
-
-
- public EchoServer(int port)
- {
- _port = port;
- _cancellationTokenSource = new CancellationTokenSource();
- }
-
- public async Task StartAsync()
- {
- _listener = new TcpListener(IPAddress.Any, _port);
- _listener.Start();
- Console.WriteLine($"Server started on port {_port}.");
-
- while (!_cancellationTokenSource.Token.IsCancellationRequested)
- {
- try
- {
- TcpClient client = await _listener.AcceptTcpClientAsync();
- Console.WriteLine("Client connected.");
-
- _ = Task.Run(() => HandleClientAsync(client, _cancellationTokenSource.Token));
- }
- catch (ObjectDisposedException)
- {
- // Listener has been closed
- break;
- }
- }
-
- Console.WriteLine("Server shutdown.");
- }
-
- private async Task HandleClientAsync(TcpClient client, CancellationToken token)
- {
- using (NetworkStream stream = client.GetStream())
- {
- try
- {
- byte[] buffer = new byte[8192];
- int bytesRead;
-
- while (!token.IsCancellationRequested && (bytesRead = await stream.ReadAsync(buffer, 0, buffer.Length, token)) > 0)
- {
- // Echo back the received message
- await stream.WriteAsync(buffer, 0, bytesRead, token);
- Console.WriteLine($"Echoed {bytesRead} bytes to the client.");
- }
- }
- catch (Exception ex) when (!(ex is OperationCanceledException))
- {
- Console.WriteLine($"Error: {ex.Message}");
- }
- finally
- {
- client.Close();
- Console.WriteLine("Client disconnected.");
- }
- }
- }
-
- public void Stop()
- {
- _cancellationTokenSource.Cancel();
- _listener.Stop();
- _cancellationTokenSource.Dispose();
- Console.WriteLine("Server stopped.");
- }
-
- public static async Task Main(string[] args)
- {
- EchoServer server = new EchoServer(5000);
-
- // Start the server in a separate task
- _ = Task.Run(() => server.StartAsync());
-
- string host = "127.0.0.1"; // Target IP
- int port = 60000; // Target Port
- int intervalMilliseconds = 5000; // Send every 3 seconds
-
- using (var sender = new UdpTimedSender(host, port))
- {
- Console.WriteLine("Press any key to stop sending...");
- sender.StartSending(intervalMilliseconds);
-
- Console.WriteLine("Press 'q' to quit...");
- while (Console.ReadKey(intercept: true).Key != ConsoleKey.Q)
- {
- // Just wait until 'q' is pressed
- }
-
- sender.StopSending();
- server.Stop();
- Console.WriteLine("Sender stopped.");
- }
- }
-}
-
-
-public class UdpTimedSender : IDisposable
-{
- private readonly string _host;
- private readonly int _port;
- private readonly UdpClient _udpClient;
- private Timer _timer;
-
- public UdpTimedSender(string host, int port)
- {
- _host = host;
- _port = port;
- _udpClient = new UdpClient();
- }
-
- public void StartSending(int intervalMilliseconds)
- {
- if (_timer != null)
- throw new InvalidOperationException("Sender is already running.");
-
- _timer = new Timer(SendMessageCallback, null, 0, intervalMilliseconds);
- }
-
- ushort i = 0;
-
- private void SendMessageCallback(object state)
- {
- try
- {
- //dummy data
- Random rnd = new Random();
- byte[] samples = new byte[1024];
- rnd.NextBytes(samples);
- i++;
-
- byte[] msg = (new byte[] { 0x04, 0x84 }).Concat(BitConverter.GetBytes(i)).Concat(samples).ToArray();
- var endpoint = new IPEndPoint(IPAddress.Parse(_host), _port);
-
- _udpClient.Send(msg, msg.Length, endpoint);
- Console.WriteLine($"Message sent to {_host}:{_port} ");
- }
- catch (Exception ex)
- {
- Console.WriteLine($"Error sending message: {ex.Message}");
- }
- }
-
- public void StopSending()
- {
- _timer?.Dispose();
- _timer = null;
- }
-
- public void Dispose()
- {
- StopSending();
- _udpClient.Dispose();
- }
-}
\ No newline at end of file
diff --git a/NetSdrClient.sln b/NetSdrClient.sln
index d8ca20f..42431fb 100644
--- a/NetSdrClient.sln
+++ b/NetSdrClient.sln
@@ -7,7 +7,7 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "NetSdrClientApp", "NetSdrCl
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "NetSdrClientAppTests", "NetSdrClientAppTests\NetSdrClientAppTests.csproj", "{D0155366-89B4-4BA4-90E2-2ECC8C1EB915}"
EndProject
-Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EchoServer", "EchoTspServer\EchoServer.csproj", "{9179F2F7-EBEE-4A5D-9FD9-F6E3C18DD263}"
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EchoServer", "EchoTcpServer\EchoServer.csproj", "{9179F2F7-EBEE-4A5D-9FD9-F6E3C18DD263}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
diff --git a/NetSdrClientApp/NetSdrClientApp.csproj b/NetSdrClientApp/NetSdrClientApp.csproj
index 2ac9100..ae220b6 100644
--- a/NetSdrClientApp/NetSdrClientApp.csproj
+++ b/NetSdrClientApp/NetSdrClientApp.csproj
@@ -8,7 +8,7 @@
-
+
diff --git a/NetSdrClientAppTests/ArchitectureTests.cs b/NetSdrClientAppTests/ArchitectureTests.cs
new file mode 100644
index 0000000..b984664
--- /dev/null
+++ b/NetSdrClientAppTests/ArchitectureTests.cs
@@ -0,0 +1,54 @@
+using NetArchTest.Rules;
+using NUnit.Framework;
+using System.Linq;
+using System.Reflection;
+using System.Text;
+
+namespace NetSdrClientAppTests
+{
+ public class ArchitectureTests
+ {
+ [Test]
+ public void App_Should_Not_Depend_On_EchoServer()
+ {
+ var result = Types.InAssembly(typeof(NetSdrClientApp.NetSdrClient).Assembly)
+ .That()
+ .ResideInNamespace("NetSdrClientApp")
+ .ShouldNot()
+ .HaveDependencyOn("EchoServer")
+ .GetResult();
+
+ Assert.That(result.IsSuccessful, Is.True);
+ }
+
+ [Test]
+ public void Messages_Should_Not_Depend_On_Networking()
+ {
+ // Arrange
+ var result = Types.InAssembly(typeof(NetSdrClientApp.Messages.NetSdrMessageHelper).Assembly)
+ .That()
+ .ResideInNamespace("NetSdrClientApp.Messages")
+ .ShouldNot()
+ .HaveDependencyOn("NetSdrClientApp.Networking")
+ .GetResult();
+
+ // Assert
+ Assert.That(result.IsSuccessful, Is.True);
+ }
+
+ [Test]
+ public void Networking_Should_Not_Depend_On_Messages()
+ {
+ // Arrange
+ var result = Types.InAssembly(typeof(NetSdrClientApp.Networking.ITcpClient).Assembly)
+ .That()
+ .ResideInNamespace("NetSdrClientApp.Networking")
+ .ShouldNot()
+ .HaveDependencyOn("NetSdrClientApp.Messages")
+ .GetResult();
+
+ // Assert
+ Assert.That(result.IsSuccessful, Is.True);
+ }
+ }
+}
\ No newline at end of file
diff --git a/NetSdrClientAppTests/NetSdrClientAppTests.csproj b/NetSdrClientAppTests/NetSdrClientAppTests.csproj
index 3cbc46a..9213cef 100644
--- a/NetSdrClientAppTests/NetSdrClientAppTests.csproj
+++ b/NetSdrClientAppTests/NetSdrClientAppTests.csproj
@@ -11,7 +11,12 @@
-
+
+ runtime; build; native; contentfiles; analyzers; buildtransitive
+ all
+
+
+