module AST.Variable where
import Data.Binary
import Control.Applicative ((<$>), (<*>))
import Text.PrettyPrint as P
import qualified AST.Helpers as Help
import AST.PrettyPrint
newtype Raw = Raw String
deriving (Eq,Ord,Show)
data Home
= BuiltIn
| Module !String
| Local
deriving (Eq,Ord,Show)
data Canonical = Canonical
{ home :: !Home
, name :: !String
} deriving (Eq,Ord,Show)
local :: String -> Canonical
local x = Canonical Local x
builtin :: String -> Canonical
builtin x = Canonical BuiltIn x
is :: String -> String -> Canonical -> Bool
is home name var =
var == Canonical (Module home) name
isJson :: Canonical -> Bool
isJson = is "Json" "Value"
isMaybe :: Canonical -> Bool
isMaybe = is "Maybe" "Maybe"
isArray :: Canonical -> Bool
isArray = is "Array" "Array"
isSignal :: Canonical -> Bool
isSignal = is "Signal" "Signal"
isList :: Canonical -> Bool
isList v = v == Canonical BuiltIn "_List"
isTuple :: Canonical -> Bool
isTuple v =
case v of
Canonical BuiltIn name -> Help.isTuple name
_ -> False
isPrimitive :: Canonical -> Bool
isPrimitive v =
case v of
Canonical BuiltIn name -> name `elem` ["Int","Float","String","Bool"]
_ -> False
isPrim :: String -> Canonical -> Bool
isPrim prim v =
case v of
Canonical BuiltIn name -> name == prim
_ -> False
isText :: Canonical -> Bool
isText = is "Text" "Text"
class ToString a where
toString :: a -> String
instance ToString Raw where
toString (Raw x) = x
instance ToString Canonical where
toString (Canonical home name) =
case home of
BuiltIn -> name
Module path -> path ++ "." ++ name
Local -> name
data Listing a = Listing
{ _explicits :: [a]
, _open :: Bool
} deriving (Eq,Ord,Show)
openListing :: Listing a
openListing = Listing [] True
data Value
= Value !String
| Alias !String
| ADT !String !(Listing String)
deriving (Eq,Ord,Show)
instance Pretty Raw where
pretty (Raw var) = variable var
instance Pretty Canonical where
pretty var = P.text (toString var)
instance Pretty a => Pretty (Listing a) where
pretty (Listing explicits open) =
P.parens (commaCat (map pretty explicits ++ dots))
where
dots = [if open then P.text ".." else P.empty]
instance Pretty Value where
pretty portable =
case portable of
Value name -> P.text name
Alias name -> P.text name
ADT name ctors ->
P.text name <> pretty (ctors { _explicits = map P.text (_explicits ctors) })
instance Binary Canonical where
put (Canonical home name) =
case home of
BuiltIn -> putWord8 0 >> put name
Module path -> putWord8 1 >> put path >> put name
Local -> putWord8 2 >> put name
get = do tag <- getWord8
case tag of
0 -> Canonical BuiltIn <$> get
1 -> Canonical . Module <$> get <*> get
2 -> Canonical Local <$> get
_ -> error "Unexpected tag when deserializing canonical variable"
instance Binary Value where
put portable =
case portable of
Value name -> putWord8 0 >> put name
Alias name -> putWord8 1 >> put name
ADT name ctors -> putWord8 2 >> put name >> put ctors
get = do tag <- getWord8
case tag of
0 -> Value <$> get
1 -> Alias <$> get
2 -> ADT <$> get <*> get
_ -> error "Error reading valid import/export information from serialized string"
instance (Binary a) => Binary (Listing a) where
put (Listing explicits open) =
put explicits >> put open
get = Listing <$> get <*> get