summaryrefslogtreecommitdiff
path: root/src/compiler/scala/tools/nsc/interpreter/InteractiveReader.scala
blob: b3957e1062cd6fd5bc9f2fcc033874e4c45e299a (plain) (blame)
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
/* NSC -- new Scala compiler
 * Copyright 2005-2010 LAMP/EPFL
 * @author Stepan Koltsov
 */
// $Id$

package scala.tools.nsc
package interpreter
import scala.util.control.Exception._

/** Reads lines from an input stream */
trait InteractiveReader {
  import InteractiveReader._
  import java.io.IOException

  protected def readOneLine(prompt: String): String
  val interactive: Boolean

  def readLine(prompt: String): String = {
    def handler: Catcher[String] = {
      case e: IOException if restartSystemCall(e) => readLine(prompt)
    }
    catching(handler) { readOneLine(prompt) }
  }

  // override if history is available
  def history: Option[History] = None
  def historyList = history map (_.asList) getOrElse Nil

  // override if completion is available
  def completion: Option[Completion] = None

  // hack necessary for OSX jvm suspension because read calls are not restarted after SIGTSTP
  private def restartSystemCall(e: Exception): Boolean =
    Properties.isMac && (e.getMessage == msgEINTR)
}


object InteractiveReader {
  val msgEINTR = "Interrupted system call"
  private val exes = List(classOf[Exception], classOf[NoClassDefFoundError])

  def createDefault(): InteractiveReader = createDefault(null)

  /** Create an interactive reader.  Uses <code>JLineReader</code> if the
   *  library is available, but otherwise uses a <code>SimpleReader</code>.
   */
  def createDefault(interpreter: Interpreter): InteractiveReader =
    try new JLineReader(interpreter)
    catch {
      case e @ (_: Exception | _: NoClassDefFoundError) =>
        // println("Failed to create JLineReader(%s): %s".format(interpreter, e))
        new SimpleReader
    }
}