![]() |
VOOZH | about |
dotnet add package Mostlylucid.Ephemeral.Patterns.PersistentWindow --version 2.6.3
NuGet\Install-Package Mostlylucid.Ephemeral.Patterns.PersistentWindow -Version 2.6.3
<PackageReference Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" Version="2.6.3" />
<PackageVersion Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" Version="2.6.3" />Directory.Packages.props
<PackageReference Include="Mostlylucid.Ephemeral.Patterns.PersistentWindow" />Project file
paket add Mostlylucid.Ephemeral.Patterns.PersistentWindow --version 2.6.3
#r "nuget: Mostlylucid.Ephemeral.Patterns.PersistentWindow, 2.6.3"
#:package Mostlylucid.Ephemeral.Patterns.PersistentWindow@2.6.3
#addin nuget:?package=Mostlylucid.Ephemeral.Patterns.PersistentWindow&version=2.6.3Install as a Cake Addin
#tool nuget:?package=Mostlylucid.Ephemeral.Patterns.PersistentWindow&version=2.6.3Install as a Cake Tool
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
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
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
)
// 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();
┌─────────────────────────────────────┐
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 │
└─────────────────────────────────────┘
| 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 |
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}");
// 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.*");
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();
}
}
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);
| 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 |
Unlicense (public domain)
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net8.0 net8.0 is compatible. net8.0-android net8.0-android was computed. net8.0-browser net8.0-browser was computed. net8.0-ios net8.0-ios was computed. net8.0-maccatalyst net8.0-maccatalyst was computed. net8.0-macos net8.0-macos was computed. net8.0-tvos net8.0-tvos was computed. net8.0-windows net8.0-windows was computed. net9.0 net9.0 is compatible. net9.0-android net9.0-android was computed. net9.0-browser net9.0-browser was computed. net9.0-ios net9.0-ios was computed. net9.0-maccatalyst net9.0-maccatalyst was computed. net9.0-macos net9.0-macos was computed. net9.0-tvos net9.0-tvos was computed. net9.0-windows net9.0-windows was computed. net10.0 net10.0 is compatible. net10.0-android net10.0-android was computed. net10.0-browser net10.0-browser was computed. net10.0-ios net10.0-ios was computed. net10.0-maccatalyst net10.0-maccatalyst was computed. net10.0-macos net10.0-macos was computed. net10.0-tvos net10.0-tvos was computed. net10.0-windows net10.0-windows was computed. |
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. |
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 2.6.3 | 101 | 5/22/2026 |
| 2.6.2 | 101 | 5/22/2026 |
| 2.6.0 | 97 | 5/22/2026 |
| 2.5.1 | 96 | 5/22/2026 |
| 2.5.0 | 100 | 5/3/2026 |
| 2.4.0 | 114 | 4/17/2026 |
| 2.3.2 | 124 | 1/9/2026 |
| 2.3.1 | 123 | 1/9/2026 |
| 2.3.1-alpha0 | 118 | 1/9/2026 |
| 2.3.0 | 1,198 | 1/8/2026 |
| 2.3.0-alpha1 | 115 | 1/8/2026 |
| 2.1.0 | 123 | 1/8/2026 |
| 2.1.0-preview | 120 | 1/8/2026 |
| 2.0.1 | 126 | 1/8/2026 |
| 2.0.0 | 158 | 1/8/2026 |
| 2.0.0-alpha1 | 118 | 1/8/2026 |
| 1.7.1 | 444 | 12/11/2025 |
| 1.6.8 | 466 | 12/9/2025 |
| 1.6.7 | 449 | 12/9/2025 |
| 1.6.6 | 458 | 12/9/2025 |