UI, S2S: Re-implement and re-enable Push activity

- When pushing to a repo, a Push activity is now automatically published
- The 'actor' is now the repo, and 'attributedTo' specifies the person who
  pushed
- No need for 'context' in the Push anymore, since it's always the 'actor'
- 'target' now specifies the branch as a Branch object rather than URI (since
  Vervis doesn't keep AS2 objects for branches anymore)
- I deleted 'pushCommitsC' (from Vervis.API) because the code for preparing and
  pushing an activity is so simple with the new delivery API, doesn't need a
  dedicated pushCommitsC function
- The generated Push activity does generate an HTML summary, unlike all other
  generated activities (in which I removed the summary generating code); I'm
  still unsure whether to bring back those summaries (extra code to write, for
  a problematic feature that may become useless when the new UI comes)
This commit is contained in:
fr33domlover 2022-10-26 10:47:38 +00:00
parent 340d1eacb1
commit 934c69daae
3 changed files with 156 additions and 179 deletions

View file

@ -31,7 +31,6 @@ module Vervis.API
--, offerDepC --, offerDepC
, resolveC , resolveC
, undoC , undoC
--, pushCommitsC
) )
where where
@ -3067,92 +3066,3 @@ undoC (Entity senderPersonID senderPerson) senderActor maybeCap localRecips remo
, AP.acceptResult = Nothing , AP.acceptResult = Nothing
} }
} }
pushCommitsC
:: Entity Person
-> Html
-> Push URIMode
-> ShrIdent
-> RpIdent
-> ExceptT Text Handler OutboxItemId
pushCommitsC eperson summary push shrRepo rpRepo = do
error "pushCommitsC temporarily disabled"
{-
let dont = Authority "dont-do.any-forwarding" Nothing
(obiid, doc, remotesHttp) <- runDBExcept $ do
(obiid, doc) <- lift $ insertToOutbox
remoteRecips <- lift $ deliverLocal obiid
federation <- getsYesod $ appFederation . appSettings
unless (federation || null remoteRecips) $
throwE "Federation disabled but remote collection members found"
remotesHttp <- lift $ deliverRemoteDB' dont obiid [] remoteRecips
return (obiid, doc, remotesHttp)
lift $ forkWorker "pushCommitsC: async HTTP delivery" $ deliverRemoteHttp dont obiid doc remotesHttp
return obiid
where
insertToOutbox :: AppDB (OutboxItemId, Doc Activity URIMode)
insertToOutbox = do
host <- getsYesod siteInstanceHost
encodeRouteLocal <- getEncodeRouteLocal
encodeRouteHome <- getEncodeRouteHome
let shrUser = sharerIdent sharer
aud = map encodeRouteHome
[ SharerFollowersR shrUser
, RepoR shrRepo rpRepo
, RepoTeamR shrRepo rpRepo
, RepoFollowersR shrRepo rpRepo
]
activity mluAct = Doc host Activity
{ activityId = mluAct
, activityActor = encodeRouteLocal $ SharerR shrUser
, activityCapability = Nothing
, activitySummary =
Just $ renderHTML summary
, activityAudience = Audience aud [] [] [] [] []
, activitySpecific = PushActivity push
}
now <- liftIO getCurrentTime
obiid <- insert OutboxItem
{ outboxItemOutbox = personOutbox $ entityVal eperson
, outboxItemActivity = persistJSONObjectFromDoc $ activity Nothing
, outboxItemPublished = now
}
obikhid <- encodeKeyHashid obiid
let luAct = encodeRouteLocal $ SharerOutboxItemR shrUser obikhid
doc = activity $ Just luAct
update obiid [OutboxItemActivity =. persistJSONObjectFromDoc doc]
return (obiid, doc)
deliverLocal
:: OutboxItemId
-> AppDB
[ ( (InstanceId, Host)
, NonEmpty RemoteRecipient
)
]
deliverLocal obiid = do
let pidAuthor = entityKey eperson
(sidRepo, repo) <- do
sid <- getKeyBy404 $ UniqueSharer shrRepo
r <- getValBy404 $ UniqueRepo rpRepo sid
return (sid, r)
(pids, remotes) <- do
(repoPids, repoRemotes) <- getRepoTeam sidRepo
(pfsPids, pfsRemotes) <-
getFollowers $ personFollowers $ entityVal eperson
(rfsPids, rfsRemotes) <- getFollowers $ repoFollowers repo
return
( L.delete pidAuthor $ union repoPids $ union pfsPids rfsPids
, repoRemotes `unionRemotes` pfsRemotes `unionRemotes` rfsRemotes
)
ibiid <- insert $ InboxItem False now
insert_ $ InboxItemLocal (repoInbox repo) obiid ibiid
for_ pids $ \ pid -> do
ibid <- personInbox <$> getJust pid
ibiid <- insert $ InboxItem True now
insert_ $ InboxItemLocal ibid obiid ibiid
return remotes
-}

