Roslyn method survey - Loading a solution with MsBuild
I wanted to survey all method signatures in a codebase. I split the problem into two: loading the solution; and indexing all methods. The first turned out to be a complete rabbit hole. The documentation around MsBuild is a morass of partially-incomplete, somewhat misleading treacle; I’ve summarised some points here. MsBuild is also passive aggressive at best. If something doesn’t work, it won’t tell you - you have to know how to ask it.
MsBuild
Using MsBuild from C# is relatively easy; however, a lot of information out there concerns itself with finding a version of MsBuild to use from code that mostly relates to older versions of MsBuild - and which are actively obstructive today. Nowadays you can just include a dependency on the NuGet Microsoft.Build.Locator
. If you’re using Rider, as I am, you might not have installed the Microsoft build tools, but they can be downloaded as a standalone installer from Microsoft’s all MsBuild downloads page.
So, to cut to the solution create a CSharp executable project, add NuGet packages:
Microsoft.Build.Framework
Microsoft.Build.Tasks.Core
Microsoft.Build
Microsoft.CodeAnalysis.Workspaces.MSBuildWorkspace
Microsoft.CodeAnalysis.CSharp
Microsoft.CodeAnalysis.CSharp.Workspaces
and then, importantly, initialise the MsBuildLocator before calling any methods that use it; a small, but interesting wrinkle that I didn’t get immediately. This example from Microsoft was crtically useful in getting me (re)started.
using Microsoft.Build.Locator; | |
namespace CleanAnalysis | |
{ | |
class Program | |
{ | |
static void Main(string[] args) | |
{ | |
MSBuildLocator.RegisterDefaults(); | |
// TODO: stuff here, importantly, in a method call | |
} | |
} | |
} |
Then add a method that we can use to load all syntax-api-supporting documents in the solution … with some NuGets and helpful comments along the way:
NuGet.Frameworks
NuGet.ProjectModel
NuGet.Packaging
using System; | |
using System.Collections.Generic; | |
using System.Collections.Immutable; | |
using System.IO; | |
using System.Linq; | |
using Microsoft.Build.Locator; | |
using Microsoft.CodeAnalysis; | |
using Microsoft.CodeAnalysis.MSBuild; | |
using Document = Microsoft.CodeAnalysis.Document; | |
namespace CleanAnalysis | |
{ | |
class Program | |
{ | |
public static IEnumerable<Document> LoadSolution(string solutionDir) | |
{ | |
var solutionFilePath = Path.GetFullPath(solutionDir); | |
var documents = new List<Document>(); | |
// Getting hold of the right MSBuild is a nightmare. You can install old | |
// versions without VS, from | |
// https://visualstudio.microsoft.com/vs/older-downloads | |
using var workspace = MSBuildWorkspace.Create(); | |
var solution = workspace.OpenSolutionAsync(solutionFilePath).Result; | |
// If you forget nuget Microsoft.CodeAnalysis.CSharp and | |
// Microsoft.CodeAnalysis.CSharp.Workspaces you'll silently get an empty list | |
// of projects here and have *no idea why* | |
foreach (var projectId in solution.ProjectIds) | |
{ | |
var project = solution.GetProject(projectId); | |
// ARGH THIS BL**DY STUFF IS SO UNCOMMUNICATIVE | |
// Turns out you can find a list of diagnostics messages here to google | |
ImmutableList<WorkspaceDiagnostic> diagnostics = workspace.Diagnostics; | |
foreach(var diagnostic in diagnostics) | |
{ | |
Console.WriteLine($"Diagnostics: {diagnostic.ToString()}"); | |
} | |
// Here - no documentIds. At all. And spurious Microsoft.NET.Sdk errors, | |
// until I started using Microsoft.Build.Locator. | |
foreach (var documentId in project.DocumentIds) | |
{ | |
Console.WriteLine(documentId); | |
Document document = solution.GetDocument(documentId); | |
if (document.SupportsSyntaxTree) documents.Add(document); | |
} | |
} | |
return documents; | |
} | |
static void Main(string[] args) | |
{ | |
MSBuildLocator.RegisterDefaults(); | |
var documents = LoadSolution(@"C:\code\Sandbox\CleanAnalysis.sln"); | |
} | |
} | |
} |
Finally - you need to mark the Microsoft.Build.*
assets with ExcludeRuntime=true
tags in your project file - this avoids having MsBuild assemblies copied to your build directory and allows the locator to reference them indirectly:
<Project Sdk="Microsoft.NET.Sdk"> | |
<PropertyGroup> | |
<OutputType>Exe</OutputType> | |
<TargetFramework>netcoreapp3.1</TargetFramework> | |
</PropertyGroup> | |
<ItemGroup> | |
<PackageReference Include="Microsoft.Build" Version="16.6.0" ExcludeAssets="runtime" /> | |
<PackageReference Include="Microsoft.Build.Framework" Version="16.6.0" ExcludeAssets="runtime" /> | |
<PackageReference Include="Microsoft.Build.Locator" Version="1.2.6" /> | |
<PackageReference Include="Microsoft.Build.Tasks.Core" Version="16.6.0" ExcludeAssets="runtime" /> | |
<PackageReference Include="Microsoft.CodeAnalysis.Workspaces.MSBuild" Version="3.7.0-4.final" /> | |
</ItemGroup> | |
</Project> |
and you’re there: congratulations! In a few minutes you’ve achieved what took me a number of grinding, painful hours :)