scsh-0.6/scsh/syslog.scm

259 lines
7.2 KiB
Scheme

; Copyright (c) 1993-2000 by Richard Kelsey and Jonathan Rees. See file COPYING.
;; Options for openlog
(define-finite-type syslog-option :syslog-option
(mask)
syslog-option?
the-syslog-options
syslog-option-name
syslog-option-index
(mask syslog-option-mask)
;; These values are known to the C code.
((console #o01)
(delay #o02)
(no-delay #o04)
(standard-error #o10)
(log-pid #o20)))
(define-record-type syslog-options :syslog-options
(really-make-syslog-options value)
syslog-options?
(value syslog-options-value))
(define syslog-options=? eq?)
(define-exported-binding "syslog-options-type" :syslog-options)
(define (syslog-options-on? options0 options1)
(= 0 (bitwise-and (syslog-options-value options1)
(bitwise-not (syslog-options-value options0)))))
(define (make-syslog-options options)
(really-make-syslog-options
(apply bitwise-ior (map syslog-option-mask options))))
(define default-syslog-options (make-syslog-options '()))
(define (make-mask-record->list get-value get-mask record-vector)
(lambda (mask-record)
(let ((value (get-value mask-record))
(n-syslog-options (vector-length record-vector)))
(let loop ((i 0) (list '()))
(cond
((>= i n-syslog-options)
list)
((zero? (bitwise-and value
(get-mask (vector-ref record-vector i))))
(loop (+ 1 i) list))
(else
(loop (+ 1 i) (cons (vector-ref record-vector i)
list))))))))
(define syslog-options->list
(make-mask-record->list syslog-options-value
syslog-option-mask
the-syslog-options))
; Simplifying syntax, e.g. (syslog-options delay console)
(define-syntax syslog-options
(syntax-rules ()
((syslog-options name ...)
(make-syslog-options (list (syslog-option name) ...)))))
(define-enumerated-type syslog-facility :syslog-facility
syslog-facility?
syslog-facilities
syslog-facility-name
syslog-facility-index
;; Options for openlog
;; The order of these is known to the C code.
(authorization
cron
daemon
kernel
lpr
mail
news
user
uucp
local0 local1 local2 local3 local4 local5 local6 local7))
(define syslog-facility=? eq?)
(define default-syslog-facility (syslog-facility user))
(define-exported-binding "syslog-facility-type" :syslog-facility)
(define-exported-binding "syslog-facilities" syslog-facilities)
(define-finite-type syslog-level :syslog-level
(mask)
syslog-level?
syslog-levels
syslog-level-name
syslog-level-index
(mask syslog-level-mask)
;; Options for syslog
;; The order and the values of these is known to the C code.
((emergency #o001)
(alert #o002)
(critical #o004)
(error #o010)
(warning #o020)
(notice #o040)
(info #o100)
(debug #o200)))
(define syslog-level=? eq?)
(define-exported-binding "syslog-level-type" :syslog-level)
(define-exported-binding "syslog-levels" syslog-levels)
(define-record-type syslog-mask :syslog-mask
(make-syslog-mask value)
syslog-mask?
(value syslog-mask-value))
(define (syslog-mask=? mask-1 mask-2)
(= (syslog-mask-value mask-1)
(syslog-mask-value mask-2)))
(define-exported-binding "syslog-mask-type" :syslog-mask)
(define (syslog-mask-levels-on? mask-1 mask-2)
(= 0 (bitwise-and (syslog-mask-value mask-2)
(bitwise-not (syslog-mask-value mask-1)))))
(define (levels->syslog-mask levels)
(make-syslog-mask
(apply bitwise-ior
(map syslog-level-mask levels))))
(define syslog-mask->levels
(make-mask-record->list syslog-mask-value
syslog-level-mask
syslog-levels))
(define-syntax syslog-mask
(syntax-rules ()
((syslog-mask name ...)
(levels->syslog-mask (list (syslog-level name) ...)))))
(define (syslog-mask-upto level)
(let loop ((index (syslog-level-index level)) (levels '()))
(if (< index 0)
(levels->syslog-mask levels)
(loop (- index 1) (cons index levels)))))
(define syslog-mask-all (levels->syslog-mask (vector->list syslog-levels)))
(define default-syslog-mask syslog-mask-all)
(import-lambda-definition openlog (ident options facility)
"sch_openlog")
(import-lambda-definition unix-syslog (level opt-facility message)
"sch_syslog")
(import-lambda-definition setlogmask! (logmask)
"sch_setlogmask")
(import-lambda-definition closelog ()
"sch_closelog")
(define-record-type syslog-channel :syslog-channel
(make-syslog-channel ident options facility mask)
syslog-channel?
(ident syslog-channel-ident)
(options syslog-channel-options)
(facility syslog-channel-facility)
(mask syslog-channel-mask))
(define (syslog-channel-equivalent? channel-1 channel-2)
(and (string=? (syslog-channel-ident channel-1)
(syslog-channel-ident channel-2))
(syslog-options=? (syslog-channel-options channel-1)
(syslog-channel-options channel-2))
;; facility can be specified with each syslog-write
(syslog-mask=? (syslog-channel-mask channel-1)
(syslog-channel-mask channel-2))))
(define current-syslog-channel #f)
(define current-syslog-channel-lock (make-lock))
(define open-syslog-channel make-syslog-channel)
(define (close-syslog-channel channel)
(obtain-lock current-syslog-channel-lock)
(if (syslog-channel-equivalent? channel
current-syslog-channel)
(closelog))
(release-lock current-syslog-channel-lock))
;; THUNK must not escape
(define (with-syslog-channel channel thunk)
(obtain-lock current-syslog-channel-lock)
(if (or (not current-syslog-channel)
(not (syslog-channel-equivalent? channel
current-syslog-channel)))
(begin
(if current-syslog-channel
(closelog))
(openlog (syslog-channel-ident channel)
(syslog-channel-options channel)
(syslog-channel-facility channel))
(if (not (syslog-mask=? (syslog-channel-mask channel)
default-syslog-mask))
(setlogmask! (syslog-channel-mask channel)))
(set! current-syslog-channel channel)))
(thunk)
(release-lock current-syslog-channel-lock))
(define (syslog-write level message channel)
(with-syslog-channel
channel
(lambda ()
(unix-syslog level (syslog-channel-facility channel) message))))
(define (list-ref-carefully list n default)
(cond
((null? list) default)
((zero? n) (car list))
(else
(list-ref-carefully (cdr list) (- n 1) default))))
(define (change-syslog-channel channel . rest)
(let ((ident (list-ref-carefully rest 0 #f))
(options (list-ref-carefully rest 1 #f))
(facility (list-ref-carefully rest 2 #f))
(mask (list-ref-carefully rest 3 #f)))
(make-syslog-channel (or ident
(syslog-channel-ident channel))
(or options
(syslog-channel-options channel))
(or facility
(syslog-channel-facility channel))
(or mask
(syslog-channel-mask channel)))))
(define dynamic-syslog-channel
(make-thread-fluid
(make-syslog-channel "scheme48"
default-syslog-options
default-syslog-facility
default-syslog-mask)))
(define (syslog level message . rest)
(syslog-write level message
(if (and (not (null? rest))
(null? (cdr rest))
(syslog-channel? (car rest)))
(car rest)
;; this might be a little excessive allocation
(apply change-syslog-channel
(thread-fluid dynamic-syslog-channel)
rest))))
(define (with-syslog-destination ident options facility mask thunk)
(let-thread-fluid dynamic-syslog-channel
(make-syslog-channel ident options facility mask)
thunk))