src/Pure/General/file.scala
author wenzelm
Mon, 02 Dec 2024 22:16:29 +0100
changeset 81541 5335b1ca6233
parent 80481 0e2b09fef3d2
permissions -rw-r--r--
more elementary operation Term.variant_bounds: only for bounds vs. frees, no consts, no tfrees;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     1
/*  Title:      Pure/General/file.scala
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     3
64698
e022a69db531 tuned comments;
wenzelm
parents: 64668
diff changeset
     4
File-system operations.
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     5
*/
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     6
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     7
package isabelle
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     8
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
     9
79980
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
    10
import java.util.{Properties => JProperties}
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
    11
import java.io.{BufferedWriter, OutputStreamWriter, FileOutputStream, BufferedOutputStream,
51504
18095684c5a6 basic support for xz files;
wenzelm
parents: 51251
diff changeset
    12
  OutputStream, InputStream, FileInputStream, BufferedInputStream, BufferedReader,
18095684c5a6 basic support for xz files;
wenzelm
parents: 51251
diff changeset
    13
  InputStreamReader, File => JFile, IOException}
73317
df49ca5da9d0 clarified modules: more like ML;
wenzelm
parents: 72698
diff changeset
    14
import java.nio.file.{StandardOpenOption, Path => JPath, Files, SimpleFileVisitor,
df49ca5da9d0 clarified modules: more like ML;
wenzelm
parents: 72698
diff changeset
    15
  FileVisitOption, FileVisitResult}
78169
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
    16
import java.nio.file.attribute.{BasicFileAttributes, PosixFilePermission}
79659
a4118f530263 clarified signature: avoid ill-defined type java.net.URL;
wenzelm
parents: 79045
diff changeset
    17
import java.net.URI
50684
12b7e0b4a66e support File.read_gzip as well, in accordance to File.write_gzip;
wenzelm
parents: 50203
diff changeset
    18
import java.util.zip.{GZIPInputStream, GZIPOutputStream}
69293
72a9860f8602 clarified find_files: follow links by default, e.g. relevant for "~/cronjob/log";
wenzelm
parents: 66693
diff changeset
    19
import java.util.EnumSet
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
    20
76353
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
    21
import org.tukaani.xz
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
    22
import com.github.luben.zstd
76348
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
    23
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
    24
import scala.collection.mutable
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
    25
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
    26
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
    27
