added Isar/isar_document.ML: Interactive Isar documents.
authorwenzelm
Tue, 13 Jan 2009 13:47:35 +0100
changeset 29459 8acad4f0a727
parent 29458 98d749ae5edc
child 29466 a905283ad03e
added Isar/isar_document.ML: Interactive Isar documents.
src/Pure/IsaMakefile
src/Pure/Isar/ROOT.ML
src/Pure/Isar/isar_document.ML
--- a/src/Pure/IsaMakefile	Tue Jan 13 13:46:30 2009 +0100
+++ b/src/Pure/IsaMakefile	Tue Jan 13 13:47:35 2009 +0100
@@ -39,9 +39,9 @@
   Isar/class_target.ML Isar/code.ML Isar/code_unit.ML			\
   Isar/constdefs.ML Isar/context_rules.ML Isar/element.ML		\
   Isar/expression.ML Isar/find_theorems.ML Isar/isar.ML			\
-  Isar/isar_cmd.ML Isar/isar_syn.ML Isar/local_defs.ML			\
-  Isar/local_syntax.ML Isar/local_theory.ML Isar/locale.ML		\
-  Isar/method.ML Isar/net_rules.ML Isar/old_locale.ML			\
+  Isar/isar_document.ML Isar/isar_cmd.ML Isar/isar_syn.ML		\
+  Isar/local_defs.ML Isar/local_syntax.ML Isar/local_theory.ML		\
+  Isar/locale.ML Isar/method.ML Isar/net_rules.ML Isar/old_locale.ML	\
   Isar/object_logic.ML Isar/obtain.ML Isar/outer_keyword.ML		\
   Isar/outer_lex.ML Isar/outer_parse.ML Isar/outer_syntax.ML		\
   Isar/overloading.ML Isar/proof.ML Isar/proof_context.ML		\
--- a/src/Pure/Isar/ROOT.ML	Tue Jan 13 13:46:30 2009 +0100
+++ b/src/Pure/Isar/ROOT.ML	Tue Jan 13 13:47:35 2009 +0100
@@ -85,6 +85,7 @@
 use "../Thy/thy_info.ML";
 use "session.ML";
 use "isar.ML";
