mirror of
https://code.sup39.dev/repos/Wqawg
synced 2025-01-10 10:36:47 +09:00
c6c41b485c
I used this chance to make some name changes, add some utils, tweak some imports, remove more `setTitle`s and so on. I also made person, repo, key and project creation forms verify CI-uniqueness.
147 lines
6.4 KiB
Haskell
147 lines
6.4 KiB
Haskell
{- This file is part of Vervis.
|
|
-
|
|
- Written in 2016 by fr33domlover <fr33domlover@riseup.net>.
|
|
-
|
|
- ♡ Copying is an act of love. Please copy, reuse and share.
|
|
-
|
|
- The author(s) have dedicated all copyright and related and neighboring
|
|
- rights to this software to the public domain worldwide. This software is
|
|
- distributed without any warranty.
|
|
-
|
|
- You should have received a copy of the CC0 Public Domain Dedication along
|
|
- with this software. If not, see
|
|
- <http://creativecommons.org/publicdomain/zero/1.0/>.
|
|
-}
|
|
|
|
module Vervis.Handler.Git
|
|
( getGitRefDiscoverR
|
|
--, postGitUploadRequestR
|
|
)
|
|
where
|
|
|
|
import Prelude
|
|
|
|
import Control.Monad.IO.Class (liftIO)
|
|
import Data.Git.Harder (ObjId (..))
|
|
import Data.Git.Harder.Pack
|
|
import Data.Git.Repository (getCommit)
|
|
import Data.Git.Storage (isRepo, withRepo)
|
|
import Data.String (fromString)
|
|
import Data.Text (Text)
|
|
import Data.Text.Encoding (encodeUtf8)
|
|
import Network.Git.Get (parseService)
|
|
import Network.Git.Transport.HTTP.Fetch.RefDiscovery
|
|
import Network.Git.Transport.HTTP.Fetch.UploadRequest
|
|
import Yesod.Core.Handler
|
|
|
|
import Vervis.BinaryBody (decodeRequestBody)
|
|
import Vervis.Content
|
|
import Vervis.Foundation (Handler)
|
|
import Vervis.Model.Ident
|
|
import Vervis.Path (askRepoDir)
|
|
|
|
getGitRefDiscoverR :: ShrIdent -> RpIdent -> Handler GitRefDiscovery
|
|
getGitRefDiscoverR shar repo = do
|
|
path <- askRepoDir shar repo
|
|
let pathG = fromString path
|
|
seemsThere <- liftIO $ isRepo pathG
|
|
if seemsThere
|
|
then do
|
|
rq <- getRequest
|
|
case reqGetParams rq of
|
|
[("service", servT)] ->
|
|
case parseService $ encodeUtf8 servT of
|
|
Just serv -> do
|
|
rd <- liftIO $ withRepo pathG $
|
|
flip buildRefDiscover' serv
|
|
setHeader "Cache-Control" "no-cache"
|
|
return $ GitRefDiscovery rd
|
|
Nothing -> permissionDenied "Service not supported"
|
|
_ -> notFound
|
|
else notFound
|
|
|
|
{- This is commented out for now because it doesn't work. The 'collectObjIds'
|
|
- function file descriptor exhaustion. I don't know whether and how I can fix
|
|
- that. Maybe dive deep into what happens under the hood in 'hit', or make a
|
|
- fork of 'hit' which streams things using 'pipes' or 'conduit'. Or perhaps
|
|
- check how 'git' and 'libgit2' do these things without resource leaks. I
|
|
- don't know, I'm exhausted.
|
|
-
|
|
- Another option is to traverse objects using gitlib, via conduit. johnw told
|
|
- me on IRC he has a version of gitlib based on pipes. Either way I don't know
|
|
- if this thing will work on top of hit, because of how hit keeps open file
|
|
- descriptors.
|
|
-
|
|
- I could also try to change collectObjIds to be a function that streams raw
|
|
- objects, and keeps a hashmap of IDs to avoid reading an object twice.
|
|
-
|
|
- Will any of that help? I don't know.
|
|
-
|
|
-
|
|
-
|
|
postGitUploadRequestR :: Text -> Text -> Handler GitUploadPackResult
|
|
postGitUploadRequestR sharer repo = do
|
|
path <- askRepoDir sharer repo
|
|
let pathG = fromString path
|
|
seemsThere <- liftIO $ isRepo pathG
|
|
if seemsThere
|
|
then do
|
|
ereq <- decodeRequestBody getUploadRequest
|
|
case ereq of
|
|
Left _ -> invalidArgs ["UploadRequest"]
|
|
Right ur -> do
|
|
-- We need to handle /have/ lines and verify the /want/ed
|
|
-- refs appear in the ref discovery we sent. But we for now
|
|
-- ignore these things. Suppose the client didn't send an
|
|
-- /have/s, what's next? It seems we now need to build and
|
|
-- send a pack.
|
|
--
|
|
-- We just send a full pack with all the ancestors of the
|
|
-- wants.
|
|
--
|
|
-- IDEA: abstract away the HTTP request part by:
|
|
--
|
|
-- (1) Read the request body in chunks and use Get to read
|
|
-- (2) Use a Put to create the response, possibly send in
|
|
-- chunks, or instead first make LBS and then send?
|
|
|
|
-- TODO currently the code assumes all of these are commits
|
|
-- but they can also be tags (are there other options?)
|
|
let wants = urWants ur
|
|
lbs <- liftIO $ withRepo pathG $ \ git -> do
|
|
-- quick hack: in the case of a clone where the client has
|
|
-- no HAVEs, the minimal set is the entire ancestor tree of
|
|
-- the wants. So let's just collect all the ancestors.
|
|
let getC oid = (oid,) <$> getCommit git (unObjId oid)
|
|
wantsP <- traverse getC wants
|
|
let collect _ oid l = do
|
|
mo <- getObject git (unObjId oid) False
|
|
case mo of
|
|
Just (ObjCommit c) -> return ((oid,c):l, Just c)
|
|
_ -> error "non-commit parent"
|
|
pairs <- loadCommitsMulti git collect wantsP $ map (fmap Just) wantsP
|
|
oidset <- catch (collectObjIds git pairs) $ \ e -> do
|
|
print (e :: SomeException)
|
|
throwIO e
|
|
|
|
--for_ oidset $ \ oid -> do
|
|
-- obj <- getObject_ git (unObjId oid) True
|
|
-- putStrLn $ take 120 $ show obj
|
|
|
|
--let isCommit r = do
|
|
-- obj <- getObject_ git r False
|
|
-- case obj of
|
|
-- ObjCommit _ -> return True
|
|
-- _ -> return False
|
|
-- allCommits c = do
|
|
-- bools <- traverse isCommit $ commitParents c
|
|
-- return $ and bools
|
|
--bools <- for pairs $ \ (_oid, c) -> allCommits c
|
|
--let nNotC = length $ filter not bools
|
|
--putStrLn $ "Total commits: " ++ show (length pairs)
|
|
--putStrLn $ "Commits with non-commit parents: " ++ show nNotC
|
|
|
|
serializePack git oidset
|
|
return $ GitUploadPackResult $ "0008NAK\n" <> lbs
|
|
else notFound
|
|
-}
|