Filter Transformation
The FilterTransformation
is used to remove rows from a data flow based on a specified condition. It evaluates each row using a predicate function — if the predicate returns true
, the row is kept; otherwise, it is filtered out and not passed on.
The FilterTransformation
is not the only method to apply filtering. You can also use predicates directly when linking components. See the Using predicates when linking section for examples.
- Type: Non-blocking transformation
- Buffering: One input buffer
- Execution: Row-by-row in memory
Example
The following example demonstrates how to filter rows based on the Id
property using a FilterTransformation
.
public class MyRow {
public int Id { get; set; }
public string Value { get; set; }
}
public static void Main() {
var source = new MemorySource<MyRow>();
source.DataAsList.Add(new MyRow() { Id = 1, Value = "Test1" });
source.DataAsList.Add(new MyRow() { Id = 2, Value = "Test2" });
source.DataAsList.Add(new MyRow() { Id = 3, Value = "Test3" });
var filter = new FilterTransformation<MyRow>();
filter.FilterPredicate = row => row.Id == 2;
var dest = new MemoryDestination<MyRow>();
source.LinkTo(filter).LinkTo(dest);
Network.Execute(source);
foreach (var row in dest.Data)
Console.WriteLine($"Id:{row.Id} Value:{row.Value}");
//Outputs
//Id:1 Value:Test1
//Id:3 Value:Test3
}
Using Predicates When Linking
Instead of using a dedicated FilterTransformation
, you can apply filtering directly when linking components:
Example – Basic Predicate Filtering
var source = new MemorySource<MyRow>();
source.DataAsList.Add(new MyRow() { Id = 1, Value = "Test1" });
source.DataAsList.Add(new MyRow() { Id = 2, Value = "Test2" });
source.DataAsList.Add(new MyRow() { Id = 3, Value = "Test3" });
var dest = new MemoryDestination<MyRow>();
var voidDest = new VoidDestination<MyRow>();
source.LinkTo(dest, row => row.Id != 2);
source.LinkTo(voidDest, row => row.Id == 2);
Network.Execute(source);
Example – Simplified with Fallback Predicate
var source = new MemorySource<MyRow>();
source.DataAsList.Add(new MyRow() { Id = 1, Value = "Test1" });
source.DataAsList.Add(new MyRow() { Id = 2, Value = "Test2" });
source.DataAsList.Add(new MyRow() { Id = 3, Value = "Test3" });
var dest = new MemoryDestination<MyRow>();
source.LinkTo(dest, row => row.Id != 2, row => row.Id == 2);
Network.Execute(source);
Dynamic Objects
FilterTransformation
also works with dynamic inputs (ExpandoObject
). You can use dynamic access within the predicate function:
public void ExampleFilterDynamic() {
var source = new MemorySource();
dynamic r1 = new ExpandoObject(); r1.Id = 1; r1.Value = "Test1";
dynamic r2 = new ExpandoObject(); r2.Id = 2; r2.Value = "Test2";
dynamic r3 = new ExpandoObject(); r3.Id = 3; r3.Value = "Test3";
source.DataAsList.Add(r1);
source.DataAsList.Add(r2);
source.DataAsList.Add(r3);
var filter = new FilterTransformation();
filter.FilterPredicate = row => ((dynamic)row).Id == 2;
var dest = new MemoryDestination();
source.LinkTo(filter).LinkTo(dest);
Network.Execute(source);
foreach (dynamic row in dest.Data)
Console.WriteLine($"Id:{row.Id} Value:{row.Value}");
//Outputs
//Id:1 Value:Test1
//Id:3 Value:Test3
}
Error Handling
FilterTransformation
supports error redirection using LinkErrorTo()
. If the predicate throws an exception, it can be caught and routed to another component:
var errorDest = new MemoryDestination<ETLBoxError>();
filter.LinkErrorTo(errorDest);
This behavior is only available when using the FilterTransformation
. If you use predicates directly in LinkTo
, exceptions are silently ignored and the affected row is discarded.
FilteredCount and PassedCount
To monitor the result of filtering, FilterTransformation
exposes the following metrics:
FilteredCount
: Number of rows that were removed (predicate returnedfalse
)PassedCount
: Number of rows passed on (predicate returnedtrue
)
Example:
var filter = new FilterTransformation<MyRow>();
filter.FilterPredicate = row => row.Id != 2;
source.LinkTo(filter).LinkTo(dest);
Network.Execute(source);
Console.WriteLine($"Rows Passed: {filter.PassedCount}");
Console.WriteLine($"Rows Filtered: {filter.FilteredCount}");
These properties are useful for auditing and debugging data flow logic.