XSLT Debugger for Visual Studio Code
A powerful Visual Studio Code extension that enables debugging support for XSLT stylesheets. Set breakpoints, step through transformations, inspect variables, and evaluate XPath expressions in real-time using a .NET-based debug adapter.
Table of Contents
Features
- Breakpoint Support: Set breakpoints in XSLT files and step through transformations
- Variable Inspection: Automatically materialises XSLT variables and context nodes inside VS Code’s VARIABLES pane
- XPath Evaluation: Evaluate XPath expressions in the current context
- Inline C# Scripting: Debug XSLT stylesheets with embedded C# code using Roslyn
- Multiple Engines: Support for compiled XSLT engine (XSLT 1.0) and Saxon engine (XSLT 2.0/3.0)
- Cross-Platform: Works on Windows, macOS, and Linux
- Probe Tagging: Instrumented breakpoints and trace messages are tagged with dbg:probe="1"so repeated runs stay idempotent
XSLT Processing Engines
The debugger supports two engines to handle different XSLT versions and use cases:
Compiled Engine (XSLT 1.0)
| Feature | Details | 
| XSLT Version | 1.0 | 
| Special Features | Inline C# scripting via msxsl:script | 
| Debugging Support | Breakpoints via dbg:break()inline instruction; noxsl:messageinjection | 
| Platform Support | Windows, macOS, Linux | 
| Best For | XSLT 1.0 stylesheets, especially those with inline C# code | 
Saxon .NET Engine (XSLT 2.0/3.0)
| Feature | Details | 
| XSLT Version | 2.0 and 3.0 | 
| Implementation | SaxonHE10Net31Api (community IKVM build) | 
| XPath Support | 2.0 and 3.0 | 
| Debugging Support | Breakpoints + variable tracing through dbg:break()andxsl:messageprobes | 
| Platform Support | Windows, macOS, Linux | 
| License | Mozilla Public License 2.0 (free and open source) | 
| Best For | Modern XSLT 2.0/3.0 stylesheets (same approach as Azure Logic Apps Data Mapper) | 
Engine Selection
The debugger can automatically detect the appropriate engine based on your XSLT version:
- XSLT 1.0 + inline C# → Compiled engine (automatic)
- XSLT 2.0/3.0 → Saxon .NET engine (automatic)
- Manual Override → Set "engine": "compiled"or"engine": "saxonnet"in launch.json
⚠️ Current Limitations
The debugger is designed to stay simple and stable, without attempting to fully parse complex XSLT constructs.
To keep it lightweight and predictable:
- Breakpoint and step debugging are currently limited to basic XSLT structures (templates, loops, and expressions). Deep or dynamic template calls are intentionally not instrumented to avoid complex XSLT parsing.
- Inline C# scripts execute as black boxes — stepping into C# code during debugging is not supported.
- XSLT 2.0/3.0 debugging is limited to transformation execution — step-through debugging is under development.
- Variable inspection covers @select-based variables; content-defined variables are skipped for now.
- Trace logging introduces minor runtime overhead (up to ~15% in traceallmode).
- Marketplace release is pending — available today via .vsixlocal install.
These tradeoffs ensure reliable, cross-platform debugging without slowing down transformations or overcomplicating the runtime.
🧩 Note: This is a complementary developer tool intended for debugging and learning — not a production-grade runtime.
Quick Start
For Users
- Install the extension from the VS Code marketplace: - Platform-Specific Extensions: - 
- macOS: Search for "XSLT Debugger Darwin" in VS Code Extensions
- Windows: Search for "XSLT Debugger Windows" in VS Code Extensions
 - Or install from - .vsixfile:
 - # macOS
code --install-extension xsltdebugger-darwin-darwin-arm64-0.5.0.vsix
# Windows
code --install-extension xsltdebugger-windows-win32-x64-0.5.0.vsix
 
- Create a debug configuration in .vscode/launch.json 
- Start debugging by pressing F5 or selecting "Debug XSLT" from the debug menu 
For Developers
- Clone and build the extension: - npm install
npm run compile
dotnet build ./XsltDebugger.DebugAdapter
dotnet test ./XsltDebugger.Tests/XsltDebugger.Tests.csproj -v minimal
 
- Run the Extension Development Host: - 
- Press F5 in VS Code to launch the extension development host
- Select the "XSLT: Launch" configuration to debug a stylesheet
 
- Package and install locally: - Build both platforms at once (recommended): - ./package-all.sh
code --install-extension xsltdebugger-darwin-darwin-arm64-0.5.0.vsix
 - Platform-specific packaging (build individually): - # For macOS only
