module Network.PagerDuty.Internal.Types where
import           Control.Applicative
import           Control.Lens                     hiding ((.=))
import           Control.Monad.IO.Class
import           Data.Aeson                       hiding (Error)
import           Data.Aeson.Types                 (Parser)
import           Data.ByteString                  (ByteString)
import qualified Data.ByteString.Char8            as BS
import           Data.ByteString.Conversion       hiding (List)
import           Data.Default.Class
import           Data.Function                    (on)
import qualified Data.HashMap.Strict              as Map
import           Data.List                        (deleteBy, intersperse)
import           Data.Monoid
import           Data.String
import           Data.Text                        (Text)
import qualified Data.Text                        as Text
import qualified Data.Text.Encoding               as Text
import           Data.Time
import           GHC.TypeLits
import           Network.HTTP.Client              (Manager)
import           Network.HTTP.Types
import           Network.HTTP.Types.QueryLike
import           Network.PagerDuty.Internal.Query
import           Network.PagerDuty.Internal.TH
import           Data.Time.Locale.Compat
newtype CSV a = CSV [a]
    deriving (Eq, Show, Monoid)
makePrisms ''CSV
instance ToByteString a => QueryValues (CSV a)
instance ToByteString a => ToByteString (CSV a) where
    builder (CSV xs) = mconcat . intersperse "," $ map builder xs
instance FromJSON a => FromJSON (CSV a) where
    parseJSON = withText "comma_separated_value" $
        fmap CSV . traverse (parseJSON . String) . Text.split (== ',')
instance ToByteString a => ToJSON (CSV a) where
    toJSON = String . Text.decodeUtf8 . toByteString'
newtype List a = L [a]
    deriving (Eq, Show, Monoid)
deriveJSON ''List
makePrisms ''List
instance QueryValues a => QueryValues (List a) where
    queryValues (L xs) = concatMap queryValues xs
newtype Bool' = B Bool
    deriving (Eq, Show)
deriveJSON ''Bool'
makePrisms ''Bool'
instance ToByteString Bool' where
    builder (B True)  = "true"
    builder (B False) = "false"
instance QueryValues Bool'
pattern T = B True
pattern F = B False
newtype Date = D UTCTime
    deriving (Eq, Ord, Show)
makePrisms ''Date
instance FromJSON Date where
    parseJSON = fmap D . parseJSON
instance ToJSON Date where
    toJSON (D d) = toJSON d
instance ToByteString Date where
    builder (D d) = builder
        (formatTime defaultTimeLocale ("%Y-%m-%dT%XZ") d)
instance QueryValues Date
newtype TZ = TZ TimeZone
    deriving (Eq, Show)
makePrisms ''TZ
instance FromJSON TZ where
    parseJSON = undefined
instance ToJSON TZ where
    toJSON = toJSON . Text.decodeUtf8 . toByteString'
instance ToByteString TZ where
    builder (TZ tz) = builder (timeZoneName tz)
instance QueryValues TZ
instance Default TZ where
    def = TZ utc
data Security = Basic | Token
    deriving (Eq, Show)
data Auth (a :: Security) where
    AuthBasic :: ByteString -> ByteString -> Auth Basic
    AuthToken :: ByteString -> Auth Token
deriving instance Eq   (Auth a)
deriving instance Show (Auth a)
newtype SubDomain = SubDomain { subDomain :: ByteString }
    deriving (Eq, Show, IsString, ToByteString)
mkSubDomain :: ByteString -> SubDomain
mkSubDomain = SubDomain
domain :: SubDomain -> ByteString
domain (SubDomain s)
    | base `BS.isSuffixOf` s = s
    | otherwise              = s <> base
  where
    base = ".pagerduty.com"
data Logger
    = None
    | Debug (Text -> IO ())
debug :: MonadIO m => Logger -> Text -> m ()
debug None      = const (return ())
debug (Debug f) = liftIO . f
data Env (s :: Security) = Env
    { _envDomain  :: SubDomain
    , _envAuth'   :: Auth s
    , _envManager :: Manager
    , _envLogger  :: Logger
    }
makeLenses ''Env
envAuth :: forall s s'. Lens (Env s) (Env s') (Auth s) (Auth s')
envAuth = lens _envAuth' (\s x -> s { _envAuth' = x })
prod :: SubDomain -> Auth s -> Manager -> Env s
prod d a m = Env d a m None
newtype Code = Code Integer
    deriving (Eq, Show, Num)
