Skip to content

Commit

Permalink
Make findProjectRoot aware of broken files
Browse files Browse the repository at this point in the history
  • Loading branch information
albertodvp committed Jul 8, 2024
1 parent 5d293c9 commit 17b083c
Show file tree
Hide file tree
Showing 4 changed files with 70 additions and 31 deletions.
76 changes: 49 additions & 27 deletions cabal-install/src/Distribution/Client/ProjectConfig.hs
Original file line number Diff line number Diff line change
Expand Up @@ -532,6 +532,8 @@ resolveBuildTimeSettings
| otherwise = False

---------------------------------------------
-- Reading and writing project config files
--

-- | Get @ProjectRootUsability@ of a given file
getProjectRootUsability :: FilePath -> IO ProjectRootUsability
Expand All @@ -543,7 +545,7 @@ getProjectRootUsability filePath = do
let isUsableAciton =
handle @IOException
-- NOTE: if any IOException is raised, we assume the file does not exist.
-- That is what happen when we call @pathIsSymbolicLink@ on an @FilePath@
-- That is what happen when we call @pathIsSymbolicLink@ on an @FilePath@ that does not exist.
(const $ pure False)
((||) <$> pathIsSymbolicLink filePath <*> doesPathExist filePath)
isUnusable <- isUsableAciton
Expand Down Expand Up @@ -574,27 +576,40 @@ findProjectRoot verbosity mprojectDir mprojectFile = do
"Specifying an absolute path to the project file is deprecated."
<> " Use --project-dir to set the project's directory."

doesFileExist file >>= \case
False -> left (BadProjectRootExplicitFile file)
True -> uncurry projectRoot =<< first dropTrailingPathSeparator . splitFileName <$> canonicalizePath file
getProjectRootUsability file >>= \case
ProjectRootUsabilityPresentAndUsable ->
uncurry projectRoot
=<< first dropTrailingPathSeparator . splitFileName <$> canonicalizePath file
ProjectRootUsabilityNotPresent ->
left (BadProjectRootExplicitFileNotFound file)
ProjectRootUsabilityPresentAndUnusable ->
left (BadProjectRootFileBroken file)
| otherwise -> probeProjectRoot mprojectFile
Just dir ->
doesDirectoryExist dir >>= \case
False -> left (BadProjectRootDir dir)
False -> left (BadProjectRootDirNotFound dir)
True -> do
projectDir <- canonicalizePath dir

case mprojectFile of
Nothing -> pure $ Right (ProjectRootExplicit projectDir defaultProjectFile)
Just projectFile
| isAbsolute projectFile ->
doesFileExist projectFile >>= \case
False -> left (BadProjectRootAbsoluteFile projectFile)
True -> Right . ProjectRootExplicitAbsolute dir <$> canonicalizePath projectFile
getProjectRootUsability projectFile >>= \case
ProjectRootUsabilityNotPresent ->
left (BadProjectRootAbsoluteFileNotFound projectFile)
ProjectRootUsabilityPresentAndUsable ->
Right . ProjectRootExplicitAbsolute dir <$> canonicalizePath projectFile
ProjectRootUsabilityPresentAndUnusable ->
left (BadProjectRootFileBroken projectFile)
| otherwise ->
doesFileExist (projectDir </> projectFile) >>= \case
False -> left (BadProjectRootDirFile dir projectFile)
True -> projectRoot projectDir projectFile
getProjectRootUsability (projectDir </> projectFile) >>= \case
ProjectRootUsabilityNotPresent ->
left (BadProjectRootDirFileNotFound dir projectFile)
ProjectRootUsabilityPresentAndUsable ->
projectRoot projectDir projectFile
ProjectRootUsabilityPresentAndUnusable ->
left (BadProjectRootFileBroken projectFile)
where
left = pure . Left

Expand All @@ -619,23 +634,28 @@ probeProjectRoot mprojectFile = do
go dir | isDrive dir || dir == homedir =
case mprojectFile of
Nothing -> return (Right (ProjectRootImplicit startdir))
Just file -> return (Left (BadProjectRootExplicitFile file))
Just file -> return (Left (BadProjectRootExplicitFileNotFound file))
go dir = do
exists <- doesFileExist (dir </> projectFileName)
if exists
then return (Right (ProjectRootExplicit dir projectFileName))
else go (takeDirectory dir)
getProjectRootUsability (dir </> projectFileName) >>= \case
ProjectRootUsabilityNotPresent ->
go (takeDirectory dir)
ProjectRootUsabilityPresentAndUsable ->
return (Right $ ProjectRootExplicit dir projectFileName)
ProjectRootUsabilityPresentAndUnusable ->
return (Left $ BadProjectRootFileBroken projectFileName)

