maintaining-binary-log-compatibility
Installation
SKILL.md
Binary Log Considerations
The binary log (.binlog) is MSBuild's primary diagnostic format and source of truth for build analysis. It captures the complete build event stream with full fidelity. Format changes have strict compatibility requirements.
For general binary log usage, see Binary-Log.md.
Core Principles
- The binlog captures everything. All meaningful build events must be represented. If your change alters build behavior, it must be observable in the binlog.
- Format changes must be backward-compatible. Older versions of MSBuild Structured Log Viewer and other tools must be able to read binlogs produced by newer MSBuild, at minimum gracefully degrading.
- Forward compatibility matters too. Newer viewers should handle binlogs from older MSBuild without crashing, even if some events are unrecognized.
Architecture Overview
Build Engine
→ BuildEventArgs (structured events)
→ BinaryLogger (serializes to .binlog)
→ ProjectImportsCollector (embeds project/targets files)
Replay:
.binlog file
→ BinaryLogReplayEventSource (deserializes)
→ Any ILogger (console, structured log viewer, analyzers)
Key source files:
src/Build/Logging/BinaryLogger/BinaryLogger.cs— the logger itselfsrc/Framework/BuildEventArgs.cs— base class for all build eventssrc/Build/Logging/BinaryLogger/ProjectImportsCollector.cs— captures imported files
Adding New Build Event Types
When adding a new BuildEventArgs subclass:
- Define the new event class inheriting from the appropriate base (
BuildMessageEventArgs,BuildWarningEventArgs, etc.) - Add serialization support — implement
WriteToStreamandCreateFromStreammethods - Increment the binary log version if the new event type changes the format
- Add a new record type constant in the binary logger's record type enum
- Handle the unknown-type case in the replay source — older readers must skip gracefully
Serialization Compatibility Rules
- Never remove fields from existing event args serialization
- New fields must be appended to the end of the serialization stream
- Use version checks when reading — if the binlog version is older, use defaults for new fields
- Nullable fields should serialize a presence flag before the value
// Pattern for backward-compatible field addition
if (logVersion >= newFieldVersion)
{
writer.Write(newField);
}
// Reading with backward compatibility
if (logVersion >= newFieldVersion)
{
newField = reader.ReadString();
}
else
{
newField = defaultValue;
}
Message Importance Levels
Importance controls what appears in console output, but everything goes to binlog regardless of importance.
| Level | Use For | Console Verbosity |
|---|---|---|
High |
Critical user-facing information | Minimal and above |
Normal |
Standard build progress | Normal and above |
Low |
Detailed diagnostic information | Detailed and above |
Diagnostic |
Internal debugging | Diagnostic only |
Rules
- Default to
Normalfor user-relevant information - Use
Lowfor information useful when debugging but noisy in normal builds Highis reserved for important warnings/status — use sparingly- Never skip logging because "it's too verbose" — log at
Lowinstead
ProjectImportsCollector
The ProjectImportsCollector embeds all imported .props, .targets, and project files into the binlog. This enables the "preprocessed view" in log viewers.
Considerations
MSBUILDLOGIMPORTS=1(or the/blswitch) enables import collection- Imported file content is captured at evaluation time — reflects the actual content used
- Large import chains increase binlog size — this is acceptable for diagnostic completeness
- Sensitive content in imported files will be embedded — document this for users
Changes That Affect Binlog Content
When modifying MSBuild behavior, verify binlog impact:
| Change Type | Binlog Consideration |
|---|---|
| New property set during evaluation | Appears in PropertyInitialValue or PropertyReassignment events |
| New target added | Produces TargetStarted/TargetFinished events |
| Changed task behavior | Task output items/properties captured in TaskFinished |
| New warning/error | Captured as BuildWarningEventArgs/BuildErrorEventArgs |
| Modified import chain | Changes which files ProjectImportsCollector captures |
Testing Binlog Changes
- Round-trip test: Write a binlog, replay it, verify all events are reconstructed
- Version compatibility test: Verify older replay sources handle new events gracefully
- Content verification: Assert specific events appear in the binlog for behavioral changes
- Size regression: Monitor binlog size for unexpectedly large increases
- Use
BinaryLogReplayEventSourcein tests to verify binlog content programmatically
Checklist
- New event types have
WriteToStream/CreateFromStreamimplementations - Binary log format version incremented if format changed
- Backward compatibility: older readers skip/degrade gracefully
- Forward compatibility: newer readers handle old format
- Importance levels set correctly for new messages
- Behavioral changes produce observable binlog events
- Round-trip test passes (serialize → deserialize → verify)