1
0
Fork 0
mirror of https://code.naskya.net/repos/ndqEd synced 2025-01-25 16:17:50 +09:00

S2S: Implement loomResolveF, allowing to close MR without Applying

This commit is contained in:
fr33domlover 2022-10-25 18:49:19 +00:00
parent 756c2952f2
commit 648204ef80
2 changed files with 117 additions and 171 deletions

View file

@ -13,6 +13,8 @@
- <http://creativecommons.org/publicdomain/zero/1.0/>. - <http://creativecommons.org/publicdomain/zero/1.0/>.
-} -}
{-# LANGUAGE RankNTypes #-}
module Vervis.Federation.Ticket module Vervis.Federation.Ticket
( --personOfferTicketF ( --personOfferTicketF
deckOfferTicketF deckOfferTicketF
@ -26,7 +28,7 @@ module Vervis.Federation.Ticket
--, repoOfferDepF --, repoOfferDepF
, deckResolveF , deckResolveF
--, repoResolveF , loomResolveF
) )
where where
@ -1864,29 +1866,50 @@ insertResolve author ltid ractid obiidAccept = do
} }
-} -}
deckResolveF trackerResolveF
:: UTCTime :: ( PersistRecordBackend r SqlBackend, ToBackendKey SqlBackend r
-> KeyHashid Deck , ToBackendKey SqlBackend wi
)
=> (Route App -> Maybe (KeyHashid wi))
-> (r -> ActorId)
-> ( Key r
-> Key wi
-> ExceptT Text AppDB
( TicketId
, Maybe
( Entity TicketResolve
, Either
(Entity TicketResolveLocal)
(Entity TicketResolveRemote)
)
)
)
-> (Key r -> GrantResourceBy Key)
-> (KeyHashid r -> LocalStageBy KeyHashid)
-> (KeyHashid r -> KeyHashid wi -> LocalStageBy KeyHashid)
-> (forall f. f r -> LocalActorBy f)
-> UTCTime
-> KeyHashid r
-> RemoteAuthor -> RemoteAuthor
-> ActivityBody -> ActivityBody
-> Maybe (RecipientRoutes, ByteString) -> Maybe (RecipientRoutes, ByteString)
-> LocalURI -> LocalURI
-> AP.Resolve URIMode -> AP.Resolve URIMode
-> ExceptT Text Handler (Text, Maybe (ExceptT Text Worker Text)) -> ExceptT Text Handler (Text, Maybe (ExceptT Text Worker Text))
deckResolveF now recipDeckHash author body mfwd luResolve (AP.Resolve uObject) = (,Nothing) <$> do trackerResolveF maybeWorkItem grabActor getWorkItem makeResource trackerFollowers itemFollowers makeLocalActor now recipHash author body mfwd luResolve (AP.Resolve uObject) = (,Nothing) <$> do
-- Check input -- Check input
recipDeckID <- decodeKeyHashid404 recipDeckHash recipID <- decodeKeyHashid404 recipHash
taskID <- nameExceptT "Resolve object" $ do wiID <- nameExceptT "Resolve object" $ do
route <- do route <- do
routeOrRemote <- parseFedURI uObject routeOrRemote <- parseFedURI uObject
case routeOrRemote of case routeOrRemote of
Left route -> pure route Left route -> pure route
Right _ -> throwE "Remote, so definitely not mine" Right _ -> throwE "Remote, so definitely not mine"
case route of case maybeWorkItem route of
TicketR deckHash taskHash | deckHash == recipDeckHash -> Nothing -> throwE "Local route but not a work item of mine"
decodeKeyHashidE taskHash "Invalid task keyhashid" Just wiHash ->
_ -> throwE "Local route but not a ticket of mine" decodeKeyHashidE wiHash "Invalid work item keyhashid"
-- Verify the capability URI is one of: -- Verify the capability URI is one of:
-- * Outbox item URI of a local actor, i.e. a local activity -- * Outbox item URI of a local actor, i.e. a local activity
@ -1900,64 +1923,62 @@ deckResolveF now recipDeckHash author body mfwd luResolve (AP.Resolve uObject) =
maybeHttp <- runDBExcept $ do maybeHttp <- runDBExcept $ do
-- Find recipient deck in DB, returning 404 if doesn't exist because -- Find recipient tracker in DB, returning 404 if doesn't exist because
-- we're in the deck's inbox post handler -- we're in the tracker's inbox post handler
recipDeck <- lift $ get404 recipDeckID recip <- lift $ get404 recipID
let recipDeckActorID = deckActor recipDeck let recipActorID = grabActor recip
recipDeckActor <- lift $ getJust recipDeckActorID recipActor <- lift $ getJust recipActorID
-- Insert the Resolve to deck's inbox -- Insert the Resolve to tracker's inbox
mractid <- lift $ insertToInbox now author body (actorInbox recipDeckActor) luResolve False mractid <- lift $ insertToInbox now author body (actorInbox recipActor) luResolve False
for mractid $ \ resolveID -> do for mractid $ \ resolveID -> do
-- Find ticket in DB, verify it's not resolved -- Find ticket in DB, verify it's not resolved
ticketID <- do ticketID <- do
maybeTicket <- lift $ getTicket recipDeckID taskID (ticketID, maybeResolve) <- getWorkItem recipID wiID
(_deck, _task, Entity ticketID _, _author, maybeResolve) <-
fromMaybeE maybeTicket "I don't have such a ticket in DB"
unless (isNothing maybeResolve) $ unless (isNothing maybeResolve) $
throwE "Ticket is already resolved" throwE "Work item is already resolved"
return ticketID return ticketID
-- Verify the sender is authorized by the deck to resolve a ticket -- Verify the sender is authorized by the tracker to resolve a ticket
capability <- capability <-
case capID of case capID of
Left (capActor, _, capItem) -> return (capActor, capItem) Left (capActor, _, capItem) -> return (capActor, capItem)
Right _ -> throwE "Capability is a remote URI, i.e. not authored by the local deck" Right _ -> throwE "Capability is a remote URI, i.e. not authored by the local tracker"
verifyCapability verifyCapability
capability capability
(Right $ remoteAuthorId author) (Right $ remoteAuthorId author)
(GrantResourceDeck recipDeckID) (makeResource recipID)
-- Forward the Resolve activity to relevant local stages, and -- Forward the Resolve activity to relevant local stages, and
-- schedule delivery for unavailable remote members of them -- schedule delivery for unavailable remote members of them
maybeHttpFwdResolve <- lift $ for mfwd $ \ (localRecips, sig) -> do maybeHttpFwdResolve <- lift $ for mfwd $ \ (localRecips, sig) -> do
taskHash <- encodeKeyHashid taskID wiHash <- encodeKeyHashid wiID
let sieve = let sieve =
makeRecipientSet makeRecipientSet
[] []
[ LocalStageDeckFollowers recipDeckHash [ trackerFollowers recipHash
, LocalStageTicketFollowers recipDeckHash taskHash , itemFollowers recipHash wiHash
] ]
forwardActivityDB forwardActivityDB
(actbBL body) localRecips sig recipDeckActorID (actbBL body) localRecips sig recipActorID
(LocalActorDeck recipDeckHash) sieve resolveID (makeLocalActor recipHash) sieve resolveID
-- Mark ticket in DB as resolved by the Resolve -- Mark ticket in DB as resolved by the Resolve
acceptID <- acceptID <-
lift $ insertEmptyOutboxItem (actorOutbox recipDeckActor) now lift $ insertEmptyOutboxItem (actorOutbox recipActor) now
lift $ insertResolve ticketID resolveID acceptID lift $ insertResolve ticketID resolveID acceptID
-- Prepare an Accept activity and insert to deck's outbox -- Prepare an Accept activity and insert to tracker's outbox
(actionAccept, localRecipsAccept, remoteRecipsAccept, fwdHostsAccept) <- (actionAccept, localRecipsAccept, remoteRecipsAccept, fwdHostsAccept) <-
lift $ prepareAccept taskID lift $ prepareAccept wiID
_luAccept <- lift $ updateOutboxItem (LocalActorDeck recipDeckID) acceptID actionAccept _luAccept <- lift $ updateOutboxItem (makeLocalActor recipID) acceptID actionAccept
-- Deliver the Accept to local recipients, and schedule delivery -- Deliver the Accept to local recipients, and schedule delivery
-- for unavailable remote recipients -- for unavailable remote recipients
deliverHttpAccept <- deliverHttpAccept <-
deliverActivityDB deliverActivityDB
(LocalActorDeck recipDeckHash) recipDeckActorID (makeLocalActor recipHash) recipActorID
localRecipsAccept remoteRecipsAccept fwdHostsAccept localRecipsAccept remoteRecipsAccept fwdHostsAccept
acceptID actionAccept acceptID actionAccept
@ -1973,8 +1994,8 @@ deckResolveF now recipDeckHash author body mfwd luResolve (AP.Resolve uObject) =
return "I already have this activity in my inbox, doing nothing" return "I already have this activity in my inbox, doing nothing"
Just (maybeHttpFwdResolve, deliverHttpAccept) -> do Just (maybeHttpFwdResolve, deliverHttpAccept) -> do
for_ maybeHttpFwdResolve $ for_ maybeHttpFwdResolve $
forkWorker "deckResolveF inbox-forwarding" forkWorker "trackerResolveF inbox-forwarding"
forkWorker "deckResolveF Accept HTTP delivery" deliverHttpAccept forkWorker "trackerResolveF Accept HTTP delivery" deliverHttpAccept
return $ return $
case maybeHttpFwdResolve of case maybeHttpFwdResolve of
Nothing -> "Resolved ticket, no inbox-forwarding to do" Nothing -> "Resolved ticket, no inbox-forwarding to do"
@ -1994,10 +2015,10 @@ deckResolveF now recipDeckHash author body mfwd luResolve (AP.Resolve uObject) =
, ticketResolveRemoteActor = remoteAuthorId author , ticketResolveRemoteActor = remoteAuthorId author
} }
prepareAccept taskID = do prepareAccept wiID = do
encodeRouteHome <- getEncodeRouteHome encodeRouteHome <- getEncodeRouteHome
taskHash <- encodeKeyHashid taskID wiHash <- encodeKeyHashid wiID
ra <- getJust $ remoteAuthorId author ra <- getJust $ remoteAuthorId author
@ -2010,8 +2031,8 @@ deckResolveF now recipDeckHash author body mfwd luResolve (AP.Resolve uObject) =
audTracker = audTracker =
AudLocal AudLocal
[] []
[ LocalStageDeckFollowers recipDeckHash [ trackerFollowers recipHash
, LocalStageTicketFollowers recipDeckHash taskHash , itemFollowers recipHash wiHash
] ]
(recipientSet, remoteActors, fwdHosts, audLocal, audRemote) = (recipientSet, remoteActors, fwdHosts, audLocal, audRemote) =
@ -2031,139 +2052,62 @@ deckResolveF now recipDeckHash author body mfwd luResolve (AP.Resolve uObject) =
return (action, recipientSet, remoteActors, fwdHosts) return (action, recipientSet, remoteActors, fwdHosts)
repoResolveF deckResolveF
:: UTCTime :: UTCTime
-> KeyHashid Repo -> KeyHashid Deck
-> RemoteAuthor -> RemoteAuthor
-> ActivityBody -> ActivityBody
-> Maybe (RecipientRoutes, ByteString) -> Maybe (RecipientRoutes, ByteString)
-> LocalURI -> LocalURI
-> Resolve URIMode -> AP.Resolve URIMode
-> ExceptT Text Handler Text -> ExceptT Text Handler (Text, Maybe (ExceptT Text Worker Text))
repoResolveF now recipHash author body mfwd luResolve (Resolve uObject) = do deckResolveF now deckHash =
error "repoResolveF temporarily disabled" trackerResolveF
(\case
TicketR trackerHash taskHash | trackerHash == deckHash ->
Just taskHash
_ -> Nothing
)
deckActor
(\ deckID taskID -> do
maybeTicket <- lift $ getTicket deckID taskID
(_deck, _task, Entity ticketID _, _author, maybeResolve) <-
fromMaybeE maybeTicket "I don't have such a ticket in DB"
return (ticketID, maybeResolve)
)
GrantResourceDeck
LocalStageDeckFollowers
LocalStageTicketFollowers
LocalActorDeck
now
deckHash
loomResolveF
:: UTCTime
{- -> KeyHashid Loom
-> RemoteAuthor
-> ActivityBody
object <- parseWorkItem "Resolve object" uObject -> Maybe (RecipientRoutes, ByteString)
mmmmhttp <- runDBExcept $ do -> LocalURI
Entity ridRecip repoRecip <- lift $ do -> AP.Resolve URIMode
sid <- getKeyBy404 $ UniqueSharer shrRecip -> ExceptT Text Handler (Text, Maybe (ExceptT Text Worker Text))
getBy404 $ UniqueRepo rpRecip sid loomResolveF now loomHash =
mltid <- trackerResolveF
case relevantObject object of (\case
Nothing -> do ClothR trackerHash clothHash | trackerHash == loomHash ->
case object of Just clothHash
Left wi -> verifyWorkItemExists wi _ -> Nothing
Right _ -> return () )
return Nothing loomActor
Just ltid -> Just . (ltid,) <$> getObjectLtid ltid (\ loomID clothID -> do
mractid <- lift $ insertToInbox now author body (repoInbox repoRecip) luResolve False maybeCloth <- lift $ getCloth loomID clothID
lift $ for mractid $ \ ractid -> for mltid $ \ (ltid, tid) -> do (_loom, _cloth, Entity ticketID _, _author, maybeResolve, _merge) <-
mremotesHttpFwd <- for mfwd $ \ (localRecips, sig) -> do fromMaybeE maybeCloth "I don't have such a MR in DB"
ltkhid <- encodeKeyHashid ltid return (ticketID, maybeResolve)
let sieve = )
makeRecipientSet GrantResourceLoom
[] LocalStageLoomFollowers
[ LocalPersonCollectionRepoProposalFollowers shrRecip rpRecip ltkhid LocalStageClothFollowers
, LocalPersonCollectionRepoTeam shrRecip rpRecip LocalActorLoom
, LocalPersonCollectionRepoFollowers shrRecip rpRecip now
] loomHash
remoteRecips <-
insertRemoteActivityToLocalInboxes
False ractid $
localRecipSieve'
sieve False False localRecips
(sig,) <$> deliverRemoteDB_R (actbBL body) ractid ridRecip sig remoteRecips
obiidAccept <- insertEmptyOutboxItem (repoOutbox repoRecip) now
mmtrrid <- insertResolve author ltid ractid obiidAccept
case mmtrrid of
Just (Just _) -> update tid [TicketStatus =. TSClosed]
_ -> delete obiidAccept
for mmtrrid $ \ mtrrid -> for mtrrid $ \ trrid -> do
(docAccept, localRecipsAccept, remoteRecipsAccept, fwdHostsAccept) <-
insertAccept luResolve ltid obiidAccept
knownRemoteRecipsAccept <-
deliverLocal'
False
(LocalActorRepo shrRecip rpRecip)
(repoInbox repoRecip)
obiidAccept
localRecipsAccept
(mremotesHttpFwd,obiidAccept,docAccept,fwdHostsAccept,) <$>
deliverRemoteDB fwdHostsAccept obiidAccept remoteRecipsAccept knownRemoteRecipsAccept
case mmmmhttp of
Nothing -> return "I already have this activity in my inbox, doing nothing"
Just mmmhttp ->
case mmmhttp of
Nothing -> return "Object not mine, just stored in inbox"
Just mmhttp ->
case mmhttp of
Nothing -> return "Ticket already resolved"
Just mhttp ->
case mhttp of
Nothing -> return "Activity already resolved a ticket"
Just (mremotesHttpFwd, obiid, doc, fwdHosts, recips) -> do
for_ mremotesHttpFwd $ \ (sig, remotes) ->
forkWorker "repoResolveF inbox-forwarding" $
deliverRemoteHTTP_R now shrRecip rpRecip (actbBL body) sig remotes
forkWorker "repoResolveF Accept HTTP delivery" $
deliverRemoteHttp' fwdHosts obiid doc recips
return $
if isJust mremotesHttpFwd
then "Ticket is mine, now resolved, did inbox-forwarding"
else "Ticket is mine, now resolved, no inbox-forwarding to do"
where
relevantObject (Left (WorkItemRepoProposal shr rp ltid))
| shr == shrRecip && rp == rpRecip = Just ltid
relevantObject _ = Nothing
getObjectLtid ltid = do
(_, _, Entity tid _, _, _, _, _, _, _) <- do
mticket <- lift $ getRepoProposal shrRecip rpRecip ltid
fromMaybeE mticket $ "Object" <> ": No such repo-patch"
return tid
insertAccept luResolve ltid obiidAccept = do
encodeRouteLocal <- getEncodeRouteLocal
encodeRouteHome <- getEncodeRouteHome
hLocal <- asksSite siteInstanceHost
obikhidAccept <- encodeKeyHashid obiidAccept
ltkhid <- encodeKeyHashid ltid
ra <- getJust $ remoteAuthorId author
let ObjURI hAuthor luAuthor = remoteAuthorURI author
audAuthor =
AudRemote hAuthor [luAuthor] (maybeToList $ remoteActorFollowers ra)
audTicket =
AudLocal
[]
[ LocalPersonCollectionRepoProposalFollowers shrRecip rpRecip ltkhid
, LocalPersonCollectionRepoTeam shrRecip rpRecip
, LocalPersonCollectionRepoFollowers shrRecip rpRecip
]
(recipientSet, remoteActors, fwdHosts, audLocal, audRemote) =
collectAudience [audAuthor, audTicket]
recips = map encodeRouteHome audLocal ++ audRemote
doc = Doc hLocal Activity
{ activityId =
Just $ encodeRouteLocal $
RepoOutboxItemR shrRecip rpRecip obikhidAccept
, activityActor = encodeRouteLocal $ RepoR shrRecip rpRecip
, activityCapability = Nothing
, activitySummary = Nothing
, activityAudience = Audience recips [] [] [] [] []
, activitySpecific = AcceptActivity Accept
{ acceptObject = ObjURI hAuthor luResolve
, acceptResult = Nothing
}
}
update obiidAccept [OutboxItemActivity =. persistJSONObjectFromDoc doc]
return (doc, recipientSet, remoteActors, fwdHosts)
-}

View file

@ -166,6 +166,8 @@ postLoomInboxR recipLoomHash =
AP.OfferTicket ticket -> AP.OfferTicket ticket ->
loomOfferTicketF now recipLoomHash author body mfwd luActivity ticket target loomOfferTicketF now recipLoomHash author body mfwd luActivity ticket target
_ -> return ("Unsupported offer object type for looms", Nothing) _ -> return ("Unsupported offer object type for looms", Nothing)
AP.ResolveActivity resolve ->
loomResolveF now recipLoomHash author body mfwd luActivity resolve
_ -> return ("Unsupported activity type for looms", Nothing) _ -> return ("Unsupported activity type for looms", Nothing)
getLoomOutboxR :: KeyHashid Loom -> Handler TypedContent getLoomOutboxR :: KeyHashid Loom -> Handler TypedContent