1
0
Fork 0
mirror of https://code.naskya.net/repos/ndqEd synced 2025-01-12 14:25:09 +09:00
vervis/src/Vervis/Handler/Repo.hs

220 lines
7.7 KiB
Haskell
Raw Normal View History

2016-02-27 14:41:36 +09:00
{- 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.Repo
( getReposR
, postReposR
, getRepoNewR
, getRepoR
, getRepoSourceR
, getRepoHeadChangesR
, getRepoChangesR
2016-02-27 14:41:36 +09:00
)
where
--TODO CONTINUE HERE
--
-- [/] maybe list project repos in personal overview too
-- [x] make repo list page
-- [x] add new repo creation link
-- [x] make new repo form
-- [x] write the git and mkdir parts that actually create the repo
2016-03-03 17:15:54 +09:00
-- [x] make repo view that shows a table of commits
2016-02-27 14:41:36 +09:00
import ClassyPrelude.Conduit hiding (last, unpack)
2016-03-03 17:15:54 +09:00
import Yesod hiding (Header, parseTime, (==.))
import Yesod.Auth
import Prelude (init, last, tail)
import Data.Git.Graph
import Data.Git.Harder
import Data.Git.Named (RefName (..))
2016-03-03 17:15:54 +09:00
import Data.Git.Ref (toHex)
import Data.Git.Repository
import Data.Git.Storage (withRepo)
import Data.Git.Storage.Object (Object (..))
import Data.Git.Types (Blob (..), Commit (..), Person (..), entName)
import Data.Graph.Inductive.Graph (noNodes)
import Data.Graph.Inductive.Query.Topsort
import Data.List (inits)
2016-03-03 17:15:54 +09:00
import Data.Text (unpack)
import Data.Text.Encoding (decodeUtf8With)
import Data.Text.Encoding.Error (lenientDecode)
2016-02-27 14:41:36 +09:00
import Database.Esqueleto
2016-03-03 17:15:54 +09:00
import Data.Hourglass (timeConvert)
2016-02-27 14:41:36 +09:00
import System.Directory (createDirectoryIfMissing)
2016-03-03 17:15:54 +09:00
import System.Hourglass (dateCurrent)
import qualified Data.DList as D
import qualified Data.Set as S (member)
import qualified Data.Text.Lazy.Encoding as L (decodeUtf8With)
2016-03-03 17:15:54 +09:00
import Data.ByteString.Char8.Local (takeLine)
import Data.Git.Local
import Text.FilePath.Local (breakExt)
2016-02-27 14:41:36 +09:00
import Vervis.Form.Repo
2016-03-03 17:15:54 +09:00
import Vervis.Foundation
import Vervis.GitOld (timeAgo')
2016-03-03 17:15:54 +09:00
import Vervis.Path
import Vervis.MediaType (chooseMediaType)
2016-03-03 17:15:54 +09:00
import Vervis.Model
import Vervis.Model.Repo
import Vervis.Readme
import Vervis.Render
2016-03-03 17:15:54 +09:00
import Vervis.Settings
import Vervis.SourceTree
2016-04-12 23:44:43 +09:00
import Vervis.Style
import Vervis.Widget.Repo
2016-02-27 14:41:36 +09:00
import qualified Darcs.Local as D (createRepo)
import qualified Data.ByteString.Lazy as BL (ByteString)
import qualified Data.Git.Local as G (createRepo)
import qualified Vervis.Darcs as D (readSourceView)
import qualified Vervis.Git as G (readSourceView, readChangesView)
getReposR :: Text -> Handler Html
getReposR user = do
repos <- runDB $ select $ from $ \ (sharer, repo) -> do
2016-02-27 14:41:36 +09:00
where_ $
sharer ^. SharerIdent ==. val user &&.
sharer ^. SharerId ==. repo ^. RepoSharer
2016-02-27 14:41:36 +09:00
orderBy [asc $ repo ^. RepoIdent]
return $ repo ^. RepoIdent
defaultLayout $ do
setTitle $ toHtml $ intercalate " > "
["Vervis", "People", user, "Repos"]
$(widgetFile "repo/repos")
2016-02-27 14:41:36 +09:00
postReposR :: Text -> Handler Html
postReposR user = do
2016-02-27 14:41:36 +09:00
Entity _pid person <- requireAuth
let sid = personIdent person
((result, widget), enctype) <- runFormPost $ newRepoForm sid
2016-02-27 14:41:36 +09:00
case result of
FormSuccess repo -> do
parent <- askSharerDir user
liftIO $ do
createDirectoryIfMissing True parent
let repoName = unpack $ repoIdent repo
case repoVcs repo of
VCSDarcs -> D.createRepo parent repoName
VCSGit -> G.createRepo parent repoName
runDB $ insert_ repo
setMessage "Repo added."
redirect $ ReposR user
2016-02-27 14:41:36 +09:00
FormMissing -> do
setMessage "Field(s) missing"
defaultLayout $(widgetFile "repo/repo-new")
FormFailure _l -> do
setMessage "Repo creation failed, see errors below"
defaultLayout $(widgetFile "repo/repo-new")
2016-02-27 14:41:36 +09:00
getRepoNewR :: Text -> Handler Html
getRepoNewR user = do
2016-02-27 14:41:36 +09:00
Entity _pid person <- requireAuth
let sid = personIdent person
((_result, widget), enctype) <- runFormPost $ newRepoForm sid
2016-02-27 14:41:36 +09:00
defaultLayout $ do
setTitle $ toHtml $ mconcat ["Vervis > People > ", user, " > New Repo"]
$(widgetFile "repo/repo-new")
2016-02-27 14:41:36 +09:00
selectRepo :: Text -> Text -> AppDB Repo
selectRepo shar repo = do
Entity sid _s <- getBy404 $ UniqueSharerIdent shar
Entity _rid r <- getBy404 $ UniqueRepo repo sid
return r
getRepoR :: Text -> Text -> Handler Html
getRepoR shar repo = do
repository <- runDB $ selectRepo shar repo
case repoVcs repository of
VCSDarcs -> getDarcsRepoSource repository shar repo []
VCSGit ->
getGitRepoSource
repository shar repo (repoMainBranch repository) []
getDarcsRepoSource :: Repo -> Text -> Text -> [Text] -> Handler Html
getDarcsRepoSource repository user repo dir = do
path <- askRepoDir user repo
msv <- liftIO $ D.readSourceView path dir
case msv of
Nothing -> notFound
Just sv -> do
let parent = if null dir then [] else init dir
dirs = zip parent (tail $ inits parent)
defaultLayout $ do
setTitle $ toHtml $ intercalate " > "
["Vervis", "People", user, "Repos", repo]
$(widgetFile "repo/source-darcs")
getGitRepoSource :: Repo -> Text -> Text -> Text -> [Text] -> Handler Html
getGitRepoSource repository user repo ref dir = do
path <- askRepoDir user repo
(branches, tags, msv) <- liftIO $ G.readSourceView path ref dir
case msv of
Nothing -> notFound
Just sv -> do
let parent = if null dir then [] else init dir
dirs = zip parent (tail $ inits parent)
defaultLayout $ do
setTitle $ toHtml $ intercalate " > " $
["Vervis", "People", user, "Repos", repo]
$(widgetFile "repo/source-git")
getRepoSourceR :: Text -> Text -> [Text] -> Handler Html
getRepoSourceR shar repo refdir = do
repository <- runDB $ selectRepo shar repo
case repoVcs repository of
VCSDarcs -> getDarcsRepoSource repository shar repo refdir
VCSGit -> case refdir of
[] -> notFound
(ref:dir) -> getGitRepoSource repository shar repo ref dir
getDarcsRepoHeadChanges :: Text -> Text -> Handler Html
getDarcsRepoHeadChanges shar repo = notFound
getGitRepoHeadChanges :: Repo -> Text -> Text -> Handler Html
getGitRepoHeadChanges repository shar repo =
getGitRepoChanges shar repo $ repoMainBranch repository
getRepoHeadChangesR :: Text -> Text -> Handler Html
getRepoHeadChangesR user repo = do
repository <- runDB $ selectRepo user repo
case repoVcs repository of
VCSDarcs -> getDarcsRepoHeadChanges user repo
VCSGit -> getGitRepoHeadChanges repository user repo
getDarcsRepoChanges :: Text -> Text -> Text -> Handler Html
getDarcsRepoChanges shar repo tag = notFound
getGitRepoChanges :: Text -> Text -> Text -> Handler Html
getGitRepoChanges shar repo ref = do
path <- askRepoDir shar repo
(branches, tags, mentries) <- liftIO $ G.readChangesView path ref
case mentries of
Nothing -> notFound
Just entries ->
let refSelect = refSelectW shar repo branches tags
changes = changesW entries
in defaultLayout $(widgetFile "repo/changes-git")
getRepoChangesR :: Text -> Text -> Text -> Handler Html
getRepoChangesR shar repo ref = do
repository <- runDB $ selectRepo shar repo
case repoVcs repository of
VCSDarcs -> getDarcsRepoChanges shar repo ref
VCSGit -> getGitRepoChanges shar repo ref