forked from dotnet/fsharp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
SyntaxTreeTests.fs
204 lines (178 loc) · 8.65 KB
/
SyntaxTreeTests.fs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
module FSharp.Compiler.Service.Tests.SyntaxTreeTests
open System.IO
open FSharp.Compiler.CodeAnalysis
open FSharp.Compiler.Diagnostics
open FSharp.Compiler.Service.Tests.Common
open FSharp.Compiler.Syntax
open FSharp.Compiler.Text
open FSharp.Test
open Xunit
let testCasesDir = __SOURCE_DIRECTORY__ ++ ".." ++ "service" ++ "data" ++ "SyntaxTree"
let allTestCases =
Directory.EnumerateFiles(testCasesDir, "*.fs?", SearchOption.AllDirectories)
|> Seq.map (fun f ->
let fileInfo = FileInfo(f)
let fileName = Path.Combine(fileInfo.Directory.Name, fileInfo.Name)
[| fileName :> obj |])
|> Seq.toArray
[<Literal>]
let RootDirectory = @"/root"
/// <summary>
/// Every time `__SOURCE_DIRECTORY__` was used in the code, the ast will contain an invalid value and range for it.
/// This should be cleaned up when the test runs during CI/CD.
/// </summary>
/// <remarks>
/// This function is incomplete and does not clean up the entire ParsedInput.
/// A shortcut was made to only support the existing use-cases.
/// </remarks>
let private sanitizeAST (sourceDirectoryValue: string) (ast: ParsedInput) : ParsedInput =
let isZero (m: range) =
m.StartLine = 0 && m.StartColumn = 0 && m.EndLine = 0 && m.EndColumn = 0
// __SOURCE_DIRECTORY__ will contain the evaluated value, so we want to replace it with a stable value instead.
let mapParsedHashDirective (ParsedHashDirective(ident, args, _) as phd) =
match args with
| [ ParsedHashDirectiveArgument.SourceIdentifier("__SOURCE_DIRECTORY__", _, mSourceDirectory) ] ->
let mZero =
Range.mkRange mSourceDirectory.FileName (Position.mkPos 0 0) (Position.mkPos 0 0)
ParsedHashDirective(
ident,
[ ParsedHashDirectiveArgument.SourceIdentifier("__SOURCE_DIRECTORY__", sourceDirectoryValue, mZero) ],
mZero
)
| _ -> phd
let (|SourceDirectoryConstant|_|) (constant: SynConst) =
match constant with
| SynConst.SourceIdentifier("__SOURCE_DIRECTORY__", _, mSourceDirectory) ->
let mZero =
Range.mkRange mSourceDirectory.FileName (Position.mkPos 0 0) (Position.mkPos 0 0)
Some(SynConst.SourceIdentifier("__SOURCE_DIRECTORY__", sourceDirectoryValue, mZero), mZero)
| _ -> None
let (|SourceDirectoryConstantExpr|_|) (expr: SynExpr) =
match expr with
| SynExpr.Const(SourceDirectoryConstant(constant, mZero), _) -> Some(SynExpr.Const(constant, mZero))
| _ -> None
let rec mapSynModuleDecl (mdl: SynModuleDecl) =
match mdl with
| SynModuleDecl.HashDirective(ParsedHashDirective(range = mZero) as hd, m) ->
let hd = mapParsedHashDirective hd
// Only update the range of SynModuleSigDecl.HashDirective if the value was updated.
let m = if isZero mZero then mZero else m
SynModuleDecl.HashDirective(hd, m)
| SynModuleDecl.NestedModule(moduleInfo, isRecursive, decls, isContinuing, range, trivia) ->
SynModuleDecl.NestedModule(moduleInfo, isRecursive, List.map mapSynModuleDecl decls, isContinuing, range, trivia)
| SynModuleDecl.Expr(SourceDirectoryConstantExpr(expr), _) -> SynModuleDecl.Expr(expr, expr.Range)
| _ -> mdl
let mapSynModuleOrNamespace (SynModuleOrNamespace(longId, isRecursive, kind, decls, xmlDoc, attribs, ao, range, trivia)) =
SynModuleOrNamespace(longId, isRecursive, kind, List.map mapSynModuleDecl decls, xmlDoc, attribs, ao, range, trivia)
let rec mapSynModuleDeclSig (msdl: SynModuleSigDecl) =
match msdl with
| SynModuleSigDecl.HashDirective(ParsedHashDirective(range = mZero) as hd, m) ->
let hd = mapParsedHashDirective hd
// Only update the range of SynModuleSigDecl.HashDirective if the value was updated.
let m = if isZero mZero then mZero else m
SynModuleSigDecl.HashDirective(hd, m)
| SynModuleSigDecl.NestedModule(moduleInfo, isRecursive, decls, range, trivia) ->
SynModuleSigDecl.NestedModule(moduleInfo, isRecursive, List.map mapSynModuleDeclSig decls, range, trivia)
| _ -> msdl
let mapSynModuleOrNamespaceSig (SynModuleOrNamespaceSig(longId, isRecursive, kind, decls, xmlDoc, attribs, ao, range, trivia)) =
SynModuleOrNamespaceSig(longId, isRecursive, kind, List.map mapSynModuleDeclSig decls, xmlDoc, attribs, ao, range, trivia)
match ast with
| ParsedInput.ImplFile(ParsedImplFileInput(fileName,
isScript,
qualifiedNameOfFile,
scopedPragmas,
hashDirectives,
contents,
flags,
trivia,
identifiers)) ->
ParsedImplFileInput(
fileName,
isScript,
qualifiedNameOfFile,
scopedPragmas,
List.map mapParsedHashDirective hashDirectives,
List.map mapSynModuleOrNamespace contents,
flags,
trivia,
identifiers
)
|> ParsedInput.ImplFile
| ParsedInput.SigFile(ParsedSigFileInput(fileName, qualifiedNameOfFile, scopedPragmas, hashDirectives, contents, trivia, identifiers)) ->
ParsedSigFileInput(
fileName,
qualifiedNameOfFile,
scopedPragmas,
List.map mapParsedHashDirective hashDirectives,
List.map mapSynModuleOrNamespaceSig contents,
trivia,
identifiers
)
|> ParsedInput.SigFile
let parseSourceCode (name: string, code: string) =
let location = Path.Combine(RootDirectory, name).Replace("\\", "/")
let parseResults =
checker.ParseFile(
location,
SourceText.ofString code,
{ FSharpParsingOptions.Default with
SourceFiles = [| location |]
IsExe = true
LangVersionText = "preview" }
)
|> Async.RunImmediate
let tree = parseResults.ParseTree
let sourceDirectoryValue = $"{RootDirectory}/{FileInfo(location).Directory.Name}"
sanitizeAST sourceDirectoryValue tree, parseResults.Diagnostics
/// Asserts the parsed untyped tree matches the expected baseline.
///
/// To update a baseline:
/// CMD: set TEST_UPDATE_BSL=1 & dotnet test --filter "ParseFile"
/// PowerShell: $env:TEST_UPDATE_BSL = "1" ; dotnet test --filter "ParseFile"
/// Linux/macOS: export TEST_UPDATE_BSL=1 & dotnet test --filter "ParseFile"
///
/// Assuming your current directory is tests/FSharp.Compiler.Service.Tests
[<Theory>]
[<MemberData(nameof allTestCases)>]
let ParseFile fileName =
let fullPath = Path.Combine(testCasesDir, fileName)
let contents = File.ReadAllText fullPath
let ast, diagnostics = parseSourceCode (fileName, contents)
let normalize (s: string) = s.Replace("\r", "")
let actual =
if Array.isEmpty diagnostics then
$"%A{ast}"
else
let diagnostics =
diagnostics
|> Array.map (fun d ->
let severity =
match d.Severity with
| FSharpDiagnosticSeverity.Warning -> "warning"
| FSharpDiagnosticSeverity.Error -> "error"
| FSharpDiagnosticSeverity.Info -> "info"
| FSharpDiagnosticSeverity.Hidden -> "hidden"
$"(%d{d.StartLine},%d{d.StartColumn})-(%d{d.EndLine},%d{d.EndColumn}) %s{d.Subcategory} %s{severity} %s{d.Message}"
)
|> String.concat "\n"
$"%A{ast}\n\n%s{diagnostics}"
|> normalize
|> sprintf "%s\n"
let bslPath = $"{fullPath}.bsl"
let actualPath = $"{fullPath}.actual"
let expected =
if File.Exists bslPath then
File.ReadAllText bslPath |> normalize
else
"No baseline was found"
let equals = expected = actual
let testUpdateBSLEnv = System.Environment.GetEnvironmentVariable("TEST_UPDATE_BSL")
if not (isNull testUpdateBSLEnv) && testUpdateBSLEnv.Trim() = "1" && not equals then
File.WriteAllText(bslPath, actual)
elif not equals then
File.WriteAllText(actualPath, actual)
else
File.Delete(actualPath)
Assert.Equal(expected, actual)
// Run type checker to assert that it doesn't fail with the tree produced by the parser
CompilerAssert.ParseAndTypeCheck([|"--langversion:preview"|], fileName, contents) |> ignore