1
0
Fork 0
mirror of https://code.naskya.net/repos/ndqEd synced 2025-01-25 20:27:49 +09:00

Provide git log in ActivityPub format

Currently it's a paged Collection where the items are merely URIs. This could
be changed to have actual Commit objects as items; for that we need to examine
the whole thing with the LogEntry type and the Patch type and have an
AP-friendly log item representation, but without commit diffs.
This commit is contained in:
fr33domlover 2019-09-01 14:19:14 +00:00
parent 7b26d5d918
commit 6ffc2c9872
2 changed files with 112 additions and 51 deletions

View file

@ -22,6 +22,7 @@ module Vervis.Handler.Repo.Git
) )
where where
import Control.Monad
import Control.Monad.IO.Class (liftIO) import Control.Monad.IO.Class (liftIO)
import Data.Git.Graph import Data.Git.Graph
import Data.Git.Harder import Data.Git.Harder
@ -67,10 +68,12 @@ import qualified Web.ActivityPub as AP
import Data.ByteString.Char8.Local (takeLine) import Data.ByteString.Char8.Local (takeLine)
import Data.Git.Local import Data.Git.Local
import Data.Paginate.Local
import Text.FilePath.Local (breakExt) import Text.FilePath.Local (breakExt)
import Vervis.ActivityPub import Vervis.ActivityPub
import Vervis.ChangeFeed (changeFeed) import Vervis.ChangeFeed (changeFeed)
import Vervis.Changes
import Vervis.Form.Repo import Vervis.Form.Repo
import Vervis.Foundation import Vervis.Foundation
import Vervis.Path import Vervis.Path
@ -134,19 +137,57 @@ getGitRepoChanges :: ShrIdent -> RpIdent -> Text -> Handler TypedContent
getGitRepoChanges shar repo ref = do getGitRepoChanges shar repo ref = do
path <- askRepoDir shar repo path <- askRepoDir shar repo
(branches, tags) <- liftIO $ G.listRefs path (branches, tags) <- liftIO $ G.listRefs path
if ref `S.member` branches || ref `S.member` tags unless (ref `S.member` branches || ref `S.member` tags)
then do notFound
(_, _, entries, navModel) <- getPageAndNavTop $ let here = RepoChangesR shar repo ref
\ o l -> liftIO $ G.readChangesView path ref o l encodeRouteLocal <- getEncodeRouteLocal
encodeRouteHome <- getEncodeRouteHome
encodeRoutePageLocal <- getEncodeRoutePageLocal
let pageUrl = encodeRoutePageLocal here
getChanges o l = liftIO $ G.readChangesView path ref o l
mpage <- getPageAndNavMaybe getChanges
case mpage of
Nothing -> do
(total, pages, _, _) <- getPageAndNavTop getChanges
let collection = Collection
{ collectionId = encodeRouteLocal here
, collectionType = CollectionTypeOrdered
, collectionTotalItems = Just total
, collectionCurrent = Nothing
, collectionFirst = Just $ pageUrl 1
, collectionLast = Just $ pageUrl pages
, collectionItems = [] :: [Text]
}
provideHtmlAndAP collection $ redirectFirstPage here
Just (_total, pages, items, navModel) ->
let current = nmCurrent navModel
page = CollectionPage
{ collectionPageId = pageUrl current
, collectionPageType = CollectionPageTypeOrdered
, collectionPageTotalItems = Nothing
, collectionPageCurrent = Just $ pageUrl current
, collectionPageFirst = Just $ pageUrl 1
, collectionPageLast = Just $ pageUrl pages
, collectionPagePartOf = encodeRouteLocal here
, collectionPagePrev =
if current > 1
then Just $ pageUrl $ current - 1
else Nothing
, collectionPageNext =
if current < pages
then Just $ pageUrl $ current + 1
else Nothing
, collectionPageStartIndex = Nothing
, collectionPageItems =
map (encodeRouteHome . RepoPatchR shar repo . leHash)
items
}
feed = changeFeed shar repo (Just ref) VCSGit items
in provideHtmlFeedAndAP page feed $
let refSelect = refSelectW shar repo branches tags let refSelect = refSelectW shar repo branches tags
changes = changesW shar repo entries changes = changesW shar repo items
pageNav = navWidget navModel pageNav = navWidget navModel
feed = changeFeed shar repo (Just ref) VCSGit entries in $(widgetFile "repo/changes-git")
selectRep $ do
provideRep $ defaultLayout $(widgetFile "repo/changes-git")
provideRep $ atomFeed feed
provideRep $ rssFeed feed
else notFound
getGitPatch :: ShrIdent -> RpIdent -> Text -> Handler TypedContent getGitPatch :: ShrIdent -> RpIdent -> Text -> Handler TypedContent
getGitPatch shr rp ref = do getGitPatch shr rp ref = do

View file