object File {
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    28
  /* standard path (Cygwin or Posix) */
60988
1d7a7e33fd67 tuned signature, according to ML version;
wenzelm
parents: 58610
diff changeset
    29
1d7a7e33fd67 tuned signature, according to ML version;
wenzelm
parents: 58610
diff changeset
    30
  def standard_path(path: Path): String = path.expand.implode
1d7a7e33fd67 tuned signature, according to ML version;
wenzelm
parents: 58610
diff changeset
    31
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    32
  def standard_path(platform_path: String): String =
73911
a8c5ee444991 clarified modules and signatures;
wenzelm
parents: 73906
diff changeset
    33
    isabelle.setup.Environment.standard_path(platform_path)
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    34
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    35
  def standard_path(file: JFile): String = standard_path(file.getPath)
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    36
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    37
  def standard_url(name: String): String =
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    38
    try {
79044
8cc1ae43e12e clarified signature: avoid deprecated URL constructors;
wenzelm
parents: 78956
diff changeset
    39
      val url = new URI(name).toURL
8cc1ae43e12e clarified signature: avoid deprecated URL constructors;
wenzelm
parents: 78956
diff changeset
    40
      if (url.getProtocol == "file" && Url.is_wellformed_file(name)) {
64775
dd3797f1e0d6 clarified file URIs;
wenzelm
parents: 64760
diff changeset
    41
        standard_path(Url.parse_file(name))
79044
8cc1ae43e12e clarified signature: avoid deprecated URL constructors;
wenzelm
parents: 78956
diff changeset
    42
      }
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    43
      else name
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    44
    }
79045
24d04dd5bf01 more robust exception handling (amending 8cc1ae43e12e);
wenzelm
parents: 79044
diff changeset
    45
    catch { case exn: Throwable if Url.is_malformed(exn) => standard_path(name) }
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    46
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    47
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    48
  /* platform path (Windows or Posix) */
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    49
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    50
  def platform_path(standard_path: String): String =
73911
a8c5ee444991 clarified modules and signatures;
wenzelm
parents: 73906
diff changeset
    51
    isabelle.setup.Environment.platform_path(standard_path)
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    52
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    53
  def platform_path(path: Path): String = platform_path(standard_path(path))
60988
1d7a7e33fd67 tuned signature, according to ML version;
wenzelm
parents: 58610
diff changeset
    54
  def platform_file(path: Path): JFile = new JFile(platform_path(path))
1d7a7e33fd67 tuned signature, according to ML version;
wenzelm
parents: 58610
diff changeset
    55
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
    56
76884
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    57
  /* symbolic path representation, e.g. "~~/src/Pure/ROOT.ML" */
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    58
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    59
  def symbolic_path(path: Path): String = {
77218
86217697863c tuned signature;
wenzelm
parents: 77201
diff changeset
    60
    val directories = space_explode(':', Isabelle_System.getenv("ISABELLE_DIRECTORIES")).reverse
76884
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    61
    val full_name = standard_path(path)
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    62
    directories.view.flatMap(a =>
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    63
      try {
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    64
        val b = standard_path(Path.explode(a))
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    65
        if (full_name == b) Some(a)
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    66
        else {
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    67
          Library.try_unprefix(b + "/", full_name) match {
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    68
            case Some(name) => Some(a + "/" + name)
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    69
            case None => None
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    70
          }
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    71
        }
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    72
      } catch { case ERROR(_) => None }).headOption.getOrElse(path.implode)
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    73
  }
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    74
a004c5322ea4 clarified modules;
wenzelm
parents: 76546
diff changeset
    75
66232
be0ab4b94c62 clarified signature;
wenzelm
parents: 65589
diff changeset
    76
  /* platform files */
be0ab4b94c62 clarified signature;
wenzelm
parents: 65589
diff changeset
    77
be0ab4b94c62 clarified signature;
wenzelm
parents: 65589
diff changeset
    78
  def absolute(file: JFile): JFile = file.toPath.toAbsolutePath.normalize.toFile
be0ab4b94c62 clarified signature;
wenzelm
parents: 65589
diff changeset
    79
  def canonical(file: JFile): JFile = file.getCanonicalFile
be0ab4b94c62 clarified signature;
wenzelm
parents: 65589
diff changeset
    80
be0ab4b94c62 clarified signature;
wenzelm
parents: 65589
diff changeset
    81
  def path(file: JFile): Path = Path.explode(standard_path(file))
76546
88cecb9f1cdc proper unzip with strip option, within the JVM;
wenzelm
parents: 76540
diff changeset
    82
  def path(java_path: JPath): Path = path(java_path.toFile)
88cecb9f1cdc proper unzip with strip option, within the JVM;
wenzelm
parents: 76540
diff changeset
    83
75701
84990c95712d clarified signature;
wenzelm
parents: 75677
diff changeset
    84
  def uri(file: JFile): URI = file.toURI
84990c95712d clarified signature;
wenzelm
parents: 75677
diff changeset
    85
  def uri(path: Path): URI = path.file.toURI
84990c95712d clarified signature;
wenzelm
parents: 75677
diff changeset
    86
79659
a4118f530263 clarified signature: avoid ill-defined type java.net.URL;
wenzelm
parents: 79045
diff changeset
    87
  def url(file: JFile): Url = Url(uri(file))
a4118f530263 clarified signature: avoid ill-defined type java.net.URL;
wenzelm
parents: 79045
diff changeset
    88
  def url(path: Path): Url = url(path.file)
75701
84990c95712d clarified signature;
wenzelm
parents: 75677
diff changeset
    89
66232
be0ab4b94c62 clarified signature;
wenzelm
parents: 65589
diff changeset
    90
75906
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    91
  /* adhoc file types */
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    92
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    93
  def is_ML(s: String): Boolean = s.endsWith(".ML")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    94
  def is_bib(s: String): Boolean = s.endsWith(".bib")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    95
  def is_dll(s: String): Boolean = s.endsWith(".dll")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    96
  def is_exe(s: String): Boolean = s.endsWith(".exe")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    97
  def is_gz(s: String): Boolean = s.endsWith(".gz")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    98
  def is_html(s: String): Boolean = s.endsWith(".html")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
    99
  def is_jar(s: String): Boolean = s.endsWith(".jar")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   100
  def is_java(s: String): Boolean = s.endsWith(".java")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   101
  def is_node(s: String): Boolean = s.endsWith(".node")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   102
  def is_pdf(s: String): Boolean = s.endsWith(".pdf")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   103
  def is_png(s: String): Boolean = s.endsWith(".png")
76540
83de6e9ae983 clarified signature: prefer Scala functions instead of shell scripts;
wenzelm
parents: 76533
diff changeset
   104
  def is_tar_bz2(s: String): Boolean = s.endsWith(".tar.bz2")
76533
2590980401b0 proper treatment of tar.gz double-extension;
wenzelm
parents: 76529
diff changeset
   105
  def is_tar_gz(s: String): Boolean = s.endsWith(".tar.gz")
76540
83de6e9ae983 clarified signature: prefer Scala functions instead of shell scripts;
wenzelm
parents: 76533
diff changeset
   106
  def is_tgz(s: String): Boolean = s.endsWith(".tgz")
75906
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   107
  def is_thy(s: String): Boolean = s.endsWith(".thy")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   108
  def is_xz(s: String): Boolean = s.endsWith(".xz")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   109
  def is_zip(s: String): Boolean = s.endsWith(".zip")
76348
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
   110
  def is_zst(s: String): Boolean = s.endsWith(".zst")
75906
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   111
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   112
  def is_backup(s: String): Boolean = s.endsWith("~") || s.endsWith(".orig")
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   113
2167b9e3157a clarified signature: support for adhoc file types;
wenzelm
parents: 75825
diff changeset
   114
66693
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   115
  /* relative paths */
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   116
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   117
  def relative_path(base: Path, other: Path): Option[Path] = {
73945
e61add9d5b5e tuned signature;
wenzelm
parents: 73911
diff changeset
   118
    val base_path = base.java_path
e61add9d5b5e tuned signature;
wenzelm
parents: 73911
diff changeset
   119
    val other_path = other.java_path
66693
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   120
    if (other_path.startsWith(base_path))
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   121
      Some(path(base_path.relativize(other_path).toFile))
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   122
    else None
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   123
  }
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   124
02588021b581 more operations;
wenzelm
parents: 66233
diff changeset
   125
62545
8ebffdaf2ce2 Bash.process always uses a closed script instead of an open argument list, for extra robustness on Windows, where quoting is not well-defined;
wenzelm
parents: 62544
diff changeset
   126
  /* bash path */
60992
89effcb342df clarified modules, like ML version;
wenzelm
parents: 60988
diff changeset
   127
64304
96bc94c87a81 clarified modules;
wenzelm
parents: 64220
diff changeset
   128
  def bash_path(path: Path): String = Bash.string(standard_path(path))
96bc94c87a81 clarified modules;
wenzelm
parents: 64220
diff changeset
   129
  def bash_path(file: JFile): String = Bash.string(standard_path(file))
60988
1d7a7e33fd67 tuned signature, according to ML version;
wenzelm
parents: 58610
diff changeset
   130
72036
e48a5b6b7554 clarified signature;
wenzelm
parents: 71601
diff changeset
   131
  def bash_platform_path(path: Path): String = Bash.string(platform_path(path))
e48a5b6b7554 clarified signature;
wenzelm
parents: 71601
diff changeset
   132
60988
1d7a7e33fd67 tuned signature, according to ML version;
wenzelm
parents: 58610
diff changeset
   133
62829
4141c2a8458b clarified Isabelle tool wrapper: bash, Scala, no perl, no ML;
wenzelm
parents: 62704
diff changeset
   134
  /* directory content */
4141c2a8458b clarified Isabelle tool wrapper: bash, Scala, no perl, no ML;
wenzelm
parents: 62704
diff changeset
   135
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   136
  def read_dir(dir: Path): List[String] = {
69300
8b6ab9989bcd is_file/is_dir/read_dir: more uniform treatment of errors and boundary cases, notably for symlinks in ssh;
wenzelm
parents: 69299
diff changeset
   137
    if (!dir.is_dir) error("No such directory: " + dir.toString)
62829
4141c2a8458b clarified Isabelle tool wrapper: bash, Scala, no perl, no ML;
wenzelm
parents: 62704
diff changeset
   138
    val files = dir.file.listFiles
4141c2a8458b clarified Isabelle tool wrapper: bash, Scala, no perl, no ML;
wenzelm
parents: 62704
diff changeset
   139
    if (files == null) Nil
69427
ff2f39a221d4 clarified operations: uniform sorting of results;
wenzelm
parents: 69405
diff changeset
   140
    else files.toList.map(_.getName).sorted
62829
4141c2a8458b clarified Isabelle tool wrapper: bash, Scala, no perl, no ML;
wenzelm
parents: 62704
diff changeset
   141
  }
48613
232652ac346e clarified directory content operations (similar to ML version);
wenzelm
parents: 48550
diff changeset
   142
76529
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   143
  def get_entry(
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   144
    dir: Path,
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   145
    pred: Path => Boolean = _ => true,
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   146
    title: String = ""
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   147
  ): Path =
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   148
    read_dir(dir).filter(name => pred(dir + Path.basic(name))) match {
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   149
      case List(entry) => dir + Path.basic(entry)
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   150
      case bad =>
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   151
        error("Bad directory content in " + (if (title.nonEmpty) title else dir.toString) +
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   152
          "\nexpected a single entry, but found" +
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   153
          (if (bad.isEmpty) " nothing"
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   154
           else bad.sorted.map(quote).mkString(":\n  ", "\n  ", "")))
72442
90868036d693 clarified signature;
wenzelm
parents: 72378
diff changeset
   155
    }
90868036d693 clarified signature;
wenzelm
parents: 72378
diff changeset
   156
76529
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   157
  def get_file(dir: Path, title: String = ""): Path =
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   158
    get_entry(dir, pred = _.is_file, title = title)
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   159
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   160
  def get_dir(dir: Path, title: String = ""): Path =
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   161
    get_entry(dir, pred = _.is_dir, title = title)
ded37aade88e clarified signature;
wenzelm
parents: 76353
diff changeset
   162
64932
89c0896a19ad clarified operation: include dirs as well;
wenzelm
parents: 64775
diff changeset
   163
  def find_files(
89c0896a19ad clarified operation: include dirs as well;
wenzelm
parents: 64775
diff changeset
   164
    start: JFile,
89c0896a19ad clarified operation: include dirs as well;
wenzelm
parents: 64775
diff changeset
   165
    pred: JFile => Boolean = _ => true,
69293
72a9860f8602 clarified find_files: follow links by default, e.g. relevant for "~/cronjob/log";
wenzelm
parents: 66693
diff changeset
   166
    include_dirs: Boolean = false,
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   167
    follow_links: Boolean = false
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   168
  ): List[JFile] = {
62443
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   169
    val result = new mutable.ListBuffer[JFile]
73340
0ffcad1f6130 tuned --- fewer warnings;
wenzelm
parents: 73318
diff changeset
   170
    def check(file: JFile): Unit = if (pred(file)) result += file
62443
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   171
64932
89c0896a19ad clarified operation: include dirs as well;
wenzelm
parents: 64775
diff changeset
   172
    if (start.isFile) check(start)
62443
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   173
    else if (start.isDirectory) {
69293
72a9860f8602 clarified find_files: follow links by default, e.g. relevant for "~/cronjob/log";
wenzelm
parents: 66693
diff changeset
   174
      val options =
72a9860f8602 clarified find_files: follow links by default, e.g. relevant for "~/cronjob/log";
wenzelm
parents: 66693
diff changeset
   175
        if (follow_links) EnumSet.of(FileVisitOption.FOLLOW_LINKS)
72a9860f8602 clarified find_files: follow links by default, e.g. relevant for "~/cronjob/log";
wenzelm
parents: 66693
diff changeset
   176
        else EnumSet.noneOf(classOf[FileVisitOption])
78243
0e221a8128e4 tuned: prefer Scala over Java;
wenzelm
parents: 78169
diff changeset
   177
      Files.walkFileTree(start.toPath, options, Int.MaxValue,
62443
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   178
        new SimpleFileVisitor[JPath] {
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   179
          override def preVisitDirectory(
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   180
            path: JPath,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   181
            attrs: BasicFileAttributes
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   182
          ): FileVisitResult = {
64932
89c0896a19ad clarified operation: include dirs as well;
wenzelm
parents: 64775
diff changeset
   183
            if (include_dirs) check(path.toFile)
89c0896a19ad clarified operation: include dirs as well;
wenzelm
parents: 64775
diff changeset
   184
            FileVisitResult.CONTINUE
89c0896a19ad clarified operation: include dirs as well;
wenzelm
parents: 64775
diff changeset
   185
          }
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   186
          override def visitFile(
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   187
            path: JPath,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   188
            attrs: BasicFileAttributes
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   189
          ): FileVisitResult = {
69301
f6c17b9e1104 more uniform find_files, notably for symlinks;
wenzelm
parents: 69300
diff changeset
   190
            val file = path.toFile
f6c17b9e1104 more uniform find_files, notably for symlinks;
wenzelm
parents: 69300
diff changeset
   191
            if (include_dirs || !file.isDirectory) check(file)
62443
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   192
            FileVisitResult.CONTINUE
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   193
          }
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   194
        }
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   195
      )
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   196
    }
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   197
133f65ac17e5 just one File.find_files, based on Java 7 Files operations;
wenzelm
parents: 62294
diff changeset
   198
    result.toList
48613
232652ac346e clarified directory content operations (similar to ML version);
wenzelm
parents: 48550
diff changeset
   199
  }
232652ac346e clarified directory content operations (similar to ML version);
wenzelm
parents: 48550
diff changeset
   200
232652ac346e clarified directory content operations (similar to ML version);
wenzelm
parents: 48550
diff changeset
   201
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   202
  /* read */
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   203
65589
f70c617e9c26 more robust treatment of non-UTF8 text files (cf. 3ed43cfc8b14), notably old log files in ISO-8859-15;
wenzelm
parents: 64934
diff changeset
   204
  def read(file: JFile): String = Bytes.read(file).text
48913
f686cb016c0c more direct File.read_bytes -- avoid cumulative copying of StringBuilder;
wenzelm
parents: 48613
diff changeset
   205
  def read(path: Path): String = read(path.file)
f686cb016c0c more direct File.read_bytes -- avoid cumulative copying of StringBuilder;
wenzelm
parents: 48613
diff changeset
   206
80481
0e2b09fef3d2 more uniform Bytes.read_stream vs. File.read_stream;
wenzelm
parents: 80441
diff changeset
   207
  def read_stream(stream: InputStream): String = Bytes.read_stream(stream).text
50684
12b7e0b4a66e support File.read_gzip as well, in accordance to File.write_gzip;
wenzelm
parents: 50203
diff changeset
   208
12b7e0b4a66e support File.read_gzip as well, in accordance to File.write_gzip;
wenzelm
parents: 50203
diff changeset
   209
  def read_gzip(file: JFile): String =
12b7e0b4a66e support File.read_gzip as well, in accordance to File.write_gzip;
wenzelm
parents: 50203
diff changeset
   210
    read_stream(new GZIPInputStream(new BufferedInputStream(new FileInputStream(file))))
64000
445b3deced8f clarified modules;
wenzelm
parents: 62854
diff changeset
   211
  def read_gzip(path: Path): String = read_gzip(path.file)
51504
18095684c5a6 basic support for xz files;
wenzelm
parents: 51251
diff changeset
   212
64002
08f89f0e8a62 clarified modules;
wenzelm
parents: 64000
diff changeset
   213
  def read_xz(file: JFile): String =
76353
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
   214
    read_stream(new xz.XZInputStream(new BufferedInputStream(new FileInputStream(file))))
64000
445b3deced8f clarified modules;
wenzelm
parents: 62854
diff changeset
   215
  def read_xz(path: Path): String = read_xz(path.file)
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   216
76348
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
   217
  def read_zstd(file: JFile): String = {
76349
b4daf7577ca0 clarified Zstd.init(): avoid accidential com.github.luben.zstd.util.Native.load() operation;
wenzelm
parents: 76348
diff changeset
   218
    Zstd.init()
76353
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
   219
    read_stream(new zstd.ZstdInputStream(new BufferedInputStream(new FileInputStream(file))))
76348
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
   220
  }
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
   221
  def read_zstd(path: Path): String = read_zstd(path.file)
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
   222
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   223
50845
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   224
  /* read lines */
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   225
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   226
  def read_line(reader: BufferedReader): Option[String] = {
69487
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   227
    val line =
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   228
      try { reader.readLine}
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   229
      catch { case _: IOException => null }
72698
6f83f7892317 clarified: more uniform;
wenzelm
parents: 72572
diff changeset
   230
    Option(line).map(Library.trim_line)
69487
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   231
  }
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   232
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   233
  def read_lines(reader: BufferedReader, progress: String => Unit): List[String] = {
50845
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   234
    val result = new mutable.ListBuffer[String]
69487
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   235
    var line: Option[String] = None
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   236
    while ({ line = read_line(reader); line.isDefined }) {
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   237
      progress(line.get)
681760f50723 clarified signature;
wenzelm
parents: 69427
diff changeset
   238
      result += line.get
50845
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   239
    }
73367
77ef8bef0593 clarified signature --- fewer warnings;
wenzelm
parents: 73340
diff changeset
   240
    reader.close()
50845
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   241
    result.toList
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   242
  }
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   243
477ca927676f immediate theory progress for build_dialog;
wenzelm
parents: 50684
diff changeset
   244
79980
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   245
  /* read properties */
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   246
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   247
  def read_props(path: Path): JProperties = {
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   248
    val props = new JProperties
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   249
    props.load(Files.newBufferedReader(path.java_path))
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   250
    props
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   251
  }
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   252
ee04ce2ac13f clarified signature;
wenzelm
parents: 79659
diff changeset
   253
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   254
  /* write */
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   255
71534
f10bffaa2bae more scalable output of YXML files;
wenzelm
parents: 71377
diff changeset
   256
  def writer(file: JFile): BufferedWriter =
f10bffaa2bae more scalable output of YXML files;
wenzelm
parents: 71377
diff changeset
   257
    new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file), UTF8.charset))