-- | Errors returned by 'findProjectRoot'.
data BadProjectRoot
= BadProjectRootExplicitFile FilePath
| BadProjectRootDir FilePath
| BadProjectRootAbsoluteFile FilePath
| BadProjectRootDirFile FilePath FilePath
= BadProjectRootExplicitFileNotFound FilePath
| BadProjectRootDirNotFound FilePath
| BadProjectRootAbsoluteFileNotFound FilePath
| BadProjectRootDirFileNotFound FilePath FilePath
| BadProjectRootFileBroken FilePath

#if MIN_VERSION_base(4,8,0)
deriving (Show, Typeable)
deriving (Show, Typeable, Eq)
#else
deriving (Typeable)
deriving (Typeable, Eq)

instance Show BadProjectRoot where
show = renderBadProjectRoot
Expand All @@ -650,14 +670,16 @@ instance Exception BadProjectRoot

renderBadProjectRoot :: BadProjectRoot -> String
renderBadProjectRoot = \case
BadProjectRootExplicitFile projectFile ->
BadProjectRootExplicitFileNotFound projectFile ->
"The given project file '" ++ projectFile ++ "' does not exist."
BadProjectRootDir dir ->
BadProjectRootDirNotFound dir ->
"The given project directory '" <> dir <> "' does not exist."
BadProjectRootAbsoluteFile file ->
BadProjectRootAbsoluteFileNotFound file ->
"The given project file '" <> file <> "' does not exist."
BadProjectRootDirFile dir file ->
"The given projectdirectory/file combination '" <> dir </> file <> "' does not exist."
BadProjectRootDirFileNotFound dir file ->
"The given project directory/file combination '" <> dir </> file <> "' does not exist."
BadProjectRootFileBroken file ->
"The given project file '" <> file <> "' is broken. Is it a broken symbolic link?"

-- | State of the project file, encodes if the file can be used
data ProjectRootUsability
Expand Down
23 changes: 21 additions & 2 deletions cabal-install/tests/UnitTests/Distribution/Client/ProjectConfig.hs
Original file line number Diff line number Diff line change
Expand Up @@ -133,8 +133,8 @@ testGetProjectRootUsability =
test name fileName expectedState =
testCase name $
withCurrentDirectory dir $
getProjectRootUsability fileName >>=
(@?= expectedState)
getProjectRootUsability fileName
>>= (@?= expectedState)

testFindProjectRoot :: TestTree
testFindProjectRoot =
Expand All @@ -146,6 +146,10 @@ testFindProjectRoot =
, test "explicit file in lib" (cd libDir) Nothing (Just file) (succeeds dir file)
, test "other file" (cd dir) Nothing (Just fileOther) (succeeds dir fileOther)
, test "other file in lib" (cd libDir) Nothing (Just fileOther) (succeeds dir fileOther)
, test "symbolic link" (cd dir) Nothing (Just fileSymlink) (succeeds dir fileSymlink)
, test "symbolic link in lib" (cd libDir) Nothing (Just fileSymlink) (succeeds dir fileSymlink)
, test "broken symbolic link" (cd dir) Nothing (Just fileSymlinkBroken) (failsWith $ BadProjectRootFileBroken fileSymlinkBroken)
, test "broken symbolic link in lib" (cd libDir) Nothing (Just fileSymlinkBroken) (failsWith $ BadProjectRootFileBroken fileSymlinkBroken)
, -- Deprecated use-case
test "absolute file" Nothing Nothing (Just absFile) (succeeds dir file)
, test "nested file" (cd dir) Nothing (Just nixFile) (succeeds dir nixFile)
Expand All @@ -167,6 +171,9 @@ testFindProjectRoot =
nixFile = "nix" </> file
nixOther = nixFile <.> "other"

fileSymlink = file <.> "symlink"
fileSymlinkBroken = fileSymlink <.> "broken"

missing path = Just (path <.> "does_not_exist")

test name wrap projectDir projectFile validate =
Expand Down Expand Up @@ -195,6 +202,18 @@ testFindProjectRoot =
Left _ -> pure ()
Right x -> assertFailure $ "Expected an error, but found " <> show x

failsWith expectedError result = case result of
Left actualError ->
if actualError == expectedError
then pure ()
else
assertFailure $
"Expected an error "
<> show expectedError
<> ", but found "
<> show actualError
Right x -> assertFailure $ "Expected an error, but found " <> show x

fixturesDir :: FilePath
fixturesDir =
unsafePerformIO $
Expand Down

This file was deleted.

This file was deleted.

0 comments on commit 17b083c

Please sign in to comment.