{-# LANGUAGE DeriveFoldable #-}
{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE DeriveTraversable #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE FunctionalDependencies #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE NoImplicitPrelude #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TupleSections #-}
{-# LANGUAGE TypeFamilies #-}
module Waargonaut.Types.CommaSep
(
CommaSeparated (..)
, Elems (..)
, HasElems (..)
, Elem (..)
, HasElem (..)
, Comma (..)
, parseComma
, commaBuilder
, parseCommaSeparated
, commaSeparatedBuilder
, _CommaSeparated
, toList
, fromList
, consCommaSep
, unconsCommaSep
) where
import Prelude (Eq, Int, Show (showsPrec), otherwise,
showString, shows, (&&), (<=), (==))
import Control.Applicative (Applicative (..), liftA2, pure, (*>),
(<*), (<*>))
import Control.Category (id, (.))
import Control.Lens (AsEmpty (..), Cons (..), Index, Iso,
Iso', IxValue, Ixed (..), Lens',
Snoc (..), cons, from, isn't, iso,
mapped, nearly, over, prism, snoc, to,
traverse, unsnoc, (%%~), (%~), (.~),
(^.), (^..), (^?), _1, _2, _Cons,
_Just, _Nothing)
import Control.Error.Util (note)
import Control.Monad (Monad)
import Data.Bifoldable (Bifoldable (bifoldMap))
import Data.Bifunctor (Bifunctor (bimap))
import Data.Bitraversable (Bitraversable (bitraverse))
import Data.Char (Char)
import Data.Either (Either (..))
import Data.Foldable (Foldable, asum, foldMap, foldr,
length)
import Data.Function (const, flip, ($), (&))
import Data.Functor (Functor, fmap, (<$), (<$>))
import Data.Functor.Classes (Eq1, Show1, eq1, showsPrec1)
import Data.Maybe (Maybe (..), fromMaybe, maybe)
import Data.Monoid (Monoid (..), mempty)
import Data.Semigroup (Semigroup ((<>)))
import Data.Traversable (Traversable)
import Data.Tuple (snd, uncurry)
import Data.Vector (Vector)
import qualified Data.Vector as V
import Data.Functor.Identity (Identity (..))
import Data.ByteString.Builder (Builder)
import qualified Data.ByteString.Builder as BB
import Text.Parser.Char (CharParsing, char)
import qualified Text.Parser.Combinators as C
import Data.Witherable (Filterable (..), Witherable (..))
data Comma = Comma
deriving (Eq, Show)
_Comma :: Iso' Comma ()
_Comma = iso (\Comma -> ()) (const Comma)
commaBuilder :: Builder
commaBuilder = BB.charUtf8 ','
{-# INLINE commaBuilder #-}
parseComma :: CharParsing f => f Comma
parseComma = Comma <$ char ','
{-# INLINE parseComma #-}
data Elem f ws a = Elem
{ _elemVal :: a
, _elemTrailing :: f (Comma, ws)
}
deriving (Functor, Foldable, Traversable)
instance (Monoid ws, Applicative f) => Applicative (Elem f ws) where
pure a = Elem a (pure (Comma, mempty))
(Elem atob _) <*> (Elem a t') = Elem (atob a) t'
instance Functor f => Bifunctor (Elem f) where
bimap f g (Elem a t) = Elem (g a) (fmap (fmap f) t)
instance Foldable f => Bifoldable (Elem f) where
bifoldMap f g (Elem a t) = g a `mappend` foldMap (foldMap f) t
instance Traversable f => Bitraversable (Elem f) where
bitraverse f g (Elem a t) = Elem <$> g a <*> traverse (traverse f) t
class HasElem c f ws a | c -> f ws a where
elem :: Lens' c (Elem f ws a)
elemTrailing :: Lens' c (f (Comma, ws))
{-# INLINE elemTrailing #-}
elemVal :: Lens' c a
{-# INLINE elemVal #-}
elemTrailing = elem . elemTrailing
elemVal = elem . elemVal
instance HasElem (Elem f ws a) f ws a where
{-# INLINE elemTrailing #-}
{-# INLINE elemVal #-}
elem = id
elemTrailing f (Elem x1 x2) = Elem x1 <$> f x2
elemVal f (Elem x1 x2) = (`Elem` x2) <$> f x1
instance (Show1 f, Show ws, Show a) => Show (Elem f ws a) where
showsPrec _ (Elem v t) =
showString "Elem {_elemVal = " . shows v .
showString ", _elemTrailing = " . showsPrec1 0 t . showString "}"
instance (Eq1 f, Eq ws, Eq a) => Eq (Elem f ws a) where
Elem v1 t1 == Elem v2 t2 = v1 == v2 && eq1 t1 t2
floopId :: Monoid ws => Iso' (Identity (Comma,ws)) (Maybe (Comma,ws))
floopId = iso (Just . runIdentity) (pure . fromMaybe (Comma, mempty))
_ElemTrailingIso
:: ( Monoid ws
, Monoid ws'
)
=> Iso (Elem Identity ws a) (Elem Identity ws' a') (Elem Maybe ws a) (Elem Maybe ws' a')
_ElemTrailingIso = iso
(\(Elem a t) -> Elem a (t ^. floopId))
(\(Elem a t) -> Elem a (t ^. from floopId))
data Elems ws a = Elems
{ _elemsElems :: Vector (Elem Identity ws a)
, _elemsLast :: Elem Maybe ws a
}
deriving (Eq, Show, Functor, Foldable, Traversable)
instance Bifunctor Elems where
bimap f g (Elems es el) = Elems (fmap (bimap f g) es) (bimap f g el)
instance Bifoldable Elems where
bifoldMap f g (Elems es el) = foldMap (bifoldMap f g) es `mappend` bifoldMap f g el
instance Bitraversable Elems where
bitraverse f g (Elems es el) = Elems <$> traverse (bitraverse f g) es <*> bitraverse f g el
class HasElems c ws a | c -> ws a where
elems :: Lens' c (Elems ws a)
elemsElems :: Lens' c (Vector (Elem Identity ws a))
{-# INLINE elemsElems #-}
elemsLast :: Lens' c (Elem Maybe ws a)
{-# INLINE elemsLast #-}
elemsElems = elems . elemsElems
elemsLast = elems . elemsLast
instance HasElems (Elems ws a) ws a where
{-# INLINE elemsElems #-}
{-# INLINE elemsLast #-}
elems = id
elemsElems f (Elems x1 x2) = fmap (`Elems` x2) (f x1)
elemsLast f (Elems x1 x2) = fmap (Elems x1) (f x2)
instance Monoid ws => Applicative (Elems ws) where
pure a = Elems mempty (pure a)
Elems atobs atob <*> Elems as a = Elems (liftA2 (<*>) atobs as) (atob <*> a)
instance Monoid ws => Semigroup (Elems ws a) where
(<>) (Elems as alast) (Elems bs blast) =
Elems (snoc as (alast ^. from _ElemTrailingIso) <> bs) blast
data CommaSeparated ws a = CommaSeparated ws (Maybe (Elems ws a))
deriving (Eq, Show, Functor, Foldable, Traversable)
instance Bifunctor CommaSeparated where
bimap f g (CommaSeparated ws c) = CommaSeparated (f ws) (fmap (bimap f g) c)
instance Bifoldable CommaSeparated where
bifoldMap f g (CommaSeparated ws c) = f ws `mappend` foldMap (bifoldMap f g) c
instance Bitraversable CommaSeparated where
bitraverse f g (CommaSeparated ws c) = CommaSeparated <$> f ws <*> traverse (bitraverse f g) c
instance Monoid ws => Cons (CommaSeparated ws a) (CommaSeparated ws a) a a where
_Cons = prism (\(a,cs) -> consCommaSep ((Comma,mempty), a) cs) (\c -> note c . over (mapped . _1) (^. _2) $ unconsCommaSep c)
{-# INLINE _Cons #-}
instance Monoid ws => Snoc (CommaSeparated ws a) (CommaSeparated ws a) a a where
_Snoc = prism f g
where
f :: (CommaSeparated ws a, a) -> CommaSeparated ws a
f (cs,a) = over (_CommaSeparated . _2 . _Just)
(\es -> es
& elemsElems %~ flip snoc (es ^. elemsLast . from _ElemTrailingIso)
& elemsLast . elemVal .~ a
) cs
g :: CommaSeparated ws a -> Either (CommaSeparated ws a) (CommaSeparated ws a, a)
g c@(CommaSeparated _ Nothing) = Left c
g (CommaSeparated w (Just es)) = Right
( CommaSeparated w $ createNewElems <$> es ^? elemsElems . _Snoc
, es ^. elemsLast . elemVal
)
where
createNewElems (newEs, newL) = es
& elemsElems .~ newEs
& elemsLast .~ newL ^. _ElemTrailingIso
consElems :: Monoid ws => ((Comma,ws), a) -> Elems ws a -> Elems ws a
consElems (ews,a) e = e & elemsElems %~ cons (Elem a (Identity ews))
{-# INLINE consElems #-}
unconsElems :: Monoid ws => Elems ws a -> ((Maybe (Comma,ws), a), Maybe (Elems ws a))
unconsElems e = maybe (e', Nothing) (\(em, ems) -> (idT em, Just $ e & elemsElems .~ ems)) es'
where
es' = e ^? elemsElems . _Cons
e' = (e ^. elemsLast . elemTrailing, e ^. elemsLast . elemVal)
idT x = (x ^. elemTrailing . to (Just . runIdentity), x ^. elemVal)
{-# INLINE unconsElems #-}
instance (Monoid ws, Semigroup ws) => Semigroup (CommaSeparated ws a) where
(CommaSeparated wsA a) <> (CommaSeparated wsB b) = CommaSeparated (wsA <> wsB) (a <> b)
instance (Monoid ws, Semigroup ws) => Monoid (CommaSeparated ws a) where
mempty = CommaSeparated mempty Nothing
mappend = (<>)
instance Monoid ws => Filterable (CommaSeparated ws) where
mapMaybe _ (CommaSeparated ws Nothing) = CommaSeparated ws Nothing
mapMaybe f (CommaSeparated ws (Just (Elems es el))) = CommaSeparated ws newElems
where
newElems = case traverse f el of
Nothing -> (\(v,l) -> Elems v (l ^. _ElemTrailingIso)) <$> unsnoc (mapMaybe (traverse f) es)
Just l' -> Just $ Elems (mapMaybe (traverse f) es) l'
instance Monoid ws => Witherable (CommaSeparated ws) where
_CommaSeparated :: Iso (CommaSeparated ws a) (CommaSeparated ws' b) (ws, Maybe (Elems ws a)) (ws', Maybe (Elems ws' b))
_CommaSeparated = iso (\(CommaSeparated ws a) -> (ws,a)) (uncurry CommaSeparated)
{-# INLINE _CommaSeparated #-}
consCommaSep :: Monoid ws => ((Comma,ws),a) -> CommaSeparated ws a -> CommaSeparated ws a
consCommaSep (ews,a) = over (_CommaSeparated . _2) (pure . maybe new (consElems (ews,a)))
where new = Elems mempty (Elem a Nothing)
{-# INLINE consCommaSep #-}
unconsCommaSep :: Monoid ws => CommaSeparated ws a -> Maybe ((Maybe (Comma,ws), a), CommaSeparated ws a)
unconsCommaSep (CommaSeparated ws es) = over _2 (CommaSeparated ws) . unconsElems <$> es
{-# INLINE unconsCommaSep #-}
instance (Semigroup ws, Monoid ws) => AsEmpty (CommaSeparated ws a) where
_Empty = nearly mempty (^. _CommaSeparated . _2 . to (isn't _Nothing))
type instance IxValue (CommaSeparated ws a) = a
type instance Index (CommaSeparated ws a) = Int
instance Ixed (CommaSeparated ws a) where
ix _ _ c@(CommaSeparated _ Nothing) = pure c
ix i f c@(CommaSeparated w (Just es))
| i == 0 && es ^. elemsElems . to V.null =
CommaSeparated w . Just <$> (es & elemsLast . traverse %%~ f)
| i <= es ^. elemsElems . to length =
CommaSeparated w . Just <$> (es & elemsElems . ix i . traverse %%~ f)
| otherwise = pure c
fromList :: (Monoid ws, Semigroup ws) => [a] -> CommaSeparated ws a
fromList = foldr cons mempty
toList :: CommaSeparated ws a -> [a]
toList = maybe [] g . (^. _CommaSeparated . _2) where
g e = snoc (e ^.. elemsElems . traverse . elemVal) (e ^. elemsLast . elemVal)
{-# INLINE toList #-}
parseCommaTrailingMaybe
:: CharParsing f
=> f ws
-> f (Maybe (Comma, ws))
parseCommaTrailingMaybe =
C.optional . liftA2 (,) parseComma
commaTrailingBuilder
:: Foldable f
=> (ws -> Builder)
-> f (Comma, ws)
-> Builder
commaTrailingBuilder wsB =
foldMap ((commaBuilder <>) . wsB . snd)
commaSeparatedBuilder
:: forall ws a. Char
-> Char
-> (ws -> Builder)
-> (a -> Builder)
-> CommaSeparated ws a
-> Builder
commaSeparatedBuilder op fin wsB aB (CommaSeparated lws sepElems) =
BB.charUtf8 op <> wsB lws <> maybe mempty buildElems sepElems <> BB.charUtf8 fin
where
elemBuilder
:: Foldable f
=> Elem f ws a -> Builder
elemBuilder (Elem e eTrailing) =
aB e <> commaTrailingBuilder wsB eTrailing
buildElems (Elems es elst) =
foldMap elemBuilder es <> elemBuilder elst
parseCommaSeparatedElems
:: ( Monad f
, CharParsing f
)
=> f ws
-> f a
-> f (Elems ws a)
parseCommaSeparatedElems ws a = do
hd <- a
sep <- parseCommaTrailingMaybe ws
maybe (pure $ Elems mempty (Elem hd sep)) (go mempty . (hd,)) sep
where
idElem e = Elem e . Identity
fin cels lj sp =
pure $ Elems cels (Elem lj sp)
go commaElems (lastJ, lastSep) = do
mJ <- C.optional a
case mJ of
Nothing -> fin commaElems lastJ (Just lastSep)
Just j -> do
msep <- parseCommaTrailingMaybe ws
let commaElems' = snoc commaElems $ idElem lastJ lastSep
maybe (fin commaElems' j Nothing) (go commaElems' . (j,)) msep
parseCommaSeparated
:: ( Monad f
, CharParsing f
)
=> f open
-> f close
-> f ws
-> f a
-> f (CommaSeparated ws a)
parseCommaSeparated op fin ws a =
op *> (
CommaSeparated <$> ws <*> asum
[ Nothing <$ fin
, Just <$> parseCommaSeparatedElems ws a <* fin
]
)