forked from facebook/pyre-check
-
Notifications
You must be signed in to change notification settings - Fork 0
/
log.ml
179 lines (148 loc) · 4.76 KB
/
log.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
(* Copyright (c) 2016-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree. *)
open Core
type section =
[ `Check
| `Coverage
| `Debug
| `Dependencies
| `DependencyGraph
| `Dotty
| `Dump
| `Environment
| `Error
| `Event
| `Fixpoint
| `Info
| `Interprocedural
| `Memory
| `Parser
| `Performance
| `Progress
| `Protocols
| `Server
| `Taint
| `Warning
]
let section_to_string = function
| `Check -> "Check"
| `Coverage -> "Coverage"
| `Debug -> "Debug"
| `Dependencies -> "Dependencies"
| `DependencyGraph -> "DependencyGraph"
| `Dotty -> "Dotty"
| `Dump -> "Dump"
| `Environment -> "Environment"
| `Error -> "Error"
| `Event -> "Event"
| `Fixpoint -> "Fixpoint"
| `Info -> "Info"
| `Interprocedural -> "Interprocedural"
| `Memory -> "Memory"
| `Parser -> "Parser"
| `Performance -> "Performance"
| `Progress -> "Progress"
| `Protocols -> "Protocols"
| `Server -> "Server"
| `Taint -> "Taint"
| `Warning -> "Warning"
module GlobalState = struct
let enabled =
String.Hash_set.of_list
["Dump"; "Error"; "Info"; "Memory"; "Progress"; "Performance"; "Warning"]
let initialize ~debug ~sections =
if debug then
Hash_set.add enabled "Debug";
let handle_section section =
let normalize section = String.lowercase section |> String.capitalize in
match String.chop_prefix ~prefix:"-" section with
| Some section -> normalize section |> Hash_set.remove enabled
| None -> normalize section |> Hash_set.add enabled
in
List.iter ~f:handle_section sections
let initialize_for_tests () =
Hash_set.clear enabled;
Hash_set.add enabled "Dump"
type t = string list
let get () = Hash_set.to_list enabled
let restore saved_state =
Hash_set.clear enabled;
List.iter saved_state ~f:(Hash_set.add enabled)
end
let is_enabled section = Hash_set.mem GlobalState.enabled (section_to_string section)
let time_zone = ref None
(* A safer version of Time.Zone.local, which defaults to UTC instead of throwing an exception if we
cannot figure out local time. See https://github.com/janestreet/core/issues/96 for one example
when this can happen *)
let get_time_zone () =
match !time_zone with
| Some zone -> zone
| None ->
let zone =
try force Time.Zone.local with
| _ -> Time.Zone.utc
in
time_zone := Some zone;
zone
let log ~section format =
let section = section_to_string section in
if Hash_set.mem GlobalState.enabled section then
let zone = get_time_zone () in
Format.fprintf
Format.err_formatter
("%s %s " ^^ format ^^ "\n%!")
(Time.format ~zone (Time.now ()) "%Y-%m-%d %H:%M:%S")
(String.uppercase section)
else
Format.ifprintf Format.err_formatter format
let debug format = log ~section:`Debug format
let dump format = log ~section:`Dump format
let info format = log ~section:`Info format
let error format = log ~section:`Error format
let warning format = log ~section:`Warning format
let print format = Printf.printf format
let log_unix_error ?(section = `Error) (error_kind, name, parameters) =
log ~section "Unix error %s: %s(%s)" (Unix.error_message error_kind) name parameters
module Color = struct
let cyan string = Format.asprintf "\027[36m%s\027[0m" string
let red string = Format.asprintf "\027[31m%s\027[0m" string
let yellow string = Format.asprintf "\027[33m%s\027[0m" string
end
let rotate ?(number_to_keep = 10) basename =
let timestamp = Time.to_filename_string ~zone:(get_time_zone ()) (Time.now ()) in
let suppress_system_error f =
try f () with
| Sys_error _
| Unix.Unix_error _ ->
()
in
let rotate_old_logs () =
Filename.dirname basename
|> Sys.ls_dir
(* The "." is to prevent us from counting a symlinked log as a log to keep. *)
|> List.filter ~f:(String.is_prefix ~prefix:(Filename.basename basename ^ "."))
|> List.sort ~compare:String.compare (* Sorts by earliest date, i.e. least recent *)
|> List.rev
|> (fun list -> List.drop list number_to_keep)
|> List.iter ~f:(fun path ->
suppress_system_error (fun () -> Unix.remove (Filename.dirname basename ^/ path)))
in
suppress_system_error rotate_old_logs;
let is_file_or_link path =
try
let { Unix.st_kind; _ } = Unix.lstat path in
match st_kind with
| Unix.S_LNK
| Unix.S_REG ->
true
| _ -> false
with
| Unix.Unix_error _ -> false
in
if is_file_or_link basename then
suppress_system_error (fun () -> Unix.unlink basename);
let actual_path = Format.sprintf "%s.%s" basename timestamp in
suppress_system_error (fun () -> Unix.symlink ~target:actual_path ~link_name:basename);
actual_path