f10bffaa2bae more scalable output of YXML files;
wenzelm
parents: 71377
diff changeset
   258
73340
0ffcad1f6130 tuned --- fewer warnings;
wenzelm
parents: 73318
diff changeset
   259
  def write_file(
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   260
    file: JFile,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   261
    text: String,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   262
    make_stream: OutputStream => OutputStream
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   263
  ): Unit = {
51504
18095684c5a6 basic support for xz files;
wenzelm
parents: 51251
diff changeset
   264
    val stream = make_stream(new FileOutputStream(file))
69393
ed0824ef337e static type for Library.using: avoid Java 11 warnings on "Illegal reflective access";
wenzelm
parents: 69301
diff changeset
   265
    using(new BufferedWriter(new OutputStreamWriter(stream, UTF8.charset)))(_.append(text))
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   266
  }
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   267
73574
12b3f78dde61 clarified signature: avoid overlap of String vs. Bytes (both are CharSequence);
wenzelm
parents: 73367
diff changeset
   268
  def write(file: JFile, text: String): Unit = write_file(file, text, s => s)
12b3f78dde61 clarified signature: avoid overlap of String vs. Bytes (both are CharSequence);
wenzelm
parents: 73367
diff changeset
   269
  def write(path: Path, text: String): Unit = write(path.file, text)
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   270
73574
12b3f78dde61 clarified signature: avoid overlap of String vs. Bytes (both are CharSequence);
wenzelm
parents: 73367
diff changeset
   271
  def write_gzip(file: JFile, text: String): Unit =
