Mostlylucid.Ephemeral.Patterns.PersistentWindow
1.7.1
dotnet add package Mostlylucid.Ephemeral.Patterns.PersistentWindow --version 1.7.1
NuGet\Install-Package Mostlylucid.Ephemeral.Patterns.PersistentWindow -Version 1.7.1
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" Version="1.7.1" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" Version="1.7.1" />
<PackageReference Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" />
For projects that support Central Package Management (CPM), copy this XML node into the solution Directory.Packages.props file to version the package.
paket add Mostlylucid.Ephemeral.Patterns.PersistentWindow --version 1.7.1
The NuGet Team does not provide support for this client. Please contact its maintainers for support.
#r "nuget: Mostlylucid.Ephemeral.Patterns.PersistentWindow, 1.7.1"
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
#:package Mostlylucid.Ephemeral.Patterns.PersistentWindow@1.7.1
#:package directive can be used in C# file-based apps starting in .NET 10 preview 4. Copy this into a .cs file before any lines of code to reference the package.
#addin nuget:?package=Mostlylucid.Ephemeral.Patterns.PersistentWindow&version=1.7.1
#tool nuget:?package=Mostlylucid.Ephemeral.Patterns.PersistentWindow&version=1.7.1
The NuGet Team does not provide support for this client. Please contact its maintainers for support.
Mostlylucid.Ephemeral.Patterns.PersistentWindow
🚨🚨 WARNING 🚨🚨 - Though in the 1.x range of version THINGS WILL STILL BREAK. This is the lab for developing this concept when stabilized it'll becoe the first styloflow release 🚨🚨🚨
Signal window that periodically persists to SQLite and restores on restart. Survives process restarts while maintaining in-memory performance.
dotnet add package mostlylucid.ephemeral.patterns.persistentwindow
Quick Start
using Mostlylucid.Ephemeral.Patterns.PersistentWindow;
await using var window = new PersistentSignalWindow(
"Data Source=signals.db",
flushInterval: TimeSpan.FromSeconds(30));
// On startup: restore previous signals
await window.LoadFromDiskAsync(maxAge: TimeSpan.FromHours(24));
// Raise signals as normal
window.Raise("order.completed", key: "order-service");
window.Raise("payment.processed", key: "payment-service");
// Query signals
var recentOrders = window.Sense("order.*");
// Signals automatically flush every 30 seconds
// Also flushes on dispose
All Options
new PersistentSignalWindow(
// Required: SQLite connection string
connectionString: "Data Source=signals.db",
// How often to flush to disk
// Default: 30 seconds
flushInterval: TimeSpan.FromSeconds(30),
// Max signals to persist per flush
// Default: 1000
maxSignalsPerFlush: 1000,
// Max signals in memory window
// Default: 10000
maxWindowSize: 10000,
// Max age of signals in memory
// Default: 10 minutes
windowMaxAge: TimeSpan.FromMinutes(10),
// Signal sampling rate for diagnostics
// Default: 10 (1 in 10)
sampleRate: 10
)
API Reference
// Raise signals
void Raise(string signal, string? key = null);
void Raise(SignalEvent evt);
// Query signals by pattern
IReadOnlyList<SignalEvent> Sense(string? pattern = null);
IReadOnlyList<SignalEvent> Sense(Func<SignalEvent, bool> predicate);
// Force immediate flush to SQLite
Task FlushAsync(CancellationToken ct = default);
// Load signals from SQLite (call on startup)
Task LoadFromDiskAsync(TimeSpan? maxAge = null, CancellationToken ct = default);
// Get statistics
WindowStats GetStats(); // (InMemoryCount, TotalRaised, LastFlushedId)
// Access underlying sink for advanced usage
SignalSink Sink { get; }
// Dispose (flushes remaining signals)
ValueTask DisposeAsync();
How It Works
┌─────────────────────────────────────┐
Raise() ───────>│ In-Memory SignalSink (fast) │
│ - maxWindowSize: 10000 │
│ - windowMaxAge: 10 minutes │
└─────────────┬───────────────────────┘
│
│ Every 30 seconds (flushInterval)
▼
┌─────────────────────────────────────┐
│ SQLite (durable) │
│ - Single-writer coordination │
│ - WAL mode for performance │
│ - Indexed by timestamp & signal │
└─────────────────────────────────────┘
│
│ On startup: LoadFromDiskAsync()
▼
┌─────────────────────────────────────┐
│ Restored signals back to memory │
└─────────────────────────────────────┘
Signals Emitted
| Signal | Description |
|---|---|
window.initialized |
SQLite schema created |
window.raise |
Signal raised (sampled) |
window.flush.start:{count} |
Starting flush |
window.flush.done:{count} |
Flush completed |
window.flush.error |
Flush failed |
window.load.done:{count} |
Signals loaded from disk |
Example: Error Monitoring with Persistence
await using var window = new PersistentSignalWindow(
"Data Source=errors.db",
flushInterval: TimeSpan.FromSeconds(10),
maxWindowSize: 50000);
// On startup: restore last 24 hours of errors
await window.LoadFromDiskAsync(maxAge: TimeSpan.FromHours(24));
// In your error handler
try
{
await ProcessRequest();
}
catch (Exception ex)
{
window.Raise($"error:{ex.GetType().Name}", key: Environment.MachineName);
}
// Dashboard query
var last5Minutes = window.Sense(s =>
s.Signal.StartsWith("error:") &&
s.Timestamp > DateTimeOffset.UtcNow.AddMinutes(-5));
Console.WriteLine($"Errors in last 5 min: {last5Minutes.Count}");
Example: Distributed Event Tracking
// Each service instance has its own window
await using var window = new PersistentSignalWindow(
$"Data Source=events_{Environment.MachineName}.db",
windowMaxAge: TimeSpan.FromHours(1));
// Restore on startup
await window.LoadFromDiskAsync(maxAge: TimeSpan.FromHours(1));
// Track events
window.Raise("user.login", key: userId);
window.Raise("order.placed", key: orderId);
// Query for patterns
var userActivity = window.Sense("user.*");
var orderEvents = window.Sense("order.*");
Example: Graceful Shutdown
var window = new PersistentSignalWindow("Data Source=app.db");
// Handle shutdown signal
Console.CancelKeyPress += async (s, e) =>
{
e.Cancel = true;
Console.WriteLine("Flushing signals...");
await window.FlushAsync();
await window.DisposeAsync();
Environment.Exit(0);
};
// Or in ASP.NET Core
public class SignalWindowService : IHostedService
{
private readonly PersistentSignalWindow _window;
public async Task StartAsync(CancellationToken ct)
{
await _window.LoadFromDiskAsync(TimeSpan.FromHours(24), ct);
}
public async Task StopAsync(CancellationToken ct)
{
await _window.FlushAsync(ct);
await _window.DisposeAsync();
}
}
Database Schema
The window creates these tables automatically:
CREATE TABLE signals (
id INTEGER PRIMARY KEY,
operation_id INTEGER NOT NULL,
signal TEXT NOT NULL,
key TEXT,
timestamp TEXT NOT NULL,
created_at TEXT DEFAULT CURRENT_TIMESTAMP
);
CREATE INDEX idx_signals_timestamp ON signals(timestamp);
CREATE INDEX idx_signals_signal ON signals(signal);
Related Packages
| Package | Description |
|---|---|
| mostlylucid.ephemeral | Core library |
| mostlylucid.ephemeral.sqlite.singlewriter | SQLite helper |
| mostlylucid.ephemeral.patterns.signallogwatcher | Signal watching |
| mostlylucid.ephemeral.complete | All in one DLL |
License
Unlicense (public domain)
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net6.0 is compatible. 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 is compatible. 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. 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. net10.0 is compatible. 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. |
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
-
net10.0
- Microsoft.Data.Sqlite (>= 8.0.0)
- mostlylucid.ephemeral (>= 1.7.1)
-
net6.0
- Microsoft.Data.Sqlite (>= 8.0.0)
- mostlylucid.ephemeral (>= 1.7.1)
-
net7.0
- Microsoft.Data.Sqlite (>= 8.0.0)
- mostlylucid.ephemeral (>= 1.7.1)
-
net8.0
- Microsoft.Data.Sqlite (>= 8.0.0)
- mostlylucid.ephemeral (>= 1.7.1)
-
net9.0
- Microsoft.Data.Sqlite (>= 8.0.0)
- mostlylucid.ephemeral (>= 1.7.1)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.