deriveJSON ''Code
description :: Code -> Text
description (Code c) =
    case c of
        2000 -> "Internal Error"
        2001 -> "Invalid Input Provided"
        2002 -> "Arguments Caused Error"
        2003 -> "Missing Arguments"
        2004 -> "Invalid 'since' or 'until' Parameter Values"
        2005 -> "Invalid Query Date Range"
        2006 -> "Authentication Failed"
        2007 -> "Account Not Found"
        2008 -> "Account Locked"
        2009 -> "Only HTTPS Allowed For This Call"
        2010 -> "Access Denied"
        2011 -> "The action requires a 'requester_id' to be specified"
        2012 -> "Your account is expired and cannot use the API"
        _    -> "Unrecognised error code"
class HasMessage s a | s -> a where
    
    message :: Lens' s a
class HasErrors s a | s -> a where
    
    
    errors :: Lens' s a
data IntegrationError = IntegrationError
    { _ieStatus  :: Text
    , _ieMessage :: Text
    , _ieErrors  :: [Text]
    } deriving (Eq, Show)
deriveRecord ''IntegrationError
instance HasMessage IntegrationError Text   where message = ieMessage
instance HasErrors  IntegrationError [Text] where errors  = ieErrors
status :: Lens' IntegrationError Text
status = ieStatus
data RESTError = RESTError
    { _reCode    :: Code
    , _reMessage :: Text
    , _reErrors  :: [Text]
    } deriving (Eq, Show)
deriveRecord ''RESTError
instance HasMessage RESTError Text   where message = reMessage
instance HasErrors  RESTError [Text] where errors  = reErrors
code :: Lens' RESTError Code
code = reCode
data Error
    = Internal    Text
    | Integration IntegrationError
    | REST        RESTError
      deriving (Eq, Show)
instance FromJSON Error where
    parseJSON o = (REST <$> parseJSON o)
       <|> (Integration <$> parseJSON o)
makePrisms ''Error
instance HasMessage Error Text where
    message = lens f g
      where
        f (Internal    x) = x
        f (Integration s) = _ieMessage s
        f (REST        s) = _reMessage s
        g (Internal    _) x = Internal    x
        g (Integration s) x = Integration $ s { _ieMessage = x }
        g (REST        s) x = REST        $ s { _reMessage = x }
data Pager = Pager
    { _pgOffset :: !Int
      
    , _pgLimit  :: !Int
      
    , _pgTotal  :: !Int
      
    , _pgQuery  :: Maybe Text
      
    } deriving (Eq, Show)
makeLenses ''Pager
instance FromJSON a => FromJSON (a, Maybe Pager) where
    parseJSON = withObject "paginated" $ \o -> (,)
        <$> parseJSON (Object o)
        <*> optional  (parse o)
      where
        parse o = Pager
           
           
           <$> o .: "offset" .!= 0
           
           
           <*> o .:  "limit"  .!= 100
           <*> o .:  "total"
           <*> o .:? "query"
instance ToJSON Pager where
    toJSON p = object
        [ "offset" .= _pgOffset p
        , "limit"  .= _pgLimit p
        ]
data Path where
    Path :: Path
    Seg  :: ToByteString a => a -> Path
instance Monoid Path where
    mempty                  = Path
    mappend x Path          = x
    mappend Path y          = y
    mappend (Seg x) (Seg y) = Seg (builder x <> "/" <> builder y)
instance IsString Path where
    fromString = Seg
instance ToByteString Path where
    builder Path    = mempty
    builder (Seg x) = builder x
(%) :: ToByteString a => Path -> a -> Path
a % b = a <> Seg b
renderPath :: Path -> ByteString
renderPath = toByteString' . mappend v1
  where
    v1 :: Path
    v1 = "/api/v1"
data Request a (s :: Security) b where
    Request :: (QueryLike a, ToJSON a)
            => { _rqMeth   :: !StdMethod
               , _rqPath   :: Path
               , _rqQuery  :: Query
               , _rqBody   :: a
               , _rqPager  :: Maybe Pager
               , _rqUnwrap :: Value -> Parser Value
               }
            -> Request a s b
instance ToJSON (Request a s b) where
    
    toJSON (Request _ _ _ b p _) = Object $
        let Object x = toJSON b
         in case toJSON p of
                (Object y) -> x <> y
                _          -> x
