1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
|
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE OverloadedStrings #-}
module TestUtils where
import Prelude ()
import Prelude.Compat
import qualified Language.PureScript as P
import qualified Language.PureScript.CST as CST
import Control.Arrow ((***), (>>>))
import Control.Monad
import Control.Monad.Reader
import Control.Monad.Trans.Except
import Control.Monad.Trans.Maybe
import Control.Exception
import Data.Char (isSpace)
import Data.Function (on)
import Data.List (sort, sortBy, stripPrefix, groupBy)
import qualified Data.Map as M
import qualified Data.Text as T
import Data.Time.Clock (UTCTime())
import Data.Tuple (swap)
import System.Process hiding (cwd)
import System.Directory
import System.Info
import System.IO.UTF8 (readUTF8FileT)
import System.Exit (exitFailure)
import System.FilePath
import qualified System.FilePath.Glob as Glob
import System.IO
import Test.Tasty.Hspec
findNodeProcess :: IO (Maybe String)
findNodeProcess = runMaybeT . msum $ map (MaybeT . findExecutable) names
where
names = ["nodejs", "node"]
-- |
-- Fetches code necessary to run the tests with. The resulting support code
-- should then be checked in, so that npm/bower etc is not required to run the
-- tests.
--
-- Simply rerun this (via ghci is probably easiest) when the support code needs
-- updating.
--
updateSupportCode :: IO ()
updateSupportCode = do
setCurrentDirectory "tests/support"
if System.Info.os == "mingw32"
then callProcess "setup-win.cmd" []
else do
callProcess "npm" ["install"]
-- bower uses shebang "/usr/bin/env node", but we might have nodejs
node <- maybe cannotFindNode pure =<< findNodeProcess
-- Sometimes we run as a root (e.g. in simple docker containers)
-- And we are non-interactive: https://github.com/bower/bower/issues/1162
callProcess node ["node_modules/.bin/bower", "--allow-root", "install", "--config.interactive=false"]
setCurrentDirectory "../.."
where
cannotFindNode :: IO a
cannotFindNode = do
hPutStrLn stderr "Cannot find node (or nodejs) executable"
exitFailure
readInput :: [FilePath] -> IO [(FilePath, T.Text)]
readInput inputFiles = forM inputFiles $ \inputFile -> do
text <- readUTF8FileT inputFile
return (inputFile, text)
-- |
-- The support modules that should be cached between test cases, to avoid
-- excessive rebuilding.
--
getSupportModuleTuples :: IO [(FilePath, P.Module)]
getSupportModuleTuples = do
cd <- getCurrentDirectory
let supportDir = cd </> "tests" </> "support"
psciFiles <- Glob.globDir1 (Glob.compile "**/*.purs") (supportDir </> "psci")
libraries <- Glob.globDir1 (Glob.compile "purescript-*/src/**/*.purs") (supportDir </> "bower_components")
let pursFiles = psciFiles ++ libraries
fileContents <- readInput pursFiles
modules <- runExceptT $ ExceptT . return $ CST.parseFromFiles id fileContents
case modules of
Right ms -> return ms
Left errs -> fail (P.prettyPrintMultipleErrors P.defaultPPEOptions errs)
getSupportModuleNames :: IO [T.Text]
getSupportModuleNames = sort . map (P.runModuleName . P.getModuleName . snd) <$> getSupportModuleTuples
pushd :: forall a. FilePath -> IO a -> IO a
pushd dir act = do
original <- getCurrentDirectory
setCurrentDirectory dir
result <- try act :: IO (Either IOException a)
setCurrentDirectory original
either throwIO return result
createOutputFile :: FilePath -> IO Handle
createOutputFile logfileName = do
tmp <- getTemporaryDirectory
createDirectoryIfMissing False (tmp </> logpath)
openFile (tmp </> logpath </> logfileName) WriteMode
setupSupportModules :: IO ([P.Module], [P.ExternsFile], M.Map P.ModuleName FilePath)
setupSupportModules = do
ms <- getSupportModuleTuples
let modules = map snd ms
supportExterns <- runExceptT $ do
foreigns <- inferForeignModules ms
externs <- ExceptT . fmap fst . runTest $ P.make (makeActions modules foreigns) (CST.pureResult <$> modules)
return (externs, foreigns)
case supportExterns of
Left errs -> fail (P.prettyPrintMultipleErrors P.defaultPPEOptions errs)
Right (externs, foreigns) -> return (modules, externs, foreigns)
getTestFiles :: FilePath -> IO [[FilePath]]
getTestFiles testDir = do
cwd <- getCurrentDirectory
let dir = cwd </> "tests" </> "purs" </> testDir
testsInPath <- getFiles dir <$> testGlob dir
let rerunPath = dir </> "RerunCompilerTests.txt"
hasRerunFile <- doesFileExist rerunPath
rerunTests <-
if hasRerunFile
then let compilerTestDir = cwd </> "tests" </> "purs" </> "passing"
textToTestFiles
= mapM (\path -> ((path ++ ".purs") :) <$> testGlob path)
. map ((compilerTestDir </>) . T.unpack)
. filter (not . T.null)
. map (T.strip . fst . T.breakOn "--")
. T.lines
in readUTF8FileT rerunPath >>= textToTestFiles
else return []
return $ testsInPath ++ rerunTests
where
-- A glob for all purs and js files within a test directory
testGlob :: FilePath -> IO [FilePath]
testGlob = Glob.globDir1 (Glob.compile "**/*.purs")
-- Groups the test files so that a top-level file can have dependencies in a
-- subdirectory of the same name. The inner tuple contains a list of the
-- .purs files and the .js files for the test case.
getFiles :: FilePath -> [FilePath] -> [[FilePath]]
getFiles baseDir
= map (filter ((== ".purs") . takeExtensions) . map (baseDir </>))
. groupBy ((==) `on` extractPrefix)
. sortBy (compare `on` extractPrefix)
. map (makeRelative baseDir)
-- Extracts the filename part of a .purs file, or if the file is in a
-- subdirectory, the first part of that directory path.
extractPrefix :: FilePath -> FilePath
extractPrefix fp =
let dir = takeDirectory fp
ext = reverse ".purs"
in if dir == "."
then maybe fp reverse $ stripPrefix ext $ reverse fp
else dir
compile
:: [P.Module]
-> [P.ExternsFile]
-> M.Map P.ModuleName FilePath
-> [FilePath]
-> ([P.Module] -> IO ())
-> IO (Either P.MultipleErrors [P.ExternsFile], P.MultipleErrors)
compile supportModules supportExterns supportForeigns inputFiles check = runTest $ do
fs <- liftIO $ readInput inputFiles
ms <- CST.parseFromFiles id fs
foreigns <- inferForeignModules ms
liftIO (check (map snd ms))
let actions = makeActions supportModules (foreigns `M.union` supportForeigns)
case ms of
[singleModule] -> pure <$> P.rebuildModule actions supportExterns (snd singleModule)
_ -> P.make actions (CST.pureResult <$> supportModules ++ map snd ms)
assert
:: [P.Module]
-> [P.ExternsFile]
-> M.Map P.ModuleName FilePath
-> [FilePath]
-> ([P.Module] -> IO ())
-> (Either P.MultipleErrors P.MultipleErrors -> IO (Maybe String))
-> Expectation
assert supportModules supportExterns supportForeigns inputFiles check f = do
(e, w) <- compile supportModules supportExterns supportForeigns inputFiles check
maybeErr <- f (const w <$> e)
maybe (return ()) expectationFailure maybeErr
checkMain :: [P.Module] -> IO ()
checkMain ms =
unless (any ((== P.moduleNameFromString "Main") . P.getModuleName) ms)
(fail "Main module missing")
makeActions :: [P.Module] -> M.Map P.ModuleName FilePath -> P.MakeActions P.Make
makeActions modules foreigns = (P.buildMakeActions modulesDir (P.internalError "makeActions: input file map was read.") foreigns False)
{ P.getInputTimestampsAndHashes = getInputTimestampsAndHashes
, P.getOutputTimestamp = getOutputTimestamp
, P.progress = const (pure ())
}
where
getInputTimestampsAndHashes :: P.ModuleName -> P.Make (Either P.RebuildPolicy a)
getInputTimestampsAndHashes mn
| isSupportModule (P.runModuleName mn) = return (Left P.RebuildNever)
| otherwise = return (Left P.RebuildAlways)
where
isSupportModule = flip elem (map (P.runModuleName . P.getModuleName) modules)
getOutputTimestamp :: P.ModuleName -> P.Make (Maybe UTCTime)
getOutputTimestamp mn = do
let filePath = modulesDir </> T.unpack (P.runModuleName mn)
exists <- liftIO $ doesDirectoryExist filePath
return (if exists then Just (P.internalError "getOutputTimestamp: read timestamp") else Nothing)
runTest :: P.Make a -> IO (Either P.MultipleErrors a, P.MultipleErrors)
runTest = P.runMake P.defaultOptions
inferForeignModules
:: MonadIO m
=> [(FilePath, P.Module)]
-> m (M.Map P.ModuleName FilePath)
inferForeignModules = P.inferForeignModules . fromList
where
fromList :: [(FilePath, P.Module)] -> M.Map P.ModuleName (Either P.RebuildPolicy FilePath)
fromList = M.fromList . map ((P.getModuleName *** Right) . swap)
trim :: String -> String
trim = dropWhile isSpace >>> reverse >>> dropWhile isSpace >>> reverse
modulesDir :: FilePath
modulesDir = ".test_modules" </> "node_modules"
logpath :: FilePath
logpath = "purescript-output"
|