S2S: sharerCreateNoteF & projectCreateNoteF can handle sharer-hosted tickets

This commit is contained in:
fr33domlover 2020-05-11 18:59:29 +00:00
parent edaa3c49b2
commit c91908941b
3 changed files with 242 additions and 191 deletions

View file

@ -149,11 +149,6 @@ parseComment luParent = do
MessageR shr hid -> (shr,) <$> decodeKeyHashidE hid "Non-existent local message hashid" MessageR shr hid -> (shr,) <$> decodeKeyHashidE hid "Non-existent local message hashid"
_ -> throwE "Not a local message route" _ -> throwE "Not a local message route"
data NoteContext
= NoteContextSharerTicket ShrIdent TicketAuthorLocalId
| NoteContextProjectTicket ShrIdent PrjIdent LocalTicketId
deriving Eq
noteC noteC
:: Entity Person :: Entity Person
-> Sharer -> Sharer

View file

@ -14,7 +14,8 @@
-} -}
module Vervis.ActivityPub module Vervis.ActivityPub
( hostIsLocal ( NoteContext (..)
, hostIsLocal
, verifyHostLocal , verifyHostLocal
, parseContext , parseContext
, parseParent , parseParent
@ -43,6 +44,7 @@ module Vervis.ActivityPub
, deliverLocal , deliverLocal
, RemoteRecipient (..) , RemoteRecipient (..)
, deliverLocal' , deliverLocal'
, insertRemoteActivityToLocalInboxes
) )
where where
@ -119,6 +121,11 @@ import Vervis.Time
import Vervis.Widget.Repo import Vervis.Widget.Repo
import Vervis.Widget.Sharer import Vervis.Widget.Sharer
data NoteContext
= NoteContextSharerTicket ShrIdent TicketAuthorLocalId
| NoteContextProjectTicket ShrIdent PrjIdent LocalTicketId
deriving Eq
hostIsLocal :: (MonadSite m, SiteEnv m ~ App) => Host -> m Bool hostIsLocal :: (MonadSite m, SiteEnv m ~ App) => Host -> m Bool
hostIsLocal h = asksSite $ (== h) . appInstanceHost . appSettings hostIsLocal h = asksSite $ (== h) . appInstanceHost . appSettings
@ -132,7 +139,7 @@ verifyHostLocal h t = do
parseContext parseContext
:: (MonadSite m, SiteEnv m ~ App) :: (MonadSite m, SiteEnv m ~ App)
=> FedURI => FedURI
-> ExceptT Text m (Either (ShrIdent, PrjIdent, KeyHashid LocalTicket) FedURI) -> ExceptT Text m (Either NoteContext FedURI)
parseContext uContext = do parseContext uContext = do
let ObjURI hContext luContext = uContext let ObjURI hContext luContext = uContext
local <- hostIsLocal hContext local <- hostIsLocal hContext
@ -142,7 +149,12 @@ parseContext uContext = do
Nothing -> throwE "Local context isn't a valid route" Nothing -> throwE "Local context isn't a valid route"
Just r -> return r Just r -> return r
case route of case route of
ProjectTicketR shr prj num -> return (shr, prj, num) SharerTicketR shr talkhid ->
NoteContextSharerTicket shr <$>
decodeKeyHashidE talkhid "Note context invalid talkhid"
ProjectTicketR shr prj ltkhid ->
NoteContextProjectTicket shr prj <$>
decodeKeyHashidE ltkhid "Note context invalid ltkhid"
_ -> throwE "Local context isn't a ticket route" _ -> throwE "Local context isn't a ticket route"
else return $ Right uContext else return $ Right uContext
@ -735,15 +747,25 @@ data RemoteRecipient = RemoteRecipient
-- * Insert activity to inboxes of actors -- * Insert activity to inboxes of actors
-- * If collections are listed, insert activity to the local members and return -- * If collections are listed, insert activity to the local members and return
-- the remote members -- the remote members
deliverLocal' insertActivityToLocalInboxes
:: Bool -- ^ Whether to deliver to collection only if owner actor is addressed :: PersistRecordBackend record SqlBackend
-> LocalActor => (InboxId -> InboxItemId -> record)
-> InboxId -- ^ Database record to insert as an new inbox item to each inbox
-> OutboxItemId -> Bool
-- ^ Whether to deliver to collection only if owner actor is addressed
-> Maybe LocalActor
-- ^ An actor whose collections are excluded from requiring an owner, i.e.
-- even if owner is required, this actor's collections will be delivered
-- to, even if this actor isn't addressed. This is meant to be the
-- activity's author.
-> Maybe InboxId
-- ^ A user person's inbox to exclude from delivery, even if this person is
-- listed in the recipient set. This is meant to be the activity's
-- author.
-> LocalRecipientSet -> LocalRecipientSet
-> AppDB [((InstanceId, Host), NonEmpty RemoteRecipient)] -> AppDB [((InstanceId, Host), NonEmpty RemoteRecipient)]
deliverLocal' requireOwner author ibidAuthor obiid recips = do insertActivityToLocalInboxes makeInboxItem requireOwner mauthor mibidAuthor recips = do
ibidsSharer <- L.delete ibidAuthor <$> getSharerInboxes recips ibidsSharer <- deleteAuthor <$> getSharerInboxes recips
ibidsOther <- concat <$> traverse getOtherInboxes recips ibidsOther <- concat <$> traverse getOtherInboxes recips
(ibidsFollowers, remotesFollowers) <- do (ibidsFollowers, remotesFollowers) <- do
@ -754,13 +776,23 @@ deliverLocal' requireOwner author ibidAuthor obiid recips = do
ibidsTeams <- foldl' LO.union [] <$> traverse getTeams recips ibidsTeams <- foldl' LO.union [] <$> traverse getTeams recips
let ibids = L.delete ibidAuthor (ibidsFollowers `LO.union` ibidsTeams `LO.union` ibidsSharer) ++ ibidsOther let ibids = deleteAuthor (ibidsFollowers `LO.union` ibidsTeams `LO.union` ibidsSharer) ++ ibidsOther
ibiids <- insertMany $ replicate (length ibids) $ InboxItem True ibiids <- insertMany $ replicate (length ibids) $ InboxItem True
insertMany_ $ insertMany_ $ zipWith makeInboxItem ibids ibiids
map (\ (ibid, ibiid) -> InboxItemLocal ibid obiid ibiid)
(zip ibids ibiids)
return remotesFollowers return remotesFollowers
where where
isAuthor :: LocalActor -> Bool
isAuthor =
case mauthor of
Nothing -> const False
Just author -> (== author)
deleteAuthor :: [InboxId] -> [InboxId]
deleteAuthor =
case mibidAuthor of
Nothing -> id
Just ibidAuthor -> L.delete ibidAuthor
getSharerInboxes :: LocalRecipientSet -> AppDB [InboxId] getSharerInboxes :: LocalRecipientSet -> AppDB [InboxId]
getSharerInboxes sharers = do getSharerInboxes sharers = do
let shrs = let shrs =
@ -801,7 +833,7 @@ deliverLocal' requireOwner author ibidAuthor obiid recips = do
[shr | (shr, s) <- sharers [shr | (shr, s) <- sharers
, let d = localRecipSharerDirect s , let d = localRecipSharerDirect s
in localRecipSharerFollowers d && in localRecipSharerFollowers d &&
(localRecipSharer d || not requireOwner || LocalActorSharer shr == author) (localRecipSharer d || not requireOwner || isAuthor (LocalActorSharer shr))
] ]
sids <- selectKeysList [SharerIdent <-. shrs] [] sids <- selectKeysList [SharerIdent <-. shrs] []
map (personFollowers . entityVal) <$> selectList [PersonIdent <-. sids] [] map (personFollowers . entityVal) <$> selectList [PersonIdent <-. sids] []
@ -838,7 +870,7 @@ deliverLocal' requireOwner author ibidAuthor obiid recips = do
[prj | (prj, j) <- projects [prj | (prj, j) <- projects
, let d = localRecipProjectDirect j , let d = localRecipProjectDirect j
in localRecipProjectFollowers d && in localRecipProjectFollowers d &&
(localRecipProject d || not requireOwner || LocalActorProject shr prj == author) (localRecipProject d || not requireOwner || isAuthor (LocalActorProject shr prj))
] ]
fsidsJ <- fsidsJ <-
map (projectFollowers . entityVal) <$> map (projectFollowers . entityVal) <$>
@ -848,7 +880,7 @@ deliverLocal' requireOwner author ibidAuthor obiid recips = do
then then
[ (prj, localRecipProjectTicketRelated j) [ (prj, localRecipProjectTicketRelated j)
| (prj, j) <- projects | (prj, j) <- projects
, localRecipProject (localRecipProjectDirect j) || LocalActorProject shr prj == author , localRecipProject (localRecipProjectDirect j) || isAuthor (LocalActorProject shr prj)
] ]
else else
map (second localRecipProjectTicketRelated) projects map (second localRecipProjectTicketRelated) projects
@ -882,7 +914,7 @@ deliverLocal' requireOwner author ibidAuthor obiid recips = do
[rp | (rp, r) <- repos [rp | (rp, r) <- repos
, let d = localRecipRepoDirect r , let d = localRecipRepoDirect r
in localRecipRepoFollowers d && in localRecipRepoFollowers d &&
(localRecipRepo d || not requireOwner || LocalActorRepo shr rp == author) (localRecipRepo d || not requireOwner || isAuthor (LocalActorRepo shr rp))
] ]
in map (repoFollowers . entityVal) <$> in map (repoFollowers . entityVal) <$>
selectList [RepoSharer ==. sid, RepoIdent <-. rps] [] selectList [RepoSharer ==. sid, RepoIdent <-. rps] []
@ -935,7 +967,7 @@ deliverLocal' requireOwner author ibidAuthor obiid recips = do
getProjectTeams sid projects = do getProjectTeams sid projects = do
let prjs = let prjs =
[prj | (prj, LocalProjectRelatedSet d ts) <- projects [prj | (prj, LocalProjectRelatedSet d ts) <- projects
, (localRecipProject d || not requireOwner || LocalActorProject shr prj == author) && , (localRecipProject d || not requireOwner || isAuthor (LocalActorProject shr prj)) &&
(localRecipProjectTeam d || any (localRecipTicketTeam . snd) ts) (localRecipProjectTeam d || any (localRecipTicketTeam . snd) ts)
] ]
jids <- selectKeysList [ProjectSharer ==. sid, ProjectIdent <-. prjs] [] jids <- selectKeysList [ProjectSharer ==. sid, ProjectIdent <-. prjs] []
@ -946,8 +978,36 @@ deliverLocal' requireOwner author ibidAuthor obiid recips = do
[rp | (rp, r) <- repos [rp | (rp, r) <- repos
, let d = localRecipRepoDirect r , let d = localRecipRepoDirect r
in localRecipRepoTeam d && in localRecipRepoTeam d &&
(localRecipRepo d || not requireOwner || LocalActorRepo shr rp == author) (localRecipRepo d || not requireOwner || isAuthor (LocalActorRepo shr rp))
] ]
rids <- selectKeysList [RepoSharer ==. sid, RepoIdent <-. rps] [] rids <- selectKeysList [RepoSharer ==. sid, RepoIdent <-. rps] []
pids <- map (repoCollabPerson . entityVal) <$> selectList [RepoCollabRepo <-. rids] [] pids <- map (repoCollabPerson . entityVal) <$> selectList [RepoCollabRepo <-. rids] []
map (personInbox . entityVal) <$> selectList [PersonId <-. pids] [Asc PersonInbox] map (personInbox . entityVal) <$> selectList [PersonId <-. pids] [Asc PersonInbox]
-- | Given a list of local recipients, which may include actors and
-- collections,
--
-- * Insert activity to inboxes of actors
-- * If collections are listed, insert activity to the local members and return
-- the remote members
deliverLocal'
:: Bool -- ^ Whether to deliver to collection only if owner actor is addressed
-> LocalActor
-> InboxId
-> OutboxItemId
-> LocalRecipientSet
-> AppDB [((InstanceId, Host), NonEmpty RemoteRecipient)]
deliverLocal' requireOwner author ibidAuthor obiid =
insertActivityToLocalInboxes makeItem requireOwner (Just author) (Just ibidAuthor)
where
makeItem ibid ibiid = InboxItemLocal ibid obiid ibiid
insertRemoteActivityToLocalInboxes
:: Bool
-> RemoteActivityId
-> LocalRecipientSet
-> AppDB [((InstanceId, Host), NonEmpty RemoteRecipient)]
insertRemoteActivityToLocalInboxes requireOwner ractid =
insertActivityToLocalInboxes makeItem requireOwner Nothing Nothing
where
makeItem ibid ibiid = InboxItemRemote ibid ractid ibiid

