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

Better list argument handling #257

Merged
merged 6 commits into from
Nov 26, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 8 additions & 2 deletions main/Main.hs
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,8 @@ data Nixfmt = Nixfmt
strict :: Bool,
verify :: Bool,
ast :: Bool,
filename :: Maybe FilePath
filename :: Maybe FilePath,
ir :: Bool
}
deriving (Show, Data, Typeable)

Expand Down Expand Up @@ -83,7 +84,11 @@ options =
Nothing
&= help
"The filename to display when the file input is given through stdin.\n\
\Useful for tools like editors and autoformatters that wish to use Nixfmt without providing it direct file access, while still providing context to where the file is."
\Useful for tools like editors and autoformatters that wish to use Nixfmt without providing it direct file access, while still providing context to where the file is.",
ir =
False
&= help
"Pretty print the internal intermediate representation, only for debugging"
}
&= summary ("nixfmt " ++ versionFromFile)
&= help "Format Nix source code"
Expand Down Expand Up @@ -164,6 +169,7 @@ type Formatter = FilePath -> Text -> Either String Text

toFormatter :: Nixfmt -> Formatter
toFormatter Nixfmt{ast = True} = Nixfmt.printAst
toFormatter Nixfmt{ir = True} = Nixfmt.printIR
toFormatter Nixfmt{width, verify = True, strict} = Nixfmt.formatVerify (layout width strict)
toFormatter Nixfmt{width, verify = False, strict} = Nixfmt.format (layout width strict)

Expand Down
9 changes: 8 additions & 1 deletion src/Nixfmt.hs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ module Nixfmt (
format,
formatVerify,
printAst,
printIR,
)
where