./package-darwin.sh
code --install-extension xsltdebugger-darwin-darwin-arm64-0.5.0.vsix
# For Windows only
./package-win.sh
code --install-extension xsltdebugger-windows-win32-x64-0.5.0.vsix
 
Usage
Setting Up a Debug Configuration
Create a .vscode/launch.json file in your project workspace:
{
  "version": "0.2.0",
  "configurations": [
    {
      "type": "xslt",
      "request": "launch",
      "name": "Debug XSLT",
      "engine": "compiled",
      "stylesheet": "${workspaceFolder}/ShipmentConf.xslt",
      "xml": "${workspaceFolder}/ShipmentConf.xml",
      "stopOnEntry": false
    }
  ]
}
Configuration Parameters
| Parameter | Type | Required | Description | Example | 
| type | string | ✅ | Must be "xslt" | "xslt" | 
| request | string | ✅ | Must be "launch" | "launch" | 
| name | string | ✅ | Display name in debug menu | "Debug XSLT" | 
| engine | string | ❌ | Engine type ( "compiled"or"saxonnet", default:"compiled") | "saxonnet" | 
| stylesheet | string | ✅ | Path to XSLT file | "${file}"or"${workspaceFolder}/transform.xslt" | 
| xml | string | ✅ | Path to input XML | "${workspaceFolder}/data.xml" | 
| stopOnEntry | boolean | ❌ | Pause at transform start | false | 
| debug | boolean | ❌ | Enable debugging mode (breakpoints and stepping, default: true) | true | 
| logLevel | string | ❌ | Logging verbosity: "none","log","trace", or"traceall"(default:"log") | "log" | 
Variable Substitutions
- ${file}: Currently open file in editor
- ${workspaceFolder}: Root of your workspace
- ${workspaceFolder}/relative/path.xslt: Specific file in workspace
Example Configurations
Debug currently open XSLT file with auto engine selection:
{
  "type": "xslt",
  "request": "launch",
  "name": "Debug Current XSLT",
  "stylesheet": "${file}",
  "xml": "${workspaceFolder}/input.xml",
  "stopOnEntry": false
}
Debug XSLT 2.0/3.0 with Saxon .NET engine:
{
  "type": "xslt",
  "request": "launch",
  "name": "Debug XSLT 2.0/3.0",
  "engine": "saxonnet",
  "stylesheet": "${workspaceFolder}/transform.xslt",
  "xml": "${workspaceFolder}/data.xml",
  "stopOnEntry": false
}
Debug with stop on entry:
{
  "type": "xslt",
  "request": "launch",
  "name": "Debug XSLT (Stop at Start)",
  "engine": "compiled",
  "stylesheet": "${workspaceFolder}/transform.xslt",
  "xml": "${workspaceFolder}/data.xml",
  "stopOnEntry": true
}
Debug with troubleshooting traces:
{
  "type": "xslt",
  "request": "launch",
  "name": "Debug XSLT (trace level)",
  "engine": "compiled",
  "stylesheet": "${workspaceFolder}/transform.xslt",
  "xml": "${workspaceFolder}/data.xml",
  "debug": true,
  "logLevel": "trace"
}
Debug with full XPath value tracking:
{
  "type": "xslt",
  "request": "launch",
  "name": "Debug XSLT (traceall level)",
  "engine": "compiled",
  "stylesheet": "${workspaceFolder}/transform.xslt",
  "xml": "${workspaceFolder}/data.xml",
  "debug": true,
  "logLevel": "traceall"
}
Run without debugging (fastest execution):
{
  "type": "xslt",
  "request": "launch",
  "name": "Run XSLT (no debugging)",
  "engine": "compiled",
  "stylesheet": "${workspaceFolder}/transform.xslt",
  "xml": "${workspaceFolder}/data.xml",
  "debug": false,
  "logLevel": "none"
}
Debugging Features
| Feature | Description | How to Use | 
| Breakpoints | Pause execution at specific XSLT instructions | Click in the gutter next to line numbers | 
| Stepping | Control execution flow | F10 (step over), F11 (step into), Shift+F11 (step out) | 
| Variables | Inspect context nodes, attributes, and XSLT variables | View in the Variables panel during debugging | 
| Watch | Monitor specific XPath expressions | Add expressions to the Watch panel | 
| Debug Console | Evaluate XPath expressions interactively | Type XPath expressions in the Debug Console | 
Variable Inspection Notes
- XSLT 2.0 and 3.0 variables are automatically captured and displayed
Log Levels
The debugger supports four hierarchical logging levels:
| Level | Purpose | Output Includes | Overhead | Best For | 
| none | Silent mode | Errors only | ~0% | Production, performance testing | 
| log (default) | General execution | Transform lifecycle, XSLT version, compilation status, file I/O | <1% | Normal development | 
| trace | Troubleshooting | Everything in log+ breakpoint hits, execution stops, instrumented lines, XPath requests | ~5-10% | Debugging breakpoints, execution flow | 
| traceall | Deep inspection | Everything in trace+ XPath locations, node values/types, expression results, attribute values | ~15-20% | Understanding data flow, complex XPath debugging | 
Common Scenarios
// Normal development
{ "logLevel": "log" }
// Breakpoint not working?
{ "logLevel": "trace" }
// Need to see actual values?
{ "logLevel": "traceall" }
// Maximum performance
{ "debug": false, "logLevel": "none" }
Inline C# Scripting
The debugger supports XSLT stylesheets with embedded C# code using msxsl:script elements:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                xmlns:msxsl="urn:schemas-microsoft-com:xslt"
                xmlns:my="urn:my-scripts">
  <msxsl:script language="C#" implements-prefix="my">
    public string Hello(string name) {
      return "Hello, " + name;
    }
  </msxsl:script>
  <xsl:template match="/">
    <output>
      <xsl:value-of select="my:Hello(/root/name)"/>
    </output>
  </xsl:template>
