Skip to content

Commit

Permalink
treewide: switch to new demand all remained cases
Browse files Browse the repository at this point in the history
Builtins, it is 44 uses.
Other also.

Some refactor in the process.

Couple of monadic binds become functors.

After this change the code allows more refactors - move `demand`s into `do`
blocks and fold the `do` blocks.
  • Loading branch information
Anton-Latukha committed Mar 5, 2021
1 parent 7712fc4 commit 80c3ec6
Show file tree
Hide file tree
Showing 9 changed files with 421 additions and 497 deletions.
91 changes: 48 additions & 43 deletions main/Repl.hs
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ import Nix hiding ( exec
)
import Nix.Scope
import Nix.Utils
import Nix.Value.Monad ( demandF )
import Nix.Value.Monad ( demand )

import qualified Data.List
import qualified Data.Maybe
Expand Down Expand Up @@ -340,69 +340,74 @@ completion = System.Console.Repline.Prefix
-- | Main completion function
--
-- Heavily inspired by Dhall Repl, with `algebraicComplete`
-- adjusted to monadic variant able to `demandF` thunks.
-- adjusted to monadic variant able to `demand` thunks.
completeFunc
:: forall e t f m . (MonadNix e t f m, MonadIO m)
=> String
-> String
-> (StateT (IState t f m) m) [Completion]
completeFunc reversedPrev word
-- Commands
| reversedPrev == ":"
= pure . listCompletion
| reversedPrev == ":" =
pure . listCompletion
$ fmap helpOptionName (helpOptions :: HelpOptions e t f m)

-- Files
| any (`Data.List.isPrefixOf` word) [ "/", "./", "../", "~/" ]
= listFiles word
| any (`Data.List.isPrefixOf` word) [ "/", "./", "../", "~/" ] =
listFiles word

-- Attributes of sets in REPL context
| var : subFields <- Data.Text.split (== '.') (Data.Text.pack word)
, not $ null subFields
= do
s <- get
case Data.HashMap.Lazy.lookup var (replCtx s) of
Nothing -> pure mempty
Just binding -> do
candidates <- lift $ algebraicComplete subFields binding
pure $ notFinished <$> listCompletion (Data.Text.unpack . (var <>) <$> candidates)
| var : subFields <- Data.Text.split (== '.') (Data.Text.pack word) , not $ null subFields =
do
s <- get
maybe
(pure mempty)
(\ binding ->
do
candidates <- lift $ algebraicComplete subFields binding
pure $ notFinished <$> listCompletion (Data.Text.unpack . (var <>) <$> candidates)
)
(Data.HashMap.Lazy.lookup var (replCtx s))

-- Builtins, context variables
| otherwise
= do
s <- get
let contextKeys = Data.HashMap.Lazy.keys (replCtx s)
(Just (NVSet builtins _)) = Data.HashMap.Lazy.lookup "builtins" (replCtx s)
shortBuiltins = Data.HashMap.Lazy.keys builtins

pure $ listCompletion
$ ["__includes"]
<> (Data.Text.unpack <$> contextKeys)
<> (Data.Text.unpack <$> shortBuiltins)
| otherwise =
do
s <- get
let contextKeys = Data.HashMap.Lazy.keys (replCtx s)
(Just (NVSet builtins _)) = Data.HashMap.Lazy.lookup "builtins" (replCtx s)
shortBuiltins = Data.HashMap.Lazy.keys builtins

pure $ listCompletion
$ ["__includes"]
<> (Data.Text.unpack <$> contextKeys)
<> (Data.Text.unpack <$> shortBuiltins)

where
listCompletion = fmap simpleCompletion . filter (word `Data.List.isPrefixOf`)

notFinished x = x { isFinished = False }

algebraicComplete :: (MonadNix e t f m)
=> [Text]
-> NValue t f m
-> m [Text]
algebraicComplete
:: (MonadNix e t f m)
=> [Text]
-> NValue t f m
-> m [Text]
algebraicComplete subFields val =
let keys = fmap ("." <>) . Data.HashMap.Lazy.keys
withMap m =
case subFields of
[] -> pure $ keys m
-- Stop on last subField (we care about the keys at this level)
[_] -> pure $ keys m
f:fs ->
maybe
(pure mempty)
(demandF (\e' -> (fmap . fmap) (("." <> f) <>) $ algebraicComplete fs e'))
(Data.HashMap.Lazy.lookup f m)

in case val of
let
keys = fmap ("." <>) . Data.HashMap.Lazy.keys

withMap m =
case subFields of
[] -> pure $ keys m
-- Stop on last subField (we care about the keys at this level)
[_] -> pure $ keys m
f:fs ->
maybe
(pure mempty)
(((fmap . fmap) (("." <> f) <>) . algebraicComplete fs) <=< demand)
(Data.HashMap.Lazy.lookup f m)
in
case val of
NVSet xs _ -> withMap xs
_ -> pure mempty

Expand Down
56 changes: 25 additions & 31 deletions src/Nix.hs
Original file line number Diff line number Diff line change
Expand Up @@ -115,14 +115,12 @@ evaluateExpression mpath evaluator handler expr = do
(second mkStr)
(argstr opts)
evaluator mpath expr >>= \f ->
demandF
(\f' ->
processResult handler =<<
case f' of
NVClosure _ g -> g (argmap args)
_ -> pure f
)
f
(\f' ->
processResult handler =<<
case f' of
NVClosure _ g -> g (argmap args)
_ -> pure f
) =<< demand f
where
parseArg s =
case parseNixText s of
Expand All @@ -149,29 +147,25 @@ processResult h val = do
go :: [Text.Text] -> NValue t f m -> m a
go [] v = h v
go ((Text.decimal -> Right (n,"")) : ks) v =
demandF
(\case
NVList xs ->
list
(\case
NVList xs ->
list
h
go
ks
(xs !! n)
_ -> errorWithoutStackTrace $ "Expected a list for selector '" <> show n <> "', but got: " <> show v
) =<< demand v
go (k : ks) v =
(\case
NVSet xs _ ->
maybe
(errorWithoutStackTrace $ "Set does not contain key '" <> Text.unpack k <> "'")
(list
h
go
ks
(xs !! n)
_ -> errorWithoutStackTrace $ "Expected a list for selector '" <> show n <> "', but got: " <> show v
)
v
go (k : ks) v =
demandF
(\case
NVSet xs _ ->
maybe
(errorWithoutStackTrace $ "Set does not contain key '" <> Text.unpack k <> "'")
(list
h
go
ks
)
(M.lookup k xs)
_ -> errorWithoutStackTrace $ "Expected a set for selector '" <> Text.unpack k <> "', but got: " <> show v
)
v
)
(M.lookup k xs)
_ -> errorWithoutStackTrace $ "Expected a set for selector '" <> Text.unpack k <> "', but got: " <> show v
) =<< demand v
Loading

0 comments on commit 80c3ec6

Please sign in to comment.