package ocsigenserver

  1. Overview
  2. Docs
Legend:
Page
Library
Module
Module type
Parameter
Class
Class type
Source

Source file ocsigen_messages.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
(* Ocsigen
 * Copyright (C) 2005 Vincent Balat
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, with linking exception;
 * either version 2.1 of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 *)

(** Writing messages in the logs *)

let ( >>= ) = Lwt.bind
let access_file = "access.log"
let warning_file = "warnings.log"
let error_file = "errors.log"
let access_sect = Lwt_log.Section.make "ocsigen:access"
let full_path f = Filename.concat (Ocsigen_config.get_logdir ()) f
let error_log_path () = full_path error_file
let stderr = Lwt_log.channel ~close_mode:`Keep ~channel:Lwt_io.stderr ()
let stdout = Lwt_log.channel ~close_mode:`Keep ~channel:Lwt_io.stdout ()
let loggers = ref []
let access_logger = ref Lwt_log_core.null

let open_files () =
  (* CHECK: we are closing asynchronously!  That should be ok, though. *)
  List.iter (fun l -> ignore (Lwt_log.close l : unit Lwt.t)) !loggers;
  match Ocsigen_config.get_syslog_facility () with
  | Some facility ->
      (* log to syslog *)
      let syslog = Lwt_log.syslog ~facility () in
      loggers := [syslog];
      Lwt_log.default := Lwt_log.broadcast [syslog; stderr];
      Lwt.return ()
  | None ->
      (* log to files *)
      let open_log path =
        let path = full_path path in
        Lwt.catch
          (fun () -> Lwt_log.file ~file_name:path ())
          (function
             | Unix.Unix_error (error, _, _) ->
                 Lwt.fail
                   (Ocsigen_config.Config_file_error
                      (Printf.sprintf "can't open log file %s: %s" path
                         (Unix.error_message error)))
             | exn -> Lwt.fail exn)
      in
      open_log access_file >>= fun acc ->
      access_logger := acc;
      open_log warning_file >>= fun war ->
      open_log error_file >>= fun err ->
      loggers := [acc; war; err];
      Lwt_log.default :=
        Lwt_log.broadcast
          [ Lwt_log.dispatch (fun _sect lev ->
              match lev with
              | Lwt_log.Error | Lwt_log.Fatal -> err
              | Lwt_log.Warning -> war
              | _ -> Lwt_log.null)
          ; Lwt_log.dispatch (fun _sect lev ->
              if Ocsigen_config.get_silent ()
              then Lwt_log.null
              else
                match lev with
                | Lwt_log.Warning | Lwt_log.Error | Lwt_log.Fatal -> stderr
                | _ -> stdout) ];
      Lwt.return ()

(****)

let accesslog s =
  (* not really fatal, but log in all cases; does not affect console *)
  Lwt_log.ign_fatal ~section:access_sect ~logger:!access_logger s;
  Lwt_log.ign_notice ~section:access_sect s

let errlog ?section s = Lwt_log.ign_error ?section s
let warning ?section s = Lwt_log.ign_warning ?section s

let unexpected_exception e s =
  Lwt_log.ign_warning_f ~exn:e "Unexpected exception in %s" s

(****)

let console =
  if not (Ocsigen_config.get_silent ())
  then fun s -> print_endline (s ())
  else fun _s -> ()

let level_of_string = function
  | "debug" -> Some Lwt_log.Debug
  | "info" -> Some Lwt_log.Info
  | "notice" -> Some Lwt_log.Notice
  | "warning" -> Some Lwt_log.Warning
  | "error" -> Some Lwt_log.Error
  | "fatal" -> Some Lwt_log.Fatal
  | _ -> None

let command_f exc _ = function
  | [sect_name] ->
      (* Lwt_log.Section.make :
       if a section with the same name
       already exists, it is returned. *)
      let sect = Lwt_log.Section.make sect_name in
      Lwt_log.Section.reset_level sect;
      Lwt.return_unit
  | [sect_name; level_name] ->
      (* Lwt_log.Section.make :
       if a section with the same name
       already exists, it is returned. *)
      let sect = Lwt_log.Section.make sect_name in
      (match level_of_string (String.lowercase_ascii level_name) with
      | None -> Lwt_log.Section.reset_level sect
      | Some l -> Lwt_log.Section.set_level sect l);
      Lwt.return ()
  | _ -> Lwt.fail exc
OCaml

Innovation. Community. Security.