{- git-annex test suite framework - - Copyright 2010-2023 Joey Hess - - Licensed under the GNU AGPL version 3 or higher. -} {-# LANGUAGE CPP #-} module Test.Framework where import Test.Tasty import Test.Tasty.Runners import Test.Tasty.HUnit import Test.Tasty.Options import Test.Tasty.Ingredients.Rerun import Test.Tasty.Ingredients.ConsoleReporter import qualified Test.Tasty.Patterns.Types as TP import Options.Applicative.Types import Control.Concurrent import Control.Concurrent.Async import Control.Concurrent.STM import System.Environment (getArgs) import System.Console.Concurrent import System.Console.ANSI import Data.Time.Clock import GHC.Conc import System.IO.Unsafe (unsafePerformIO) import System.PosixCompat.Files (isSymbolicLink, isRegularFile, fileMode, unionFileModes, ownerWriteMode) import Common import Types.Test import Types.Concurrency import qualified Utility.RawFilePath as R import qualified Annex import qualified Annex.UUID import qualified Types.RepoVersion import qualified Backend import qualified Git.CurrentRepo import qualified Git.Construct import qualified Git.Types import qualified Git.Branch import qualified Git.Ref import qualified Types.KeySource import qualified Types.Backend import qualified Types import qualified Remote import qualified Key import qualified Types.Key import qualified Types.Messages import qualified Config import qualified Annex.WorkTree import qualified Annex.Link import qualified Annex.Path import qualified Annex.Action import qualified Annex.AdjustedBranch import qualified Annex.Init import qualified Utility.Process import qualified Utility.Process.Transcript import qualified Utility.Env import qualified Utility.Env.Set import qualified Utility.Exception import qualified Utility.ThreadScheduler import qualified Utility.Tmp.Dir import qualified Utility.Metered import qualified Utility.HumanTime import qualified Command.Uninit -- Run a process. The output and stderr is captured, and is only -- displayed if the process does not return the expected value. -- -- In debug mode, the output is allowed to pass through. -- So the output does not get checked in debug mode. testProcess :: String -> [String] -> Maybe [(String, String)] -> (Bool -> Bool) -> (String -> Bool) -> String -> Assertion testProcess command params environ expectedret expectedtranscript faildesc = void $ testProcess' command params environ expectedret expectedtranscript faildesc testProcess' :: String -> [String] -> Maybe [(String, String)] -> (Bool -> Bool) -> (String -> Bool) -> String -> IO String testProcess' command params environ expectedret expectedtranscript faildesc = do let p = (proc command params) { env = environ } debug <- testDebug . testOptions <$> getTestMode if debug then do ret <- withCreateProcess p $ \_ _ _ pid -> waitForProcess pid (expectedret (ret == ExitSuccess)) @? (faildesc ++ " failed with unexpected exit code") return "" else do (transcript, ret) <- Utility.Process.Transcript.processTranscript' p Nothing (expectedret ret) @? (faildesc ++ " failed with unexpected exit code (transcript follows)\n" ++ transcript) (expectedtranscript transcript) @? (faildesc ++ " failed with unexpected output (transcript follows)\n" ++ transcript) return transcript -- Run git. (Do not use to run git-annex as the one being tested -- may not be in path.) git :: String -> [String] -> String -> Assertion git command params = testProcess "git" (command:params) Nothing (== True) (const True) -- For when git is expected to fail. git_shouldfail :: String -> [String] -> String -> Assertion git_shouldfail command params = testProcess "git" (command:params) Nothing (== False) (const True) -- Run git-annex. git_annex :: String -> [String] -> String -> Assertion git_annex command params faildesc = git_annex' command params Nothing faildesc -- Runs git-annex with some environment. git_annex' :: String -> [String] -> Maybe [(String, String)] -> String -> Assertion git_annex' = git_annex'' (== True) (const True) -- For when git-annex is expected to fail. git_annex_shouldfail :: String -> [String] -> String -> Assertion git_annex_shouldfail command params faildesc = git_annex_shouldfail' command params Nothing faildesc git_annex_shouldfail' :: String -> [String] -> Maybe [(String, String)] -> String -> Assertion git_annex_shouldfail' = git_annex'' (== False) (const True) git_annex'' :: (Bool -> Bool) -> (String -> Bool) -> String -> [String] -> Maybe [(String, String)] -> String -> Assertion git_annex'' expectedret expectedtranscript command params environ faildesc = do pp <- Annex.Path.programPath debug <- testDebug . testOptions <$> getTestMode let params' = if debug then "--debug":params else params testProcess pp (command:params') environ expectedret expectedtranscript faildesc {- Runs git-annex and returns its standard output. -} git_annex_output :: String -> [String] -> IO String git_annex_output command params = do pp <- Annex.Path.programPath Utility.Process.readProcess pp (command:params) git_annex_expectoutput :: String -> [String] -> [String] -> Assertion git_annex_expectoutput command params expected = do got <- lines <$> git_annex_output command params got == expected @? ("unexpected value running " ++ command ++ " " ++ show params ++ " -- got: " ++ show got ++ " expected: " ++ show expected) -- Runs an action in the current annex. Note that shutdown actions -- are not run; this should only be used for actions that query state. annexeval :: Types.Annex a -> IO a annexeval a = do s <- Annex.new =<< Git.CurrentRepo.get Annex.eval s $ do Annex.setOutput Types.Messages.QuietOutput a `finally` Annex.Action.stopCoProcesses innewrepo :: IO () -> IO () innewrepo a = withgitrepo $ \r -> intopdir r a inmainrepo :: IO a -> IO a inmainrepo a = do d <- mainrepodir intopdir d a with_ssh_origin :: (Assertion -> Assertion) -> (Assertion -> Assertion) with_ssh_origin cloner a = cloner $ do let k = Git.Types.ConfigKey (encodeBS config) let v = Git.Types.ConfigValue (toRawFilePath "/dev/null") origindir <- absPath . Git.Types.fromConfigValue =<< annexeval (Config.getConfig k v) let originurl = "localhost:" ++ fromRawFilePath origindir git "config" [config, originurl] "git config failed" a where config = "remote.origin.url" intmpclonerepo :: Assertion -> Assertion intmpclonerepo a = withtmpclonerepo $ \r -> intopdir r a checkRepo :: Types.Annex a -> FilePath -> IO a checkRepo getval d = do s <- Annex.new =<< Git.Construct.fromPath (toRawFilePath d) Annex.eval s $ getval `finally` Annex.Action.stopCoProcesses intmpbareclonerepo :: Assertion -> Assertion intmpbareclonerepo a = withtmpclonerepo' (newCloneRepoConfig { bareClone = True } ) $ \r -> intopdir r a intmpsharedclonerepo :: Assertion -> Assertion intmpsharedclonerepo a = withtmpclonerepo' (newCloneRepoConfig { sharedClone = True } ) $ \r -> intopdir r a withtmpclonerepo :: (FilePath -> Assertion) -> Assertion withtmpclonerepo = withtmpclonerepo' newCloneRepoConfig withtmpclonerepo' :: CloneRepoConfig -> (FilePath -> Assertion) -> Assertion withtmpclonerepo' cfg a = do dir <- tmprepodir maindir <- mainrepodir clone <- clonerepo maindir dir cfg r <- tryNonAsync (a clone) case r of Right () -> return () Left e -> do whenM (keepFailuresOption . testOptions <$> getTestMode) $ putStrLn $ "** Preserving repo for failure analysis in " ++ clone throwM e disconnectOrigin :: Assertion disconnectOrigin = git "remote" ["rm", "origin"] "remote rm" withgitrepo :: (FilePath -> Assertion) -> Assertion withgitrepo a = do maindir <- mainrepodir bracket (setuprepo maindir) return a intopdir :: FilePath -> IO a -> IO a intopdir dir a = do topdir <- Utility.Env.getEnvDefault "TOPDIR" (error "TOPDIR not set") inpath (topdir ++ "/" ++ dir) a inpath :: FilePath -> IO a -> IO a inpath path a = do currdir <- getCurrentDirectory -- Assertion failures throw non-IO errors; catch -- any type of error and change back to currdir before -- rethrowing. r <- bracket_ (setCurrentDirectory path) (setCurrentDirectory currdir) (tryNonAsync a) case r of Right v -> return v Left e -> throwM e adjustedbranchsupported :: FilePath -> IO Bool adjustedbranchsupported repo = intopdir repo $ Annex.AdjustedBranch.isGitVersionSupported setuprepo :: FilePath -> IO FilePath setuprepo dir = do cleanup dir git "init" ["-q", dir] "git init" configrepo dir return dir data CloneRepoConfig = CloneRepoConfig { bareClone :: Bool , sharedClone :: Bool } newCloneRepoConfig :: CloneRepoConfig newCloneRepoConfig = CloneRepoConfig { bareClone = False , sharedClone = False } -- clones are always done as local clones; we cannot test ssh clones clonerepo :: FilePath -> FilePath -> CloneRepoConfig -> IO FilePath clonerepo old new cfg = do cleanup new let cloneparams = catMaybes [ Just "-q" , if bareClone cfg then Just "--bare" else Nothing , if sharedClone cfg then Just "--shared" else Nothing , Just old , Just new ] git "clone" cloneparams "git clone" configrepo new intopdir new $ do ver <- annexVersion <$> getTestMode git_annex "init" [ "-q" , new, "--version" , show (Types.RepoVersion.fromRepoVersion ver) ] "git annex init" unless (bareClone cfg) $ intopdir new $ setupTestMode return new configrepo :: FilePath -> IO () configrepo dir = intopdir dir $ do -- ensure git is set up to let commits happen git "config" ["user.name", "Test User"] "git config" git "config" ["user.email", "test@example.com"] "git config" -- avoid signed commits by test suite git "config" ["commit.gpgsign", "false"] "git config" -- tell git-annex to not annex the ingitfile git "config" ["annex.largefiles", "exclude=" ++ ingitfile] "git config annex.largefiles" -- set any additional git configs the user wants to test with gc <- testGitConfig . testOptions <$> getTestMode forM_ gc $ \case (Git.Types.ConfigKey k, Git.Types.ConfigValue v) -> git "config" [decodeBS k, decodeBS v] "git config from test options" (Git.Types.ConfigKey _, Git.Types.NoConfigValue) -> noop ensuredir :: FilePath -> IO () ensuredir d = do e <- doesDirectoryExist d unless e $ createDirectory d {- This is the only place in the test suite that can use setEnv. - Using it elsewhere can conflict with tasty's use of getEnv, which can - happen concurrently with a test case running, and would be a problem - since setEnv is not thread safe. This is run before tasty. -} setTestEnv :: IO a -> IO a setTestEnv a = Utility.Tmp.Dir.withTmpDir "testhome" $ \tmphome -> do tmphomeabs <- fromRawFilePath <$> absPath (toRawFilePath tmphome) {- Prevent global git configs from affecting the test suite. -} Utility.Env.Set.setEnv "HOME" tmphomeabs True Utility.Env.Set.setEnv "XDG_CONFIG_HOME" tmphomeabs True Utility.Env.Set.setEnv "GIT_CONFIG_NOSYSTEM" "1" True -- Ensure that the same git-annex binary that is running -- git-annex test is at the front of the PATH. p <- Utility.Env.getEnvDefault "PATH" "" pp <- Annex.Path.programPath Utility.Env.Set.setEnv "PATH" (takeDirectory pp ++ [searchPathSeparator] ++ p) True -- Avoid git complaining if it cannot determine the user's -- email address, or exploding if it doesn't know the user's name. Utility.Env.Set.setEnv "GIT_AUTHOR_EMAIL" "test@example.com" True Utility.Env.Set.setEnv "GIT_AUTHOR_NAME" "git-annex test" True Utility.Env.Set.setEnv "GIT_COMMITTER_EMAIL" "test@example.com" True Utility.Env.Set.setEnv "GIT_COMMITTER_NAME" "git-annex test" True -- force gpg into batch mode for the tests Utility.Env.Set.setEnv "GPG_BATCH" "1" True -- Make git and git-annex access ssh remotes on the local -- filesystem, without using ssh at all. Utility.Env.Set.setEnv "GIT_SSH_COMMAND" "git-annex test --fakessh --" True Utility.Env.Set.setEnv "GIT_ANNEX_USE_GIT_SSH" "1" True -- Record top directory. currdir <- getCurrentDirectory Utility.Env.Set.setEnv "TOPDIR" currdir True a removeDirectoryForCleanup :: FilePath -> IO () removeDirectoryForCleanup = removePathForcibly cleanup :: FilePath -> IO () cleanup dir = whenM (doesDirectoryExist dir) $ do Command.Uninit.prepareRemoveAnnexDir' dir -- This can fail if files in the directory are still open by a -- subprocess. void $ tryIO $ removeDirectoryForCleanup dir finalCleanup :: IO () finalCleanup = whenM (doesDirectoryExist tmpdir) $ do Command.Uninit.prepareRemoveAnnexDir' tmpdir catchIO (removeDirectoryForCleanup tmpdir) $ \e -> do print e putStrLn "sleeping 10 seconds and will retry directory cleanup" Utility.ThreadScheduler.threadDelaySeconds $ Utility.ThreadScheduler.Seconds 10 whenM (doesDirectoryExist tmpdir) $ removeDirectoryForCleanup tmpdir checklink :: FilePath -> Assertion checklink f = ifM (annexeval Config.crippledFileSystem) ( (isJust <$> annexeval (Annex.Link.getAnnexLinkTarget (toRawFilePath f))) @? f ++ " is not a (crippled) symlink" , do s <- R.getSymbolicLinkStatus (toRawFilePath f) isSymbolicLink s @? f ++ " is not a symlink" ) checkregularfile :: FilePath -> Assertion checkregularfile f = do s <- R.getSymbolicLinkStatus (toRawFilePath f) isRegularFile s @? f ++ " is not a normal file" return () checkdoesnotexist :: FilePath -> Assertion checkdoesnotexist f = (either (const True) (const False) <$> Utility.Exception.tryIO (R.getSymbolicLinkStatus (toRawFilePath f))) @? f ++ " exists unexpectedly" checkexists :: FilePath -> Assertion checkexists f = (either (const False) (const True) <$> Utility.Exception.tryIO (R.getSymbolicLinkStatus (toRawFilePath f))) @? f ++ " does not exist" checkcontent :: FilePath -> Assertion checkcontent f = do c <- Utility.Exception.catchDefaultIO "could not read file" $ readFile f assertEqual ("checkcontent " ++ f) (content f) c checkunwritable :: FilePath -> Assertion checkunwritable f = do -- Look at permissions bits rather than trying to write or -- using fileAccess because if run as root, any file can be -- modified despite permissions. s <- R.getFileStatus (toRawFilePath f) let mode = fileMode s when (mode == mode `unionFileModes` ownerWriteMode) $ assertFailure $ "able to modify annexed file's " ++ f ++ " content" checkwritable :: FilePath -> Assertion checkwritable f = do s <- R.getFileStatus (toRawFilePath f) let mode = fileMode s unless (mode == mode `unionFileModes` ownerWriteMode) $ assertFailure $ "unable to modify " ++ f checkdangling :: FilePath -> Assertion checkdangling f = ifM (annexeval Config.crippledFileSystem) ( return () -- probably no real symlinks to test , do r <- tryIO $ readFile f case r of Left _ -> return () -- expected; dangling link Right _ -> assertFailure $ f ++ " was not a dangling link as expected" ) checklocationlog :: FilePath -> Bool -> Assertion checklocationlog f expected = do thisuuid <- annexeval Annex.UUID.getUUID r <- annexeval $ Annex.WorkTree.lookupKey (toRawFilePath f) case r of Just k -> do uuids <- annexeval $ Remote.keyLocations k assertEqual ("bad content in location log for " ++ f ++ " key " ++ Key.serializeKey k ++ " uuid " ++ show thisuuid) expected (thisuuid `elem` uuids) _ -> assertFailure $ f ++ " failed to look up key" checkbackend :: FilePath -> Types.Backend -> Assertion checkbackend file expected = do b <- annexeval $ maybe (return Nothing) (Backend.getBackend file) =<< Annex.WorkTree.lookupKey (toRawFilePath file) assertEqual ("backend for " ++ file) (Just expected) b checkispointerfile :: FilePath -> Assertion checkispointerfile f = unlessM (isJust <$> Annex.Link.isPointerFile (toRawFilePath f)) $ assertFailure $ f ++ " is not a pointer file" inlocationlog :: FilePath -> Assertion inlocationlog f = checklocationlog f True notinlocationlog :: FilePath -> Assertion notinlocationlog f = checklocationlog f False runchecks :: [FilePath -> Assertion] -> FilePath -> Assertion runchecks [] _ = return () runchecks (a:as) f = do a f runchecks as f annexed_notpresent :: FilePath -> Assertion annexed_notpresent f = ifM (hasUnlockedFiles <$> getTestMode) ( annexed_notpresent_unlocked f , annexed_notpresent_locked f ) annexed_notpresent_locked :: FilePath -> Assertion annexed_notpresent_locked = runchecks [checklink, checkdangling, notinlocationlog] annexed_notpresent_unlocked :: FilePath -> Assertion annexed_notpresent_unlocked = runchecks [checkregularfile, checkispointerfile, notinlocationlog] annexed_present :: FilePath -> Assertion annexed_present f = ifM (hasUnlockedFiles <$> getTestMode) ( annexed_present_unlocked f , annexed_present_locked f ) annexed_present_locked :: FilePath -> Assertion annexed_present_locked f = ifM (annexeval Config.crippledFileSystem) ( runchecks [checklink, inlocationlog] f , runchecks [checklink, checkcontent, checkunwritable, inlocationlog] f ) annexed_present_unlocked :: FilePath -> Assertion annexed_present_unlocked = runchecks [checkregularfile, checkcontent, checkwritable, inlocationlog] annexed_present_imported :: FilePath -> Assertion annexed_present_imported f = ifM (annexeval Config.crippledFileSystem) ( annexed_present_unlocked f , ifM (adjustedUnlockedBranch <$> getTestMode) ( annexed_present_unlocked f , annexed_present_locked f ) ) annexed_notpresent_imported :: FilePath -> Assertion annexed_notpresent_imported f = ifM (annexeval Config.crippledFileSystem) ( annexed_notpresent_unlocked f , ifM (adjustedUnlockedBranch <$> getTestMode) ( annexed_notpresent_unlocked f , annexed_notpresent_locked f ) ) unannexed :: FilePath -> Assertion unannexed = runchecks [checkregularfile, checkcontent, checkwritable] -- Check that a file is unannexed, but also that what's recorded in git -- is not an annexed file. unannexed_in_git :: FilePath -> Assertion unannexed_in_git f = do unannexed f r <- annexeval $ Annex.WorkTree.lookupKey (toRawFilePath f) case r of Just _k -> assertFailure $ f ++ " is annexed in git" Nothing -> return () add_annex :: FilePath -> String -> Assertion add_annex f faildesc = ifM (unlockedFiles <$> getTestMode) ( git "add" [f] faildesc , git_annex "add" [f] faildesc ) data TestMode = TestMode { unlockedFiles :: Bool , adjustedUnlockedBranch :: Bool , annexVersion :: Types.RepoVersion.RepoVersion , testOptions :: TestOptions } testMode :: TestOptions -> Types.RepoVersion.RepoVersion -> TestMode testMode opts v = TestMode { unlockedFiles = False , adjustedUnlockedBranch = False , annexVersion = v , testOptions = opts } hasUnlockedFiles :: TestMode -> Bool hasUnlockedFiles m = unlockedFiles m || adjustedUnlockedBranch m withTestMode :: TestMode -> TestTree -> TestTree withTestMode testmode = withResource prepare release . const where prepare = setTestMode testmode release _ = noop {- The current test mode is stored here while a test is running. - - Only one test can be running at a time by a process; running a - test also involves chdir into a test repository. -} {-# NOINLINE currentTestMode #-} currentTestMode :: TMVar TestMode currentTestMode = unsafePerformIO newEmptyTMVarIO currentMainRepoDir :: TMVar FilePath currentMainRepoDir = unsafePerformIO newEmptyTMVarIO setTestMode :: TestMode -> IO () setTestMode testmode = do atomically $ do _ <- tryTakeTMVar currentTestMode putTMVar currentTestMode testmode setmainrepodir =<< newmainrepodir getTestMode :: IO TestMode getTestMode = atomically (tryReadTMVar currentTestMode) >>= \case Just tm -> return tm Nothing -> error "getTestMode without setTestMode" setupTestMode :: IO () setupTestMode = do testmode <- getTestMode when (adjustedUnlockedBranch testmode) $ do git "commit" ["--allow-empty", "-m", "empty"] "git commit failed" git_annex "adjust" ["--unlock"] "git annex adjust failed" tmpdir :: String tmpdir = ".t" setmainrepodir :: FilePath -> IO () setmainrepodir mrd = atomically $ do _ <- tryTakeTMVar currentMainRepoDir putTMVar currentMainRepoDir mrd mainrepodir :: IO FilePath mainrepodir = atomically (tryReadTMVar currentMainRepoDir) >>= \case Just tm -> return tm Nothing -> error "mainrepodir without setmainrepodir" newmainrepodir :: IO FilePath newmainrepodir = go (0 :: Int) where go n = do let d = "main" ++ show n ifM (doesDirectoryExist d) ( go $ n + 1 , do createDirectory d return d ) tmprepodir :: IO FilePath tmprepodir = go (0 :: Int) where go n = do let d = "tmprepo" ++ show n ifM (doesDirectoryExist d) ( go $ n + 1 , return d ) annexedfile :: String annexedfile = "foo" annexedfiledup :: String annexedfiledup = "foodup" wormannexedfile :: String wormannexedfile = "apple" sha1annexedfile :: String sha1annexedfile = "sha1foo" sha1annexedfiledup :: String sha1annexedfiledup = "sha1foodup" ingitfile :: String ingitfile = "bar.c" content :: FilePath -> String content f | f == annexedfile = "annexed file content" | f == ingitfile = "normal file content" | f == sha1annexedfile ="sha1 annexed file content" | f == annexedfiledup = content annexedfile | f == sha1annexedfiledup = content sha1annexedfile | f == wormannexedfile = "worm annexed file content" | "import" `isPrefixOf` f = "imported content" | otherwise = "unknown file " ++ f -- Writes new content to a file, and makes sure that it has a different -- mtime than it did before writecontent :: FilePath -> String -> IO () writecontent f c = go (10000000 :: Integer) where go ticsleft = do oldmtime <- catchMaybeIO $ getModificationTime f writeFile f c newmtime <- getModificationTime f if Just newmtime == oldmtime then do threadDelay 100000 let ticsleft' = ticsleft - 100000 if ticsleft' > 0 then go ticsleft' else do hPutStrLn stderr "file mtimes do not seem to be changing (tried for 10 seconds)" hFlush stderr return () else return () changecontent :: FilePath -> IO () changecontent f = writecontent f $ changedcontent f changedcontent :: FilePath -> String changedcontent f = content f ++ " (modified)" backendSHA1 :: Types.Backend backendSHA1 = backend_ "SHA1" backendSHA256 :: Types.Backend backendSHA256 = backend_ "SHA256" backendSHA256E :: Types.Backend backendSHA256E = backend_ "SHA256E" backendWORM :: Types.Backend backendWORM = backend_ "WORM" backend_ :: String -> Types.Backend backend_ = Backend.lookupBuiltinBackendVariety . Types.Key.parseKeyVariety . encodeBS getKey :: Types.Backend -> FilePath -> IO Types.Key getKey b f = case Types.Backend.genKey b of Just a -> annexeval $ a ks Utility.Metered.nullMeterUpdate Nothing -> error "internal" where ks = Types.KeySource.KeySource { Types.KeySource.keyFilename = toRawFilePath f , Types.KeySource.contentLocation = toRawFilePath f , Types.KeySource.inodeCache = Nothing } {- Get the name of the original branch, eg the current branch, or - if in an adjusted branch, the parent branch. -} origBranch :: Types.Annex String origBranch = maybe "foo" (Git.Types.fromRef . Git.Ref.base . Annex.AdjustedBranch.fromAdjustedBranch) <$> Annex.inRepo Git.Branch.current {- Set up repos as remotes of each other. -} pair :: FilePath -> FilePath -> Assertion pair r1 r2 = forM_ [r1, r2] $ \r -> intopdir r $ do when (r /= r1) $ git "remote" ["add", "r1", "../" ++ r1] "remote add" when (r /= r2) $ git "remote" ["add", "r2", "../" ++ r2] "remote add" {- Runs a query in the current repository, but first makes the repository - read-only. The write bit is added back at the end, so when possible, - include multiple tests within a single call for efficiency. -} readonly_query :: Assertion -> Assertion readonly_query = bracket_ (make_readonly ".") (make_writeable ".") {- Not guaranteed to do anything: - chmod may fail, or not be available, or the filesystem not support - permissions. -} make_readonly :: FilePath -> IO () make_readonly d = void $ Utility.Process.Transcript.processTranscript "chmod" ["-R", "-w", d] Nothing {- The write bit is added back for the current user, but not for other - users, even though make_readonly removes any other user's write bits. -} make_writeable :: FilePath -> IO () make_writeable d = void $ Utility.Process.Transcript.processTranscript "chmod" ["-R", "u+w", d] Nothing runFakeSsh :: [String] -> IO () runFakeSsh ("-n":ps) = runFakeSsh ps runFakeSsh (_host:cmd:[]) = withCreateProcess (shell cmd) $ \_ _ _ pid -> exitWith =<< waitForProcess pid runFakeSsh ps = error $ "fake ssh option parse error: " ++ show ps {- Tests each TestTree in parallel, and exits with success/failure. - - Tasty supports parallel tests, but this does not use it, because - many tests need to be run in test repos, and chdir would not be - thread safe. Instead, this starts one child process for each TestTree. - - An added benefit of using child processes is that any files they may - leave open are closed before finalCleanup is run at the end. This - prevents some failures to clean up after the test suite. -} parallelTestRunner :: TestOptions -> (Int -> Bool -> Bool -> TestOptions -> [TestTree]) -> IO () parallelTestRunner opts mkts = do numjobs <- case concurrentJobs opts of Just NonConcurrent -> pure 1 Just (Concurrent n) -> pure n Just ConcurrentPerCpu -> getNumProcessors Nothing -> getNumProcessors parallelTestRunner' numjobs opts mkts parallelTestRunner' :: Int -> TestOptions -> (Int -> Bool -> Bool -> TestOptions -> [TestTree]) -> IO () parallelTestRunner' numjobs opts mkts | fakeSsh opts = runFakeSsh (internalData opts) | otherwise = go =<< Utility.Env.getEnv subenv where subenv = "GIT_ANNEX_TEST_SUBPROCESS" -- Make more parts than there are jobs, because some parts -- are larger, and this allows the smaller parts to be packed -- in more efficiently, speeding up the test suite overall. -- -- When there is a pattern, splitting into parts will cause -- extra work. numparts = if haspattern then 1 else numjobs * 2 worker rs nvar a = do (n, m) <- atomically $ do (n, m) <- readTVar nvar writeTVar nvar (n+1, m) return (n, m) if n > m then return rs else do r <- a n worker (r:rs) nvar a summarizeresults a = do starttime <- getCurrentTime (numts, exitcodes) <- a duration <- Utility.HumanTime.durationSince starttime case nub (filter (/= ExitSuccess) (concat exitcodes)) of [] -> do putStrLn "" putStrLn $ "All tests succeeded. (Ran " ++ show numts ++ " test groups in " ++ Utility.HumanTime.fromDuration duration ++ ")" exitSuccess [ExitFailure 1] -> do putStrLn " (Failures above could be due to a bug in git-annex, or an incompatibility" putStrLn " with utilities, such as git, installed on this system.)" exitFailure _ -> do putStrLn $ " Test subprocesses exited with unexpected exit codes: " ++ show (concat exitcodes) exitFailure go Nothing = summarizeresults $ withConcurrentOutput $ do ensuredir tmpdir crippledfilesystem <- fst <$> Annex.Init.probeCrippledFileSystem' (toRawFilePath tmpdir) Nothing Nothing False adjustedbranchok <- Annex.AdjustedBranch.isGitVersionSupported let ts = mkts numparts crippledfilesystem adjustedbranchok opts let warnings = fst (tastyParser ts) unless (null warnings) $ do hPutStrLn stderr "warnings from tasty:" mapM_ (hPutStrLn stderr) warnings environ <- Utility.Env.getEnvironment args <- getArgs pp <- Annex.Path.programPath termcolor <- hSupportsANSIColor stdout let ps = if useColor (lookupOption tastyopts) termcolor then "--color=always":args else "--color=never":args let runone n = do let subdir = tmpdir show n ensuredir subdir let p = (proc pp ps) { env = Just ((subenv, show (n, crippledfilesystem, adjustedbranchok)):environ) , cwd = Just subdir } (_, _, _, pid) <- createProcessConcurrent p waitForProcess pid nvar <- newTVarIO (1, length ts) exitcodes <- forConcurrently [1..numjobs] $ \_ -> worker [] nvar runone unless (keepFailuresOption opts) finalCleanup return (length ts, exitcodes) go (Just subenvval) = case readish subenvval of Nothing -> error ("Bad " ++ subenv) Just (n, crippledfilesystem, adjustedbranchok) -> setTestEnv $ do let ts = mkts numparts crippledfilesystem adjustedbranchok opts let t = topLevelTestGroup [ ts !! (n - 1) ] case tryIngredients ingredients tastyopts t of Nothing -> error "No tests found!?" Just act -> ifM act ( exitSuccess , exitFailure ) (haspattern, tastyopts) = case lookupOption (tastyOptionSet opts) of -- Work around limitation of tasty; when tests to run -- are limited to a pattern, it does not include their -- dependencies. So, add another pattern including the -- init tests, which are a dependency of most tests. TestPattern (Just p) -> (True, setOption (TestPattern (Just (TP.Or p (TP.ERE initTestsName)))) (tastyOptionSet opts)) TestPattern Nothing -> (False, tastyOptionSet opts) topLevelTestGroup :: [TestTree] -> TestTree topLevelTestGroup = testGroup "Tests" initTestsName :: String initTestsName = "Init Tests" tastyParser :: [TestTree] -> ([String], Parser Test.Tasty.Options.OptionSet) #if MIN_VERSION_tasty(1,3,0) tastyParser ts = go #else tastyParser ts = ([], go) #endif where go = suiteOptionParser ingredients (topLevelTestGroup ts) ingredients :: [Ingredient] ingredients = [ listingTests , rerunningTests [consoleTestReporter] ]