{-# LANGUAGE ForeignFunctionInterface #-}
module Darcs.Repository.Job
( RepoJob(..)
, IsPrimV1(..)
, withRepoLock
, withOldRepoLock
, withRepoLockCanFail
, withRepository
, withRepositoryLocation
, checkRepoIsNoRebase
, withUMaskFlag
) where
import Darcs.Prelude
import Darcs.Patch.Apply ( ApplyState )
import Darcs.Patch.V1 ( RepoPatchV1 )
import Darcs.Patch.V2 ( RepoPatchV2 )
import Darcs.Patch.V3 ( RepoPatchV3 )
import qualified Darcs.Patch.V1.Prim as V1 ( Prim(..) )
import qualified Darcs.Patch.V2.Prim as V2 ( Prim(..) )
import Darcs.Patch ( PrimOf )
import Darcs.Patch.Prim.V1 ( Prim )
import Darcs.Patch.RepoPatch ( RepoPatch )
import Darcs.Patch.RepoType
( RepoType(..), SRepoType(..), IsRepoType
, RebaseType(..), SRebaseType(..), IsRebaseType
, singletonRepoType
)
import Darcs.Repository.Flags
( UseCache(..), UpdatePending(..), DryRun(..), UMask (..)
)
import Darcs.Repository.Format
( RepoProperty( Darcs2
, Darcs3
, RebaseInProgress
, RebaseInProgress_2_16
, HashedInventory
)
, formatHas
, writeProblem
)
import Darcs.Repository.Identify ( identifyRepository )
import Darcs.Repository.Hashed( revertRepositoryChanges )
import Darcs.Repository.InternalTypes
( Repository
, repoFormat
, repoLocation
, unsafeCoerceRepoType
, unsafeCoercePatchType
)
import Darcs.Repository.Paths ( lockPath )
import Darcs.Repository.Rebase
( startRebaseJob
, rebaseJob
, maybeDisplaySuspendedStatus
, checkOldStyleRebaseStatus
)
import Darcs.Util.Lock ( withLock, withLockCanFail )
import Darcs.Util.Progress ( debugMessage )
import Control.Monad ( when )
import Control.Exception ( bracket_, finally )
import Data.List ( intercalate )
import Foreign.C.String ( CString, withCString )
import Foreign.C.Error ( throwErrno )
import Foreign.C.Types ( CInt(..) )
import Darcs.Util.Tree ( Tree )
withUMaskFlag :: UMask -> IO a -> IO a
withUMaskFlag NoUMask = id
withUMaskFlag (YesUMask umask) = withUMask umask
foreign import ccall unsafe "umask.h set_umask" set_umask
:: CString -> IO CInt
foreign import ccall unsafe "umask.h reset_umask" reset_umask
:: CInt -> IO CInt
withUMask :: String
-> IO a
-> IO a
withUMask umask job =
do rc <- withCString umask set_umask
when (rc < 0) (throwErrno "Couldn't set umask")
bracket_
(return ())
(reset_umask rc)
job
data RepoJob a
=
RepoJob (forall rt p wR wU . (IsRepoType rt, RepoPatch p, ApplyState p ~ Tree)
=> Repository rt p wR wU wR -> IO a)
| V1Job (forall wR wU . Repository ('RepoType 'NoRebase) (RepoPatchV1 V1.Prim) wR wU wR -> IO a)
| V2Job (forall rt wR wU . IsRepoType rt => Repository rt (RepoPatchV2 V2.Prim) wR wU wR -> IO a)
| PrimV1Job (forall rt p wR wU . (IsRepoType rt, RepoPatch p, ApplyState p ~ Tree, IsPrimV1 (PrimOf p))
=> Repository rt p wR wU wR -> IO a)
| RebaseAwareJob (forall rt p wR wU . (IsRepoType rt, RepoPatch p, ApplyState p ~ Tree) => Repository rt p wR wU wR -> IO a)
| RebaseJob (forall p wR wU . (RepoPatch p, ApplyState p ~ Tree) => Repository ('RepoType 'IsRebase) p wR wU wR -> IO a)
| OldRebaseJob (forall p wR wU . (RepoPatch p, ApplyState p ~ Tree) => Repository ('RepoType 'IsRebase) p wR wU wR -> IO a)
| StartRebaseJob (forall p wR wU . (RepoPatch p, ApplyState p ~ Tree) => Repository ('RepoType 'IsRebase) p wR wU wR -> IO a)
onRepoJob :: RepoJob a
-> (forall rt p wR wU . (RepoPatch p, ApplyState p ~ Tree) => (Repository rt p wR wU wR -> IO a) -> Repository rt p wR wU wR -> IO a)
-> RepoJob a
onRepoJob (RepoJob job) f = RepoJob (f job)
onRepoJob (V1Job job) f = V1Job (f job)
onRepoJob (V2Job job) f = V2Job (f job)
onRepoJob (PrimV1Job job) f = PrimV1Job (f job)
onRepoJob (RebaseAwareJob job) f = RebaseAwareJob (f job)
onRepoJob (RebaseJob job) f = RebaseJob (f job)
onRepoJob (OldRebaseJob job) f = OldRebaseJob (f job)
onRepoJob (StartRebaseJob job) f = StartRebaseJob (f job)
withRepository :: UseCache -> RepoJob a -> IO a
withRepository useCache = withRepositoryLocation useCache "."
data RepoPatchType p where
RepoV1 :: RepoPatchType (RepoPatchV1 V1.Prim)
RepoV2 :: RepoPatchType (RepoPatchV2 V2.Prim)
RepoV3 :: RepoPatchType (RepoPatchV3 V2.Prim)
data IsTree p where
IsTree :: (ApplyState p ~ Tree) => IsTree p
checkTree :: RepoPatchType p -> IsTree p
checkTree RepoV1 = IsTree
checkTree RepoV2 = IsTree
checkTree RepoV3 = IsTree
class ApplyState p ~ Tree => IsPrimV1 p where
toPrimV1 :: p wX wY -> Prim wX wY
instance IsPrimV1 V1.Prim where
toPrimV1 = V1.unPrim
instance IsPrimV1 V2.Prim where
toPrimV1 = V2.unPrim
data UsesPrimV1 p where
UsesPrimV1 :: (ApplyState p ~ Tree, IsPrimV1 (PrimOf p)) => UsesPrimV1 p
checkPrimV1 :: RepoPatchType p -> UsesPrimV1 p
checkPrimV1 RepoV1 = UsesPrimV1
checkPrimV1 RepoV2 = UsesPrimV1
checkPrimV1 RepoV3 = UsesPrimV1
withRepositoryLocation :: UseCache -> String -> RepoJob a -> IO a
withRepositoryLocation useCache url repojob = do
repo <- identifyRepository useCache url
let
rf = repoFormat repo
startRebase =
case repojob of
StartRebaseJob {} -> True
_ -> False
runJob1
:: IsRebaseType rebaseType
=> SRebaseType rebaseType -> Repository rtDummy pDummy wR wU wR -> RepoJob a -> IO a
runJob1 isRebase =
if formatHas Darcs3 rf
then runJob RepoV3 (SRepoType isRebase)
else
if formatHas Darcs2 rf
then runJob RepoV2 (SRepoType isRebase)
else runJob RepoV1 (SRepoType isRebase)
runJob2 :: Repository rtDummy pDummy wR wU wR -> RepoJob a -> IO a
runJob2 =
if startRebase ||
formatHas RebaseInProgress rf || formatHas RebaseInProgress_2_16 rf
then runJob1 SIsRebase
else runJob1 SNoRebase
runJob2 repo repojob
runJob
:: forall rt p rtDummy pDummy wR wU a
. (IsRepoType rt, RepoPatch p)
=> RepoPatchType p
-> SRepoType rt
-> Repository rtDummy pDummy wR wU wR
-> RepoJob a
-> IO a
runJob patchType (SRepoType isRebase) repo repojob = do
let
therepo = unsafeCoercePatchType (unsafeCoerceRepoType repo) :: Repository rt p wR wU wR
patchTypeString :: String
patchTypeString =
case patchType of
RepoV3 -> "darcs-3"
RepoV2 -> "darcs-2"
RepoV1 -> "darcs-1"
repoAttributes :: [String]
repoAttributes =
case isRebase of
SIsRebase -> ["rebase"]
SNoRebase -> []
repoAttributesString :: String
repoAttributesString =
case repoAttributes of
[] -> ""
_ -> " " ++ intercalate "+" repoAttributes
debugMessage $ "Identified " ++ patchTypeString ++ repoAttributesString ++
" repo: " ++ repoLocation repo
case repojob of
RepoJob job ->
case checkTree patchType of
IsTree -> do
checkOldStyleRebaseStatus isRebase therepo
job therepo
`finally`
maybeDisplaySuspendedStatus isRebase therepo
PrimV1Job job ->
case checkPrimV1 patchType of
UsesPrimV1 -> do
checkOldStyleRebaseStatus isRebase therepo
job therepo
`finally`
maybeDisplaySuspendedStatus isRebase therepo
V2Job job ->
case (patchType, isRebase) of
(RepoV2, SNoRebase) -> job therepo
(RepoV2, SIsRebase) ->
fail "This command is not supported while a rebase is in progress."
(RepoV1, _ ) ->
fail $ "This repository contains darcs v1 patches,"
++ " but the command requires darcs v2 patches."
(RepoV3, _ ) ->
fail $ "This repository contains darcs v3 patches,"
++ " but the command requires darcs v2 patches."
V1Job job ->
case (patchType, isRebase) of
(RepoV1, SNoRebase) -> job therepo
(RepoV1, SIsRebase) ->
fail "This command is not supported while a rebase is in progress."
(RepoV2, _ ) ->
fail $ "This repository contains darcs v2 patches,"
++ " but the command requires darcs v1 patches."
(RepoV3, _ ) ->
fail $ "This repository contains darcs v3 patches,"
++ " but the command requires darcs v1 patches."
RebaseAwareJob job ->
case (checkTree patchType, isRebase) of
(IsTree, SNoRebase) -> job therepo
(IsTree, SIsRebase) -> do
checkOldStyleRebaseStatus isRebase therepo
rebaseJob job therepo
RebaseJob job ->
case (checkTree patchType, isRebase) of
(_ , SNoRebase) -> fail "No rebase in progress. Try 'darcs rebase suspend' first."
(IsTree, SIsRebase) -> do
checkOldStyleRebaseStatus isRebase therepo
rebaseJob job therepo
OldRebaseJob job ->
case (checkTree patchType, isRebase) of
(_ , SNoRebase) -> fail "No rebase in progress."
(IsTree, SIsRebase) -> do
job therepo
`finally`
maybeDisplaySuspendedStatus isRebase therepo
StartRebaseJob job ->
case (checkTree patchType, isRebase) of
(_ , SNoRebase) -> error "impossible case"
(IsTree, SIsRebase) -> do
startRebaseJob job therepo
withRepoLock :: DryRun -> UseCache -> UpdatePending -> UMask -> RepoJob a -> IO a
withRepoLock YesDryRun useCache _ _ repojob =
withRepository useCache $ onRepoJob repojob $ \job repository -> job repository
withRepoLock NoDryRun useCache upe um repojob =
withLock lockPath $
withRepository useCache $ onRepoJob repojob $ \job repository -> do
maybe (return ()) fail $ writeProblem (repoFormat repository)
withUMaskFlag um $ revertRepositoryChanges repository upe >>= job
withOldRepoLock :: RepoJob a -> IO a
withOldRepoLock repojob =
withRepository NoUseCache $ onRepoJob repojob $ \job repository ->
withLock lockPath $ job repository
withRepoLockCanFail :: UseCache -> RepoJob () -> IO ()
withRepoLockCanFail useCache repojob = do
eitherDone <-
withLockCanFail lockPath $
withRepository useCache $ onRepoJob repojob $ \job repository -> do
let rf = repoFormat repository
if formatHas HashedInventory rf then do
maybe (return ()) fail $ writeProblem rf
job repository
else
debugMessage
"Not doing the job because this is an old-fashioned repository."
case eitherDone of
Left _ -> debugMessage "Lock could not be obtained, not doing the job."
Right _ -> return ()
checkRepoIsNoRebase :: forall rt p wR wU wT. IsRepoType rt
=> Repository rt p wR wU wT
-> Maybe (Repository ('RepoType 'NoRebase) p wR wU wT)
checkRepoIsNoRebase repo =
case singletonRepoType :: SRepoType rt of
SRepoType SNoRebase -> Just repo
SRepoType SIsRebase -> Nothing