</xsl:stylesheet>
Requirements
For Users
- Visual Studio Code 1.105.0 or higher
- No additional dependencies (the extension includes the .NET debug adapter)
For Developers
- Visual Studio Code 1.105.0 or higher
- .NET 8.0 SDK (for building the debug adapter)
- Node.js 18 or higher (for building the extension)
Architecture
The workflow is split between a lightweight VS Code entry point and a .NET debug adapter that owns execution and instrumentation.
┌─────────────────────────────┐
│   VS Code Extension (TS)    │
│   src/extension.ts          │
│   • resolves configs        │
│   • launches adapter        │
└───────────────┬─────────────┘
                │ DAP over stdio
┌───────────────▼─────────────┐
│     .NET Debug Adapter       │
│  XsltDebugger.DebugAdapter   │
│ ┌──────────────────────────┐ │
│ │        DapServer         │ │
│ │  maps DAP ⇄ engine APIs  │ │
│ └───────────┬──────────────┘ │
│             │                 │
│    ┌────────▼────────┐        │
│    │XsltEngineManager│        │
│    │ state + logging │        │
│    └────────┬────────┘        │
│             │                 │
│ ┌───────────▼──────────┐      │
│ │  XsltEngineFactory   │      │
│ └──────┬────────┬──────┘      │
│        │        │             │
│  ┌─────▼┐   ┌───▼───────┐     │
│  │Compiled│ │  Saxon    │     │
│  │Engine  │ │  Engine   │     │
│  └──┬────┘ └──┬────────┘     │
│     │         │               │
│  dbg:break via │speedy XPath  │
│  XsltDebugExt  │SaxonDebugExt │
└─────┴──────────┴──────────────┘
- TypeScript extension wires the debug type, resolves ${workspace}paths, and launches the adapter (dotnet XsltDebugger.DebugAdapter.dll).
- DapServer owns the Debug Adapter Protocol: launch,setBreakpoints,variables,evaluate, etc., and forwards them throughXsltEngineManager.
- XsltEngineManager tracks the current engine, breakpoints, stop reasons, and captured variables. It exposes events the adapter translates into DAP notifications.
- Engine factory chooses between:
- XsltCompiledEngine(XslCompiledTransform) for XSLT 1.0 and inline C#. It rewrites the DOM with- <xsl:value-of select="dbg:break(...)">plus tagged- xsl:messageinstructions.
- SaxonEngine(Saxon HE via IKVM) for XSLT 2.0/3.0. It injects- <xsl:sequence select="dbg:break(...) dbg:probe='1'"/>so functions and advanced constructs remain side-effect free.- SaxonDebugExtensionexposes the- dbg:break()function that raises breakpoints back to the adapter.
 
