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

argparse claims '*' positional argument is required in error output #72795

Open
rrt mannequin opened this issue Nov 4, 2016 · 13 comments
Open

argparse claims '*' positional argument is required in error output #72795

rrt mannequin opened this issue Nov 4, 2016 · 13 comments
Labels
3.9 only security fixes 3.10 only security fixes 3.11 only security fixes stdlib Python modules in the Lib dir type-bug An unexpected behavior, bug, or error

Comments

@rrt
Copy link
Mannequin

rrt mannequin commented Nov 4, 2016

BPO 28609
Nosy @benhoyt, @charlierproctor, @iritkatriel
Files
  • optional_arguments.patch: Check that action.nargs not in [OPTIONAL, ZERO_OR_MORE]
  • Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.

    Show more details

    GitHub fields:

    assignee = None
    closed_at = None
    created_at = <Date 2016-11-04.15:19:23.095>
    labels = ['type-bug', 'library', '3.9', '3.10', '3.11']
    title = "argparse claims '*' positional argument is required in error output"
    updated_at = <Date 2021-12-10.17:07:21.760>
    user = 'https://bugs.python.org/rrt'

    bugs.python.org fields:

    activity = <Date 2021-12-10.17:07:21.760>
    actor = 'iritkatriel'
    assignee = 'none'
    closed = False
    closed_date = None
    closer = None
    components = ['Library (Lib)']
    creation = <Date 2016-11-04.15:19:23.095>
    creator = 'rrt'
    dependencies = []
    files = ['45357']
    hgrepos = []
    issue_num = 28609
    keywords = ['patch']
    message_count = 12.0
    messages = ['280052', '280072', '280074', '280161', '280169', '280171', '280180', '280181', '280230', '288644', '343614', '408225']
    nosy_count = 6.0
    nosy_names = ['benhoyt', 'paul.j3', 'charlie.proctor', 'rrt', 'svenevs', 'iritkatriel']
    pr_nums = []
    priority = 'normal'
    resolution = None
    stage = None
    status = 'open'
    superseder = None
    type = 'behavior'
    url = 'https://bugs.python.org/issue28609'
    versions = ['Python 3.9', 'Python 3.10', 'Python 3.11']

    @rrt
    Copy link
    Mannequin Author

    rrt mannequin commented Nov 4, 2016

    In Python 3.5.2, with a positional argument with nargs='*', running the program with no arguments gives an error like this:

    usage: caffeinate [-h] [-V] COMMAND [ARGUMENT [ARGUMENT ...]]
    caffeinate: error: the following arguments are required: COMMAND, ARGUMENT

    Here it is clear from the (correct, though redundant) syntax that "ARGUMENT" is optional, but the error message claims, incorrectly, that it is required.

    The add_argument calls used were:

    parser.add_argument('COMMAND', help='command to run')
    parser.add_argument('ARGUMENT', nargs='*', help='arguments to COMMAND')
    parser.add_argument('-V', '--version', action='version', version=PROGRAM_NAME + ' ' + VERSION)

    @rrt rrt mannequin added stdlib Python modules in the Lib dir type-bug An unexpected behavior, bug, or error labels Nov 4, 2016
    @charlierproctor
    Copy link
    Mannequin

    charlierproctor mannequin commented Nov 4, 2016

    I agree that the message is slightly misleading.

    Uploading one possible solution. I'm sure someone more familiar with the library might have a better approach...

    @rrt
    Copy link
    Mannequin Author

    rrt mannequin commented Nov 4, 2016

    Thanks very much for this. It would be great if the redundancy I referred to in the usage message could also be removed, so that instead of "[ARGUMENT [ARGUMENT ...]]" it just said "[ARGUMENT ...]".

    (Similarly, for a '+' argument, it could say just

    ARGUMENT ...

    )

    @paulj3
    Copy link
    Mannequin

    paulj3 mannequin commented Nov 6, 2016

    The current error message is the result of http://bugs.python.org/issue10424 and http://bugs.python.org/issue12776

    Before the test was just:

       if positionals:
           self.error(_('too few arguments'))

    The 2nd patch reworked the test to include the revised handling of defaults.

    So the current error message just lists all the positionals which haven't been consumed. ARGUMENT wasn't consumed because COMMAND wasn't consumed. And technically that would be true even if ARGUMENT required arguments. Well, to be pickier, it as a 're' pattern like 'AA*' that failed.

    The proposed patch looks like it would work, but I haven't tested or looked at the unittests. But I wonder if such a patch is really needed. Are users really misled by the the current message?

    ===============

    As for the usage, the current version allows you to give a tuple METAVAR, producing lines like:

    In [359]: a.metavar=('A','B')
    In [360]: parser.print_usage()
    usage: ipython3 [-h] [A [B ...]]
    In [361]: a.nargs='+'
    In [362]: parser.print_usage()
    usage: ipython3 [-h] A [B ...]
    

    This display pattern is generated in HelpFormater._format_args, with these lines

        elif action.nargs == ZERO_OR_MORE:
            result = '[%s [%s ...]]' % get_metavar(2)
        elif action.nargs == ONE_OR_MORE:
            result = '%s [%s ...]' % get_metavar(2)
        elif action.nargs == REMAINDER:
            result = '...'
    

    You could subclass HelpFormatter, and replace this method with one that performs as you want, (I just tried this)

    result = '[%s ...]' % get_metavar(1)
    

    I wouldn't recommend this as default change, but if there was a enough demand it could added as another HelpFormatter subclass.

    METAVAR lets me approximate your shorter version:

    In [4]: p.print_usage()
    usage: ipython3 [-h] [pos [pos ...]]
    In [5]: a.metavar=('pos','')
    In [6]: p.print_usage()
    usage: ipython3 [-h] [pos [...]]
    In [7]: a.nargs='+'
    In [8]: p.print_usage()
    usage: ipython3 [-h] pos [...]
    

    It still has the [], but the name is gone.

    @paulj3
    Copy link
    Mannequin

    paulj3 mannequin commented Nov 6, 2016

    Try nargs='?' or try providing a default along with the '*'.

    Including your ARGUMENT action in the error message is intentional.

    The test for this error message is:

            required_actions = []
            for action in self._actions:
                if action not in seen_actions:
                    if action.required:

    Originally the code just checked if positionals was empty. That was the list of positional Actions. Actions were popped as they were parsed. Now it maintains a set seen_actions, and checks the required attribute. This test applies to both positionals and optionals.

    For optionals, required is set by the programmer. But for positionals it is set with:

    def _get_positional_kwargs
         ...
        # mark positional arguments as required if at least one is
        # always required
        if kwargs.get('nargs') not in [OPTIONAL, ZERO_OR_MORE]:
            kwargs['required'] = True
        if kwargs.get('nargs') == ZERO_OR_MORE and 'default' not in kwargs:
            kwargs['required'] = True
    

    So for '?' argument, required is False. But for '*', it must also have a 'default' parameter (not just the default None).

    So the proposed patch is overriding the 'required' value that was set during 'add_argument'. And issuing this error message is the main purpose of the 'required' attribute.

    I would not implement this patch.

    But it would be a good idea to check if this method of setting the required attribute has been discussed in other bug/issues. (There is an open issue concerning how the 'required' is set/or not for the subparsers positional.)

    Off hand I don't see anything about this in the documentation. Maybe that's what needs to be patched. (It's easier and safer to change the documentation than the working code. Backward compatibility is a major concern when changing the code.)

    @rrt
    Copy link
    Mannequin Author

    rrt mannequin commented Nov 6, 2016

    Try nargs='?' or try providing a default along with the '*'.

    Why would I do that? I want 0 or more arguments, and there's no default value.

    Including your ARGUMENT action in the error message is intentional.

    It will likely confuse the user. The syntax message says that it's optional, the error suggests (wrongly) that it is required.

    Patching the Python documentation will not help in this case, because the user of my program will not be reading the Python documentation to understand how it works, only the program's own documentation.

    Note that I did not suggest that the behavior be changed, only the message that is output.

    The analysis of why the message is output is useful, but it does not demonstrate that the error message is correct; the error message, as I've already demonstrated, is undeniably wrong. In no sense is "ARGUMENT" missing, because 0 occurrences are entirely legal. Therefore although in terms of the API the argument is "required", it makes no sense to tell the user this (the user will assume that "required" has its colloquial sense, not a technical sense).

    I entirely sympathise with the argument that the behavior of argparse should not change; I see nothing wrong with the behavior, in any case.

    The problems are purely cosmetic:

    1. The syntax message says, redundantly and confusingly, "[ARGUMENT [ARGUMENT ...]]" where it should say just "[ARGUMENT ...]".

    2. The error message says that ARGUMENT is "required", whereas, from the user's point of view, it clearly is not.

    These are serious annoyances from the developer's point of view (in this case, from my point of view), because they mean that in order to release a production-quality tool, I have to hack around argparse's shortcomings.

    So in fact, you're quite right that the documentation should be fixed; only in this case it is the documentation generated by argparse, not the documentation of argparse (which, again, is fine as far as I can see).

    @paulj3
    Copy link
    Mannequin

    paulj3 mannequin commented Nov 7, 2016

    Simply including a default parameter, even with the default default None, changes the error message

     In [395]: parser=argparse.ArgumentParser()
     In [396]: parser.add_argument('cmd');
     In [397]: a=parser.add_argument('args',nargs='*',default=None)
     In [398]: a.required
     Out[398]: False
     In [399]: parser.parse_args([])
    
     usage: ipython3 [-h] cmd [args [args ...]]
     ipython3: error: the following arguments are required: cmd
    

    You shouldn't see any other changes in behavior (except maybe if the positional is in a mutually_exclusive_group). The code that sets 'required' for positionals only looks for the presence of the parameter in kwargs, not its value: 'default' not in kwargs.

    An alternative is to change the value of 'required' after creation:

     a.required = False
    

    Anyways I remain convinced that changing the 'required' attribute is the correct way to change the error message, not adding more tests to the message generator.

    @paulj3
    Copy link
    Mannequin

    paulj3 mannequin commented Nov 7, 2016

    My suggestion to use metavar=('A','') to streamline the usage creates problems with the help code

    http://bugs.python.org/issue14074

    The tuple metavar does not work right for positionals. That's a old issue that should have been fixed long ago.

    So streamlining the usage has to be done with a custom HelpFormatter subclass.

    @rrt
    Copy link
    Mannequin Author

    rrt mannequin commented Nov 7, 2016

    Thanks, that's a simple, robust workaround.

    I'll duck out now and leave the Python experts to sort out the underlying problem, if they can; I think it's still well worth sorting out, though documenting the problem and workaround would be much better than nothing.

    @benhoyt
    Copy link
    Mannequin

    benhoyt mannequin commented Feb 27, 2017

    I definitely agree with Reuben here -- I just ran into this issue while creating a "production quality" tool, and the help message produced by argparse with nargs='*' confused me too. It's pretty clear that this is a simple bug in argparse's production of that usage message: it says ARGUMENT is required, but it isn't.

    However, the workaround of specifying an (unused) default is a reasonable workaround in the meantime -- thanks Paul.

    @svenevs
    Copy link
    Mannequin

    svenevs mannequin commented May 27, 2019

    For optionals, required is set by the programmer. But for positionals it is set with: ...
    So for '?' argument, required is False. But for '*', it must also have a 'default' parameter (not just the default None).
    So the proposed patch is overriding the 'required' value that was set during 'add_argument'. And issuing this error message is the main purpose of the 'required' attribute.

    nargs='*' being marked as required is incorrect though, isn't it?

    I was also very confused by this behavior, the only reason I found this bug was to search before opening new, and had a patch prepared that is nearly identical to the one here.

    1. It's really helpful to know about explicitly setting default=None|[] depending on use case. Would a docs PR briefly explaining at the bottom of the nargs [a] docs explaining how to change the error message via default be welcome? This is a slightly challenging problem to search for.

    2. My understanding is the ultimate rejection of the patch is because it's bypassing the required attribute. So to fix this adequately, changing ? and * to not show up as required (when no default provided) should be done?

    [a] https://docs.python.org/3/library/argparse.html#nargs

    @iritkatriel
    Copy link
    Member

    Reproduced on 3.11.

    >>> parser.parse_args([])
    usage: [-h] [-V] COMMAND [ARGUMENT ...]
    : error: the following arguments are required: COMMAND, ARGUMENT

    @iritkatriel iritkatriel added 3.9 only security fixes 3.10 only security fixes 3.11 only security fixes labels Dec 10, 2021
    @ezio-melotti ezio-melotti transferred this issue from another repository Apr 10, 2022
    @vkoukis
    Copy link

    vkoukis commented Jan 26, 2023

    I would like to say that I also bumped into this, the usage text / error message is self-contradictory. The end result is very confusing to users.

    I understand there is a workaround, but I couldn't find any mention of it anywhere in the documentation, and Googling for this buggy argparse behavior didn't come up with this GitHub issue.

    vkoukis added a commit to arrikto/kubessh that referenced this issue Feb 1, 2023
    Mimic what 'ssh' does more closely, by invoking a shell inside the
    container and using it to run the specified command. To do this,
    concatenate the full argument list into a space-separated list and pass
    it to the shell via its '-c' argument, e.g.:
    
    /bin/sh -c 'command arg1 arg2'
    
    Callers of the 'ssh' binary expect this behavior, and it really helps
    when using the tool interactively, e.g., one can specify shell pipelines
    directly. But it diverges from what 'kubectl exec' supports, so add an
    argument, --no-shell, to disable this default behavior.
    
    Also disable prefix matching when passing arguments, and work around a
    longstanding upstream bug with Python's argparse, see:
    
    python/cpython#72795
    
    Signed-off-by: Vangelis Koukis <vkoukis@arrikto.com>
    Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
    Labels
    3.9 only security fixes 3.10 only security fixes 3.11 only security fixes stdlib Python modules in the Lib dir type-bug An unexpected behavior, bug, or error
    Projects
    Status: Bugs
    Development

    No branches or pull requests

    2 participants