2
0
mirror of https://github.com/esiur/esiur-dotnet.git synced 2026-04-29 06:48:41 +00:00
This commit is contained in:
2026-04-13 15:46:44 +03:00
parent fefe76c726
commit b45c205d12
8 changed files with 751 additions and 0 deletions
@@ -0,0 +1,14 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\..\..\..\Libraries\Esiur\Esiur.csproj" OutputItemType="Analyzer"/>
</ItemGroup>
</Project>
@@ -0,0 +1,40 @@
// ============================================================
// Test 4: Fork-Join Queueing Test — SERVER NODE//
// Usage: dotnet run -- --port 10901
// ============================================================
using Esiur.Resource;
using Esiur.Stores;
using Esiur.Protocol;
using Esiur.Tests.Queueing.Server;
var port = int.Parse(GetArg(args, "--port", "10901"));
Console.WriteLine($"[Server] Listening on port {port}...");
var wh = Warehouse.Default;
var mem = await wh.Put("sys", new MemoryStore());
var service = await wh.Put("sys/queueing", new QueueingService());
var server = await wh.Put("sys/server", new EpServer() { Port = (ushort)port,
EntryPoint = service });
long memBefore = GC.GetTotalMemory(forceFullCollection: true);
await wh.Open();
long memAfter = GC.GetTotalMemory(forceFullCollection: true);
double memMB = (memAfter - memBefore) / (1024.0 * 1024.0);
Console.WriteLine("Press ENTER to stop.");
Console.ReadLine();
await wh.Close();
static string GetArg(string[] args, string key, string def)
{
int i = Array.IndexOf(args, key);
return (i >= 0 && i + 1 < args.Length) ? args[i + 1] : def;
}
@@ -0,0 +1,204 @@
using Esiur.Core;
using Esiur.Data;
using Esiur.Protocol;
using Esiur.Resource;
using System;
using System.Collections.Generic;
using System.Text;
namespace Esiur.Tests.Queueing.Server
{
[Resource]
public partial class QueueingService : EntryPoint
{
static Random rand = new Random(322221);
public List<TestObject> TestObjects = new List<TestObject>();
[Export]
public object testProperty;
[Export]
public async AsyncReply<TestObject> StartUpdatesLocal(int interval, int count, double localProbability)
{
var dis = GenerateRandomBoolSequence(count, localProbability, new Random(2222));
for (var i = 0; i < count; i++)
{
if (dis[i])
{
var o = await Warehouse.Default.New<TestObject>("sys/anything");
o.Value = i;
o.Name = "Update " + i;
TestObjects.Add(o);
TestProperty = o;
}
else
{
TestProperty = i;
}
await Task.Delay(interval);
}
return null;
}
[Export]
public async AsyncReply<ResourceLink<TestObject>> StartUpdatesRemote(int interval, int count, double remoteProbability, string remoteLink)
{
for (var i = 0; i < count; i++)
{
var probability = rand.NextDouble();
if (probability <= remoteProbability)
{
TestProperty = remoteLink;
}
else
{
TestProperty = i;
}
await Task.Delay(interval);
}
return null;
}
[Export]
public async AsyncReply<TestObject> StartUpdatesMirror(int interval, int count, double remoteProbability, string remoteNode, string remoteLink)
{
var remoteCon = await Warehouse.Default.Get<EpConnection>(remoteNode);
for (var i = 0; i < count; i++)
{
var probability = rand.NextDouble();
if (probability <= remoteProbability)
{
var o = await remoteCon.Get(remoteLink);
TestObjects.Add(o as TestObject);
TestProperty = o;
}
else
{
TestProperty = i;
}
await Task.Delay(interval);
}
return null;
}
[Export]
public async AsyncReply<ResourceLink<TestObject>> StartUpdates(int interval, int count, double localProbability, double remoteProbability, string remoteHostLink)
{
for (var i = 0; i < count; i++)
{
var probability = rand.NextDouble();
if ((localProbability != 0 && probability <= localProbability) || localProbability == 1)
{
var o = await Warehouse.Default.New<TestObject>("sys/anything");
o.Value = i;
o.Name = "Update " + i;
TestObjects.Add(o);
TestProperty = o;
}
else if (probability < localProbability + remoteProbability)
{
TestProperty = new ResourceLink(remoteHostLink);
}
else
{
TestProperty = i;
}
await Task.Delay(interval);
}
TestObjects.Clear();
return null;
}
public override async AsyncReply<IResource> Query(string path, EpConnection sender)
{
if (path == "gen")
{
var o = await Warehouse.Default.New<TestObject>("sys/anything");
o.Value = rand.Next();
o.Name = "Update " + o.Value;
TestObjects.Add(o);
return o;
}
else
{
if (this.Instance != null)
return await this.Instance.Warehouse.Query(path);
else
return null;
}
}
public static bool[] GenerateRandomBoolSequence(
int length,
double probabilityTrue,
Random? rng = null)
{
if (length <= 0)
throw new ArgumentOutOfRangeException(nameof(length));
if (probabilityTrue < 0.0 || probabilityTrue > 1.0)
throw new ArgumentOutOfRangeException(nameof(probabilityTrue));
rng ??= Random.Shared;
int trueTarget = (int)Math.Round(length * probabilityTrue);
int remaining = length;
int remainingTrue = trueTarget;
var result = new bool[length];
for (int i = 0; i < length; i++)
{
// Probability adjusted to guarantee exact total
double p = (double)remainingTrue / remaining;
bool value = rng.NextDouble() < p;
result[i] = value;
if (value)
remainingTrue--;
remaining--;
}
return result;
}
protected override bool Create()
{
return true;
}
}
}
@@ -0,0 +1,15 @@
using Esiur.Resource;
using System;
using System.Collections.Generic;
using System.Text;
namespace Esiur.Tests.Queueing.Server
{
[Resource]
public partial class TestObject
{
[Export] int size;
[Export] string name;
[Export] object value;
}
}