51504
18095684c5a6 basic support for xz files;
wenzelm
parents: 51251
diff changeset
   272
    write_file(file, text, (s: OutputStream) => new GZIPOutputStream(new BufferedOutputStream(s)))
73574
12b3f78dde61 clarified signature: avoid overlap of String vs. Bytes (both are CharSequence);
wenzelm
parents: 73367
diff changeset
   273
  def write_gzip(path: Path, text: String): Unit = write_gzip(path.file, text)
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   274
76351
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   275
  def write_xz(file: JFile, text: String, options: Compress.Options_XZ): Unit =
76353
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
   276
    File.write_file(file, text,
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
   277
      s => new xz.XZOutputStream(new BufferedOutputStream(s), options.make))
76351
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   278
  def write_xz(file: JFile, text: String): Unit = write_xz(file, text, Compress.Options_XZ())
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   279
  def write_xz(path: Path, text: String, options: Compress.Options_XZ): Unit =
64002
08f89f0e8a62 clarified modules;
wenzelm
parents: 64000
diff changeset
   280
    write_xz(path.file, text, options)
76351
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   281
  def write_xz(path: Path, text: String): Unit = write_xz(path, text, Compress.Options_XZ())
64000
445b3deced8f clarified modules;
wenzelm
parents: 62854
diff changeset
   282
