{-# LANGUAGE RecordWildCards #-} {-# LANGUAGE TupleSections #-} module TestPsci where import Prelude () import Prelude.Compat import Control.Monad.Trans.State.Strict (evalStateT) import Control.Monad (when) import Data.List (sort) import System.Exit (exitFailure) import System.Console.Haskeline import System.FilePath (()) import System.Directory (getCurrentDirectory) import qualified System.FilePath.Glob as Glob import Test.HUnit import qualified Language.PureScript as P import Language.PureScript.Interactive.Module (loadAllModules) import Language.PureScript.Interactive.Completion import Language.PureScript.Interactive.Types import TestUtils (supportModules) main :: IO () main = do Counts{..} <- runTestTT allTests when (errors + failures > 0) exitFailure allTests :: Test allTests = completionTests completionTests :: Test completionTests = TestLabel "completionTests" (TestList (map (TestCase . assertCompletedOk) completionTestData)) -- If the cursor is at the right end of the line, with the 1st element of the -- pair as the text in the line, then pressing tab should offer all the -- elements of the list (which is the 2nd element) as completions. completionTestData :: [(String, [String])] completionTestData = -- basic directives [ (":h", [":help"]) , (":re", [":reset"]) , (":q", [":quit"]) , (":b", [":browse"]) -- :browse should complete module names , (":b Control.Monad.E", map (":b Control.Monad.Eff" ++) ["", ".Unsafe", ".Class", ".Console"]) , (":b Control.Monad.Eff.", map (":b Control.Monad.Eff" ++) [".Unsafe", ".Class", ".Console"]) -- import should complete module names , ("import Control.Monad.E", map ("import Control.Monad.Eff" ++) ["", ".Unsafe", ".Class", ".Console"]) , ("import Control.Monad.Eff.", map ("import Control.Monad.Eff" ++) [".Unsafe", ".Class", ".Console"]) -- :quit, :help, :reset should not complete , (":help ", []) , (":quit ", []) , (":reset ", []) -- :show should complete to "loaded" and "import" , (":show ", [":show import", ":show loaded"]) , (":show a", []) -- :type should complete values and data constructors in scope , (":type Control.Monad.Eff.Console.lo", [":type Control.Monad.Eff.Console.log", ":type Control.Monad.Eff.Console.logShow"]) --, (":type uni", [":type unit"]) --, (":type E", [":type EQ"]) -- :kind should complete types in scope --, (":kind C", [":kind Control.Monad.Eff.Pure"]) --, (":kind O", [":kind Ordering"]) -- Only one argument for directives should be completed , (":show import ", []) , (":type EQ ", []) , (":kind Ordering ", []) -- a few other import tests , ("impor", ["import"]) , ("import ", map ("import " ++) supportModules) , ("import Prelude ", []) -- String and number literals should not be completed , ("\"hi", []) , ("34", []) -- Identifiers and data constructors should be completed --, ("uni", ["unit"]) , ("Control.Monad.Eff.Class.", ["Control.Monad.Eff.Class.liftEff"]) --, ("G", ["GT"]) , ("Data.Ordering.L", ["Data.Ordering.LT"]) -- if a module is imported qualified, values should complete under the -- qualified name, as well as the original name. , ("ST.new", ["ST.newSTRef"]) , ("Control.Monad.ST.new", ["Control.Monad.ST.newSTRef"]) ] assertCompletedOk :: (String, [String]) -> Assertion assertCompletedOk (line, expecteds) = do (unusedR, completions) <- runCM (completion' (reverse line, "")) let unused = reverse unusedR let actuals = map ((unused ++) . replacement) completions sort expecteds @=? sort actuals runCM :: CompletionM a -> IO a runCM act = do psciState <- getPSCiState evalStateT (liftCompletionM act) psciState getPSCiState :: IO PSCiState getPSCiState = do cwd <- getCurrentDirectory let supportDir = cwd "tests" "support" "bower_components" let supportFiles ext = Glob.globDir1 (Glob.compile ("purescript-*/**/*." ++ ext)) supportDir pursFiles <- supportFiles "purs" modulesOrFirstError <- loadAllModules pursFiles case modulesOrFirstError of Left err -> print err >> exitFailure Right modules -> let imports = [controlMonadSTasST, (P.ModuleName [P.ProperName "Prelude"], P.Implicit, Nothing)] dummyExterns = P.internalError "TestPsci: dummyExterns should not be used" in return (PSCiState imports [] (zip (map snd modules) (repeat dummyExterns))) controlMonadSTasST :: ImportedModule controlMonadSTasST = (s "Control.Monad.ST", P.Implicit, Just (s "ST")) where s = P.moduleNameFromString