author | wenzelm |
Mon, 20 Feb 2023 17:10:22 +0100 | |
changeset 77316 | d17b0851a61a |
parent 76617 | d5adc9126ae8 |
child 77517 | ede77a627b3a |
permissions | -rw-r--r-- |
75555
197a5b3a1ea2
promote "isabelle sync" to regular user-space tool, with proper documentation;
wenzelm
parents:
75553
diff
changeset
|
1 |
/* Title: Pure/Tools/sync.scala |
75481 | 2 |
Author: Makarius |
3 |
||
4 |
Synchronize Isabelle + AFP repositories. |
|
5 |
*/ |
|
6 |
||
7 |
package isabelle |
|
8 |
||
9 |
||
75549 | 10 |
object Sync { |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
11 |
/* session images */ |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
12 |
|
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
13 |
def find_images(options: Options, session_images: List[String], |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
14 |
dirs: List[Path] = Nil |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
15 |
): List[String] = { |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
16 |
if (session_images.isEmpty) Nil |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
17 |
else { |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
18 |
val store = Sessions.store(options) |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
19 |
val sessions_structure = Sessions.load_structure(options, dirs = dirs) |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
20 |
sessions_structure.build_requirements(session_images).flatMap { session => |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
21 |
val heap = |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
22 |
store.find_heap(session).map(_.expand).map(path => |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
23 |
File.standard_path(path.dir.dir) + "/./" + path.dir.file_name + "/" + path.file_name) |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
24 |
val db = |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
25 |
store.find_database(session).map(_.expand).map(path => |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
26 |
File.standard_path(path.dir.dir.dir) + "/./" + path.dir.dir.file_name + "/" + |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
27 |
path.dir.file_name + "/" + path.file_name) |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
28 |
heap.toList ::: db.toList |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
29 |
} |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
30 |
} |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
31 |
} |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
32 |
|
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
33 |
|
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
34 |
/* sync */ |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
35 |
|
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
36 |
def sync(options: Options, context: Rsync.Context, target: String, |
75481 | 37 |
verbose: Boolean = false, |
75487 | 38 |
thorough: Boolean = false, |
75553 | 39 |
purge_heaps: Boolean = false, |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
40 |
session_images: List[String] = Nil, |
75492 | 41 |
preserve_jars: Boolean = false, |
75481 | 42 |
dry_run: Boolean = false, |
43 |
rev: String = "", |
|
44 |
afp_root: Option[Path] = None, |
|
45 |
afp_rev: String = "" |
|
46 |
): Unit = { |
|
75506 | 47 |
val hg = Mercurial.self_repository() |
75481 | 48 |
val afp_hg = afp_root.map(Mercurial.repository(_)) |
49 |
||
75492 | 50 |
val more_filter = if (preserve_jars) List("include *.jar", "protect *.jar") else Nil |
51 |
||
75511 | 52 |
def sync(hg: Mercurial.Repository, dest: String, r: String, |
53 |
contents: List[File.Content] = Nil, filter: List[String] = Nil |
|
54 |
): Unit = { |
|
75525
68162e4f60a7
clarified signature: more explicit type Rsync.Context;
wenzelm
parents:
75524
diff
changeset
|
55 |
hg.sync(context, dest, rev = r, verbose = verbose, thorough = thorough, dry_run = dry_run, |
68162e4f60a7
clarified signature: more explicit type Rsync.Context;
wenzelm
parents:
75524
diff
changeset
|
56 |
contents = contents, filter = filter ::: more_filter) |
75511 | 57 |
} |
75481 | 58 |
|
75525
68162e4f60a7
clarified signature: more explicit type Rsync.Context;
wenzelm
parents:
75524
diff
changeset
|
59 |
context.progress.echo_if(verbose, "\n* Isabelle repository:") |
75553 | 60 |
val filter_heaps = if (purge_heaps) Nil else List("protect /heaps", "protect /heaps/**") |
75511 | 61 |
sync(hg, target, rev, |
75824 | 62 |
contents = List(File.content(Path.explode("etc/ISABELLE_ID"), hg.id(rev = rev))), |
75553 | 63 |
filter = filter_heaps ::: List("protect /AFP")) |
75481 | 64 |
|
65 |
for (hg <- afp_hg) { |
|
75525
68162e4f60a7
clarified signature: more explicit type Rsync.Context;
wenzelm
parents:
75524
diff
changeset
|
66 |
context.progress.echo_if(verbose, "\n* AFP repository:") |
76617 | 67 |
sync(hg, Url.append_path(target, "AFP"), afp_rev) |
75481 | 68 |
} |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
69 |
|
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
70 |
val images = |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
71 |
find_images(options, session_images, |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
72 |
dirs = afp_root.map(_ + Path.explode("thys")).toList) |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
73 |
if (images.nonEmpty) { |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
74 |
context.progress.echo_if(verbose, "\n* Session images:") |
76617 | 75 |
val heaps = Url.append_path(target, "heaps/") |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
76 |
Rsync.exec(context, verbose = verbose, thorough = thorough, dry_run = dry_run, |
76617 | 77 |
args = List("--relative", "--") ::: images ::: List(heaps)).check |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
78 |
} |
75481 | 79 |
} |
80 |
||
81 |
val isabelle_tool = |
|
75549 | 82 |
Isabelle_Tool("sync", "synchronize Isabelle + AFP repositories", |
75481 | 83 |
Scala_Project.here, { args => |
84 |
var afp_root: Option[Path] = None |
|
75553 | 85 |
var purge_heaps = false |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
86 |
var session_images = List.empty[String] |
75492 | 87 |
var preserve_jars = false |
75536
7cdeed5dc96d
more robust treatment of rsync on macOS (see also 96fb1f9a4042);
wenzelm
parents:
75525
diff
changeset
|
88 |
var protect_args = false |
75487 | 89 |
var thorough = false |
75481 | 90 |
var afp_rev = "" |
91 |
var dry_run = false |
|
76136
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
92 |
var ssh_port = 0 |
75481 | 93 |
var rev = "" |
76136
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
94 |
var ssh_control_path = "" |
75481 | 95 |
var verbose = false |
96 |
||
97 |
val getopts = Getopts(""" |
|
75549 | 98 |
Usage: isabelle sync [OPTIONS] TARGET |
75481 | 99 |
|
100 |
Options are: |
|
75497 | 101 |
-A ROOT include AFP with given root directory (":" for """ + AFP.BASE.implode + """) |
75553 | 102 |
-H purge heaps directory on target |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
103 |
-I NAME include session heap image and build database |
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
104 |
(based on accidental local state) |
75492 | 105 |
-J preserve *.jar files |
76134 | 106 |
-P protect spaces in target file names: more robust, less portable |
76136
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
107 |
-S PATH SSH control path for connection multiplexing |
75493 | 108 |
-T thorough treatment of file content and directory times |
75481 | 109 |
-a REV explicit AFP revision (default: state of working directory) |
110 |
-n no changes: dry-run |
|
76136
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
111 |
-p PORT SSH port |
75481 | 112 |
-r REV explicit revision (default: state of working directory) |
113 |
-v verbose |
|
114 |
||
75555
197a5b3a1ea2
promote "isabelle sync" to regular user-space tool, with proper documentation;
wenzelm
parents:
75553
diff
changeset
|
115 |
Synchronize Isabelle + AFP repositories, based on "isabelle hg_sync". |
75481 | 116 |
""", |
75497 | 117 |
"A:" -> (arg => afp_root = Some(if (arg == ":") AFP.BASE else Path.explode(arg))), |
75553 | 118 |
"H" -> (_ => purge_heaps = true), |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
119 |
"I:" -> (arg => session_images = session_images ::: List(arg)), |
75492 | 120 |
"J" -> (_ => preserve_jars = true), |
76134 | 121 |
"P" -> (_ => protect_args = true), |
76136
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
122 |
"S:" -> (arg => ssh_control_path = arg), |
75487 | 123 |
"T" -> (_ => thorough = true), |
75481 | 124 |
"a:" -> (arg => afp_rev = arg), |
125 |
"n" -> (_ => dry_run = true), |
|
76136
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
126 |
"p:" -> (arg => ssh_port = Value.Int.parse(arg)), |
75481 | 127 |
"r:" -> (arg => rev = arg), |
128 |
"v" -> (_ => verbose = true)) |
|
129 |
||
130 |
val more_args = getopts(args) |
|
131 |
val target = |
|
132 |
more_args match { |
|
133 |
case List(target) => target |
|
134 |
case _ => getopts.usage() |
|
135 |
} |
|
136 |
||
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
137 |
val options = Options.init() |
75481 | 138 |
val progress = new Console_Progress |
76136
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
139 |
val context = |
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
140 |
Rsync.Context(progress, ssh_port = ssh_port, ssh_control_path = ssh_control_path, |
1bb677cceea4
let rsync re-use ssh connection via control path;
wenzelm
parents:
76134
diff
changeset
|
141 |
protect_args = protect_args) |
75552
4aa3da02fd4d
sync session images, based on accidental local state;
wenzelm
parents:
75549
diff
changeset
|
142 |
sync(options, context, target, verbose = verbose, thorough = thorough, |
75553 | 143 |
purge_heaps = purge_heaps, session_images = session_images, preserve_jars = preserve_jars, |
144 |
dry_run = dry_run, rev = rev, afp_root = afp_root, afp_rev = afp_rev) |
|
75481 | 145 |
} |
146 |
) |
|
147 |
} |