76351
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   283
  def write_zstd(file: JFile, text: String, options: Compress.Options_Zstd): Unit = {
76349
b4daf7577ca0 clarified Zstd.init(): avoid accidential com.github.luben.zstd.util.Native.load() operation;
wenzelm
parents: 76348
diff changeset
   284
    Zstd.init()
76353
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
   285
    File.write_file(file, text,
3698d0f3da18 clarified signature;
wenzelm
parents: 76351
diff changeset
   286
      s => new zstd.ZstdOutputStream(new BufferedOutputStream(s), options.level))
76348
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
   287
  }
76351
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   288
  def write_zstd(file: JFile, text: String): Unit =
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   289
    write_zstd(file, text, Compress.Options_Zstd())
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   290
  def write_zstd(path: Path, text: String, options: Compress.Options_Zstd): Unit =
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   291
    write_zstd(path.file, text, options)
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   292
  def write_zstd(path: Path, text: String): Unit =
2cee31cd92f0 generic support for XZ and Zstd compression in Isabelle/Scala;
wenzelm
parents: 76349
diff changeset
   293
    write_zstd(path, text, Compress.Options_Zstd())
76348
a15f16e8ad18 support for Zstd data compression;
wenzelm
parents: 75906
diff changeset
   294
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   295
  def write_backup(path: Path, text: String): Unit = {
73317
df49ca5da9d0 clarified modules: more like ML;
wenzelm
parents: 72698
diff changeset
   296
    if (path.is_file) Isabelle_System.move_file(path, path.backup)
62444
wenzelm
parents: 62443
diff changeset
   297
    write(path, text)
53336
b3bf6d72fea5 more general backup files;
wenzelm
parents: 52671
diff changeset
   298
  }
