src/Pure/System/progress.scala
author wenzelm
Mon, 06 Mar 2023 12:08:33 +0100
changeset 77538 fcda9a009213
parent 77524 a3dda42cd110
child 78155 54d6b2f75806
permissions -rw-r--r--
more database content: formal end_build;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
61276
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     1
/*  Title:      Pure/System/progress.scala
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     3
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     4
Progress context for system processes.
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     5
*/
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     6
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     7
package isabelle
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     8
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
     9
73897
0ddb5de0506e clarified signature: prefer Java interfaces;
wenzelm
parents: 73367
diff changeset
    10
import java.util.{Map => JMap}
64201
wenzelm
parents: 64117
diff changeset
    11
import java.io.{File => JFile}
wenzelm
parents: 64117
diff changeset
    12
wenzelm
parents: 64117
diff changeset
    13
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 73897
diff changeset
    14
object Progress {
77502
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    15
  object Kind extends Enumeration { val writeln, warning, error_message = Value }
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    16
  sealed case class Message(kind: Kind.Value, text: String, verbose: Boolean = false) {
77502
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    17
    def output_text: String =
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    18
      kind match {
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    19
        case Kind.writeln => Output.writeln_text(text)
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    20
        case Kind.warning => Output.warning_text(text)
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    21
        case Kind.error_message => Output.error_message_text(text)
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    22
      }
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    23
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    24
    override def toString: String = output_text
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    25
  }
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    26
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 73897
diff changeset
    27
  sealed case class Theory(theory: String, session: String = "", percentage: Option[Int] = None) {
77502
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    28
    def message: Message =
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    29
      Message(Kind.writeln, print_session + print_theory + print_percentage, verbose = true)
68987
e86d6e869b96 tuned signature;
wenzelm
parents: 68957
diff changeset
    30
77504
wenzelm
parents: 77503
diff changeset
    31
    def print_session: String = if_proper(session, session + ": ")
68987
e86d6e869b96 tuned signature;
wenzelm
parents: 68957
diff changeset
    32
    def print_theory: String = "theory " + theory
e86d6e869b96 tuned signature;
wenzelm
parents: 68957
diff changeset
    33
    def print_percentage: String =
e86d6e869b96 tuned signature;
wenzelm
parents: 68957
diff changeset
    34
      percentage match { case None => "" case Some(p) => " " + p + "%" }
68957
eef4e983fd9d clarified theory progress;
wenzelm
parents: 68951
diff changeset
    35
  }
68410
4e27f5c361d2 clarified signature: more uniform theory_message (see also d7920eb7de54);
wenzelm
parents: 68330
diff changeset
    36
}
4e27f5c361d2 clarified signature: more uniform theory_message (see also d7920eb7de54);
wenzelm
parents: 68330
diff changeset
    37
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 73897
diff changeset
    38
