{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE ViewPatterns #-} {-# LANGUAGE TupleSections #-} {-# LANGUAGE TypeFamilies #-} -- {-# LANGUAGE GADTs #-} module XMPPServer where -- ( listenForXmppClients ) where import Data.HList.TypeEqGeneric1() import Data.HList.TypeCastGeneric1() import ByteStringOperators import Server import Data.ByteString.Lazy.Char8 as L ( hPutStrLn , unlines , ByteString , pack , unpack ) import qualified Data.ByteString.Lazy.Char8 as L ( putStrLn ) import System.IO ( Handle ) import Data.HList import AdaptServer import Text.XML.HaXml.Lex (xmlLex) import Text.XML.HaXml.Parse (XParser,xmlParseWith,element,{-doctypedecl,-}processinginstruction,elemOpenTag,elemCloseTag) import Text.XML.HaXml.Types as Hax hiding (Element) -- (ElemTag,QName(..),Namespace(..),Element(..),Content(..),AttValue(..)) import qualified Text.XML.HaXml.Types as Hax (Element(..)) import Text.XML.HaXml.Posn (Posn) import Text.XML.HaXml.Lex (TokenT) import qualified Text.XML.HaXml.Pretty as PP import Text.PrettyPrint import Data.Maybe import Debug.Trace import Control.Arrow import LocalPeerCred import Network.Socket import Data.String import Control.Monad.Trans.Maybe import Control.Monad.IO.Class import Control.DeepSeq import Control.Concurrent.STM import Control.Concurrent import Control.Exception -- | Jabber ID (JID) datatype data JID = JID { name :: Maybe ByteString , server :: ByteString , resource :: Maybe ByteString } deriving (Ord,Eq) instance Show JID where show (JID n s r ) = L.unpack $ fmap (<++>"@") n s <++?> fmap ("/"<++>) r instance NFData JID where rnf v@(JID n s r) = n `seq` s `seq` r `seq` () jid user host rsrc = JID (Just user) host (Just rsrc) data JabberShow = Offline | Away | Available deriving (Show,Enum,Ord,Eq,Read) data Presence = Presence JID JabberShow deriving Show xmlifyPresence (Presence jid stat) = L.unlines [ " typ stat <++> ">" , "" <++> shw stat <++> "" , "" ] where typ Offline = " type='unavailable'" typ _ = "" shw Available = "chat" shw Away = "away" shw Offline = "away" -- Is this right? instance NFData Presence where rnf (Presence jid stat) = rnf jid `seq` stat `seq` () class XMPPSession session where data XMPPClass session newSession :: XMPPClass session -> Socket -> Handle -> IO session setResource :: session -> ByteString -> IO () getJID :: session -> IO ByteString closeSession :: session -> IO () subscribe :: session -> Maybe JID -> IO (TChan Presence) greet host = L.unlines [ "" , "" , "" , " " {- -- , " " , " " -- , " DIGEST-MD5" , " PLAIN" , " " -} , "" ] -- data TaggedXMPPSession s = TaggedXMPPSession s data Commands = Send ByteString deriving Show startCon session_factory sock st = do let h = hOccursFst st :: Handle cred <- getLocalPeerCred sock Prelude.putStrLn $ "PEER CRED: "++show cred pname <- getPeerName sock session <- newSession session_factory sock h Prelude.putStrLn $ "PEER NAME: "++show pname pchan <- subscribe session Nothing cmdChan <- atomically newTChan reader <- forkIO $ handle (\(SomeException _) -> L.putStrLn "quit reader via exception.") $ fix $ \loop -> do event <- atomically $ (fmap Left $ readTChan pchan) `orElse` (fmap Right $ readTChan cmdChan) case event of Left presence -> do L.putStrLn $ "PRESENCE: " <++> bshow presence -- TODO: it violates spec to send presence information before -- a resource is bound. let r = xmlifyPresence presence hPutStrLn h r L.putStrLn $ "\nOUT:\n" <++> r Right (Send r) -> hPutStrLn h r loop let quit = do killThread reader closeSession session return ( (session,cmdChan) .*. ConnectionFinalizer quit .*. st) iq_query_unavailable host id mjid xmlns kind = L.unlines $ [ " " to='" <++> jid <++> "'" Nothing -> "" , " id='" <++> id <++> "'>" , " <" <++> kind <++> " xmlns='" <++> xmlns <++> "'/>" , " " , " " , " " , "" ] tagattrs tag content = Prelude.concatMap (\(CElem (Elem _ a _) _)->a) $ Prelude.filter (bindElem tag) content anytagattrs content = Prelude.concatMap (\(CElem (Elem n a _) _)->map (second (n,)) a) content bindElem tag (CElem (Elem (N n) _ _) _) | n==tag = True bindElem _ _ = False hasElem tag content = not . Prelude.null . Prelude.filter (bindElem tag) $ content unattr (AttValue as) = listToMaybe $ Prelude.concatMap left as where left (Left x) = [x] left _ = [] astring (AttValue [Left s]) = [s] tagcontent tag content = Prelude.concatMap (\(CElem (Elem _ _ c) _)->c) $ Prelude.filter (bindElem tag) content iq_bind_reply id jid = L.unlines $ [ "" , "" , "" <++> jid <++> "" , "" , " " ] iq_session_reply host id = L.unlines $ [ " " ] iqresult_info host id mjid = L.unlines $ [ " " to='" <++> jid <++> "'" Nothing -> "" , " id='" <++> id <++> "'>" , " " , " " , " " , " " , " " , "" ] iqresponse session host (Elem _ attrs content) = runMaybeT $ do id <- MaybeT . return $ fmap pack (lookup (N "id") attrs >>= unattr) typ <- MaybeT . return $ fmap pack (lookup (N "type") attrs >>= unattr) case typ of "set" -> do let string (CString _ s _) = [s] mplus (do rsrc <- MaybeT . return . fmap pack $ listToMaybe . Prelude.concatMap string . tagcontent "resource" . tagcontent "bind" $ content -- let jid = "TODO" <++> "@" <++> host <++> "/" <++> pack rsrc liftIO $ do setResource session rsrc jid <- getJID session return $ iq_bind_reply id jid ) (do guard (hasElem "session" content) return (iq_session_reply host id)) "get" -> {- trace ("iq-get "++show (attrs,content)) $ -} do (tag,as) <- MaybeT . return $ lookup (N "xmlns") (anytagattrs content) xmlns <- MaybeT . return $ fmap pack $ listToMaybe . astring $ as let servicekind = case tag of { (N s) -> pack s ; _ -> "query" } case xmlns of "urn:xmpp:ping" -> do let to = case fmap pack (lookup (N "from") attrs >>= unattr) of Just jid -> "to='" <++> jid <++> "' " Nothing -> "" return $ " to <++> "id='" <++> id <++> "' type='result'/>" _ -> return (iq_query_unavailable host id Nothing xmlns servicekind) _ -> MaybeT (return Nothing) presence_response host (Elem _ attrs content) = do -- let id = fmap pack (lookup (N "id") attrs >>= unattr) typ <- fmap pack (lookup (N "type") attrs >>= unattr) case typ of "subscribe" -> do -- to <- fmap pack (lookup (N "to") attrs >>= unattr) Just $ "" _ -> Nothing doCon st elem cont = do let h = hOccursFst st :: Handle host = "localhost" (session,cmdChan) = hHead st hsend r = do atomically $ writeTChan cmdChan (Send r) -- hPutStrLn h r L.putStrLn $ "\nOUT:\n" <++> r putStrLn $ (show $ hang (text "\nIN:") 2 $ pp elem) ++ "\n" case elem of OpenTag _ -> hsend (greet host) Element e@(Elem (N "iq") _ _) -> do rpns <- iqresponse session host e case rpns of Nothing -> trace "IGNORE: no response to " $ return () Just r -> hsend r Element e@(Elem (N "presence") _ _) -> case presence_response host e of Nothing -> trace "IGNORE: no response to " $ return () Just r -> hsend r _ -> return () -- putStrLn $ "unhandled: "++show v cont () instance Show Hax.ElemTag where show _ = "elemtag" data XmppObject = Element (Hax.Element Posn) | ProcessingInstruction Hax.ProcessingInstruction | OpenTag ElemTag | CloseTag () deriving Show pp (Element e) = PP.element e pp o = fromString (show o) streamName = QN (Namespace {nsPrefix= "stream" , nsURI="http://etherx.jabber.org/streams" }) "stream" newtype TryParse p x e = Try (Either e p,[x]) instance Monad (TryParse p x) where return v = Try (Left v,[]) Try (Left m,xs) >>= k = k m Try (Right e,xs) >>= _ = Try (Right e,xs) runTryParse (Try p) = p mapRight f (Right x,ls) = (Right (f x),ls) mapRight f (Left y,ls) = (Left y ,ls) xmppParse :: [(Posn, TokenT)] -> (Either String XmppObject, [(Posn, TokenT)]) xmppParse ls = runTryParse $ do let xml :: (t -> b) -> XParser t -> (Either String b, [(Posn, TokenT)]) xml tag = mapRight tag . flip xmlParseWith ls Try . xml Element $ element Try . xml OpenTag $ elemOpenTag Try . xml CloseTag $ elemCloseTag streamName Try . xml ProcessingInstruction $ processinginstruction listenForXmppClients session_factory port st = do -- standard port: 5222 let (start,dopkt) = adaptServer ( xmlLex "local-client" . unpack , xmppParse) (startCon session_factory,doCon) doServer (port .*. st) dopkt start startPeer session_factory sock st = do let h = hOccursFst st :: Handle name <- fmap bshow $ getPeerName sock L.putStrLn $ "REMOTE: connected " <++> name let quit = L.putStrLn $ "REMOTE: disconnected " <++> name return ( ConnectionFinalizer quit .*. st ) doPeer st elem cont = do cont () listenForRemotePeers session_factory port st = do -- standard port: 5269 let (start,dopkt) = adaptServer ( xmlLex "remote-peer" . unpack , xmppParse) (startPeer session_factory,doPeer) doServer (port .*. st) dopkt start