b3bf6d72fea5 more general backup files;
wenzelm
parents: 52671
diff changeset
   299
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   300
  def write_backup2(path: Path, text: String): Unit = {
73317
df49ca5da9d0 clarified modules: more like ML;
wenzelm
parents: 72698
diff changeset
   301
    if (path.is_file) Isabelle_System.move_file(path, path.backup2)
62444
wenzelm
parents: 62443
diff changeset
   302
    write(path, text)
58610
fffdbce036db added update_cartouches tool;
wenzelm
parents: 56783
diff changeset
   303
  }
fffdbce036db added update_cartouches tool;
wenzelm
parents: 56783
diff changeset
   304
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   305
62703
01b71da798dd more operations;
wenzelm
parents: 62636
diff changeset
   306
  /* append */
01b71da798dd more operations;
wenzelm
parents: 62636
diff changeset
   307
73574
12b3f78dde61 clarified signature: avoid overlap of String vs. Bytes (both are CharSequence);
wenzelm
parents: 73367
diff changeset
   308
  def append(file: JFile, text: String): Unit =
73627
27659455c592 clarified signature;
wenzelm
parents: 73574
diff changeset
   309
    Files.write(file.toPath, UTF8.bytes(text),
62703
01b71da798dd more operations;
wenzelm
parents: 62636
diff changeset
   310
      StandardOpenOption.APPEND, StandardOpenOption.CREATE)
01b71da798dd more operations;
wenzelm
parents: 62636
diff changeset
   311
73574
12b3f78dde61 clarified signature: avoid overlap of String vs. Bytes (both are CharSequence);
wenzelm
parents: 73367
diff changeset
   312
  def append(path: Path, text: String): Unit = append(path.file, text)
62703
01b71da798dd more operations;
wenzelm
parents: 62636
diff changeset
   313
01b71da798dd more operations;
wenzelm
parents: 62636
diff changeset
   314
