Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow specifying a stack.yaml for stack configurations #230

Merged
merged 15 commits into from
Aug 23, 2020
Merged
4 changes: 2 additions & 2 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
dist/
dist-newstyle/
.stack-work/
stack.yaml
stack.yaml.lock
tests/projects/**/stack*.yaml
tests/projects/**/stack*.yaml.lock
cabal.project.local*
.vscode/
49 changes: 49 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,16 @@ cradle:

This configuration suffices if your whole project can be loaded by the command `stack repl`. As a rule of thumb, this works if the project consists of only one executable, one library and one test-suite.

Some projects have multiple `stack-*.yaml` files for multiple versions of ghc/resolvers. In this case you
can specify an alternate relative file to use by using the `stackYaml` option. The path is relative to the
configuration file.

```yaml
cradle:
stack:
stackYaml: "./stack-8.8.3.yaml"
```

If your project is more complicated, you need to specify which [components](https://docs.haskellstack.org/en/stable/build_command/#components) you want to load. A component is, roughly speaking, a library/executable/test-suite or benchmark in `stack`. You can view the components/targets of a stack project by executing the command:
``` sh
$ stack ide targets
Expand Down Expand Up @@ -126,6 +136,26 @@ Here you can see two important features:

This way we specified which component needs to be compiled given a source file for our whole project.

If you use both, multiple components and an alternate `stack.yaml` file, there is a way to specify defaults
for the path-specific configurations.

```yaml
cradle:
stack:
stackYaml: "stack-8.3.3.yaml"
components:
- path: "./src"
component: "hie-bios:lib"
- path: "./exe"
component: "hie-bios:exe:hie-bios"
- path: "./tests/BiosTests.hs"
component: "hie-bios:test:hie-bios"
- path: "./tests/ParserTests.hs"
component: "hie-bios:test:parser-tests"
```
WorldSEnder marked this conversation as resolved.
Show resolved Hide resolved

A word of warning: Due to current restrictions in the language server, as mentioned in [this bug report](https://github.com/haskell/haskell-language-server/issues/268#issuecomment-667640809) all referenced stack.yaml files must specify the same version of GHC, as only one version of ghcide is loaded at a time per workspace root. This restriction might be lifted in the future.

#### Debugging a `stack` cradle

If you find that `hie-bios` can't load a certain component or file, run `stack repl` and `stack repl <component name>` to see if `stack` succeeds in building your project. Chances are that there is a problem in your project and if you fix that, `hie-bios` will succeed to load it.
Expand Down Expand Up @@ -267,6 +297,25 @@ Here you can see two important features:
* The filepath can be a file.
* This is convenient if components are overlapping.

Similarly to `multi-stack` configurations, you can also specify multiple components using a `components` subkey.
While this is currently not used for anything, this syntax gives you a place to put defaults, directly under
the `cabal` entry.

```yaml
cradle:
cabal:
# Reserved for future default options
components:
- path: "./src"
component: "lib:hie-bios"
- path: "./exe"
component: "exe:hie-bios"
- path: "./tests/BiosTests.hs"
component: "test:hie-bios"
- path: "./tests/ParserTests.hs"
component: "test:parser-tests"
```

This way we specified which component needs to be compiled given a certain source file for our whole project.

#### Debugging a `cabal` cradle
Expand Down
18 changes: 17 additions & 1 deletion hie-bios.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -31,12 +31,16 @@ Extra-Source-Files: ChangeLog.md
tests/configs/direct.yaml
tests/configs/multi.yaml
tests/configs/multi-ch.yaml
tests/configs/multi-stack-with-yaml.yaml
tests/configs/keys-not-unique-fails.yaml
tests/configs/nested-cabal-multi.yaml
tests/configs/nested-stack-multi.yaml
tests/configs/none.yaml
tests/configs/obelisk.yaml
tests/configs/stack-config.yaml
tests/configs/stack-multi.yaml
tests/configs/stack-with-both.yaml
tests/configs/stack-with-yaml.yaml
tests/projects/symlink-test/a/A.hs
tests/projects/symlink-test/hie.yaml
tests/projects/multi-direct/A.hs
Expand Down Expand Up @@ -145,7 +149,18 @@ Extra-Source-Files: ChangeLog.md
tests/projects/implicit-stack-multi/other-package/Setup.hs
tests/projects/implicit-stack-multi/other-package/other-package.cabal
tests/projects/implicit-stack-multi/other-package/Main.hs

tests/projects/multi-stack-with-yaml/appA/Setup.hs
tests/projects/multi-stack-with-yaml/appA/appA.cabal
tests/projects/multi-stack-with-yaml/appA/src/Lib.hs
tests/projects/multi-stack-with-yaml/appB/Setup.hs
tests/projects/multi-stack-with-yaml/appB/appB.cabal
tests/projects/multi-stack-with-yaml/appB/src/Lib.hs
tests/projects/multi-stack-with-yaml/hie.yaml
tests/projects/stack-with-yaml/Setup.hs
tests/projects/stack-with-yaml/app/Main.hs
tests/projects/stack-with-yaml/hie.yaml
tests/projects/stack-with-yaml/stack-with-yaml.cabal
tests/projects/stack-with-yaml/src/Lib.hs

tested-with: GHC==8.10.1, GHC==8.8.3, GHC==8.6.5, GHC==8.4.4

Expand Down Expand Up @@ -216,6 +231,7 @@ test-suite parser-tests
base,
filepath,
hie-bios,
hspec-expectations,
tasty,
tasty-hunit,
text,
Expand Down
160 changes: 128 additions & 32 deletions src/HIE/Bios/Config.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2,20 +2,46 @@
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TupleSections #-}
{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE PatternSynonyms #-}
{-# LANGUAGE ViewPatterns #-}
{-# LANGUAGE LambdaCase #-}
-- | Logic and datatypes for parsing @hie.yaml@ files.
module HIE.Bios.Config(
readConfig,
Config(..),
CradleConfig(..),
CabalType,
pattern CabalType,
cabalComponent,
StackType,
pattern StackType,
stackComponent,
stackYaml,
CradleType(..),
Callable(..)
) where

import Control.Exception
import qualified Data.Text as T
import qualified Data.Vector as V
import qualified Data.HashMap.Strict as Map
import Data.Maybe (mapMaybe)
import Data.Semigroup
import Data.Foldable (foldrM)
import Data.Aeson (JSONPath)
import Data.Yaml
import Data.Yaml.Internal (Warning(..))

type MLast a = Maybe (Last a)

viewLast :: MLast a -> Maybe a
viewLast (Just l) = Just $ getLast l
viewLast Nothing = Nothing

pattern MLast :: Maybe a -> MLast a
pattern MLast m <- (viewLast -> m) where
MLast (Just l) = Just $ Last l
MLast Nothing = Nothing

-- | Configuration that can be used to load a 'Cradle'.
-- A configuration has roughly the following form:
Expand Down Expand Up @@ -43,11 +69,45 @@ data CradleConfig a =
data Callable = Program FilePath | Command String
deriving (Show, Eq)

data CabalType
= CabalType_ { _cabalComponent :: !(MLast String) }
deriving (Eq)

instance Semigroup CabalType where
CabalType_ cr <> CabalType_ cl = CabalType_ (cr <> cl)

instance Monoid CabalType where
mempty = CabalType_ mempty

pattern CabalType :: Maybe String -> CabalType
pattern CabalType { cabalComponent } = CabalType_ (MLast cabalComponent)
{-# COMPLETE CabalType #-}

instance Show CabalType where
show = show . Cabal

data StackType
= StackType_ { _stackComponent :: !(MLast String) , _stackYaml :: !(MLast String) }
deriving (Eq)

instance Semigroup StackType where
StackType_ cr yr <> StackType_ cl yl = StackType_ (cr <> cl) (yr <> yl)

instance Monoid StackType where
mempty = StackType_ mempty mempty

pattern StackType :: Maybe String -> Maybe String -> StackType
pattern StackType { stackComponent, stackYaml } = StackType_ (MLast stackComponent) (MLast stackYaml)
{-# COMPLETE StackType #-}

instance Show StackType where
show = show . Stack

data CradleType a
= Cabal { component :: Maybe String }
| CabalMulti [ (FilePath, String) ]
| Stack { component :: Maybe String }
| StackMulti [ (FilePath, String) ]
= Cabal { cabalType :: !CabalType }
| CabalMulti { defaultCabal :: !CabalType, subCabalComponents :: [ (FilePath, CabalType) ] }
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This also allows the config file:

cradle:
  cabal:
    components:
      - path: "./src"
        component: "lib:hie-bios"
      - path: "./"
        component: "lib:hie-bios"

While I dont expect any harm of it (and maybe we want soonish a similar field for cabal) but I just wanted to document it.

| Stack { stackType :: !StackType }
| StackMulti { defaultStack :: !StackType, subStackComponents :: [ (FilePath, StackType) ] }
-- Bazel and Obelisk used to be supported but bit-rotted and no users have complained.
-- They can be added back if a user
-- | Bazel
Expand All @@ -73,10 +133,10 @@ instance FromJSON a => FromJSON (CradleType a) where
parseJSON _ = fail "Not a known cradle type. Possible are: cabal, stack, bios, direct, default, none, multi"

instance Show (CradleType a) where
show (Cabal comp) = "Cabal {component = " ++ show comp ++ "}"
show (CabalMulti a) = "CabalMulti " ++ show a
show (Stack comp) = "Stack {component = " ++ show comp ++ "}"
show (StackMulti a) = "StackMulti " ++ show a
show (Cabal comp) = "Cabal {component = " ++ show (cabalComponent comp) ++ "}"
show (CabalMulti d a) = "CabalMulti {defaultCabal = " ++ show d ++ ", subCabalComponents = " ++ show a ++ "}"
show (Stack comp) = "Stack {component = " ++ show (stackComponent comp) ++ ", stackYaml = " ++ show (stackYaml comp) ++ "}"
show (StackMulti d a) = "StackMulti {defaultStack = " ++ show d ++ ", subStackComponents = " ++ show a ++ "}"
show Bios { call, depsCall } = "Bios {call = " ++ show call ++ ", depsCall = " ++ show depsCall ++ "}"
show (Direct args) = "Direct {arguments = " ++ show args ++ "}"
show None = "None"
Expand All @@ -96,38 +156,61 @@ parseCradleType o
| Just val <- Map.lookup "other" o = Other <$> parseJSON val <*> pure val
parseCradleType o = fail $ "Unknown cradle type: " ++ show o

parseStackOrCabal
:: (Maybe String -> CradleType a)
-> ([(FilePath, String)] -> CradleType a)
parseSingleOrMultiple
:: Monoid x
=> (x -> CradleType a)
-> (x -> [(FilePath, x)] -> CradleType a)
-> (Map.HashMap T.Text Value -> Parser x)
-> Value
-> Parser (CradleType a)
parseStackOrCabal singleConstructor _ (Object x)
| Map.size x == 1, Just (String stackComponent) <- Map.lookup "component" x
= return $ singleConstructor $ Just $ T.unpack stackComponent
| Map.null x
= return $ singleConstructor Nothing
| otherwise
= fail "Not a valid Configuration type, following keys are allowed: component"
parseStackOrCabal _ multiConstructor (Array x) = do
let parseOne e
parseSingleOrMultiple single multiple parse = doParse where
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Interestingly, this allows this configuration:

cradle:
  cabal:
    components:
      - path: "./src"
        component: "lib:hie-bios"
      - path: "./"
        component: "lib:hie-bios"
    components:
      - path: "./src"
        component: "exe:hie-bios"
      - path: "./"
        component: "exe:hie-bios"

with this result:

Config
  { cradle =
      CradleConfig
        { cradleDependencies = []
        , cradleType =
            CabalMulti
              { defaultCabal = Cabal { component = Nothing }
              , subCabalComponents =
                  [ ( "./src" , Cabal { component = Just "exe:hie-bios" } )
                  , ( "./" , Cabal { component = Just "exe:hie-bios" } )
                  ]
              }
        }
  }

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's a good catch, didn't think about multiple components entries. I'll fix this before the PR is merged.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thinking about this a little bit more, isn't a duplicate key (components) an error according to the yaml spec?

Mapping nodes are somewhat tricky because their keys are unordered and must be unique.

In that case, I suppose this PR is relevant and https://github.com/mpickering/hie-bios/pull/230/files#diff-0478bef4574ee638b0c136cccc31479aR288 should use decodeFileWithWarnings

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added a test case that's supposed to fail on the config with duplicate components key.

parseOne e
| Object v <- e
, Just (String prefix) <- Map.lookup "path" v
, Just (String comp) <- Map.lookup "component" v
, Map.size v == 2
= return (T.unpack prefix, T.unpack comp)
= (T.unpack prefix,) <$> parse (Map.delete "path" v)
| otherwise
= fail "Expected an object with path and component keys"

xs <- foldrM (\v cs -> (: cs) <$> parseOne v) [] x
return $ multiConstructor xs
parseStackOrCabal singleConstructor _ Null = return $ singleConstructor Nothing
parseStackOrCabal _ _ _ = fail "Configuration is expected to be an object."
= fail "Expected an object with a path key"
parseArray = foldrM (\v cs -> (: cs) <$> parseOne v) []
doParse (Object v)
| Just (Array x) <- Map.lookup "components" v
= do
d <- parse (Map.delete "components" v)
xs <- parseArray x
return $ multiple d xs
| Just _ <- Map.lookup "components" v
= fail "Expected components to be an array of subcomponents"
| Nothing <- Map.lookup "components" v
= single <$> parse v
doParse (Array x)
= do
xs <- parseArray x
return $ multiple mempty xs
doParse Null = single <$> parse Map.empty
doParse _ = fail "Configuration is expected to be an object or an array of objects."

parseStack :: Value -> Parser (CradleType a)
parseStack = parseStackOrCabal Stack StackMulti
parseStack = parseSingleOrMultiple Stack StackMulti $
\case x | Map.size x == 2
, Just (String component) <- Map.lookup "component" x
, Just (String syaml) <- Map.lookup "stackYaml" x
-> return $ StackType (Just $ T.unpack component) (Just $ T.unpack syaml)
| Map.size x == 1, Just (String component) <- Map.lookup "component" x
-> return $ StackType (Just $ T.unpack component) Nothing
| Map.size x == 1, Just (String syaml) <- Map.lookup "stackYaml" x
-> return $ StackType Nothing (Just $ T.unpack syaml)
| Map.null x
-> return $ StackType Nothing Nothing
| otherwise
-> fail "Not a valid Stack configuration, following keys are allowed: component, stackYaml"

parseCabal :: Value -> Parser (CradleType a)
parseCabal = parseStackOrCabal Cabal CabalMulti
parseCabal = parseSingleOrMultiple Cabal CabalMulti $
\case x | Map.size x == 1, Just (String component) <- Map.lookup "component" x
-> return $ CabalType (Just $ T.unpack component)
| Map.null x
-> return $ CabalType Nothing
| otherwise
-> fail "Not a valid Cabal configuration, following keys are allowed: component"

parseBios :: Value -> Parser (CradleType a)
parseBios (Object x) =
Expand Down Expand Up @@ -206,4 +289,17 @@ instance FromJSON a => FromJSON (Config a) where
-- If the contents of the file is not a valid 'Config a',
-- an 'Control.Exception.IOException' is thrown.
readConfig :: FromJSON a => FilePath -> IO (Config a)
readConfig = decodeFileThrow
readConfig fp = do
result <- decodeFileWithWarnings fp
either throwIO failOnAnyDuplicate result
where
failOnAnyDuplicate :: ([Warning], Config a) -> IO (Config a)
failOnAnyDuplicate (warnings, config) = do
_ <- case mapMaybe failOnDuplicate warnings of
dups@(_:_) -> throwIO $ InvalidYaml $ Just $ YamlException
$ "Duplicate keys are not allowed, found: " ++ show dups
_ -> return ()
return config
-- future proofing in case more warnings are added
failOnDuplicate :: Warning -> Maybe JSONPath
failOnDuplicate (DuplicateKey a) = Just a
Loading