class Progress {
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    39
  def verbose: Boolean = false
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    40
  final def do_output(message: Progress.Message): Boolean = !message.verbose || verbose
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    41
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    42
  def output(message: Progress.Message) = {}
77502
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    43
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    44
  final def echo(msg: String, verbose: Boolean = false): Unit =
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    45
    output(Progress.Message(Progress.Kind.writeln, msg, verbose = verbose))
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    46
  final def echo_warning(msg: String, verbose: Boolean = false): Unit =
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    47
    output(Progress.Message(Progress.Kind.warning, msg, verbose = verbose))
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    48
  final def echo_error_message(msg: String, verbose: Boolean = false): Unit =
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    49
    output(Progress.Message(Progress.Kind.error_message, msg, verbose = verbose))
77502
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
    50
77506
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    51
  final def echo_if(cond: Boolean, msg: String): Unit = if (cond) echo(msg)
77499
8fc94efa954a clarified signature: more uniform Progress.verbose, avoid adhoc "override def theory()";
wenzelm
parents: 77498
diff changeset
    52
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    53
  def theory(theory: Progress.Theory): Unit = output(theory.message)
73340
0ffcad1f6130 tuned --- fewer warnings;
wenzelm
parents: 72699
diff changeset
    54
  def nodes_status(nodes_status: Document_Status.Nodes_Status): Unit = {}
64909
8007f10195af tuned signature;
wenzelm
parents: 64201
diff changeset
    55
77506
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    56
  final def timeit[A](
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    57
    body: => A,
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    58
    message: Exn.Result[A] => String = null,
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    59
    enabled: Boolean = true
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    60
  ): A = Timing.timeit(body, message = message, enabled = enabled, output = echo(_))
65921
5b42937d3b2d more operations;
wenzelm
parents: 65910
diff changeset
    61
77524
a3dda42cd110 more robust: disallow override;
wenzelm
parents: 77509
diff changeset
    62
  @volatile private var is_stopped = false
73367
77ef8bef0593 clarified signature --- fewer warnings;
wenzelm
parents: 73340
diff changeset
    63
  def stop(): Unit = { is_stopped = true }
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 73897
diff changeset
    64
  def stopped: Boolean = {
73367
77ef8bef0593 clarified signature --- fewer warnings;
wenzelm
parents: 73340
diff changeset
    65
    if (Thread.interrupted()) is_stopped = true
71726
a5fda30edae2 clarified signature: more uniform treatment of stopped/interrupted state;
wenzelm
parents: 71601
diff changeset
    66
    is_stopped
a5fda30edae2 clarified signature: more uniform treatment of stopped/interrupted state;
wenzelm
parents: 71601
diff changeset
    67
  }
a5fda30edae2 clarified signature: more uniform treatment of stopped/interrupted state;
wenzelm
parents: 71601
diff changeset
    68
77506
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    69
  final def interrupt_handler[A](e: => A): A = POSIX_Interrupt.handler { stop() } { e }
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    70
  final def expose_interrupt(): Unit = if (stopped) throw Exn.Interrupt()
61276
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
    71
  override def toString: String = if (stopped) "Progress(stopped)" else "Progress"
64201
wenzelm
parents: 64117
diff changeset
    72
77506
a8175b950173 more robust signature: avoid totally adhoc overriding (see also Build_Process.progress vs. build_progress);
wenzelm
parents: 77504
diff changeset
    73
  final def bash(script: String,
64201
wenzelm
parents: 64117
diff changeset
    74
    cwd: JFile = null,
73897
0ddb5de0506e clarified signature: prefer Java interfaces;
wenzelm
parents: 73367
diff changeset
    75
    env: JMap[String, String] = Isabelle_System.settings(),
64201
wenzelm
parents: 64117
diff changeset
    76
    redirect: Boolean = false,
65930
9a28fc03c3fe tuned signature;
wenzelm
parents: 65921
diff changeset
    77
    echo: Boolean = false,
72599
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    78
    watchdog: Time = Time.zero,
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 73897
diff changeset
    79
    strict: Boolean = true
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 73897
diff changeset
    80
  ): Process_Result = {
72599
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    81
    val result =
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    82
      Isabelle_System.bash(script, cwd = cwd, env = env, redirect = redirect,
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    83
        progress_stdout = echo_if(echo, _),
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    84
        progress_stderr = echo_if(echo, _),
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    85
        watchdog = if (watchdog.is_zero) None else Some((watchdog, _ => stopped)),
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    86
        strict = strict)
76550282267f more interrupts, notably for running latex;
wenzelm
parents: 72574
diff changeset
    87
    if (strict && stopped) throw Exn.Interrupt() else result
64201
wenzelm
parents: 64117
diff changeset
    88
  }
61276
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
    89
}
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
    90
77499
8fc94efa954a clarified signature: more uniform Progress.verbose, avoid adhoc "override def theory()";
wenzelm
parents: 77498
diff changeset
    91
class Console_Progress(override val verbose: Boolean = false, stderr: Boolean = false)
8fc94efa954a clarified signature: more uniform Progress.verbose, avoid adhoc "override def theory()";
wenzelm
parents: 77498
diff changeset
    92
extends Progress {
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    93
  override def output(message: Progress.Message): Unit =
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    94
    if (do_output(message)) {
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    95
      Output.output(message.output_text, stdout = !stderr, include_empty = true)
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
    96
    }
77503
daf632e9ce7e tuned messages;
wenzelm
parents: 77502
diff changeset
    97
daf632e9ce7e tuned messages;
wenzelm
parents: 77502
diff changeset
    98
  override def toString: String = super.toString + ": console"
61276
8a4bd05c1735 clarified modules;
wenzelm
parents:
diff changeset
    99
}
65888
29a31cf0b4bc more Progress variations;
wenzelm
parents: 65828
diff changeset
   100
77507
8dfe2fbc98e7 tuned whitespace;
wenzelm
parents: 77506
diff changeset
   101
class File_Progress(path: Path, override val verbose: Boolean = false)
8dfe2fbc98e7 tuned whitespace;
wenzelm
parents: 77506
diff changeset
   102
