module Binary ( toBinary , fromBinary , toBitString , fromBitString ) where import Data.Binary ( decode , encode ) import qualified Data.BitString as Bit import qualified Data.ByteString.Lazy as Byte import Data.Int ( Int8 ) import Helper toBinary :: Expression -> String toBinary (Bruijn x ) = (replicate (x + 1) '1') ++ "0" toBinary (Abstraction e ) = "00" ++ toBinary e toBinary (Application exp1 exp2) = "01" ++ (toBinary exp1) ++ (toBinary exp2) toBinary _ = "" -- shouldn't happen fromBinary' :: String -> (Expression, String) fromBinary' = \case '0' : '0' : rst -> let (e, es) = fromBinary' rst in (Abstraction e, es) '0' : '1' : rst -> let (exp1, rst1) = fromBinary' rst (exp2, rst2) = fromBinary' rst1 in (Application exp1 exp2, rst2) '1' : '0' : rst -> binaryBruijn rst '1' : '1' : rst -> binaryBruijn rst _ -> (Bruijn (-1), "") where binaryBruijn rst = let idx = length . takeWhile (== '1') in case length rst of 0 -> (Bruijn $ idx rst, "") _ -> (Bruijn $ idx rst, drop ((idx rst) + 1) rst) fromBinary :: String -> Expression fromBinary = fst . fromBinary' -- 1 byte indicating bit-padding at end + n bytes filled with bits -- TODO: technically only 1 nibble is needed (versioning/sth?) toBitString :: String -> Bit.BitString toBitString str = Bit.concat [ Bit.bitString $ Byte.toStrict $ encode (fromIntegral $ length str `mod` 8 :: Int8) , Bit.fromList $ map (\case '0' -> False '1' -> True _ -> error "invalid bit" ) str ] -- TODO: Fix this fromBitString :: Bit.BitString -> String fromBitString bits = map (\case False -> '0' True -> '1' ) $ Bit.toList $ Bit.take (Bit.length bits - pad bits) $ Bit.drop 8 bits where pad = decode . Bit.realizeBitStringLazy . Bit.take 8