import qualified Control.Monad as Monad import qualified Data.ByteString as ByteString import qualified GHC.Clock as Clock import qualified Rattletrap import qualified System.Exit as Exit import qualified System.FilePath as FilePath import qualified Test.HUnit as Test import qualified Text.Printf as Printf main :: IO () main = runTests makeTests runTests :: Test.Test -> IO () runTests test = do Rattletrap.rattletrap "" ["--schema", "--output", FilePath.combine directory "schema.json"] (result, elapsed) <- withElapsed $ Test.runTestTT test Printf.printf "Total time: %.3f seconds\n" elapsed Monad.when (Test.errors result > 0 || Test.failures result > 0) Exit.exitFailure makeTests :: Test.Test makeTests = Test.TestList $ fmap toTest replays toTest :: (String, String) -> Test.Test toTest (uuid, name) = Test.TestLabel (toLabel uuid name) . Test.TestCase $ toAssertion uuid toLabel :: String -> String -> String toLabel uuid name = uuid <> ": " <> name toAssertion :: String -> Test.Assertion toAssertion uuid = do let inputFile = FilePath.combine "replays" $ uuid <> ".replay" jsonFile = FilePath.combine directory $ uuid <> ".json" outputFile = FilePath.combine directory $ uuid <> ".replay" input <- ByteString.readFile inputFile decode inputFile jsonFile encode jsonFile outputFile output <- ByteString.readFile outputFile Monad.when (output /= input) $ Test.assertFailure "output does not match input" decode :: FilePath -> FilePath -> IO () decode input output = Rattletrap.rattletrap "" ["--compact", "--input", input, "--output", output] encode :: FilePath -> FilePath -> IO () encode input output = Rattletrap.rattletrap "" ["--input", input, "--output", output] withElapsed :: IO a -> IO (a, Double) withElapsed action = do before <- Clock.getMonotonicTime result <- action after <- Clock.getMonotonicTime pure (result, after - before) directory :: FilePath directory = "output" replays :: [(String, String)] replays = [ ("0008", "a flip time") , ("000b", "nintendo switch") , ("0416", "v1.78 demolition") , ("07e9", "a game mode before Neo Tokyo") , ("0ad2", "some Latin-1 text") , ("0e76", "v1.95 rumble") , ("1205", "rumble mode") , ("160c", "a dedicated server IP") , ("16d5", "new property types") , ("18d6", "an online loadout attribute") , ("1a12", "overtime") , ("1ae4", "a game time") , ("1bc2", "no padding after the frames") , ("1d1d", "a camera pitch") , ("1ec9", "a V1.63 match") , ("1ef9", "a private hoops match") , ("1f37", "splitscreen players") , ("2114", "a match save") , ("21a8", "v1.66") , ("2266", "dropshot") , ("22ba", "a vote to forfeit") , ("27b6", "some UTF-16 text") , ("29f5", "frames") , ("2cfe", "a new playstation id") , ("3381", "patch 1.37") , ("372d", "a camera yaw attribute") , ("383e", "older unknown content field") , ("387f", "a frozen attribute") , ("3abd", "rlcs") , ("3ea1", "a custom team name") , ("4126", "a game mode after Neo Tokyo") , ("419a", "a club match") , ("42f0", "reservations after Neo Tokyo") , ("42f2", "anniversary ball") , ("4bc3", "with timed out attribute") , ("504e", "some messages") , ("520e", "no pickup attribute") , ("524f", "quat edge case") , ("52aa", "a match-ending attribute") , ("540d", "a demolish attribute") , ("551c", "private match settings") , ("5a06", "esports items") , ("6210", "different player history key") , ("6320", "a forfeit attribute") , ("6688", "a malformed byte property") , ("6b0d", "patch 1.37") , ("6d1b", "a flip right") , ("6f7c", "a map with numbers") , ("7083", "weird basketball capitalization") , ("7109", "a boost modifier") , ("7256", "special edition") , ("75ce", "primary and secondary titles") , ("7bf6", "an online loadouts attribute") , ("81d1", "gridiron") , ("89cb", "remote user data") , ("8ae5", "new painted items") , ("92a6", "with server performance state") , ("946f", "patch 1.43") , ("9704", "a batarang") , ("98e5", "a player using behind view") , ("9a2c", "ghost hunt") , ("9e35", "spike rush") , ("9eaa", "newer replay without trailing bytes") , ("a09e", "a tournament") , ("a128", "a round count down") , ("a184", "max score") , ("a1c0", "epic system id") , ("a52f", "some more mutators") , ("a558", "extended explosion data") , ("a671", "a waiting player") , ("a676", "new user color") , ("a7f0", "a ready attribute") , ("a9df", "salty shores patch 1.45") , ("aa70", "patch 1.50 - TitleID attribute") , ("afb1", "patch 1.37") , ("b9f9", "a party leader") , ("c14f", "some mutators") , ("c23b", "new psynet id") , ("c837", "a spectator") , ("cc4c", "after Starbase ARC") , ("d044", "hoops mutators") , ("d1d5", "v1.68") , ("d236", "rlcs s2") , ("d428", "a private hockey match") , ("d44c", "ranked tournament") , ("d52e", "psynet system id") , ("d7fb", "an explosion attribute") , ("d818", "heatseeker") , ("db70", "new lag indicator") , ("dcab", "weird ball attribute value") , ("dcb3", "a pawn type attribute") , ("dd14", "v1.88") , ("de56", "a problematic product attribute") , ("e80d", "unlimited time") , ("e978", "distracted") , ("eae3", "an actor/object ID collision") , ("eae8", "custom team colors") , ("edbb", "remote role") , ("f299", "a location attribute") , ("f7b9", "a hockey game event") , ("f811", "no frames") , ("fdc7", "an MVP") , ("ffb7", "with difficulty") ]