mirror of
https://github.com/anoma/juvix.git
synced 2025-01-05 22:46:08 +03:00
135 lines
3.8 KiB
Haskell
135 lines
3.8 KiB
Haskell
module Commands.Init where
|
|
|
|
import Data.Text qualified as Text
|
|
import Data.Versions
|
|
import Data.Yaml (encodeFile)
|
|
import Juvix.Compiler.Pipeline.Package
|
|
import Juvix.Data.Effect.Fail.Extra qualified as Fail
|
|
import Juvix.Extra.Paths
|
|
import Juvix.Prelude
|
|
import Juvix.Prelude.Pretty
|
|
import Text.Megaparsec (Parsec)
|
|
import Text.Megaparsec qualified as P
|
|
import Text.Megaparsec.Char qualified as P
|
|
|
|
type Err = Text
|
|
|
|
parse :: Parsec Void Text a -> Text -> Either Err a
|
|
parse p t = mapLeft ppErr (P.runParser p "<stdin>" t)
|
|
|
|
ppErr :: P.ParseErrorBundle Text Void -> Text
|
|
ppErr = pack . errorBundlePretty
|
|
|
|
init :: forall r. Members '[Embed IO] r => Sem r ()
|
|
init = do
|
|
checkNotInProject
|
|
say "✨ Your next Juvix adventure is about to begin! ✨"
|
|
say "I will help you set it up"
|
|
pkg <- getPackage
|
|
say ("creating " <> pack juvixYamlFile)
|
|
embed (encodeFile juvixYamlFile pkg)
|
|
say "you are all set"
|
|
|
|
checkNotInProject :: forall r. Members '[Embed IO] r => Sem r ()
|
|
checkNotInProject =
|
|
whenM (embed (doesFileExist juvixYamlFile)) err
|
|
where
|
|
err :: Sem r ()
|
|
err = do
|
|
say "You are already in a Juvix project"
|
|
embed exitFailure
|
|
|
|
getPackage :: forall r. Members '[Embed IO] r => Sem r Package
|
|
getPackage = do
|
|
tproj <- getProjName
|
|
say "Tell me the version of your project [leave empty for 0.0.0]"
|
|
tversion <- getVersion
|
|
return
|
|
Package
|
|
{ _packageName = Just tproj,
|
|
_packageVersion = Just (prettySemVer tversion)
|
|
}
|
|
|
|
getProjName :: forall r. Members '[Embed IO] r => Sem r Text
|
|
getProjName = do
|
|
d <- getDefault
|
|
let defMsg :: Text
|
|
defMsg = case d of
|
|
Nothing -> mempty
|
|
Just d' -> " [leave empty for '" <> d' <> "']"
|
|
say
|
|
( "Tell me the name of your project"
|
|
<> defMsg
|
|
<> " (lower case letters, numbers and dashes are allowed): "
|
|
)
|
|
readName d
|
|
where
|
|
getDefault :: Sem r (Maybe Text)
|
|
getDefault = runFail $ do
|
|
dir <- map toLower <$> (embed getCurrentDirectory >>= Fail.last . splitDirectories)
|
|
Fail.fromRight (parse projectNameParser (pack dir))
|
|
readName :: Maybe Text -> Sem r Text
|
|
readName def = go
|
|
where
|
|
go :: Sem r Text
|
|
go = do
|
|
txt <- embed getLine
|
|
if
|
|
| Text.null txt, Just def' <- def -> return def'
|
|
| otherwise ->
|
|
case parse projectNameParser txt of
|
|
Right p
|
|
| Text.length p <= projextNameMaxLength -> return p
|
|
| otherwise -> do
|
|
say ("The project name cannot exceed " <> prettyText projextNameMaxLength <> " characters")
|
|
retry
|
|
Left err -> do
|
|
say err
|
|
retry
|
|
where
|
|
retry :: Sem r Text
|
|
retry = do
|
|
tryAgain
|
|
go
|
|
|
|
say :: Members '[Embed IO] r => Text -> Sem r ()
|
|
say = embed . putStrLn
|
|
|
|
tryAgain :: Members '[Embed IO] r => Sem r ()
|
|
tryAgain = say "Please, try again:"
|
|
|
|
getVersion :: forall r. Members '[Embed IO] r => Sem r SemVer
|
|
getVersion = do
|
|
txt <- embed getLine
|
|
if
|
|
| Text.null txt -> return mempty
|
|
| otherwise -> case parse semver' txt of
|
|
Right r -> return r
|
|
Left err -> do
|
|
say err
|
|
say "The version must follow the 'Semantic Versioning 2.0.0' specification"
|
|
retry
|
|
where
|
|
retry :: Sem r SemVer
|
|
retry = do
|
|
tryAgain
|
|
getVersion
|
|
|
|
projextNameMaxLength :: Int
|
|
projextNameMaxLength = 100
|
|
|
|
projectNameParser :: Parsec Void Text Text
|
|
projectNameParser = do
|
|
h <- P.satisfy validFirstChar
|
|
t <- P.takeWhileP (Just "project name character") validChar
|
|
P.hspace
|
|
P.eof
|
|
return (Text.cons h t)
|
|
where
|
|
validFirstChar :: Char -> Bool
|
|
validFirstChar c =
|
|
isAscii c
|
|
&& (isLower c || isNumber c)
|
|
validChar :: Char -> Bool
|
|
validChar c = c == '-' || validFirstChar c
|