@ -23,11 +23,13 @@ module Yesod.ActivityPub
, provideHtmlAndAP , provideHtmlAndAP
, provideHtmlAndAP' , provideHtmlAndAP'
, provideHtmlAndAP'' , provideHtmlAndAP''
, provideHtmlFeedAndAP
) )
where where
import Control.Exception import Control.Exception
import Control.Monad.Logger.CallStack import Control.Monad.Logger.CallStack
import Control.Monad.Trans.Writer
import Data.Aeson import Data.Aeson
import Data.Aeson.Encode.Pretty import Data.Aeson.Encode.Pretty
import Data.ByteString (ByteString) import Data.ByteString (ByteString)
@ -35,10 +37,14 @@ import Data.Foldable
import Data.Function import Data.Function
import Data.List import Data.List
import Data.List.NonEmpty (NonEmpty) import Data.List.NonEmpty (NonEmpty)
import Data.Semigroup
import Data.Text (Text) import Data.Text (Text)
import Network.HTTP.Client import Network.HTTP.Client
import Network.HTTP.Types.Header import Network.HTTP.Types.Header
import Yesod.AtomFeed
import Yesod.Core hiding (logError, logDebug) import Yesod.Core hiding (logError, logDebug)
import Yesod.Feed
import Yesod.RssFeed
import qualified Data.ByteString.Lazy as BL import qualified Data.ByteString.Lazy as BL
import qualified Data.HashMap.Strict as M import qualified Data.HashMap.Strict as M
@ -192,12 +198,17 @@ provideHtmlAndAP object widget = do
host <- getsYesod siteInstanceHost host <- getsYesod siteInstanceHost
provideHtmlAndAP' host object widget provideHtmlAndAP' host object widget
provideHtmlAndAP' provideHtmlAndAP_
:: (YesodActivityPub site, SiteFedURIMode site ~ u, ActivityPub a) :: Yesod site
=> Authority u -> a u -> WidgetFor site () -> HandlerFor site TypedContent => (a -> Writer (Endo [ProvidedRep (HandlerFor site)]) ())
provideHtmlAndAP' host object widget = selectRep $ do -> (a -> WidgetFor site ())
let doc = Doc host object -> (a -> WidgetFor site ())
provideAP $ pure doc -> a
-> WidgetFor site ()
-> Maybe (Feed (Route site))
-> HandlerFor site TypedContent
provideHtmlAndAP_ provide renderSky renderHl doc widget mfeed = selectRep $ do
provide doc
provideRep $ do provideRep $ do
mval <- lookupGetParam "prettyjson" mval <- lookupGetParam "prettyjson"
defaultLayout $ defaultLayout $
@ -210,8 +221,8 @@ provideHtmlAndAP' host object widget = selectRep $ do
Just "sky" -> True Just "sky" -> True
Just _ -> error "Invalid highlight style" Just _ -> error "Invalid highlight style"
if sky if sky
then renderPrettyJSONSkylighting doc then renderSky doc
else renderPrettyJSON doc else renderHl doc
mroute <- getCurrentRoute mroute <- getCurrentRoute
for_ mroute $ \ route -> do for_ mroute $ \ route -> do
params <- reqGetParams <$> getRequest params <- reqGetParams <$> getRequest
@ -236,39 +247,48 @@ provideHtmlAndAP' host object widget = selectRep $ do
<a href=@?{(route, params')}> <a href=@?{(route, params')}>
[See JSON] [See JSON]
|] |]
for_ mfeed $ \ feed -> do
provideRep $ atomFeed feed
provideRep $ rssFeed feed
where where
delete' t = deleteBy ((==) `on` fst) (t, "") delete' t = deleteBy ((==) `on` fst) (t, "")
provideHtmlAndAP'
:: (YesodActivityPub site, SiteFedURIMode site ~ u, ActivityPub a)
=> Authority u -> a u -> WidgetFor site () -> HandlerFor site TypedContent
provideHtmlAndAP' host object widget =
provideHtmlAndAP_
(provideAP . pure)
renderPrettyJSONSkylighting
renderPrettyJSON
(Doc host object)
widget
Nothing
provideHtmlAndAP'' provideHtmlAndAP''
:: Yesod site :: Yesod site
=> PersistJSON a -> WidgetFor site () -> HandlerFor site TypedContent => PersistJSON a -> WidgetFor site () -> HandlerFor site TypedContent
provideHtmlAndAP'' body widget = selectRep $ do provideHtmlAndAP'' body widget =
provideAP' $ pure $ persistJSONBytes body provideHtmlAndAP_
provideRep $ do (provideAP' . pure . persistJSONBytes)
mval <- lookupGetParam "prettyjson" (renderPrettyJSONSkylighting' . encodePretty . persistJSONObject)
defaultLayout $ (renderPrettyJSON' . encodePretty . persistJSONObject)
case mval of body
Just "true" -> do
mhl <- lookupGetParam "highlight"
let sky = case mhl of
Nothing -> True
Just "hl2" -> False
Just "sky" -> True
Just _ -> error "Invalid highlight style"
pretty = encodePretty $ persistJSONObject body
if sky
then renderPrettyJSONSkylighting' pretty
else renderPrettyJSON' pretty
_ -> do
widget widget
mroute <- getCurrentRoute Nothing
for_ mroute $ \ route -> do
params <- reqGetParams <$> getRequest provideHtmlFeedAndAP
let pj = ("prettyjson", "true") :: (YesodActivityPub site, SiteFedURIMode site ~ u, ActivityPub a)
hl = ("highlight", "sky") => a u
params' = pj : hl : params -> Feed (Route site)
[whamlet| -> WidgetFor site ()
<div> -> HandlerFor site TypedContent
<a href=@?{(route, params')}> provideHtmlFeedAndAP object feed widget = do
[See JSON] host <- getsYesod siteInstanceHost
|] provideHtmlAndAP_
(provideAP . pure)
renderPrettyJSONSkylighting
renderPrettyJSON
(Doc host object)
widget
(Just feed)