File: logger.rkt

package info (click to toggle)
racket-mode 20251013~git.b9a4f51-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 2,036 kB
  • sloc: lisp: 17,282; makefile: 106
file content (77 lines) | stat: -rw-r--r-- 2,963 bytes parent folder | download | duplicates (2)
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
;; Copyright (c) 2013-2022, 2025 by Greg Hendershott.
;; SPDX-License-Identifier: GPL-3.0-or-later

#lang at-exp racket/base

(require racket/match
         racket/format)

(provide (rename-out [command-channel logger-command-channel]
                     [notify-channel logger-notify-channel]))

;; "On start-up, Racket creates an initial logger that is used to
;; record events from the core run-time system. For example, an 'debug
;; event is reported for each garbage collection (see Garbage
;; Collection)." Use that; don't create new one. See issue #325.
(define global-logger (current-logger))

(define command-channel (make-channel))
(define notify-channel (make-channel))

;; Go ahead and start our log receiver thread early so we can see our
;; own racket-mode topic's 'debug level ouput in the front end.
;;
;; On the other hand (see #631) set all other topics to the 'fatal
;; level (least noisy). This avoids sending excessive logger
;; notifications to the front end, until/unless it gives us the user's
;; logger configuration, with whatever verbosity they desire.
(define (racket-mode-log-receiver-thread)
  (let wait ([receiver (make-receiver '((racket-mode . debug)
                                        (*           . fatal)))])
    (sync
     (handle-evt command-channel
                 (λ (v)
                   (wait (make-receiver v))))
     (handle-evt receiver
                 (match-lambda
                   [(vector level message _v topic)
                    (channel-put notify-channel
                                 `(logger
                                   ,(cons level
                                          (topic+message topic message))))
                    (wait receiver)])))))
(void (thread racket-mode-log-receiver-thread))

(define (topic+message topic message)
  (match message
    [(pregexp (format "^~a: (.*)$" (regexp-quote (~a topic)))
              (list _ message))
     (list topic
           message)]
    [message-without-topic
     (list (or topic '*)
           message-without-topic)]))

(module+ test
  (require rackunit)
  (check-equal? (topic+message 'topic "message")
                (list 'topic "message"))
  (check-equal? (topic+message 'topic "topic: message")
                (list 'topic "message"))
  (check-equal? (topic+message #f "message")
                (list '* "message")))

(define (make-receiver alist)
  (apply make-log-receiver (list* global-logger
                                  (alist->spec alist))))

;; Convert from ([logger . level] ...) alist to the format used by
;; make-log-receiver: (level logger ... ... default-level). In the
;; alist, treat the logger '* as the default level.
(define (alist->spec xs) ;(Listof (Pairof Symbol Symbol)) -> (Listof Symbol)
  (for/fold ([spec '()])
            ([x (in-list xs)])
    (append spec
            (match x
              [(cons '*     level) (list level)]
              [(cons logger level) (list level logger)]))))