View file

@ -129,6 +129,7 @@ import qualified Data.ByteString.Lazy.Internal as BLI
import qualified Data.CaseInsensitive as CI (foldedCase) import qualified Data.CaseInsensitive as CI (foldedCase)
import qualified Data.DList as D import qualified Data.DList as D
import qualified Data.Set as S (member) import qualified Data.Set as S (member)
import qualified Data.Text.Encoding as TE
import qualified Data.Text as T import qualified Data.Text as T
import qualified Data.Text.Lazy.Encoding as L (decodeUtf8With) import qualified Data.Text.Lazy.Encoding as L (decodeUtf8With)
import qualified Database.Esqueleto as E import qualified Database.Esqueleto as E
@ -137,6 +138,7 @@ import Data.MediaType
import Database.Persist.JSON import Database.Persist.JSON
import Development.PatchMediaType import Development.PatchMediaType
import Network.FedURI import Network.FedURI
import Web.Text
import Yesod.ActivityPub import Yesod.ActivityPub
import Yesod.FedURI import Yesod.FedURI
import Yesod.Hashids import Yesod.Hashids
@ -150,6 +152,7 @@ import Data.Either.Local
import Data.Git.Local import Data.Git.Local
import Database.Persist.Local import Database.Persist.Local
import Text.FilePath.Local (breakExt) import Text.FilePath.Local (breakExt)
import Web.Hashids.Local
import Yesod.Form.Local import Yesod.Form.Local
import Yesod.Persist.Local import Yesod.Persist.Local
@ -157,6 +160,7 @@ import qualified Data.Git.Local as G (createRepo)
import qualified Darcs.Local.Repository as D (createRepo) import qualified Darcs.Local.Repository as D (createRepo)
import Vervis.Access import Vervis.Access
import Vervis.ActivityPub
import Vervis.API import Vervis.API
import Vervis.Federation.Auth import Vervis.Federation.Auth
import Vervis.Federation.Collab import Vervis.Federation.Collab
@ -168,6 +172,7 @@ import Vervis.Path
import Vervis.Model import Vervis.Model
import Vervis.Model.Ident import Vervis.Model.Ident
import Vervis.Paginate import Vervis.Paginate
import Vervis.Persist.Actor
import Vervis.Readme import Vervis.Readme
import Vervis.Recipient import Vervis.Recipient
import Vervis.Settings import Vervis.Settings
@ -175,6 +180,7 @@ import Vervis.SourceTree
import Vervis.Style import Vervis.Style
import Vervis.Web.Actor import Vervis.Web.Actor
import Vervis.Web.Darcs import Vervis.Web.Darcs
import Vervis.Web.Delivery
import Vervis.Web.Git import Vervis.Web.Git
import qualified Vervis.Client as C import qualified Vervis.Client as C
@ -538,111 +544,165 @@ postRepoUnfollowR :: KeyHashid Repo -> Handler ()
postRepoUnfollowR _ = error "Temporarily disabled" postRepoUnfollowR _ = error "Temporarily disabled"
postPostReceiveR :: Handler Text postPostReceiveR :: Handler Text
postPostReceiveR = return "Temporarily disabled, no Push activity published" postPostReceiveR = do
{- -- Parse the push object that the hook sent
push <- requireCheckJsonBody push <- requireCheckJsonBody
(pushAP, shr, rp) <- push2ap push
user <- runDB $ do errorOrPush <- runExceptT $ do
p <- getJustEntity $ toSqlKey $ H.pushUser push
s <- getJust $ personIdent $ entityVal p -- Compose an ActivityPub Push activity
return (p, s) (pushAP, repoID, repoHash) <- lift $ push2ap push
let shrUser = sharerIdent $ snd user
summary <- do -- Find repo and person in DB
let mbranch = H.pushBranch push let pusherID = toSqlKey $ H.pushUser push
total = pushCommitsTotal pushAP (Entity actorID actor, pusher) <- runDBExcept $ do
lasts = pushCommitsLast pushAP
rest firsts = total - length firsts - length lasts repoActorEntity <- do
hashText (Hash b) = decodeUtf8 b repo <- getE repoID "Repo not found in DB"
commitW c = lift $ getJustEntity $ repoActor repo
[hamlet|
<a href=@{RepoCommitR shr rp $ hashText $ commitHash c}> person <- getE pusherID "Pusher person not found in DB"
#{commitTitle c} let actorID = personActor person
|] actor <- lift $ getJust actorID
withUrlRenderer let pusher = (Entity pusherID person, actor)
[hamlet|
<p> return (repoActorEntity, pusher)
<a href=@{SharerR shrUser}>#{shr2text shrUser}
\ pushed #{total} # -- Compose summary and audience
\ #{commitsText mbranch total} to repo # let repoName = actorName actor
<a href=@{RepoR shr rp}>#{rp2text rp}</a>^{branchText shr rp mbranch}: summary <-
<ul> lift $ renderHTML <$> makeSummary push pushAP repoHash repoName pusher
$maybe firsts <- pushCommitsFirst pushAP let audience = [AudLocal [] [LocalStageRepoFollowers repoHash]]
$forall c <- firsts (localRecips, remoteRecips, fwdHosts, action) <-
<li>^{commitW c} lift $ C.makeServerInput Nothing (Just summary) audience (AP.PushActivity pushAP)
<li>#{rest firsts}
$forall c <- lasts -- Publish and deliver Push activity
<li>^{commitW c} now <- liftIO getCurrentTime
|] runDBExcept $ do
eid <- runExceptT $ pushCommitsC user summary pushAP shr rp pushID <- lift $ insertEmptyOutboxItem (actorOutbox actor) now
case eid of luPush <- lift $ updateOutboxItem (LocalActorRepo repoID) pushID action
deliverHttpPush <-
deliverActivityDB
(LocalActorRepo repoHash) actorID localRecips remoteRecips
fwdHosts pushID action
return (luPush, deliverHttpPush)
-- HTTP delivery to remote recipients
case errorOrPush of
Left e -> liftIO $ throwIO $ userError $ T.unpack e Left e -> liftIO $ throwIO $ userError $ T.unpack e
Right obiid -> do Right (luPush, deliverHttpPush) -> do
renderUrl <- askUrlRender forkWorker "PostReceiveR: async HTTP Push delivery" deliverHttpPush
obikhid <- encodeKeyHashid obiid hLocal <- asksSite siteInstanceHost
return $ return $
"Push activity published: " <> "Push activity published: " <>
renderUrl (SharerOutboxItemR shrUser obikhid) renderObjURI (ObjURI hLocal luPush)
where where
push2ap (H.Push secret _ sharer repo mbranch mbefore after early mlate) = do
encodeRouteLocal <- getEncodeRouteLocal push2ap (H.Push secret personNum repo mbranch mbefore after early mlate) = do
let shr = text2shr sharer secret' <- asksSite appHookSecret
rp = text2rp repo unless (secret == H.hookSecretText secret') $
commit2ap' = commit2ap shr rp error "Inavlid hook secret"
repoID <- do
ctx <- asksSite siteHashidsContext
case decodeInt64 ctx $ TE.encodeUtf8 repo of
Nothing -> error "Invalid repo keyhashid"
Just repoNum -> return $ toSqlKey repoNum
repoHash <- do
repoHash <- encodeKeyHashid repoID
unless (keyHashidText repoHash == repo) $
error "decode-encode repo hash returned a different value"
return repoHash
let commit2ap' = commit2ap repoHash
(commitsLast, commitsFirst) <- (commitsLast, commitsFirst) <-
runDB $ case mlate of runDB $ case mlate of
Nothing -> (,) <$> traverse commit2ap' early <*> pure Nothing Nothing -> (,) <$> traverse commit2ap' early <*> pure Nothing
Just (_omitted, late) -> Just (_omitted, late) ->
(,) <$> traverse commit2ap' late (,) <$> traverse commit2ap' late
<*> (Just <$> traverse commit2ap' early) <*> (Just <$> traverse commit2ap' early)
encodeRouteLocal <- getEncodeRouteLocal
encodeRouteHome <- getEncodeRouteHome
let pusherID = toSqlKey personNum
pusherHash <- encodeKeyHashid pusherID
let luRepo = encodeRouteLocal $ RepoR repoHash
return return
( Push ( AP.Push
{ pushCommitsLast = commitsLast { AP.pushCommitsLast = commitsLast
, pushCommitsFirst = commitsFirst , AP.pushCommitsFirst = commitsFirst
, pushCommitsTotal = , AP.pushCommitsTotal =
case mlate of case mlate of
Nothing -> length early Nothing -> length early
Just (omitted, late) -> Just (omitted, late) ->
length early + omitted + length late length early + omitted + length late
, pushTarget = , AP.pushTarget =
encodeRouteLocal $
case mbranch of case mbranch of
Nothing -> RepoR shr rp Nothing -> Left luRepo
Just b -> RepoBranchR shr rp b Just b ->
, pushContext = encodeRouteLocal $ RepoR shr rp Right $ AP.Branch b ("refs/heads/" <> b) luRepo
, pushHashBefore = mbefore , AP.pushAttrib = encodeRouteHome $ PersonR pusherHash
, pushHashAfter = after , AP.pushHashBefore = mbefore
, AP.pushHashAfter = after
} }
, shr , repoID
, rp , repoHash
) )
where where
commit2ap shr rp (H.Commit (wauthor, wtime) mcommitted hash title desc) = do commit2ap repoHash (H.Commit (wauthor, wtime) mcommitted hash title desc) = do
encodeRouteLocal <- getEncodeRouteLocal encodeRouteLocal <- getEncodeRouteLocal
encodeRouteHome <- getEncodeRouteHome encodeRouteHome <- getEncodeRouteHome
author <- authorByEmail wauthor author <- authorByEmail wauthor
mcommitter <- traverse (authorByEmail . fst) mcommitted mcommitter <- traverse (authorByEmail . fst) mcommitted
return Commit return AP.Commit
{ commitId = encodeRouteLocal $ RepoCommitR shr rp hash { AP.commitId = encodeRouteLocal $ RepoCommitR repoHash hash
, commitRepository = encodeRouteLocal $ RepoR shr rp , AP.commitRepository = encodeRouteLocal $ RepoR repoHash
, commitAuthor = second (encodeRouteHome . SharerR) author , AP.commitAuthor = second (encodeRouteHome . PersonR) author
, commitCommitter = , AP.commitCommitter =
second (encodeRouteHome . SharerR) <$> mcommitter second (encodeRouteHome . PersonR) <$> mcommitter
, commitTitle = title , AP.commitTitle = title
, commitHash = Hash $ encodeUtf8 hash , AP.commitHash = AP.Hash $ TE.encodeUtf8 hash
, commitDescription = , AP.commitDescription =
if T.null desc if T.null desc
then Nothing then Nothing
else Just desc else Just desc
, commitWritten = wtime , AP.commitWritten = wtime
, commitCommitted = snd <$> mcommitted , AP.commitCommitted = snd <$> mcommitted
} }
where where
authorByEmail (H.Author name email) = do authorByEmail (H.Author name email) = do
mperson <- getValBy $ UniquePersonEmail email mperson <- getKeyBy $ UniquePersonEmail email
case mperson of case mperson of
Nothing -> return $ Left $ Author name email Nothing -> return $ Left $ AP.Author name email
Just person -> Just person -> Right <$> encodeKeyHashid person
Right . sharerIdent <$> getJust (personIdent person)
makeSummary push pushAP repoHash repoName (Entity personID person, actor) = do
let mbranch = H.pushBranch push
total = AP.pushCommitsTotal pushAP
lasts = AP.pushCommitsLast pushAP
rest firsts = total - length firsts - length lasts
hashText (AP.Hash b) = decodeUtf8 b
commitW c =
[hamlet|
<a href=@{RepoCommitR repoHash $ hashText $ AP.commitHash c}>
#{AP.commitTitle c}
|]
personHash <- encodeKeyHashid personID
withUrlRenderer
[hamlet|
<p>
<a href=@{PersonR personHash}>
#{actorName actor} ~#{username2text $ personUsername person}
\ pushed #{total} #
\ #{commitsText mbranch total} to repo #
<a href=@{RepoR repoHash}>^#{keyHashidText repoHash} #{repoName}</a>^{branchText repoHash mbranch}:
<ul>
$maybe firsts <- AP.pushCommitsFirst pushAP
$forall c <- firsts
<li>^{commitW c}
<li>#{rest firsts}
$forall c <- lasts
<li>^{commitW c}
|]
commitsText :: Maybe a -> Int -> Text commitsText :: Maybe a -> Int -> Text
commitsText Nothing n = commitsText Nothing n =
if n > 1 if n > 1
@ -652,14 +712,14 @@ postPostReceiveR = return "Temporarily disabled, no Push activity published"
if n > 1 if n > 1
then "commits" then "commits"
else "commit" else "commit"
--branchText :: ShrIdent -> RpIdent -> Maybe Text -> HtmlUrl (Route App) --branchText :: ShrIdent -> RpIdent -> Maybe Text -> HtmlUrl (Route App)
branchText _ _ Nothing = const mempty branchText _ Nothing = const mempty
branchText shr rp (Just branch) = branchText r (Just branch) =
[hamlet| [hamlet|
, branch # , branch #
<a href=@{RepoBranchR shr rp branch}>#{branch} <a href=@{RepoBranchCommitsR r branch}>#{branch}
|] |]
-}
postRepoLinkR :: KeyHashid Repo -> KeyHashid Loom -> Handler Html postRepoLinkR :: KeyHashid Repo -> KeyHashid Loom -> Handler Html
postRepoLinkR repoHash loomHash = do postRepoLinkR repoHash loomHash = do

View file

@ -1610,8 +1610,8 @@ data Push u = Push
{ pushCommitsLast :: NonEmpty (Commit u) { pushCommitsLast :: NonEmpty (Commit u)
, pushCommitsFirst :: Maybe (NonEmpty (Commit u)) , pushCommitsFirst :: Maybe (NonEmpty (Commit u))
, pushCommitsTotal :: Int , pushCommitsTotal :: Int
, pushTarget :: LocalURI , pushTarget :: Either LocalURI (Branch u)
, pushContext :: LocalURI , pushAttrib :: ObjURI u
, pushHashBefore :: Maybe Text , pushHashBefore :: Maybe Text
, pushHashAfter :: Maybe Text , pushHashAfter :: Maybe Text
} }
@ -1623,21 +1623,28 @@ parsePush a o = do
<$> (traverse (withAuthorityT a . parseObject) =<< c .: "items" <|> c .: "orderedItems") <$> (traverse (withAuthorityT a . parseObject) =<< c .: "items" <|> c .: "orderedItems")
<*> (traverse (traverse $ withAuthorityT a . parseObject) =<< c .:? "earlyItems") <*> (traverse (traverse $ withAuthorityT a . parseObject) =<< c .:? "earlyItems")
<*> c .: "totalItems" <*> c .: "totalItems"
<*> withAuthorityO a (o .: "target") <*> (do target <- o .:+ "target"
<*> withAuthorityO a (o .: "context") let (h, target') =
case target of
Left (ObjURI h lu) -> (h, Left lu)
Right (Doc h branch) -> (h, Right branch)
unless (h == a) $ fail "target host != Push host"
return target'
)
<*> o .: "attributedTo"
<*> o .:? "hashBefore" <*> o .:? "hashBefore"
<*> o .:? "hashAfter" <*> o .:? "hashAfter"
encodePush :: UriMode u => Authority u -> Push u -> Series encodePush :: UriMode u => Authority u -> Push u -> Series
encodePush a (Push lateCommits earlyCommits total target context before after) encodePush a (Push lateCommits earlyCommits total target attrib before after)
= "object" `pair` pairs = "object" `pair` pairs
( "type" .= ("OrderedCollection" :: Text) ( "type" .= ("OrderedCollection" :: Text)
<> pair "orderedItems" (objectList lateCommits) <> pair "orderedItems" (objectList lateCommits)
<> maybe mempty (pair "earlyItems" . objectList) earlyCommits <> maybe mempty (pair "earlyItems" . objectList) earlyCommits
<> "totalItems" .= total <> "totalItems" .= total
) )
<> "target" .= ObjURI a target <> "target" .=+ bimap (ObjURI a) (Doc a) target
<> "context" .= ObjURI a context <> "attributedTo" .= attrib
<> "hashBefore" .=? before <> "hashBefore" .=? before
<> "hashAfter" .=? after <> "hashAfter" .=? after
where where