1
1
mirror of https://github.com/anoma/juvix.git synced 2025-01-07 16:22:14 +03:00

Do not call git fetch on git dependency if ref already exists in clone (#2379)

During path resolution, `git fetch` is called on every git dependency
clone. It's desirable to reduce the number of `git fetch` calls because
it makes a network call.

This PR changes the git dependency resolution to only call `git fetch`
if the existing clone does not already contain the specified ref.
This commit is contained in:
Paul Cadman 2023-09-21 18:27:36 +01:00 committed by GitHub
parent 6c08cbb8d8
commit 135a170b21
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 91 additions and 15 deletions

View File

@ -121,11 +121,8 @@ getDependencyPath i = case i ^. packageDepdendencyInfoDependency of
r <- rootBuildDir <$> asks (^. envRoot) r <- rootBuildDir <$> asks (^. envRoot)
let cloneDir = r <//> relDependenciesDir <//> relDir (T.unpack (g ^. gitDependencyName)) let cloneDir = r <//> relDependenciesDir <//> relDir (T.unpack (g ^. gitDependencyName))
cloneArgs = CloneArgs {_cloneArgsCloneDir = cloneDir, _cloneArgsRepoUrl = g ^. gitDependencyUrl} cloneArgs = CloneArgs {_cloneArgsCloneDir = cloneDir, _cloneArgsRepoUrl = g ^. gitDependencyUrl}
errorHandler' = errorHandler cloneDir scoped @CloneArgs @Git cloneArgs $
scoped @CloneArgs @Git cloneArgs $ do fetchOnNoSuchRefAndRetry (errorHandler cloneDir) (`checkout` (g ^. gitDependencyRef)) >> return cloneDir
fetch errorHandler'
checkout errorHandler' (g ^. gitDependencyRef)
return cloneDir
where where
errorHandler :: Path Abs Dir -> GitError -> Sem (Git ': r) a errorHandler :: Path Abs Dir -> GitError -> Sem (Git ': r) a
errorHandler p c = errorHandler p c =

View File

@ -22,8 +22,20 @@ data GitError
data Git m a where data Git m a where
Fetch :: (GitError -> m ()) -> Git m () Fetch :: (GitError -> m ()) -> Git m ()
Checkout :: (GitError -> m ()) -> GitRef -> Git m () Checkout :: (GitError -> m ()) -> GitRef -> Git m ()
HeadRef :: (GitError -> m GitRef) -> Git m GitRef NormalizeRef :: (GitError -> m GitRef) -> GitRef -> Git m GitRef
makeSem ''Git makeSem ''Git
type GitClone = Scoped CloneArgs Git type GitClone = Scoped CloneArgs Git
headRef :: (Member Git r) => (GitError -> Sem r GitRef) -> Sem r GitRef
headRef h = normalizeRef h "HEAD"
-- | If an action fails because a ref does not exist in the clone, first do a fetch and then retry.
fetchOnNoSuchRefAndRetry :: forall r a. (Member Git r) => (GitError -> Sem r a) -> ((GitError -> Sem r a) -> Sem r a) -> Sem r a
fetchOnNoSuchRefAndRetry handler action = action retryHandler
where
retryHandler :: GitError -> Sem r a
retryHandler = \case
NoSuchRef _ -> fetch (void . handler) >> action handler
e -> handler e

View File

@ -53,9 +53,13 @@ checkValidGitClone = void gitHeadRef
isValidGitClone :: (Members '[Process, Reader CloneEnv] r) => Sem r Bool isValidGitClone :: (Members '[Process, Reader CloneEnv] r) => Sem r Bool
isValidGitClone = isRight <$> runError @GitProcessError checkValidGitClone isValidGitClone = isRight <$> runError @GitProcessError checkValidGitClone
-- | Return the normal form of the passed git reference
gitNormalizeRef :: forall r. (Members '[Process, Error GitProcessError, Reader CloneEnv] r) => Text -> Sem r Text
gitNormalizeRef ref = T.strip <$> runGitCmdInDir' ["rev-parse", "--verify", ref <> "^{commit}"]
-- | Return the HEAD ref of the clone -- | Return the HEAD ref of the clone
gitHeadRef :: (Members '[Process, Error GitProcessError, Reader CloneEnv] r) => Sem r Text gitHeadRef :: (Members '[Process, Error GitProcessError, Reader CloneEnv] r) => Sem r Text
gitHeadRef = T.strip <$> runGitCmdInDir' ["rev-parse", "HEAD"] gitHeadRef = gitNormalizeRef "HEAD"
-- | Checkout the clone at a particular ref -- | Checkout the clone at a particular ref
gitCheckout :: (Members '[Process, Error GitProcessError, Reader CloneEnv] r) => Text -> Sem r () gitCheckout :: (Members '[Process, Error GitProcessError, Reader CloneEnv] r) => Text -> Sem r ()
@ -88,14 +92,11 @@ handleNotACloneError errorHandler eff = catch @GitProcessError eff $ \case
GitCmdError GitCmdErrorDetails {_gitCmdErrorDetailsExitCode = ExitFailure 128} -> runTSimple (return NotAClone) >>= bindTSimple errorHandler GitCmdError GitCmdErrorDetails {_gitCmdErrorDetailsExitCode = ExitFailure 128} -> runTSimple (return NotAClone) >>= bindTSimple errorHandler
e -> throw e e -> throw e
handleNoSuchRefError :: (Member (Error GitProcessError) r, Monad m) => (GitError -> m x) -> GitRef -> Tactical e m r x -> Tactical e m r x handleNormalizeRefError :: (Member (Error GitProcessError) r, Monad m) => (GitError -> m x) -> GitRef -> Tactical e m r x -> Tactical e m r x
handleNoSuchRefError errorHandler ref eff = catch @GitProcessError eff $ \case handleNormalizeRefError errorHandler ref eff = catch @GitProcessError eff $ \case
GitCmdError GitCmdErrorDetails {_gitCmdErrorDetailsExitCode = ExitFailure 1} -> runTSimple (return (NoSuchRef ref)) >>= bindTSimple errorHandler GitCmdError GitCmdErrorDetails {_gitCmdErrorDetailsExitCode = ExitFailure 128} -> runTSimple (return (NoSuchRef ref)) >>= bindTSimple errorHandler
e -> throw e e -> throw e
handleCheckoutError :: (Member (Error GitProcessError) r, Monad m) => (GitError -> m x) -> GitRef -> Tactical e m r x -> Tactical e m r x
handleCheckoutError errorHandler ref eff = handleNoSuchRefError errorHandler ref (handleNotACloneError errorHandler eff)
runGitProcess :: runGitProcess ::
forall r a. forall r a.
(Members '[Log, Files, Process, Error GitProcessError, Internet] r) => (Members '[Log, Files, Process, Error GitProcessError, Internet] r) =>
@ -110,9 +111,11 @@ runGitProcess = interpretScopedH allocator handler
handler :: forall q r0 x. Path Abs Dir -> Git (Sem r0) x -> Tactical Git (Sem r0) (Opaque q ': r) x handler :: forall q r0 x. Path Abs Dir -> Git (Sem r0) x -> Tactical Git (Sem r0) (Opaque q ': r) x
handler p eff = case eff of handler p eff = case eff of
HeadRef errorHandler -> handleNotACloneError errorHandler (runReader env gitHeadRef >>= pureT)
Fetch errorHandler -> handleNotACloneError errorHandler (runReader env gitFetch >>= pureT) Fetch errorHandler -> handleNotACloneError errorHandler (runReader env gitFetch >>= pureT)
Checkout errorHandler ref -> handleCheckoutError errorHandler ref (runReader env (gitCheckout ref) >>= pureT) Checkout errorHandler ref -> do
void (handleNormalizeRefError errorHandler ref (runReader env (void (gitNormalizeRef ref)) >>= pureT))
handleNotACloneError errorHandler (runReader env (gitCheckout ref) >>= pureT)
NormalizeRef errorHandler ref -> handleNormalizeRefError errorHandler ref (runReader env (gitNormalizeRef ref) >>= pureT)
where where
env :: CloneEnv env :: CloneEnv
env = CloneEnv {_cloneEnvDir = p} env = CloneEnv {_cloneEnvDir = p}

View File

@ -561,3 +561,67 @@ tests:
stdout: stdout:
contains: duplicate contains: duplicate
exit-status: 1 exit-status: 1
- name: git-dependencies-no-fetch-if-ref-exists-in-clone
command:
shell:
- bash
script: |
temp=$(mktemp -d)
trap 'rm -rf -- "$temp"' EXIT
# create dependency
mkdir $temp/dep
cd $temp/dep
git init
cat <<-EOF > HelloDep.juvix
module HelloDep;
import Stdlib.Prelude open;
main : IO := printStringLn "Hello from dep";
EOF
touch juvix.yaml
git add -A
git commit -m "commit1"
dep1hash=$(git rev-parse HEAD)
# create project that uses dependency
mkdir $temp/base
cd $temp/base
cat <<-EOF > juvix.yaml
name: HelloWorld
main: HelloWorld.juvix
dependencies:
- .juvix-build/stdlib
- git:
url: $temp/dep
name: dep1
ref: $dep1hash
version: 0.1.0
EOF
cat <<-EOF > HelloWorld.juvix
-- HelloWorld.juvix
module HelloWorld;
import Stdlib.Prelude open;
import HelloDep;
main : IO := HelloDep.main;
EOF
# compile project
juvix compile HelloWorld.juvix
# delete the dependency to check that it's not required
rm -rf $temp/dep
# compile project
juvix compile HelloWorld.juvix
stderr: ""
stdout:
contains: cloning
exit-status: 0