| author | desharna | 
| Sat, 04 Jun 2022 18:32:30 +0200 | |
| changeset 75540 | 02719bd7b4e6 | 
| parent 75406 | 85e8b4c2b9a9 | 
| child 75552 | 4aa3da02fd4d | 
| permissions | -rw-r--r-- | 
| 62631 | 1 | /* Title: Pure/Thy/sessions.scala | 
| 2 | Author: Makarius | |
| 3 | ||
| 65430 | 4 | Cumulative session information. | 
| 62631 | 5 | */ | 
| 6 | ||
| 7 | package isabelle | |
| 8 | ||
| 65461 
b6c2e30dc018
support for known theories files (according to multiple uses);
 wenzelm parents: 
65457diff
changeset | 9 | import java.io.{File => JFile}
 | 
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 10 | import java.nio.ByteBuffer | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 11 | import java.nio.channels.FileChannel | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 12 | import java.nio.file.StandardOpenOption | 
| 62631 | 13 | |
| 73364 | 14 | import scala.collection.immutable.{SortedSet, SortedMap}
 | 
| 62631 | 15 | import scala.collection.mutable | 
| 16 | ||
| 17 | ||
| 75393 | 18 | object Sessions {
 | 
| 67284 | 19 | /* session and theory names */ | 
| 62883 | 20 | |
| 72837 | 21 |   val ROOTS: Path = Path.explode("ROOTS")
 | 
| 72832 | 22 |   val ROOT: Path = Path.explode("ROOT")
 | 
| 23 | ||
| 72837 | 24 | val roots_name: String = "ROOTS" | 
| 67215 | 25 | val root_name: String = "ROOT" | 
| 75406 | 26 | val theory_import: String = "Pure.Sessions" | 
| 67215 | 27 | |
| 69762 
58fb0d779583
support for session information via virtual file-system;
 wenzelm parents: 
69671diff
changeset | 28 | val UNSORTED = "Unsorted" | 
| 65445 
e9e7f5f5794c
more qualifier treatment, but in the end it is still ignored;
 wenzelm parents: 
65441diff
changeset | 29 | val DRAFT = "Draft" | 
| 
e9e7f5f5794c
more qualifier treatment, but in the end it is still ignored;
 wenzelm parents: 
65441diff
changeset | 30 | |
| 65360 | 31 | def is_pure(name: String): Boolean = name == Thy_Header.PURE | 
| 64856 | 32 | |
| 67284 | 33 | |
| 34 | def exclude_session(name: String): Boolean = name == "" || name == DRAFT | |
| 35 | ||
| 67286 | 36 | def exclude_theory(name: String): Boolean = | 
| 67290 | 37 | name == root_name || name == "README" || name == "index" || name == "bib" | 
| 67284 | 38 | |
| 39 | ||
| 72837 | 40 | /* ROOTS file format */ | 
| 41 | ||
| 75393 | 42 |   class File_Format extends isabelle.File_Format {
 | 
| 72837 | 43 | val format_name: String = roots_name | 
| 44 | val file_ext = "" | |
| 72839 
a597300290de
clarified File_Format.detect: needs to operate on full node name;
 wenzelm parents: 
72837diff
changeset | 45 | |
| 
a597300290de
clarified File_Format.detect: needs to operate on full node name;
 wenzelm parents: 
72837diff
changeset | 46 | override def detect(name: String): Boolean = | 
| 
a597300290de
clarified File_Format.detect: needs to operate on full node name;
 wenzelm parents: 
72837diff
changeset | 47 |       Thy_Header.split_file_name(name) match {
 | 
| 
a597300290de
clarified File_Format.detect: needs to operate on full node name;
 wenzelm parents: 
72837diff
changeset | 48 | case Some((_, file_name)) => file_name == roots_name | 
| 
a597300290de
clarified File_Format.detect: needs to operate on full node name;
 wenzelm parents: 
72837diff
changeset | 49 | case None => false | 
| 
a597300290de
clarified File_Format.detect: needs to operate on full node name;
 wenzelm parents: 
72837diff
changeset | 50 | } | 
| 72837 | 51 | |
| 52 | override def theory_suffix: String = "ROOTS_file" | |
| 53 | override def theory_content(name: String): String = | |
| 54 | """theory "ROOTS" imports Pure begin ROOTS_file """ + | |
| 55 | Outer_Syntax.quote_string(name) + """ end""" | |
| 56 | } | |
| 57 | ||
| 58 | ||
| 67284 | 59 | /* base info and source dependencies */ | 
| 60 | ||
| 65495 | 61 | sealed case class Base( | 
| 66571 | 62 | pos: Position.T = Position.none, | 
| 70681 | 63 | session_directories: Map[JFile, String] = Map.empty, | 
| 65495 | 64 | global_theories: Map[String, String] = Map.empty, | 
| 72323 
e36f94e2eb6b
some support for document preparation in Isabelle/Scala;
 wenzelm parents: 
72068diff
changeset | 65 | session_theories: List[Document.Node.Name] = Nil, | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 66 | document_theories: List[Document.Node.Name] = Nil, | 
| 66717 
67dbf5cdc056
more informative loaded_theories: dependencies and syntax;
 wenzelm parents: 
66716diff
changeset | 67 | loaded_theories: Graph[String, Outer_Syntax] = Graph.string, | 
| 70687 
086575316fd5
eliminated pointless theory graph (reverting parts of a56eab490f4e): it caused problems with loaded vs. non-loaded node names, e.g. for theory Pure (see also 29bb1ebb188f);
 wenzelm parents: 
70686diff
changeset | 68 | used_theories: List[(Document.Node.Name, Options)] = Nil, | 
| 72816 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 69 | load_commands: Map[Document.Node.Name, List[Command_Span.Span]] = Map.empty, | 
| 70740 | 70 | known_theories: Map[String, Document.Node.Entry] = Map.empty, | 
| 71 | known_loaded_files: Map[String, List[Path]] = Map.empty, | |
| 66720 | 72 | overall_syntax: Outer_Syntax = Outer_Syntax.empty, | 
| 66744 | 73 | imported_sources: List[(Path, SHA1.Digest)] = Nil, | 
| 65495 | 74 | sources: List[(Path, SHA1.Digest)] = Nil, | 
| 66822 | 75 | session_graph_display: Graph_Display.Graph = Graph_Display.empty_graph, | 
| 75393 | 76 | errors: List[String] = Nil | 
| 77 |   ) {
 | |
| 69102 
4b06a20b13b5
tuned output -- avoid bombing of Scala toplevel, e.g. for AFP deps;
 wenzelm parents: 
69080diff
changeset | 78 | override def toString: String = | 
| 
4b06a20b13b5
tuned output -- avoid bombing of Scala toplevel, e.g. for AFP deps;
 wenzelm parents: 
69080diff
changeset | 79 | "Sessions.Base(loaded_theories = " + loaded_theories.size + | 
| 
4b06a20b13b5
tuned output -- avoid bombing of Scala toplevel, e.g. for AFP deps;
 wenzelm parents: 
69080diff
changeset | 80 | ", used_theories = " + used_theories.length + ")" | 
| 
4b06a20b13b5
tuned output -- avoid bombing of Scala toplevel, e.g. for AFP deps;
 wenzelm parents: 
69080diff
changeset | 81 | |
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 82 | def theory_qualifier(name: String): String = | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 83 | global_theories.getOrElse(name, Long_Name.qualifier(name)) | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 84 | def theory_qualifier(name: Document.Node.Name): String = theory_qualifier(name.theory) | 
| 66966 | 85 | |
| 66717 
67dbf5cdc056
more informative loaded_theories: dependencies and syntax;
 wenzelm parents: 
66716diff
changeset | 86 | def loaded_theory(name: String): Boolean = loaded_theories.defined(name) | 
| 66712 | 87 | def loaded_theory(name: Document.Node.Name): Boolean = loaded_theory(name.theory) | 
| 65432 | 88 | |
| 66717 
67dbf5cdc056
more informative loaded_theories: dependencies and syntax;
 wenzelm parents: 
66716diff
changeset | 89 | def loaded_theory_syntax(name: String): Option[Outer_Syntax] = | 
| 
67dbf5cdc056
more informative loaded_theories: dependencies and syntax;
 wenzelm parents: 
66716diff
changeset | 90 | if (loaded_theory(name)) Some(loaded_theories.get_node(name)) else None | 
| 
67dbf5cdc056
more informative loaded_theories: dependencies and syntax;
 wenzelm parents: 
66716diff
changeset | 91 | def loaded_theory_syntax(name: Document.Node.Name): Option[Outer_Syntax] = | 
| 
67dbf5cdc056
more informative loaded_theories: dependencies and syntax;
 wenzelm parents: 
66716diff
changeset | 92 | loaded_theory_syntax(name.theory) | 
| 
67dbf5cdc056
more informative loaded_theories: dependencies and syntax;
 wenzelm parents: 
66716diff
changeset | 93 | |
| 72669 | 94 | def theory_syntax(name: Document.Node.Name): Outer_Syntax = | 
| 95 | loaded_theory_syntax(name) getOrElse overall_syntax | |
| 96 | ||
| 66770 
122df1fde073
clarified node_syntax (amending ae38b8c0fdd9): default to overall_syntax, e.g. relevant for command spans wrt. bad header;
 wenzelm parents: 
66764diff
changeset | 97 | def node_syntax(nodes: Document.Nodes, name: Document.Node.Name): Outer_Syntax = | 
| 
122df1fde073
clarified node_syntax (amending ae38b8c0fdd9): default to overall_syntax, e.g. relevant for command spans wrt. bad header;
 wenzelm parents: 
66764diff
changeset | 98 | nodes(name).syntax orElse loaded_theory_syntax(name) getOrElse overall_syntax | 
| 65355 | 99 | } | 
| 64856 | 100 | |
| 75393 | 101 |   sealed case class Deps(sessions_structure: Structure, session_bases: Map[String, Base]) {
 | 
| 69102 
4b06a20b13b5
tuned output -- avoid bombing of Scala toplevel, e.g. for AFP deps;
 wenzelm parents: 
69080diff
changeset | 102 |     override def toString: String = "Sessions.Deps(" + sessions_structure + ")"
 | 
| 
4b06a20b13b5
tuned output -- avoid bombing of Scala toplevel, e.g. for AFP deps;
 wenzelm parents: 
69080diff
changeset | 103 | |
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 104 | def is_empty: Boolean = session_bases.isEmpty | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 105 | def apply(name: String): Base = session_bases(name) | 
| 66988 | 106 | def get(name: String): Option[Base] = session_bases.get(name) | 
| 66744 | 107 | |
| 108 | def imported_sources(name: String): List[SHA1.Digest] = | |
| 109 | session_bases(name).imported_sources.map(_._2) | |
| 110 | ||
| 111 | def sources(name: String): List[SHA1.Digest] = | |
| 112 | session_bases(name).sources.map(_._2) | |
| 66571 | 113 | |
| 114 | def errors: List[String] = | |
| 115 |       (for {
 | |
| 116 | (name, base) <- session_bases.iterator | |
| 117 | if base.errors.nonEmpty | |
| 118 | } yield cat_lines(base.errors) + | |
| 119 | "\nThe error(s) above occurred in session " + quote(name) + Position.here(base.pos) | |
| 120 | ).toList | |
| 121 | ||
| 122 | def check_errors: Deps = | |
| 123 |       errors match {
 | |
| 124 | case Nil => this | |
| 125 | case errs => error(cat_lines(errs)) | |
| 126 | } | |
| 65251 | 127 | } | 
| 64856 | 128 | |
| 67052 | 129 | def deps(sessions_structure: Structure, | 
| 75393 | 130 | progress: Progress = new Progress, | 
| 131 | inlined_files: Boolean = false, | |
| 132 | verbose: Boolean = false, | |
| 133 | list_files: Boolean = false, | |
| 134 | check_keywords: Set[String] = Set.empty | |
| 135 |   ): Deps = {
 | |
| 66743 | 136 | var cache_sources = Map.empty[JFile, SHA1.Digest] | 
| 75393 | 137 |     def check_sources(paths: List[Path]): List[(Path, SHA1.Digest)] = {
 | 
| 66743 | 138 |       for {
 | 
| 139 | path <- paths | |
| 140 | file = path.file | |
| 141 | if cache_sources.isDefinedAt(file) || file.isFile | |
| 142 | } | |
| 143 |       yield {
 | |
| 144 |         cache_sources.get(file) match {
 | |
| 145 | case Some(digest) => (path, digest) | |
| 146 | case None => | |
| 147 | val digest = SHA1.digest(file) | |
| 148 | cache_sources = cache_sources + (file -> digest) | |
| 149 | (path, digest) | |
| 150 | } | |
| 151 | } | |
| 152 | } | |
| 153 | ||
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 154 | val session_bases = | 
| 73359 | 155 |       sessions_structure.imports_topological_order.foldLeft(Map("" -> sessions_structure.bootstrap)) {
 | 
| 67023 | 156 | case (session_bases, session_name) => | 
| 67880 | 157 | progress.expose_interrupt() | 
| 65251 | 158 | |
| 67024 | 159 | val info = sessions_structure(session_name) | 
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 160 |           try {
 | 
| 72066 | 161 | val deps_base = info.deps_base(session_bases) | 
| 162 | val resources = new Resources(sessions_structure, deps_base) | |
| 65251 | 163 | |
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 164 |             if (verbose || list_files) {
 | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 165 | val groups = | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 166 | if (info.groups.isEmpty) "" | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 167 |                 else info.groups.mkString(" (", " ", ")")
 | 
| 72670 | 168 |               progress.echo("Session " + info.chapter_session + groups)
 | 
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 169 | } | 
| 65251 | 170 | |
| 69008 
d55783ea6cf6
more detailed session dependencies, with conditions for theories;
 wenzelm parents: 
68841diff
changeset | 171 | val dependencies = resources.session_dependencies(info) | 
| 65251 | 172 | |
| 67053 | 173 | val overall_syntax = dependencies.overall_syntax | 
| 65251 | 174 | |
| 72323 
e36f94e2eb6b
some support for document preparation in Isabelle/Scala;
 wenzelm parents: 
72068diff
changeset | 175 | val session_theories = | 
| 72565 | 176 | dependencies.theories.filter(name => deps_base.theory_qualifier(name) == session_name) | 
| 72323 
e36f94e2eb6b
some support for document preparation in Isabelle/Scala;
 wenzelm parents: 
72068diff
changeset | 177 | |
| 67059 | 178 | val theory_files = dependencies.theories.map(_.path) | 
| 72068 
4768b1facec2
clarified errors: avoid hiding of import_errors/dir_errors by their consequences (file-access problems);
 wenzelm parents: 
72067diff
changeset | 179 | |
| 72816 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 180 | dependencies.load_commands | 
| 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 181 | |
| 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 182 | val (load_commands, load_commands_errors) = | 
| 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 183 |               try { if (inlined_files) (dependencies.load_commands, Nil) else (Nil, Nil) }
 | 
| 72068 
4768b1facec2
clarified errors: avoid hiding of import_errors/dir_errors by their consequences (file-access problems);
 wenzelm parents: 
72067diff
changeset | 184 |               catch { case ERROR(msg) => (Nil, List(msg)) }
 | 
| 65251 | 185 | |
| 72816 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 186 | val loaded_files = | 
| 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 187 |               load_commands.map({ case (name, spans) => dependencies.loaded_files(name, spans) })
 | 
| 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 188 | |
| 66742 | 189 | val session_files = | 
| 66701 | 190 | (theory_files ::: loaded_files.flatMap(_._2) ::: | 
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 191 | info.document_files.map(file => info.dir + file._1 + file._2)).map(_.expand) | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 192 | |
| 67053 | 193 | val imported_files = if (inlined_files) dependencies.imported_files else Nil | 
| 66743 | 194 | |
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 195 | if (list_files) | 
| 66742 | 196 |               progress.echo(cat_lines(session_files.map(_.implode).sorted.map("  " + _)))
 | 
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 197 | |
| 66719 
d37efafd55b5
clarified theory syntax vs. overall session syntax;
 wenzelm parents: 
66718diff
changeset | 198 |             if (check_keywords.nonEmpty) {
 | 
| 
d37efafd55b5
clarified theory syntax vs. overall session syntax;
 wenzelm parents: 
66718diff
changeset | 199 | Check_Keywords.check_keywords( | 
| 66720 | 200 | progress, overall_syntax.keywords, check_keywords, theory_files) | 
| 66719 
d37efafd55b5
clarified theory syntax vs. overall session syntax;
 wenzelm parents: 
66718diff
changeset | 201 | } | 
| 65251 | 202 | |
| 75393 | 203 |             val session_graph_display: Graph_Display.Graph = {
 | 
| 65507 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 204 | def session_node(name: String): Graph_Display.Node = | 
| 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 205 |                 Graph_Display.Node("[" + name + "]", "session." + name)
 | 
| 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 206 | |
| 75393 | 207 |               def node(name: Document.Node.Name): Graph_Display.Node = {
 | 
| 72066 | 208 | val qualifier = deps_base.theory_qualifier(name) | 
| 66780 
bf54ca580bf2
theory qualifier is always session name (see also 31e8a86971a8);
 wenzelm parents: 
66770diff
changeset | 209 | if (qualifier == info.name) | 
| 65507 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 210 | Graph_Display.Node(name.theory_base_name, "theory." + name.theory) | 
| 65528 | 211 | else session_node(qualifier) | 
| 65507 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 212 | } | 
| 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 213 | |
| 70946 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 214 | val required_sessions = | 
| 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 215 | dependencies.loaded_theories.all_preds(dependencies.theories.map(_.theory)) | 
| 72066 | 216 | .map(theory => deps_base.theory_qualifier(theory)) | 
| 71574 
95460356d633
avoid premature crash due to missing session parents/imports;
 wenzelm parents: 
71569diff
changeset | 217 | .filter(name => name != info.name && sessions_structure.defined(name)) | 
| 70946 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 218 | |
| 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 219 | val required_subgraph = | 
| 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 220 | sessions_structure.imports_graph | 
| 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 221 | .restrict(sessions_structure.imports_graph.all_preds(required_sessions).toSet) | 
| 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 222 | .transitive_closure | 
| 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 223 | .restrict(required_sessions.toSet) | 
| 
79d23e6436d0
clarified session_graph_display: restrict sessions to actually required theories;
 wenzelm parents: 
70869diff
changeset | 224 | .transitive_reduction_acyclic | 
| 65507 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 225 | |
| 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 226 | val graph0 = | 
| 73359 | 227 |                 required_subgraph.topological_order.foldLeft(Graph_Display.empty_graph) {
 | 
| 228 | case (g, session) => | |
| 229 | val a = session_node(session) | |
| 230 | val bs = required_subgraph.imm_preds(session).toList.map(session_node) | |
| 231 | bs.foldLeft((a :: bs).foldLeft(g)(_.default_node(_, Nil)))(_.add_edge(_, a)) | |
| 232 | } | |
| 65507 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 233 | |
| 73359 | 234 |               dependencies.entries.foldLeft(graph0) {
 | 
| 235 | case (g, entry) => | |
| 236 | val a = node(entry.name) | |
| 237 | val bs = entry.header.imports.map(node).filterNot(_ == a) | |
| 238 | bs.foldLeft((a :: bs).foldLeft(g)(_.default_node(_, Nil)))(_.add_edge(_, a)) | |
| 239 | } | |
| 65507 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 240 | } | 
| 
decdb95bd007
clarified session graph: collapse theories from other sessions;
 wenzelm parents: 
65500diff
changeset | 241 | |
| 70740 | 242 | val known_theories = | 
| 73359 | 243 | dependencies.entries.iterator.map(entry => entry.name.theory -> entry). | 
| 244 | foldLeft(deps_base.known_theories)(_ + _) | |
| 70740 | 245 | |
| 72066 | 246 | val known_loaded_files = deps_base.known_loaded_files ++ loaded_files | 
| 66701 | 247 | |
| 75393 | 248 |             val import_errors = {
 | 
| 72067 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 249 | val known_sessions = | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 250 | sessions_structure.imports_requirements(List(session_name)).toSet | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 251 |               for {
 | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 252 | name <- dependencies.theories | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 253 | qualifier = deps_base.theory_qualifier(name) | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 254 | if !known_sessions(qualifier) | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 255 | } yield "Bad import of theory " + quote(name.toString) + | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 256 | ": need to include sessions " + quote(qualifier) + " in ROOT" | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 257 | } | 
| 
17507b48b6f5
clarified errors: avoid accidental import from other session that happens to be within overall selection (notably "isabelle build -a");
 wenzelm parents: 
72066diff
changeset | 258 | |
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 259 | val document_errors = | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 260 | info.document_theories.flatMap( | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 261 |               {
 | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 262 | case (thy, pos) => | 
| 74812 | 263 | val build_hierarchy = | 
| 72647 
fd6dc1a4b9ca
more robust, e.g. for "isabelle build_doc system";
 wenzelm parents: 
72634diff
changeset | 264 |                     if (sessions_structure.build_graph.defined(session_name)) {
 | 
| 74812 | 265 | sessions_structure.build_hierarchy(session_name) | 
| 72647 
fd6dc1a4b9ca
more robust, e.g. for "isabelle build_doc system";
 wenzelm parents: 
72634diff
changeset | 266 | } | 
| 
fd6dc1a4b9ca
more robust, e.g. for "isabelle build_doc system";
 wenzelm parents: 
72634diff
changeset | 267 | else Nil | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 268 | |
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 269 | def err(msg: String): Option[String] = | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 270 | Some(msg + " " + quote(thy) + Position.here(pos)) | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 271 | |
| 72604 | 272 |                   known_theories.get(thy).map(_.name) match {
 | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 273 |                     case None => err("Unknown document theory")
 | 
| 72604 | 274 | case Some(name) => | 
| 275 | val qualifier = deps_base.theory_qualifier(name) | |
| 276 |                       if (session_theories.contains(name)) {
 | |
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 277 |                         err("Redundant document theory from this session:")
 | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 278 | } | 
| 74812 | 279 | else if (build_hierarchy.contains(qualifier)) None | 
| 72604 | 280 | else if (dependencies.theories.contains(name)) None | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 281 |                       else err("Document theory from other session not imported properly:")
 | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 282 | } | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 283 | }) | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 284 | val document_theories = | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 285 |               info.document_theories.map({ case (thy, _) => known_theories(thy).name })
 | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 286 | |
| 75393 | 287 |             val dir_errors = {
 | 
| 70681 | 288 | val ok = info.dirs.map(_.canonical_file).toSet | 
| 70679 | 289 | val bad = | 
| 290 |                 (for {
 | |
| 72565 | 291 | name <- session_theories.iterator | 
| 70719 | 292 | path = name.master_dir_path | 
| 70676 | 293 | if !ok(path.canonical_file) | 
| 70679 | 294 | path1 = File.relative_path(info.dir.canonical, path).getOrElse(path) | 
| 295 | } yield (path1, name)).toList | |
| 71602 | 296 |               val bad_dirs = (for { (path1, _) <- bad } yield path1.toString).distinct.sorted
 | 
| 70679 | 297 | |
| 298 | val errs1 = | |
| 299 |                 for { (path1, name) <- bad }
 | |
| 300 | yield "Implicit use of directory " + path1 + " for theory " + quote(name.toString) | |
| 301 | val errs2 = | |
| 302 | if (bad_dirs.isEmpty) Nil | |
| 303 |                 else List("Implicit use of session directories: " + commas(bad_dirs))
 | |
| 70693 | 304 | val errs3 = for (p <- info.dirs if !p.is_dir) yield "No such directory: " + p | 
| 70719 | 305 | val errs4 = | 
| 306 |                 (for {
 | |
| 72565 | 307 | name <- session_theories.iterator | 
| 70719 | 308 | name1 <- resources.find_theory_node(name.theory) | 
| 309 | if name.node != name1.node | |
| 310 | } yield "Incoherent theory file import:\n " + name.path + " vs. \n " + name1.path) | |
| 311 | .toList | |
| 70693 | 312 | |
| 70719 | 313 | errs1 ::: errs2 ::: errs3 ::: errs4 | 
| 70676 | 314 | } | 
| 315 | ||
| 66743 | 316 | val sources_errors = | 
| 317 | for (p <- session_files if !p.is_file) yield "No such file: " + p | |
| 66604 | 318 | |
| 69904 | 319 | val path_errors = | 
| 320 |               try { Path.check_case_insensitive(session_files ::: imported_files); Nil }
 | |
| 321 |               catch { case ERROR(msg) => List(msg) }
 | |
| 322 | ||
| 67297 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 323 | val bibtex_errors = | 
| 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 324 |               try { info.bibtex_entries; Nil }
 | 
| 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 325 |               catch { case ERROR(msg) => List(msg) }
 | 
| 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 326 | |
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 327 | val base = | 
| 66571 | 328 | Base( | 
| 329 | pos = info.pos, | |
| 70673 
b0172698d0d3
theory_name based on session_directories: no need for expensive all_known;
 wenzelm parents: 
70672diff
changeset | 330 | session_directories = sessions_structure.session_directories, | 
| 70671 
cb1776c8e216
clarified signature: retain global session information, unaffected by later restriction;
 wenzelm parents: 
70668diff
changeset | 331 | global_theories = sessions_structure.global_theories, | 
| 72323 
e36f94e2eb6b
some support for document preparation in Isabelle/Scala;
 wenzelm parents: 
72068diff
changeset | 332 | session_theories = session_theories, | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 333 | document_theories = document_theories, | 
| 67053 | 334 | loaded_theories = dependencies.loaded_theories, | 
| 72062 | 335 | used_theories = dependencies.theories_adjunct, | 
| 72816 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 336 | load_commands = load_commands.toMap, | 
| 70740 | 337 | known_theories = known_theories, | 
| 338 | known_loaded_files = known_loaded_files, | |
| 66720 | 339 | overall_syntax = overall_syntax, | 
| 66744 | 340 | imported_sources = check_sources(imported_files), | 
| 66743 | 341 | sources = check_sources(session_files), | 
| 66822 | 342 | session_graph_display = session_graph_display, | 
| 72816 
ea4f86914cb2
support for PIDE markup for auxiliary files ("blobs");
 wenzelm parents: 
72799diff
changeset | 343 | errors = dependencies.errors ::: load_commands_errors ::: import_errors ::: | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 344 | document_errors ::: dir_errors ::: sources_errors ::: path_errors ::: | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 345 | bibtex_errors) | 
| 65251 | 346 | |
| 65519 | 347 | session_bases + (info.name -> base) | 
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 348 | } | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 349 |           catch {
 | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 350 | case ERROR(msg) => | 
| 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 351 | cat_error(msg, "The error(s) above occurred in session " + | 
| 65519 | 352 | quote(info.name) + Position.here(info.pos)) | 
| 65463 
104502de757c
more informative known_files: known_theories within the local session directory come first;
 wenzelm parents: 
65461diff
changeset | 353 | } | 
| 73359 | 354 | } | 
| 65251 | 355 | |
| 70686 
9cde8c4ea5a5
discontinued obsolete "isabelle imports" and all_known data;
 wenzelm parents: 
70685diff
changeset | 356 | Deps(sessions_structure, session_bases) | 
| 65372 
b722ee40c26c
refer to global_theories from all sessions, before selection;
 wenzelm parents: 
65371diff
changeset | 357 | } | 
| 65251 | 358 | |
| 66963 | 359 | |
| 360 | /* base info */ | |
| 361 | ||
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 362 | sealed case class Base_Info( | 
| 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 363 | session: String, | 
| 67052 | 364 | sessions_structure: Structure, | 
| 66975 | 365 | errors: List[String], | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 366 | base: Base, | 
| 75393 | 367 | infos: List[Info] | 
| 368 |   ) {
 | |
| 72627 | 369 | def check: Base_Info = if (errors.isEmpty) this else error(cat_lines(errors)) | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 370 | } | 
| 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 371 | |
| 67846 | 372 | def base_info(options: Options, | 
| 373 | session: String, | |
| 71726 
a5fda30edae2
clarified signature: more uniform treatment of stopped/interrupted state;
 wenzelm parents: 
71642diff
changeset | 374 | progress: Progress = new Progress, | 
| 65428 | 375 | dirs: List[Path] = Nil, | 
| 67922 
9e668ae81f97
clarified signature: prefer selective include_sessions;
 wenzelm parents: 
67880diff
changeset | 376 | include_sessions: List[String] = Nil, | 
| 68370 | 377 | session_ancestor: Option[String] = None, | 
| 75393 | 378 | session_requirements: Boolean = false | 
| 379 |   ): Base_Info = {
 | |
| 67026 | 380 | val full_sessions = load_structure(options, dirs = dirs) | 
| 65372 
b722ee40c26c
refer to global_theories from all sessions, before selection;
 wenzelm parents: 
65371diff
changeset | 381 | |
| 66990 | 382 | val selected_sessions = | 
| 68370 | 383 | full_sessions.selection(Selection(sessions = session :: session_ancestor.toList)) | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 384 | val info = selected_sessions(session) | 
| 68370 | 385 | val ancestor = session_ancestor orElse info.parent | 
| 66963 | 386 | |
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 387 | val (session1, infos1) = | 
| 68370 | 388 |       if (session_requirements && ancestor.isDefined) {
 | 
| 70671 
cb1776c8e216
clarified signature: retain global session information, unaffected by later restriction;
 wenzelm parents: 
70668diff
changeset | 389 | val deps = Sessions.deps(selected_sessions, progress = progress) | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 390 | val base = deps(session) | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 391 | |
| 66988 | 392 | val ancestor_loaded = | 
| 393 |           deps.get(ancestor.get) match {
 | |
| 66990 | 394 | case Some(ancestor_base) | 
| 395 | if !selected_sessions.imports_requirements(List(ancestor.get)).contains(session) => | |
| 71601 | 396 | ancestor_base.loaded_theories.defined _ | 
| 66990 | 397 | case _ => | 
| 66988 | 398 |               error("Bad ancestor " + quote(ancestor.get) + " for session " + quote(session))
 | 
| 399 | } | |
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 400 | |
| 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 401 | val required_theories = | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 402 |           for {
 | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 403 | thy <- base.loaded_theories.keys | 
| 66988 | 404 | if !ancestor_loaded(thy) && base.theory_qualifier(thy) != session | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 405 | } | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 406 | yield thy | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 407 | |
| 66988 | 408 | if (required_theories.isEmpty) (ancestor.get, Nil) | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 409 |         else {
 | 
| 68370 | 410 |           val other_name = info.name + "_requirements(" + ancestor.get + ")"
 | 
| 71569 
391ea80ff27c
more robust re-use of $ISABELLE_TMP_PREFIX (amending c1597167563e);
 wenzelm parents: 
70946diff
changeset | 411 | Isabelle_System.isabelle_tmp_prefix() | 
| 
391ea80ff27c
more robust re-use of $ISABELLE_TMP_PREFIX (amending c1597167563e);
 wenzelm parents: 
70946diff
changeset | 412 | |
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 413 | (other_name, | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 414 | List( | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 415 | make_info(info.options, | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 416 | dir_selected = false, | 
| 70685 
c1597167563e
avoid duplicate directories wrt. synthetic session;
 wenzelm parents: 
70684diff
changeset | 417 |                 dir = Path.explode("$ISABELLE_TMP_PREFIX"),
 | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 418 | chapter = info.chapter, | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 419 | Session_Entry( | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 420 | pos = info.pos, | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 421 | name = other_name, | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 422 | groups = info.groups, | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 423 | path = ".", | 
| 66988 | 424 | parent = ancestor, | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 425 | description = "Required theory imports from other sessions", | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 426 | options = Nil, | 
| 66991 | 427 | imports = info.deps, | 
| 70678 
36c8c32346cb
clarified syntax: 'directories' and 'theories' belong together;
 wenzelm parents: 
70676diff
changeset | 428 | directories = Nil, | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 429 | theories = List((Nil, required_theories.map(thy => ((thy, Position.none), false)))), | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 430 | document_theories = Nil, | 
| 68292 | 431 | document_files = Nil, | 
| 432 | export_files = Nil)))) | |
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 433 | } | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 434 | } | 
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 435 | else (session, Nil) | 
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 436 | |
| 66974 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 437 | val full_sessions1 = | 
| 
b14c24b31f45
avoid duplicate invocation of expensive Sessions.deps on full_sessions;
 wenzelm parents: 
66970diff
changeset | 438 | if (infos1.isEmpty) full_sessions | 
| 67026 | 439 | else load_structure(options, dirs = dirs, infos = infos1) | 
| 66987 
352b23c97ac8
support focus_session, for much faster startup of Isabelle/jEdit;
 wenzelm parents: 
66984diff
changeset | 440 | |
| 
352b23c97ac8
support focus_session, for much faster startup of Isabelle/jEdit;
 wenzelm parents: 
66984diff
changeset | 441 | val selected_sessions1 = | 
| 70683 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 442 | full_sessions1.selection(Selection(sessions = session1 :: session :: include_sessions)) | 
| 66964 | 443 | |
| 70683 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 444 | val deps1 = Sessions.deps(selected_sessions1, progress = progress) | 
| 66964 | 445 | |
| 71599 | 446 | Base_Info(session1, full_sessions1, deps1.errors, deps1(session1), infos1) | 
| 65251 | 447 | } | 
| 448 | ||
| 449 | ||
| 65415 
8cd54b18b68b
clarified signature: tree structure is not essential;
 wenzelm parents: 
65410diff
changeset | 450 | /* cumulative session info */ | 
| 62631 | 451 | |
| 452 | sealed case class Info( | |
| 65519 | 453 | name: String, | 
| 62631 | 454 | chapter: String, | 
| 66829 | 455 | dir_selected: Boolean, | 
| 62631 | 456 | pos: Position.T, | 
| 457 | groups: List[String], | |
| 458 | dir: Path, | |
| 459 | parent: Option[String], | |
| 460 | description: String, | |
| 70681 | 461 | directories: List[Path], | 
| 62631 | 462 | options: Options, | 
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 463 | imports: List[String], | 
| 65517 | 464 | theories: List[(Options, List[(String, Position.T)])], | 
| 65374 | 465 | global_theories: List[String], | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 466 | document_theories: List[(String, Position.T)], | 
| 62631 | 467 | document_files: List[(Path, Path)], | 
| 69671 | 468 | export_files: List[(Path, Int, List[String])], | 
| 75393 | 469 | meta_digest: SHA1.Digest | 
| 470 |   ) {
 | |
| 72670 | 471 | def chapter_session: String = chapter + "/" + name | 
| 472 | ||
| 73042 | 473 | def relative_path(info1: Info): String = | 
| 474 | if (name == info1.name) "" | |
| 475 | else if (chapter == info1.chapter) "../" + info1.name + "/" | |
| 476 | else "../../" + info1.chapter_session + "/" | |
| 477 | ||
| 66828 | 478 | def deps: List[String] = parent.toList ::: imports | 
| 479 | ||
| 75393 | 480 |     def deps_base(session_bases: String => Base): Base = {
 | 
| 72066 | 481 |       val parent_base = session_bases(parent.getOrElse(""))
 | 
| 482 | val imports_bases = imports.map(session_bases) | |
| 483 | parent_base.copy( | |
| 484 | known_theories = | |
| 73359 | 485 |           (for {
 | 
| 486 | base <- imports_bases.iterator | |
| 487 | (_, entry) <- base.known_theories.iterator | |
| 488 | } yield (entry.name.theory -> entry)).foldLeft(parent_base.known_theories)(_ + _), | |
| 72066 | 489 | known_loaded_files = | 
| 73359 | 490 | imports_bases.iterator.map(_.known_loaded_files). | 
| 491 | foldLeft(parent_base.known_loaded_files)(_ ++ _)) | |
| 72066 | 492 | } | 
| 493 | ||
| 70681 | 494 | def dirs: List[Path] = dir :: directories | 
| 70668 | 495 | |
| 73701 | 496 | def timeout_ignored: Boolean = | 
| 497 |       !options.bool("timeout_build") || Time.seconds(options.real("timeout")) < Time.ms(1)
 | |
| 498 | ||
| 62631 | 499 |     def timeout: Time = Time.seconds(options.real("timeout") * options.real("timeout_scale"))
 | 
| 67297 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 500 | |
| 72574 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 501 | def document_enabled: Boolean = | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 502 |       options.string("document") match {
 | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 503 | case "" | "false" => false | 
| 73826 
72900f34dbb3
allow system option short form NAME for NAME=true for type string, not just bool;
 wenzelm parents: 
73815diff
changeset | 504 | case "pdf" | "true" => true | 
| 72574 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 505 |         case doc => error("Bad document specification " + quote(doc))
 | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 506 | } | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 507 | |
| 75393 | 508 |     def document_variants: List[Document_Build.Document_Variant] = {
 | 
| 72574 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 509 | val variants = | 
| 72649 | 510 |         Library.space_explode(':', options.string("document_variants")).
 | 
| 73718 | 511 | map(Document_Build.Document_Variant.parse) | 
| 72649 | 512 | |
| 513 | val dups = Library.duplicates(variants.map(_.name)) | |
| 72574 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 514 |       if (dups.nonEmpty) error("Duplicate document variants: " + commas_quote(dups))
 | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 515 | |
| 72649 | 516 | variants | 
| 517 | } | |
| 518 | ||
| 75393 | 519 |     def documents: List[Document_Build.Document_Variant] = {
 | 
| 72672 | 520 | val variants = document_variants | 
| 72574 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 521 | if (!document_enabled || document_files.isEmpty) Nil else variants | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 522 | } | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 523 | |
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 524 | def document_output: Option[Path] = | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 525 |       options.string("document_output") match {
 | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 526 | case "" => None | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 527 | case s => Some(dir + Path.explode(s)) | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 528 | } | 
| 
d892f6d66402
build documents in Isabelle/Scala, based on generated tex files as session exports;
 wenzelm parents: 
72571diff
changeset | 529 | |
| 72648 | 530 |     def browser_info: Boolean = options.bool("browser_info")
 | 
| 531 | ||
| 72613 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 532 | lazy val bibtex_entries: List[Text.Info[String]] = | 
| 67297 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 533 |       (for {
 | 
| 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 534 | (document_dir, file) <- document_files.iterator | 
| 69366 | 535 | if Bibtex.is_bibtex(file.file_name) | 
| 67297 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 536 | info <- Bibtex.entries(File.read(dir + document_dir + file)).iterator | 
| 
86a099f896fc
formal check of @{cite} bibtex entries -- only in batch-mode session builds;
 wenzelm parents: 
67290diff
changeset | 537 | } yield info).toList | 
| 70859 
6e6254bbce1f
split into standard partitions, for improved scalability;
 wenzelm parents: 
70796diff
changeset | 538 | |
| 70869 | 539 |     def record_proofs: Boolean = options.int("record_proofs") >= 2
 | 
| 540 | ||
| 70859 
6e6254bbce1f
split into standard partitions, for improved scalability;
 wenzelm parents: 
70796diff
changeset | 541 | def is_afp: Boolean = chapter == AFP.chapter | 
| 
6e6254bbce1f
split into standard partitions, for improved scalability;
 wenzelm parents: 
70796diff
changeset | 542 | def is_afp_bulky: Boolean = is_afp && groups.exists(AFP.groups_bulky.contains) | 
| 62631 | 543 | } | 
| 544 | ||
| 75393 | 545 | def make_info( | 
| 546 | options: Options, | |
| 547 | dir_selected: Boolean, | |
| 548 | dir: Path, | |
| 549 | chapter: String, | |
| 550 | entry: Session_Entry | |
| 551 |   ): Info = {
 | |
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 552 |     try {
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 553 | val name = entry.name | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 554 | |
| 67284 | 555 |       if (exclude_session(name)) error("Bad session name")
 | 
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 556 |       if (is_pure(name) && entry.parent.isDefined) error("Illegal parent session")
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 557 |       if (!is_pure(name) && !entry.parent.isDefined) error("Missing parent session")
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 558 | |
| 70672 
e4bba654d085
clarified session_directories: relative to session_path, with overlapping information;
 wenzelm parents: 
70671diff
changeset | 559 | val session_path = dir + Path.explode(entry.path) | 
| 70681 | 560 | val directories = entry.directories.map(dir => session_path + Path.explode(dir)) | 
| 70668 | 561 | |
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 562 | val session_options = options ++ entry.options | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 563 | |
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 564 | val theories = | 
| 67216 | 565 |         entry.theories.map({ case (opts, thys) =>
 | 
| 566 | (session_options ++ opts, | |
| 567 |             thys.map({ case ((thy, pos), _) =>
 | |
| 67284 | 568 | if (exclude_theory(thy)) | 
| 67216 | 569 |                 error("Bad theory name " + quote(thy) + Position.here(pos))
 | 
| 570 | else (thy, pos) })) }) | |
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 571 | |
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 572 | val global_theories = | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 573 |         for { (_, thys) <- entry.theories; ((thy, pos), global) <- thys if global }
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 574 |         yield {
 | 
| 69366 | 575 | val thy_name = Path.explode(thy).file_name | 
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 576 | if (Long_Name.is_qualified(thy_name)) | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 577 |             error("Bad qualified name for global theory " +
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 578 | quote(thy_name) + Position.here(pos)) | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 579 | else thy_name | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 580 | } | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 581 | |
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 582 | val conditions = | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 583 |         theories.flatMap(thys => space_explode(',', thys._1.string("condition"))).distinct.sorted.
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 584 | map(x => (x, Isabelle_System.getenv(x) != "")) | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 585 | |
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 586 | val document_files = | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 587 |         entry.document_files.map({ case (s1, s2) => (Path.explode(s1), Path.explode(s2)) })
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 588 | |
| 68292 | 589 | val export_files = | 
| 69671 | 590 |         entry.export_files.map({ case (dir, prune, pats) => (Path.explode(dir), prune, pats) })
 | 
| 68292 | 591 | |
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 592 | val meta_digest = | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 593 | SHA1.digest( | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 594 | (name, chapter, entry.parent, entry.directories, entry.options, entry.imports, | 
| 72666 
945cee776e79
proper meta_digest: avoid non-portable position information;
 wenzelm parents: 
72653diff
changeset | 595 | entry.theories_no_position, conditions, entry.document_theories_no_position, | 
| 
945cee776e79
proper meta_digest: avoid non-portable position information;
 wenzelm parents: 
72653diff
changeset | 596 | entry.document_files) | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 597 | .toString) | 
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 598 | |
| 70672 
e4bba654d085
clarified session_directories: relative to session_path, with overlapping information;
 wenzelm parents: 
70671diff
changeset | 599 | Info(name, chapter, dir_selected, entry.pos, entry.groups, session_path, | 
| 70668 | 600 | entry.parent, entry.description, directories, session_options, | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 601 | entry.imports, theories, global_theories, entry.document_theories, document_files, | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 602 | export_files, meta_digest) | 
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 603 | } | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 604 |     catch {
 | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 605 | case ERROR(msg) => | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 606 | error(msg + "\nThe error(s) above occurred in session entry " + | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 607 | quote(entry.name) + Position.here(entry.pos)) | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 608 | } | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 609 | } | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 610 | |
| 75393 | 611 |   object Selection {
 | 
| 65422 | 612 | val empty: Selection = Selection() | 
| 65525 | 613 | val all: Selection = Selection(all_sessions = true) | 
| 70788 | 614 | def session(session: String): Selection = Selection(sessions = List(session)) | 
| 65419 | 615 | } | 
| 616 | ||
| 617 | sealed case class Selection( | |
| 618 | requirements: Boolean = false, | |
| 619 | all_sessions: Boolean = false, | |
| 66737 
2edc0c42c883
option -B for "isabelle build" and "isabelle imports";
 wenzelm parents: 
66736diff
changeset | 620 | base_sessions: List[String] = Nil, | 
| 65419 | 621 | exclude_session_groups: List[String] = Nil, | 
| 622 | exclude_sessions: List[String] = Nil, | |
| 623 | session_groups: List[String] = Nil, | |
| 75393 | 624 | sessions: List[String] = Nil | 
| 625 |   ) {
 | |
| 66736 | 626 | def ++ (other: Selection): Selection = | 
| 65422 | 627 | Selection( | 
| 628 | requirements = requirements || other.requirements, | |
| 629 | all_sessions = all_sessions || other.all_sessions, | |
| 66737 
2edc0c42c883
option -B for "isabelle build" and "isabelle imports";
 wenzelm parents: 
66736diff
changeset | 630 | base_sessions = Library.merge(base_sessions, other.base_sessions), | 
| 66736 | 631 | exclude_session_groups = Library.merge(exclude_session_groups, other.exclude_session_groups), | 
| 632 | exclude_sessions = Library.merge(exclude_sessions, other.exclude_sessions), | |
| 633 | session_groups = Library.merge(session_groups, other.session_groups), | |
| 634 | sessions = Library.merge(sessions, other.sessions)) | |
| 65419 | 635 | } | 
| 636 | ||
| 75393 | 637 |   object Structure {
 | 
| 72858 
cb0c407fbc6e
added "isabelle log": print messages from build database;
 wenzelm parents: 
72855diff
changeset | 638 | val empty: Structure = make(Nil) | 
| 
cb0c407fbc6e
added "isabelle log": print messages from build database;
 wenzelm parents: 
72855diff
changeset | 639 | |
| 75393 | 640 |     def make(infos: List[Info]): Structure = {
 | 
| 641 | def add_edges( | |
| 642 | graph: Graph[String, Info], | |
| 643 | kind: String, | |
| 644 | edges: Info => Iterable[String] | |
| 645 |       ) : Graph[String, Info] = {
 | |
| 646 |         def add_edge(pos: Position.T, name: String, g: Graph[String, Info], parent: String) = {
 | |
| 72855 | 647 | if (!g.defined(parent)) | 
| 648 |             error("Bad " + kind + " session " + quote(parent) + " for " +
 | |
| 649 | quote(name) + Position.here(pos)) | |
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 650 | |
| 72855 | 651 |           try { g.add_edge_acyclic(parent, name) }
 | 
| 652 |           catch {
 | |
| 653 | case exn: Graph.Cycles[_] => | |
| 654 | error(cat_lines(exn.cycles.map(cycle => | |
| 655 | "Cyclic session dependency of " + | |
| 656 |                   cycle.map(c => quote(c.toString)).mkString(" via "))) + Position.here(pos))
 | |
| 657 | } | |
| 658 | } | |
| 73359 | 659 |         graph.iterator.foldLeft(graph) {
 | 
| 660 | case (g, (name, (info, _))) => | |
| 661 | edges(info).foldLeft(g)(add_edge(info.pos, name, _, _)) | |
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 662 | } | 
| 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 663 | } | 
| 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 664 | |
| 72855 | 665 | val info_graph = | 
| 73359 | 666 |         infos.foldLeft(Graph.string[Info]) {
 | 
| 72855 | 667 | case (graph, info) => | 
| 668 | if (graph.defined(info.name)) | |
| 669 |               error("Duplicate session " + quote(info.name) + Position.here(info.pos) +
 | |
| 670 | Position.here(graph.get_node(info.name).pos)) | |
| 671 | else graph.new_node(info.name, info) | |
| 672 | } | |
| 673 | val build_graph = add_edges(info_graph, "parent", _.parent) | |
| 674 | val imports_graph = add_edges(build_graph, "imports", _.imports) | |
| 62631 | 675 | |
| 72855 | 676 | val session_positions: List[(String, Position.T)] = | 
| 677 | (for ((name, (info, _)) <- info_graph.iterator) yield (name, info.pos)).toList | |
| 70683 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 678 | |
| 72855 | 679 | val session_directories: Map[JFile, String] = | 
| 73359 | 680 |         (for {
 | 
| 681 | session <- imports_graph.topological_order.iterator | |
| 682 | info = info_graph.get_node(session) | |
| 683 | dir <- info.dirs.iterator | |
| 684 |         } yield (info, dir)).foldLeft(Map.empty[JFile, String]) {
 | |
| 685 | case (dirs, (info, dir)) => | |
| 72855 | 686 | val session = info.name | 
| 687 | val canonical_dir = dir.canonical_file | |
| 688 |               dirs.get(canonical_dir) match {
 | |
| 689 | case Some(session1) => | |
| 690 | val info1 = info_graph.get_node(session1) | |
| 691 |                   error("Duplicate use of directory " + dir +
 | |
| 692 | "\n for session " + quote(session1) + Position.here(info1.pos) + | |
| 693 | "\n vs. session " + quote(session) + Position.here(info.pos)) | |
| 694 | case None => dirs + (canonical_dir -> session) | |
| 695 | } | |
| 73359 | 696 | } | 
| 70671 
cb1776c8e216
clarified signature: retain global session information, unaffected by later restriction;
 wenzelm parents: 
70668diff
changeset | 697 | |
| 72855 | 698 | val global_theories: Map[String, String] = | 
| 73359 | 699 |         (for {
 | 
| 700 | session <- imports_graph.topological_order.iterator | |
| 701 | info = info_graph.get_node(session) | |
| 702 | thy <- info.global_theories.iterator } | |
| 703 |           yield (info, thy)).foldLeft(Thy_Header.bootstrap_global_theories.toMap) {
 | |
| 704 | case (global, (info, thy)) => | |
| 72855 | 705 | val qualifier = info.name | 
| 706 |               global.get(thy) match {
 | |
| 707 | case Some(qualifier1) if qualifier != qualifier1 => | |
| 708 |                   error("Duplicate global theory " + quote(thy) + Position.here(info.pos))
 | |
| 709 | case _ => global + (thy -> qualifier) | |
| 710 | } | |
| 73359 | 711 | } | 
| 70671 
cb1776c8e216
clarified signature: retain global session information, unaffected by later restriction;
 wenzelm parents: 
70668diff
changeset | 712 | |
| 72855 | 713 | new Structure( | 
| 714 | session_positions, session_directories, global_theories, build_graph, imports_graph) | |
| 715 | } | |
| 62631 | 716 | } | 
| 717 | ||
| 67052 | 718 | final class Structure private[Sessions]( | 
| 75393 | 719 | val session_positions: List[(String, Position.T)], | 
| 720 | val session_directories: Map[JFile, String], | |
| 721 | val global_theories: Map[String, String], | |
| 722 | val build_graph: Graph[String, Info], | |
| 723 | val imports_graph: Graph[String, Info] | |
| 724 |   ) {
 | |
| 69524 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 725 | sessions_structure => | 
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 726 | |
| 72845 | 727 | def bootstrap: Base = | 
| 728 | Base( | |
| 729 | session_directories = session_directories, | |
| 730 | global_theories = global_theories, | |
| 731 | overall_syntax = Thy_Header.bootstrap_syntax) | |
| 732 | ||
| 70683 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 733 | def dest_session_directories: List[(String, String)] = | 
| 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 734 | for ((file, session) <- session_directories.toList) | 
| 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 735 | yield (File.standard_path(file), session) | 
| 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 736 | |
| 69762 
58fb0d779583
support for session information via virtual file-system;
 wenzelm parents: 
69671diff
changeset | 737 | lazy val chapters: SortedMap[String, List[Info]] = | 
| 73359 | 738 |       build_graph.iterator.foldLeft(SortedMap.empty[String, List[Info]]) {
 | 
| 739 | case (chs, (_, (info, _))) => | |
| 740 | chs + (info.chapter -> (info :: chs.getOrElse(info.chapter, Nil))) | |
| 741 | } | |
| 69762 
58fb0d779583
support for session information via virtual file-system;
 wenzelm parents: 
69671diff
changeset | 742 | |
| 66823 | 743 | def build_graph_display: Graph_Display.Graph = Graph_Display.make_graph(build_graph) | 
| 744 | def imports_graph_display: Graph_Display.Graph = Graph_Display.make_graph(imports_graph) | |
| 745 | ||
| 67025 
961285f581e6
clarifified selection: always wrt. build_graph structure;
 wenzelm parents: 
67024diff
changeset | 746 | def defined(name: String): Boolean = imports_graph.defined(name) | 
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 747 | def apply(name: String): Info = imports_graph.get_node(name) | 
| 67025 
961285f581e6
clarifified selection: always wrt. build_graph structure;
 wenzelm parents: 
67024diff
changeset | 748 | def get(name: String): Option[Info] = if (defined(name)) Some(apply(name)) else None | 
| 62631 | 749 | |
| 70859 
6e6254bbce1f
split into standard partitions, for improved scalability;
 wenzelm parents: 
70796diff
changeset | 750 | def theory_qualifier(name: String): String = | 
| 
6e6254bbce1f
split into standard partitions, for improved scalability;
 wenzelm parents: 
70796diff
changeset | 751 | global_theories.getOrElse(name, Long_Name.qualifier(name)) | 
| 
6e6254bbce1f
split into standard partitions, for improved scalability;
 wenzelm parents: 
70796diff
changeset | 752 | |
| 75393 | 753 |     def check_sessions(names: List[String]): Unit = {
 | 
| 71601 | 754 | val bad_sessions = SortedSet(names.filterNot(defined): _*).toList | 
| 68542 | 755 | if (bad_sessions.nonEmpty) | 
| 756 |         error("Undefined session(s): " + commas_quote(bad_sessions))
 | |
| 757 | } | |
| 758 | ||
| 68733 | 759 | def check_sessions(sel: Selection): Unit = | 
| 760 | check_sessions(sel.base_sessions ::: sel.exclude_sessions ::: sel.sessions) | |
| 761 | ||
| 75393 | 762 |     private def selected(graph: Graph[String, Info], sel: Selection): List[String] = {
 | 
| 68733 | 763 | check_sessions(sel) | 
| 764 | ||
| 68732 | 765 | val select_group = sel.session_groups.toSet | 
| 68734 
c14a2cc9b5ef
isabelle build options -c -x -B refer to imports_graph;
 wenzelm parents: 
68733diff
changeset | 766 | val select_session = sel.sessions.toSet ++ imports_graph.all_succs(sel.base_sessions) | 
| 68732 | 767 | |
| 768 | val selected0 = | |
| 769 | if (sel.all_sessions) graph.keys | |
| 770 |         else {
 | |
| 771 |           (for {
 | |
| 772 | (name, (info, _)) <- graph.iterator | |
| 74810 | 773 | if info.dir_selected || select_session(name) || info.groups.exists(select_group) | 
| 68732 | 774 | } yield name).toList | 
| 775 | } | |
| 776 | ||
| 777 | if (sel.requirements) (graph.all_preds(selected0).toSet -- selected0).toList | |
| 778 | else selected0 | |
| 779 | } | |
| 780 | ||
| 75393 | 781 |     def selection(sel: Selection): Structure = {
 | 
| 68733 | 782 | check_sessions(sel) | 
| 67025 
961285f581e6
clarifified selection: always wrt. build_graph structure;
 wenzelm parents: 
67024diff
changeset | 783 | |
| 75393 | 784 |       val excluded = {
 | 
| 68732 | 785 | val exclude_group = sel.exclude_session_groups.toSet | 
| 786 | val exclude_group_sessions = | |
| 787 |           (for {
 | |
| 68734 
c14a2cc9b5ef
isabelle build options -c -x -B refer to imports_graph;
 wenzelm parents: 
68733diff
changeset | 788 | (name, (info, _)) <- imports_graph.iterator | 
| 
c14a2cc9b5ef
isabelle build options -c -x -B refer to imports_graph;
 wenzelm parents: 
68733diff
changeset | 789 | if imports_graph.get_node(name).groups.exists(exclude_group) | 
| 68732 | 790 | } yield name).toList | 
| 68734 
c14a2cc9b5ef
isabelle build options -c -x -B refer to imports_graph;
 wenzelm parents: 
68733diff
changeset | 791 | imports_graph.all_succs(exclude_group_sessions ::: sel.exclude_sessions).toSet | 
| 68732 | 792 | } | 
| 67027 | 793 | |
| 75393 | 794 |       def restrict(graph: Graph[String, Info]): Graph[String, Info] = {
 | 
| 68732 | 795 | val sessions = graph.all_preds(selected(graph, sel)).filterNot(excluded) | 
| 67029 | 796 | graph.restrict(graph.all_preds(sessions).toSet) | 
| 67025 
961285f581e6
clarifified selection: always wrt. build_graph structure;
 wenzelm parents: 
67024diff
changeset | 797 | } | 
| 
961285f581e6
clarifified selection: always wrt. build_graph structure;
 wenzelm parents: 
67024diff
changeset | 798 | |
| 70671 
cb1776c8e216
clarified signature: retain global session information, unaffected by later restriction;
 wenzelm parents: 
70668diff
changeset | 799 | new Structure( | 
| 70683 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 800 | session_positions, session_directories, global_theories, | 
| 
8c7706b053c7
find theory files via session structure: much faster Prover IDE startup;
 wenzelm parents: 
70681diff
changeset | 801 | restrict(build_graph), restrict(imports_graph)) | 
| 62631 | 802 | } | 
| 803 | ||
| 72571 | 804 | def selection(session: String): Structure = selection(Selection.session(session)) | 
| 805 | ||
| 69524 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 806 | def selection_deps( | 
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 807 | selection: Selection, | 
| 71726 
a5fda30edae2
clarified signature: more uniform treatment of stopped/interrupted state;
 wenzelm parents: 
71642diff
changeset | 808 | progress: Progress = new Progress, | 
| 69524 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 809 | loading_sessions: Boolean = false, | 
| 68304 | 810 | inlined_files: Boolean = false, | 
| 75393 | 811 | verbose: Boolean = false | 
| 812 |     ): Deps = {
 | |
| 69524 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 813 | val deps = | 
| 70869 | 814 | Sessions.deps(sessions_structure.selection(selection), | 
| 69524 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 815 | progress = progress, inlined_files = inlined_files, verbose = verbose) | 
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 816 | |
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 817 |       if (loading_sessions) {
 | 
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 818 | val selection_size = deps.sessions_structure.build_graph.size | 
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 819 |         if (selection_size > 1) progress.echo("Loading " + selection_size + " sessions ...")
 | 
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 820 | } | 
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 821 | |
| 
fa94f2b2a877
clarified sessions_deps, according to Isabelle/MMT usage;
 wenzelm parents: 
69393diff
changeset | 822 | deps | 
| 68304 | 823 | } | 
| 824 | ||
| 68732 | 825 | def build_selection(sel: Selection): List[String] = selected(build_graph, sel) | 
| 67029 | 826 | def build_descendants(ss: List[String]): List[String] = build_graph.all_succs(ss) | 
| 72065 
11dc8929832d
clarified order --- proper sorting of requirements;
 wenzelm parents: 
72064diff
changeset | 827 | def build_requirements(ss: List[String]): List[String] = build_graph.all_preds_rev(ss) | 
| 67029 | 828 | def build_topological_order: List[String] = build_graph.topological_order | 
| 74809 | 829 | def build_hierarchy(session: String): List[String] = build_graph.all_preds(List(session)) | 
| 62631 | 830 | |
| 68732 | 831 | def imports_selection(sel: Selection): List[String] = selected(imports_graph, sel) | 
| 67029 | 832 | def imports_descendants(ss: List[String]): List[String] = imports_graph.all_succs(ss) | 
| 72065 
11dc8929832d
clarified order --- proper sorting of requirements;
 wenzelm parents: 
72064diff
changeset | 833 | def imports_requirements(ss: List[String]): List[String] = imports_graph.all_preds_rev(ss) | 
| 67029 | 834 | def imports_topological_order: List[String] = imports_graph.topological_order | 
| 74809 | 835 | def imports_hierarchy(session: String): List[String] = imports_graph.all_preds(List(session)) | 
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 836 | |
| 72613 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 837 | def bibtex_entries: List[(String, List[String])] = | 
| 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 838 | build_topological_order.flatMap(name => | 
| 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 839 |         apply(name).bibtex_entries match {
 | 
| 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 840 | case Nil => None | 
| 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 841 | case entries => Some(name -> entries.map(_.info)) | 
| 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 842 | }) | 
| 
d01ea9e3bd2d
clarified bibtex_entries: refer to overall session structure;
 wenzelm parents: 
72604diff
changeset | 843 | |
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 844 | override def toString: String = | 
| 67052 | 845 |       imports_graph.keys_iterator.mkString("Sessions.Structure(", ", ", ")")
 | 
| 62631 | 846 | } | 
| 847 | ||
| 848 | ||
| 849 | /* parser */ | |
| 850 | ||
| 851 | private val CHAPTER = "chapter" | |
| 852 | private val SESSION = "session" | |
| 853 | private val IN = "in" | |
| 854 | private val DESCRIPTION = "description" | |
| 70668 | 855 | private val DIRECTORIES = "directories" | 
| 62631 | 856 | private val OPTIONS = "options" | 
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 857 | private val SESSIONS = "sessions" | 
| 62631 | 858 | private val THEORIES = "theories" | 
| 65374 | 859 | private val GLOBAL = "global" | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 860 | private val DOCUMENT_THEORIES = "document_theories" | 
| 62631 | 861 | private val DOCUMENT_FILES = "document_files" | 
| 68292 | 862 | private val EXPORT_FILES = "export_files" | 
| 62631 | 863 | |
| 71601 | 864 | val root_syntax: Outer_Syntax = | 
| 70668 | 865 |     Outer_Syntax.empty + "(" + ")" + "+" + "," + "=" + "[" + "]" +
 | 
| 70681 | 866 | GLOBAL + IN + | 
| 63443 | 867 | (CHAPTER, Keyword.THY_DECL) + | 
| 868 | (SESSION, Keyword.THY_DECL) + | |
| 869 | (DESCRIPTION, Keyword.QUASI_COMMAND) + | |
| 70668 | 870 | (DIRECTORIES, Keyword.QUASI_COMMAND) + | 
| 63443 | 871 | (OPTIONS, Keyword.QUASI_COMMAND) + | 
| 65420 
695d4e22345a
support for static session imports, without affect build hierarchy;
 wenzelm parents: 
65419diff
changeset | 872 | (SESSIONS, Keyword.QUASI_COMMAND) + | 
| 63443 | 873 | (THEORIES, Keyword.QUASI_COMMAND) + | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 874 | (DOCUMENT_THEORIES, Keyword.QUASI_COMMAND) + | 
| 68292 | 875 | (DOCUMENT_FILES, Keyword.QUASI_COMMAND) + | 
| 876 | (EXPORT_FILES, Keyword.QUASI_COMMAND) | |
| 62631 | 877 | |
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 878 | abstract class Entry | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 879 | sealed case class Chapter(name: String) extends Entry | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 880 | sealed case class Session_Entry( | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 881 | pos: Position.T, | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 882 | name: String, | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 883 | groups: List[String], | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 884 | path: String, | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 885 | parent: Option[String], | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 886 | description: String, | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 887 | options: List[Options.Spec], | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 888 | imports: List[String], | 
| 70681 | 889 | directories: List[String], | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 890 | theories: List[(List[Options.Spec], List[((String, Position.T), Boolean)])], | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 891 | document_theories: List[(String, Position.T)], | 
| 68292 | 892 | document_files: List[(String, String)], | 
| 75393 | 893 | export_files: List[(String, Int, List[String])] | 
| 894 |   ) extends Entry {
 | |
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 895 | def theories_no_position: List[(List[Options.Spec], List[(String, Boolean)])] = | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 896 |       theories.map({ case (a, b) => (a, b.map({ case ((c, _), d) => (c, d) })) })
 | 
| 72666 
945cee776e79
proper meta_digest: avoid non-portable position information;
 wenzelm parents: 
72653diff
changeset | 897 | def document_theories_no_position: List[String] = | 
| 
945cee776e79
proper meta_digest: avoid non-portable position information;
 wenzelm parents: 
72653diff
changeset | 898 | document_theories.map(_._1) | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 899 | } | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 900 | |
| 75405 | 901 |   private object Parsers extends Options.Parsers {
 | 
| 75393 | 902 |     private val chapter: Parser[Chapter] = {
 | 
| 62631 | 903 |       val chapter_name = atom("chapter name", _.is_name)
 | 
| 904 | ||
| 905 |       command(CHAPTER) ~! chapter_name ^^ { case _ ~ a => Chapter(a) }
 | |
| 906 | } | |
| 907 | ||
| 75393 | 908 |     private val session_entry: Parser[Session_Entry] = {
 | 
| 62631 | 909 | val option = | 
| 67210 | 910 |         option_name ~ opt($$$("=") ~! option_value ^^ { case _ ~ x => x }) ^^
 | 
| 911 |           { case x ~ y => (x, y) }
 | |
| 62631 | 912 |       val options = $$$("[") ~> rep1sep(option, $$$(",")) <~ $$$("]")
 | 
| 913 | ||
| 65374 | 914 | val theory_entry = | 
| 70668 | 915 |         position(theory_name) ~ opt_keyword(GLOBAL) ^^ { case x ~ y => (x, y) }
 | 
| 65374 | 916 | |
| 62631 | 917 | val theories = | 
| 65374 | 918 | $$$(THEORIES) ~! | 
| 66970 
13857f49d215
clarified ROOT syntax: 'sessions' and 'theories' are optional, but need to be non-empty;
 wenzelm parents: 
66969diff
changeset | 919 | ((options | success(Nil)) ~ rep1(theory_entry)) ^^ | 
| 65374 | 920 |           { case _ ~ (x ~ y) => (x, y) }
 | 
| 62631 | 921 | |
| 68292 | 922 |       val in_path = $$$("(") ~! ($$$(IN) ~ path ~ $$$(")")) ^^ { case _ ~ (_ ~ x ~ _) => x }
 | 
| 923 | ||
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 924 | val document_theories = | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 925 |         $$$(DOCUMENT_THEORIES) ~! rep1(position(name)) ^^ { case _ ~ x => x }
 | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 926 | |
| 62631 | 927 | val document_files = | 
| 928 | $$$(DOCUMENT_FILES) ~! | |
| 68292 | 929 |           ((in_path | success("document")) ~ rep1(path)) ^^ { case _ ~ (x ~ y) => y.map((x, _)) }
 | 
| 930 | ||
| 69671 | 931 |       val prune = $$$("[") ~! (nat ~ $$$("]")) ^^ { case _ ~ (x ~ _) => x } | success(0)
 | 
| 932 | ||
| 68292 | 933 | val export_files = | 
| 69671 | 934 |         $$$(EXPORT_FILES) ~! ((in_path | success("export")) ~ prune ~ rep1(embedded)) ^^
 | 
| 935 |           { case _ ~ (x ~ y ~ z) => (x, y, z) }
 | |
| 62631 | 936 | |
| 937 | command(SESSION) ~! | |
| 938 | (position(session_name) ~ | |
| 939 |           (($$$("(") ~! (rep1(name) <~ $$$(")")) ^^ { case _ ~ x => x }) | success(Nil)) ~
 | |
| 940 |           (($$$(IN) ~! path ^^ { case _ ~ x => x }) | success(".")) ~
 | |
| 941 |           ($$$("=") ~!
 | |
| 942 |             (opt(session_name ~! $$$("+") ^^ { case x ~ _ => x }) ~
 | |
| 943 |               (($$$(DESCRIPTION) ~! text ^^ { case _ ~ x => x }) | success("")) ~
 | |
| 944 |               (($$$(OPTIONS) ~! options ^^ { case _ ~ x => x }) | success(Nil)) ~
 | |
| 66970 
13857f49d215
clarified ROOT syntax: 'sessions' and 'theories' are optional, but need to be non-empty;
 wenzelm parents: 
66969diff
changeset | 945 |               (($$$(SESSIONS) ~! rep1(session_name)  ^^ { case _ ~ x => x }) | success(Nil)) ~
 | 
| 70681 | 946 |               (($$$(DIRECTORIES) ~! rep1(path) ^^ { case _ ~ x => x }) | success(Nil)) ~
 | 
| 66970 
13857f49d215
clarified ROOT syntax: 'sessions' and 'theories' are optional, but need to be non-empty;
 wenzelm parents: 
66969diff
changeset | 947 | rep(theories) ~ | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 948 | (opt(document_theories) ^^ (x => x.getOrElse(Nil))) ~ | 
| 68292 | 949 | (rep(document_files) ^^ (x => x.flatten)) ~ | 
| 72600 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 950 | rep(export_files)))) ^^ | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 951 |         { case _ ~ ((a, pos) ~ b ~ c ~ (_ ~ (d ~ e ~ f ~ g ~ h ~ i ~ j ~ k ~ l))) =>
 | 
| 
2fa4f25d9d07
official support for document theories from other sessions;
 wenzelm parents: 
72574diff
changeset | 952 | Session_Entry(pos, a, b, c, d, e, f, g, h, i, j, k, l) } | 
| 62631 | 953 | } | 
| 954 | ||
| 75393 | 955 |     def parse_root(path: Path): List[Entry] = {
 | 
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 956 | val toks = Token.explode(root_syntax.keywords, File.read(path)) | 
| 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 957 | val start = Token.Pos.file(path.implode) | 
| 62631 | 958 | |
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 959 |       parse_all(rep(chapter | session_entry), Token.reader(toks, start)) match {
 | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 960 | case Success(result, _) => result | 
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 961 | case bad => error(bad.toString) | 
| 62631 | 962 | } | 
| 963 | } | |
| 964 | } | |
| 965 | ||
| 75405 | 966 | def parse_root(path: Path): List[Entry] = Parsers.parse_root(path) | 
| 66819 | 967 | |
| 968 | def parse_root_entries(path: Path): List[Session_Entry] = | |
| 75405 | 969 | for (entry <- Parsers.parse_root(path) if entry.isInstanceOf[Session_Entry]) | 
| 66819 | 970 | yield entry.asInstanceOf[Session_Entry] | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 971 | |
| 75393 | 972 |   def read_root(options: Options, select: Boolean, path: Path): List[Info] = {
 | 
| 69762 
58fb0d779583
support for session information via virtual file-system;
 wenzelm parents: 
69671diff
changeset | 973 | var entry_chapter = UNSORTED | 
| 66960 | 974 | val infos = new mutable.ListBuffer[Info] | 
| 66819 | 975 |     parse_root(path).foreach {
 | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 976 | case Chapter(name) => entry_chapter = name | 
| 66967 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 977 | case entry: Session_Entry => | 
| 
e365c91c72a9
synthesize session with all required theories from other session imports;
 wenzelm parents: 
66966diff
changeset | 978 | infos += make_info(options, select, path.dir, entry_chapter, entry) | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 979 | } | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 980 | infos.toList | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 981 | } | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 982 | |
| 75393 | 983 |   def parse_roots(roots: Path): List[String] = {
 | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 984 |     for {
 | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 985 | line <- split_lines(File.read(roots)) | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 986 |       if !(line == "" || line.startsWith("#"))
 | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 987 | } yield line | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 988 | } | 
| 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 989 | |
| 62631 | 990 | |
| 62635 | 991 | /* load sessions from certain directories */ | 
| 62631 | 992 | |
| 73814 
c8b4a4f69068
clarified check (refining fc828f64da5b): etc/settings or etc/components is not strictly required according to "init_component", and notable components only have session ROOTS (e.g. AFP/thys);
 wenzelm parents: 
73718diff
changeset | 993 | def is_session_dir(dir: Path): Boolean = | 
| 62631 | 994 | (dir + ROOT).is_file || (dir + ROOTS).is_file | 
| 995 | ||
| 75297 
fc4d07587695
more robust errors -- on foreground process instead of background server;
 wenzelm parents: 
74812diff
changeset | 996 | def check_session_dir(dir: Path): Path = | 
| 65468 
c41791ad75c3
early check and normalization of session directory, e.g. relevant for path information passed to ML process, which may have a different CWD;
 wenzelm parents: 
65463diff
changeset | 997 | if (is_session_dir(dir)) File.pwd() + dir.expand | 
| 73311 | 998 |     else error("Bad session root directory (missing ROOT or ROOTS): " + dir.expand.toString)
 | 
| 62631 | 999 | |
| 75393 | 1000 |   def directories(dirs: List[Path], select_dirs: List[Path]): List[(Boolean, Path)] = {
 | 
| 73815 | 1001 | val default_dirs = Components.directories().filter(is_session_dir) | 
| 68746 
f95e2f145ea5
canonical session directories in correspondence to Known.files;
 wenzelm parents: 
68734diff
changeset | 1002 |     for { (select, dir) <- (default_dirs ::: dirs).map((false, _)) ::: select_dirs.map((true, _)) }
 | 
| 
f95e2f145ea5
canonical session directories in correspondence to Known.files;
 wenzelm parents: 
68734diff
changeset | 1003 | yield (select, dir.canonical) | 
| 65561 | 1004 | } | 
| 1005 | ||
| 75393 | 1006 | def load_structure( | 
| 1007 | options: Options, | |
| 66968 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 1008 | dirs: List[Path] = Nil, | 
| 
9991671c98aa
allow to augment session context via explicit session infos;
 wenzelm parents: 
66967diff
changeset | 1009 | select_dirs: List[Path] = Nil, | 
| 75393 | 1010 | infos: List[Info] = Nil | 
| 1011 |   ): Structure = {
 | |
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1012 | def load_dir(select: Boolean, dir: Path): List[(Boolean, Path)] = | 
| 62635 | 1013 | load_root(select, dir) ::: load_roots(select, dir) | 
| 62631 | 1014 | |
| 75393 | 1015 |     def load_root(select: Boolean, dir: Path): List[(Boolean, Path)] = {
 | 
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1016 | val root = dir + ROOT | 
| 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1017 | if (root.is_file) List((select, root)) else Nil | 
| 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1018 | } | 
| 62631 | 1019 | |
| 75393 | 1020 |     def load_roots(select: Boolean, dir: Path): List[(Boolean, Path)] = {
 | 
| 62631 | 1021 | val roots = dir + ROOTS | 
| 1022 |       if (roots.is_file) {
 | |
| 1023 |         for {
 | |
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 1024 | entry <- parse_roots(roots) | 
| 62631 | 1025 | dir1 = | 
| 66818 
5bc903a60932
clarified signature: public access to ROOT file syntax;
 wenzelm parents: 
66780diff
changeset | 1026 |             try { check_session_dir(dir + Path.explode(entry)) }
 | 
| 62631 | 1027 |             catch {
 | 
| 1028 | case ERROR(msg) => | |
| 1029 | error(msg + "\nThe error(s) above occurred in session catalog " + roots.toString) | |
| 1030 | } | |
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1031 | res <- load_dir(select, dir1) | 
| 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1032 | } yield res | 
| 62631 | 1033 | } | 
| 1034 | else Nil | |
| 1035 | } | |
| 1036 | ||
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1037 | val roots = | 
| 62631 | 1038 |       for {
 | 
| 65561 | 1039 | (select, dir) <- directories(dirs, select_dirs) | 
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1040 | res <- load_dir(select, check_session_dir(dir)) | 
| 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1041 | } yield res | 
| 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1042 | |
| 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1043 | val unique_roots = | 
| 73359 | 1044 |       roots.foldLeft(Map.empty[JFile, (Boolean, Path)]) {
 | 
| 1045 | case (m, (select, path)) => | |
| 1046 | val file = path.canonical_file | |
| 1047 |           m.get(file) match {
 | |
| 1048 | case None => m + (file -> (select, path)) | |
| 1049 | case Some((select1, path1)) => m + (file -> (select1 || select, path1)) | |
| 1050 | } | |
| 1051 | }.toList.map(_._2) | |
| 66764 
006deaf5c3dc
process ROOT files only once, which allows duplicate (or overlapping) session root directories;
 wenzelm parents: 
66759diff
changeset | 1052 | |
| 72855 | 1053 | Structure.make(unique_roots.flatMap(p => read_root(options, p._1, p._2)) ::: infos) | 
| 62631 | 1054 | } | 
| 62632 | 1055 | |
| 1056 | ||
| 71808 | 1057 | /* Isabelle tool wrapper */ | 
| 1058 | ||
| 72763 | 1059 |   val isabelle_tool = Isabelle_Tool("sessions", "explore structure of Isabelle sessions",
 | 
| 75394 | 1060 | Scala_Project.here, | 
| 1061 |     { args =>
 | |
| 1062 | var base_sessions: List[String] = Nil | |
| 1063 | var select_dirs: List[Path] = Nil | |
| 1064 | var requirements = false | |
| 1065 | var exclude_session_groups: List[String] = Nil | |
| 1066 | var all_sessions = false | |
| 1067 | var dirs: List[Path] = Nil | |
| 1068 | var session_groups: List[String] = Nil | |
| 1069 | var exclude_sessions: List[String] = Nil | |
| 71808 | 1070 | |
| 75394 | 1071 |       val getopts = Getopts("""
 | 
| 71808 | 1072 | Usage: isabelle sessions [OPTIONS] [SESSIONS ...] | 
| 1073 | ||
| 1074 | Options are: | |
| 1075 | -B NAME include session NAME and all descendants | |
| 1076 | -D DIR include session directory and select its sessions | |
| 1077 | -R refer to requirements of selected sessions | |
| 1078 | -X NAME exclude sessions from group NAME and all descendants | |
| 1079 | -a select all sessions | |
| 1080 | -d DIR include session directory | |
| 1081 | -g NAME select session group NAME | |
| 1082 | -x NAME exclude session NAME and all descendants | |
| 1083 | ||
| 1084 | Explore the structure of Isabelle sessions and print result names in | |
| 1085 | topological order (on stdout). | |
| 1086 | """, | |
| 75394 | 1087 | "B:" -> (arg => base_sessions = base_sessions ::: List(arg)), | 
| 1088 | "D:" -> (arg => select_dirs = select_dirs ::: List(Path.explode(arg))), | |
| 1089 | "R" -> (_ => requirements = true), | |
| 1090 | "X:" -> (arg => exclude_session_groups = exclude_session_groups ::: List(arg)), | |
| 1091 | "a" -> (_ => all_sessions = true), | |
| 1092 | "d:" -> (arg => dirs = dirs ::: List(Path.explode(arg))), | |
| 1093 | "g:" -> (arg => session_groups = session_groups ::: List(arg)), | |
| 1094 | "x:" -> (arg => exclude_sessions = exclude_sessions ::: List(arg))) | |
| 71808 | 1095 | |
| 75394 | 1096 | val sessions = getopts(args) | 
| 71808 | 1097 | |
| 75394 | 1098 | val options = Options.init() | 
| 71808 | 1099 | |
| 75394 | 1100 | val selection = | 
| 1101 | Selection(requirements = requirements, all_sessions = all_sessions, base_sessions = base_sessions, | |
| 1102 | exclude_session_groups = exclude_session_groups, exclude_sessions = exclude_sessions, | |
| 1103 | session_groups = session_groups, sessions = sessions) | |
| 1104 | val sessions_structure = | |
| 1105 | load_structure(options, dirs = dirs, select_dirs = select_dirs).selection(selection) | |
| 71808 | 1106 | |
| 75394 | 1107 |       for (name <- sessions_structure.imports_topological_order) {
 | 
| 1108 | Output.writeln(name, stdout = true) | |
| 1109 | } | |
| 1110 | }) | |
| 71808 | 1111 | |
| 1112 | ||
| 62637 
0189fe0f6452
support for Poly/ML heap hierarchy, which saves a lot of disk space;
 wenzelm parents: 
62636diff
changeset | 1113 | |
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1114 | /** heap file with SHA1 digest **/ | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1115 | |
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1116 | private val sha1_prefix = "SHA1:" | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1117 | |
| 75393 | 1118 |   def read_heap_digest(heap: Path): Option[String] = {
 | 
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1119 |     if (heap.is_file) {
 | 
| 75394 | 1120 |       using(FileChannel.open(heap.java_path, StandardOpenOption.READ)) { file =>
 | 
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1121 | val len = file.size | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1122 | val n = sha1_prefix.length + SHA1.digest_length | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1123 |         if (len >= n) {
 | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1124 | file.position(len - n) | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1125 | |
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1126 | val buf = ByteBuffer.allocate(n) | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1127 | var i = 0 | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1128 | var m = 0 | 
| 75382 
81673c441ce3
tuned: eliminted do-while for the sake of scala3;
 wenzelm parents: 
75310diff
changeset | 1129 | var cont = true | 
| 
81673c441ce3
tuned: eliminted do-while for the sake of scala3;
 wenzelm parents: 
75310diff
changeset | 1130 |           while (cont) {
 | 
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1131 | m = file.read(buf) | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1132 | if (m != -1) i += m | 
| 75382 
81673c441ce3
tuned: eliminted do-while for the sake of scala3;
 wenzelm parents: 
75310diff
changeset | 1133 | cont = (m != -1 && n > i) | 
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1134 | } | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1135 | |
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1136 |           if (i == n) {
 | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1137 | val prefix = new String(buf.array(), 0, sha1_prefix.length, UTF8.charset) | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1138 | val s = new String(buf.array(), sha1_prefix.length, SHA1.digest_length, UTF8.charset) | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1139 | if (prefix == sha1_prefix) Some(s) else None | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1140 | } | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1141 | else None | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1142 | } | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1143 | else None | 
| 75394 | 1144 | } | 
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1145 | } | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1146 | else None | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1147 | } | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1148 | |
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1149 | def write_heap_digest(heap: Path): String = | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1150 |     read_heap_digest(heap) match {
 | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1151 | case None => | 
| 75310 | 1152 | val s = SHA1.digest(heap).toString | 
| 62704 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1153 | File.append(heap, sha1_prefix + s) | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1154 | s | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1155 | case Some(s) => s | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1156 | } | 
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1157 | |
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1158 | |
| 
478b49f0d726
proper SHA1 digest as annex to heap file: Poly/ML reads precise segment length;
 wenzelm parents: 
62637diff
changeset | 1159 | |
| 62637 
0189fe0f6452
support for Poly/ML heap hierarchy, which saves a lot of disk space;
 wenzelm parents: 
62636diff
changeset | 1160 | /** persistent store **/ | 
| 62632 | 1161 | |
| 75393 | 1162 |   object Session_Info {
 | 
| 66857 | 1163 |     val session_name = SQL.Column.string("session_name").make_primary_key
 | 
| 65326 | 1164 | |
| 65296 | 1165 | // Build_Log.Session_Info | 
| 1166 |     val session_timing = SQL.Column.bytes("session_timing")
 | |
| 1167 |     val command_timings = SQL.Column.bytes("command_timings")
 | |
| 66873 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1168 |     val theory_timings = SQL.Column.bytes("theory_timings")
 | 
| 65296 | 1169 |     val ml_statistics = SQL.Column.bytes("ml_statistics")
 | 
| 1170 |     val task_statistics = SQL.Column.bytes("task_statistics")
 | |
| 65934 | 1171 |     val errors = SQL.Column.bytes("errors")
 | 
| 65296 | 1172 | val build_log_columns = | 
| 66873 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1173 | List(session_name, session_timing, command_timings, theory_timings, | 
| 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1174 | ml_statistics, task_statistics, errors) | 
| 65296 | 1175 | |
| 1176 | // Build.Session_Info | |
| 1177 |     val sources = SQL.Column.string("sources")
 | |
| 1178 |     val input_heaps = SQL.Column.string("input_heaps")
 | |
| 1179 |     val output_heap = SQL.Column.string("output_heap")
 | |
| 1180 |     val return_code = SQL.Column.int("return_code")
 | |
| 66746 | 1181 | val build_columns = List(sources, input_heaps, output_heap, return_code) | 
| 65296 | 1182 | |
| 1183 |     val table = SQL.Table("isabelle_session_info", build_log_columns ::: build_columns)
 | |
| 1184 | } | |
| 1185 | ||
| 72682 | 1186 | class Database_Context private[Sessions]( | 
| 1187 | val store: Sessions.Store, | |
| 75393 | 1188 | database_server: Option[SQL.Database] | 
| 1189 |   ) extends AutoCloseable {
 | |
| 74731 
161e84e6b40a
just one cache, via HTML_Context, via Sessions.Store or Session;
 wenzelm parents: 
74696diff
changeset | 1190 | def cache: Term.Cache = store.cache | 
| 72683 | 1191 | |
| 73367 | 1192 | def close(): Unit = database_server.foreach(_.close()) | 
| 72682 | 1193 | |
| 72715 
2615b8c05337
clarified signature --- avoid repeated open_database on server;
 wenzelm parents: 
72696diff
changeset | 1194 | def output_database[A](session: String)(f: SQL.Database => A): A = | 
| 
2615b8c05337
clarified signature --- avoid repeated open_database on server;
 wenzelm parents: 
72696diff
changeset | 1195 |       database_server match {
 | 
| 
2615b8c05337
clarified signature --- avoid repeated open_database on server;
 wenzelm parents: 
72696diff
changeset | 1196 | case Some(db) => f(db) | 
| 
2615b8c05337
clarified signature --- avoid repeated open_database on server;
 wenzelm parents: 
72696diff
changeset | 1197 | case None => using(store.open_database(session, output = true))(f) | 
| 
2615b8c05337
clarified signature --- avoid repeated open_database on server;
 wenzelm parents: 
72696diff
changeset | 1198 | } | 
| 
2615b8c05337
clarified signature --- avoid repeated open_database on server;
 wenzelm parents: 
72696diff
changeset | 1199 | |
| 72716 | 1200 | def input_database[A](session: String)(f: (SQL.Database, String) => Option[A]): Option[A] = | 
| 1201 |       database_server match {
 | |
| 1202 | case Some(db) => f(db, session) | |
| 1203 | case None => | |
| 1204 |           store.try_open_database(session) match {
 | |
| 1205 | case Some(db) => using(db)(f(_, session)) | |
| 1206 | case None => None | |
| 1207 | } | |
| 1208 | } | |
| 1209 | ||
| 72854 | 1210 | def read_export( | 
| 75393 | 1211 | sessions: List[String], | 
| 1212 | theory_name: String, | |
| 1213 | name: String | |
| 1214 |     ): Option[Export.Entry] = {
 | |
| 72682 | 1215 | val attempts = | 
| 1216 |         database_server match {
 | |
| 1217 | case Some(db) => | |
| 72854 | 1218 | sessions.view.map(session_name => | 
| 73031 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1219 | Export.read_entry(db, store.cache, session_name, theory_name, name)) | 
| 72682 | 1220 | case None => | 
| 72854 | 1221 | sessions.view.map(session_name => | 
| 72682 | 1222 |               store.try_open_database(session_name) match {
 | 
| 72847 
9dda93a753b1
clarified signature: provide XZ.Cache where Export.Entry is created;
 wenzelm parents: 
72845diff
changeset | 1223 | case Some(db) => | 
| 73031 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1224 | using(db)(Export.read_entry(_, store.cache, session_name, theory_name, name)) | 
| 72682 | 1225 | case None => None | 
| 1226 | }) | |
| 1227 | } | |
| 1228 |       attempts.collectFirst({ case Some(entry) => entry })
 | |
| 1229 | } | |
| 1230 | ||
| 72854 | 1231 | def get_export( | 
| 1232 | session_hierarchy: List[String], theory_name: String, name: String): Export.Entry = | |
| 1233 | read_export(session_hierarchy, theory_name, name) getOrElse | |
| 1234 | Export.empty_entry(theory_name, name) | |
| 72682 | 1235 | |
| 75393 | 1236 |     override def toString: String = {
 | 
| 72682 | 1237 | val s = | 
| 1238 |         database_server match {
 | |
| 1239 | case Some(db) => db.toString | |
| 1240 |           case None => "input_dirs = " + store.input_dirs.map(_.absolute).mkString(", ")
 | |
| 1241 | } | |
| 1242 |       "Database_Context(" + s + ")"
 | |
| 1243 | } | |
| 1244 | } | |
| 1245 | ||
| 74731 
161e84e6b40a
just one cache, via HTML_Context, via Sessions.Store or Session;
 wenzelm parents: 
74696diff
changeset | 1246 | def store(options: Options, cache: Term.Cache = Term.Cache.make()): Store = | 
| 73031 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1247 | new Store(options, cache) | 
| 62632 | 1248 | |
| 75393 | 1249 |   class Store private[Sessions](val options: Options, val cache: Term.Cache) {
 | 
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1250 | store => | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1251 | |
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1252 | override def toString: String = "Store(output_dir = " + output_dir.absolute + ")" | 
| 68219 | 1253 | |
| 1254 | ||
| 1255 | /* directories */ | |
| 1256 | ||
| 68523 
ccacc84e0251
clarified settings -- avoid hard-wired directories;
 wenzelm parents: 
68483diff
changeset | 1257 |     val system_output_dir: Path = Path.explode("$ISABELLE_HEAPS_SYSTEM/$ML_IDENTIFIER")
 | 
| 
ccacc84e0251
clarified settings -- avoid hard-wired directories;
 wenzelm parents: 
68483diff
changeset | 1258 |     val user_output_dir: Path = Path.explode("$ISABELLE_HEAPS/$ML_IDENTIFIER")
 | 
| 68219 | 1259 | |
| 69854 
cc0b3e177b49
system option "system_heaps" supersedes various command-line options for "system build mode";
 wenzelm parents: 
69812diff
changeset | 1260 |     def system_heaps: Boolean = options.bool("system_heaps")
 | 
| 
cc0b3e177b49
system option "system_heaps" supersedes various command-line options for "system build mode";
 wenzelm parents: 
69812diff
changeset | 1261 | |
| 68221 | 1262 | val output_dir: Path = | 
| 69854 
cc0b3e177b49
system option "system_heaps" supersedes various command-line options for "system build mode";
 wenzelm parents: 
69812diff
changeset | 1263 | if (system_heaps) system_output_dir else user_output_dir | 
| 68219 | 1264 | |
| 68221 | 1265 | val input_dirs: List[Path] = | 
| 69854 
cc0b3e177b49
system option "system_heaps" supersedes various command-line options for "system build mode";
 wenzelm parents: 
69812diff
changeset | 1266 | if (system_heaps) List(system_output_dir) | 
| 68219 | 1267 | else List(user_output_dir, system_output_dir) | 
| 62632 | 1268 | |
| 72648 | 1269 | def presentation_dir: Path = | 
| 69854 
cc0b3e177b49
system option "system_heaps" supersedes various command-line options for "system build mode";
 wenzelm parents: 
69812diff
changeset | 1270 |       if (system_heaps) Path.explode("$ISABELLE_BROWSER_INFO_SYSTEM")
 | 
| 62632 | 1271 |       else Path.explode("$ISABELLE_BROWSER_INFO")
 | 
| 1272 | ||
| 62637 
0189fe0f6452
support for Poly/ML heap hierarchy, which saves a lot of disk space;
 wenzelm parents: 
62636diff
changeset | 1273 | |
| 68221 | 1274 | /* file names */ | 
| 65298 | 1275 | |
| 68221 | 1276 | def heap(name: String): Path = Path.basic(name) | 
| 1277 |     def database(name: String): Path = Path.basic("log") + Path.basic(name).ext("db")
 | |
| 1278 |     def log(name: String): Path = Path.basic("log") + Path.basic(name)
 | |
| 1279 |     def log_gz(name: String): Path = log(name).ext("gz")
 | |
| 62637 
0189fe0f6452
support for Poly/ML heap hierarchy, which saves a lot of disk space;
 wenzelm parents: 
62636diff
changeset | 1280 | |
| 68221 | 1281 | def output_heap(name: String): Path = output_dir + heap(name) | 
| 1282 | def output_database(name: String): Path = output_dir + database(name) | |
| 68086 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1283 | def output_log(name: String): Path = output_dir + log(name) | 
| 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1284 | def output_log_gz(name: String): Path = output_dir + log_gz(name) | 
| 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1285 | |
| 73340 | 1286 |     def prepare_output_dir(): Unit = Isabelle_System.make_directory(output_dir + Path.basic("log"))
 | 
| 68220 
8fc4e3d1df86
clarified store.clean_output: cleanup user_output_dir even in system_mode;
 wenzelm parents: 
68219diff
changeset | 1287 | |
| 62637 
0189fe0f6452
support for Poly/ML heap hierarchy, which saves a lot of disk space;
 wenzelm parents: 
62636diff
changeset | 1288 | |
| 68221 | 1289 | /* heap */ | 
| 62637 
0189fe0f6452
support for Poly/ML heap hierarchy, which saves a lot of disk space;
 wenzelm parents: 
62636diff
changeset | 1290 | |
| 68212 
5a59fded83c7
clarified heap vs. database operations: discontinued correlation of directory;
 wenzelm parents: 
68210diff
changeset | 1291 | def find_heap(name: String): Option[Path] = | 
| 68221 | 1292 | input_dirs.map(_ + heap(name)).find(_.is_file) | 
| 68212 
5a59fded83c7
clarified heap vs. database operations: discontinued correlation of directory;
 wenzelm parents: 
68210diff
changeset | 1293 | |
| 
5a59fded83c7
clarified heap vs. database operations: discontinued correlation of directory;
 wenzelm parents: 
68210diff
changeset | 1294 | def find_heap_digest(name: String): Option[String] = | 
| 71601 | 1295 | find_heap(name).flatMap(read_heap_digest) | 
| 62632 | 1296 | |
| 68221 | 1297 | def the_heap(name: String): Path = | 
| 1298 | find_heap(name) getOrElse | |
| 1299 |         error("Missing heap image for session " + quote(name) + " -- expected in:\n" +
 | |
| 1300 | cat_lines(input_dirs.map(dir => " " + dir.expand.implode))) | |
| 1301 | ||
| 1302 | ||
| 1303 | /* database */ | |
| 1304 | ||
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1305 |     def database_server: Boolean = options.bool("build_database_server")
 | 
| 65281 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1306 | |
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1307 | def open_database_server(): SQL.Database = | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1308 | PostgreSQL.open_database( | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1309 |         user = options.string("build_database_user"),
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1310 |         password = options.string("build_database_password"),
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1311 |         database = options.string("build_database_name"),
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1312 |         host = options.string("build_database_host"),
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1313 |         port = options.int("build_database_port"),
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1314 | ssh = | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1315 |           options.proper_string("build_database_ssh_host").map(ssh_host =>
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1316 | SSH.open_session(options, | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1317 | host = ssh_host, | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1318 |               user = options.string("build_database_ssh_user"),
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1319 |               port = options.int("build_database_ssh_port"))),
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1320 | ssh_close = true) | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1321 | |
| 72854 | 1322 | def open_database_context(): Database_Context = | 
| 1323 | new Database_Context(store, if (database_server) Some(open_database_server()) else None) | |
| 72682 | 1324 | |
| 75393 | 1325 |     def try_open_database(name: String, output: Boolean = false): Option[SQL.Database] = {
 | 
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1326 | def check(db: SQL.Database): Option[SQL.Database] = | 
| 73367 | 1327 |         if (output || session_info_exists(db)) Some(db) else { db.close(); None }
 | 
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1328 | |
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1329 | if (database_server) check(open_database_server()) | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1330 | else if (output) Some(SQLite.open_database(output_database(name))) | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1331 |       else {
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1332 |         (for {
 | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1333 | dir <- input_dirs.view | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1334 | path = dir + database(name) if path.is_file | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1335 | db <- check(SQLite.open_database(path)) | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1336 | } yield db).headOption | 
| 68221 | 1337 | } | 
| 1338 | } | |
| 68212 
5a59fded83c7
clarified heap vs. database operations: discontinued correlation of directory;
 wenzelm parents: 
68210diff
changeset | 1339 | |
| 68221 | 1340 | def open_database(name: String, output: Boolean = false): SQL.Database = | 
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1341 | try_open_database(name, output = output) getOrElse | 
| 68221 | 1342 |         error("Missing build database for session " + quote(name))
 | 
| 68205 | 1343 | |
| 75393 | 1344 |     def clean_output(name: String): (Boolean, Boolean) = {
 | 
| 68221 | 1345 | val relevant_db = | 
| 75393 | 1346 |         database_server && {
 | 
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1347 |           try_open_database(name) match {
 | 
| 68221 | 1348 | case Some(db) => | 
| 1349 |               try {
 | |
| 1350 |                 db.transaction {
 | |
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1351 | val relevant_db = session_info_defined(db, name) | 
| 68221 | 1352 | init_session_info(db, name) | 
| 1353 | relevant_db | |
| 1354 | } | |
| 73367 | 1355 |               } finally { db.close() }
 | 
| 68221 | 1356 | case None => false | 
| 1357 | } | |
| 1358 | } | |
| 68212 
5a59fded83c7
clarified heap vs. database operations: discontinued correlation of directory;
 wenzelm parents: 
68210diff
changeset | 1359 | |
| 68221 | 1360 | val del = | 
| 1361 |         for {
 | |
| 69854 
cc0b3e177b49
system option "system_heaps" supersedes various command-line options for "system build mode";
 wenzelm parents: 
69812diff
changeset | 1362 | dir <- | 
| 
cc0b3e177b49
system option "system_heaps" supersedes various command-line options for "system build mode";
 wenzelm parents: 
69812diff
changeset | 1363 | (if (system_heaps) List(user_output_dir, system_output_dir) else List(user_output_dir)) | 
| 68221 | 1364 | file <- List(heap(name), database(name), log(name), log_gz(name)) | 
| 1365 | path = dir + file if path.is_file | |
| 1366 | } yield path.file.delete | |
| 1367 | ||
| 1368 | val relevant = relevant_db || del.nonEmpty | |
| 1369 | val ok = del.forall(b => b) | |
| 1370 | (relevant, ok) | |
| 1371 | } | |
| 65287 | 1372 | |
| 1373 | ||
| 68218 | 1374 | /* SQL database content */ | 
| 1375 | ||
| 1376 | def read_bytes(db: SQL.Database, name: String, column: SQL.Column): Bytes = | |
| 1377 | db.using_statement(Session_Info.table.select(List(column), | |
| 75394 | 1378 |         Session_Info.session_name.where_equal(name))) { stmt =>
 | 
| 68218 | 1379 | val res = stmt.execute_query() | 
| 1380 | if (!res.next()) Bytes.empty else res.bytes(column) | |
| 75394 | 1381 | } | 
| 68218 | 1382 | |
| 1383 | def read_properties(db: SQL.Database, name: String, column: SQL.Column): List[Properties.T] = | |
| 73031 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1384 | Properties.uncompress(read_bytes(db, name, column), cache = cache) | 
| 68218 | 1385 | |
| 1386 | ||
| 65296 | 1387 | /* session info */ | 
| 65281 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1388 | |
| 75393 | 1389 |     def init_session_info(db: SQL.Database, name: String): Unit = {
 | 
| 68086 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1390 |       db.transaction {
 | 
| 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1391 | db.create_table(Session_Info.table) | 
| 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1392 | db.using_statement( | 
| 73367 | 1393 | Session_Info.table.delete(Session_Info.session_name.where_equal(name)))(_.execute()) | 
| 68221 | 1394 | |
| 1395 | db.create_table(Export.Data.table) | |
| 1396 | db.using_statement( | |
| 73367 | 1397 | Export.Data.table.delete(Export.Data.session_name.where_equal(name)))(_.execute()) | 
| 72653 
ea35afdb1366
store documents within session database, instead of browser_info directory;
 wenzelm parents: 
72652diff
changeset | 1398 | |
| 73718 | 1399 | db.create_table(Document_Build.Data.table) | 
| 72653 
ea35afdb1366
store documents within session database, instead of browser_info directory;
 wenzelm parents: 
72652diff
changeset | 1400 | db.using_statement( | 
| 73718 | 1401 | Document_Build.Data.table.delete( | 
| 1402 | Document_Build.Data.session_name.where_equal(name)))(_.execute()) | |
| 68221 | 1403 | } | 
| 1404 | } | |
| 1405 | ||
| 75393 | 1406 |     def session_info_exists(db: SQL.Database): Boolean = {
 | 
| 72634 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1407 | val tables = db.tables | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1408 | tables.contains(Session_Info.table.name) && | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1409 | tables.contains(Export.Data.table.name) | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1410 | } | 
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1411 | |
| 
5cea0993ee4f
clarified access to single database server vs. collection of database files;
 wenzelm parents: 
72627diff
changeset | 1412 | def session_info_defined(db: SQL.Database, name: String): Boolean = | 
| 68221 | 1413 |       db.transaction {
 | 
| 75393 | 1414 |         session_info_exists(db) && {
 | 
| 68221 | 1415 | db.using_statement( | 
| 1416 | Session_Info.table.select(List(Session_Info.session_name), | |
| 1417 | Session_Info.session_name.where_equal(name)))(stmt => stmt.execute_query().next()) | |
| 1418 | } | |
| 68086 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1419 | } | 
| 
9e1c670301b8
cleanup session output before starting build job;
 wenzelm parents: 
68018diff
changeset | 1420 | |
| 65296 | 1421 | def write_session_info( | 
| 65318 
342efc382558
eliminated somewhat redundant inlined name (despite a7aa17a1f721);
 wenzelm parents: 
65298diff
changeset | 1422 | db: SQL.Database, | 
| 65320 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1423 | name: String, | 
| 65318 
342efc382558
eliminated somewhat redundant inlined name (despite a7aa17a1f721);
 wenzelm parents: 
65298diff
changeset | 1424 | build_log: Build_Log.Session_Info, | 
| 75393 | 1425 | build: Build.Session_Info | 
| 1426 |     ): Unit = {
 | |
| 65281 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1427 |       db.transaction {
 | 
| 75394 | 1428 |         db.using_statement(Session_Info.table.insert()) { stmt =>
 | 
| 65748 | 1429 | stmt.string(1) = name | 
| 65857 | 1430 | stmt.bytes(2) = Properties.encode(build_log.session_timing) | 
| 73031 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1431 | stmt.bytes(3) = Properties.compress(build_log.command_timings, cache = cache.xz) | 
| 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1432 | stmt.bytes(4) = Properties.compress(build_log.theory_timings, cache = cache.xz) | 
| 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1433 | stmt.bytes(5) = Properties.compress(build_log.ml_statistics, cache = cache.xz) | 
| 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1434 | stmt.bytes(6) = Properties.compress(build_log.task_statistics, cache = cache.xz) | 
| 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1435 | stmt.bytes(7) = Build_Log.compress_errors(build_log.errors, cache = cache.xz) | 
| 66873 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1436 | stmt.string(8) = build.sources | 
| 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1437 | stmt.string(9) = cat_lines(build.input_heaps) | 
| 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1438 | stmt.string(10) = build.output_heap getOrElse "" | 
| 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1439 | stmt.int(11) = build.return_code | 
| 65281 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1440 | stmt.execute() | 
| 75394 | 1441 | } | 
| 65281 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1442 | } | 
| 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1443 | } | 
| 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1444 | |
| 65320 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1445 | def read_session_timing(db: SQL.Database, name: String): Properties.T = | 
| 73031 
f93f0597f4fb
clarified signature: absorb XZ.Cache into XML.Cache;
 wenzelm parents: 
73024diff
changeset | 1446 | Properties.decode(read_bytes(db, name, Session_Info.session_timing), cache = cache) | 
| 65286 | 1447 | |
| 65320 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1448 | def read_command_timings(db: SQL.Database, name: String): List[Properties.T] = | 
| 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1449 | read_properties(db, name, Session_Info.command_timings) | 
| 65286 | 1450 | |
| 66873 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1451 | def read_theory_timings(db: SQL.Database, name: String): List[Properties.T] = | 
| 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1452 | read_properties(db, name, Session_Info.theory_timings) | 
| 
9953ae603a23
provide theory timing information, similar to command timing but always considered relevant;
 wenzelm parents: 
66857diff
changeset | 1453 | |
| 65320 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1454 | def read_ml_statistics(db: SQL.Database, name: String): List[Properties.T] = | 
| 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1455 | read_properties(db, name, Session_Info.ml_statistics) | 
| 65286 | 1456 | |
| 65320 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1457 | def read_task_statistics(db: SQL.Database, name: String): List[Properties.T] = | 
| 
52861eebf58d
access table via session_name: db may in principle contain multiple entries;
 wenzelm parents: 
65318diff
changeset | 1458 | read_properties(db, name, Session_Info.task_statistics) | 
| 65286 | 1459 | |
| 72738 
a4d7da18ac5c
store timings for used_theories in canonical order, with reconstructed store.read_theories;
 wenzelm parents: 
72716diff
changeset | 1460 | def read_theories(db: SQL.Database, name: String): List[String] = | 
| 
a4d7da18ac5c
store timings for used_theories in canonical order, with reconstructed store.read_theories;
 wenzelm parents: 
72716diff
changeset | 1461 | read_theory_timings(db, name).flatMap(Markup.Name.unapply) | 
| 
a4d7da18ac5c
store timings for used_theories in canonical order, with reconstructed store.read_theories;
 wenzelm parents: 
72716diff
changeset | 1462 | |
| 65937 | 1463 | def read_errors(db: SQL.Database, name: String): List[String] = | 
| 73033 | 1464 | Build_Log.uncompress_errors(read_bytes(db, name, Session_Info.errors), cache = cache) | 
| 65937 | 1465 | |
| 75393 | 1466 |     def read_build(db: SQL.Database, name: String): Option[Build.Session_Info] = {
 | 
| 66957 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1467 |       if (db.tables.contains(Session_Info.table.name)) {
 | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1468 | db.using_statement(Session_Info.table.select(Session_Info.build_columns, | 
| 75394 | 1469 |           Session_Info.session_name.where_equal(name))) { stmt =>
 | 
| 66957 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1470 | val res = stmt.execute_query() | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1471 | if (!res.next()) None | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1472 |           else {
 | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1473 | Some( | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1474 | Build.Session_Info( | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1475 | res.string(Session_Info.sources), | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1476 | split_lines(res.string(Session_Info.input_heaps)), | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1477 |                 res.string(Session_Info.output_heap) match { case "" => None case s => Some(s) },
 | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1478 | res.int(Session_Info.return_code))) | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1479 | } | 
| 75394 | 1480 | } | 
| 66957 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1481 | } | 
| 
82d13ba817b2
more permissive: db could be empty after hard crash;
 wenzelm parents: 
66914diff
changeset | 1482 | else None | 
| 66744 | 1483 | } | 
| 65281 
c70e7d24a16d
SQL database operations for combined session info;
 wenzelm parents: 
65278diff
changeset | 1484 | } | 
| 62631 | 1485 | } |