summaryrefslogtreecommitdiff
path: root/Presence/main.hs
blob: 63ebf8171a67d31e64b1cdc52f3bc20345e54bed (plain)
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
{-# LANGUAGE CPP               #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TypeFamilies      #-}

import System.Directory
import Control.Monad
import System.Posix.Signals
import System.Posix.Types
import System.Posix.Process
import Data.Maybe

import System.INotify
#ifndef NOUTMP
import UTmp
#endif
import FGConsole
import XMPPServer
import Data.HList
import Network.Socket (sClose)
import Control.Exception

jid user host rsrc = user ++ "@" ++ host ++ "/" ++ rsrc

toJabberId :: String -> (String,String,t) -> Maybe String
toJabberId host (user,tty,_) = 
    if take 3 tty == "tty"
      then Just (jid user host tty)
      else Nothing

#ifndef NOUTMP
utmp_event e = do
    -- print e
    putStrLn "---"
    us <- users
    let ids = mapMaybe (toJabberId "localhost") us
        ids :: [String]
    forM_ ids putStrLn
#endif

data UnixSession = UnixSession

instance XMPPSession UnixSession where
    data XMPPClass UnixSession = UnixSessions
    newSession   _ sock handle = return UnixSession
    setResource  _ resource    = return ()
    getJID       _             = return "nobody@fake.bad"
    closeSession _             = return ()

on_chvt vtnum = do
    putStrLn $ "changed vt to "++ show vtnum

start :: IO ()
start = do
#ifndef NOUTMP
    installHandler sigUSR1 (Catch (utmp_event (userError "signaled"))) Nothing
#endif
    -- installHandler sigTERM (CatchOnce (utmp_event (userError "term signaled"))) Nothing
    inotify <- initINotify
    print inotify
#ifndef NOUTMP
    wd <- addWatch
            inotify
            [CloseWrite] -- [Open,Close,Access,Modify,Move]
            utmp_file
            utmp_event
    print wd
#endif
    mtty <- monitorTTY on_chvt
    sock <- listenForXmppClients UnixSessions 5222 HNil
    putStrLn "Hit enter to terminate..."
    getLine
    {-
    let doException (SomeException e) = Prelude.putStrLn ("\n\nexception: " ++ show e ++ "\n\n")
    handle doException $ do
    -}
    sClose sock
    -- threadDelay 1000
    putStrLn "closed listener."
    unmonitorTTY mtty
    putStrLn "unhooked tty monitor."
#ifndef NOUTMP
    removeWatch wd
#endif
    putStrLn "Normal termination."

sendUSR1 pid = do
    signalProcess sigUSR1 pid

getStartupAction []     = throw (userError "pid file?") >> return (Right "")
getStartupAction (p:ps) = do
    handle onEr $
        ( do
            pid <- fmap CPid (readFile p >>= readIO)
            -- signal pid
            return (Left pid) )
 where
    onEr (SomeException _) = do
        pid <- getProcessID
        putStrLn $ "starting pid = "++show pid
        handle (\(SomeException _) -> getStartupAction ps)
            (do 
                writeFile p (show pid)
                putStrLn $ "writing "++show p
                -- start daemon
                return (Right p) )

runOnce ps run notify = getStartupAction ps >>= doit
 where
    doit (Left pid      ) = notify pid
    doit (Right pidfile ) = do 
        run
        removeFile pidfile


main = do
    runOnce ["/var/run/presence.pid","/tmp/presence.pid"] start sendUSR1