-- SPDX-FileCopyrightText: Copyright (c) 2025 Objectionary.com -- SPDX-License-Identifier: MIT module DataizeSpec (spec) where import AST import Control.Monad import Dataize (DataizeContext (DataizeContext), dataize, dataize', morph) import Deps (dontSaveStep) import Functions (buildTerm) import Parser (parseExpressionThrows, parseProgramThrows) import Printer (printProgram) import Rewriter (Rewritten) import Test.Hspec defaultDataizeContext :: Program -> DataizeContext defaultDataizeContext prog = DataizeContext prog 25 25 False buildTerm dontSaveStep test :: (Eq a, Show a) => ((Expression, [Rewritten]) -> DataizeContext -> IO (Maybe a, [Rewritten])) -> [(String, Expression, Expression, Maybe a)] -> Spec test func useCases = forM_ useCases $ \(desc, input, prog, output) -> it desc $ do (res, _) <- func (input, []) (defaultDataizeContext (Program prog)) res `shouldBe` output test' :: (Eq a, Show a) => ((Expression, [Rewritten]) -> DataizeContext -> IO (a, [Rewritten])) -> [(String, Expression, Expression, a)] -> Spec test' func useCases = forM_ useCases $ \(desc, input, prog, output) -> it desc $ do (res, _) <- func (input, [(Program prog, Nothing)]) (defaultDataizeContext (Program prog)) res `shouldBe` output testDataize :: [(String, String, String, Bytes)] -> Spec testDataize useCases = forM_ useCases $ \(name, loc, prog, res) -> it name $ do prog' <- parseProgramThrows prog loc' <- parseExpressionThrows loc putStrLn (printProgram prog') (value, _) <- dataize loc' (defaultDataizeContext prog') value `shouldBe` Just res spec :: Spec spec = do describe "morph" $ test' morph [ ("[[ D> 00- ]] => [[ D> 00- ]]", ExFormation [BiDelta (BtOne "00")], ExGlobal, ExFormation [BiDelta (BtOne "00")]) , ("T => T", ExTermination, ExGlobal, ExTermination) , ("$ => X", ExThis, ExGlobal, ExTermination) , ("Q => X", ExGlobal, ExGlobal, ExTermination) , ( "Q.x (Q -> [[ x -> [[]] ]]) => [[]]" , ExDispatch ExGlobal (AtLabel "x") , ExFormation [BiTau (AtLabel "x") (ExFormation [])] , ExFormation [] ) ] describe "dataize" $ test dataize' [ ("[[ D> 00- ]] => 00-", ExFormation [BiDelta (BtOne "00")], ExGlobal, Just (BtOne "00")) , ("T => X", ExTermination, ExGlobal, Nothing) , ( "[[ @ -> [[ D> 00-]] ]] => 00-" , ExFormation [BiTau AtPhi (ExFormation [BiDelta (BtOne "00"), BiVoid AtRho]), BiVoid AtRho] , ExGlobal , Just (BtOne "00") ) , ( "[[ x -> [[ D> 01- ]] ]].x => 01-" , ExDispatch (ExFormation [BiTau (AtLabel "x") (ExFormation [BiDelta (BtOne "01"), BiVoid AtRho]), BiVoid AtRho]) (AtLabel "x") , ExGlobal , Just (BtOne "01") ) , ( "[[ @ -> [[ x -> [[ D> 01-, y -> ? ]](y -> [[ ]]) ]].x ]] => 01-" , ExFormation [ BiTau AtPhi ( ExDispatch ( ExFormation [ BiTau (AtLabel "x") ( ExApplication ( ExFormation [ BiDelta (BtOne "01") , BiVoid (AtLabel "y") , BiVoid AtRho ] ) (BiTau (AtLabel "y") (ExFormation [])) ) ] ) (AtLabel "x") ) ] , ExGlobal , Just (BtOne "01") ) ] testDataize [ ( "5.plus(5)" , "Q" , unlines [ "Q -> [[" , " org -> [[" , " eolang -> [[" , " bytes -> [[" , " data -> ?," , " @ -> $.data" , " ]]," , " number -> [[" , " as-bytes -> ?," , " @ -> $.as-bytes," , " plus -> [[ x -> ?, L> L_org_eolang_number_plus ]]" , " ]]" , " ]]" , " ]]," , " @ -> 5.plus(5)" , "]]" ] , BtMany ["40", "24", "00", "00", "00", "00", "00", "00"] ) , ( "Fahrenheit" , "Q" , unlines [ "Q -> [[" , " org -> [[" , " eolang -> [[" , " bytes -> [[" , " data -> ?," , " @ -> $.data" , " ]]," , " number -> [[" , " as-bytes -> ?," , " @ -> $.as-bytes," , " plus -> [[ x -> ?, L> L_org_eolang_number_plus ]]," , " times -> [[ x -> ?, L> L_org_eolang_number_times ]]" , " ]]" , " ]]" , " ]]," , " @ -> $.c.times(1.8).plus(32)," , " c -> 25" , "]]" ] , BtMany ["40", "53", "40", "00", "00", "00", "00", "00"] ) , ( "Factorial" , "Q" , unlines [ "Q -> [[" , " org -> [[" , " eolang -> [[" , " bytes -> [[" , " data -> ?," , " @ -> $.data" , " ]]," , " number -> [[" , " as-bytes -> ?," , " @ -> $.as-bytes," , " times -> [[ x -> ?, L> L_org_eolang_number_times ]]," , " plus -> [[ x -> ?, L> L_org_eolang_number_plus ]]," , " eq -> [[ x -> ?, y -> ?, L> L_org_eolang_number_eq ]]" , " ]]" , " ]]" , " ]]," , " fac -> [[" , " x -> ?," , " @ -> $.x.eq(" , " 1," , " $.x.times($.^.fac($.x.plus(-1)))" , " )" , " ]]," , " @ -> $.fac(3)" , "]]" ] , BtMany ["40", "18", "00", "00", "00", "00", "00", "00"] ) , ( "Located" , "Q.foo.bar" , unlines [ "Q -> [[" , " foo -> [[" , " bar -> [[" , " @ -> Q.x" , " ]]" , " ]]," , " x -> [[ D> 42- ]]" , "]]" ] , BtOne "42" ) ]