========================================== Configurable objects with traitlets.config ========================================== .. module:: traitlets.config This document describes :mod:`traitlets.config`, the traitlets-based configuration system used by IPython and Jupyter. The main concepts ================= There are a number of abstractions that the IPython configuration system uses. Each of these abstractions is represented by a Python class. Configuration object: :class:`~traitlets.config.Config` A configuration object is a simple dictionary-like class that holds configuration attributes and sub-configuration objects. These classes support dotted attribute style access (``cfg.Foo.bar``) in addition to the regular dictionary style access (``cfg['Foo']['bar']``). The Config object is a wrapper around a simple dictionary with some convenience methods, such as merging and automatic section creation. Application: :class:`~traitlets.config.Application` An application is a process that does a specific job. The most obvious application is the :command:`ipython` command line program. Each application may read configuration files and a single set of command line options and then produces a master configuration object for the application. This configuration object is then passed to the configurable objects that the application creates, usually either via the `config` or `parent` constructor arguments. These configurable objects implement the actual logic of the application and know how to configure themselves given the configuration object. Applications always have a `log` attribute that is a configured Logger. This allows centralized logging configuration per-application. Configurable: :class:`~traitlets.config.Configurable` A configurable is a regular Python class that serves as a base class for all main classes in an application. The :class:`~traitlets.config.Configurable` base class is lightweight and only does one thing. This :class:`~traitlets.config.Configurable` is a subclass of :class:`~traitlets.HasTraits` that knows how to configure itself. Class level traits with the metadata ``config=True`` become values that can be configured from the command line and configuration files. Developers create :class:`~traitlets.config.Configurable` subclasses that implement all of the logic in the application. Each of these subclasses has its own configuration information that controls how instances are created. When constructing a :class:`~traitlets.config.Configurable`, the `config` or `parent` arguments can be passed to the constructor (respectively a :class:`~traitlets.config.Config` and a Configurable object with a ``.config``). Singletons: :class:`~traitlets.config.SingletonConfigurable` Any object for which there is a single canonical instance. These are just like Configurables, except they have a class method :meth:`~traitlets.config.SingletonConfigurable.instance`, that returns the current active instance (or creates one if it does not exist). :class:`~traitlets.config.Application` is a singleton. This lets objects easily connect to the current running Application without passing objects around everywhere. For instance, to get the current running Application instance, simply do: ``app = Application.instance()``. .. note:: Singletons are not strictly enforced - you can have many instances of a given singleton class, but the :meth:`instance` method will always return the same one. Having described these main concepts, we can now state the main idea in our configuration system: *"configuration" allows the default values of class attributes to be controlled on a class by class basis*. Thus all instances of a given class are configured in the same way. Furthermore, if two instances need to be configured differently, they need to be instances of two different classes. While this model may seem a bit restrictive, we have found that it expresses most things that need to be configured extremely well. However, it is possible to create two instances of the same class that have different trait values. This is done by overriding the configuration. Now, we show what our configuration objects and files look like. Configuration objects and files =============================== A configuration object is little more than a wrapper around a dictionary. A configuration *file* is simply a mechanism for producing that object. Configuration files currently can be a plain Python script or a JSON file. The former has the benefit that it can perform extensive logic to populate the config object, while the latter is just a direct JSON serialization of the config dictionary and can be easily processed by external software. When both Python and JSON configuration file are present, both will be loaded, with JSON configuration having higher priority. The configuration files can be loaded by calling :meth:`Application.load_config_file()`, which takes the relative path to the config file (with or without file extension) and the directories in which to search for the config file. All found configuration files will be loaded in reverse order, so that configs in earlier directories will have higher priority. Python configuration Files -------------------------- A Python configuration file is a pure Python file that populates a configuration object. This configuration object is a :class:`~traitlets.config.Config` instance. It is available inside the config file as ``c``, and you simply set attributes on this. All you have to know is: * The name of the class to configure. * The name of the attribute. * The type of each attribute. The answers to these questions are provided by the various :class:`~traitlets.config.Configurable` subclasses that an application uses. Let's look at how this would work for a simple configurable subclass .. code-block:: python # Sample configurable: from traitlets.config.configurable import Configurable from traitlets import Int, Float, Unicode, Bool class School(Configurable): name = Unicode("defaultname", help="the name of the object").tag(config=True) ranking = Integer(0, help="the class's ranking").tag(config=True) value = Float(99.0) # The rest of the class implementation would go here.. # Construct from config via School(config=..) In this example, we see that :class:`School` has three attributes, two of which (``name``, ``ranking``) can be configured. All of the attributes are given types and default values. If a :class:`School` is instantiated, but not configured, these default values will be used. But let's see how to configure this class in a configuration file .. code-block:: python # Sample config file c.School.name = "coolname" c.School.ranking = 10 After this configuration file is loaded, the values set in it will override the class defaults anytime a :class:`School` is created. Furthermore, these attributes will be type checked and validated anytime they are set. This type checking is handled by the :mod:`traitlets` module, which provides the :class:`~traitlets.Unicode`, :class:`~traitlets.Integer` and :class:`~traitlets.Float` types; see :doc:`trait_types` for the full list. It should be very clear at this point what the naming convention is for configuration attributes:: c.ClassName.attribute_name = attribute_value Here, ``ClassName`` is the name of the class whose configuration attribute you want to set, ``attribute_name`` is the name of the attribute you want to set and ``attribute_value`` the value you want it to have. The ``ClassName`` attribute of ``c`` is not the actual class, but instead is another :class:`~traitlets.config.Config` instance. .. note:: The careful reader may wonder how the ``ClassName`` (``School`` in the above example) attribute of the configuration object ``c`` gets created. These attributes are created on the fly by the :class:`~traitlets.config.Config` instance, using a simple naming convention. Any attribute of a :class:`~traitlets.config.Config` instance whose name begins with an uppercase character is assumed to be a sub-configuration and a new empty :class:`~traitlets.config.Config` instance is dynamically created for that attribute. This allows deeply hierarchical information created easily (``c.Foo.Bar.value``) on the fly. JSON configuration Files ------------------------ A JSON configuration file is simply a file that contains a :class:`~traitlets.config.Config` dictionary serialized to JSON. A JSON configuration file has the same base name as a Python configuration file, but with a .json extension. Configuration described in previous section could be written as follows in a JSON configuration file: .. code-block:: json { "School": { "name": "coolname", "ranking": 10 } } JSON configuration files can be more easily generated or processed by programs or other languages. Configuration files inheritance =============================== .. note:: This section only applies to Python configuration files. Let's say you want to have different configuration files for various purposes. Our configuration system makes it easy for one configuration file to inherit the information in another configuration file. The :func:`load_subconfig` command can be used in a configuration file for this purpose. Here is a simple example that loads all of the values from the file :file:`base_config.py`: .. code-block:: python :caption: examples/docs/configs/base_config.py c = get_config() # noqa c.School.name = "Harvard" c.School.ranking = 100 into the configuration file :file:`main_config.py`: .. code-block:: python :caption: examples/docs/configs/main_config.py :emphasize-lines: 4 c = get_config() # noqa # Load everything from base_config.py load_subconfig("base_config.py") # noqa # Now override one of the values c.School.name = "bettername" In a situation like this the :func:`load_subconfig` makes sure that the search path for sub-configuration files is inherited from that of the parent. Thus, you can typically put the two in the same directory and everything will just work. An example app using these configuration files can be found at `examples/docs/load_config_app.py `__. Class based configuration inheritance ===================================== There is another aspect of configuration where inheritance comes into play. Sometimes, your classes will have an inheritance hierarchy that you want to be reflected in the configuration system. Here is a simple example: .. code-block:: python from traitlets.config import Application, Configurable from traitlets import Integer, Float, Unicode, Bool class Foo(Configurable): name = Unicode("fooname", config=True) value = Float(100.0, config=True) class Bar(Foo): name = Unicode("barname", config=True) othervalue = Int(0, config=True) # construct Bar(config=..) Now, we can create a configuration file to configure instances of :class:`Foo` and :class:`Bar`: .. code-block:: python # config file c = get_config() # noqa c.Foo.name = "bestname" c.Bar.othervalue = 10 This class hierarchy and configuration file accomplishes the following: * The default value for :attr:`Foo.name` and :attr:`Bar.name` will be ``'bestname'``. Because :class:`Bar` is a :class:`Foo` subclass it also picks up the configuration information for :class:`Foo`. * The default value for :attr:`Foo.value` and :attr:`Bar.value` will be ``100.0``, which is the value specified as the class default. * The default value for :attr:`Bar.othervalue` will be 10 as set in the configuration file. Because :class:`Foo` is the parent of :class:`Bar` it doesn't know anything about the :attr:`othervalue` attribute. .. _commandline: Command-line arguments ====================== All configurable options can also be supplied at the command line when launching the application. Internally, when :meth:`Application.initialize()` is called, a :class:`~traitlets.config.loader.KVArgParseConfigLoader` instance is constructed to load values into a :class:`~traitlets.config.Config` object. (For advanced users, this can be overridden in the helper method :meth:`Application._create_loader()`.) Most command-line scripts simply need to call :meth:`Application.launch_instance()`, which will create the Application singleton, parse the command-line arguments, and start the application: .. code-block:: python :emphasize-lines: 8 from traitlets.config import Application class MyApp(Application): def start(self): pass # app logic goes here if __name__ == "__main__": MyApp.launch_instance() By default, config values are assigned from command-line arguments in much the same way as in a config file: .. code-block:: bash $ ipython --InteractiveShell.autoindent=False --BaseIPythonApplication.profile='myprofile' is the same as adding: .. code-block:: python c.InteractiveShell.autoindent = False c.BaseIPythonApplication.profile = "myprofile" to your configuration file. Command-line arguments take precedence over values read from configuration files. (This is done in :meth:`Application.load_config_file()` by merging ``Application.cli_config`` over values read from configuration files.) Note that even though :class:`Application` is a :class:`SingletonConfigurable`, multiple applications could still be started and called from each other by constructing them as one would with any other :class:`Configurable`: .. code-block:: python :caption: examples/docs/multiple_apps.py :emphasize-lines: 11,12,13 from traitlets.config import Application class OtherApp(Application): def start(self): print("other") class MyApp(Application): classes = [OtherApp] def start(self): # similar to OtherApp.launch_instance(), but without singleton self.other_app = OtherApp(config=self.config) self.other_app.initialize(["--OtherApp.log_level", "INFO"]) self.other_app.start() if __name__ == "__main__": MyApp.launch_instance() .. versionchanged:: 5.0 Prior to 5.0, fully specified ``--Class.trait=value`` arguments required an equals sign and no space separating the key and value. But after 5.0, these arguments can be separated by space as with aliases. .. versionchanged:: 5.0 extra quotes around strings and literal prefixes are no longer required. .. seealso:: :ref:`cli_strings` .. versionchanged:: 5.0 If a scalar (:class:`~.Unicode`, :class:`~.Integer`, etc.) is specified multiple times on the command-line, this will now raise. Prior to 5.0, all instances of the option before the last would be ignored. .. versionchanged:: 5.0 In 5.0, positional extra arguments (typically a list of files) must be contiguous, for example:: mycommand file1 file2 --flag or:: mycommand --flag file1 file2 whereas prior to 5.0, these "extra arguments" be distributed among other arguments:: mycommand file1 --flag file2 .. note:: By default, an error in a configuration file will cause the configuration file to be ignored and a warning logged. Application subclasses may specify `raise_config_file_errors = True` to exit on failure to load config files instead. .. versionadded:: 4.3 The environment variable ``TRAITLETS_APPLICATION_RAISE_CONFIG_FILE_ERROR`` to ``'1'`` or ``'true'`` to change the default value of ``raise_config_file_errors``. Common Arguments ---------------- Since the strictness and verbosity of the full ``--Class.trait=value`` form are not ideal for everyday use, common arguments can be specified as flags_ or aliases_. In general, flags and aliases are prefixed by ``--``, except for those that are single characters, in which case they can be specified with a single ``-``, e.g.: .. code-block:: bash $ ipython -i -c "import numpy; x=numpy.linspace(0,1)" --profile testing --colors=lightbg Flags and aliases are declared by specifying ``flags`` and ``aliases`` attributes as dictionaries on subclasses of :class:`~traitlets.config.Application`. A key in both those dictionaries might be a string or tuple of strings. One-character strings are converted into "short" options (like ``-v``); longer strings are "long" options (like ``--verbose``). Aliases ******* For convenience, applications have a mapping of commonly used traits, so you don't have to specify the whole class name: .. code-block:: bash $ ipython --profile myprofile # and $ ipython --profile='myprofile' # are equivalent to $ ipython --BaseIPythonApplication.profile='myprofile' When specifying ``alias`` dictionary in code, the values might be the strings like ``'Class.trait'`` or two-tuples like ``('Class.trait', "Some help message")``. For example: .. code-block:: python :caption: examples/docs/aliases.py :emphasize-lines: 11,12 from traitlets import Bool from traitlets.config import Application, Configurable class Foo(Configurable): enabled = Bool(False, help="whether enabled").tag(config=True) class App(Application): classes = [Foo] dry_run = Bool(False, help="dry run test").tag(config=True) aliases = { "dry-run": "App.dry_run", ("f", "foo-enabled"): ("Foo.enabled", "whether foo is enabled"), } if __name__ == "__main__": App.launch_instance() By default, the ``--log-level`` alias will be set up for ``Application.log_level``. Flags ***** Applications can also be passed **flags**. Flags are options that take no arguments. They are simply wrappers for setting one or more configurables with predefined values, often True/False. For instance: .. code-block:: bash $ ipcontroller --debug # is equivalent to $ ipcontroller --Application.log_level=DEBUG # and $ ipython --matplotlib # is equivalent to $ ipython --matplotlib auto # or $ ipython --no-banner # is equivalent to $ ipython --TerminalIPythonApp.display_banner=False And a runnable code example: .. code-block:: python :caption: examples/docs/flags.py :emphasize-lines: 11,12,13,14,15,16,17 from traitlets import Bool from traitlets.config import Application, Configurable class Foo(Configurable): enabled = Bool(False, help="whether enabled").tag(config=True) class App(Application): classes = [Foo] dry_run = Bool(False, help="dry run test").tag(config=True) flags = { "dry-run": ({"App": {"dry_run": True}}, dry_run.help), ("f", "enable-foo"): ( { "Foo": {"enabled": True}, }, "Enable foo", ), ("disable-foo"): ( { "Foo": {"enabled": False}, }, "Disable foo", ), } if __name__ == "__main__": App.launch_instance() Since flags are a bit more complicated to set up, there are a couple of common patterns implemented in helper methods. For example, :func:`traitlets.config.boolean_flag()` sets up the flags ``--x`` and ``--no-x``. By default, the following few flags are set up: ``--debug`` (setting ``log_level=DEBUG``), ``--show-config``, and ``--show-config-json`` (print config to stdout and exit). Subcommands ----------- Configurable applications can also have **subcommands**. Subcommands are modeled after :command:`git`, and are called with the form :command:`command subcommand [...args]`. For instance, the QtConsole is a subcommand of terminal IPython: .. code-block:: bash $ jupyter qtconsole --profile myprofile .. currentmodule:: traitlets.config Subcommands are specified as a dictionary assigned to a ``subcommands`` class member of :class:`~traitlets.config.Application` instances. This dictionary maps *subcommand names* to two-tuples containing these: 1. A subclass of :class:`Application` to handle the subcommand. This can be specified as: - simply as a class, where its :meth:`SingletonConfigurable.instance()` will be invoked (straight-forward, but loads subclasses on import time); - as a string which can be imported to produce the above class; - as a factory function accepting a single argument like that:: app_factory(parent_app: Application) -> Application .. note:: The return value of the factory above is an *instance*, not a class, so the :meth:`SingletonConfigurable.instance()` is not invoked in this case. In all cases, the instantiated app is stored in :attr:`Application.subapp` and its :meth:`Application.initialize()` is invoked. 2. A short description of the subcommand for use in help output. For example (refer to ``examples/subcommands_app.py`` for a more complete example): .. code-block:: python :caption: examples/docs/subcommands.py :emphasize-lines: 14,15 from traitlets.config import Application class SubApp1(Application): pass class SubApp2(Application): @classmethod def get_subapp_instance(cls, app: Application) -> Application: app.clear_instance() # since Application is singleton, need to clear main app return cls.instance(parent=app) class MainApp(Application): subcommands = { "subapp1": (SubApp1, "First subapp"), "subapp2": (SubApp2.get_subapp_instance, "Second subapp"), } if __name__ == "__main__": MainApp.launch_instance() To see a list of the available aliases, flags, and subcommands for a configurable application, simply pass ``-h`` or ``--help``. To see the full list of configurable options (*very* long), pass ``--help-all``. For more complete examples of setting up :class:`~traitlets.config.Application`, refer to the `application examples `__. Other ``Application`` members ----------------------------- The following are typically set as class variables of :class:`~traitlets.config.Application` subclasses, but can also be set as instance variables. * ``.classes``: A list of :class:`~traitlets.config.Configurable` classes. Similar to configs, any class name can be used in ``--Class.trait=value`` arguments, including classes that the :class:`~traitlets.config.Application` might not know about. However, the ``--help-all`` menu will only enumerate ``config`` traits of classes in ``Application.classes``. Similarly, ``.classes`` is used in other places where an application wants to list all configurable traits; examples include :meth:`Application.generate_config_file()` and the :ref:`argcomplete` handling. * ``.name``, ``.description``, ``.option_description``, ``.keyvalue_description``, ``.subcommand_description``, ``.examples``, ``.version``: Various strings used in the ``--help`` menu and other messages * ``.log_level``, ``.log_datefmt``, ``.log_format``, ``.logging_config``: Configurable options to control application logging, which is emitted via the logger ``Application.log``. For more information about these, refer to their respective traits' ``.help``. * ``.show_config``, ``.show_config_json``: Configurable boolean options, which if set to ``True``, will cause the application to print the config to stdout instead of calling :meth:`Application.start()` Additionally, the following are set by :class:`~traitlets.config.Application`: * ``.cli_config``: The :class:`~traitlets.config.Config` created from the command-line arguments. This is saved to override any config values loaded from configuration files called by :meth:`Application.load_config_file()`. * ``.extra_args``: This is a list holding any positional arguments remaining from the command-line arguments parsed during :meth:`Application.initialize()`. As noted earlier, these must be contiguous in the command-line. .. _cli_strings: Interpreting command-line strings --------------------------------- .. versionadded:: 5.0 :meth:`~.TraitType.from_string`, :meth:`~.List.from_string_list`, and :meth:`~.List.item_from_string`. Prior to 5.0, we only had good support for Unicode or similar string types on the command-line. Other types were supported via :py:func:`ast.literal_eval`, which meant that simple types such as integers were well supported, too. The downside of this implementation was that the :py:func:`literal_eval` happened before the type of the target trait was known, meaning that strings that could be interpreted as literals could end up with the wrong type, famously:: $ ipython -c 1 ... [TerminalIPythonApp] CRITICAL | Bad config encountered during initialization: [TerminalIPythonApp] CRITICAL | The 'code_to_run' trait of a TerminalIPythonApp instance must be a unicode string, but a value of 1 was specified. This resulted in requiring redundant "double-quoting" of strings in many cases. That gets confusing when the shell *also* interprets quotes, so one had to:: $ ipython -c "'1'" in order to set a string that looks like an integer. traitlets 5.0 defers parsing of interpreting command-line strings to :meth:`~.TraitType.from_string`, which is an arbitrary function that will be called with the string given on the command-line. This eliminates the need to 'guess' how to interpret strings before we know what they are configuring. Backward compatibility ********************** It is not feasible to be perfectly backward-compatible when fixing behavior as problematic as this. However, we are doing our best to ensure that folks who had workarounds for this funky behavior are disrupted as little as we can manage. That means that we have kept what look like literals working wherever we could, so if you were double-quoting strings to ensure the were interpreted as strings, that will continue to work with warnings for the foreseeable future. If you have an example command-line call that used to work with traitlets 4 but does not any more with traitlets 5, please `let us know `__. Custom traits ************* .. versionadded:: 5.0 Custom trait types can override :meth:`~.TraitType.from_string` to specify how strings should be interpreted. This could for example allow specifying hex-encoded bytes on the command-line: .. code-block:: python :caption: examples/docs/from_string.py :emphasize-lines: 6,7 from binascii import a2b_hex from traitlets.config import Application from traitlets import Bytes class HexBytes(Bytes): def from_string(self, s): return a2b_hex(s) class App(Application): aliases = {"key": "App.key"} key = HexBytes( help=""" Key to be used. Specify as hex on the command-line. """, config=True, ) def start(self): print(f"key={self.key}") if __name__ == "__main__": App.launch_instance() .. code-block:: bash $ examples/docs/from_string.py --key=a1b2 key=b'\xa2\xb2' Container traits ---------------- In traitlets 5.0, items for container traits can be specified by passing the key multiple times, e.g.:: myprogram -l a -l b to produce the list ``["a", "b"]`` or for dictionaries use ``key=value``:: myprogram -d a=5 -d b=10 to produce the dict ``{"a": 5, "b": 10}``. In traitlets prior to 5.0, container traits (List, Dict) could *technically* be configured on the command-line by specifying a repr of a Python list or dict, e.g:: ipython --ScriptMagics.script_paths='{"perl": "/usr/bin/perl"}' but that gets pretty tedious, especially with more than a couple of fields. This still works with a FutureWarning, but the new way allows container items to be specified by passing the argument multiple times:: ipython \ --ScriptMagics.script_paths perl=/usr/bin/perl \ --ScriptMagics.script_paths ruby=/usr/local/opt/bin/ruby This handling is good enough that we can recommend defining aliases for container traits for the first time! For example: .. code-block:: python :caption: examples/docs/container.py from traitlets.config import Application from traitlets import Dict, Integer, List, Unicode class App(Application): aliases = {"x": "App.x", "y": "App.y"} x = List(Unicode(), config=True) y = Dict(Integer(), config=True) def start(self): print(f"x={self.x}") print(f"y={self.y}") if __name__ == "__main__": App.launch_instance() produces: .. code-block:: bash $ examples/docs/container.py -x a -x b -y a=10 -y b=5 x=['a', 'b'] y={'a': 10, 'b': 5} .. note:: Specifying the value trait of Dict was necessary to cast the values in `y` to integers. Otherwise, they values of `y` would have been the strings ``'10'`` and ``'5'``. For container types, :meth:`.List.from_string_list` is called with the list of all values specified on the command-line and is responsible for turning the list of strings into the appropriate type. Each item is then passed to :meth:`.List.item_from_string` which is responsible for handling the item, such as casting to integer or parsing ``key=value`` in the case of a Dict. The deprecated :py:func:`ast.literal_eval` handling is preserved for backward-compatibility in the event of a single item that 'looks like' a list or dict literal. If you would prefer, you can also use custom container traits which define :meth`~.TraitType.from_string` to expand a single string into a list, for example: .. code-block:: python class PathList(List): def from_string(self, s): return s.split(os.pathsep) which would allow:: myprogram --path /bin:/usr/local/bin:/opt/bin to set a ``PathList`` trait with ``["/bin", "/usr/local/bin", "/opt/bin"]``. .. _argcomplete: Command-line tab completion with ``argcomplete`` ------------------------------------------------ .. versionadded:: 5.8 ``traitlets`` has limited support for command-line tab completion for scripts based on :class:`~traitlets.config.Application` using `argcomplete `__. To use this, follow the instructions for setting up argcomplete; you will likely want to `activate global completion `__ by doing something alone the lines of: .. code-block:: bash # pip install argcomplete mkdir -p ~/.bash_completion.d/ activate-global-python-argcomplete --dest=~/.bash_completion.d/argcomplete # source ~/.bash_completion.d/argcomplete from your ~/.bashrc (Follow relevant instructions for your shell.) For any script you want tab-completion to work on, include the line: .. code-block:: python # PYTHON_ARGCOMPLETE_OK in the first 1024 bytes of the script. The following options can be tab-completed: * Flags and aliases * The classes in ``Application.classes``, which can be initially completed as ``--Class.`` * Once a completion is narrows to a single class, the individual ``config`` traits of the class will be tab-completable, as ``--Class.trait``. * The available values for :class:`traitlets.Bool` and :class:`traitlets.Enum` will be completable, as well as any other custom :class:`traitlets.TraitType` which defines a ``argcompleter()`` method returning a list of available string completions. * Custom completer methods can be assigned to a trait by tagging an ``argcompleter`` metadata tag. Refer to `argcomplete's documentation `__ for examples of creating custom completer methods. Detailed examples of these can be found in the docstring of `examples/argcomplete_app.py `__. Caveats with ``argcomplete`` handling ************************************* The support for ``argcomplete`` is still relatively new and may not work with all ways in which an :class:`~traitlets.config.Application` is used. Some known caveats: * ``argcomplete`` is called when any ``Application`` first constructs and uses a :class:`~traitlets.config.KVArgParseConfigLoader` instance, which constructs a ``argparse.ArgumentParser`` instance. We assume that this is usually first done in scripts when parsing the command-line arguments, but technically a script can first call ``Application.initialize(["other", "args"])`` for some other reason. * ``traitlets`` does not actually add ``"--Class.trait"`` options to the ``ArgumentParser``, but instead directly parses them from ``argv``. In order to complete these, a custom :class:`~traitlets.config.argcomplete_config.CompletionFinder` is subclassed from ``argcomplete.CompletionFinder``, which dynamically inserts the ``"--Class.""`` and ``"--Class.trait"`` completions when it thinks suitable. However, this handling may be a bit fragile. * Because ``traitlets`` initializes configs from ``argv`` and not from ``ArgumentParser``, it may be more difficult to write custom completers which dynamically provide completions based on the state of other parsed arguments. * Subcommand handling is especially tricky. ``argcomplete`` libraries' strategy is to call the python script with no arguments e.g. ``len(sys.argv) == 1``, run until ``argcomplete`` is called on an ``ArgumentParser`` and determine what completions are available. On the other hand, the ``traitlet`` subcommand-handling strategy is to check ``sys.argv[1]`` and see if it matches a subcommand, and if so then dynamically load the subcommand app and initialize it with ``sys.argv[1:]``. To reconcile these two different approaches, some hacking was done to get ``traitlets`` to recognize the current command-line as seen by ``argcomplete``, and to get ``argcomplete`` to start parsing command-line arguments after subcommands have been evaluated. * Currently, completing subcommands themselves is not yet supported. * Some applications like ``Jupyter`` have custom ways of constructing subcommands or parsing ``argv`` which complicates matters even further. More details about these caveats can be found in the `original pull request `__. Design requirements =================== Here are the main requirements we wanted our configuration system to have: * Support for hierarchical configuration information. * Full integration with command line option parsers. Often, you want to read a configuration file, but then override some of the values with command line options. Our configuration system automates this process and allows each command line option to be linked to a particular attribute in the configuration hierarchy that it will override. * Configuration files that are themselves valid Python code. This accomplishes many things. First, it becomes possible to put logic in your configuration files that sets attributes based on your operating system, network setup, Python version, etc. Second, Python has a super simple syntax for accessing hierarchical data structures, namely regular attribute access (``Foo.Bar.Bam.name``). Third, using Python makes it easy for users to import configuration attributes from one configuration file to another. Fourth, even though Python is dynamically typed, it does have types that can be checked at runtime. Thus, a ``1`` in a config file is the integer '1', while a ``'1'`` is a string. * A fully automated method for getting the configuration information to the classes that need it at runtime. Writing code that walks a configuration hierarchy to extract a particular attribute is painful. When you have complex configuration information with hundreds of attributes, this makes you want to cry. * Type checking and validation that doesn't require the entire configuration hierarchy to be specified statically before runtime. Python is a very dynamic language and you don't always know everything that needs to be configured when a program starts.