ExpressionTreeToString 3.4.71

dotnet add package ExpressionTreeToString --version 3.4.71                
NuGet\Install-Package ExpressionTreeToString -Version 3.4.71                
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="ExpressionTreeToString" Version="3.4.71" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add ExpressionTreeToString --version 3.4.71                
#r "nuget: ExpressionTreeToString, 3.4.71"                
#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.
// Install ExpressionTreeToString as a Cake Addin
#addin nuget:?package=ExpressionTreeToString&version=3.4.71

// Install ExpressionTreeToString as a Cake Tool
#tool nuget:?package=ExpressionTreeToString&version=3.4.71                

Expression Tree To String

AppVeyor build status Tests NuGet Status       Test objects: NuGet TestObjects Status

Targets .NET Standard 2.0 and above / Framework 4.5.2 and above

Provides a ToString extension method which returns a string representation of an expression tree (an object inheriting from System.Linq.Expressions.Expression).

Expression<Func<bool>> expr = () => true;

Console.WriteLine(expr.ToString("C#"));
// prints: () => true

Console.WriteLine(expr.ToString("Visual Basic"));
// prints: Function() True

Console.WriteLine(expr.ToString("Factory methods", "C#"));
// prints:
/*
    // using static System.Linq.Expressions.Expression

    Lambda(
        Constant(true)
    )
*/

Console.WriteLine(expr.ToString("Object notation", "C#"));
// prints:
/*
    new Expression<Func<bool>> {
        NodeType = ExpressionType.Lambda,
        Type = typeof(Func<bool>),
        Body = new ConstantExpression {
            Type = typeof(bool),
            Value = true
        },
        ReturnType = typeof(bool)
    }
*/

Console.WriteLine(expr.ToString("Object notation", "Visual Basic"));
// prints:
/*
    New Expression(Of Func(Of Boolean)) With {
        .NodeType = ExpressionType.Lambda,
        .Type = GetType(Func(Of Boolean)),
        .Body = New ConstantExpression With {
            .Type = GetType(Boolean),
            .Value = True
        },
        .ReturnType = GetType(Boolean)
    }
*/

Console.WriteLine(expr.ToString("Textual tree"));
// prints:
/*
    Lambda (Func<bool>)
        Body - Constant (bool) = True
*/

var b = true;
Expression<Func<int>> expr1 = () => b ? 1 : 0;
Console.WriteLine(expr1.ToString("ToString"));
// prints:
/*
    () => IIF(value(_tests.Program+<>c__DisplayClass0_0).b, 1, 0)
*/

Console.WriteLine(expr1.ToString("DebugView"));
// prints:
/*
    .Lambda #Lambda1<System.Func`1[System.Int32]>() {
        .If (
            .Constant<_tests.Program+<>c__DisplayClass0_0>(_tests.Program+<>c__DisplayClass0_0).b
        ) {
            1
        } .Else {
            0
        }
    }
*/

Expression<Func<Person, bool>> filter = p => p => p.LastName == "A" || p.FirstName == "B" || p.DOB == DateTime.MinValue || p.LastName == "C" || p.FirstName == "D";
Console.WriteLine(filter.ToString("Dynamic LINQ", "C#"));
// prints:
/*
    "LastName in (\"A\", \"C\") || FirstName in (\"B\", \"D\") || DOB == DateTime.MinValue"
*/

