Skip to content

Latest commit

 

History

History
759 lines (488 loc) · 45.3 KB

click.md

File metadata and controls

759 lines (488 loc) · 45.3 KB

Click

  • Welcome to Click — Click Documentation (7.x)

    • Click is a Python package for creating beautiful command line interfaces in a COMPOSABLE way with as little code as necessary. It’s the “Command Line Interface Creation Kit”. It’s HIGHLY CONFIGURABLE but comes with SENSIBLE DEFAULTS out of the box.

      Composability - Wikipedia 的說法 -- "A highly composable system provides components that can be selected and ASSEMBLED IN VARIOUS COMBINATIONS to satisfy specific user requirements.",就是可以任意組合的設計,猜想文件一直提到的 composable/composibility 指的就是 "subcommand/option/argument 編輯的自由度"?

      不過其中的 highly configurable,不包含 help message/page,下面會說明。

    • It aims to make the process of writing command line tools quick and fun while also preventing any frustration caused by the INABILITY to implement an intended CLI API.

    • Click in three points:

      • arbitrary nesting of commands 也就是 subcommand
      • automatic help page generation 下面 $ python hello.py --help 輸出還滿像樣的,沒有像 Python Fire help message 很鳥的問題。
      • supports lazy loading of subcommands at runtime 如果 subcommand 的實作很大,但一般情況沒什麼差。
  • Why Click? — Click Documentation (7.x)

    • There are so many libraries out there for writing command line utilities; why does Click exist? This question is easy to answer: because there is not a single command line utility for Python out there which ticks the following boxes:

      • is LAZILY COMPOSABLE without restrictions
      • supports implementation of Unix/POSIX command line CONVENTIONS 符合使用慣例很重要,CLI 該有的樣子
      • supports loading values from ENVIRONMENT VARIABLES out of the box 可惜環境變數不會出現在 help message 裡;可以自行補充嗎??
      • supports for prompting of custom values 尤其需要輸入密碼時
      • is fully nestable and composable
      • works the same in Python 2 and 3
      • supports file handling out of the box ??
      • comes with useful common HELPERS (getting terminal dimensions, ANSI colors, fetching direct keyboard input, screen clearing, finding config paths, launching apps and editors, etc.) 省掉很多事
    • There are many alternatives to Click and you can have a look at them if you enjoy them better. The obvious ones are optparse and argparse from the standard library. 雖然 Click 是基於 optparse,但根本不能比吧?

    • Click actually implements ITS OWN PARSING of arguments and does not use optparse or argparse following the optparse parsing behavior. The reason it’s not based on argparse is that argparse does not allow proper NESTING OF COMMANDS by design and has some deficiencies when it comes to POSIX COMPLIANT ARGUMENT HANDLING.

      這跟下面 "internally based on optparse" 的說法衝突,不過 argument handling 不足的地方是指??

    • Click is designed to be fun to work with and at the same time not stand in your way. It’s not OVERLY FLEXIBLE either. Currently, for instance, it does not allow you to customize the help pages too much. This is intentional because Click is designed to allow you to NEST command line utilities. The idea is that you can have a system that works together with another system by tacking two Click instances together and they will continue working as they should. 不懂為何 nest command (也就是 subcommand) 的支援跟 "works together with another system" 有關?? 又自訂 help page 跟 nesting 有什麼關係??

      Too much customizability would break this promise. 所幸 help page 看起來還不錯,不太需要自訂。

    • Click was written to support the Flask microframework ecosystem because no tool could provide it with the functionality it needed.

      所以 Flask CLI 的那一塊是其他 library 不支援的??

    • To get an understanding of what Click is all about, I strongly recommend looking at the Complex Applications chapter to see what it’s useful for. 複雜的應用才能看出 Click 的能耐

    Why not Argparse?

    • Click is internally based on optparse instead of argparse. This however is an implementation detail that a user does not have to be concerned with. The reason however Click is not using argparse is that it has some problematic behaviors that make handling arbitrary command line interfaces hard:

      • argparse has built-in magic behavior to GUESS if something is an ARGUMENT or an OPTION. This becomes a problem when dealing with INCOMPLETE command lines as it’s not possible to know without having a full understanding of the command line how the parser is going to behave. This goes against Click’s ambitions of dispatching to SUBPARSERS. 若 subcommand 的用法不完整,也要能由 subcommand 指出錯誤的意思??
      • argparse currently does not support disabling of interspersed (散置的) arguments. Without this feature it’s not possible to safely implement Click’s NESTED PARSING nature.

    Why not Docopt etc.?

    • Docopt and many tools like it are cool in how they work, but very few of these tools deal with nesting of commands and composability in a way like Click. To the best of the developer’s knowledge, Click is the first Python library that aims to create a level of COMPOSABILITY of applications that goes beyond what the system itself supports. 超越 OS 對 CLI 的支持??

    • Docopt, for instance, acts by PARSING YOUR HELP PAGES and then parsing according to those rules. The side effect of this is that docopt is quite rigid (死板的) in how it handles the command line interface. The upside of docopt is that it gives you STRONG CONTROL OVER YOUR HELP PAGE; the downside is that due to this it cannot REWRAP your output for the current terminal width and it makes translations hard. On top of that docopt is restricted to basic parsing. It does not handle ARGUMENT DISPATCHING and callback invocation or TYPES. This means there is a lot of code that needs to be written in addition to the basic help page to handle the parsing results.

      為了美美的 help page 導致許多雜事都要自己來,好像不怎麼划算? 畢竟 CLI 的重點是 command line 的 UX。

    • Most of all, however, it makes composability hard. While docopt does support dispatching to subcommands, it for instance does not directly support any kind of AUTOMATIC SUBCOMMAND ENUMERATION?? based on what’s available or it does not enforce subcommands to work in a consistent way.

    • This is fine, but it’s different from how Click wants to work. Click aims to support fully composable command line user interfaces by doing the following:

      • Click does not just parse, it also DISPATCHES to the appropriate code. 從首頁的範例就可以看出,直接對應到 function。
      • Click has a strong concept of an INVOCATION CONTEXT?? that allows subcommands to respond to DATA FROM THE PARENT COMMAND. 什麼情況下需要拿 parent data??
      • Click has strong information available for all parameters and commands so that it can generate UNIFIED help pages for the full CLI and to assist the user in converting the input data as necessary.
      • Click has a strong understanding of what types are and can give the user CONSISTENT ERROR MESSAGES if something goes wrong. (也要符合慣例) A subcommand written by a different developer will not suddenly die with a different error message because it’s manually handled.
      • Click has enough META INFORMATION available for its whole program that it can evolve over time to improve the user experience without forcing developers to adjust their programs. For instance, if Click decides to change how help pages are formatted, all Click programs will automatically benefit from this.
    • The aim of Click is to make COMPOSABLE systems, whereas the aim of docopt is to build the most BEAUTIFUL and hand-crafted command line interfaces. (針對 help page,但重點還是 elegant UX) These two goals conflict with one another in subtle ways. Click actively prevents people from implementing certain patterns in order to achieve UNIFIED command line interfaces. You have very little input on reformatting your help pages for instance.

    Why Hardcoded Behaviors?

    • The other question is why Click goes away from optparse and hardcodes certain behaviors instead of staying configurable. There are multiple reasons for this. The biggest one is that too much configurability makes it hard to achieve a CONSISTENT COMMAND LINE EXPERIENCE.
    • The best example for this is optparse’s callback functionality for accepting an arbitrary number of arguments. Due to syntactical ambiguities on the command line, there is no way to implement fully variadic?? arguments. There are always tradeoffs that need to be made and in case of argparse these tradeoffs have been critical enough, that a system like Click cannot even be implemented on top of it. 採用 argparse 會失去某些彈性?
    • In this particular case, Click attempts to stay with A HANDFUL OF ACCEPTED PARADIGMS for building command line interfaces that can be well documented and tested.

    Why No Auto Correction?

    • The question came up why Click does not auto correct parameters given that even optparse and argparse support automatic expansion?? of long arguments. The reason for this is that it’s a liability (不利條件) for backwards compatibility. If people start relying on automatically modified parameters and someone adds a new parameter in the future, the script might stop working. These kinds of problems are hard to find so Click does not attempt to be magical about this. 聽起來 "auto correction" 有點危險??
    • This sort of behavior however can be implemented on a higher level to support things such as explicit ALIASES. For more information see Command Aliases. #ril

