This is the repository for creating and consuming compiler log files. These are files created from a MSBuild binary log that contain information necessary to recreate all of the Compilation instances from that build.
The compiler log files are self contained. They must be created on the same machine where the binary log was created but after creation they can be freely copied between machines. That enables a number of scenarios:
- GitHub pipelines can cleanly separate build and build analysis into different legs. The analysis can be done on a separate machine entirely independent of where the build happens.
- Allows for easier customer investigations by the C# / VB compiler teams. Instead of trying to re-create a customer build environment, customers can provide a compiler log file that developers can easily open with a call to the API.
This global tool can be installed via
dotnet tool install --global complog
From there the following commands are available:
create
: create a complog file from an existing binary logreplay
: replay the builds from the complogexport
: export complete compilations to diskref
: export references for a compilation to diskrsp
: generate rsp files for compilation eventsprint
: print the summary of a complog on the command line
A compiler log file contains all of the information necessary to recreate a Compilation
. That includes all source, resources, references, strong name keys, etc .... That will be visible to anyone you provide a compiler log to.
There are a number of ways to create a compiler log. The first step is to install the complog
global tool as that will be used to create the compiler log.
> dotnet tool install -g complog
The easiest is to create it off of a binary log file from a previous build.
> msbuild -bl MySolution.sln
> complog create msbuild.binlog
By default this will include every project in the binary log. If there are a lot of projects this can produce a large compiler log. You can use the -p
option to limit the compiler log to a specific set of projects.
> complog create msbuild.binlog -p MyProject.csproj
For solutions or projects that can be built with dotnet build
a compiler log can be created by just running create
against the solution or project file directly.
> complog create MyProject.csproj
When trying to get a compiler log from a build that occurs in a GitHub action you can use the complog-action
action to simplify creating and uploading the compiler log.
- name: Build .NET app
run: dotnet build -bl
- name: Create and upload the compiler log
uses: jaredpar/compilerlog-action@v1
with:
binlog: msbuild.binlog
To re-run all of the compilations in a compiler log use the replay
command
> complog replay build.complog
Microsoft.VisualStudio.IntegrationTest.IntegrationService.csproj (net472) ...Success
Roslyn.Test.Performance.Utilities.csproj (net472) ...Success
Microsoft.CodeAnalysis.XunitHook.csproj (net472) ...Success
Passing the -export
argument will cause all failed compilations to be exported to the local disk for easy analysis.
To debug a compilation in Visual Studio first export it to disk:
> complog export build.complog
That will write out all the artifacts necessary to run a command line build to disk. Use the --project
option to limit the output to specific projects. For each project it will generate a build.rsp command that uses the exported arguments. It will also generate several build*.cmd
files. Those will execute dotnet exec csc.dll @build.rsp
on the build for every SDK installed on the machine.
The next step is to setup csc / vbc to use the build.rsp file for debugging. Open the debug settings for csc / vbc and set them to have the argument @build.rsp
and make the working directory the location of that file.
Then launch csc / vbc and it will debug that project.
The samples below for creating Roslyn objects work for both compiler and binary logs. For a binary log though it must be done on the same machine where the build occurred.
Log files can be used to recreate a Compilation
instance. This is done by calling the Compilation.Create
method and passing in the path to the compiler log file.
using var reader = CompilerCallReaderUtil.Create(logFilePath);
foreach (var compilationData in reader.ReadAllCompilationData())
{
var compilation = compilationData.GetCompilationAfterGenerators();
// ...
}
The SolutionReader
type can be used to create a SolutionInfo
instance from the log file:
var reader = SolutionReader.Create(logFilePath);
var workspace = new AdhocWorkspace();
var solution = workspace.AddSolution(reader.ReadSolutionInfo());