type Unwrap = Getting (First Value) Value Value
mk :: (QueryLike a, ToJSON a) => a -> Request a s b
mk x = Request GET mempty mempty x Nothing pure
empty :: Request Empty s r
empty = mk Empty
upd :: (QueryLike a, ToJSON a) => Lens' (Request a s b) a
upd = lens _rqBody (\(Request m p q _ g u) x -> Request m p q x g u)
auth :: Request a s b -> Request a t b
auth (Request x m p q g u) = Request x m p q g u
meth :: Lens' (Request a s b) StdMethod
meth = lens _rqMeth (\r x -> r { _rqMeth = x })
path :: Lens' (Request a s b) Path
path = lens _rqPath (\r x -> r { _rqPath = x })
query :: QueryValueLike v
      => Lens (Request a s b) (Request a s b) Query [(ByteString, v)]
query = lens _rqQuery (\r x -> r { _rqQuery = toQuery x })
pager :: Lens' (Request a s b) (Maybe Pager)
pager = lens _rqPager (\r x -> r { _rqPager = x })
unwrap :: Setter (Request a s b) (Request a s b) (Value -> Parser Value) Unwrap
unwrap f r = f (_rqUnwrap r) <&> \k -> r { _rqUnwrap = g k }
  where
    g k x = maybe (fail "Failed to extract nested keys.") return (x ^? k)
class Paginate a where
    next :: Request a s b -> Maybe Pager -> Maybe (Request a s b)
    next rq = maybe Nothing go
      where
        go x | x ^. pgTotal == 0 = Nothing
             | otherwise         = Just $
                 rq & pager ?~ (x & pgOffset +~ x ^. pgTotal)
                    & query %~ (add . clear)
          where
            add :: Query -> Query
            add = maybe id ((:) . (k,) . Just . Text.encodeUtf8) (x ^. pgQuery)
        clear :: Query -> Query
        clear = deleteBy ((==) `on` fst) (k, Nothing)
        k :: ByteString
        k = "query"
newtype Key (a :: Symbol) = Key Text
    deriving (Eq, Show, IsString)
mkKey :: Text -> Key a
mkKey = Key
instance FromJSON (Key a) where
    parseJSON = withText "key" (return . Key)
instance ToJSON (Key a) where
    toJSON (Key k) = toJSON k
instance ToByteString (Key a) where
    builder (Key k) = builder k
instance QueryValues (Key a)
instance QueryValueLike (Key a) where
    toQueryValue = Just . toByteString'
type ServiceKey  = Key "service"
type IncidentKey = Key "incident"
newtype Id (a :: Symbol) = Id Text
    deriving (Eq, Show, IsString)
mkId :: Text -> Id a
mkId = Id
instance FromJSON (Id a) where
    parseJSON = withText "id" (return . Id)
instance ToJSON (Id a) where
    toJSON (Id i) = toJSON i
instance ToByteString (Id a) where
    builder (Id i) = builder i
instance QueryValues (Id a)
instance QueryValueLike (Id a) where
    toQueryValue = Just . toByteString'
type AlertId            = Id "alert"
type ContactId          = Id "contact"
type EmailFilterId      = Id "email-filter"
type EscalationPolicyId = Id "escalation-policy"
type EscalationRuleId   = Id "escalation-rule"
type IncidentId         = Id "incident"
type LogEntryId         = Id "log-entry"
type NoteId             = Id "note"
type NotificationRuleId = Id "notification-rule"
type OverrideId         = Id "schedule-override"
type RequesterId        = Id "requester"
type ScheduleId         = Id "schedule"
type ServiceId          = Id "service"
type UserId             = Id "user"
type VendorId           = Id "vendor"
type WebhookId          = Id "webhook"
type WindowId           = Id "maintenance-window"
data Empty = Empty
instance ToJSON Empty where
    toJSON = const (object [])
instance FromJSON Empty where
    parseJSON = withObject "empty" f
      where
        f !o | Map.null o = pure Empty
             | otherwise  = fail "Unexpected non-empty JSON object."
instance QueryLike Empty where
    toQuery = const []
newtype Address = Address Text
    deriving (Eq, Show, IsString)
mkAddress :: Text -> Address
mkAddress = Address
deriveJSON ''Address
makePrisms ''Address
instance ToByteString Address where
    builder (Address a) = builder a
instance QueryValues Address