75206
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   315
  /* change */
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   316
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   317
  def change(
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   318
    path: Path,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   319
    init: Boolean = false,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   320
    strict: Boolean = false
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   321
  )(f: String => String): Unit = {
75208
31c5a22d50ef proper init of non-existing file;
wenzelm
parents: 75206
diff changeset
   322
    if (!path.is_file && init) write(path, "")
75206
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   323
    val x = read(path)
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   324
    val y = f(x)
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   325
    if (x != y) write(path, y)
75213
e3475e1d5094 tuned signature: more robust operation;
wenzelm
parents: 75208
diff changeset
   326
    else if (strict) error("Unchanged file: " + path)
75206
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   327
  }
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   328
75213
e3475e1d5094 tuned signature: more robust operation;
wenzelm
parents: 75208
diff changeset
   329
  def change_lines(path: Path, init: Boolean = false, strict: Boolean = false)(
e3475e1d5094 tuned signature: more robust operation;
wenzelm
parents: 75208
diff changeset
   330
      f: List[String] => List[String]): Unit =
e3475e1d5094 tuned signature: more robust operation;
wenzelm
parents: 75208
diff changeset
   331
    change(path, init = init, strict = strict)(text => cat_lines(f(split_lines(text))))
75206
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   332
481ad7da73a9 clarified signature;
wenzelm
parents: 75203
diff changeset
   333
64213
b265dd04d57d clarified file operations;
wenzelm
parents: 64003
diff changeset
   334
  /* eq */
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   335
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   336
  def eq(file1: JFile, file2: JFile): Boolean =
73318
wenzelm
parents: 73317
diff changeset
   337
    try { Files.isSameFile(file1.toPath, file2.toPath) }
49673
2a088cff1e7b more robust File.eq, and thus File.copy of "~~/lib/logo/isabelle.gif";
wenzelm
parents: 49610
diff changeset
   338
    catch { case ERROR(_) => false }
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   339
64213
b265dd04d57d clarified file operations;
wenzelm
parents: 64003
diff changeset
   340
  def eq(path1: Path, path2: Path): Boolean = eq(path1.file, path2.file)
b265dd04d57d clarified file operations;
wenzelm
parents: 64003
diff changeset
   341
b265dd04d57d clarified file operations;
wenzelm
parents: 64003
diff changeset
   342
64934
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   343
  /* eq_content */
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   344
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   345
  def eq_content(file1: JFile, file2: JFile): Boolean =
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   346
    if (eq(file1, file2)) true
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   347
    else if (file1.length != file2.length) false
80378
ab4badc7db7f more efficient equals: avoid somewhat slow sha1_digest (see also 29b761e290c5, 306f273c91ec);
wenzelm
parents: 80365
diff changeset
   348
    else Bytes.read(file1) == Bytes.read(file2)
64934
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   349
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   350
  def eq_content(path1: Path, path2: Path): Boolean = eq_content(path1.file, path2.file)
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   351
795055a0be98 hardlink within JVM;
wenzelm
parents: 64932
diff changeset
   352
69405
22428643351f more direct File.executable operation: avoid external process (on Unix);
wenzelm
parents: 69402
diff changeset
   353
  /* permissions */
22428643351f more direct File.executable operation: avoid external process (on Unix);
wenzelm
parents: 69402
diff changeset
   354
78169
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   355
  private val restrict_perms: List[PosixFilePermission] =
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   356
    List(
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   357
      PosixFilePermission.GROUP_READ,
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   358
      PosixFilePermission.GROUP_WRITE,
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   359
      PosixFilePermission.GROUP_EXECUTE,
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   360
      PosixFilePermission.OTHERS_READ,
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   361
      PosixFilePermission.OTHERS_WRITE,
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   362
      PosixFilePermission.OTHERS_EXECUTE)
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   363
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   364
  def restrict(path: Path): Unit =
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   365
    if (Platform.is_windows) Isabelle_System.chmod("g-rwx,o-rwx", path)
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   366
    else {
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   367
      val perms = Files.getPosixFilePermissions(path.java_path)
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   368
      var perms_changed = false
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   369
      for (p <- restrict_perms if perms.contains(p)) {
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   370
        perms.remove(p)
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   371
        perms_changed = true
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   372
      }
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   373
      if (perms_changed) Files.setPosixFilePermissions(path.java_path, perms)
5ad1ae8626de minor performance tuning: avoid external process;
wenzelm
parents: 78161
diff changeset
   374
    }
78161
4b1b7cbb3e9a clarified signature;
wenzelm
parents: 78158
diff changeset
   375
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 75213
diff changeset
   376
  def is_executable(path: Path): Boolean = {
69788
c175499a7537 added executable flag for exports;
wenzelm
parents: 69487
diff changeset
   377
    if (Platform.is_windows) Isabelle_System.bash("test -x " + bash_path(path)).check.ok
c175499a7537 added executable flag for exports;
wenzelm
parents: 69487
diff changeset
   378
    else path.file.canExecute
c175499a7537 added executable flag for exports;
wenzelm
parents: 69487
diff changeset
   379
  }
