This repository has been archived by the owner on Jun 4, 2019. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 204
/
Copy pathmain_codemap.ml
705 lines (631 loc) · 24.6 KB
/
main_codemap.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
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
(*s: main_codemap.ml *)
(*
* Please imagine a long and boring gnu-style copyright notice
* appearing just here.
*)
open Common
module Flag = Flag_visual
module FT = File_type
module Model = Model2
(*****************************************************************************)
(* Prelude *)
(*****************************************************************************)
(*
* This is the main entry point of codemap, a semantic source code visualizer
* using treemaps and code thumbnails. The focus here is code understanding
* not editing, so for instance even if features like autocompletion are
* great for editing, they are not really helpful for understanding an existing
* codebase. What can help is completion to help navigate and go from one
* place to another, and this is one of the feature of this tool.
*
* requirements:
* - get a bird's eye view of all the code (hence treemaps)
* - get a bird's eye view of a file (hence code thumbnails)
* - better syntax highlighting than Emacs, using real parsers so
* we can colorize differently identifiers (a function vs a field vs
* a constant)
* - important code should be bigger. Just like in google maps
* the important roads are more visible. So need some sort of
* global semantic analysis.
* - show the data (the source code), but also show the relations
* (hence codegraph integration)
* - look at the code through different views (hence layers)
* - SEMI deep semantic analysis, control flow + data flow so can answer
* flow questions such as where a callback is called (hence datalog)
*
* history:
* - saw Aspect Browser while working on aspects as an intern at IRISA
* - work on Poffs and idea of visualizing the same code through
* different views
* - talked about mixing sgrep/spatch with code visualization,
* highlighting with a certain color different architecture aspects
* of the Linux kernel (influenced by work on aspect browser)
* - talked about fancy code visualizer while at cleanmake with YY,
* Spiros, etc.
* - saw SeeSoft code visualizer while doing some bibliographic work
* - saw code thumbnails by MSR, and Rob Deline
* - saw treemap of Linux kernel by Fekete => idea of mixing
* tree-map+code-thumbnails+seesoft = codemap
* - saw talk at CC'09 about improving javadoc by putting in bigger fonts
* really often used API functions => idea of light db and semantic
* visual feedback
* - read hierarchical edge bundling paper and its d3 implementation to
* visualize on top of a treemap the call graph
*
* related work:
* - racket IDE (was called DrScheme before), had arrows long time ago
* between occurences of a variable and its definition
* - http://peaker.github.io/lamdu/, but focused more on AST pretty printing
* - light table, interesting visualization slice but now focused more
* on live programming a la Bret Victor
* - http://www.kickstarter.com/projects/296054304/zeta-code, mostly focused
* on code relations, so related more to codegraph
* - textmate, nice fuzzy file find, very quick to go to a place
* - sublime, has thumbnails, but people don't really care about it
* - http://www.hello2morrow.com/products/sotoarc ?
* - http://scg.unibe.ch/codemap ?
* - http://scg.unibe.ch/wiki/projects/rbcrawler, class blueprint, very nice
* when exploring tangled object code abusing inheritance
* - moose http://youtu.be/yvXm9LC17vk at 14min
* - http://redotheweb.com/CodeFlower/ ?
* - code swarm, visualize git history, focused on people more than code
* https://code.google.com/p/gource/
* http://artzub.com/ghv/#repo=d3&climit=100&user=mbostock
* - http://www.codetrails.com/ctrlflow, smarter completion by infering
* importance of method (like I do, by #times this entity is globally used)
* - codesonar, very nice interactive zoomable call graph visualizer
* https://www.youtube.com/watch?v=EqDhtRoorGU also very deep semantic analysis
* with control flow and data flow
*
* features of IDE we do want (e.g. see the list at http://xamarin.com/studio):
* - smart syntax highlighting (we do even more: semantic highlighting!)
* - go to definition (=~ TAGS, light db and search bar completion provides it)
* - code navigation (directory, files, also "hypertext" go to def/uses)
* - find uses (funcs, classes, TODO tricky for methods in dynamic languages)
* - code tooltip, hover on use of an entity to display information about
* it (#uses, TODO: type/args, comments, code, age, methods, etc)
* - unified "fuzzy" search (files, entities, TODO but also content),
* SEMI project-wide fuzzy search is really useful for filenames but also content!
* - debugger? it helps understand code so a coverage layer or TODO live
* coverage tracing would be nice (as in tracegl),
* also a way to see the actual concrete values of variables/parameters
* inline would be nice too (as in intellij 14) (always-on visualization!)
* - source control? can extract age, number of authors, churn information in
* layers
* - SEMI dataflow from here, dataflow to here (intellij)
*
* features of IDE we care less about:
* - folding/outline? thumbnails make this less important
* - auto completion? One nice thing of autocomplete is that
* it proposes all the possible methods of an object, the overriden
* as well as not overriden parent methods. We don't need autocomplete
* but we want the ability to understand a class by TODO "inlining" parent
* methods that are relevant to understand the local code of the class
* (e.g. the short command of Eiffel)
* - code snippet? This is similar to auto completion, it's good for writing
* new code, but not that useful to understand existing code.
* - refactoring? no (but some spatch integration could be nice)
* - UI designer? no
* - deploy assistant, cloud assistant? no
*)
(*****************************************************************************)
(* Flags *)
(*****************************************************************************)
(*s: main flags *)
(* on Macos lion, X11 resizes the window to a smaller size so
* no point in starting with a big screen_size :(
*)
let screen_size = ref 1
let legend = ref true
(* you can also put this in your ~/gtkrc-2.0
* gtk-icon-theme-name = "Tango"
* gtk-theme-name = "Murrine-Gray"
* gtk-font-name = "DejaVu Sans 16"
*)
(* if not specified, codemap will try to use files in the current directory *)
let db_file = ref (None: Common.filename option)
let graph_file = ref (None: Common.filename option)
let layer_file = ref (None: Common.filename option)
let layer_dir = ref (None: Common.dirname option)
(* See also Gui.synchronous_actions *)
let test_mode = ref (None: string option)
(*e: main flags *)
(* see filters below, which files we are interested in *)
let filter = ref (fun _file -> true)
let skip_file = ref (None: Common.filename option)
(* less: a config file: GtkMain.Rc.add_default_file "/.../pfff_browser.rc"; *)
(* action mode *)
let action = ref ""
(*****************************************************************************)
(* Shortcuts *)
(*****************************************************************************)
let filters = [
(* pad specific, ocaml related *)
"pfff", (fun file ->
match FT.file_type_of_file file with
| FT.PL (
(FT.ML _) | FT.Makefile | FT.Opa | FT.Prolog _ | FT.Web (FT.Php _)) ->
not (
(* file =~ ".*commons/" || *)
(* file =~ ".*external/" || *)
file =~ ".*_build/")
| _ -> false
);
"xix", (fun file ->
match FT.file_type_of_file file with
| FT.PL ((FT.ML _) | FT.Makefile | (FT.C _ | FT.Asm)) -> true
| _ -> false
);
"ocaml", (fun file ->
match File_type.file_type_of_file file with
| FT.PL (FT.ML _) | FT.PL (FT.Makefile) -> true
| _ -> false
);
"mli", (fun file ->
match File_type.file_type_of_file file with
| FT.PL (FT.ML "mli") | FT.PL (FT.Makefile) ->
not (file =~ ".*/commons/")
| _ -> false
);
"nw", (fun file ->
match FT.file_type_of_file file with
| FT.Text "nw" -> true | _ -> false
);
"doc", (fun file ->
match FT.file_type_of_file file with
| FT.Text _ -> true | _ -> false
);
(* other languages *)
"php", (fun file ->
match File_type.file_type_of_file file with
| FT.PL (FT.Web (FT.Php _)) -> true | _ -> false
);
"js", (fun file ->
match File_type.file_type_of_file file with
| FT.PL (FT.Web (FT.Js)) -> true | _ -> false
);
"cpp", (let x = ref false in (fun file ->
Common2.once x (fun () ->
(* TODO: also add possible pfff_macros.h when there *)
Parse_cpp.init_defs !Flag_parsing_cpp.macros_h
);
match FT.file_type_of_file file with
| FT.PL (FT.C _ | FT.Cplusplus _) -> true
| FT.PL FT.Asm -> true
| _ -> false
));
(* exotic languages *)
"exotic", (fun file ->
match FT.file_type_of_file file with
| FT.PL (FT.Opa | FT.Rust) -> true
(* | FT.PL (FT.Web (_)) -> true *)
| _ -> false
);
(* general categories *)
"pl", (fun file ->
match File_type.file_type_of_file file with
| FT.PL _ -> true | _ -> false
);
]
(*****************************************************************************)
(* Helpers *)
(*****************************************************************************)
let set_gc () =
if !Flag.debug_gc
then Gc.set { (Gc.get()) with Gc.verbose = 0x01F };
(* only relevant in bytecode, in native the stacklimit is the os stacklimit *)
Gc.set {(Gc.get ()) with Gc.stack_limit = 1000 * 1024 * 1024};
(* see www.elehack.net/michael/blog/2010/06/ocaml-memory-tuning *)
Gc.set { (Gc.get()) with Gc.minor_heap_size = 4_000_000 };
Gc.set { (Gc.get()) with Gc.major_heap_increment = 8_000_000 };
Gc.set { (Gc.get()) with Gc.space_overhead = 300 };
()
(*****************************************************************************)
(* Model helpers *)
(*****************************************************************************)
(*s: treemap_generator *)
(* this is called each time we go in a new directory (or set of dirs) *)
let treemap_generator ~filter_file =
fun paths ->
let treemap = Treemap_pl.code_treemap ~filter_file paths in
let algo = Treemap.Ordered Treemap.PivotByMiddle in
let big_borders = !Flag.boost_label_size in
let rects = Treemap.render_treemap ~algo ~big_borders treemap in
Common.pr2 (spf "%d rectangles to draw" (List.length rects));
rects
(*e: treemap_generator *)
(*s: build_model *)
(* this is currently called in the background *)
let build_model2 root dbfile_opt graphfile_opt =
let db_opt = dbfile_opt +> Common.map_opt Database_code.load_database in
(* todo: and skip_list?
* less: opti by factorizing the 'find' with treemap_generator?
*)
let files =
Common.files_of_dir_or_files_no_vcs_nofilter [root] +> List.filter !filter
in
let hentities = Model_database_code.hentities root db_opt in
let all_entities = Model_database_code.all_entities ~root files db_opt in
let big_grep_idx = Completion2.build_completion_defs_index all_entities in
let g_opt = graphfile_opt +> Common.map_opt Graph_code.load in
let hfile_deps_of_node, hentities_of_file =
match g_opt with
| None -> Hashtbl.create 0, Hashtbl.create 0
| Some g ->
let a = Model_graph_code.build_filedeps_of_dir_or_file g in
let b = Model_graph_code.build_entities_of_file g in
let b = Model_graph_code.add_headers_files_entities_of_file root b in
a, Common.hash_of_list b
in
let model = { Model.
root = root;
db = db_opt;
hentities; big_grep_idx;
g = g_opt;
hfile_deps_of_node; hentities_of_file;
}
in
model
let build_model a b c =
Common.profile_code "View.build_model" (fun () ->
build_model2 a b c)
(*e: build_model *)
(* could also to parse all json files and filter the one which do not parse *)
let layers_in_dir dir =
Common2.readdir_to_file_list dir +> Common.map_filter (fun file ->
if file =~ "layer.*json"
then Some (Filename.concat dir file)
else None
)
(*****************************************************************************)
(* Main action *)
(*****************************************************************************)
(*s: main_action() *)
let main_action xs =
set_gc ();
Logger.log Config_pfff.logger "codemap" None;
(* this used to be done by linking with gtkInit.cmo, but better like this *)
let _locale = GtkMain.Main.init () in
pr2 (spf "Using Cairo version: %s" Cairo.compile_time_version_string);
let root = Common2.common_prefix_of_files_or_dirs xs in
pr2 (spf "Using root = %s" root);
let async_model = Async.async_make () in
let layers =
match !layer_file, !layer_dir, xs with
| Some file, _, _ ->
[Layer_code.load_layer file]
| None, Some dir, _ | None, None, [dir] ->
layers_in_dir dir +> List.map Layer_code.load_layer
| _ -> []
in
let layers_with_index =
Layer_code.build_index_of_layers ~root
(match !layer_file, layers with
| Some _, [layer] -> [layer, true]
| _ -> layers +> List.map (fun x -> x, false)
)
in
let db_file =
match !db_file, xs with
| Some file, _ -> Some file
| None, [dir] ->
let candidates = [
Filename.concat dir Database_code.default_db_name;
Filename.concat dir Database_code.default_db_name ^ ".json";
] in
(try
Some (candidates +> List.find (fun file -> Sys.file_exists file))
with Not_found -> None
)
| _ -> None
in
db_file +> Common.do_option (fun db ->
pr2 (spf "Using pfff light db: %s" db)
);
let graph_file =
match !graph_file, xs with
| Some file, _ -> Some file
| None, [dir] ->
let candidates = [
Filename.concat dir Graph_code.default_filename;
] in
(try
Some (candidates +> List.find (fun file -> Sys.file_exists file))
with Not_found -> None
)
| _ -> None
in
graph_file +> Common.do_option (fun db ->
pr2 (spf "Using graphcode: %s" db)
);
let skip_file = !skip_file ||| Filename.concat root "skip_list.txt" in
let skip_list =
if Sys.file_exists skip_file
then begin
pr2 (spf "Using skip file: %s" skip_file);
Skip_code.load skip_file
end
else []
in
let filter_files_skip_list = Skip_code.filter_files skip_list root in
let filter_file = (fun file ->
!filter file &&
(skip_list = [] || filter_files_skip_list [file] <> []))
in
let treemap_func = treemap_generator ~filter_file in
let dw = Model.init_drawing treemap_func layers_with_index xs root in
(* This can require lots of stack. Make sure to have ulimit -s 40000 *)
Thread.create (fun () ->
(* heavy computation are not *fairly* scheduled apparently by the OCaml
* runtime, so let's do the heavy computation in another process
* and here just have the thread waiting for it to be done.
* This thread used to cause some Bus error on MacOS but now that
* we use invoke and do the job in another process things seems better :)
*)
let job () = build_model root db_file graph_file in
let res = Parallel.invoke job () () in
Async.async_set res async_model;
) ()
+> ignore;
let w = { Model.
dw;
dw_stack = ref [dw];
model = async_model;
treemap_func;
current_node = None;
current_node_selected = None;
current_entity = None;
settings = { Model.
(* todo: too fuzzy for now *)
draw_summary = false;
draw_searched_rectangles = true;
};
root_orig = root;
}
in
View2.mk_gui ~screen_size:!screen_size ~legend:!legend !test_mode w
(*e: main_action() *)
(*****************************************************************************)
(* Extra actions *)
(*****************************************************************************)
(* related work: http://cloc.sourceforge.net/ but have skip list
* and archi_code_lexer.mll which lower the important of some files?
*)
let test_loc print_top30 xs =
let xs = xs |> List.map Common.realpath in
let root = Common2.common_prefix_of_files_or_dirs xs in
let skip_file = !skip_file ||| Filename.concat root "skip_list.txt" in
let skip_list =
if Sys.file_exists skip_file
then begin
pr2 (spf "Using skip file: %s" skip_file);
Skip_code.load skip_file
end
else []
in
let filter_files_skip_list = Skip_code.filter_files skip_list root in
let filter_file = (fun file ->
!filter file && (skip_list = [] || filter_files_skip_list [file] <> []))
in
let treemap = Treemap_pl.code_treemap ~filter_file xs in
let res = ref [] in
let rec aux tree =
match tree with
| Common2.Node (_dir, xs) ->
List.iter aux xs
| Common2.Leaf (leaf, _) ->
let file = leaf.Treemap.label in
let size = leaf.Treemap.size in
let unix_size = (Common2.unix_stat_eff file).Unix.st_size in
if unix_size > 0
then begin
let multiplier = (float_of_int size /. float_of_int unix_size) in
let multiplier = min multiplier 1.0 in
let loc = Common2.nblines_with_wc file in
Common.push ((Common.readable ~root file),
(float_of_int loc *. multiplier)) res;
end
in
aux treemap;
let total = !res +> List.map snd +> List.map int_of_float +> Common2.sum in
pr2 (spf "LOC = %d (%d files)" total (List.length !res));
if print_top30 then begin
let topx = 30 in
pr2 (spf "Top %d:" topx);
!res +> Common.sort_by_val_highfirst +> Common.take_safe topx
+> List.iter (fun (file, f) ->
pr2 (spf "%-40s: %d" file (int_of_float f))
)
end
let test_treemap_dirs () =
let paths =
["commons/common.ml"; "h_visualization"; "code_graph"]
+> List.map Common.realpath in
let paths = List.sort String.compare paths in
let tree =
paths +> Treemap.tree_of_dirs_or_files
~filter_dir:Lib_vcs.filter_vcs_dir
~filter_file:(fun file -> file =~ ".*\\.ml")
~file_hook:(fun _file -> 10)
in
pr2_gen tree
(* update: try to put ocamlgtk related tests in widgets/test_widgets.ml, not
* here. Here it's for ... well it's for nothing I think because it's not
* really easy to test a gui.
*)
(*s: visual_commitid() action *)
let test_visual_commitid id =
let files = Common.cmd_to_list
(spf "git show --pretty=\"format:\" --name-only %s"
id)
(* not sure why git adds an extra empty line at the beginning but we
* have to filter it
*)
+> Common.exclude Common.null_string
in
pr2_gen files;
main_action files
(*e: visual_commitid() action *)
let width = 500
let height = 500
let test_draw cr =
(* [0,0][1,1] world scaled to a width x height screen *)
Cairo.scale cr (float_of_int width) (float_of_int height);
Cairo.set_source_rgba cr ~red:0.5 ~green:0.5 ~blue:0.5 ~alpha:0.5;
Cairo.set_line_width cr 0.001;
Cairo.move_to cr 0.5 0.5;
Cairo.line_to cr 0.6 0.6;
Cairo.stroke cr;
Cairo.select_font_face cr "serif"
Cairo.FONT_SLANT_NORMAL Cairo.FONT_WEIGHT_BOLD;
Cairo.set_font_size cr 0.1;
Cairo.move_to cr 0.1 0.1;
Cairo.show_text cr "THIS IS SOME TEXT";
Cairo.move_to cr 0.1 0.2;
Cairo.show_text cr "THIS IS SOME TEXT";
Cairo.set_font_size cr 0.05;
Cairo.move_to cr 0.1 0.3;
Cairo.show_text cr "THIS IS SOME TEXT";
Cairo.set_source_rgb cr ~red:0.1 ~green:0.1 ~blue:0.1;
Cairo.move_to cr 0.1 0.1;
Cairo.line_to cr 0.1 0.2;
Cairo.stroke cr;
let start = ref 0.0 in
for _i = 0 to 3 do
let end_ = !start +. 0.5 in
Cairo.arc cr ~xc:0.5 ~yc:0.5 ~radius:0.3 ~angle1:!start
~angle2:end_;
Cairo.stroke cr;
start := end_;
done;
()
let test_cairo () =
let _locale = GtkMain.Main.init () in
let w = GWindow.window ~title:"test" () in
(w#connect#destroy GMain.quit) +> ignore;
let px = GDraw.pixmap ~width ~height ~window:w () in
px#set_foreground `WHITE;
px#rectangle ~x:0 ~y:0 ~width ~height ~filled:true ();
let cr = Cairo_lablgtk.create px#pixmap in
test_draw cr;
(GMisc.pixmap px ~packing:w#add ()) +> ignore;
w#show ();
GMain.main()
(*---------------------------------------------------------------------------*)
(* the command line flags *)
(*---------------------------------------------------------------------------*)
let extra_actions () = [
(*s: actions *)
"-test_loc", " ",
Common.mk_action_n_arg (test_loc true);
"-test_loc2", " ",
Common.mk_action_n_arg (test_loc false);
"-test_cairo", " ",
Common.mk_action_0_arg (test_cairo);
"-test_commitid", " <id>",
Common.mk_action_1_arg (test_visual_commitid);
"-test_treemap_dirs", " <id>",
Common.mk_action_0_arg (test_treemap_dirs);
(*e: actions *)
]
(*****************************************************************************)
(* The options *)
(*****************************************************************************)
let all_actions () =
extra_actions()@
[]
let options () = ([
(*s: options *)
"-screen_size", Arg.Set_int screen_size,
" <int> (1 = small, 2 = big)";
"-no_legend", Arg.Clear legend,
" do not display the legend";
"-symlinks", Arg.Unit (fun () -> Treemap.follow_symlinks := true;),
" follow symlinks";
"-no_symlinks", Arg.Unit (fun () -> Treemap.follow_symlinks := false),
" do not follow symlinks";
"-with_graph", Arg.String (fun s -> graph_file := Some s),
" <graph_file> dependency semantic information";
"-with_db", Arg.String (fun s -> db_file := Some s),
" <db_file> generic semantic information";
"-with_layer", Arg.String (fun s -> layer_file := Some s),
" <layer_file>";
"-with_layers", Arg.String (fun s -> layer_dir := Some s),
" <dir_with_layers>";
"-filter", Arg.String (fun s -> filter := List.assoc s filters;),
spf " filter certain files (available = %s)"
(filters +> List.map fst +> Common.join ", ");
"-extra_filter", Arg.String (fun s -> Flag.extra_filter := Some s),
" ";
"-skip_list", Arg.String (fun s -> skip_file := Some s),
" <file> skip files or directories";
"-ft", Arg.Set_float Flag.threshold_draw_content_font_size_real,
" <float> threshold to draw content";
"-nblines_file", Arg.Set_float Flag.threshold_draw_content_nblines,
" <float>";
"-boost_lbl", Arg.Set Flag.boost_label_size,
" boost size of labels";
"-emacs_client", Arg.Set_string Editor_connection.emacsclient_path,
" <path>";
(*-------------------------------------------------------------------------*)
(* debugging helpers *)
(*-------------------------------------------------------------------------*)
"-test", Arg.String (fun s -> test_mode := Some s),
" <str> execute an internal script";
"-verbose", Arg.Set Flag.verbose_visual,
" ";
"-debug_gc", Arg.Set Flag.debug_gc,
" ";
"-debug_handlers", Arg.Set Gui.synchronous_actions,
" ";
(* "-disable_ancient", Arg.Clear Flag.use_ancient, " "; *)
"-disable_fonts", Arg.Set Flag.disable_fonts,
" ";
(*e: options *)
] @
Common.options_of_actions action (all_actions()) @
Common2.cmdline_flags_devel () @
[
"-version", Arg.Unit (fun () ->
pr2 (spf "CodeMap version: %s" Config_pfff.version);
exit 0;
),
" guess what";
]) |> Arg.align
(*****************************************************************************)
(* The main entry point *)
(*****************************************************************************)
let main () =
let usage_msg =
spf "Usage: %s [options] <file or dir> \nDoc: %s\nOptions:"
(Filename.basename Sys.argv.(0))
"https://github.com/facebook/pfff/wiki/Codemap"
in
let args = Common.parse_options (options()) usage_msg Sys.argv in
(* must be done after Arg.parse, because Common.profile is set by it *)
Common.profile_code "Main total" (fun () ->
(match args with
(* --------------------------------------------------------- *)
(* actions, useful to debug subpart *)
(* --------------------------------------------------------- *)
| xs when List.mem !action (Common.action_list (all_actions())) ->
Common.do_action !action xs (all_actions())
| _ when not (Common.null_string !action) ->
failwith ("unrecognized action or wrong params: " ^ !action)
(* --------------------------------------------------------- *)
(* main entry *)
(* --------------------------------------------------------- *)
| (x::xs) ->
main_action (x::xs)
(* --------------------------------------------------------- *)
(* empty entry *)
(* --------------------------------------------------------- *)
| [] -> Arg.usage (options()) usage_msg;
);
)
(*****************************************************************************)
let _ =
Common.main_boilerplate (fun () ->
main ()
)
(*e: main_codemap.ml *)