Gress 2.0.1
See the version list below for details.
dotnet add package Gress --version 2.0.1
NuGet\Install-Package Gress -Version 2.0.1
<PackageReference Include="Gress" Version="2.0.1" />
paket add Gress --version 2.0.1
#r "nuget: Gress, 2.0.1"
// Install Gress as a Cake Addin #addin nuget:?package=Gress&version=2.0.1 // Install Gress as a Cake Tool #tool nuget:?package=Gress&version=2.0.1
Gress
✅ Project status: active. What does it mean?
Gress is a library that extends the standard IProgress<T>
interface with a set of utilities for collecting, transforming, filtering, and multiplexing progress updates in your code.
💬 If you want to chat, join my Discord server.
Download
📦 NuGet: dotnet add package Gress
Screenshots
Usage
Percentage type
To make progress updates more explicit, Gress provides a universal progress unit -- the Percentage
type.
Unlike raw numeric values commonly used with IProgress<T>
, this type unambiguously represents reported progress as a portion of work that has been completed so far.
An instance of Percentage
can be created from either a value or a fraction:
using Gress;
// Mapped from value
var fiftyPercent = Percentage.FromValue(50); // 50%
// Mapped from fractional representation
var twentyPercent = Percentage.FromFraction(0.2); // 20%
Similarly, both value and fraction can be extracted from an initialized Percentage
by accessing the corresponding properties:
using Gress;
var fiftyPercent = Percentage.FromValue(50);
var asValue = fiftyPercent.Value; // 50.0 (double)
var asFraction = fiftyPercent.Fraction; // 0.5 (double)
Using Percentage
in your IProgress<T>
handlers lets you communicate progress updates without making any assumptions about their semantics:
using Gress;
async Task PerformWorkAsync(IProgress<Percentage> progrss)
{
await Task.Delay(100);
// Half-way done
progress.Report(Percentage.FromValue(50));
await Task.Delay(100);
// Finished
progress.Report(Percentage.FromFraction(1));
}
// ...
var progress = new Progress<Percentage>(p => Console.WriteLine(p));
await PerformWorkAsync(progress);
// Console output:
// 50,0%
// 100,0%
When interfacing with external methods, however, you may need to provide a specific progress handler required by their signature. In such cases, you can convert an existing percentage-based handler into another type using one of the available extension methods:
using Gress;
async Task FooAsync(IProgress<double> progress) { /* ... */ }
async Task BarAsync(IProgress<int> progress) { /* ... */ }
var progress = new Progress<Percentage>(p => /* ... */);
await FooAsync(progress.ToDoubleBased());
await BarAsync(progress.ToInt32Based());
Likewise, you can also perform conversions in the other direction, which can be useful for preserving backwards-compatibility in your own methods:
using Gress;
async Task FooAsync(IProgress<double> progress)
{
var actualProgress = progress.ToPercentageBased();
// Reports 0.5 on the original progress handler
actualProgress.Report(Percentage.FromFraction(0.5));
}
async Task BarAsync(IProgress<int> progress)
{
var actualProgress = progress.ToPercentageBased();
// Reports 50 on the original progress handler
actualProgress.Report(Percentage.FromFraction(0.5));
}
💡 When converting between percentage-based and double-based handlers, percentages are mapped using their fractional form by default. To override this behavior and map by value instead, use
ToDoubleBased(false)
andToPercentageBased(false)
.
💡 For more complex conversion scenarios, consider using the
WithTransform(...)
method.
Terminal handlers
Every progress reporting chain ultimately ends with a terminal handler, which usually relays the information to the user or stores it somewhere else. To simplify some of the most common scenarios, Gress comes with two terminal handlers built in.
Progress container
This handler simply represents an object with a single property, whose value is overwritten every time a new progress update is reported.
It also implements the INotifyPropertyChanged
interface, allowing the property to be bound from XAML-based user interfaces.
Here's a very basic example of how you would use it in a typical WPF application:
public class MainViewModel
{
public ProgressContainer<Percentage> Progress { get; } = new();
public IRelayCommand PerformWorkCommand { get; }
public MainViewModel() =>
PerformWorkCommand = new RelayCommand(PerformWork);
public async void PerformWork()
{
for (var i = 1; i <= 100; i++)
{
await Task.Delay(200);
Progress.Report(Percentage.FromValue(i));
}
}
}
<Window
x:Class="MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
d:DataContext="{d:DesignInstance Type=MainViewModel}">
<StackPanel>
<Button
Margin="32"
Content="Execute"
Command="{Binding PerformWorkCommand}" />
<ProgressBar
Margin="32"
Height="10"
Minimum="0"
Maximum="100"
Value="{Binding Progress.Current.Value, Mode=OneWay}" />
</StackPanel>
</Window>
Progress collector
This handler works by storing all reported progress updates in a collection, whose values can be retrieved later. It's primarily designed for testing purposes.
Here's how you can use it to verify that a method reported its progress correctly:
[Fact]
public async Task My_method_reports_progress_correctly()
{
// Arrange
var progress = new ProgressCollector<Percentage>();
var worker = new Worker();
// Act
await worker.PerformWorkAsync(progress);
// Assert
var values = progress.GetValues();
values.Should().NotBeEmpty(); // not empty
values.Should().OnlyHaveUniqueItems(); // no redundant updates
}
Composing handlers
Existing progress handlers can be composed into more complex handlers using some of the extension methods that Gress offers. These can be used to easily apply transformations, inject filtering logic, or merge multiple handlers together.
Transformation
You can use WithTransform(...)
to create a handler that transforms all reported progress updates into a different form:
using Gress;
enum Status { Started, HalfWay, Completed }
var progress = new Progress<Percentage>(p => /* ... */);
// Transform into a progress handler that accepts an enum value and maps
// it into a value of the original type
var progressTransformed = progress.WithTransform((Status s) => s switch
{
Status.Completed => Percentage.FromValue(100), // 100%
Status.HalfWay => Percentage.FromValue(50), // 50%
_ => Percentage.FromValue(0) // 0%
});
// This effectively reports 50% on the original handler
progressTransformed.Report(Status.HalfWay);
A simpler overload of the above method can also be used when transforming between values of the same type:
using Gress;
var progress = new Progress<int>(p => /* ... */);
var progressTransformed = progress.WithTransform(p => 5 * p);
// This effectively reports 50 on the original handler
progressTransformed.Report(10);
💡 Method
WithTransform(...)
bears some resemblance to LINQ'sSelect(...)
, however they are not completely equivalent. The main difference is that the flow of data inIProgress<T>
is inverse to that ofIEnumerable<T>
, which means that the transformations inWithTransform(...)
are applied in the opposite direction.
Filtering
You can use WithFilter(...)
to create a handler that drops progress updates that don't satisfy a predicate:
using Gress;
var progress = new Progress<Percentage>(p => /* ... */);
// Filter out values below 10%
var progressFiltered = progress.WithFilter(p => p.Fraction >= 0.1);
// ✖
progressFiltered.Report(Percentage.FromFraction(0.05));
// ✓
progressFiltered.Report(Percentage.FromFraction(0.25));
Deduplication
You can use WithDeduplication(...)
to create a handler that filters out consecutive progress updates with the same value:
using Gress;
var progress = new Progress<Percentage>(p => /* ... */);
var progressDeduplicated = progress.WithDeduplication();
progressDeduplicated.Report(Percentage.FromFraction(0.1)); // ✓
progressDeduplicated.Report(Percentage.FromFraction(0.3)); // ✓
progressDeduplicated.Report(Percentage.FromFraction(0.3)); // ✖
progressDeduplicated.Report(Percentage.FromFraction(0.3)); // ✖
progressDeduplicated.Report(Percentage.FromFraction(0.5)); // ✓
Merging
You can use Merge(...)
to combine multiple progress handlers into one:
using Gress;
var progress1 = new Progress<Percentage>(p => /* ... */);
var progress2 = new Progress<Percentage>(p => /* ... */);
var progressMerged = progress1.Merge(progress2);
// Reports 50% on both progress handlers
progressMerged.Report(Percentage.FromFraction(0.5));
This method can also be called on collections:
using Gress;
var progresses = new[]
{
new Progress<Percentage>(p => /* ... */),
new Progress<Percentage>(p => /* ... */),
new Progress<Percentage>(p => /* ... */),
new Progress<Percentage>(p => /* ... */)
};
var progressMerged = progresses.Merge();
// Reports 50% on all progress handlers
progressMerged.Report(Percentage.FromFraction(0.5));
Multiplexing
Multiplexing allows a single handler to aggregate progress reports from multiple input sources. This is useful when you want to encapsulate several progress-reporting operations in a single higher-order operation.
To do this, create a muxer for the target progress handler and use it to create an input for each operation:
using Gress;
var progress = new Progress<Percentage>(p => /* ... */);
var muxer = progress.CreateMuxer();
var progressSub1 = muxer.CreateInput();
var progressSub2 = muxer.CreateInput();
var progressSub3 = muxer.CreateInput();
When a progress update is reported on any of these inputs, all of the updates up to that point are aggregated into one and routed to the target handler. The sample below illustrates this process in detail:
// ...
progressSub1.Report(Percentage.FromFraction(0.5));
// Input 1 -> 50%
// Input 2 -> 0%
// Input 3 -> 0%
// Total -> ~17%
progressSub1.Report(Percentage.FromFraction(1));
progressSub2.Report(Percentage.FromFraction(0.75));
// Input 1 -> 100%
// Input 2 -> 75%
// Input 3 -> 0%
// Total -> ~58%
progressSub2.Report(Percentage.FromFraction(1));
progressSub3.Report(Percentage.FromFraction(0.9));
// Input 1 -> 100%
// Input 2 -> 100%
// Input 3 -> 90%
// Total -> ~97%
progressSub3.Report(Percentage.FromFraction(1));
// Input 1 -> 100%
// Input 2 -> 100%
// Input 3 -> 100%
// Total -> 100%
Additionally, since muxer inputs are progress handlers themselves, they can be multiplexed as well. Doing this allows you to create a progress reporting chain that form a hierarchy:
using Gress;
async Task PerformWorkAsync(IProgress<Percentage> progress)
{
for (var i = 1; i <= 100; i++)
{
await Task.Delay(200);
progress.Report(Percentage.FromValue(i));
}
}
async Task FooAsync(IProgress<Percentage> progress)
{
var muxer = progress.CreateMuxer();
var progressSub1 = muxer.CreateInput();
var progressSub2 = muxer.CreateInput();
await Task.WhenAll(
PerformWorkAsync(progressSub1),
PerformWorkAsync(progressSub2)
);
}
async Task BarAsync(IProgress<Percentage> progress)
{
var muxer = progress.CreateMuxer();
var progressSub1 = muxer.CreateInput();
var progressSub2 = muxer.CreateInput();
var progressSub3 = muxer.CreateInput();
await Task.WhenAll(
FooAsync(progressSub1),
FooAsync(progressSub2),
FooAsync(progressSub3)
);
}
⚠️ Muxing is only available on percentage-based handlers because it relies on their ability to represent progress as a relative fraction. If required, you can convert certain other handlers into percentage-based handlers using the
ToPercentageBased()
extension method.
With custom weight
A muxer input may be assigned a custom weight modifier, which determines its priority in relation to others. Progress reported on an input with higher weight influences the aggregated progress to a greater degree and vice versa.
You can specify the input weight by passing it to the CreateInput(...)
method:
using Gress;
var progress = new Progress<Percentage>(p => /* ... */);
var muxer = progress.CreateMuxer();
var progressSub1 = muxer.CreateInput(1);
var progressSub2 = muxer.CreateInput(4);
// Weight split:
// Input 1 -> 20% of total
// Input 2 -> 80% of total
progressSub1.Report(Percentage.FromFraction(0.9));
progressSub2.Report(Percentage.FromFraction(0.3));
// Input 1 -> 90% (less important)
// Input 2 -> 30% (more important)
// Total -> 42% (would've been 60% without weights)
With auto-reset
In some cases, you may need to report progress on an infinite workflow where new operations are started and completed in a continuous fashion. This can be achieved by using an auto-reset muxer.
Inputs to an auto-reset muxer implement the ICompletableProgress<T>
interface and are capable of reporting completion after all of the underlying work is finished.
Once all connected inputs report completion, they are disconnected and the muxer resets back to its initial state.
To create an auto-reset muxer, call WithAutoReset()
on an existing instance:
using Gress;
using Gress.Completable;
var progress = new Progress<Percentage>(p => /* ... */);
var muxer = progress.CreateMuxer().WithAutoReset();
var progressSub1 = muxer.CreateInput();
var progressSub2 = muxer.CreateInput();
progressSub1.Report(Percentage.FromFraction(0.3));
progressSub2.Report(Percentage.FromFraction(0.9));
// Input 1 -> 30%
// Input 2 -> 90%
// Total -> 60%
progressSub1.Report(Percentage.FromFraction(1));
progressSub1.ReportCompletion();
// Input 1 -> 100% (completed)
// Input 2 -> 90%
// Total -> 95%
progressSub2.Report(Percentage.FromFraction(1));
progressSub2.ReportCompletion();
// All inputs disconnected
// Total -> 0%
var progressSub3 = muxer.CreateInput();
progressSub3.Report(Percentage.FromFraction(0.5));
// Input 3 -> 50%
// Total -> 50%
💡 You can wrap an instance of
ICompletableProgress<T>
in a disposable container by callingToDisposable()
. This allows you to place the handler in ausing (...)
block, which ensures that the completion is always reported at the end.
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 was computed. 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. |
.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
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories (4)
Showing the top 4 popular GitHub repositories that depend on Gress:
Repository | Stars |
---|---|
Tyrrrz/YoutubeDownloader
Downloads videos and playlists from YouTube
|
|
Tyrrrz/DiscordChatExporter
Exports Discord chat logs to a file
|
|
Tyrrrz/YoutubeExplode
Abstraction layer over YouTube's internal API
|
|
Tyrrrz/OsuHelper
Beatmap suggester for osu!
|