{- arch-tag: Simple log handlers Copyright (C) 2004-2006 John Goerzen 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; 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA -} {- | Module : System.Log.Handler.Simple Copyright : Copyright (C) 2004-2006 John Goerzen License : GNU LGPL, version 2.1 or above Maintainer : John Goerzen Stability : provisional Portability: portable Simple log handlers Written by John Goerzen, jgoerzen\@complete.org -} module System.Log.Handler.Simple(streamHandler, fileHandler, GenericHandler (..), verboseStreamHandler) where import System.Log import System.Log.Handler import System.IO import Control.Concurrent.MVar {- | A helper data type. -} data GenericHandler a = GenericHandler {priority :: Priority, privData :: a, writeFunc :: a -> LogRecord -> String -> IO (), closeFunc :: a -> IO () } instance LogHandler (GenericHandler a) where setLevel sh p = sh{priority = p} getLevel sh = priority sh emit sh lr loggername = (writeFunc sh) (privData sh) lr loggername close sh = (closeFunc sh) (privData sh) {- | Create a stream log handler. Log messages sent to this handler will be sent to the stream used initially. Note that the 'close' method will have no effect on stream handlers; it does not actually close the underlying stream. -} streamHandler :: Handle -> Priority -> IO (GenericHandler Handle) streamHandler h pri = do lock <- newMVar () let mywritefunc hdl (_, msg) _ = withMVar lock (\_ -> do hPutStrLn hdl msg hFlush hdl ) return (GenericHandler {priority = pri, privData = h, writeFunc = mywritefunc, closeFunc = \x -> return ()}) {- | Create a file log handler. Log messages sent to this handler will be sent to the filename specified, which will be opened in Append mode. Calling 'close' on the handler will close the file. -} fileHandler :: FilePath -> Priority -> IO (GenericHandler Handle) fileHandler fp pri = do h <- openFile fp AppendMode sh <- streamHandler h pri return (sh{closeFunc = hClose}) {- | Like 'streamHandler', but note the priority and logger name along with each message. -} verboseStreamHandler :: Handle -> Priority -> IO (GenericHandler Handle) verboseStreamHandler h pri = do lock <- newMVar () let mywritefunc hdl (prio, msg) loggername = withMVar lock (\_ -> do hPutStrLn hdl ("[" ++ loggername ++ "/" ++ show prio ++ "] " ++ msg) hFlush hdl ) return (GenericHandler {priority = pri, privData = h, writeFunc = mywritefunc, closeFunc = \x -> return ()})