module ZM.Types (
module Data.Model.Types,
AbsTypeModel,
AbsType,
AbsRef(..),
absRef,
AbsADT,
AbsEnv,
ADTRef(..),
Identifier(..),
UnicodeLetter(..),
UnicodeLetterOrNumberOrLine(..),
UnicodeSymbol(..),
SHA3_256_6(..),
SHAKE128_48(..),
NonEmptyList(..),
nonEmptyList,
Word7,
FlatEncoding(..), UTF8Encoding(..), UTF16LEEncoding(..), NoEncoding(..)
,TypedDecoded,
TypedDecodeException(..),
NFData(),
Flat,
ZigZag(..),
LeastSignificantFirst(..),
MostSignificantFirst(..),
Value(..),
Label(..),
label,
) where
import Control.DeepSeq
import Control.Exception
import qualified Data.ByteString as B
import Data.Char
import Data.Digest.Keccak
import Data.Flat
import Data.Foldable (toList)
import qualified Data.ListLike.String as L
import qualified Data.Map as M
import Data.Model hiding (Name)
import Data.Model.Types hiding (Name)
import ZM.Model()
import ZM.Type.BLOB
import ZM.Type.NonEmptyList
import ZM.Type.Words (LeastSignificantFirst (..),
MostSignificantFirst (..), Word7,
ZigZag (..))
import Data.Word
type AbsType = Type AbsRef
data AbsRef = AbsRef (SHAKE128_48 AbsADT) deriving (Eq, Ord, Show, NFData, Generic, Flat)
absRef :: Flat r => r -> AbsRef
absRef a = let [w1,w2,w3,w4,w5,w6] = B.unpack . shake_128 6 . flat $ a
in AbsRef $ SHAKE128_48 w1 w2 w3 w4 w5 w6
data SHA3_256_6 a = SHA3_256_6 Word8 Word8 Word8 Word8 Word8 Word8
deriving (Eq, Ord, Show, NFData, Generic, Flat)
data SHAKE128_48 a = SHAKE128_48 Word8 Word8 Word8 Word8 Word8 Word8
deriving (Eq, Ord, Show, NFData, Generic, Flat)
type AbsADT = ADT Identifier Identifier (ADTRef AbsRef)
type AbsTypeModel = TypeModel Identifier Identifier (ADTRef AbsRef) AbsRef
type AbsEnv = TypeEnv Identifier Identifier (ADTRef AbsRef) AbsRef
data ADTRef r = Var Word8
| Rec
| Ext r
deriving (Eq, Ord, Show, NFData, Generic, Functor, Foldable, Traversable ,Flat)
data Identifier = Name UnicodeLetter [UnicodeLetterOrNumberOrLine]
| Symbol (NonEmptyList UnicodeSymbol)
deriving (Eq, Ord, Show, NFData, Generic, Flat)
instance Flat [UnicodeLetterOrNumberOrLine]
instance L.StringLike Identifier where
fromString = identifier
toString (Name (UnicodeLetter h) t) = h : map (\(UnicodeLetterOrNumberOrLine s) -> s) t
toString (Symbol l) = map (\(UnicodeSymbol s) -> s) . toList $ l
data UnicodeLetterOrNumberOrLine = UnicodeLetterOrNumberOrLine Char deriving (Eq, Ord, Show, NFData, Generic, Flat)
data UnicodeLetter = UnicodeLetter Char deriving (Eq, Ord, Show, NFData, Generic, Flat)
data UnicodeNumber = UnicodeNumber Char deriving (Eq, Ord, Show, NFData, Generic, Flat)
data UnicodeSymbol = UnicodeSymbol Char deriving (Eq, Ord, Show, NFData, Generic, Flat)
identifier :: String -> Identifier
identifier [] = error "identifier cannot be empty"
identifier s@(h:t) = if isLetter h
then Name (asLetter h) (map asLetterOrNumber t)
else Symbol (nonEmptyList $ map asSymbol s)
asSymbol :: Char -> UnicodeSymbol
asSymbol c | isSymbol c = UnicodeSymbol c
| otherwise = error . unwords $ [show c,"is not an Unicode Symbol"]
asLetter :: Char -> UnicodeLetter
asLetter c | isLetter c = UnicodeLetter c
| otherwise = error . unwords $ [show c,"is not an Unicode Letter"]
asLetterOrNumber :: Char -> UnicodeLetterOrNumberOrLine
asLetterOrNumber c | isLetter c || isNumber c || isAlsoOK c = UnicodeLetterOrNumberOrLine c
| otherwise = error . unwords $ [show c,"is not an Unicode Letter or Number"]
isAlsoOK :: Char -> Bool
isAlsoOK '_' = True
isAlsoOK _ = False
data Value = Value {valType::AbsType
,valName::String
,valBits::[Bool]
,valFields::[Value]
} deriving (Eq,Ord,Show,NFData, Generic, Flat)
data Label a label = Label a (Maybe label) deriving (Eq, Ord, Show, NFData, Generic, Flat)
label :: (Functor f, Ord k) => M.Map k a -> (a -> l) -> f k -> f (Label k l)
label env f o = (\ref -> Label ref (f <$> M.lookup ref env)) <$> o
type TypedDecoded a = Either TypedDecodeException a
data TypedDecodeException = UnknownMetaModel AbsType
| WrongType {expectedType::AbsType,actualType::AbsType}
| DecodeError DecodeException deriving (Show,Eq,Ord)
instance Exception TypedDecodeException
instance (Flat adtName, Flat consName, Flat inRef, Flat exRef,Ord exRef) => Flat (TypeModel adtName consName inRef exRef)
instance (Flat a,Flat b,Flat c) => Flat (ADT a b c)
instance (Flat a,Flat b) => Flat (ConTree a b)
instance (Flat a,Flat b) => Flat [(a,Type b)]
instance Flat a => Flat [Type a]
instance Flat a => Flat (Type a)
instance Flat a => Flat (TypeRef a)
instance (Model a,Model b,Model c) => Model (ADT a b c)
instance (Model a,Model b) => Model (ConTree a b)
instance Model a => Model (ADTRef a)
instance Model a => Model (Type a)
instance Model a => Model (TypeRef a)
instance (Model adtName, Model consName, Model inRef, Model exRef) => Model (TypeModel adtName consName inRef exRef)
instance Model Identifier
instance Model UnicodeLetter
instance Model UnicodeLetterOrNumberOrLine
instance Model UnicodeSymbol
instance Model a => Model (SHA3_256_6 a)
instance Model a => Model (SHAKE128_48 a)
instance Model AbsRef
instance Model a => Model (PostAligned a)