- Tests (XsltDebugger.Tests) run the adapter engines end-to-end with real XSLT samples (guardrails, advanced v2/v3, inline C#) to ensure instrumentation and logging stay consistent.
What's New
See the CHANGELOG for detailed version history.
Latest Release: v0.5.0
Test Infrastructure & Code Quality Improvements
- Centralized test data to TestData/Integration/folder at repository root for better organization
- All test projects now reference common test data location
- Enhanced test coverage with 105 passing integration and unit tests
- Improved ConsoleTest project with unified engine support (both Compiled and Saxon)
Variable Debugging Enhancements
- Enhanced variable instrumentation for both Compiled and Saxon engines
- Improved variable capture and display in VS Code Variables panel
- Better support for XSLT 2.0/3.0 variable debugging with Saxon engine
- Added CompiledMessageHandlerfor enhanced compiled engine debugging
Engine Improvements
- Unified console testing with ProgramUsingEngineType.cssupporting both engines
- Better breakpoint context information and handling
- Enhanced xsl:message support for debugging output
- Improved XSLT 2.0/3.0 features support including accumulators
Developer Experience
- Platform-specific packaging with optimized binary sizes
- Updated build scripts for both macOS (darwin-arm64) and Windows (win32-x64)
- Comprehensive integration tests for both engines
- Better documentation and code organization
Contributing
Contributions are welcome! Here's how to get started:
- Fork the repository
- Create a feature branch (git checkout -b feature/amazing-feature)
- Make your changes
- Add tests for new functionality
- Ensure all tests pass: dotnet test XsltDebugger.Tests/
- Build and test the extension: npm run compile
- Submit a pull request
Development Setup
# Clone the repository
git clone <repository-url>
cd XsltDebugger
# Install dependencies
npm install
dotnet restore
# Build the project
npm run compile
dotnet build XsltDebugger.DebugAdapter/
# Run tests
dotnet test XsltDebugger.Tests/
# Package for testing
./package-all.sh         # Build both macOS and Windows (recommended)
./package-darwin.sh      # macOS only
./package-win.sh         # Windows only
The extension supports platform-specific packaging to reduce file size by including only the necessary binaries for each platform:
Package Scripts
| Script | Platform | Extension Name | Includes | 
| package-all.sh | macOS + Windows | Both platforms | Builds both packages | 
| package-darwin.sh | macOS | xsltdebugger-darwin | osx-arm64 binaries only | 
| package-win.sh | Windows | xsltdebugger-windows | win-x64 binaries only | 
How It Works
Each packaging script:
- Temporarily modifies package.jsonto use a platform-specific extension name
- Removes unnecessary runtime binaries for other platforms
- Packages the extension with --targetflag for the specific platform
- Restores the original package.json
Publishing to Marketplace
# Build both platform packages
./package-all.sh
# Publish each as a separate extension
vsce publish -p YOUR_TOKEN --packagePath xsltdebugger-darwin-darwin-arm64-0.5.0.vsix
vsce publish -p YOUR_TOKEN --packagePath xsltdebugger-windows-win32-x64-0.5.0.vsix
Benefits:
- Smaller download size for users (only includes their platform's binaries)
- Separate marketplace listings prevent package conflicts
- Each platform can be updated independently
### Project Structure
XsltDebugger/
├── src/                           # TypeScript extension source
│   ├── extension.ts               # Main extension entry point
│   └── test/                      # Extension tests
├── XsltDebugger.DebugAdapter/     # C# debug adapter
│   ├── Program.cs                 # Debug adapter entry point
│   ├── DapServer.cs               # DAP protocol implementation
│   ├── XsltCompiledEngine.cs      # XSLT 1.0 engine with inline C#
│   ├── SaxonEngine.cs             # XSLT 2.0/3.0 Saxon engine
│   ├── CompiledMessageHandler.cs  # Message handling for compiled engine
│   ├── SaxonDebugExtension.cs     # Debug extension for Saxon
│   └── XsltEngineManager.cs       # Engine state management
├── XsltDebugger.Tests/            # C# integration & unit tests (105 tests)
│   ├── CompiledEngineIntegrationTests.cs
│   ├── SaxonEngineIntegrationTests.cs
│   └── [other test files]
├── XsltDebugger.ConsoleTest/      # Console test harness
│   └── ProgramUsingEngineType.cs  # Unified engine testing
├── TestData/                      # Centralized test data
│   └── Integration/               # Integration test XSLT & XML files
└── package.json                   # Extension manifest
## License
This project is licensed under the MIT License. See the [LICENSE](https://github.com/imdj360/XsltDebugger/blob/HEAD/LICENSE) file for details.
### Third-Party Licenses
- **SaxonHE10Net31Api**: Mozilla Public License 2.0 (Martin Honnen's community IKVM build)
- Variable Capture via xsl:message: For the Saxon engine,<xsl:message select="('[DBG]', '$var', …)" dbg:probe="1"/>is emitted around safe variables and loops so captured values appear in the VARIABLES view without mutating the result tree.