{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE TemplateHaskell #-} module Main where import Data.Foldable qualified import Data.Maybe (fromMaybe) import Data.Text (Text, pack) import Data.Text qualified as T import Data.Text.IO qualified as Text import GitHash (GitInfo, giBranch, giHash, tGitInfoCwdTry) import Options.Applicative import Swarm.App (appMain) import Swarm.DocGen (EditorType (..), GenerateDocs (..), PageAddress (..), SheetType (..), generateDocs) import Swarm.Language.LSP (lspMain) import Swarm.Language.Pipeline (processTerm) import Swarm.TUI.Model (AppOpts (..)) import Swarm.Version import Swarm.Web (defaultPort) import System.Exit (exitFailure, exitSuccess) import System.IO (hPrint, stderr) gitInfo :: Maybe GitInfo gitInfo = either (const Nothing) Just ($$tGitInfoCwdTry) commitInfo :: String commitInfo = case gitInfo of Nothing -> "" Just git -> " (" <> giBranch git <> "@" <> take 10 (giHash git) <> ")" data CLI = Run AppOpts | Format Input | DocGen GenerateDocs | LSP | Version cliParser :: Parser CLI cliParser = subparser ( mconcat [ command "format" (info (format <**> helper) (progDesc "Format a file")) , command "generate" (info (DocGen <$> docgen <**> helper) (progDesc "Generate docs")) , command "lsp" (info (pure LSP) (progDesc "Start the LSP")) , command "version" (info (pure Version) (progDesc "Get current and upstream version.")) ] ) <|> Run <$> (AppOpts <$> seed <*> scenario <*> run <*> autoplay <*> cheat <*> webPort <*> pure gitInfo) where format :: Parser CLI format = (Format Stdin <$ switch (long "stdin" <> help "Read code from stdin")) <|> (Format . File <$> strArgument (metavar "FILE")) docgen :: Parser GenerateDocs docgen = subparser . mconcat $ [ command "recipes" (info (pure RecipeGraph) $ progDesc "Output graphviz dotfile of entity dependencies based on recipes") , command "editors" (info (EditorKeywords <$> editor <**> helper) $ progDesc "Output editor keywords") , command "cheatsheet" (info (CheatSheet <$> address <*> cheatsheet <**> helper) $ progDesc "Output nice Wiki tables") ] editor :: Parser (Maybe EditorType) editor = Data.Foldable.asum [ pure Nothing , Just VSCode <$ switch (long "code" <> help "Generate for the VS Code editor") , Just Emacs <$ switch (long "emacs" <> help "Generate for the Emacs editor") ] address :: Parser PageAddress address = let replace a b = T.unpack . T.replace a b . T.pack opt n = fmap (fromMaybe "") . optional $ option str ( long n <> metavar "ADDRESS" <> help ("Set the address of " <> replace "-" " " n <> ". Default no link.") ) in PageAddress <$> opt "entities-page" <*> opt "commands-page" <*> opt "capabilities-page" <*> opt "recipes-page" cheatsheet :: Parser (Maybe SheetType) cheatsheet = Data.Foldable.asum [ pure Nothing , Just Entities <$ switch (long "entities" <> help "Generate entities page (uses data from entities.yaml)") , Just Recipes <$ switch (long "recipes" <> help "Generate recipes page (uses data from recipes.yaml)") , Just Capabilities <$ switch (long "capabilities" <> help "Generate capabilities page (uses entity map)") , Just Commands <$ switch (long "commands" <> help "Generate commands page (uses constInfo, constCaps and inferConst)") ] seed :: Parser (Maybe Int) seed = optional $ option auto (long "seed" <> short 's' <> metavar "INT" <> help "Seed to use for world generation") webPort :: Parser (Maybe Int) webPort = optional $ option auto ( long "web" <> metavar "PORT" <> help ("Set the web service port (or disable it with 0). Default to " <> show defaultPort <> ".") ) scenario :: Parser (Maybe String) scenario = optional $ strOption (long "scenario" <> short 'c' <> metavar "FILE" <> help "Name of a scenario to load") run :: Parser (Maybe String) run = optional $ strOption (long "run" <> short 'r' <> metavar "FILE" <> help "Run the commands in a file at startup") autoplay :: Parser Bool autoplay = switch (long "autoplay" <> short 'a' <> help "Automatically run the solution defined in the scenario, if there is one. Mutually exclusive with --run.") cheat :: Parser Bool cheat = switch (long "cheat" <> short 'x' <> help "Enable cheat mode. This allows toggling Creative Mode with Ctrl+v and unlocks \"Testing\" scenarios in the menu.") cliInfo :: ParserInfo CLI cliInfo = info (cliParser <**> helper) ( header ("Swarm game - " <> version <> commitInfo) <> progDesc "To play the game simply run without any command." <> fullDesc ) data Input = Stdin | File FilePath getInput :: Input -> IO Text getInput Stdin = Text.getContents getInput (File fp) = Text.readFile fp showInput :: Input -> Text showInput Stdin = "(input)" showInput (File fp) = pack fp -- | Utility function to validate and format swarm-lang code formatFile :: Input -> IO () formatFile input = do content <- getInput input case processTerm content of Right _ -> do Text.putStrLn content exitSuccess Left e -> do Text.hPutStrLn stderr $ showInput input <> ":" <> e exitFailure showVersion :: IO () showVersion = do putStrLn $ "Swarm game - " <> version <> commitInfo up <- getNewerReleaseVersion gitInfo either (hPrint stderr) (putStrLn . ("New upstream release: " <>)) up main :: IO () main = do cli <- execParser cliInfo case cli of Run opts -> appMain opts DocGen g -> generateDocs g Format fo -> formatFile fo LSP -> lspMain Version -> showVersion