{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE StandaloneDeriving #-} {-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE TypeFamilies #-} module Main (main) where import Chronos.Types import Data.List (intercalate) import Test.QuickCheck (Gen, Arbitrary(..), choose, arbitraryBoundedEnum, genericShrink, elements) import Test.QuickCheck.Property (failed,succeeded,Result(..)) import Test.Framework (defaultMain, defaultMainWithOpts, testGroup, Test) import qualified Test.Framework as TF import Test.Framework.Providers.QuickCheck2 (testProperty) import Test.Framework.Providers.HUnit (testCase) import Test.HUnit (Assertion,(@?=),assertBool) import Data.Text (Text) import Data.Text.Lazy.Builder (Builder) import qualified Chronos.Internal.Conversion as Conv import qualified Chronos.Calendar as Month import qualified Chronos.Posix as Posix import qualified Data.Text as Text import qualified Data.Text.Lazy as LText import qualified Data.Text.Lazy.Builder as Builder import qualified Data.Attoparsec.Text as Atto import qualified Chronos.TimeOfDay.Text as TimeOfDayText import qualified Chronos.Datetime.Text as DatetimeText import qualified Chronos.OffsetDatetime.Text as OffsetDatetimeText import qualified Chronos.Date.Text as DateText -- We increase the default number of property-based tests (provided -- by quickcheck) to 1000. Some of the encoding and decoding functions -- we test have special behavior when the minute is zero, which only -- happens in 1/60 of the generated scenarios. If we only generate 100 -- scenarios, there is a 18.62% chance that none of these will have zero -- as the minute. If we increase this to 1000, that probability drops to -- almost nothing. main :: IO () main = defaultMainWithOpts tests mempty { TF.ropt_test_options = Just mempty { TF.topt_maximum_generated_tests = Just 1000 } } tests :: [Test] tests = [ testGroup "Time of Day" [ testGroup "Parser Spec Tests" $ [ testCase "No Separator + microseconds" (timeOfDayParse Nothing "165956.246052" (TimeOfDay 16 59 56246052000)) , testCase "Separator + microseconds" (timeOfDayParse (Just ':') "16:59:56.246052" (TimeOfDay 16 59 56246052000)) , testCase "Separator + milliseconds" (timeOfDayParse (Just ':') "05:00:58.675" (TimeOfDay 05 00 58675000000)) , testCase "Separator + deciseconds" (timeOfDayParse (Just ':') "05:00:58.9" (TimeOfDay 05 00 58900000000)) , testCase "Separator + no subseconds" (timeOfDayParse (Just ':') "23:08:01" (TimeOfDay 23 8 1000000000)) , testCase "Separator + nanoseconds" (timeOfDayParse (Just ':') "05:00:58.111222999" (TimeOfDay 05 00 58111222999)) , testCase "Separator + 10e-18 seconds (truncate)" (timeOfDayParse (Just ':') "05:00:58.111222333444555666" (TimeOfDay 05 00 58111222333)) , testCase "Separator + opt seconds (absent)" (parseMatch (TimeOfDayText.parser_HMS_opt_S (Just ':')) "00:01" (TimeOfDay 0 1 0)) , testCase "Separator + opt seconds (present)" (parseMatch (TimeOfDayText.parser_HMS_opt_S (Just ':')) "00:01:05" (TimeOfDay 0 1 5000000000)) , testCase "No Separator + opt seconds (absent)" (parseMatch (TimeOfDayText.parser_HMS_opt_S Nothing) "0001" (TimeOfDay 0 1 0)) , testCase "No Separator + opt seconds (present)" (parseMatch (TimeOfDayText.parser_HMS_opt_S Nothing) "000105" (TimeOfDay 0 1 5000000000)) ] , testGroup "Builder Spec Tests" [ testCase "No Separator + microseconds" (timeOfDayBuilder (SubsecondPrecisionFixed 6) Nothing "165956.246052" (TimeOfDay 16 59 56246052000)) , testCase "Separator + microseconds" (timeOfDayBuilder (SubsecondPrecisionFixed 6) (Just ':') "16:59:56.246052" (TimeOfDay 16 59 56246052000)) , testCase "Separator + no subseconds" (timeOfDayBuilder (SubsecondPrecisionFixed 0) (Just ':') "23:08:01" (TimeOfDay 23 8 1000000000)) ] , testProperty "Builder Parser Isomorphism (H:M:S)" $ propEncodeDecodeIso (LText.toStrict . Builder.toLazyText . TimeOfDayText.builder_HMS (SubsecondPrecisionFixed 9) (Just ':')) (either (const Nothing) Just . Atto.parseOnly (TimeOfDayText.parser_HMS (Just ':'))) ] , testGroup "Date" [ testGroup "Parser Spec Tests" $ [ testCase "No Separator" (dateParse Nothing "20160101" (Date (Year 2016) (Month 0) (DayOfMonth 1))) , testCase "Separator 1" (dateParse (Just '-') "2016-01-01" (Date (Year 2016) (Month 0) (DayOfMonth 1))) , testCase "Separator 2" (dateParse (Just '-') "1876-09-27" (Date (Year 1876) (Month 8) (DayOfMonth 27))) ] , testGroup "Builder Spec Tests" $ [ testCase "No Separator" (dateBuilder Nothing "20160101" (Date (Year 2016) (Month 0) (DayOfMonth 1))) , testCase "Separator 1" (dateBuilder (Just '-') "2016-01-01" (Date (Year 2016) (Month 0) (DayOfMonth 1))) , testCase "Separator 2" (dateBuilder (Just '-') "1876-09-27" (Date (Year 1876) (Month 8) (DayOfMonth 27))) ] , testProperty "Builder Parser Isomorphism (Y-m-d)" $ propEncodeDecodeIso (LText.toStrict . Builder.toLazyText . DateText.builder_Ymd (Just '-')) (either (const Nothing) Just . Atto.parseOnly (DateText.parser_Ymd (Just '-'))) ] , testGroup "Datetime" [ testProperty "Builder Parser Isomorphism (Y-m-dTH:M:S)" $ propEncodeDecodeIsoSettings (\format -> LText.toStrict . Builder.toLazyText . DatetimeText.builder_YmdHMS (SubsecondPrecisionFixed 9) format) (\format -> either (const Nothing) Just . Atto.parseOnly (DatetimeText.parser_YmdHMS format)) , testProperty "Builder Parser Isomorphism (YmdHMS)" $ propEncodeDecodeIso (LText.toStrict . Builder.toLazyText . DatetimeText.builder_YmdHMS (SubsecondPrecisionFixed 9) (DatetimeFormat Nothing Nothing Nothing)) (either (const Nothing) Just . Atto.parseOnly (DatetimeText.parser_YmdHMS (DatetimeFormat Nothing Nothing Nothing))) ] , testGroup "Offset Datetime" [ testGroup "Builder Spec Tests" $ [ testCase "W3C" $ matchBuilder "1997-07-16T19:20:30.450+01:00" $ OffsetDatetimeText.builderW3 $ OffsetDatetime ( Datetime ( Date (Year 1997) Month.july (DayOfMonth 16) ) ( TimeOfDay 19 20 30450000000 ) ) (Offset 60) ] , testProperty "Builder Parser Isomorphism (YmdHMSz)" $ propEncodeDecodeIsoSettings (\(offsetFormat,datetimeFormat) offsetDatetime -> LText.toStrict $ Builder.toLazyText $ OffsetDatetimeText.builder_YmdHMSz offsetFormat (SubsecondPrecisionFixed 9) datetimeFormat offsetDatetime ) (\(offsetFormat,datetimeFormat) input -> either (const Nothing) Just $ flip Atto.parseOnly input $ OffsetDatetimeText.parser_YmdHMSz offsetFormat datetimeFormat ) ] , testGroup "Posix Time" [ testCase "Get now" $ do now <- Posix.now assertBool "Current time is the beginning of the epoch." (now /= Posix.epoch) ] , testGroup "Conversion" [ testGroup "POSIX to UTC" [ -- Technically, this should fail, but it is very unlikely that quickcheck -- will pick a leapsecond. Actually, since we are going from posix to utc -- first, I think that this cannot fail. testProperty "Isomorphism" $ propEncodeDecodeFullIso Posix.toUtc Posix.fromUtc ] , testGroup "UTC to Datetime" [ testProperty "Isomorphism" $ propEncodeDecodeFullIso Conv.utcTimeToDatetime Conv.datetimeToUtcTime ] , testGroup "POSIX to Datetime" [ testCase "Epoch" $ Posix.toDatetime (PosixTime 0) @?= Datetime (Date (Year 1970) Month.january (DayOfMonth 1)) (TimeOfDay 0 0 0) , testCase "Billion Seconds" $ Posix.toDatetime (PosixTime $ 10 ^ 18) @?= Datetime (Date (Year 2001) Month.september (DayOfMonth 9)) (TimeOfDay 1 46 (40 * 10 ^ 9)) , testProperty "Isomorphism" $ propEncodeDecodeFullIso Posix.toDatetime Posix.fromDatetime ] ] ] failure :: String -> Result failure msg = failed { reason = msg , theException = Nothing } propEncodeDecodeFullIso :: (Eq a,Show a,Show b) => (a -> b) -> (b -> a) -> a -> Result propEncodeDecodeFullIso f g a = let fa = f a gfa = g fa in if gfa == a then succeeded else failure $ concat [ "x: ", show a, "\n" , "f(x): ", show fa, "\n" , "g(f(x)): ", show gfa, "\n" ] propEncodeDecodeIso :: Eq a => (a -> b) -> (b -> Maybe a) -> a -> Bool propEncodeDecodeIso f g a = g (f a) == Just a propEncodeDecodeIsoSettings :: (Eq a,Show a,Show b,Show e) => (e -> a -> b) -> (e -> b -> Maybe a) -> e -> a -> Result propEncodeDecodeIsoSettings f g e a = let fa = f e a gfa = g e fa in if gfa == Just a then succeeded else failure $ concat [ "env: ", show e, "\n" , "x: ", show a, "\n" , "f(x): ", show fa, "\n" , "g(f(x)): ", show gfa, "\n" ] parseMatch :: (Eq a, Show a) => Atto.Parser a -> Text -> a -> Assertion parseMatch p t expected = do Atto.parseOnly (p <* Atto.endOfInput) t @?= Right expected timeOfDayParse :: Maybe Char -> Text -> TimeOfDay -> Assertion timeOfDayParse m t expected = Atto.parseOnly (TimeOfDayText.parser_HMS m <* Atto.endOfInput) t @?= Right expected timeOfDayBuilder :: SubsecondPrecision -> Maybe Char -> Text -> TimeOfDay -> Assertion timeOfDayBuilder sp m expected tod = LText.toStrict (Builder.toLazyText (TimeOfDayText.builder_HMS sp m tod)) @?= expected dateParse :: Maybe Char -> Text -> Date -> Assertion dateParse m t expected = Atto.parseOnly (DateText.parser_Ymd m <* Atto.endOfInput) t @?= Right expected dateBuilder :: Maybe Char -> Text -> Date -> Assertion dateBuilder m expected tod = LText.toStrict (Builder.toLazyText (DateText.builder_Ymd m tod)) @?= expected matchBuilder :: Text -> Builder -> Assertion matchBuilder a b = LText.toStrict (Builder.toLazyText b) @?= a instance Arbitrary TimeOfDay where arbitrary = TimeOfDay <$> choose (0,23) <*> choose (0,59) -- never use leap seconds for property-based tests <*> choose (0,60000000000 - 1) instance Arbitrary Date where arbitrary = Date <$> fmap Year (choose (1800,2100)) <*> fmap Month (choose (0,11)) <*> fmap DayOfMonth (choose (1,28)) instance Arbitrary Datetime where arbitrary = Datetime <$> arbitrary <*> arbitrary instance Arbitrary UtcTime where arbitrary = UtcTime <$> fmap Day (choose (-100000,100000)) <*> choose (0,24 * 60 * 60 * 1000000000 - 1) instance Arbitrary OffsetDatetime where arbitrary = OffsetDatetime <$> arbitrary <*> arbitrary instance (Arbitrary a, a ~ Char) => Arbitrary (DatetimeFormat a) where arbitrary = DatetimeFormat <$> arbitrary <*> elements [Nothing, Just '/', Just ':', Just '-'] <*> arbitrary instance Arbitrary OffsetFormat where arbitrary = arbitraryBoundedEnum shrink = genericShrink deriving instance Arbitrary PosixTime instance Arbitrary Offset where arbitrary = fmap Offset (choose ((-24) * 60, 24 * 60))