Skip to content

Commit

Permalink
Merge pull request haskell#1046 from haskell/issue-272-tz-offsets
Browse files Browse the repository at this point in the history
Resolve haskell#272. Allow 23:59 as maximum/minimum tz offset
  • Loading branch information
phadej authored Jun 20, 2023
2 parents 9a48ced + 3774a8d commit f8b5413
Show file tree
Hide file tree
Showing 2 changed files with 16 additions and 7 deletions.
11 changes: 6 additions & 5 deletions text-iso8601/src/Data/Time/FromText.hs
Original file line number Diff line number Diff line change
Expand Up @@ -131,6 +131,9 @@ parseTimeOfDay = parseTimeOfDay_ kontEOF $ \_ _ _ c _ -> unexpectedChar c "end-o
--
-- The accepted formats are @Z@, @+HH@, @+HHMM@, or @+HH:MM@. (@+@ can be @-@).
--
-- Accepts @-23:59..23:59@ range, i.e. @HH < 24@ and @MM < 59@.
-- (This is consistent with grammar, and with what Python, Clojure, joda-time do).
--
parseTimeZone :: Text -> Either String Local.TimeZone
parseTimeZone = parseTimeZone_ Right

Expand Down Expand Up @@ -485,11 +488,9 @@ parseTimeZone__ x kont c t0 = case c of

withResult :: (Int -> Int) -> Int -> Int -> (Local.TimeZone -> Either String b) -> Either String b
withResult posNeg hh mm kontR =
let off = posNeg (hh * 60 + mm)
in if off < (-720) || off > 840 || mm > 59
then Left $ "Invalid TimeZone:" ++ show (hh, mm)
else kontR (Local.minutesToTimeZone off)

if hh < 24 && mm < 60
then kontR (Local.minutesToTimeZone (posNeg (hh * 60 + mm)))
else Left $ "Invalid TimeZone:" ++ show (hh, mm)

{-# INLINE parseLocalTime_ #-}
parseLocalTime_
Expand Down
12 changes: 10 additions & 2 deletions text-iso8601/tests/text-iso8601-tests.hs
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,10 @@ main = defaultMain $ testGroup "text-iso8601"
-- ISO8601 allows various offsets, while RFC3339 only +-HH:MM
, accepts T.parseUTCTime "1990-12-31T15:59:60-0800" -- no colon
, accepts T.parseUTCTime "1990-12-31T15:59:60-08" -- just hour

-- accepts +23:59
, accepts T.parseUTCTime "1937-01-01T12:00:00+23:59"
, accepts T.parseUTCTime "1937-01-01T12:00:00-23:59"
]

, testGroup "rejected"
Expand All @@ -70,6 +74,10 @@ main = defaultMain $ testGroup "text-iso8601"
-- RFC3339 says we MAY limit, i.e. requiring they should be uppercase.
, rejects T.parseUTCTime "2023-06-09T02:35:33z"
, rejects T.parseUTCTime "2023-06-09t02:35:33Z"

-- accepts +23:59, but not 24 or 60
, rejects T.parseUTCTime "1937-01-01T12:00:00+24:59"
, rejects T.parseUTCTime "1937-01-01T12:00:00-23:60"
]
]

Expand All @@ -91,13 +99,13 @@ roundtrip eq build parse = testProperty (show (typeRep (Proxy :: Proxy a))) $ \x
property (liftEq eq y (Right x))

rejects :: forall a. (Typeable a, Show a) => (Text -> Either String a) -> String -> TestTree
rejects parse inp = testCase (show (typeRep (Proxy :: Proxy a)) ++ " accepts " ++ show inp) $ do
rejects parse inp = testCase (show (typeRep (Proxy :: Proxy a)) ++ " rejects " ++ show inp) $ do
case parse (T.pack inp) of
Left _ -> return ()
Right a -> assertFailure $ "Unexpectedly accepted: " ++ show a

accepts :: forall a. (Typeable a, Show a) => (Text -> Either String a) -> String -> TestTree
accepts parse inp = testCase (show (typeRep (Proxy :: Proxy a)) ++ " rejects " ++ show inp) $ do
accepts parse inp = testCase (show (typeRep (Proxy :: Proxy a)) ++ " accepts " ++ show inp) $ do
case parse (T.pack inp) of
Left err -> assertFailure $ "Unexpectedly rejected: " ++ err
Right _ -> return ()

0 comments on commit f8b5413

Please sign in to comment.