forked from xavierleroy/coq2html
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathgenerate_index.ml
365 lines (335 loc) · 13.5 KB
/
generate_index.ml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
(* *********************************************************************)
(* *)
(* Addition to the the Coq2HTML documentation generator *)
(* *)
(* Copyright National Institute of Advanced Industrial Science and *)
(* Technology. All rights reserved. This file is distributed *)
(* under the terms of the GNU General Public License as published by *)
(* the Free Software Foundation, either version 2 of the License, or *)
(* (at your option) any later version. *)
(* *)
(* *********************************************************************)
type range = Range.t
let (!%) s = Printf.sprintf s
let use_file filename f =
let ch = open_in filename in
try
let y = f ch in
close_in ch; y
with
| e -> close_in ch; raise e
let read_file filename = use_file filename (fun ch ->
really_input_string ch (in_channel_length ch))
let escaped =
let buff = Buffer.create 5 in
fun s ->
Buffer.clear buff;
for i = 0 to String.length s - 1 do
match s.[i] with
| '<' -> Buffer.add_string buff "<"
| '>' -> Buffer.add_string buff ">"
| '&' -> Buffer.add_string buff "&"
| '\"' -> Buffer.add_string buff """
| c -> Buffer.add_char buff c
done;
Buffer.contents buff
let sanitize_linkname s =
let rec loop esc i =
if i < 0 then if esc then escaped s else s
else match s.[i] with
| 'a'..'z' | 'A'..'Z' | '0'..'9' | '.' | '_' -> loop esc (i-1)
| '<' | '>' | '&' | '\'' | '\"' -> loop true (i-1)
| '-' | ':' -> loop esc (i-1) (* should be safe in HTML5 attribute name syntax *)
| _ ->
(* This name contains complex characters:
this is probably a notation string, we simply hash it. *)
Digest.to_hex (Digest.string s)
in loop false (String.length s - 1)
let alphabets = (* ['A'; ...; 'Z'; '_'] *)
let rec iter code store =
if code <= Char.code 'Z' then iter (succ code) (Char.chr code :: store)
else store
in
List.rev ('*' :: '_' :: iter (Char.code 'A') [])
type file_path =
| Dir of (string * file_path list)
| File of string
let sidebar_files all_files =
let sort_for_directories files =
let comp x y =
match x, y with
| Dir _, File _ -> -1
| File _, Dir _ -> 1
| _, _ -> compare x y
in
List.sort comp files
in
let rec tag_of_file_path parents = function
| File name ->
let link = (String.concat "." (List.rev (name :: parents))) ^ ".html" in
!%{|<li><a href="%s">%s</a></li>|} link name
| Dir (name, fs) ->
let current_path = List.rev (name :: parents) |> String.concat "." in
let children =
sort_for_directories fs
|> List.map (tag_of_file_path (name :: parents))
in
!%{|<li><details id="%s"><summary>%s</summary>
<ul>
%s
</ul>
</details>
</li>|} current_path name (String.concat "\n" children)
in
sort_for_directories all_files
|> List.map (tag_of_file_path [])
|> String.concat "\n"
let write_html_file all_files txt filename title =
let oc = open_out filename in
let header =
Str.global_replace (Str.regexp "<h1.*</h1>") (!%"<h1>%s</h1>" title) Resources.header
|> Str.global_replace (Str.regexp "<title>.*</title>") (!%"<title>%s</title>" title)
|> Str.global_replace (Str.regexp_string "$FILES") (sidebar_files all_files)
in
output_string oc header;
output_string oc txt;
output_string oc Resources.footer;
close_out oc
type kind = Global | EntryKind of string
let kinds = [EntryKind "file";
EntryKind "def";
EntryKind "prf";
EntryKind "abbrev";
Global;
]
let skind = function Global -> "Global Index"
| EntryKind "def" -> "Definitions"
| EntryKind "prf" -> "Lemmas"
| EntryKind "abbrev" -> "Abbreviations"
| EntryKind "file" -> "Files"
| EntryKind other -> other
let is_kind = function
| Global -> fun _ -> true
| EntryKind k -> fun s -> s = k
let linkname_of_kind = function Global -> "global"
| EntryKind s -> s
type item = {kind: kind; name: string; linkname: string; module_: string}
let table citems =
let mkrow kind =
(!%"<td>%s</td>" (skind kind))
^ (List.map (fun (c, items) ->
if List.exists (fun item -> kind = Global || item.kind = kind) items then
!%{|<td><a href="index_%s_%c.html">%c</a></td>|} (linkname_of_kind kind) c c
else
!%{|<td>%c</td>|} c) citems
|> String.concat "")
|> fun s -> "<tr>" ^ s ^ "</tr>"
in
"<table><tbody>"
^ (List.map mkrow kinds |> String.concat "")
^ "</tbody></table>"
let html_of_notation_item item =
let (scope, notation) =
match Str.(bounded_split_delim (regexp ":") item.name 4) with
| [_; _; ""; notation] -> ("<span class=\"warning\">no scope</span>", notation)
| [_; _; scope; notation] -> ("in " ^ scope, notation)
| _ ->
failwith (!%"unexpected notation format in glob file: name=%s" item.name)
in
let show notation =
let len = String.length notation in
let rec iter pos tags =
let text_of_placeholder s =
Str.(global_replace (regexp_string "_") s " ")
(* |> fun s -> Str.(global_replace (regexp_string "x") s "x")*)
in
if pos < len then
match String.index_from_opt notation pos '\'' with
| Some pos' when pos = pos' -> quoted (pos+1) tags []
| Some pos' ->
quoted (pos'+1)
(text_of_placeholder (String.sub notation pos (pos'-pos)) :: tags) []
| None ->
List.rev (
text_of_placeholder (String.sub notation pos (len - pos)) :: tags)
else List.rev tags
and quoted pos tags store =
let tag_of_quoted ss =
String.concat "" (List.rev ss)
|> !%"<span class=\"notation-symbol\">%s</span>"
in
if pos < len then
match String.index_from_opt notation pos '\'' with
| Some pos' when pos' = len - 1 ->
tag_of_quoted (String.sub notation pos (pos'-pos) :: store) :: tags
|> List.rev
| Some pos' when String.get notation (pos'+1) = '\'' ->
(* two contiguous quotations *)
let s = String.sub notation pos (pos' - pos) in
quoted (pos'+2) tags ("\'" :: s :: store)
| Some pos' ->
(* termination of the quote *)
let tag = tag_of_quoted (String.sub notation pos (pos' - pos) :: store) in
iter (pos' + 1) (tag :: tags)
| None ->
failwith "unclosed quote"
else
List.rev (tag_of_quoted store :: tags)
in
String.concat "" (iter 0 [])
in
!%{|<a href="%s">%s</a> [%s, in %s] (%s)|} item.linkname (show notation) (linkname_of_kind item.kind) item.module_ scope
let compare_case_insensitive s1 s2 =
String.(compare (lowercase_ascii s1) (lowercase_ascii s2))
(*
* generate an html file, e.g., mathcomp.classical.functions.html
*)
let generate_with_capital output_dir table all_files kind (c, items) =
let html_of_item item =
if item.kind = EntryKind "not" then
html_of_notation_item item
else
!%{|<a href="%s">%s</a> [%s, in %s]|} item.linkname item.name (linkname_of_kind item.kind) item.module_
in
if items = [] then () else
let body =
let h2 = if kind = Global then !%"%c" c else !%"%c (%s)" c (skind kind) in
List.filter (fun item -> kind = Global || item.kind = kind) items
|> List.map html_of_item
|> String.concat "<br>"
|> (^) (!%"%s<h2>%s</h2>" table h2)
in
let title = !%"%C (%s)" c (skind kind) in
write_html_file all_files body (Filename.concat output_dir (!%"index_%s_%c.html" (linkname_of_kind kind) c)) title
let overwrite_dot_file_with_url xref_table dot_file = (* dirty *)
let dot_content = String.concat "\n" (Common.read_lines dot_file) in
let is_exists_in_dot_file name = Common.grep name dot_content in
let all_hb_defs =
XrefTable.fold (fun (mod_,_) (_, xref) store ->
match xref with
| Defs ds ->
begin match List.find_opt (fun (path,typ) ->
String.ends_with ~suffix:".pack_" path) ds with
| Some (path,typ) ->
(mod_, path) :: store
| None -> store
end
| _ -> store)
xref_table []
in
let hb_defs =
all_hb_defs
|> List.map (fun (mod_, path) ->
(mod_, String.sub path 0 (String.length path - String.length ".pack_")))
|> List.filter (fun (_, name) -> is_exists_in_dot_file name)
in
let node_with_node (mod_, name) =
let url = mod_ ^ ".html#" ^ name in
!%{|"%s" [URL="%s"]|} name url
in
let links = String.concat "; " (List.map node_with_node hb_defs) in
let lines = Common.read_lines dot_file in
let lines = match lines with (* insert links to second line *)
| line1 :: rest -> line1 :: links :: rest
| [] ->
failwith ("empty lines: " ^ dot_file)
in
Common.write_lines dot_file lines
let generate_hierarchy_graph title xref_table output_dir dot_file =
overwrite_dot_file_with_url xref_table dot_file;
let png_filename = "hierarchy_graph.png" in
let png_path = Filename.concat output_dir png_filename in
let map_path = Filename.concat output_dir "hierarchy_graph.map" in
Graphviz.from_file dot_file
|> Graphviz.generate_file png_path map_path;
let map = read_file map_path in
(*TODO: ↓ The map id (#Hierarchy) should be taken from dot file *)
Printf.sprintf {|<h2>Mathematical Structures (%s only)</h2><img src="%s" title usemap="#Hierarchy"/>
%s|} title png_filename map
let generate_dependency_graph xref_table output_dir dot_file =
let png_filename = "dependency_graph.png" in
let png_path = Filename.concat output_dir png_filename in
let map_path = Filename.concat output_dir "dependency_graph.map" in
Graphviz.from_file dot_file
|> Graphviz.generate_file png_path map_path;
let map = read_file map_path in
Printf.sprintf {|<h2>Clickable Dependency Graph of Files</h2><img src="%s" usemap="#depend"/>%s|} png_filename map
(*
* generate index.html
*)
let generate_topfile output_dir all_files xrefs title xref_table hierarchy_graph_dot_file dependency_dot_file =
let hierarchy_graph =
if hierarchy_graph_dot_file = "" then "" else
generate_hierarchy_graph title xref_table output_dir hierarchy_graph_dot_file
in
let dependency_graph =
if dependency_dot_file = "" then "" else
generate_dependency_graph xref_table output_dir dependency_dot_file
in
let body = table xrefs ^ hierarchy_graph ^ dependency_graph in
write_html_file all_files body (Filename.concat output_dir "index.html") title
let is_initial c s =
if s = "" then false else
match c, String.get s 0 with
| _, '_' -> c = '_'
| _, ('a'..'z' as s0) -> Char.uppercase_ascii s0 = c
| _, ('A'..'Z' as s0) -> s0 = c
| '*', _ -> true
| _, _ -> false
let all_files xref_modules =
let rec iter = function
| [] -> []
| [single_name] :: rest ->
File single_name :: iter rest
| (dir_name :: path) :: rest ->
let (brothers, rest) =
List.partition (fun p -> List.hd p = dir_name) rest
in
let fs =
(path :: List.map List.tl brothers)
|> iter
in
Dir (dir_name, fs) :: iter rest
| [] :: _ ->
failwith "Generate_index.all_files: Please report: This is an unexpected case."
in
Hashtbl.to_seq_keys xref_modules
|> List.of_seq
|> List.sort compare_case_insensitive
|> List.map (String.split_on_char '.')
|> iter
let is_subproof path = String.ends_with ~suffix:"_subproof" path
let generate output_dir (xref_table:XrefTable.t) xref_modules title hierarchy_dot_file dependency_dot_file =
let indexed_items =
List.map (fun c ->
let items =
XrefTable.fold (fun (name, pos) xref store ->
match xref with
| range, XrefTable.Defs defs ->
List.filter (fun (path, _) -> is_initial c path) defs
|> List.filter (fun (_, typ) -> typ <> "binder")
|> List.filter (fun (_, typ) -> typ <> "var")
|> List.filter (fun (path, _) -> not (is_subproof path))
|> List.map (fun (path, typ) ->
let linkname = !%"%s.html#%s" name (sanitize_linkname path) in
let module_ = name in
{kind=EntryKind typ; name=path; linkname; module_})
|> fun is -> is @ store
| range, Ref _ -> store) xref_table []
in
Hashtbl.fold (fun filename _ store ->
let basename = Str.(split (regexp_string ".") filename) |> List.rev |> List.hd in
if is_initial c basename then
let linkname = !%"%s.html" filename in
{kind=EntryKind "file"; name=basename; linkname; module_=filename} :: store
else store) xref_modules items
|> List.sort (fun x y -> compare (String.lowercase_ascii x.name)
(String.lowercase_ascii y.name))
|> fun items -> (c, items))
alphabets
in
let all_files = all_files xref_modules in
List.iter (fun kind ->
List.iter (generate_with_capital output_dir (table indexed_items) all_files kind) indexed_items)
kinds;
generate_topfile output_dir all_files indexed_items title xref_table hierarchy_dot_file dependency_dot_file