-
Notifications
You must be signed in to change notification settings - Fork 1.1k
/
Copy pathParseResult.scala
236 lines (203 loc) · 7.2 KB
/
ParseResult.scala
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
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
package dotty.tools
package repl
import dotc.CompilationUnit
import dotc.ast.untpd
import dotc.core.Contexts.*
import dotc.core.StdNames.str
import dotc.parsing.Parsers.Parser
import dotc.parsing.Tokens
import dotc.reporting.{Diagnostic, StoreReporter}
import dotc.util.SourceFile
import scala.annotation.internal.sharable
/** A parsing result from string input */
sealed trait ParseResult
/** An error free parsing resulting in a list of untyped trees */
case class Parsed(source: SourceFile, trees: List[untpd.Tree], reporter: StoreReporter) extends ParseResult
/** A parsing result containing syntax `errors` */
case class SyntaxErrors(sourceCode: String,
errors: List[Diagnostic],
trees: List[untpd.Tree]) extends ParseResult
/** Parsed result is simply a newline */
case object Newline extends ParseResult
/** `ctrl-c` obtained from input string */
case object SigKill extends ParseResult
/** A command is on the format:
*
* ```none
* :commandName <optional arguments...>
* ```
* The `Command` trait denotes these commands
*/
sealed trait Command extends ParseResult
/** An unknown command that will not be handled by the REPL */
case class UnknownCommand(cmd: String) extends Command
/** An ambiguous prefix that matches multiple commands */
case class AmbiguousCommand(cmd: String, matchingCommands: List[String]) extends Command
/** `:load <path>` interprets a scala file as if entered line-by-line into
* the REPL
*/
case class Load(path: String) extends Command
object Load {
val command: String = ":load"
}
/** `:require` is a deprecated alias for :jar`
*/
case class Require(path: String) extends Command
object Require {
val command: String = ":require"
}
/** `:jar <path>` adds a jar to the classpath
*/
case class JarCmd(path: String) extends Command
object JarCmd {
val command: String = ":jar"
}
/** `:kind <type>` display the kind of a type. see also :help kind
*/
case class KindOf(expr: String) extends Command
object KindOf {
val command: String = ":kind"
}
/** To find out the type of an expression you may simply do:
*
* ```
* scala> :type (1 * 54).toString
* String
* ```
*/
case class TypeOf(expr: String) extends Command
object TypeOf {
val command: String = ":type"
}
/**
* A command that is used to display the documentation associated with
* the given expression.
*/
case class DocOf(expr: String) extends Command
object DocOf {
val command: String = ":doc"
}
/** `:imports` lists the imports that have been explicitly imported during the
* session
*/
case object Imports extends Command {
val command: String = ":imports"
}
case class Settings(arg: String) extends Command
object Settings {
val command: String = ":settings"
}
/** Reset the session to the initial state from when the repl program was
* started
*/
case class Reset(arg: String) extends Command
object Reset {
val command: String = ":reset"
}
/** `:sh <command line>` run a shell command (result is implicitly => List[String]) */
case class Sh(expr: String) extends Command
object Sh {
val command: String = ":sh"
}
/** Toggle automatic printing of results */
case object Silent extends Command:
val command: String = ":silent"
/** `:quit` exits the repl */
case object Quit extends Command {
val command: String = ":quit"
val alias: String = ":exit"
}
/** `:help` shows the different commands implemented by the Dotty repl */
case object Help extends Command {
val command: String = ":help"
val text: String =
"""The REPL has several commands available:
|
|:help print this summary
|:load <path> interpret lines in a file
|:quit exit the interpreter
|:type <expression> evaluate the type of the given expression
|:doc <expression> print the documentation for the given expression
|:imports show import history
|:reset [options] reset the repl to its initial state, forgetting all session entries
|:settings <options> update compiler options, if possible
|:silent disable/enable automatic printing of results
""".stripMargin
}
object ParseResult {
@sharable private val CommandExtract = """(:[\S]+)\s*(.*)""".r
private def parseStats(using Context): List[untpd.Tree] = {
val parser = new Parser(ctx.source)
val stats = parser.blockStatSeq(outermost = true)
parser.accept(Tokens.EOF)
stats
}
private[repl] val commands: List[(String, String => ParseResult)] = List(
Quit.command -> (_ => Quit),
Quit.alias -> (_ => Quit),
Help.command -> (_ => Help),
Reset.command -> (arg => Reset(arg)),
Imports.command -> (_ => Imports),
JarCmd.command -> (arg => JarCmd(arg)),
KindOf.command -> (arg => KindOf(arg)),
Load.command -> (arg => Load(arg)),
Require.command -> (arg => Require(arg)),
TypeOf.command -> (arg => TypeOf(arg)),
DocOf.command -> (arg => DocOf(arg)),
Settings.command -> (arg => Settings(arg)),
Sh.command -> (arg => Sh(arg)),
Silent.command -> (_ => Silent),
)
def apply(source: SourceFile)(using state: State): ParseResult = {
val sourceCode = source.content().mkString
sourceCode match {
case "" => Newline
case CommandExtract(cmd, arg) => {
val matchingCommands = commands.filter((command, _) => command.startsWith(cmd))
matchingCommands match {
case Nil => UnknownCommand(cmd)
case (_, f) :: Nil => f(arg)
case multiple => AmbiguousCommand(cmd, multiple.map(_._1))
}
}
case _ =>
inContext(state.context) {
val reporter = newStoreReporter
val stats = parseStats(using state.context.fresh.setReporter(reporter).withSource(source))
if (reporter.hasErrors)
SyntaxErrors(
sourceCode,
reporter.removeBufferedMessages,
stats)
else
Parsed(source, stats, reporter)
}
}
}
def apply(sourceCode: String)(using state: State): ParseResult =
maybeIncomplete(sourceCode, maybeIncomplete = true)
def complete(sourceCode: String)(using state: State): ParseResult =
maybeIncomplete(sourceCode, maybeIncomplete = false)
private def maybeIncomplete(sourceCode: String, maybeIncomplete: Boolean)(using state: State): ParseResult =
apply:
SourceFile.virtual(str.REPL_SESSION_LINE + (state.objectIndex + 1), sourceCode, maybeIncomplete)
/** Check if the input is incomplete.
*
* This can be used in order to check if a newline can be inserted without
* having to evaluate the expression.
*/
def isIncomplete(sourceCode: String)(using Context): Boolean =
sourceCode match
case CommandExtract(_) | "" => false
case _ =>
val reporter = newStoreReporter
val source = SourceFile.virtual("<incomplete-handler>", sourceCode, maybeIncomplete = true)
val unit = CompilationUnit(source, mustExist = false)
val localCtx = ctx.fresh
.setCompilationUnit(unit)
.setReporter(reporter)
var needsMore = false
reporter.withIncompleteHandler((_, _) => needsMore = true):
parseStats(using localCtx)
!reporter.hasErrors && needsMore
}