+use "isar_document.ML";
 
 (*theory and proof operations*)
 use "rule_insts.ML";
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/Pure/Isar/isar_document.ML	Tue Jan 13 13:47:35 2009 +0100
@@ -0,0 +1,198 @@
+(*  Title:      Pure/Isar/isar_document.ML
+    Author:     Makarius
+
+Interactive Isar documents.
+*)
+
+signature ISAR_DOCUMENT =
+sig
+  type state_id = string
+  type command_id = string
+  type document_id = string
+  val define_command: command_id -> Toplevel.transition -> unit
+  val begin_document: document_id -> Path.T -> unit
+  val edit_document: document_id -> document_id -> (command_id * command_id option) list -> unit
+  val end_document: document_id -> unit
+end;
+
+structure IsarDocument: ISAR_DOCUMENT =
+struct
+
+(* unique identifiers *)
+
+type document_id = string;
+type command_id = string;
+type state_id = string;
+
+
+(** execution states **)
+
+(* command status *)
+
+datatype status =
+  Unprocessed |
+  Running of Toplevel.state option future |
+  Failed |
+  Finished of Toplevel.state;
+
+fun status_markup Unprocessed = Markup.unprocessed
+  | status_markup (Running future) = Markup.running (Task_Queue.str_of_task (Future.task_of future))
+  | status_markup Failed = Markup.failed
+  | status_markup (Finished _) = Markup.finished;
+
+fun status_update tr state status =
+  (case status of
+    Unprocessed => SOME (Running (Future.fork_pri 1 (fn () =>
+      (case Toplevel.transition true tr state of
+        NONE => (Toplevel.error_msg tr (SYS_ERROR "Cannot terminate here!", ""); NONE)
+      | SOME (_, SOME err) => (Toplevel.error_msg tr err; NONE)
+      | SOME (state', NONE) => SOME state'))))
+  | Running future =>
+      (case Future.peek future of
+        NONE => NONE
+      | SOME (Exn.Result NONE) => SOME Failed
+      | SOME (Exn.Result (SOME state')) => SOME (Finished state')
+      | SOME (Exn.Exn exn) => (Toplevel.error_msg tr (exn, ""); SOME Failed))
+  | _ => NONE);
+
+
+(* state *)
+
+datatype state = State of
+ {prev: state_id option,
+  command: command_id,
+  status: status};
+
+fun make_state prev command status = State {prev = prev, command = command, status = status};
+
+
+
+(** documents **)
+
+(* command entries *)
+
+datatype entry = Entry of
+ {prev: command_id option,
+  next: command_id option,
+  state: state_id};
+
+fun make_entry prev next state = Entry {prev = prev, next = next, state = state};
+
+
+(* document *)
+
+datatype document = Document of
+ {dir: Path.T,                    (*master directory*)
+  name: string,                   (*theory name*)
+  commands: entry Symtab.table};  (*unique command entries indexed by command_id*)
+
+fun make_document dir name commands = Document {dir = dir, name = name, commands = commands};
+
+
+
+(** global configuration **)
+
+fun err_dup kind id = sys_error ("Duplicate " ^ kind ^ ": " ^ quote id);
+fun err_undef kind id = sys_error ("Unknown " ^ kind ^ ": " ^ quote id);
+
+local
+  val global_states = ref (Symtab.empty: state Symtab.table);
+  val global_commands = ref (Symtab.empty: Toplevel.transition Symtab.table);
+  val global_documents = ref (Symtab.empty: document Symtab.table);
+in
+
+fun change_states f = NAMED_CRITICAL "Isar" (fn () => change global_states f);
+fun get_states () = NAMED_CRITICAL "Isar" (fn () => ! global_states);
+
+fun change_commands f = NAMED_CRITICAL "Isar" (fn () => change global_commands f);
+fun get_commands () = NAMED_CRITICAL "Isar" (fn () => ! global_commands);
+
+fun change_documents f = NAMED_CRITICAL "Isar" (fn () => change global_documents f);
+fun get_documents () = NAMED_CRITICAL "Isar" (fn () => ! global_documents);
+
+fun init () = NAMED_CRITICAL "Isar" (fn () =>
+ (global_states := Symtab.empty;
+  global_commands := Symtab.empty;
+  global_documents := Symtab.empty));
+
+end;
+
+
+fun define_state (id: state_id) state =
+  change_states (Symtab.update_new (id, state))
+    handle Symtab.DUP dup => err_dup "state" dup;
+
+fun the_state (id: state_id) =
+  (case Symtab.lookup (get_states ()) id of
+    NONE => err_undef "state" id
+  | SOME (State {status as Finished state, ...}) => state
+  | _ => sys_error ("Unfinished state " ^ id));
+
+
+fun define_command id tr =
+  change_commands (Symtab.update_new (id, Toplevel.put_id id tr))
+    handle Symtab.DUP dup => err_dup ("command " ^ Toplevel.name_of tr) dup;
+
+fun the_command (id: command_id) =
+  (case Symtab.lookup (get_commands ()) id of
+    NONE => err_undef "command" id
+  | SOME cmd => cmd);
+
+
+fun define_document (id: document_id) document =
+  change_documents (Symtab.update_new (id, document))
+    handle Symtab.DUP dup => err_dup "document" dup;
+
+fun the_document (id: document_id) =
+  (case Symtab.lookup (get_documents ()) id of
+    NONE => err_undef "document" id
+  | SOME doc => doc);
+
+
+fun begin_document (id: document_id) path =
+  let
+    val (dir, name) = ThyLoad.split_thy_path path;
+    val _ = define_command id Toplevel.empty;
+    val _ = define_state id (make_state NONE id (Finished Toplevel.toplevel));
+    val commands = Symtab.make [(id, make_entry NONE NONE id)];
+    val _ = define_document id (make_document dir name commands);
+  in () end;
+
+fun edit_document (id: document_id) (new_id: document_id) edits =
+  let
+    val Document {dir, name, commands} = the_document id;
+    val commands' = sys_error "FIXME";
+    val _ = define_document new_id (make_document dir name commands');
+  in () end;
+
+fun end_document (id: document_id) = sys_error "FIXME";
+
+
+
+(** concrete syntax **)
+
+local structure P = OuterParse structure V = ValueParse in
+
+val _ =
+  OuterSyntax.internal_command "Isar.define_command"
+    (P.string -- P.string >> (fn (id, text) =>
+      Toplevel.imperative (fn () =>
+        define_command id (OuterSyntax.prepare_command (Position.id id) text))));
+
+val _ =
+  OuterSyntax.internal_command "Isar.begin_document"
+    (P.string -- P.string >> (fn (id, path) =>
+      Toplevel.imperative (fn () => begin_document id (Path.explode path))));
+
+val _ =
+  OuterSyntax.internal_command "Isar.edit_document"
+    (P.string -- P.string -- V.list (P.string -- (P.string >> SOME) || P.string >> rpair NONE)
+      >> (fn ((id, new_id), edits) => Toplevel.imperative (fn () => end_document id new_id edits)));
+
+val _ =
+  OuterSyntax.internal_command "Isar.end_document"
+    (P.string >> (fn id => Toplevel.imperative (fn () => end_document id)));
+
+end;
+
+end;