Guide to write test cases in the parser’s test suite and how to debug failures.
The general approach to fixing bugs in the parser is to first write an automated test case that reproduces the test case in isolation. This allows you to invoke the parser with minimal dependencies and allows you to set breakpoints inside of it.
Once you’ve written a test case (see below), set a breakpoint in Parser.parseSourceFile
and navigate the debugger to the place where the parser behaves unexpectedly. While the debugger is stopped at an instance function in doc:SwiftParser/Parser, po self.currentToken
can show you the next token that will be parsed. po self.lexemes
can show the next tokens that will be parsed.
- Add a new test case in
SwiftParserTest
that looks like the followingassertParse( """ <#your code that does not round trip#> """ )
- Run the test case, read the error message to figure out which part of the source file does not round-trip
Diagnostics are produced when the parsed syntax tree contains missing or unexpected nodes. The test case you should start with is identical to the one described in Round-Trip Failure. You want to navigate the debugger to the place that incorrectly produces a missing or unexpected node.
- Add a test case in
SwiftParserTest
that looks like the followingassertParse( """ <#your code that produces an invalid syntax tree#> """, substructure: <#create a syntax node that you expect the tree to have#> )
- Run the test case and navigate the debugger to the place that produced the invalid syntax node.
Unhelpful diagnostics can result from two reasons:
- The parser does a bad job at recovering from invalid source code and produces a syntax tree that does not match what the developer meant to write
- The parser recovers reasonably well from the parse failure but complains about missing and unexpected text where a more contextual error would be more helpful.
To distinguish these cases run the following command and look at the dumped syntax tree. Use your own judgment to decide whether this models the intended meaning of the source code reasonably well.
swift-parser-cli print-tree /path/to/file/with/unhelpful/diagnostic.swift
Fixing the first case where the parser does not recover according to the user’s intent is similar to Parse of Valid Source Code Produced an Invalid Syntax Tree. See doc:SwiftParser/ParserRecovery for documentation how parser recovery works and determine how to recover better from the invalid source code.
To add a new, more contextual diagnostic, perform the following steps.
-
Add a test case in
SwiftParserTest
that looks like the followingassertParse( """ <#your code that produced the unhelpful diagnostic#> """, diagnostics: [ DiagnosticSpec(message: "<#expected diagnostic message#>") ] )
-
Mark the location at which you expect the diagnostic to be produced with
1️⃣
. If you expect multiple diagnostics to be produced, you can use multiple of these markers with different names and use these markers by passing alocationMarker
toDiagnosticSpec
. The list of valid marker emojis can be found inCharacter.isMarkerEmoji
. -
Determine which node encompasses all information that is necessary to produce the improved diagnostic – for example
FunctionSignatureSyntax
contains all information to diagnose if thethrows
keyword was written after the->
instead of in front of it. -
If the diagnostic message you want to emit does not exist yet, add a case to doc:SwiftParser/DiagnosticKind for the new diagnostic.
-
If the function does not already exist, write a new visit method on doc:SwiftParser/ParseDiagnosticsGenerator.
-
In that visitation method, detect the pattern for which the improved diagnostic should be emitted and emit it using
diagnostics.append
. -
Mark the missing or garbage nodes that are covered by the new diagnostic as handled by adding their
SyntaxIdentifier
s tohandledNodes
. -
If the diagnostic produces Fix-Its assert that they are generated by adding the Fix-It's message to the
DiagnosticSpec
with thefixIt
parameter and asserting that applying the Fix-Its produces the correct source code by adding thefixedSource
parameter toassertParse
.
💡 Tip: To make typing the marker emojis more convienient. you can set up code snippets in Xcode. To do this, perform the following steps:
- Type the marker in any Xcode window or find it in some test case
- Select the emoji, right-click it and select “Create Code Snippet…”
- Replace “My Code Snippet” by the emoji marker character and in for “Availability” select “Comment or String”
- Now you can invoke code completion in string literals and get presented with a list of marker emojis