新手上路 {: #getting-started }

  • Welcome to Click — Click Documentation (7.x)

    • What does it look like? Here is an example of a simple Click program:

      import click
      
      @click.command() <-- 大量的 decorator
      @click.option('--count', default=1, help='Number of greetings.')
      @click.option('--name', prompt='Your name',
                    help='The person to greet.')
      def hello(count, name): <-- 把 function 直接當 command 用,這點跟 Python Fire 很像
          """Simple program that greets NAME for a total of COUNT times."""
          for x in range(count):
              click.echo('Hello %s!' % name)
      
      if __name__ == '__main__':
          hello() <-- 雖然 hello() 要兩個參數,但呼叫的對像是 decorator 包裝過的版本
      

      And what it looks like when run:

      $ python hello.py --count=3
      Your name: John <-- prompt 參數在作用
      Hello John!
      Hello John!
      Hello John!
      

      It automatically generates nicely formatted help pages: 看起來滿像樣的!!

      $ python hello.py --help
      Usage: hello.py [OPTIONS]
      
        Simple program that greets NAME for a total of COUNT times.
      
      Options:
        --count INTEGER  Number of greetings.
        --name TEXT      The person to greet.
        --help           Show this message and exit.
      
  • Complex Applications — Click Documentation (7.x) #ril

Parameter, Option, Argument ??

  • Class click.Parameter - API — Click Documentation (7.x) #ril

    class click.Parameter(param_decls=None, type=None, required=False, default=None, callback=None, nargs=None, metavar=None, expose_value=True, is_eager=False, envvar=None, autocompletion=None)
    
    • A parameter to a command comes in two versions: they are either Options or Arguments. Other subclasses are currently not supported by design as some of the internals for parsing are intentionally NOT FINALIZED.

      Some settings are supported by both options and arguments.

      泛稱做 parameter,再進一步區分為前面通常是 optional 的 option,與後面通常是 required 的 argument;也因此 option 與 argument 有一些通用的特性。

    • Changed in version 2.0: Changed signature for parameter callback to also be passed the parameter. In Click 2.0, the old callback format will still work, but it will raise a warning to give you change to migrate the code easier. ??

    Parameters

    • param_decls

      the parameter declarations for this option or argument. This is a list of FLAGS or ARGUMENT NAMES.

      其中 "flag" 的說法是針對 option,因為是 -x--xxx 的用法。

    • type

      the type that should be used. Either a ParamType or a Python type. The later is converted into the former automatically if supported.

      似乎沒有現成的 ParamType 可用 ??

    • required

      controls if this is optional or not.

      如果不是 required 又沒有 default,會拿到 None ??

    • default

      the default value if omitted. This can also be a CALLABLE, in which case it’s invoked when the default is needed WITHOUT ANY ARGUMENTS.

      支持 callable 這一點還滿酷的,不用在 import time 就決定 default value。

    • callback

      a callback that should be executed AFTER the parameter was MATCHED. This is called as fn(ctx, param, value) and needs to return the value. Before Click 2.0, the signature was (ctx, value).

      看似有機會把傳進來的 value 加工過,不過 value 是經過 type 轉換的 ??

    • nargs

      the number of arguments to match. If not 1 the return value is a TUPLE instead of single value. The default for nargs is 1 (except if the type is a tuple, then it’s the arity of the tuple).

    • metavar

      how the value is REPRESENTED IN THE HELP PAGE.

    • expose_value

      if this is True then the value is passed onwards to the COMMAND CALLBACK and stored on the context, otherwise it’s skipped. ??

    • is_eager

      eager values are PROCESSED BEFORE NON EAGER ONES. This should not be set for arguments or it will inverse the order of processing.

      感覺跨多個 paramter 的驗證可以利用這個 ??

    • envvar

      a string or LIST OF STRINGS that are environment variables that should be checked.

      對應多個環境變數是什麼概念?

  • Class click.Option - API — Click Documentation (7.x) #ril

    class click.Option(param_decls=None, show_default=False, prompt=False, confirmation_prompt=False, hide_input=False, is_flag=None, flag_value=None, multiple=False, count=False, allow_from_autoenv=True, type=None, help=None, hidden=False, show_choices=True, show_envvar=False, **attrs)
    
    • Options are USUALLY OPTIONAL values on the command line and have some extra features that arguments don’t have.

      All other parameters are passed onwards to the parameter constructor.

    Parameters

    • show_default

      controls if the default value should be shown on the help page. Normally, defaults are NOT shown. If this value is a string, it shows the string INSTEAD OF THE VALUE. This is particularly useful for DYNAMIC OPTIONS. ??

    • show_envvar

      controls if an environment variable should be shown on the help page. Normally, environment variables are NOT shown.

    • prompt

      if set to True or a non empty string then the user will be prompted for input. If set to True the prompt will be the option name capitalized.

      主要是用來自訂提示文件,但設為 True 時會自動從 option name 轉換出提示文字。

      prompt 在控制使用者沒有提供 option 時要不要提示輸入,跟有沒有預設值無關;搭配 default 使用時,只是會一併提示預設值而已,除非 hide_input 設為 True

    • confirmation_prompt

      if set then the value will need to be confirmed if it was prompted for.

    • hide_input

      if this is True then the input on the PROMPT will be hidden from the user. This is useful for password input.

    • is_flag

      forces this option to act as a flag. The default is auto detection.

      所以 option 不一定是 flag ?? flag 是 boolean ??

    • flag_value

      which value should be used for this flag if it’s enabled. This is set to a boolean automatically if the option string contains a slash to mark two options. ??

    • multiple

      if this is set to True then the argument is accepted multiple times and recorded. This is similar to nargs in how it works but supports arbitrary number of arguments. 這有什麼差別 ??

    • count

      this flag makes an option INCREMENT AN INTEGER.

      用多次,數字就會疊加 ??

    • allow_from_autoenv

      if this is enabled then the value of this parameter will be pulled from an environment variable in case a PREFIX is defined on the context. ??

    • help

      the help string.

    • hidden

      hide this option from help outputs.

Validation ??

Nesting Commands ??

  • Commands and Groups — Click Documentation (7.x) #ril

    • The most important feature of Click is the concept of ARBITRARILY NESTING command line utilities. This is implemented through the Command and Group (actually MultiCommand).

    Callback Invocation

    • For a regular command, the callback is executed WHENEVER THE COMMAND RUNS. If the script is the only command, it will always fire (unless a parameter callback prevents it. This for instance happens if someone passes --help to the script).

      所謂 script 就是使用者在 command line 一長串最原始的輸入;不過都加 --help 了,為什麼要執行??

    • For groups and multi commands, the situation looks different. In this case, the callback fires whenever a subcommand fires (unless this behavior is changed). What this means in practice is that AN OUTER COMMAND RUNS WHEN AN INNER COMMAND RUNS:

      @click.group()
      @click.option('--debug/--no-debug', default=False)
      def cli(debug):
          click.echo('Debug mode is %s' % ('on' if debug else 'off'))
      
      @cli.command()  # @cli, not @click!
      def sync():
          click.echo('Syncing')
      

      這個特性可以讓 parent command 將 option 先記錄下來。不過除了 sync --help 看不到 --debug 的說明之外,sync subcomand 如何拿到 debug 的值也是個問題?下面示範了透過 Context.obj 在 command/subommand 間交換資料的做法。

      其實更大的問題是 --debug 只能寫在 sync subcommand 前,否則會出現 no such option: ... 的錯誤,因為那個 option 是宣告在 parent command:

      $ python sync.py --no-debug sync
      Debug mode is off
      Syncing
      
      $ python sync.py sync --no-debug
      Debug mode is off
      Usage: sync.py sync [OPTIONS]
      Try "sync.py sync --help" for help.
      
      Error: no such option: --no-debug
      
    • Here is what this looks like:

      $ tool.py
      Usage: tool.py [OPTIONS] COMMAND [ARGS]...
      
      Options:
        --debug / --no-debug
        --help                Show this message and exit.
      
      Commands:
        sync
      
      $ tool.py --debug sync
      Debug mode is on
      Syncing
      

      這裡避開了 sync subcommand 如何拿到 debug 的問題 ...

    Nested Handling and Contexts

    • As you can see from the earlier example, the basic command group accepts a debug argument which is passed to its callback, but not to the sync command itself. The sync command only accepts its own arguments.

      @click.group()
      @click.option('--debug/--no-debug', default=False)
      def cli(debug):
          click.echo('Debug mode is %s' % ('on' if debug else 'off'))
      
      @cli.command()  # @cli, not @click!
      def sync():
          click.echo('Syncing')
      
    • This allows tools to act COMPLETELY INDEPENDENT of each other, but how does one command talk to a nested one? The answer to this is the Context.

      Each time a command is invoked, a new context is created and linked with the PARENT CONTEXT. Normally, you can’t see these contexts, but they are there. Contexts are passed to parameter callbacks together with the value automatically. Commands can also ask for the context to be passed by marking themselves with the pass_context() decorator. In that case, the context is passed as FIRST ARGUMENT.

      Command/subcommand 之間可以透過 context 交換資料,但這裡卻完全沒提到 Context.meta?? 從 "created and linked with the parent context" 看來,每一層 command 都有自己的 context (透過 context.parent 取得 parent context),雖然最後提到不一定要走 context,因為 Python 的 global/module 就可以當做媒介。

    • The context can also carry a PROGRAM SPECIFIED OBJECT that can be used for the program’s purposes. What this means is that you can build a script like this:

      @click.group()
      @click.option('--debug/--no-debug', default=False)
      @click.pass_context
      def cli(ctx, debug):
          # ensure that ctx.obj exists and is a dict (in case `cli()` is called
          # by means other than the `if` block below
          ctx.ensure_object(dict)
      
          ctx.obj['DEBUG'] = debug
      
      @cli.command()
      @click.pass_context <-- 注意這裡又用回 @click 了
      def sync(ctx):
          click.echo('Debug is %s' % (ctx.obj['DEBUG'] and 'on' or 'off'))
      
      if __name__ == '__main__':
          cli(obj={})
      

      If the object is provided, EACH CONTEXT will PASS the object onwards to its CHILDREN, but at any level a context’s object can be overridden. To reach to a parent, context.parent can be used.

      若將其初始化為 dict,用起來就跟 Context.meta 一樣了,似乎沒什麼意思?

      In addition to that, instead of passing an object down, nothing stops the application from modifying GLOBAL STATE. For instance, you could just flip a global DEBUG variable and be done with it.

  • Context.meta - API — Click Documentation (7.x)

    • This is a dictionary which is SHARED with all the contexts that are nested. It exists so that CLICK UTILITIES can store some state here if they need to. It is however the responsibility of that code to manage this dictionary well.

    • The keys are supposed to be UNIQUE DOTTED STRINGS. For instance MODULE PATHS are a good choice for it. What is stored in there is irrelevant for the operation of click. However what is important is that code that places data here adheres to the general semantics of the system.

      LANG_KEY = __name__ + '.lang'
      
      def set_language(value):
          ctx = get_current_context()
          ctx.meta[LANG_KEY] = value
      
      def get_language():
          return get_current_context().meta.get(LANG_KEY, 'en_US')
      

      意思是 Click 自己也會用,命名上採 module name 為 prefix 可以避免衝突? 難怪範例會用 __name__ 做為 prefix。

  • python - Click group with options and commands at the same time - Stack Overflow @click.group(invoke_without_command=True) 搭配 @cmd.command(default_command=True) 就可以省略 subcommand #ril

  • Python Click - only execute subcommand if parent command executed successfully - Stack Overflow 出現 ctx.obj['xxx'] 的用法 #ril

  • In Python Click how do I see --help for Subcommands whose parents have required arguments? - Stack Overflow #ril

Help Page ??

  • Documenting Scripts — Click Documentation (7.x) #ril

    • Click makes it very easy to document your command line tools. First of all, it automatically generates help pages for you. While these are currently NOT CUSTOMIZABLE IN TERMS OF THEIR LAYOUT, all of the text can be changed.

    Help Texts

    • Commands and options accept help arguments. In the case of commands, the DOCSTRING of the function is automatically used if provided.

      Simple example:

      @click.command()
      @click.option('--count', default=1, help='number of greetings')
      @click.argument('name')
      def hello(count, name):
          """This script prints hello NAME COUNT times."""
          for x in range(count):
              click.echo('Hello %s!' % name)
      

      And what it looks like:

      $ hello --help
      Usage: hello [OPTIONS] NAME
      
        This script prints hello NAME COUNT times.
      
      Options:
        --count INTEGER  number of greetings
        --help           Show this message and exit.
      
    • Arguments cannot be documented this way. This is to follow the GENERAL CONVENTION of Unix tools of using arguments for only the most necessary things and to DOCUMENT THEM IN THE INTRODUCTION text by referring to them by name.

      add help to click argument · Issue #587 · pallets/click davidism: (member) As the docs quoted earlier say, Click INTENTIONALLY does not implement this.

在 Parent/Sub Command 間共用 Option {: #share-options }

docker--debug 為例,不能放在 subcommand 後面,而且 subcommand 的 help page 也看不到 --debug 的說明。

$ docker ps --debug
unknown flag: --debug
See 'docker ps --help'.

雖然有點違反直覺 (見仁見智?),cmd [OPTIONS] subcmd subsubcmd [OPTIONS] 這樣的安排還可以,但中間再插個 subcmd [OPTIONS] 就太多了? 也就是說 option 可以初現在開頭或結尾,遇到這種狀況時,將通用的 option 在 subsubcmd 重新宣告一次即可,至於實作細節如何共用邏輯,在 #108 Support for shared arguments? 的討論裡,mikenerone 提出的做法很實用:

_global_test_options = [
    click.option('--verbose', '-v', 'verbosity', flag_value=2, default=1, help='Verbose output'),
    click.option('--quiet', '-q', 'verbosity', flag_value=0, help='Minimal output'),
    click.option('--fail-fast', '--failfast', '-f', 'fail_fast', is_flag=True, default=False, help='Stop on failure'),
]

def global_test_options(func):
    for option in reversed(_global_test_options):
        func = option(func)
    return func

@click.command()
@global_test_options
@click.option('--start-directory', '-s', default='test', help='Directory (or module path) to start discovery ("test" default)')
def test(verbosity, fail_fast, start_directory):
    # Run tests here

@click.command()
@click.option(
    '--format', '-f', type=click.Choice(['html', 'xml', 'text']), default='html', show_default=True,
    help='Coverage report output format',
)
@global_test_options
@click.pass_context
def cover(ctx, format, verbosity, fail_fast):
    # Setup coverage, ctx.invoke() the test command above, generate report

參考資料:

  • Support for shared arguments? · Issue #108 · pallets/click

    • mahmoudimus: A very simple and trivial example is the verbose example. Assume you have more than one subcommand in a CLI app. An ideal user experience on the CLI would be:

      python script.py subcmd -vvv
      

      However, this wouldn't be the case with click, SINCE subcmd DOESN'T DEFINE A verbose OPTION. You'd have to invoke it as follows:

      python script.py -vvv subcmd
      

      This example is very simple, but when there are many subcommands, sometimes a root support option would go a long way to make something simple and EASY TO USE. Let me know if you'd like further clarification.

    • mitsuhiko (member): This is already really simple to implement through decorators. As an example:

      import click
      
      class State(object):
      
          def __init__(self):
              self.verbosity = 0
              self.debug = False
      
      pass_state = click.make_pass_decorator(State, ensure=True) # 用 global 來存狀態
      
      def verbosity_option(f):
          def callback(ctx, param, value):
              state = ctx.ensure_object(State)
              state.verbosity = value
              return value
          return click.option('-v', '--verbose', count=True,
                              expose_value=False,
                              help='Enables verbosity.',
                              callback=callback)(f) # 用 parameter callback 存進 global state
      
      def debug_option(f):
          def callback(ctx, param, value):
              state = ctx.ensure_object(State)
              state.debug = value
              return value
          return click.option('--debug/--no-debug',
                              expose_value=False,
                              help='Enables or disables debug mode.',
                              callback=callback)(f)
      
      def common_options(f): # 通用的 options
          f = verbosity_option(f)
          f = debug_option(f)
          return f
      
      @click.group()
      def cli():
          pass
      
      @cli.command()
      @common_options
      @pass_state
      def cmd1(state):
          click.echo('Verbosity: %s' % state.verbosity)
          click.echo('Debug: %s' % state.debug)
      
    • mahmoudimus: From your example, is the intention that users build their own common options and annotate all relevant commands? Maybe a better user experience is to register these COMMON OPTIONS to the group and have the group TRANSITIVELY PROPAGATE THEM TO ITS SUBCOMMANDS? I could see arguments for either or.

      mitsuhiko (member): If the option is available on all commands it really does not belong to the option but instead to the group that encloses it. It makes no sense that an option conceptually belongs to the group but is attached to an option in my mind. ??

    • mahmoudimus: Right, and I'm on board with your logic. However, this translates to position dependence for options which causes COGNITIVE LOAD on the user of the cli app, unless the approach above is used to get, what I would consider, the desirable and expected UX. 確實,是 UX 的問題!!

      That's why I'm wondering if it makes sense to have a parameter on the group class which propagates options down to commands to get the desired behavior WITHOUT SURPRISING THE USER.

      If my use case is the outlier (門外漢) in terms of what is desired and expected, then I guess the option of using a similar idiom to what you've demonstrated above is the right way to go.

    • mitsuhiko: Doing this by magic will not happen, that stands against one of the core principles of Click which is to NEVER BREAK BACKWARDS COMPATIBILITY with scripts by adding new parameters/options later. The correct solution is to use a CUSTOM DECORATOR for this. :)

      untitaker (member): How about adding such a decorator to click or a click-contrib package? 團隊內部有不同的看法 ...

    • Stiivi: I'm giving my vote for this feature, as it makes the CLI EXPERIENCE LESS COMPLEX. Even though technically cmd -a subcmd -b subsubcmd -c is correct, cmd subcmd subsubcmd -a -b -c is analogous to have cmd_subcmd_subsubcmd -a -b -c.

    • mikenerone: I think this can be done even more trivially than the given example. Here's a snippet from a helper command I have for running unit tests and/or coverage. Note that several of the options are SHARED between the test and cover subcommands:

      _global_test_options = [
          click.option('--verbose', '-v', 'verbosity', flag_value=2, default=1, help='Verbose output'),
          click.option('--quiet', '-q', 'verbosity', flag_value=0, help='Minimal output'),
          click.option('--fail-fast', '--failfast', '-f', 'fail_fast', is_flag=True, default=False, help='Stop on failure'),
      ]
      
      def global_test_options(func):
          for option in reversed(_global_test_options):
              func = option(func)
          return func
      
      @click.command()
      @global_test_options
      @click.option('--start-directory', '-s', default='test', help='Directory (or module path) to start discovery ("test" default)')
      def test(verbosity, fail_fast, start_directory):
          # Run tests here
      
      @click.command()
      @click.option(
          '--format', '-f', type=click.Choice(['html', 'xml', 'text']), default='html', show_default=True,
          help='Coverage report output format',
      )
      @global_test_options
      @click.pass_context
      def cover(ctx, format, verbosity, fail_fast):
          # Setup coverage, ctx.invoke() the test command above, generate report
      

      這用起來會更直覺,有趣的是 --verbose--quiet 共用一個參數 verbosity,跟 fail_fast 一起出現在參數裡,很直覺!!

  • What is the most elegant way of accepting an option on both the group and subcommands? · Issue #1034 · pallets/click #ril

用 Class 來實作 Subcommand ?? {: #}

Testing ??

  • Testing Click Applications — Click Documentation (7.x) #ril

    • For basic testing, Click provides the click.testing module which provides test functionality that helps you INVOKE command line applications and check their BEHAVIOR.
    • These tools should really only be used for testing as they change the entire INTERPRETER STATE?? for simplicity and are NOT in any way thread-safe!

    Basic Testing

    • The basic functionality for testing Click applications is the CliRunner which can invoke commands as command line scripts. The CliRunner.invoke() method runs the command line script IN ISOLATION and CAPTURES THE OUTPUT as both bytes and binary data. The return value is a Result object, which has the captured output data, exit code, and optional exception attached.

      import click
      from click.testing import CliRunner
      
      def test_hello_world():
          @click.command()
          @click.argument('name')
          def hello(name):
              click.echo('Hello %s!' % name)
      
          runner = CliRunner()
          result = runner.invoke(hello, ['Peter']) # 為什麼 invoke 的對象是 command,而非使用者在 command line 最原始的輸入??
          assert result.exit_code == 0
          assert result.output == 'Hello Peter!\n'
      
  • Testing - API — Click Documentation (7.x) #ril

Slack Slash Command ??

  • 若想在同一個 (Python) process 裡做事 (不想調用外部 CLI),可能會卡在 CLI framework 設計上的限制,例如要攔截 SystemExit、將 STDOUT/STDERR 轉向等。
  • 若調用外部 CLI,上面的問題就沒了,只要把 STDOUT/STDERR 包裝成 Slack message 回傳給使用者即可;多人同時執行也沒問題,只要把 working directory 拆開。
  • 讓 Slack Slash Command 做為 CLI 的 thin wrapper 有幾個好處 -- 在本地端開發時也好測試,除了 Slash command,使用者也可以直接用 CLI,而且這方法也適用於非 Python-based 的 CLI。
  • 但 CLI 與 Slack Slash Command 的 UI (指 command line parsing) 也可能有些微的差異,例如 pipeline、本地檔案操作、密碼輸入提示等,或許應該拉出一層 facade 讓多種 UI 共用?

參考資料:

  • Testing Click Applications — Click Documentation (7.x)

    • These tools should really only be used for testing as they CHANGE THE ENTIRE INTERPRETER STATE for simplicity and are NOT in any way thread-safe!

      看起來很適合用來實作 Slack Slash Command,但上面又說不該用在 testing 以外的地方? 只要 worker 不走 thread 就沒有問題??

      若是把 Slash Command 視為 CLI 的另一層包裝? 也就是 CLI 可以在本地端使用,也可以透過 Slash Command 調用 CLI 外部程式,事情會單純一點?

  • CliRunnter.invoke() - click/testing.py at master · pallets/click #ril

    def invoke(self, cli, args=None, input=None, env=None,
               catch_exceptions=True, color=False, mix_stderr=False, **extra):
    
        exc_info = None
        with self.isolation(input=input, env=env, color=color) as outstreams: # (1)
            exception = None
            exit_code = 0
    
            if isinstance(args, string_types):
                args = shlex.split(args)
    
            try:
                prog_name = extra.pop("prog_name")
            except KeyError:
                prog_name = self.get_default_prog_name(cli)
    
            try:
                cli.main(args=args or (), prog_name=prog_name, **extra) # (2)
            except SystemExit as e:
                exc_info = sys.exc_info()
                exit_code = e.code
                if exit_code is None:
                    exit_code = 0
    
                if exit_code != 0:
                    exception = e
    
                if not isinstance(exit_code, int):
                    sys.stdout.write(str(exit_code))
                    sys.stdout.write('\n')
                    exit_code = 1
    
            except Exception as e:
                if not catch_exceptions:
                    raise
                exception = e
                exit_code = 1
                exc_info = sys.exc_info()
            finally:
                sys.stdout.flush()
                stdout = outstreams[0].getvalue()
                stderr = outstreams[1] and outstreams[1].getvalue()
    
        return Result(runner=self,
                      stdout_bytes=stdout,
                      stderr_bytes=stderr,
                      exit_code=exit_code,
                      exception=exception,
                      exc_info=exc_info)
    
        try:
            cli.main(args=args or (), prog_name=prog_name, **extra)
        except SystemExit as e:
            exc_info = sys.exc_info()
            exit_code = e.code
    
    1. isolation() 會暫時將 STDIN, STDOUT, STDERR 換掉。

      不過這會影響到同一個 process 的其他輸出入,例如 logging 在 STDERR 的輸出。

      或許更不具侵入性的做法是自訂 echo() 內部再轉呼叫 click.echo() ?? 在真正的 CLI 下就送 STDERR/STDOUT,在 Slash Command 下則改送 Slack message。

    2. 只是呼叫 BaseCommand.main()? 從 click.core.main() 的原始碼看來,背後也是調用 BaseCommand.invoke()

      with self.make_context(prog_name, args, **extra) as ctx:
          rv = self.invoke(ctx)
      
  • BaseCommand.main() - API — Click Documentation (7.x) #ril

    • main(args=None, prog_name=None, complete_var=None, standalone_mode=True, **extra)

    • This is the way to invoke a script with all the bells and whistles as a command line application. This will always terminate the application after a call. If this is not wanted, SystemExit needs to be caught.

      明確指出可以攔截 SystemExit,沒有問題。

    • This method is also available by directly calling the instance of a Command.

    • New in version 3.0: Added the standalone_mode flag to control the standalone mode??.

    Parameters

    • args – the arguments that should be used for parsing. If not provided, sys.argv[1:] is used. (不含 program name 本身)
    • prog_name – the program name that should be used. By default the program name is constructed by taking the file name from sys.argv[0].
    • complete_var – the environment variable that controls the bash completion support. The default is "_<prog_name>_COMPLETE" with prog_name in uppercase.
    • standalone_mode – the default behavior is to invoke the script in STANDALONE MODE. Click will then handle exceptions and convert them into error messages and the function will never return but SHUT DOWN the interpreter. (也就是會丟出 SystemExit) If this is set to False they will be propagated to the caller and the return value of this function is the return value of invoke().
    • extra – extra keyword arguments are forwarded to the context constructor. See Context for more information.
  • Context API — Click Documentation (7.x) #ril

    invoke(**kwargs)

    • Invokes a command callback in exactly the way it expects. There are two ways to invoke this method:
      • the first argument can be a CALLBACK and all other arguments and keyword arguments are forwarded directly to the function.
      • the first argument is a click COMMAND OBJECT. In that case all arguments are forwarded as well but proper click parameters (options and click arguments) must be keyword arguments and Click will fill in defaults.

安裝設置 {: #setup }

參考資料 {: #reference }

社群:

手冊: