From ed9968c3196c8aa0ba94a0365bad60d50d8a803b Mon Sep 17 00:00:00 2001 From: Rebecca Turner Date: Fri, 6 Dec 2024 15:56:16 -0800 Subject: [PATCH] Do not use shallow clones for short revisions When the `tag` of a `source-repository-stanza` is not a full hash, we cannot do a shallow clone and fetch it separately. Therefore, in those cases, we fall back to doing a full clone. Fixes #10605 Includes a regression test by @9999years. Co-authored-by: Rebecca Turner --- cabal-install/src/Distribution/Client/VCS.hs | 32 +++++++++++++++++-- .../SourceRepositoryPackageShallow/cabal.out | 10 ++++++ .../cabal.project | 8 +++++ .../cabal.test.hs | 4 +++ .../puppy.cabal | 9 ++++++ .../src/Puppy.hs | 1 + 6 files changed, 62 insertions(+), 2 deletions(-) create mode 100644 cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.out create mode 100644 cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.project create mode 100644 cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.test.hs create mode 100644 cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/puppy.cabal create mode 100644 cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/src/Puppy.hs diff --git a/cabal-install/src/Distribution/Client/VCS.hs b/cabal-install/src/Distribution/Client/VCS.hs index 98b8251d9c1..afd6dfea43c 100644 --- a/cabal-install/src/Distribution/Client/VCS.hs +++ b/cabal-install/src/Distribution/Client/VCS.hs @@ -549,7 +549,10 @@ vcsGit = -- If we want a particular branch or tag, fetch it. ref <- case srpBranch `mplus` srpTag of Nothing -> pure "HEAD" - Just ref -> do + -- `doShallow` controls whether we use a shallow clone. + -- If the clone is shallow, make sure to fetch specified revisions + -- before using them. + Just ref | doShallow -> do -- /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ -- /!\ MULTIPLE HOURS HAVE BEEN LOST HERE!! /!\ -- /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ /!\ @@ -582,6 +585,9 @@ vcsGit = -- for now. Option 1 is possible but seems to have little benefit. git localDir ("fetch" : verboseArg ++ ["origin", ref]) pure "FETCH_HEAD" + Just ref + | otherwise -> + pure ref -- Then, reset to the appropriate ref. git localDir $ @@ -608,8 +614,30 @@ vcsGit = { progInvokeCwd = Just cwd } + -- Beware: if the user supplied revision for the source repository + -- package is /not/ a full hash, then we cannot fetch it, which means + -- we cannot do a shallow clone (--depth=1). + -- See #10605. + doShallow + | Nothing <- srpTag = + -- No tag, OK for shallow + True + -- full hashes are exactly 40 characters + | Just tg <- srpTag + , length tg == 40 + , all (`elem` (['0' .. '9'] ++ ['a' .. 'f'] ++ ['A' .. 'F'])) tg = + True + | otherwise = + False + + depthIs1 + | doShallow = ["--depth=1"] + | otherwise = [] + cloneArgs = - ["clone", "--depth=1", "--no-checkout", loc, localDir] + ["clone"] + ++ depthIs1 + ++ [ "--no-checkout", loc, localDir] ++ case peer of Nothing -> [] Just peerLocalDir -> ["--reference", peerLocalDir] diff --git a/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.out b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.out new file mode 100644 index 00000000000..b5579a8defc --- /dev/null +++ b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.out @@ -0,0 +1,10 @@ +# cabal v2-build +Configuration is affected by the following files: +- cabal.project +Resolving dependencies... +Build profile: -w ghc- -O1 +In order, the following will be built: + - puppy-1.0 (lib) (first run) +Configuring library for puppy-1.0... +Preprocessing library for puppy-1.0... +Building library for puppy-1.0... diff --git a/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.project b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.project new file mode 100644 index 00000000000..275c4fc4bb6 --- /dev/null +++ b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.project @@ -0,0 +1,8 @@ +packages: . + +-- Regression for https://github.com/haskell/cabal/issues/10605 +-- This is `my-lib` 0.9. +source-repository-package + type: git + location: https://github.com/9999years/cabal-testsuite-my-lib.git + tag: 9a0af0aa diff --git a/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.test.hs b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.test.hs new file mode 100644 index 00000000000..d8024a44c0a --- /dev/null +++ b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/cabal.test.hs @@ -0,0 +1,4 @@ +import Test.Cabal.Prelude + +main = cabalTest $ do + cabal "v2-build" [] diff --git a/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/puppy.cabal b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/puppy.cabal new file mode 100644 index 00000000000..d4dbe921e99 --- /dev/null +++ b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/puppy.cabal @@ -0,0 +1,9 @@ +cabal-version: 3.0 +name: puppy +version: 1.0 + +library + default-language: Haskell2010 + hs-source-dirs: src + build-depends: base + exposed-modules: Puppy diff --git a/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/src/Puppy.hs b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/src/Puppy.hs new file mode 100644 index 00000000000..47209451314 --- /dev/null +++ b/cabal-testsuite/PackageTests/SourceRepositoryPackageShallow/src/Puppy.hs @@ -0,0 +1 @@ +module Puppy () where