From 169ffaee061f0a53ad7fc4da7aaced0d96a05536 Mon Sep 17 00:00:00 2001 From: Irakli Safareli Date: Wed, 20 Dec 2017 16:05:45 +0100 Subject: [PATCH] add Gen module and propery tests --- src/Data/Path/Pathy/Gen.purs | 45 ++++++++++++++++++++++++++++++++++++ test/Main.purs | 32 ++++++++++++++++++++++++- 2 files changed, 76 insertions(+), 1 deletion(-) create mode 100644 src/Data/Path/Pathy/Gen.purs diff --git a/src/Data/Path/Pathy/Gen.purs b/src/Data/Path/Pathy/Gen.purs new file mode 100644 index 0000000..c6d163f --- /dev/null +++ b/src/Data/Path/Pathy/Gen.purs @@ -0,0 +1,45 @@ +module Data.Path.Pathy.Gen where + +import Prelude + +import Control.Monad.Gen (class MonadGen) +import Control.Monad.Gen as Gen +import Control.Monad.Rec.Class (class MonadRec) +import Data.Either (Either(..)) +import Data.Foldable (foldr) +import Data.List as L +import Data.NonEmpty ((:|)) +import Data.Path.Pathy (AbsPath, AbsFile, AbsDir, RelDir, RelFile, RelPath, Sandboxed, ()) +import Data.Path.Pathy as P + +genAbsDirPath :: forall m. MonadGen m => MonadRec m => m String -> m (AbsDir Sandboxed) +genAbsDirPath genName = Gen.sized \size → do + newSize ← Gen.chooseInt 0 size + Gen.resize (const newSize) do + parts ∷ L.List String ← Gen.unfoldable genName + pure $ foldr (flip P.appendPath <<< P.dir) P.rootDir parts + +genAbsFilePath :: forall m. MonadGen m => MonadRec m => m String -> m (AbsFile Sandboxed) +genAbsFilePath genName = do + dir ← genAbsDirPath genName + file ← genName + pure $ dir P.file file + +genAbsAnyPath :: forall m. MonadGen m => MonadRec m => m String -> m (AbsPath Sandboxed) +genAbsAnyPath genName = Gen.oneOf $ (Left <$> genAbsDirPath genName) :| [Right <$> genAbsFilePath genName] + +genRelDirPath :: forall m. MonadGen m => MonadRec m => m String -> m (RelDir Sandboxed) +genRelDirPath genName = Gen.sized \size → do + newSize ← Gen.chooseInt 0 size + Gen.resize (const newSize) do + parts ∷ L.List String ← Gen.unfoldable genName + pure $ foldr (flip P.appendPath <<< P.dir) P.currentDir parts + +genRelFilePath :: forall m. MonadGen m => MonadRec m => m String -> m (RelFile Sandboxed) +genRelFilePath genName = do + dir ← genRelDirPath genName + file ← genName + pure $ dir P.file file + +genRelAnyPath :: forall m. MonadGen m => MonadRec m => m String -> m (RelPath Sandboxed) +genRelAnyPath genName = Gen.oneOf $ (Left <$> genRelDirPath genName) :| [Right <$> genRelFilePath genName] diff --git a/test/Main.purs b/test/Main.purs index ca41e0e..bcfa1eb 100644 --- a/test/Main.purs +++ b/test/Main.purs @@ -5,7 +5,8 @@ import Control.Monad.Eff (Eff) import Control.Monad.Eff.Console (CONSOLE, info, infoShow) import Data.Foldable (foldl) import Data.Maybe (Maybe(..), fromJust) -import Data.Path.Pathy (Path, Abs, Rel, Dir, File, Sandboxed, dir, rootDir, parseAbsDir, parseRelDir, currentDir, file, parseAbsFile, parseRelFile, parentDir', depth, sandbox, dropExtension, renameFile, canonicalize, unsandbox, unsafePrintPath, (), (<..>), (<.>)) +import Data.Path.Pathy (Path, Abs, Rel, Dir, File, Unsandboxed, Sandboxed, dir, rootDir, parseAbsDir, parseRelDir, currentDir, file, parseAbsFile, parseRelFile, parentDir', depth, sandbox, dropExtension, renameFile, canonicalize, unsandbox, unsafePrintPath, (), (<..>), (<.>)) +import Data.Path.Pathy.Gen as PG import Data.String as Str import Partial.Unsafe (unsafePartial) import Test.QuickCheck as QC @@ -39,8 +40,37 @@ instance arbitraryArbPath ∷ QC.Arbitrary ArbPath where pathPart ∷ Gen.Gen String pathPart = Gen.suchThat QC.arbitrary (not <<< Str.null) +parsePrintCheck :: forall a b. Path a b Sandboxed -> Maybe (Path a b Unsandboxed) -> QC.Result +parsePrintCheck input parsed = + if parsed == Just (unsandbox input) + then QC.Success + else QC.Failed + $ "`parse (print path) != Just path` for path: `" <> show input <> "` which was re-parsed into `" <> show parsed <> "`" + <> "\n\tPrinted path: " <> show (unsafePrintPath input) + <> "\n\tPrinted path': `" <> show (map unsafePrintPath parsed) <> "`" + +parsePrintAbsDirPath :: Gen.Gen QC.Result +parsePrintAbsDirPath = PG.genAbsDirPath pathPart <#> \path -> + parsePrintCheck path (parseAbsDir $ unsafePrintPath path) + +parsePrintAbsFilePath :: Gen.Gen QC.Result +parsePrintAbsFilePath = PG.genAbsFilePath pathPart <#> \path -> + parsePrintCheck path (parseAbsFile $ unsafePrintPath path) + +parsePrintRelDirPath :: Gen.Gen QC.Result +parsePrintRelDirPath = PG.genRelDirPath pathPart <#> \path -> + parsePrintCheck path (parseRelDir $ unsafePrintPath path) + +parsePrintRelFilePath :: Gen.Gen QC.Result +parsePrintRelFilePath = PG.genRelFilePath pathPart <#> \path -> + parsePrintCheck path (parseRelFile $ unsafePrintPath path) + main :: QC.QC () Unit main = do + info "checking `parse <<< print` for `AbsDir``" *> QC.quickCheck parsePrintAbsDirPath + info "checking `parse <<< print` for `AbsFile``" *> QC.quickCheck parsePrintAbsFilePath + info "checking `parse <<< print` for `RelDir``" *> QC.quickCheck parsePrintRelDirPath + info "checking `parse <<< print` for `RelFile``" *> QC.quickCheck parsePrintRelFilePath -- Should not compile: -- test "() - file in dir" (printPath (file "image.png" dir "foo")) "./image.png/foo"