{-# OPTIONS -Wall #-} {-# OPTIONS -Wcompat #-} {-# OPTIONS -Wincomplete-record-updates #-} {-# OPTIONS -Wincomplete-uni-patterns #-} {-# OPTIONS -Wno-type-defaults #-} {-# LANGUAGE TypeOperators #-} {-# LANGUAGE AllowAmbiguousTypes #-} {-# LANGUAGE TypeApplications #-} {-# LANGUAGE DataKinds #-} {-# LANGUAGE GADTs #-} {-# LANGUAGE PolyKinds #-} {-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE NoOverloadedLists #-} -- overloaded lists breaks some predicates {-# LANGUAGE ViewPatterns #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE DeriveGeneric #-} {-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE LambdaCase #-} module TestRefined where import TastyExtras import Test.Tasty import Test.Tasty.HUnit import Test.Tasty.QuickCheck import Predicate --import Predicate.Refined import Predicate.Util_TH import Predicate.TH_Orphans () -- need this else refined*TH' fails for dates import Control.Lens import Data.Aeson import Control.Monad.Cont import Text.Show.Functions () --import GHC.TypeNats (Nat) import Data.Semigroup (Semigroup(..)) suite :: TestTree suite = let s = "TestRefined" in testGroup s (namedTests <> orderTests s unnamedTests <> allProps) namedTests :: [TestTree] namedTests = [ testCase "always true" $ (@?=) ($$(refinedTH 7) :: Refined 'True Int) (unsafeRefined 7) , testCase "between5and9" $ (@?=) ($$(refinedTH 7) :: Refined (Between 5 9 Id) Int) (unsafeRefined 7) ] unnamedTests :: [IO ()] unnamedTests = [ (@?=) (unsafeRefined @'True ("1.2.3.4" :: String)) $$(refinedTH "1.2.3.4") , (@?=) (unsafeRefined @((Len == 4) && Luhn Id) [1,2,3,0]) $$(refinedTH [1,2,3,0]) , (@?=) (unsafeRefined @(Not ((Len == 4) && Luhn Id)) [1,2,3,1]) $$(refinedTH [1,2,3,1]) , (@?=) [(unsafeRefined 7, "")] (reads @(Refined (Between 2 10 Id) Int) "Refined {unRefined = 7}") , (@?=) [] (reads @(Refined (Between 2 10 Id) Int) "Refined {unRefined = 0}") , (@?=) [(unsafeRefined "abcaaaabb", "")] (reads @(Refined (Re "^[abc]+$" Id) String) "Refined {unRefined = \"abcaaaabb\"}") , (@?=) [] (reads @(Refined (Re "^[abc]+$" Id) String) "Refined {unRefined = \"abcaaaabbx\"}") , expectJ (Left ["Error in $: Refined:FalseP"]) (toFrom (unsafeRefined @(Between 4 7 Id || Gt 14) 12)) , expectJ (Right (unsafeRefined 22)) (toFrom (unsafeRefined @(Between 4 7 Id || Gt 14) 22)) , expectJ (Left ["Error in $: Refined:FailP \"someval\""]) (toFrom (unsafeRefined @(Between 4 7 Id || Gt 14 || Failt _ "someval") 12)) , (fst <$> unRavelT (tst2 ol 10 200)) >>= (@?= Right (10,200)) , (fst <$> unRavelT (tst2 ol 11 12)) >>= (@?= Left "FalseP") , (fst <$> unRavelT (tst1 ol 10 200)) >>= (@?= Right (10,200)) , (fst <$> unRavelT (tst1 ol 11 12)) >>= (@?= Left "FalseP") ] allProps :: [TestTree] allProps = [ testProperty "readshow" $ forAll (arbRefined @(Between 10 45 Id) ol) (\r -> read @(Refined (Between 10 45 Id) Int) (show r) === r) , testProperty "jsonroundtrip" $ forAll (arbRefined @(Between 10 45 Id) ol) (\r -> testRefinedJ @(Between 10 45 Id) ol (unRefined r) === Right r) ] tst1 :: Monad m => POpts -> Int -> Int -> RefinedT m (Int,Int) tst1 opts i j = withRefinedT @(Between 2 11 Id) opts i $ \x -> withRefinedT @(Between 200 211 Id) opts j $ \y -> return (unRefined x, unRefined y) tst2 :: MonadIO m => POpts -> Int -> Int -> RefinedT m (Int,Int) tst2 opts i j = withRefinedTIO @(Between 2 11 Id) opts i $ \x -> withRefinedTIO @(Stderr "startio..." |> Between 200 211 Id >| Stderr "...endio") opts j $ \y -> return (unRefined x, unRefined y) -- roundtrip tojson then fromjson testRefinedJ :: forall p a . (ToJSON a, FromJSON a, RefinedC p a) => POpts -> a -> Either String (Refined p a) testRefinedJ opts a = let ((bp,(e,_top)),mr) = runIdentity $ newRefined @p opts a in case mr of Nothing -> error $ show bp ++ "\n" ++ e Just r -> eitherDecode @(Refined p a) $ encode r