Expand All @@ -15,7 +16,7 @@ import Data.Either (fromRight)
import Data.Text (Text, unpack)
import Data.Text.Lazy (toStrict)
import qualified Nixfmt.Parser as Parser
import Nixfmt.Predoc (Pretty)
import Nixfmt.Predoc (Pretty, fixup, pretty)
import Nixfmt.Pretty ()
import Nixfmt.Types (Expression, LanguageElement, ParseErrorBundle, Whole (..), walkSubprograms)
import qualified Text.Megaparsec as Megaparsec (parse)
Expand All @@ -41,6 +42,12 @@ printAst path unformatted = do
Whole unformattedParsed' _ <- first errorBundlePretty . Megaparsec.parse Parser.file path $ unformatted
Left (unpack $ toStrict $ pShow unformattedParsed')

-- | Pretty print the internal IR for debugging
printIR :: FilePath -> Text -> Either String Text
printIR filename =
bimap errorBundlePretty (toStrict . pShow . fixup . pretty)
. Megaparsec.parse Parser.file filename

-- Same functionality as `format`, but add sanity checks to guarantee the following properties of the formatter:
-- - Correctness: The formatted output parses, and the parse tree is identical to the input's
-- - Idempotency: Formatting the output again will not modify it
Expand Down
40 changes: 33 additions & 7 deletions src/Nixfmt/Pretty.hs
Original file line number Diff line number Diff line change
Expand Up @@ -381,6 +381,12 @@ prettyApp indentFunction pre hasPost f a =
-- because if they get expanded before anything else,
-- only the `.`-and-after part gets to a new line, which looks very odd
absorbApp (Application f' a'@(Term Selection{})) = group' Transparent (absorbApp f') <> line <> nest (group' RegularG $ absorbInner a')
-- If two consecutive arguments are lists, treat them specially: Don't priority expand, and also
-- if one does not fit onto the line then put both on a new line each.
-- Note that this does not handle the case where the two last arguments are lists, as the last argument
-- is handled elsewhere and cannot be pattern-matched here.
absorbApp (Application (Application f' l1@(Term List{})) l2@(Term List{})) =
group' Transparent (group' Transparent (absorbApp f') <> nest (group' RegularG $ line <> group (absorbInner l1) <> line <> group (absorbInner l2)))
absorbApp (Application f' a') = group' Transparent (absorbApp f') <> line <> nest (group' Priority $ absorbInner a')
-- First argument
absorbApp expr
Expand All @@ -392,7 +398,7 @@ prettyApp indentFunction pre hasPost f a =
-- If lists have only simple items, try to render them single-line instead of expanding
-- This is just a copy of the list rendering code, but with `sepBy line` instead of `sepBy hardline`
absorbInner (Term (List paropen@Ann{trailComment = post'} items parclose))
| length (unItems items) <= 4 && all (isSimple . Term) items =
| length (unItems items) <= 6 && all (isSimple . Term) items =
pretty (paropen{trailComment = Nothing})
<> surroundWith sur (nest $ pretty post' <> sepBy line (unItems items))
<> pretty parclose
Expand Down Expand Up @@ -451,15 +457,35 @@ prettyApp indentFunction pre hasPost f a =
((\a'@Ann{preTrivia} -> (a'{preTrivia = []}, preTrivia)) . moveTrailingCommentUp)
f

renderedF = pre <> group' Transparent (absorbApp fWithoutComment)
renderedFUnexpanded = unexpandSpacing' Nothing renderedF
-- renderSimple will take a document to render, and call one of two callbacks depending on whether
-- it can take a simplified layout (with removed line breaks) or not.
renderSimple :: Doc -> (Doc -> Doc) -> (Doc -> Doc) -> Doc
renderSimple toRender renderIfSimple renderOtherwise =
let renderedF = pre <> group' Transparent toRender
renderedFUnexpanded = unexpandSpacing' Nothing renderedF
in if isSimple (Application f a) && isJust renderedFUnexpanded
then renderIfSimple (fromJust renderedFUnexpanded)
else renderOtherwise renderedF

post = if hasPost then line' else mempty
in pretty comment'
<> ( if isSimple (Application f a) && isJust renderedFUnexpanded
then group' RegularG $ fromJust renderedFUnexpanded <> hardspace <> absorbLast a
else group' RegularG $ renderedF <> line <> absorbLast a <> post
)
<> case (fWithoutComment, a) of
-- When the two last arguments are lists, render these specially (same as above)
-- Also no need to wrap in renderSimple here, because we know that these kinds of arguments
-- are never "simple" by definition.
(Application fWithoutCommandAndWithoutArg l1@(Term List{}), l2@(Term List{})) ->
group' RegularG $
(pre <> group' Transparent (absorbApp fWithoutCommandAndWithoutArg))
<> line
<> nest (group (absorbInner l1))
<> line
<> nest (group (absorbInner l2))
<> post
_ ->
renderSimple
(absorbApp fWithoutComment)
(\fRendered -> group' RegularG $ fRendered <> hardspace <> absorbLast a)
(\fRendered -> group' RegularG $ fRendered <> line <> absorbLast a <> post)
<> (if hasPost && not (null comment') then hardline else mempty)

prettyWith :: Bool -> Expression -> Doc
Expand Down
5 changes: 1 addition & 4 deletions test/diff/apply/out-pure.nix
Original file line number Diff line number Diff line change
Expand Up @@ -160,10 +160,7 @@
''"''
"\${"
];
escapeMultiline = libStr.replaceStrings [ "\${" "''" ] [
"''\${"
"'''"
];
escapeMultiline = libStr.replaceStrings [ "\${" "''" ] [ "''\${" "'''" ];
test =
foo
[
Expand Down
78 changes: 78 additions & 0 deletions test/diff/apply_with_lists/in.nix
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
# This file contains an assortment of test cases involving list-heavy function calls

[
(f [ ] [ rhs lhs ])
(lib.mkMerge [ false false ])
(replaceStrings
[ "\${" "''" ]
#force multiline
[ "''\${" "'''" ])
(replaceStrings [ ''"'' "\\" ] [ ''\"'' "\\\\" ] name)
(replaceStrings
[ ''"'' "\\" ]
# force multiline
[ ''\"'' "\\\\" ]
name)
(replaceStrings
[ "@" ":" "\\" "[" "]" ]
[ "-" "-" "-" "" "" ])
(lists.removePrefix [ 1 2 ] [ ])
(lists.removePrefix aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa [ 1 2 ] [ ])
(builtins.replaceStrings
[ "@NIX_STORE_VERITY@" ]
[partitionTypes.usr-verity]
(builtins.readFile ./assert_uki_repart_match.py))
(replaceStrings [ "-" ] [ "_" ] (toUpper final.rust.cargoShortTarget))
(lib.mkChangedOptionModule
[ "security" "acme" "validMin" ]
[ "security" "acme" "defaults" "validMinDays" ]
(config: config.security.acme.validMin / (24 * 3600)))
(lib.replaceStrings
[ "https://registry" ".io/providers" ]
[ "registry" ".io" ]
homepage)
(lib.mkRenamedOptionModule [ "boot" "extraTTYs" ] [ "console" "extraTTYs" ])
# This line is engineered to exactly hit the line length limit
(lib.mkRenamedOptionModule [ "hardware" "package234" ] [ "hardware" "graphics" ])
(mkRenamedOptionModule [
"services"
"xserver"
"displayManager"
"sddm"
"enable"
] [ "services" "displayManager" "sddm" "enable" ])
(map (buildAllowCommand "allow" [ "snapshot" "mount" "destroy" ]))
(map (x: "${x} ${escapeShellArgs [ stateDir workDir logsDir ]}") [
"+${unconfigureRunner}" # runs as root
configureRunner
setupWorkDir
])
(lib.checkListOfEnum "${pname}: theme accent"
[
"Blue"
"Flamingo"
"Green"
]
[ accent ]
lib.checkListOfEnum
"${pname}: color variant"
[ "Latte" "Frappe" "Macchiato" "Mocha" ]
[ variant ]
)
(lib.switch [ coq.coq-version ssreflect.version ] [
{
cases = [
(lib.versions.range "8.15" "8.20")
lib.pred.true
];
out = "2.0.4";
}
{
cases = [
"8.5"
lib.pred.true
];
out = "20170512";
}
] null)
]
110 changes: 110 additions & 0 deletions test/diff/apply_with_lists/out-pure.nix
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
# This file contains an assortment of test cases involving list-heavy function calls

[
(f [ ] [ rhs lhs ])
(lib.mkMerge [
false
false
])
(replaceStrings
[ "\${" "''" ]
#force multiline
[ "''\${" "'''" ]
)
(replaceStrings [ ''"'' "\\" ] [ ''\"'' "\\\\" ] name)
(replaceStrings
[ ''"'' "\\" ]
# force multiline
[ ''\"'' "\\\\" ]
name
)
(replaceStrings [ "@" ":" "\\" "[" "]" ] [ "-" "-" "-" "" "" ])
(lists.removePrefix
[
1
2
]
[ ]
)
(lists.removePrefix aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
[
1
2
]
[ ]
)
(builtins.replaceStrings [ "@NIX_STORE_VERITY@" ] [ partitionTypes.usr-verity ]
(builtins.readFile ./assert_uki_repart_match.py)
)
(replaceStrings [ "-" ] [ "_" ] (toUpper final.rust.cargoShortTarget))
(lib.mkChangedOptionModule
[ "security" "acme" "validMin" ]
[ "security" "acme" "defaults" "validMinDays" ]
(config: config.security.acme.validMin / (24 * 3600))
)
(lib.replaceStrings [ "https://registry" ".io/providers" ] [ "registry" ".io" ]
homepage
)
(lib.mkRenamedOptionModule [ "boot" "extraTTYs" ] [ "console" "extraTTYs" ])
# This line is engineered to exactly hit the line length limit
(lib.mkRenamedOptionModule
[ "hardware" "package234" ]
[ "hardware" "graphics" ]
)
(mkRenamedOptionModule
[ "services" "xserver" "displayManager" "sddm" "enable" ]
[ "services" "displayManager" "sddm" "enable" ]
)
(map (
buildAllowCommand "allow" [
"snapshot"
"mount"
"destroy"
]
))
(map
(
x:
"${x} ${
escapeShellArgs [
stateDir
workDir
logsDir
]
}"
)
[
"+${unconfigureRunner}" # runs as root
configureRunner
setupWorkDir
]
)
(lib.checkListOfEnum "${pname}: theme accent"
[ "Blue" "Flamingo" "Green" ]
[ accent ]
lib.checkListOfEnum
"${pname}: color variant"
[ "Latte" "Frappe" "Macchiato" "Mocha" ]
[ variant ]
)
(lib.switch
[ coq.coq-version ssreflect.version ]
[
{
cases = [
(lib.versions.range "8.15" "8.20")
lib.pred.true
];
out = "2.0.4";
}
{
cases = [
"8.5"
lib.pred.true
];
out = "20170512";
}
]
null
)
]
Loading