c175499a7537 added executable flag for exports;
wenzelm
parents: 69487
diff changeset
   380
78298
3b0f8f1010f2 clarified signature, with subtle change of semantics (amending 8b5a2e4b16d4);
wenzelm
parents: 78243
diff changeset
   381
  def set_executable(path: Path, reset: Boolean = false): Unit = {
3b0f8f1010f2 clarified signature, with subtle change of semantics (amending 8b5a2e4b16d4);
wenzelm
parents: 78243
diff changeset
   382
    if (Platform.is_windows) Isabelle_System.chmod(if (reset) "a-x" else "a+x", path)
3b0f8f1010f2 clarified signature, with subtle change of semantics (amending 8b5a2e4b16d4);
wenzelm
parents: 78243
diff changeset
   383
    else path.file.setExecutable(!reset, false)
69405
22428643351f more direct File.executable operation: avoid external process (on Unix);
wenzelm
parents: 69402
diff changeset
   384
  }
74811
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   385
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   386
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   387
  /* content */
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   388
75825
ad00fbf64bff clarified signature --- simplified types;
wenzelm
parents: 75824
diff changeset
   389
  def content(path: Path, content: Bytes): Content = new Content(path, content)
ad00fbf64bff clarified signature --- simplified types;
wenzelm
parents: 75824
diff changeset
   390
  def content(path: Path, content: String): Content = new Content(path, Bytes(content))
75824
a2b2e8964e1a tuned signature;
wenzelm
parents: 75823
diff changeset
   391
  def content(path: Path, content: XML.Body): Content_XML = new Content_XML(path, content)
74811
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   392
75825
ad00fbf64bff clarified signature --- simplified types;
wenzelm
parents: 75824
diff changeset
   393
  final class Content private[File](val path: Path, val content: Bytes) {
75677
347f9fde03dd clarified signature;
wenzelm
parents: 75676
diff changeset
   394
    override def toString: String = path.toString
74811
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   395
77852
df35b5b7b6a4 more direct hg_sync init via ssh (see also 721b3278c8e4);
wenzelm
parents: 77218
diff changeset
   396
    def write(dir: Path, ssh: SSH.System = SSH.Local): Unit = {
75508
64d48fb1b37b more robust;
wenzelm
parents: 75393
diff changeset
   397
      val full_path = dir + path
77852
df35b5b7b6a4 more direct hg_sync init via ssh (see also 721b3278c8e4);
wenzelm
parents: 77218
diff changeset
   398
      ssh.make_directory(ssh.expand_path(full_path).dir)
df35b5b7b6a4 more direct hg_sync init via ssh (see also 721b3278c8e4);
wenzelm
parents: 77218
diff changeset
   399
      ssh.write_bytes(full_path, content)
75508
64d48fb1b37b more robust;
wenzelm
parents: 75393
diff changeset
   400
    }
74811
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   401
  }
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   402
75676
23fbdac78310 clarified signature;
wenzelm
parents: 75508
diff changeset
   403
  final class Content_XML private[File](val path: Path, val content: XML.Body) {
75823
6eb8d6cdb686 proper toString for Content_XML, which is not covered by trait Content;
wenzelm
parents: 75701
diff changeset
   404
    override def toString: String = path.toString
6eb8d6cdb686 proper toString for Content_XML, which is not covered by trait Content;
wenzelm
parents: 75701
diff changeset
   405
75825
ad00fbf64bff clarified signature --- simplified types;
wenzelm
parents: 75824
diff changeset
   406
    def output(out: XML.Body => String): Content = new Content(path, Bytes(out(content)))
74811
1f40ded31b78 clarified modules;
wenzelm
parents: 73945
diff changeset
   407
  }
77109
e3a2b3536030 prefer typed bytes count, but retain toString of original Long for robustness of Java/Scala string composition;
wenzelm
parents: 76884
diff changeset
   408
e3a2b3536030 prefer typed bytes count, but retain toString of original Long for robustness of Java/Scala string composition;
wenzelm
parents: 76884
diff changeset
   409
78956
12abaffb0346 tuned signature: more operations;
wenzelm
parents: 78952
diff changeset
   410
  /* strict file size */
77109
e3a2b3536030 prefer typed bytes count, but retain toString of original Long for robustness of Java/Scala string composition;
wenzelm
parents: 76884
diff changeset
   411
78956
12abaffb0346 tuned signature: more operations;
wenzelm
parents: 78952
diff changeset
   412
  def size(path: Path): Long = path.check_file.file.length
12abaffb0346 tuned signature: more operations;
wenzelm
parents: 78952
diff changeset
   413
  def space(path: Path): Space = Space.bytes(size(path))
48411
5b3440850d36 more abstract file system operations in Scala, corresponding to ML version;
wenzelm
parents:
diff changeset
   414
}