extends Progress {
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
   103
  override def output(message: Progress.Message): Unit =
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
   104
    if (do_output(message)) File.append(path, message.output_text + "\n")
65888
29a31cf0b4bc more Progress variations;
wenzelm
parents: 65828
diff changeset
   105
77503
daf632e9ce7e tuned messages;
wenzelm
parents: 77502
diff changeset
   106
  override def toString: String = super.toString + ": " + path.toString
65888
29a31cf0b4bc more Progress variations;
wenzelm
parents: 65828
diff changeset
   107
}
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   108
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   109
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   110
/* structured program progress */
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   111
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   112
object Program_Progress {
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   113
  class Program private[Program_Progress](heading: String, title: String) {
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   114
    private val output_buffer = new StringBuffer(256)  // synchronized
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   115
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
   116
    def output(message: Progress.Message): Unit = synchronized {
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   117
      if (output_buffer.length() > 0) output_buffer.append('\n')
77502
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
   118
      output_buffer.append(message.output_text)
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   119
    }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   120
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   121
    val start_time: Time = Time.now()
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   122
    private var stop_time: Option[Time] = None
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   123
    def stop_now(): Unit = synchronized { stop_time = Some(Time.now()) }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   124
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   125
    def output(): (Command.Results, XML.Body) = synchronized {
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   126
      val output_text = output_buffer.toString
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   127
      val elapsed_time = stop_time.map(t => t - start_time)
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   128
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   129
      val message_prefix = heading + " "
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   130
      val message_suffix =
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   131
        elapsed_time match {
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   132
          case None => " ..."
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   133
          case Some(t) => " ... (" + t.message + " elapsed time)"
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   134
        }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   135
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   136
      val (results, message) =
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   137
        if (output_text.isEmpty) {
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   138
          (Command.Results.empty, XML.string(message_prefix + title + message_suffix))
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   139
        }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   140
        else {
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   141
          val i = Document_ID.make()
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   142
          val results = Command.Results.make(List(i -> Protocol.writeln_message(output_text)))
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   143
          val message =
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   144
            XML.string(message_prefix) :::
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   145
            List(XML.Elem(Markup(Markup.WRITELN, Markup.Serial(i)), XML.string(title))) :::
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   146
            XML.string(message_suffix)
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   147
          (results, message)
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   148
        }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   149
76997
wenzelm
parents: 76994
diff changeset
   150
      (results, List(XML.elem(Markup.TRACING_MESSAGE, message)))
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   151
    }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   152
  }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   153
}
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   154
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   155
abstract class Program_Progress(
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   156
  default_heading: String = "Running",
77499
8fc94efa954a clarified signature: more uniform Progress.verbose, avoid adhoc "override def theory()";
wenzelm
parents: 77498
diff changeset
   157
  default_title: String = "program",
8fc94efa954a clarified signature: more uniform Progress.verbose, avoid adhoc "override def theory()";
wenzelm
parents: 77498
diff changeset
   158
  override val verbose: Boolean = false
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   159
) extends Progress {
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   160
  private var _finished_programs: List[Program_Progress.Program] = Nil
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   161
  private var _running_program: Option[Program_Progress.Program] = None
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   162
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   163
  def output(): (Command.Results, XML.Body) = synchronized {
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   164
    val programs = (_running_program.toList ::: _finished_programs).reverse
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   165
    val programs_output = programs.map(_.output())
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   166
    val results = Command.Results.merge(programs_output.map(_._1))
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   167
    val body = Library.separate(Pretty.Separator, programs_output.map(_._2)).flatten
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   168
    (results, body)
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   169
  }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   170
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   171
  private def start_program(heading: String, title: String): Unit = synchronized {
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   172
    _running_program = Some(new Program_Progress.Program(heading, title))
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   173
  }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   174
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   175
  def stop_program(): Unit = synchronized {
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   176
    _running_program match {
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   177
      case Some(program) =>
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   178
        program.stop_now()
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   179
        _finished_programs ::= program
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   180
        _running_program = None
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   181
      case None =>
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   182
    }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   183
  }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   184
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   185
  def detect_program(s: String): Option[String]
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   186
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
   187
  override def output(message: Progress.Message): Unit = synchronized {
77502
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
   188
    val writeln_msg = if (message.kind == Progress.Kind.writeln) message.text else ""
2e2b2bd6b2d2 clarified signature: require just one "override def echo(message: Progress.Message): Unit";
wenzelm
parents: 77499
diff changeset
   189
    detect_program(writeln_msg).map(Word.explode) match {
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   190
      case Some(a :: bs) =>
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   191
        stop_program()
77174
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   192
        start_program(a, Word.implode(bs))
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   193
      case _ =>
1eb55d6809b3 more general program start message;
wenzelm
parents: 77158
diff changeset
   194
        if (_running_program.isEmpty) start_program(default_heading, default_title)
77509
3bc49507bae5 clarified treatment of "verbose" messages, e.g. Progress.theory();
wenzelm
parents: 77507
diff changeset
   195
        if (do_output(message)) _running_program.get.output(message)
76994
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   196
    }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   197
  }
7c23db6b857b more detailed Program_Progress / Log_Progress: each program gets its own log output, which is attached to the document via markup;
wenzelm
parents: 76593
diff changeset
   198
}