Mostlylucid.Ephemeral.Patterns.PersistentWindow
1.7.1
See the version list below for details.
dotnet add package Mostlylucid.Ephemeral.Patterns.PersistentWindow --version 1.7.1
NuGet\Install-Package Mostlylucid.Ephemeral.Patterns.PersistentWindow -Version 1.7.1
<PackageReference Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" Version="1.7.1" />
<PackageVersion Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" Version="1.7.1" />
<PackageReference Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" />
paket add Mostlylucid.Ephemeral.Patterns.PersistentWindow --version 1.7.1
#r "nuget: Mostlylucid.Ephemeral.Patterns.PersistentWindow, 1.7.1"
#:package Mostlylucid.Ephemeral.Patterns.PersistentWindow@1.7.1
#addin nuget:?package=Mostlylucid.Ephemeral.Patterns.PersistentWindow&version=1.7.1
#tool nuget:?package=Mostlylucid.Ephemeral.Patterns.PersistentWindow&version=1.7.1
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. |
-
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 (1)
Showing the top 1 NuGet packages that depend on Mostlylucid.Ephemeral.Patterns.PersistentWindow:
| Package | Downloads |
|---|---|
|
mostlylucid.ephemeral.complete
Meta-package that references all Mostlylucid.Ephemeral packages - bounded async execution with signals, atoms, and patterns. Install this single package to get everything. |
GitHub repositories
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 2.3.2 | 89 | 1/9/2026 |
| 2.3.1 | 87 | 1/9/2026 |
| 2.3.1-alpha0 | 86 | 1/9/2026 |
| 2.3.0 | 326 | 1/8/2026 |
| 2.3.0-alpha1 | 79 | 1/8/2026 |
| 2.1.0 | 88 | 1/8/2026 |
| 2.1.0-preview | 86 | 1/8/2026 |
| 2.0.1 | 93 | 1/8/2026 |
| 2.0.0 | 124 | 1/8/2026 |
| 2.0.0-alpha1 | 87 | 1/8/2026 |
| 1.7.1 | 417 | 12/11/2025 |
| 1.6.8 | 438 | 12/9/2025 |
| 1.6.7 | 424 | 12/9/2025 |
| 1.6.6 | 434 | 12/9/2025 |
| 1.6.5 | 431 | 12/9/2025 |
| 1.6.0 | 415 | 12/8/2025 |
| 1.5.0 | 413 | 12/8/2025 |
| 1.3.0 | 290 | 12/7/2025 |
| 1.2.2 | 286 | 12/7/2025 |
| 1.1.0-preview2 | 201 | 12/7/2025 |