Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: NixOS/nixpkgs
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: 5e4c49463607
Choose a base ref
...
head repository: NixOS/nixpkgs
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: 07eb21ceaf7a
Choose a head ref
  • 8 commits
  • 5 files changed
  • 1 contributor

Commits on Jan 23, 2020

  1. lib/cli: unexport symbols & sort with generators

    lib/cli is very similar to generators, so it should follow largely the
    same interface. Similar to how generators isn’t exported, we should
    also namespace cli by default (plus “cli” is only three characters to
    type).
    Profpatsch committed Jan 23, 2020
    Copy the full SHA
    88a7f65 View commit details
  2. lib/cli: encodeGNUCommandLine -> toGNUCommandLineShell

    The semantic difference between `encode` and `to` is not apparent.
    Users are likely to confuse both functions (which leads to unexpected
    error messages about the wrong types). Like in `generators.nix`, all
    functions should be prefixed by `to`.
    
    Furthermore, converting to a string depends on the target context. In
    this case, it’s a POSIX shell, so we should name it that (compare
    `escapeShellArg` in `strings.nix`).
    
    We can later add versions that escape for embedding in e.g. python
    scripts or similar.
    Profpatsch committed Jan 23, 2020
    Copy the full SHA
    582354d View commit details
  3. Copy the full SHA
    6841f40 View commit details
  4. Copy the full SHA
    b2654c2 View commit details
  5. lib/cli: rename renderX options to mkX

    Mirrors the naming scheme in `generators.nix`, for consistency.
    
    Also rename `key` to `k` and value to `v` to aid readability to the
    code structure.
    Profpatsch committed Jan 23, 2020
    Copy the full SHA
    e71e1be View commit details
  6. lib/generators: floats are not supported in mkValueStringDefault

    They are cut off after a few decimal places; we cannot in good faith
    define a default string representation with that.
    Profpatsch committed Jan 23, 2020
    Copy the full SHA
    18520b7 View commit details
  7. lib/cli: mkKey -> mkOptionName, use generators.mkValueStringDefault

    Let’s call them by what they are, option names.
    
    `generators.mkValueStringDefault` is a better value string renderer
    than plain `toString`.
    
    Also add docs to all options.
    Profpatsch committed Jan 23, 2020
    Copy the full SHA
    7228a3c View commit details

Commits on Jan 24, 2020

  1. Merge pull request #78337 from Profpatsch/lib-improve-cli-module

    lib: improve cli module
    Profpatsch authored Jan 24, 2020

    Verified

    This commit was created on GitHub.com and signed with GitHub’s verified signature. The key has expired.
    Copy the full SHA
    07eb21c View commit details
Showing with 100 additions and 51 deletions.
  1. +2 −0 .github/CODEOWNERS
  2. +60 −33 lib/cli.nix
  3. +4 −3 lib/default.nix
  4. +3 −0 lib/generators.nix
  5. +31 −15 lib/tests/misc.nix
2 changes: 2 additions & 0 deletions .github/CODEOWNERS
Original file line number Diff line number Diff line change
@@ -14,7 +14,9 @@
/lib @edolstra @nbp @infinisil
/lib/systems @nbp @ericson2314 @matthewbauer
/lib/generators.nix @edolstra @nbp @Profpatsch
/lib/cli.nix @edolstra @nbp @Profpatsch
/lib/debug.nix @edolstra @nbp @Profpatsch
/lib/asserts.nix @edolstra @nbp @Profpatsch