Features:

  • Multiple writers:

    • Pseudo-code in C# or VB.NET
    • Factory method calls which generate this expression
    • Object notation, using object initializer and collection initializer syntax to describe objects
    • Textual tree, focusing on the properties related to the structure of the tree
    • ToString and DebugView reimplementation
    • Dynamic LINQ equivalent to the expression
  • For C# and VB pseudo-code representations:

    • Extension methods are rendered as instance methods

      Expression<Func<int, int>> expr = x => Enumerable.Range(1, x).Select(y => x * y).Count();
      Console.WriteLine(expr.ToString("C#"));
      // prints: (int x) => Enumerable.Range(1, x).Select((int y) => x * y).Count()
      
    • Closed-over variables are rendered as simple identifiers (instead of member access on the hidden compiler-generated class)

      var i = 7;
      var j = 8;
      Expression<Func<int>> expr = () => i + j;
      Console.WriteLine(expr.ToString("C#"));
      // prints: () => i + j
      
    • Calls to String.Concat and String.Format are mapped to the + operator and string interpolation, respectively (where possible):

      var name = "World";
      Expression<Func<string>> expr = () => string.Format("Hello, {0}!", name);
      Console.WriteLine(expr.ToString("C#"));
      // prints: () => $"Hello, {name}!"
      
    • Unnecessary conversions are not rendered:

      Expression<Func<IEnumerable<char>>> expr = () => (IEnumerable<char>)"abcd";
      Console.WriteLine(expr.ToString("C#"));
      // prints: () => "abcd"
      
    • Comparisons against an enum or char are rendered properly, not as comparison to int-converted value:

      var dow = DayOfWeek.Sunday;
      Expression<Func<bool>> expr = () => DateTime.Today.DayOfWeek == dow;
      
      Console.WriteLine(expr.ToString("Textual tree", "C#"));
      // prints:
      /*
        Lambda (Func<bool>)
            · Body - Equal (bool) = false
                · Left - Convert (int) = 3
                    · Operand - MemberAccess (DayOfWeek) DayOfWeek = DayOfWeek.Wednesday
                        · Expression - MemberAccess (DateTime) DateTime.Today = 30/09/2020 12:00:00 am
                · Right - Convert (int) = 0
                    · Operand - MemberAccess (DayOfWeek) dow = DayOfWeek.Sunday
                        · Expression - Constant (<closure>) = #<closure>      
      */
      
      Console.WriteLine(expr.ToString("C#"));
      // prints: () => DateTime.Today.DayOfWeek == dow
      
  • Each representation (including the ToString and DebugView renderers) can return the start and length of the substring corresponding to any of the paths of the tree's nodes, which can be used to find the substring corresponding to a given node in the tree:

    var s = expr.ToString("C#", out var pathSpans);
    Console.WriteLine(s);
    // prints: (Person p) => p.DOB.DayOfWeek == DayOfWeek.Tuesday
    
    (int start, int length) = pathSpans["Body.Left.Operand"];
    Console.WriteLine(s.Substring(start, length));
    // prints: p.DOB.DayOfWeek
    
  • Type names are rendered using language syntax and keywords, instead of the Type.Name property; e.g. List<string> or List(Of Date) instead of List`1

  • Supports the full range of types in System.Linq.Expressions, including .NET 4 expression types, and DynamicExpression

  • Extensibility -- allows creating custom renderers, or inheriting from existing renderers, to handle your own Expression-derived types

For more information, see the wiki.

Feedback

  • Star the project
  • File an issue
Product Compatible and additional computed target framework versions.
.NET net5.0 is compatible.  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.  net9.0 was computed.  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. 
.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 net452 is compatible.  net46 was computed.  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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (8)

Showing the top 5 NuGet packages that depend on ExpressionTreeToString:

Package Downloads
RaaLabs.Edge

Package Description

NCPC.Utils

Usefull methods of common use

ExpressionTreeVisualizer.UI

UI pieces for displaying expression trees

NiuX.Common

csharp common, utils, helpers, tools etc.

GCore.Logging.Expressions

Useful C# Stuff for .NET Standard 2.0

GitHub repositories (3)

Showing the top 3 popular GitHub repositories that depend on ExpressionTreeToString:

Repository Stars
yangzhongke/NETBookMaterials
zspitz/ExpressionTreeVisualizer
Debugging visualizer for expression trees
NeVeSpl/RevitDBExplorer
Interactive Revit database exploration tool to view and edit BIM element parameters, properties and relationships.
Version Downloads Last updated
3.4.71 208,504 2/25/2022
3.4.70 4,128 12/23/2021
3.4.68 1,072 12/14/2021
3.4.67 317 12/14/2021
3.4.65 23,926 7/14/2021
3.4.64 390 7/14/2021
3.4.63 18,389 7/5/2021
3.4.62 669 6/29/2021
3.4.61 587 6/23/2021
3.4.60 397 6/23/2021
3.3.59 437 6/17/2021
3.3.58 497 6/7/2021
3.3.57 396 6/6/2021
3.3.56 454 5/31/2021
3.2.54 42,293 1/16/2021
3.2.53 6,850 12/6/2020
3.2.52 558 12/5/2020
3.2.51 559 12/5/2020
3.2.50 547 12/5/2020
3.2.49 442 12/5/2020
3.2.48 2,219 10/28/2020
3.1.47 510 10/8/2020
3.1.45 745 9/30/2020
3.1.44 559 9/30/2020
3.1.43 493 9/24/2020
3.1.42 517 9/23/2020
3.0.41 627 9/16/2020
2.2.40 505 9/15/2020
2.2.39 490 9/15/2020
2.2.38 475 9/14/2020
2.2.37 578 8/31/2020
2.2.36 493 8/26/2020
2.1.34 513 8/24/2020
2.1.33 441 8/24/2020
2.0.32 546 8/18/2020
2.0.29 925 8/5/2020
2.0.28 536 7/19/2020
2.0.27 4,542 7/19/2020
2.0.26 588 7/19/2020
2.0.25 488 7/16/2020
2.0.23 567 6/28/2020
2.0.22 587 6/27/2020
2.0.21 569 6/27/2020
2.0.20 6,170 3/29/2020
2.0.19 626 3/29/2020
2.0.15 764 12/18/2019
2.0.14 561 12/18/2019
2.0.13 556 12/18/2019
2.0.12 530 12/18/2019
2.0.11 534 12/18/2019
2.0.10 582 12/16/2019
2.0.4 533 12/15/2019
1.6.2 573 12/15/2019
1.5.67 2,894 8/15/2019
1.5.66 668 8/14/2019
1.5.65 699 8/13/2019
1.4.64 691 8/1/2019
1.4.63 686 7/4/2019
1.4.62 668 7/4/2019
1.4.57 651 6/26/2019
1.3.55 680 6/25/2019
1.3.54 672 6/25/2019
1.3.53 648 6/24/2019
1.3.52 684 6/23/2019
1.3.51 673 6/21/2019
1.3.50 701 6/21/2019
1.3.49 686 6/21/2019
1.3.41 661 6/20/2019
1.3.35 667 6/18/2019
1.3.34 702 5/30/2019
1.3.33 686 5/30/2019
1.2.32 689 5/30/2019
1.2.30 702 5/30/2019
1.2.29 659 5/29/2019
1.2.28 719 5/21/2019
1.2.27 700 5/14/2019
1.2.19 3,816 5/3/2019
1.1.10 719 3/24/2019
1.1.7 715 3/15/2019
1.1.6 684 3/14/2019
1.1.5 667 3/14/2019
1.0.1 786 2/17/2019
1.0.0 799 2/14/2019

Non-statement expression support is complete, escaped string literals, exceptions written into source