CP.TwinCATRx
2.1.2
dotnet add package CP.TwinCATRx --version 2.1.2
NuGet\Install-Package CP.TwinCATRx -Version 2.1.2
<PackageReference Include="CP.TwinCATRx" Version="2.1.2" />
<PackageVersion Include="CP.TwinCATRx" Version="2.1.2" />
<PackageReference Include="CP.TwinCATRx" />
paket add CP.TwinCATRx --version 2.1.2
#r "nuget: CP.TwinCATRx, 2.1.2"
#:package CP.TwinCATRx@2.1.2
#addin nuget:?package=CP.TwinCATRx&version=2.1.2
#tool nuget:?package=CP.TwinCATRx&version=2.1.2
TwinCATRx
A reactive, cross-platform wrapper for Beckhoff TwinCAT ADS built on System.Reactive (Rx). It lets you observe PLC variables as IObservable<T>, write values, and work with structured tags using a HashTable-like API.
Packages
- CP.TwinCATRx: main reactive client and extension APIs.
- CP.TwinCATRx.Core: shared helpers (settings, code generation, Rx extensions, ADS observables).
Supported frameworks
- .NET Standard 2.0
- .NET 8
- .NET 9
- Windows-specific features (service monitoring) are enabled for net8.0-windows10.0.17763.0 and net9.0-windows10.0.17763.0.
Install
# Main package
dotnet add package CP.TwinCATRx
# Optional low-level helpers (usually not required directly)
dotnet add package CP.TwinCATRx.Core
🚀 Quick Start
using CP.TwinCatRx;
using CP.TwinCatRx.Core;
// Create client and settings
var client = new RxTcAdsClient();
var settings = new Settings { AdsAddress = "5.35.59.10.1.1", Port = 801, SettingsId = "Default" };
// Register tags to observe (notifications) and write
settings.AddNotification(".Tag1"); // structure
settings.AddNotification(".AString", arraySize: 80); // string length required for string
settings.AddNotification(".AInt");
settings.AddWriteVariable(".ArrInt", 11); // arrays require a length
client.Connect(settings);
// Wait for PLC to be ready
client.InitializeComplete.Subscribe(_ =>
{
// One-shot reads of simple types
client.Read(".AString");
client.Read(".AInt");
// Write a value
client.Write(".AInt", 42);
});
// Observe tag changes as streams
client.Observe<string>(".AString").Subscribe(v => Console.WriteLine($"AString: {v}"));
client.Observe<short>(".AInt").Subscribe(v => Console.WriteLine($"AInt: {v}"));
// Observe arrays (triggered by reads)
client.Observe<short[]>(".ArrInt").Subscribe(arr => Console.WriteLine($"ArrInt[{arr.Length}]: {string.Join(",", arr)}"));
📖 API Reference
IRxTcAdsClient API
- Code: IObservable<string[]> of generated code artifacts (internal diagnostics).
- InitializeComplete: IObservable<Unit> signaled when connected and initialized.
- DataReceived: IObservable<(string Variable, object? Data, string? Id)> of raw variable updates.
- ErrorReceived: IObservable<Exception> of client errors.
- OnWrite: IObservable<string?> of write results (e.g., "Success" or error text).
- ReadWriteHandleInfo: IDictionary<string, uint?> of handles for notification variables.
- WriteHandleInfo: IDictionary<string, (uint? Handle, int ArrayLength)> of handles for write variables.
- Settings: ISettings? used during Connect.
- IsPaused: bool indicating WriteValuesAsync throttling state.
- IsPausedObservable: IObservable<bool> of pause/resume state.
- Connect(ISettings settings): connect and initialize.
- Disconnect(): stop the client.
- Read(string variable, int? arrayLength = null, string? id = null): one-shot read for simple or array variables.
- Write(string variable, object value, string? id = null): write a value to a variable.
- Pause(TimeSpan time): temporarily pause WriteValuesAsync scheduling.
Examples
Observe variable updates
client.Observe<bool>(".ABool").Subscribe(v => Console.WriteLine($"ABool: {v}"));
client.Observe<int>(".ADInt").Subscribe(v => Console.WriteLine($"ADInt: {v}"));
// With correlation id
client.Observe<int>(".AInt", id: "R1").Subscribe(v => Console.WriteLine($"AInt[R1]: {v}"));
Read and write values
// Simple types
client.Read(".ALReal");
client.Write(".ABool", true);
// Arrays
// For arrays registered via AddWriteVariable(".ArrInt", 11) you can read with the configured length
client.Read(".ArrInt");
// Or supply a length for arrays not registered as write variables
client.Read(".ArrInt", arrayLength: 11);
Structured tags with HashTableRx
TwinCatRxExtensions provides helpers to work with structured PLC tags using CP.Collections.HashTableRx.
// Create a live structure wrapper for a tag (structure or UDT)
var tag1 = client.CreateStruct(".Tag1");
// Wait until the first payload is received
await tag1!.StructureReady();
// Stream inner fields
tag1.Observe<bool>("ABool").Subscribe(v => Console.WriteLine($"Tag1.ABool: {v}"));
tag1.Observe<short>("AInt").Subscribe(v => Console.WriteLine($"Tag1.AInt: {v}"));
// Clone, set values, and write back atomically
var ok = tag1.WriteValues(ht =>
{
var current = ht.Value<short>("AInt");
ht.Value("AInt", (short)(current + 10));
ht.Value("AString", $"Int Value {current + 10}");
});
// Async write with a pause window
var okAsync = await tag1.WriteValuesAsync(ht =>
{
ht.Value("AInt", 100);
ht.Value("AString", "Updated");
}, TimeSpan.FromMilliseconds(300));
TwinCatRxExtensions (main) reference
- Observe<T>(this IRxTcAdsClient, string variable)
- Observe<T>(this IRxTcAdsClient, string variable, string id)
- CreateStruct(this IRxTcAdsClient, string variable): HashTableRx
- WriteValues(this HashTableRx, Action<HashTableRx>): bool
- WriteValuesAsync(this HashTableRx, Action<HashTableRx>, TimeSpan): Task<bool>
- StructureReady(this HashTableRx): IObservable<HashTableRx>
- CreateClone(this HashTableRx): HashTableRx
Settings and configuration
var settings = new Settings
{
AdsAddress = "5.35.59.10.1.1", // or leave null/empty to use local port-only Connect
Port = 801,
SettingsId = "Default"
};
// Notifications (observed tags)
settings.AddNotification(".Tag1"); // structure
settings.AddNotification(".AString", arraySize: 80); // strings require a size
settings.AddNotification(".AInt");
// Write variables (readable and writable)
settings.AddWriteVariable(".ArrInt", 11); // arrays require a length
Core helpers (CP.TwinCATRx.Core)
Reactive retry helpers
using CP.TwinCatRx.Core;
// Retry forever
source.OnErrorRetry();
// Retry with handler
source.OnErrorRetry<SomeType, TimeoutException>(ex => Log(ex));
// Retry with delay and count
source.OnErrorRetry<SomeType, Exception>(ex => Log(ex), retryCount: 5, delay: TimeSpan.FromSeconds(1));
ADS observables
using CP.TwinCatRx.Core;
using TwinCAT.Ads;
var ads = new AdsClient();
ads.Connect(801);
// State changed events
ads.AdsStateChangedObserver().Subscribe(e => Console.WriteLine($"ADS state changed: {e.State}"));
// Polling observer for StateInfo
ads.AdsStateObserver().Subscribe(si => Console.WriteLine($"ADS: {si.AdsState}"));
Advanced (dynamic code generation)
TwinCATRx can generate types for complex structures at runtime to simplify marshaling. This is handled internally, but you can hook into emitted content via IRxTcAdsClient.Code. Dynamic emit/reflective APIs carry AOT/trimming caveats and are annotated accordingly.
Windows-only: service monitoring
Available on Windows-targeted TFMs.
using CP.TwinCatRx;
ObservableServiceController.GetServices()
.Where(s => s.DisplayName is "TwinCAT System Service" or "TwinCAT3 System Service")
.Subscribe(s =>
{
Console.WriteLine($"{s.DisplayName}: {s.Status}");
s.StatusObserver.Subscribe(st => Console.WriteLine($"Status: {st}"));
if (s.Status != ServiceControllerStatus.Running) s.Start();
});
Error handling
client.ErrorReceived.Subscribe(ex => Console.WriteLine($"Error: {ex}"));
client.OnWrite.Subscribe(msg => Console.WriteLine($"Write: {msg}"));
Performance and AOT notes
- Case-insensitive maps are used to avoid repeated string allocations.
- Reads use O(1) reverse handle lookups.
- Library is trimmable; dynamic features are annotated for AOT.
Limitations
- Arrays of string are not supported at this time (including arrays of string inside structures).
📄 License
This project is licensed under the MIT License - see the LICENSE file for details.
🤝 Contributing
Contributions are welcome! Please feel free to submit a Pull Request. For major changes, please open an issue first to discuss what you would like to change.
TwinCATRx - Empowering Industrial Automation with Reactive Technology ⚡🏭
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net8.0-windows10.0.17763 is compatible. net9.0 is compatible. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. net9.0-windows10.0.17763 is compatible. net10.0 was computed. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- CP.TwinCATRx.Core (>= 2.1.2)
- HashTableRx (>= 2.2.2)
- Nerdbank.GitVersioning (>= 3.7.115)
-
net8.0
- CP.TwinCATRx.Core (>= 2.1.2)
- HashTableRx (>= 2.2.2)
- Nerdbank.GitVersioning (>= 3.7.115)
-
net8.0-windows10.0.17763
- CP.TwinCATRx.Core (>= 2.1.2)
- HashTableRx (>= 2.2.2)
- Nerdbank.GitVersioning (>= 3.7.115)
-
net9.0
- CP.TwinCATRx.Core (>= 2.1.2)
- HashTableRx (>= 2.2.2)
- Nerdbank.GitVersioning (>= 3.7.115)
-
net9.0-windows10.0.17763
- CP.TwinCATRx.Core (>= 2.1.2)
- HashTableRx (>= 2.2.2)
- Nerdbank.GitVersioning (>= 3.7.115)
NuGet packages (1)
Showing the top 1 NuGet packages that depend on CP.TwinCATRx:
Package | Downloads |
---|---|
MQTTnet.Rx.TwinCAT
Reactive extensions for MQTTnet Broker |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last Updated |
---|---|---|
2.1.2 | 156 | 9/9/2025 |
2.0.4 | 164 | 8/20/2025 |
1.6.4 | 184 | 7/11/2025 |
1.6.3 | 184 | 6/24/2025 |
1.6.2 | 166 | 4/19/2025 |
1.6.1 | 213 | 2/2/2025 |
1.6.0 | 141 | 1/15/2025 |
1.5.0 | 159 | 12/20/2024 |
1.4.3 | 194 | 11/4/2024 |
1.4.2 | 148 | 10/28/2024 |
1.4.1 | 366 | 9/9/2024 |
1.4.0 | 329 | 8/20/2024 |
1.3.1 | 228 | 8/6/2024 |
1.3.0 | 194 | 6/5/2024 |
1.2.1 | 190 | 3/22/2024 |
1.2.0 | 319 | 12/27/2023 |
1.1.0 | 684 | 7/7/2023 |
1.0.1 | 294 | 4/15/2023 |
1.0.0 | 593 | 2/14/2023 |
0.9.0 | 376 | 2/2/2023 |
0.8.1 | 369 | 2/2/2023 |
0.8.0 | 338 | 2/1/2023 |
0.7.0 | 524 | 1/27/2023 |
0.6.0 | 529 | 1/24/2023 |
0.5.0 | 358 | 1/24/2023 |
0.4.0 | 557 | 1/22/2023 |
0.3.0 | 389 | 1/5/2023 |
0.2.0 | 388 | 1/4/2023 |
Compatability with Net 8, Net 9 and netstandard2.0