# Nixpkgs Internals
/default.nix @nbp
93 changes: 60 additions & 33 deletions lib/cli.nix
Original file line number Diff line number Diff line change
@@ -6,50 +6,77 @@ rec {
This helps protect against malformed command lines and also to reduce
boilerplate related to command-line construction for simple use cases.
Example:
encodeGNUCommandLine
{ }
{ data = builtins.toJSON { id = 0; };
X = "PUT";
retry = 3;
`toGNUCommandLine` returns a list of nix strings.
`toGNUCommandLineShell` returns an escaped shell string.
retry-delay = null;
url = [ "https://example.com/foo" "https://example.com/bar" ];
silent = false;
Example:
cli.toGNUCommandLine {} {
data = builtins.toJSON { id = 0; };
X = "PUT";
retry = 3;
retry-delay = null;
url = [ "https://example.com/foo" "https://example.com/bar" ];
silent = false;
verbose = true;
}
=> [
"-X" "PUT"
"--data" "{\"id\":0}"
"--retry" "3"
"--url" "https://example.com/foo"
"--url" "https://example.com/bar"
"--verbose"
]
verbose = true;
};
=> "'-X' 'PUT' '--data' '{\"id\":0}' '--retry' '3' '--url' 'https://example.com/foo' '--url' 'https://example.com/bar' '--verbose'"
cli.toGNUCommandLineShell {} {
data = builtins.toJSON { id = 0; };
X = "PUT";
retry = 3;
retry-delay = null;
url = [ "https://example.com/foo" "https://example.com/bar" ];
silent = false;
verbose = true;
}
=> "'-X' 'PUT' '--data' '{\"id\":0}' '--retry' '3' '--url' 'https://example.com/foo' '--url' 'https://example.com/bar' '--verbose'";
*/
encodeGNUCommandLine =
toGNUCommandLineShell =
options: attrs: lib.escapeShellArgs (toGNUCommandLine options attrs);

toGNUCommandLine =
{ renderKey ?
key: if builtins.stringLength key == 1 then "-${key}" else "--${key}"
toGNUCommandLine = {
# how to string-format the option name;
# by default one character is a short option (`-`),
# more than one characters a long option (`--`).
mkOptionName ?
k: if builtins.stringLength k == 1
then "-${k}"
else "--${k}",

, renderOption ?
key: value:
if value == null
then []
else [ (renderKey key) (builtins.toString value) ]
# how to format a boolean value to a command list;
# by default it’s a flag option
# (only the option name if true, left out completely if false).
mkBool ? k: v: lib.optional v (mkOptionName k),

, renderBool ? key: value: lib.optional value (renderKey key)
# how to format a list value to a command list;
# by default the option name is repeated for each value
# and `mkOption` is applied to the values themselves.
mkList ? k: v: lib.concatMap (mkOption k) v,

, renderList ? key: value: lib.concatMap (renderOption key) value
# how to format any remaining value to a command list;
# on the toplevel, booleans and lists are handled by `mkBool` and `mkList`,
# though they can still appear as values of a list.
# By default, everything is printed verbatim and complex types
# are forbidden (lists, attrsets, functions). `null` values are omitted.
mkOption ?
k: v: if v == null
then []
else [ (mkOptionName k) (lib.generators.mkValueStringDefault {} v) ]
}:
options:
let
render = key: value:
if builtins.isBool value
then renderBool key value
else if builtins.isList value
then renderList key value
else renderOption key value;
render = k: v:
if builtins.isBool v then mkBool k v
else if builtins.isList v then mkList k v
else mkOption k v;

in
builtins.concatLists (lib.mapAttrsToList render options);
7 changes: 4 additions & 3 deletions lib/default.nix
Original file line number Diff line number Diff line change
@@ -37,11 +37,13 @@ let
licenses = callLibs ./licenses.nix;
systems = callLibs ./systems;

# serialization
cli = callLibs ./cli.nix;
generators = callLibs ./generators.nix;

# misc
asserts = callLibs ./asserts.nix;
cli = callLibs ./cli.nix;
debug = callLibs ./debug.nix;
generators = callLibs ./generators.nix;
misc = callLibs ./deprecated.nix;

# domain-specific
@@ -121,7 +123,6 @@ let
isOptionType mkOptionType;
inherit (asserts)
assertMsg assertOneOf;
inherit (cli) encodeGNUCommandLine toGNUCommandLine;
inherit (debug) addErrorContextToAttrs traceIf traceVal traceValFn
traceXMLVal traceXMLValMarked traceSeq traceSeqN traceValSeq
traceValSeqFn traceValSeqN traceValSeqNFn traceShowVal
3 changes: 3 additions & 0 deletions lib/generators.nix
Original file line number Diff line number Diff line change
@@ -46,7 +46,10 @@ rec {
else if isList v then err "lists" v
# same as for lists, might want to replace
else if isAttrs v then err "attrsets" v
# functions can’t be printed of course
else if isFunction v then err "functions" v
# let’s not talk about floats. There is no sensible `toString` for them.
else if isFloat v then err "floats" v
else err "this value is" (toString v);


46 changes: 31 additions & 15 deletions lib/tests/misc.nix
Original file line number Diff line number Diff line change
@@ -441,24 +441,40 @@ runTests {
expected = "«foo»";
};

testRenderOptions = {
expr =
encodeGNUCommandLine
{ }
{ data = builtins.toJSON { id = 0; };

X = "PUT";

retry = 3;

retry-delay = null;

url = [ "https://example.com/foo" "https://example.com/bar" ];
# CLI

testToGNUCommandLine = {
expr = cli.toGNUCommandLine {} {
data = builtins.toJSON { id = 0; };
X = "PUT";
retry = 3;
retry-delay = null;
url = [ "https://example.com/foo" "https://example.com/bar" ];
silent = false;
verbose = true;
};

silent = false;
expected = [
"-X" "PUT"
"--data" "{\"id\":0}"
"--retry" "3"
"--url" "https://example.com/foo"
"--url" "https://example.com/bar"
"--verbose"
];
};

verbose = true;
};
testToGNUCommandLineShell = {
expr = cli.toGNUCommandLineShell {} {
data = builtins.toJSON { id = 0; };
X = "PUT";
retry = 3;
retry-delay = null;
url = [ "https://example.com/foo" "https://example.com/bar" ];
silent = false;
verbose = true;
};

expected = "'-X' 'PUT' '--data' '{\"id\":0}' '--retry' '3' '--url' 'https://example.com/foo' '--url' 'https://example.com/bar' '--verbose'";
};