View file

@ -27,6 +27,7 @@ import Control.Monad.Trans.Except
import Control.Monad.Trans.Maybe import Control.Monad.Trans.Maybe
import Data.Bifunctor import Data.Bifunctor
import Data.ByteString (ByteString) import Data.ByteString (ByteString)
import Data.Either
import Data.Foldable import Data.Foldable
import Data.Function import Data.Function
import Data.List (sort, deleteBy, nub, union, unionBy, partition) import Data.List (sort, deleteBy, nub, union, unionBy, partition)
@ -55,6 +56,7 @@ import Web.ActivityPub
import Yesod.ActivityPub import Yesod.ActivityPub
import Yesod.FedURI import Yesod.FedURI
import Yesod.Hashids import Yesod.Hashids
import Yesod.MonadSite
import Control.Monad.Trans.Except.Local import Control.Monad.Trans.Except.Local
import Data.Tuple.Local import Data.Tuple.Local
@ -69,6 +71,7 @@ import Vervis.Foundation
import Vervis.Model import Vervis.Model
import Vervis.Model.Ident import Vervis.Model.Ident
import Vervis.Settings import Vervis.Settings
import Vervis.Ticket
sharerCreateNoteF sharerCreateNoteF
:: UTCTime :: UTCTime
@ -101,56 +104,55 @@ sharerCreateNoteF now shrRecip author body (Note mluNote _ _ muParent muContext
Right _ -> Right _ ->
Right <$> insertToInbox luCreate (personInbox personRecip) Right <$> insertToInbox luCreate (personInbox personRecip)
where where
checkContextParent context mparent = runExceptT $ do checkContextParent (Left context) mparent = runExceptT $ do
case context of did <-
Left (shr, prj, ltkhid) -> do case context of
mdid <- lift $ runMaybeT $ do NoteContextSharerTicket shr talid -> do
sid <- MaybeT $ getKeyBy $ UniqueSharer shr (_, Entity _ lt, _, project) <- do
jid <- MaybeT $ getKeyBy $ UniqueProject prj sid mticket <- lift $ getSharerTicket shr talid
ltid <- decodeKeyHashidM ltkhid fromMaybeE mticket "Note context no such local sharer-hosted ticket"
lt <- MaybeT $ get ltid
tpl <-
MaybeT $ getValBy $
UniqueTicketProjectLocal $ localTicketTicket lt
guard $ ticketProjectLocalProject tpl == jid
return $ localTicketDiscuss lt return $ localTicketDiscuss lt
did <- fromMaybeE mdid "Context: No such local ticket" NoteContextProjectTicket shr prj ltid -> do
for_ mparent $ \ parent -> (_, _, _, Entity _ lt, _, _) <- do
case parent of mticket <- lift $ getProjectTicket shr prj ltid
Left (shrP, lmidP) -> fromMaybeE mticket "Note context no such local project-hosted ticket"
void $ getLocalParentMessageId did shrP lmidP return $ localTicketDiscuss lt
Right (ObjURI hParent luParent) -> do for_ mparent $ \ parent ->
mrm <- lift $ runMaybeT $ do case parent of
iid <- MaybeT $ getKeyBy $ UniqueInstance hParent Left (shrP, lmidP) ->
roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luParent void $ getLocalParentMessageId did shrP lmidP
MaybeT $ getValBy $ UniqueRemoteMessageIdent roid Right (ObjURI hParent luParent) -> do
for_ mrm $ \ rm -> do mrm <- lift $ runMaybeT $ do
let mid = remoteMessageRest rm iid <- MaybeT $ getKeyBy $ UniqueInstance hParent
m <- lift $ getJust mid roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luParent
unless (messageRoot m == did) $ MaybeT $ getValBy $ UniqueRemoteMessageIdent roid
throwE "Remote parent belongs to a different discussion" for_ mrm $ \ rm -> do
Right (ObjURI hContext luContext) -> do let mid = remoteMessageRest rm
mdid <- lift $ runMaybeT $ do m <- lift $ getJust mid
iid <- MaybeT $ getKeyBy $ UniqueInstance hContext unless (messageRoot m == did) $
roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luContext throwE "Remote parent belongs to a different discussion"
rd <- MaybeT $ getValBy $ UniqueRemoteDiscussionIdent roid checkContextParent (Right (ObjURI hContext luContext)) mparent = runExceptT $ do
return $ remoteDiscussionDiscuss rd mdid <- lift $ runMaybeT $ do
for_ mparent $ \ parent -> iid <- MaybeT $ getKeyBy $ UniqueInstance hContext
case parent of roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luContext
Left (shrP, lmidP) -> do rd <- MaybeT $ getValBy $ UniqueRemoteDiscussionIdent roid
did <- fromMaybeE mdid "Local parent inexistent, no RemoteDiscussion" return $ remoteDiscussionDiscuss rd
void $ getLocalParentMessageId did shrP lmidP for_ mparent $ \ parent ->
Right (ObjURI hParent luParent) -> do case parent of
mrm <- lift $ runMaybeT $ do Left (shrP, lmidP) -> do
iid <- MaybeT $ getKeyBy $ UniqueInstance hParent did <- fromMaybeE mdid "Local parent inexistent, no RemoteDiscussion"
roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luParent void $ getLocalParentMessageId did shrP lmidP
MaybeT $ getValBy $ UniqueRemoteMessageIdent roid Right (ObjURI hParent luParent) -> do
for_ mrm $ \ rm -> do mrm <- lift $ runMaybeT $ do
let mid = remoteMessageRest rm iid <- MaybeT $ getKeyBy $ UniqueInstance hParent
m <- lift $ getJust mid roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luParent
did <- fromMaybeE mdid "Remote parent known, but no context RemoteDiscussion" MaybeT $ getValBy $ UniqueRemoteMessageIdent roid
unless (messageRoot m == did) $ for_ mrm $ \ rm -> do
throwE "Remote parent belongs to a different discussion" let mid = remoteMessageRest rm
m <- lift $ getJust mid
did <- fromMaybeE mdid "Remote parent known, but no context RemoteDiscussion"
unless (messageRoot m == did) $
throwE "Remote parent belongs to a different discussion"
insertToInbox luCreate ibidRecip = do insertToInbox luCreate ibidRecip = do
let iidAuthor = remoteAuthorInstance author let iidAuthor = remoteAuthorInstance author
roid <- roid <-
@ -167,12 +169,6 @@ sharerCreateNoteF now shrRecip author body (Note mluNote _ _ muParent muContext
return $ "Activity already exists in inbox of /s/" <> recip return $ "Activity already exists in inbox of /s/" <> recip
Just _ -> return $ "Activity inserted to inbox of /s/" <> recip Just _ -> return $ "Activity inserted to inbox of /s/" <> recip
data CreateNoteRecipColl
= CreateNoteRecipProjectFollowers
| CreateNoteRecipTicketParticipants
| CreateNoteRecipTicketTeam
deriving Eq
projectCreateNoteF projectCreateNoteF
:: UTCTime :: UTCTime
-> ShrIdent -> ShrIdent
@ -195,84 +191,94 @@ projectCreateNoteF now shrRecip prjRecip author body (Note mluNote _ _ muParent
if uParent == uContext if uParent == uContext
then return Nothing then return Nothing
else Just <$> parseParent uParent else Just <$> parseParent uParent
(localRecips, _remoteRecips) <- do
mrecips <- parseAudience $ activityAudience $ actbActivity body
fromMaybeE mrecips "Create Note with no recipients"
msig <- checkForward shrRecip prjRecip
case context of case context of
Right _ -> return $ recip <> " not using; context isn't local" Right _ -> return "Not using; context isn't local"
Left (shr, prj, ltkhid) -> Left (NoteContextSharerTicket shr talid) -> do
if shr /= shrRecip || prj /= prjRecip mremotesHttp <- runDBExcept $ do
then return $ recip <> " not using; context is a different project" (jid, ibid) <- lift getProjectRecip404
else do (_, _, _, project) <- do
msig <- checkForward shrRecip prjRecip mticket <- lift $ getSharerTicket shr talid
hLocal <- getsYesod $ appInstanceHost . appSettings fromMaybeE mticket "Context: No such sharer-ticket"
let colls = case project of
findRelevantCollections hLocal ltkhid $ Left (Entity _ tpl)
activityAudience $ actbActivity body | ticketProjectLocalProject tpl == jid -> do
mremotesHttp <- runDBExcept $ do mractid <- lift $ insertToProjectInbox ibid luCreate
(sid, fsidProject, fsidTicket, jid, ibid, did, meparent) <- getContextAndParent ltkhid mparent case mractid of
lift $ join <$> do Nothing -> return $ Left "Activity already in my inbox"
mmid <- insertToDiscussion luCreate luNote published ibid did meparent fsidTicket Just ractid ->
for mmid $ \ (ractid, mid) -> do case msig of
updateOrphans luNote did mid Nothing ->
for msig $ \ sig -> do return $ Left
remoteRecips <- deliverLocal ractid colls sid fsidProject fsidTicket "Context is a sharer-ticket, \
(sig,) <$> deliverRemoteDB (actbBL body) ractid jid sig remoteRecips \but no inbox forwarding \
lift $ for_ mremotesHttp $ \ (sig, remotesHttp) -> do \header for me, so doing \
let handler e = logError $ "Project inbox handler: delivery failed! " <> T.pack (displayException e) \nothing, just storing in inbox"
forkHandler handler $ Just sig -> lift $ Right <$> do
deliverRemoteHTTP now shrRecip prjRecip (actbBL body) sig remotesHttp let sieve =
return $ recip <> " inserted new ticket comment" makeRecipientSet
[]
[ LocalPersonCollectionProjectFollowers shrRecip prjRecip
, LocalPersonCollectionProjectTeam shrRecip prjRecip
]
remoteRecips <- insertRemoteActivityToLocalInboxes False ractid $ localRecipSieve' sieve False False localRecips
(sig,) <$> deliverRemoteDB (actbBL body) ractid jid sig remoteRecips
_ -> return $ Left "Context is a sharer-ticket of another project"
case mremotesHttp of
Left msg -> return msg
Right (sig, remotesHttp) -> do
forkWorker "projectCreateNoteF inbox forwarding by http" $ deliverRemoteHTTP now shrRecip prjRecip (actbBL body) sig remotesHttp
return "Stored to inbox and did inbox forwarding"
Left (NoteContextProjectTicket shr prj ltid) -> do
mremotesHttp <- runDBExcept $ do
(jid, ibid) <- lift getProjectRecip404
(_, _, _, Entity _ lt, Entity _ tpl, _) <- do
mticket <- lift $ getProjectTicket shr prj ltid
fromMaybeE mticket "Context: No such project-ticket"
if ticketProjectLocalProject tpl == jid
then do
mractid <- lift $ insertToProjectInbox ibid luCreate
case mractid of
Nothing -> return $ Left "Activity already in my inbox"
Just ractid -> do
let did = localTicketDiscuss lt
meparent <- traverse (getParent did) mparent
mmid <- lift $ insertToDiscussion luNote published did meparent ractid
case mmid of
Nothing -> return $ Left "I already have this comment, just storing in inbox"
Just mid -> lift $ do
updateOrphans luNote did mid
case msig of
Nothing ->
return $ Left "Storing in inbox, caching comment, no inbox forwarding header"
Just sig -> Right <$> do
ltkhid <- encodeKeyHashid ltid
let sieve =
makeRecipientSet
[]
[ LocalPersonCollectionProjectFollowers shrRecip prjRecip
, LocalPersonCollectionProjectTeam shrRecip prjRecip
, LocalPersonCollectionProjectTicketFollowers shrRecip prjRecip ltkhid
, LocalPersonCollectionProjectTicketTeam shrRecip prjRecip ltkhid
]
remoteRecips <- insertRemoteActivityToLocalInboxes False ractid $ localRecipSieve' sieve False False localRecips
(sig,) <$> deliverRemoteDB (actbBL body) ractid jid sig remoteRecips
else return $ Left "Context is a project-ticket of another project"
case mremotesHttp of
Left msg -> return msg
Right (sig, remotesHttp) -> do
forkWorker "projectCreateNoteF inbox forwarding by http" $ deliverRemoteHTTP now shrRecip prjRecip (actbBL body) sig remotesHttp
return "Stored to inbox, cached comment, and did inbox forwarding"
where where
findRelevantCollections hLocal ctx = nub . mapMaybe decide . concatRecipients getProjectRecip404 = do
where sid <- getKeyBy404 $ UniqueSharer shrRecip
decide u = do Entity jid j <- getBy404 $ UniqueProject prjRecip sid
let ObjURI h lu = u return (jid, projectInbox j)
guard $ h == hLocal insertToProjectInbox ibid luCreate = do
route <- decodeRouteLocal lu
case route of
ProjectFollowersR shr prj
| shr == shrRecip && prj == prjRecip
-> Just CreateNoteRecipProjectFollowers
ProjectTicketParticipantsR shr prj tkhid
| shr == shrRecip && prj == prjRecip && tkhid == ctx
-> Just CreateNoteRecipTicketParticipants
ProjectTicketTeamR shr prj tkhid
| shr == shrRecip && prj == prjRecip && tkhid == ctx
-> Just CreateNoteRecipTicketTeam
_ -> Nothing
recip = T.concat ["/s/", shr2text shrRecip, "/p/", prj2text prjRecip]
getContextAndParent ltkhid mparent = do
mt <- do
sid <- lift $ getKeyBy404 $ UniqueSharer shrRecip
Entity jid j <- lift $ getBy404 $ UniqueProject prjRecip sid
ltid <- decodeKeyHashidE ltkhid "Context: Not a valid ticket khid"
mlt <- lift $ get ltid
for mlt $ \ lt -> do
mtpl <- lift $ getValBy $ UniqueTicketProjectLocal $ localTicketTicket lt
tpl <- fromMaybeE mtpl "No TPL"
unless (ticketProjectLocalProject tpl == jid) $
throwE "Context: Local ticket khid belongs to different project"
return (jid, projectInbox j, projectFollowers j, sid, lt)
(jid, ibid, fsidProject, sid, lt) <- fromMaybeE mt "Context: No such local ticket"
let did = localTicketDiscuss lt
meparent <- for mparent $ \ parent ->
case parent of
Left (shrParent, lmidParent) -> Left <$> getLocalParentMessageId did shrParent lmidParent
Right p@(ObjURI hParent luParent) -> do
mrm <- lift $ runMaybeT $ do
iid <- MaybeT $ getKeyBy $ UniqueInstance hParent
roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luParent
MaybeT $ getValBy $ UniqueRemoteMessageIdent roid
case mrm of
Just rm -> Left <$> do
let mid = remoteMessageRest rm
m <- lift $ getJust mid
unless (messageRoot m == did) $
throwE "Remote parent belongs to a different discussion"
return mid
Nothing -> return $ Right p
return (sid, fsidProject, localTicketFollowers lt, jid, ibid, did, meparent)
insertToDiscussion luCreate luNote published ibid did meparent fsid = do
let iidAuthor = remoteAuthorInstance author let iidAuthor = remoteAuthorInstance author
raidAuthor = remoteAuthorId author
roid <- roid <-
either entityKey id <$> insertBy' (RemoteObject iidAuthor luCreate) either entityKey id <$> insertBy' (RemoteObject iidAuthor luCreate)
ractid <- either entityKey id <$> insertBy' RemoteActivity ractid <- either entityKey id <$> insertBy' RemoteActivity
@ -280,6 +286,29 @@ projectCreateNoteF now shrRecip prjRecip author body (Note mluNote _ _ muParent
, remoteActivityContent = persistJSONFromBL $ actbBL body , remoteActivityContent = persistJSONFromBL $ actbBL body
, remoteActivityReceived = now , remoteActivityReceived = now
} }
ibiid <- insert $ InboxItem False
new <- isRight <$> insertBy' (InboxItemRemote ibid ractid ibiid)
return $
if new
then Just ractid
else Nothing
getParent did (Left (shrParent, lmidParent)) = Left <$> getLocalParentMessageId did shrParent lmidParent
getParent did (Right p@(ObjURI hParent luParent)) = do
mrm <- lift $ runMaybeT $ do
iid <- MaybeT $ getKeyBy $ UniqueInstance hParent
roid <- MaybeT $ getKeyBy $ UniqueRemoteObject iid luParent
MaybeT $ getValBy $ UniqueRemoteMessageIdent roid
case mrm of
Just rm -> Left <$> do
let mid = remoteMessageRest rm
m <- lift $ getJust mid
unless (messageRoot m == did) $
throwE "Remote parent belongs to a different discussion"
return mid
Nothing -> return $ Right p
insertToDiscussion luNote published did meparent ractid = do
let iidAuthor = remoteAuthorInstance author
raidAuthor = remoteAuthorId author
mid <- insert Message mid <- insert Message
{ messageCreated = published { messageCreated = published
, messageSource = src , messageSource = src
@ -290,11 +319,11 @@ projectCreateNoteF now shrRecip prjRecip author body (Note mluNote _ _ muParent
_ -> Nothing _ -> Nothing
, messageRoot = did , messageRoot = did
} }
roid2 <- roidNote <-
either entityKey id <$> insertBy' (RemoteObject iidAuthor luNote) either entityKey id <$> insertBy' (RemoteObject iidAuthor luNote)
mrmid <- insertUnique RemoteMessage mrmid <- insertUnique RemoteMessage
{ remoteMessageAuthor = raidAuthor { remoteMessageAuthor = raidAuthor
, remoteMessageIdent = roid2 , remoteMessageIdent = roidNote
, remoteMessageRest = mid , remoteMessageRest = mid
, remoteMessageCreate = ractid , remoteMessageCreate = ractid
, remoteMessageLostParent = , remoteMessageLostParent =
@ -306,11 +335,7 @@ projectCreateNoteF now shrRecip prjRecip author body (Note mluNote _ _ muParent
Nothing -> do Nothing -> do
delete mid delete mid
return Nothing return Nothing
Just _ -> do Just _ -> return $ Just mid
-- insertUnique_ $ RemoteFollow raidAuthor fsid False True
ibiid <- insert $ InboxItem False
insert_ $ InboxItemRemote ibid ractid ibiid
return $ Just (ractid, mid)
updateOrphans luNote did mid = do updateOrphans luNote did mid = do
let hAuthor = objUriAuthority $ remoteAuthorURI author let hAuthor = objUriAuthority $ remoteAuthorURI author
uNote = ObjURI hAuthor luNote uNote = ObjURI hAuthor luNote
@ -341,32 +366,3 @@ projectCreateNoteF now shrRecip prjRecip author body (Note mluNote _ _ muParent
rm E.^. RemoteMessageLostParent E.==. E.just (E.val uNote) E.&&. rm E.^. RemoteMessageLostParent E.==. E.just (E.val uNote) E.&&.
m E.^. MessageRoot `op` E.val did m E.^. MessageRoot `op` E.val did
return (rm E.^. RemoteMessageId, m E.^. MessageId) return (rm E.^. RemoteMessageId, m E.^. MessageId)
deliverLocal
:: RemoteActivityId
-> [CreateNoteRecipColl]
-> SharerId
-> FollowerSetId
-> FollowerSetId
-> AppDB [((InstanceId, Host), NonEmpty RemoteRecipient)]
deliverLocal ractid recips sid fsidProject fsidTicket = do
(teamPids, teamRemotes) <-
if CreateNoteRecipTicketTeam `elem` recips
then getTicketTeam sid
else return ([], [])
(tfsPids, tfsRemotes) <-
if CreateNoteRecipTicketParticipants `elem` recips
then getFollowers fsidTicket
else return ([], [])
(jfsPids, jfsRemotes) <-
if CreateNoteRecipProjectFollowers `elem` recips
then getFollowers fsidProject
else return ([], [])
let pids = union teamPids tfsPids `union` jfsPids
remotes = teamRemotes `unionRemotes` tfsRemotes `unionRemotes` jfsRemotes
for_ pids $ \ pid -> do
ibid <- personInbox <$> getJust pid
ibiid <- insert $ InboxItem True
mibrid <- insertUnique $ InboxItemRemote ibid ractid ibiid
when (isNothing mibrid) $
delete ibiid
return remotes