diff --git a/Assistant/Threads/XMPPClient.hs b/Assistant/Threads/XMPPClient.hs
index 4f41fdb30b..0836fa0f65 100644
--- a/Assistant/Threads/XMPPClient.hs
+++ b/Assistant/Threads/XMPPClient.hs
@@ -76,7 +76,7 @@ xmppClientThread urlrenderer = NamedThread "XMPPClient" $ do
- the client connects, so that stale info
- is not retained. -}
void $ liftIO ioemptybuddies
- putStanza $ gitAnnexPresence gitAnnexSignature
+ putStanza gitAnnexSignature
xmppThread $ receivenotifications selfjid
forever $ do
a <- liftIO $ iorelay selfjid
@@ -90,13 +90,11 @@ xmppClientThread urlrenderer = NamedThread "XMPPClient" $ do
handle (PresenceMessage p) =
void $ liftIO $ ioupdatebuddies p
handle (GotNetMessage QueryPresence) =
- putStanza $ gitAnnexPresence gitAnnexSignature
+ putStanza gitAnnexSignature
handle (GotNetMessage (NotifyPush us)) =
void $ liftIO $ iopull us
handle (GotNetMessage (PairingNotification stage t u)) =
maybe noop (handlePairing stage u) (parseJID t)
- handle (GotNetMessage (SelfPairingNotification stage t u)) =
- error "TODO"
handle (Ignorable _) = noop
handle (Unknown _) = noop
handle (ProtocolError _) = noop
@@ -119,22 +117,24 @@ decodeStanza selfjid s@(ReceivedPresence p)
| presenceType p == PresenceError = [ProtocolError s]
| presenceFrom p == Nothing = [Ignorable p]
| presenceFrom p == Just selfjid = [Ignorable p]
- | not (null pushed) = impliedp $ GotNetMessage $ NotifyPush pushed
- | isPresenceQuery p = impliedp $ GotNetMessage QueryPresence
- | otherwise = case decodePairingNotification p of
- Nothing -> [PresenceMessage p]
- Just pn -> impliedp $ GotNetMessage pn
+ | otherwise = maybe [PresenceMessage p] decode (getGitAnnexAttrValue p)
where
- -- Things sent via presence imply a presence message,
- -- along with their real meaning.
+ decode (attr, v)
+ | attr == pushAttr = impliedp $ GotNetMessage $ NotifyPush $
+ decodePushNotification v
+ | attr == queryAttr = impliedp $ GotNetMessage QueryPresence
+ | otherwise = [Unknown s]
+ {- Things sent via presence imply a presence message,
+ - along with their real meaning. -}
impliedp v = [PresenceMessage p, v]
- pushed = concat $ catMaybes $ map decodePushNotification $
- presencePayloads p
-decodeStanza _ s@(ReceivedIQ iq)
- | iqType iq == IQError = [ProtocolError s]
- | otherwise = case decodeSelfPairingNotification iq of
- Nothing -> [Unknown s]
- Just pn -> [GotNetMessage pn]
+decodeStanza _ s@(ReceivedMessage m)
+ | messageType m == MessageError = [ProtocolError s]
+ | otherwise = maybe [Unknown s] decode (getGitAnnexAttrValue m)
+ where
+ decode (attr, v)
+ | attr == pairAttr =
+ [maybe (Unknown s) GotNetMessage (decodePairingNotification v m)]
+ | otherwise = [Unknown s]
decodeStanza _ s = [Unknown s]
{- Waits for a NetMessager message to be sent, and relays it to XMPP. -}
@@ -145,12 +145,10 @@ relayNetMessage selfjid = convert <$> waitNetMessage
convert QueryPresence = putStanza $ presenceQuery
convert (PairingNotification stage t u) = case parseJID t of
Nothing -> noop
- Just tojid -> mapM_ putStanza $
- encodePairingNotification stage u tojid selfjid
- convert (SelfPairingNotification stage t u) = case parseJID t of
- Nothing -> noop
- Just tojid -> putStanza $
- encodeSelfPairingNotification stage u tojid selfjid
+ Just tojid
+ | tojid == selfjid -> noop
+ | otherwise -> putStanza $
+ pairingNotification stage u tojid selfjid
{- Runs the client, handing restart events. -}
restartableClient :: IO () -> Assistant ()
diff --git a/Assistant/Types/NetMessager.hs b/Assistant/Types/NetMessager.hs
index 3df1782f88..f84247d6cf 100644
--- a/Assistant/Types/NetMessager.hs
+++ b/Assistant/Types/NetMessager.hs
@@ -23,9 +23,6 @@ data NetMessage
-- notification about a stage in the pairing process,
-- involving a client identified by the Text, and a UUID.
| PairingNotification PairStage Text UUID
- -- notification about a stage in the pairing process with
- -- other clients using the same account.
- | SelfPairingNotification PairStage Text UUID
deriving (Show)
data NetMessagerControl = NetMessagerControl
diff --git a/Assistant/WebApp/Configurators/Pairing.hs b/Assistant/WebApp/Configurators/Pairing.hs
index 4257f4151b..98781fc77b 100644
--- a/Assistant/WebApp/Configurators/Pairing.hs
+++ b/Assistant/WebApp/Configurators/Pairing.hs
@@ -81,26 +81,24 @@ getStartPairR = noPairing "local or jabber"
getStartXMPPPairR :: BuddyKey -> Handler RepHtml
#ifdef WITH_XMPP
getStartXMPPPairR bid = do
- creds <- runAnnex Nothing getXMPPCreds
- let ourjid = fromJust $ parseJID =<< xmppJID <$> creds
buddy <- liftAssistant $ getBuddy bid <<~ buddyList
- case S.toList . buddyAssistants <$> buddy of
- -- A buddy could have logged out, or the XMPP client restarted;
- -- so handle unforseen by going back.
- Nothing -> redirect StartPairR
- (Just []) -> redirect StartPairR
- (Just clients@((Client exemplar):_)) -> do
- let samejid = baseJID ourjid == baseJID exemplar
- let account = formatJID $ baseJID exemplar
- liftAssistant $ do
- u <- liftAnnex getUUID
- if samejid
- then forM_ clients $ \(Client c) ->
- sendNetMessage $ SelfPairingNotification PairReq (formatJID c) u
- else sendNetMessage $ PairingNotification PairReq account u
- pairPage $ do
- let name = buddyName exemplar
- $(widgetFile "configurators/pairing/xmpp/inprogress")
+ go $ S.toList . buddyAssistants <$> buddy
+ where
+ go (Just (clients@((Client exemplar):_))) = do
+ creds <- runAnnex Nothing getXMPPCreds
+ let ourjid = fromJust $ parseJID =<< xmppJID <$> creds
+ let samejid = baseJID ourjid == baseJID exemplar
+ let account = formatJID $ baseJID exemplar
+ liftAssistant $ do
+ u <- liftAnnex getUUID
+ forM_ clients $ \(Client c) -> sendNetMessage $
+ PairingNotification PairReq (formatJID c) u
+ pairPage $ do
+ let name = buddyName exemplar
+ $(widgetFile "configurators/pairing/xmpp/inprogress")
+ -- A buddy could have logged out, or the XMPP client restarted,
+ -- and there be no clients to message; handle unforseen by going back.
+ go _ = redirect StartPairR
#else
getStartXMPPPairR _ = noXMPPPairing
diff --git a/Assistant/XMPP.hs b/Assistant/XMPP.hs
index e3013a92f1..37cf003748 100644
--- a/Assistant/XMPP.hs
+++ b/Assistant/XMPP.hs
@@ -12,157 +12,133 @@ import Assistant.Types.NetMessager
import Assistant.Pairing
import Network.Protocol.XMPP
+import Data.Text (Text)
import qualified Data.Text as T
import Data.XML.Types
-{- A presence with a git-annex tag in it. -}
-gitAnnexPresence :: Element -> Presence
-gitAnnexPresence tag = (emptyPresence PresenceAvailable)
- { presencePayloads = [extendedAway, tag] }
- where
- extendedAway = Element (Name (T.pack "show") Nothing Nothing) []
- [NodeContent $ ContentText $ T.pack "xa"]
-
-{- Does a presence contain a git-annex tag? -}
-isGitAnnexPresence :: Presence -> Bool
-isGitAnnexPresence p = any isGitAnnexTag (presencePayloads p)
-
-{- Name of a git-annex tag, in our own XML namespace.
+{- Name of the git-annex tag, in our own XML namespace.
- (Not using a namespace URL to avoid unnecessary bloat.) -}
gitAnnexTagName :: Name
gitAnnexTagName = Name (T.pack "git-annex") (Just $ T.pack "git-annex") Nothing
+{- Creates a git-annex tag containing a particular attribute and value. -}
+gitAnnexTag :: Name -> Text -> Element
+gitAnnexTag attr val = Element gitAnnexTagName [(attr, [ContentText val])] []
+
isGitAnnexTag :: Element -> Bool
isGitAnnexTag t = elementName t == gitAnnexTagName
-{- A git-annex tag, to let other clients know we're a git-annex client too. -}
-gitAnnexSignature :: Element
-gitAnnexSignature = Element gitAnnexTagName [] []
+{- Things that a git-annex tag can inserted into. -}
+class GitAnnexTaggable a where
+ insertGitAnnexTag :: a -> Element -> a
-queryAttr :: Name
-queryAttr = Name (T.pack "query") Nothing Nothing
+ extractGitAnnexTag :: a -> Maybe Element
-pushAttr :: Name
-pushAttr = Name (T.pack "push") Nothing Nothing
+ hasGitAnnexTag :: a -> Bool
+ hasGitAnnexTag = isJust . extractGitAnnexTag
-pairAttr :: Name
-pairAttr = Name (T.pack "pair") Nothing Nothing
+instance GitAnnexTaggable Message where
+ insertGitAnnexTag m e = m { messagePayloads = e : messagePayloads m }
+ extractGitAnnexTag = headMaybe . filter isGitAnnexTag . messagePayloads
-isAttr :: Name -> (Name, [Content]) -> Bool
-isAttr attr (k, _) = k == attr
+instance GitAnnexTaggable Presence where
+ -- always mark extended away
+ insertGitAnnexTag p e = p { presencePayloads = extendedAway : e : presencePayloads p }
+ extractGitAnnexTag = headMaybe . filter isGitAnnexTag . presencePayloads
-getAttr :: Element -> Name -> Maybe T.Text
-getAttr (Element _name attrs _nodes) name =
- T.concat . map unpack . snd <$> headMaybe (filter (isAttr name) attrs)
+{- Gets the attr and value from a git-annex tag. -}
+getGitAnnexAttrValue :: GitAnnexTaggable a => a -> Maybe (Name, Text)
+getGitAnnexAttrValue a = case extractGitAnnexTag a of
+ Just (Element _ [(attr, content)] []) -> Just $
+ (attr, T.concat $ map unpack content)
+ _ -> Nothing
where
unpack (ContentText t) = t
unpack (ContentEntity t) = t
+{- A presence with a git-annex tag in it. -}
+gitAnnexPresence :: Element -> Presence
+gitAnnexPresence = insertGitAnnexTag $ emptyPresence PresenceAvailable
+
+{- A presence with an empty git-annex tag in it, used for letting other
+ - clients know we're around and are a git-annex client. -}
+gitAnnexSignature :: Presence
+gitAnnexSignature = gitAnnexPresence $ Element gitAnnexTagName [] []
+
+{- A message with a git-annex tag in it. -}
+gitAnnexMessage :: Element -> Message
+gitAnnexMessage = insertGitAnnexTag silentMessage
+
+{- A notification that we've pushed to some repositories, listing their
+ - UUIDs. -}
+pushNotification :: [UUID] -> Presence
+pushNotification = gitAnnexPresence . gitAnnexTag pushAttr . encodePushNotification
+
+pushAttr :: Name
+pushAttr = Name (T.pack "push") Nothing Nothing
+
uuidSep :: T.Text
uuidSep = T.pack ","
-{- git-annex tag with one push attribute per UUID pushed to. -}
-encodePushNotification :: [UUID] -> Element
-encodePushNotification us = Element gitAnnexTagName
- [(pushAttr, [ContentText pushvalue])] []
- where
- pushvalue = T.intercalate uuidSep $
- map (T.pack . fromUUID) us
+encodePushNotification :: [UUID] -> Text
+encodePushNotification = T.intercalate uuidSep . map (T.pack . fromUUID)
-decodePushNotification :: Element -> Maybe [UUID]
-decodePushNotification (Element name attrs _nodes)
- | name == gitAnnexTagName && not (null us) = Just us
- | otherwise = Nothing
- where
- us = map (toUUID . T.unpack) $
- concatMap (T.splitOn uuidSep . T.concat . map fromContent . snd) $
- filter ispush attrs
- ispush (k, _) = k == pushAttr
- fromContent (ContentText t) = t
- fromContent (ContentEntity t) = t
-
-pushNotification :: [UUID] -> Presence
-pushNotification = gitAnnexPresence . encodePushNotification
+decodePushNotification :: Text -> [UUID]
+decodePushNotification = map (toUUID . T.unpack) . T.splitOn uuidSep
{- A request for other git-annex clients to send presence. -}
presenceQuery :: Presence
-presenceQuery = gitAnnexPresence $ Element gitAnnexTagName
- [ (queryAttr, [ContentText T.empty]) ]
- []
+presenceQuery = gitAnnexPresence $ gitAnnexTag queryAttr T.empty
-isPresenceQuery :: Presence -> Bool
-isPresenceQuery p = case filter isGitAnnexTag (presencePayloads p) of
- [] -> False
- ((Element _name attrs _nodes):_) -> any (isAttr queryAttr) attrs
+queryAttr :: Name
+queryAttr = Name (T.pack "query") Nothing Nothing
-{- A notification about a stage of pairing, sent as directed presence
- - to all clients of a jid.
- -
- - For PairReq, the directed presence is followed by a second presence
- - without the pair notification. This is done because XMPP servers
- - resend the last directed presence periodically, which can make
- - the pair request alert be re-displayed annoyingly. For PairAck and
- - PairDone, that resending is a desirable feature, as it helps ensure
- - clients see them.
- -}
-encodePairingNotification :: PairStage -> UUID -> JID -> JID -> [Presence]
-encodePairingNotification pairstage u tojid fromjid
- | pairstage == PairReq = [send, clear]
- | otherwise = [send]
- where
- send = directed $ gitAnnexPresence $ Element gitAnnexTagName
- [(pairAttr, [ContentText content])] []
- clear = directed $ gitAnnexPresence gitAnnexSignature
+{- A notification about a stage of pairing. -}
+pairingNotification :: PairStage -> UUID -> JID -> JID -> Message
+pairingNotification pairstage u tojid fromjid =
+ (gitAnnexMessage tag)
+ { messageTo = Just tojid
+ , messageFrom = Just fromjid
+ }
+ where
+ tag = gitAnnexTag pairAttr $
+ encodePairingNotification pairstage u
- directed p = p
- { presenceTo = Just $ baseJID tojid
- , presenceFrom = Just fromjid
- }
+pairAttr :: Name
+pairAttr = Name (T.pack "pair") Nothing Nothing
- content = mkPairingContent pairstage u
-
-{- A notification about a stage of pairing. Sent to self as an XMPP IQ.
- - Directed presence is not used for self-messaging presence because
- - some XMPP clients seem very confused by it. Google Talk has been
- - observed leaking self-directed presence to other friends, seeming
- - to think it sets the visible presence.
- -
- - The pairing info is sent using its id attribute; it also has a git-annex
- - tag to identify it as from us. -}
-encodeSelfPairingNotification :: PairStage -> UUID -> JID -> JID -> IQ
-encodeSelfPairingNotification pairstage u tojid fromjid = (emptyIQ IQGet)
- { iqTo = Just tojid
- , iqFrom = Just fromjid
- , iqID = Just $ mkPairingContent pairstage u
- , iqPayload = Just gitAnnexSignature
- }
-
-decodePairingNotification :: Presence -> Maybe NetMessage
-decodePairingNotification p = case filter isGitAnnexTag (presencePayloads p) of
- [] -> Nothing
- (elt:_) -> parsePairingContent (presenceFrom p) =<< getAttr elt pairAttr
-
-decodeSelfPairingNotification :: IQ -> Maybe NetMessage
-decodeSelfPairingNotification iq@(IQ { iqPayload = Just elt })
- | isGitAnnexTag elt = parsePairingContent (iqFrom iq) =<< iqID iq
- | otherwise = Nothing
-decodeSelfPairingNotification _ = Nothing
-
-mkPairingContent :: PairStage -> UUID -> T.Text
-mkPairingContent pairstage u = T.unwords $ map T.pack
+encodePairingNotification :: PairStage -> UUID -> Text
+encodePairingNotification pairstage u = T.unwords $ map T.pack
[ show pairstage
, fromUUID u
]
-parsePairingContent :: Maybe JID -> T.Text -> Maybe NetMessage
-parsePairingContent jid t = parse $ words $ T.unpack t
+decodePairingNotification :: Text -> Message -> Maybe NetMessage
+decodePairingNotification t msg = parse $ words $ T.unpack t
where
parse [stage, u] = PairingNotification
<$> readish stage
- <*> (formatJID <$> jid)
+ <*> (formatJID <$> messageFrom msg)
<*> pure (toUUID u)
parse _ = Nothing
{- The JID without the client part. -}
baseJID :: JID -> JID
baseJID j = JID (jidNode j) (jidDomain j) Nothing
+
+{- An XMPP chat message with an empty body. This should not be displayed
+ - by clients, but can be used for communications. -}
+silentMessage :: Message
+silentMessage = (emptyMessage MessageChat)
+ { messagePayloads = [ emptybody ] }
+ where
+ emptybody = Element
+ { elementName = Name (T.pack "body") Nothing Nothing
+ , elementAttributes = []
+ , elementNodes = []
+ }
+
+{- Add to a presence to mark its client as extended away. -}
+extendedAway :: Element
+extendedAway = Element (Name (T.pack "show") Nothing Nothing) []
+ [NodeContent $ ContentText $ T.pack "xa"]
diff --git a/Assistant/XMPP/Buddies.hs b/Assistant/XMPP/Buddies.hs
index 77e506ee94..fe5d8c6a9f 100644
--- a/Assistant/XMPP/Buddies.hs
+++ b/Assistant/XMPP/Buddies.hs
@@ -67,7 +67,7 @@ applyPresence p b = fromMaybe b $! go <$> presenceFrom p
, buddyPresent = removefrom $ buddyPresent b
, buddyAssistants = removefrom $ buddyAssistants b
}
- | isGitAnnexPresence p = b
+ | hasGitAnnexTag p = b
{ buddyAssistants = addto $ buddyAssistants b
, buddyAway = removefrom $ buddyAway b }
| presenceType p == PresenceAvailable = b
diff --git a/doc/design/assistant/xmpp.mdwn b/doc/design/assistant/xmpp.mdwn
index 570084dda9..284e56fa3b 100644
--- a/doc/design/assistant/xmpp.mdwn
+++ b/doc/design/assistant/xmpp.mdwn
@@ -29,26 +29,26 @@ who share a repository, that is stored in the [[cloud]].
## protocol
To avoid relying on XMPP extensions, git-annex communicates
-using presence messages. These always mark it as extended away.
-To this, it adds its own tag as [extended content](http://xmpp.org/rfcs/rfc6121.html#presence-extended).
+using presence messages (which always mark it as extended away),
+and chat messages (with empty body tags, so clients don't display them).
+
+To these messages, it adds its own tag as
+[extended content](http://xmpp.org/rfcs/rfc6121.html#presence-extended).
The xml namespace is "git-annex" (not an URL because I hate wasting bandwidth).
-To indicate it's pushed changes to a git repo with a given UUID, a client uses:
+To indicate it's pushed changes to a git repo with a given UUID,
+a message that should be sent to all buddies and other clients using the account
+(no explicit pairing needed), a client uses a broadcast presence message with:
Multiple UUIDs can be listed when multiple clients were pushed. If the
git repo does not have a git-annex UUID, an empty string is used.
-For pairing, a directed presence message is sent, also using the git-annex tag:
+For pairing, a chat message is sent, also using the git-annex tag:
-For pairing with other clients using the same XMPP account, git-annex uses
-IQ messages, also containing a git-annex tag. The id attribute of the iq
-tag contains the pairing information. This is done because self-directed
-presence is not handled correctly by Google Talk. (Or is ill-specified.)
-
### security
Data git-annex sends over XMPP will be visible to the XMPP