module Data.Time.Format.RFC733
( RFC733
)
where
import Control.Applicative
import Data.Ascii (Ascii, AsciiBuilder)
import qualified Data.Ascii as A
import Data.Attoparsec.Char8
import Data.Convertible.Base
import Data.Default
import Data.Monoid.Unicode
import Data.Tagged
import Data.Time
import Data.Time.Calendar.WeekDate
import Data.Time.Format.HTTP.Common
import Data.Time.Format.RFC822
import Prelude.Unicode
data RFC733
instance ConvertSuccess (Tagged RFC733 ZonedTime) Ascii where
convertSuccess = A.fromAsciiBuilder ∘ cs
instance ConvertSuccess (Tagged RFC733 ZonedTime) AsciiBuilder where
convertSuccess = toAsciiBuilder
instance ConvertAttempt Ascii (Tagged RFC733 ZonedTime) where
convertAttempt = parseAttempt' def
instance Default (Parser (Tagged RFC733 ZonedTime)) where
def = do weekDay ← optionMaybe $
do w ← longWeekDayNameP
<|>
shortWeekDayNameP
string ", " *> pure w
gregDay ← date
case weekDay of
Nothing
→ return ()
Just givenWD
→ assertWeekDayIsGood givenWD gregDay
(tod, timeZone) ← time
let lt = LocalTime gregDay tod
zt = ZonedTime lt timeZone
pure $ Tagged zt
date ∷ Parser Day
date = do day ← read2
_ ← char '-' <|> char ' '
month ← try longMonthNameP
<|>
shortMonthNameP
_ ← char '-' <|> char ' '
year ← try read4
<|>
(+ 1900) <$> read2
_ ← char ' '
assertGregorianDateIsGood year month day
time ∷ Parser (TimeOfDay, TimeZone)
time = do tod ← hms
_ ← char '-' <|> char ' '
tz ← zone
return (tod, tz)
hms ∷ Parser TimeOfDay
hms = do hour ← read2
_ ← optional (char ':')
minute ← read2
second ← option 0 $
do _ ← optional (char ':')
read2
assertTimeOfDayIsGood hour minute second
zone ∷ Parser TimeZone
zone = choice [ string "GMT" *> return (TimeZone 0 False "GMT")
, char 'N'
*> choice [ string "ST" *> return (TimeZone ((3) * 60 30) False "NST")
, return (TimeZone (1 * 60) False "N")
]
, char 'A'
*> choice [ string "ST" *> return (TimeZone ((4) * 60) False "AST")
, string "DT" *> return (TimeZone ((3) * 60) False "AST")
, return (TimeZone ((1) * 60) False "A")
]
, char 'E'
*> choice [ string "ST" *> return (TimeZone ((5) * 60) False "EST")
, string "DT" *> return (TimeZone ((4) * 60) True "EDT")
]
, char 'C'
*> choice [ string "ST" *> return (TimeZone ((6) * 60) False "CST")
, string "DT" *> return (TimeZone ((5) * 60) True "CDT")
]
, char 'M'
*> choice [ string "ST" *> return (TimeZone ((7) * 60) False "MST")
, string "DT" *> return (TimeZone ((6) * 60) True "MDT")
, return (TimeZone ((12) * 60) False "M")
]
, char 'P'
*> choice [ string "ST" *> return (TimeZone ((8) * 60) False "PST")
, string "DT" *> return (TimeZone ((7) * 60) True "PDT")
]
, char 'Y'
*> choice [ string "ST" *> return (TimeZone ((9) * 60) False "YST")
, string "DT" *> return (TimeZone ((8) * 60) True "YDT")
, return (TimeZone ( 12 * 60) False "Y")
]
, char 'H'
*> choice [ string "ST" *> return (TimeZone ((10) * 60) False "HST")
, string "DT" *> return (TimeZone (( 9) * 60) True "HDT")
]
, char 'B'
*> choice [ string "ST" *> return (TimeZone ((11) * 60) False "BST")
, string "DT" *> return (TimeZone ((10) * 60) True "BDT")
]
, char 'Z' *> return (TimeZone 0 False "Z")
, read4digitsTZ
]
toAsciiBuilder ∷ Tagged RFC733 ZonedTime → AsciiBuilder
toAsciiBuilder zonedTime
= let localTime = zonedTimeToLocalTime $ untag zonedTime
timeZone = zonedTimeZone <$> retag' zonedTime
(year, month, day) = toGregorian (localDay localTime)
(_, _, week) = toWeekDate (localDay localTime)
timeOfDay = localTimeOfDay localTime
in
longWeekDayName week
⊕ A.toAsciiBuilder ", "
⊕ show2 day
⊕ A.toAsciiBuilder "-"
⊕ shortMonthName month
⊕ A.toAsciiBuilder "-"
⊕ show4 year
⊕ A.toAsciiBuilder " "
⊕ show2 (todHour timeOfDay)
⊕ A.toAsciiBuilder ":"
⊕ show2 (todMin timeOfDay)
⊕ A.toAsciiBuilder ":"
⊕ show2 (floor (todSec timeOfDay) ∷ Int)
⊕ A.toAsciiBuilder " "
⊕ cs timeZone
where
retag' ∷ Tagged τ α → Tagged RFC822 α
retag' = retag
deriveAttempts [ ([t| Tagged RFC733 ZonedTime |], [t| Ascii |])
, ([t| Tagged RFC733 ZonedTime |], [t| AsciiBuilder |])
]