# Configuration options The `few` package can be tuned through the use of optional configuration options which can be defined by either: - A `few.ini` configuration file - Environment variables - Command-line arguments when using a compatible command-line utility - A Python object named the [`ConfigurationSetter`](few.utils.globals.ConfigurationSetter) which can be used right after importing the `few` module for the first time The `few.ini` file is searched, in priority order: - In your current working directory - In your [`platformdirs.user_config_path()`](https://github.com/tox-dev/platformdirs/blob/main/README.rst) (e.g. `~/.config/few.ini`) - In your [`platformdirs.site_config_path()`](https://github.com/tox-dev/platformdirs/blob/main/README.rst) (e.g. `/etc/xdg/few/v1.5/few.ini`) The path to `few.ini` can be enforced by the `FEW_CONFIG_FILE` environment variable, or in command-line contexts, by the `-C` or `--config-file` argument. ## Tunable utilities These configuration options act on ### Log level `few` handles logging of operations using the python `logging` standard module. Its logger can be accessed either directly through a call to ```py3 import logging logger = logging.getLogger("few") ``` or by accessing `few` global states with ```py3 import few logger = few.get_logger() ``` The default log level is `logging.WARNING` but this can be modified through the `log_level` configuration option. ### Backends management Backends are plugin-like entities which can be used to delegate heavy computations to specific hardware like GPUs. By default, FEW will try to use the best available backend for each class that the user instanciates. It is however possible to enforce or prevent the use of specific backends through the `enabled_backends` options. The list of available backends is: - `cpu`: Use the CPU itself for accelerated computations - `cuda11x`: Use a NVIDIA GPU with CUDA 11.x drivers - `cuda12x`: Use a NVIDIA GPU with CUDA 12.x drivers By default, all these backends can be used provided they are installed and have required sotware and hardware available. A class that supports only CPU will only attempt to use he `cpu` backend, whilst a class with hybrid GPU/CPU support will (usually) first attempt to use the `cuda12x` backend, in case of failure it will attempt using the `cuda11x` and finally fallback to the `cpu` one. When setting the `enabled_backends` option, all items present in that list will be loaded (and FEW will fail initializing if any in not loadable) while other items will be strictly disabled. Example: - On a computer with only a CPU, setting `enabled_backends = ['cpu']` will speed-up FEW loading process by disabling the `cuda11x` and `cuda12x` backends and not even try loading them - On a computer with a NVIDIA GPU and CUDA 12.x drivers, setting `enabled_backends = ['cuda12x', 'cpu']` ensures that the CUDA 12.x backend will be loaded and that CPU only classes are still supported. If for any reason, the CUDA 12.x backend cannot be loaded, FEW will fail to run (and the message error should explain the failure and suggest mitigation strategies). ### File manager The `few` package requires external files of pre-computed coefficients which are too large to be bundled with the source code. These files are accessed through the `FileManager` global entity which takes care of locating these files, checking their integrity, and downloading missing files on request. The file manager is accessed through `few.get_file_manager()` This `FileManager` is highly tunable and propose the following options: #### `file_storage_path` The *storage path* is the directory where the `FileManager` will first look for files. Its default value is built relative to [`platformdirs.user_data_dir()`](https://github.com/tox-dev/platformdirs/blob/main/README.rst) (e.g. `~/.local/share/few/v1.5.1` on Linux systems) #### `file_download_path` The *download path* is the directory where the `FileManager` will download missing files. By default, it is a subdirectory of the *storage path* named `download`. #### `file_extra_paths` Extra paths are supplementary read-only directories where the file manager will search for requested files. They are provided as a ";"-separated list of paths. #### `file_registry_path` The *file registry* is a YAML file which lists known files which can be downloaded from FEW data repositories. It defines these data repositories and then, for each file, lists the repositories they can be found in, their tags and their checksums. A default `registry.yml` file is embedded with each version of `few` but one can provide a custom registry by setting its path through the `file_registry_path` option. #### `file_integrity_check` By default, the first time a file is requested to the `FileManager`, its integrity is checked and cached so that future request to that file are fast. This behaviour can be tuned to disable entirely integrity checks, or to perform these checks each and everytime a given file is requested. The `file_integrity_check` option can thus take on the following values: - `always`: perform the integrity check each and every time - `once` (default): perform the integrity check once for the lifetime of the file manager - `never`: never check for integrity (even on download) #### `file_allow_download` If a file is requested but not found locally, the file manager can (and by default will) attempt to download it from data repositories defined in the file registry. This behavior can be switched off if FEW needs to run offline (for instance, on compute node with filtered network access). ## Summary of configuration options General configuration options: | Option name | Config file entry | Environment variable | Command-line option | Authorized values | Comment | |---|---|---|---|---|---| | `log_level` | `log-level` | `FEW_LOG_LEVEL` | `--log-level` | `DEBUG`, `INFO`, `WARNING`, `ERROR`, `CRITICAL` | | | `log_format` | `log-format` | `FEW_LOG_FORMAT` | `--log-format` | Any format string supported by [`logging.Formatter`](https://docs.python.org/3/library/logging.html#logging.Formatter) | | | `enabled_backends` | `enbaled-backends` | `FEW_ENABLED_BACKENDS` | `--enable-backend` | `cpu`, `cuda11x`, `cuda12x` | ";"-separated list of values. The CLI parameter can be used multiple time for each enabled backend. | | `file_registry_path` | `file-registry` | `FEW_FILE_REGISTRY` | `--file-registry` | Path to a `registry.yml` file | | | `file_storage_path` | `file-storage-dir` | `FEW_FILE_STORAGE_DIR` | `--storage-dir` | Absolute path, or relative to current working directory | Directory must already exist | | `file_download_path` | `file-download-dir` | `FEW_FILE_DOWNLOAD_DIR` | `--download-dir` | Absolute path, or relative to the storage directory | | | `file_allow_download` | `file-allow-download` | `FEW_FILE_ALLOW_DOWNLOAD` | `--(no-)file-download` | Truthy (`yes`, `true`, `on`, `1`) or Falsy (`no`, `false`, `off`, `0`) value | | | `file_integrity_check` | `file-integrity-check` | `FEW_FILE_INTEGRITY_CHECK` | `--file-integrity-check` | `never`, `once`, `always` | | | `file_extra_paths` | `file-extra-paths` | `FEW_FILE_EXTRA_PATHS` | `--extra-path` | ";"-separated list of directories | (1) | (1) Extra paths are cumulated between option sources. *i.e.* the final options contains all paths from the config file, the environment variable and the CLI option. In command line contexts, the help message will usually only contain the options specific to the utility you are currently running. To check if a utility is compatible with the previous CLI options, use the `-H` or `--config-help` argument to obtain the help message corresponding to those options. One may also disable reading options from the config file either by setting the `FEW_IGNORE_CFG_FILE` to a *Truthy* value (yes/on/true/1) or by using the `--ignore-config-file` argument. Similarly, one may disable options from the environment using the `--ignore-env` CLI option (in command line contexts only). In your program, you may access the values of these configuration option by accessing the global configuration: ```{eval-rst} >>> import few >>> cfg = few.get_config() >>> cfg.log_level 30 >>> cfg.file_integrity_check 'once' ``` Unless stated otherwise, command-line arguments take precedence over environment variable which in turn take precedence over configuration file entries. ## Configuration setter If you need to update configuration settings in a given script, or in an interactive Python context (terminal, notebook), this can be done using the [*configuration setter*](few.utils.globals.ConfigurationSetter). ```{important} The Configuration Setter can only be used right after importing `few`. As soon as any `few` entity makes use of a configuration option (for instance, when initializing the first backend-accelerated object), the setter cannot be used anymore to change an option. ``` The *configuration setter* is accessed by `few.get_config_setter`and offers multiple methods to customize configuration options. Note that these methods can be chained directly: ```{eval-rst} .. testcode:: cfg-setter # import few # Access the setter setter = few.get_config_setter() # Use a single method setter.disable_file_download() # Chain methods setter.set_log_level( "debug" ).add_file_extra_paths("/tmp/few_data") ``` Options defined in the [*configuration setter*](few.utils.globals.ConfigurationSetter) can be applied immediately by calling the `finalize()` method like: ```{eval-rst} .. testcode:: cfg-explicit-finalize import few # Access the setter, set an option and finalize options few.get_config_setter().set_log_level("info").finalize() ``` But if `finalize()` is not explicitely called, the options set will still be taken into account automatically at a latter stage: ```{eval-rst} .. testcode:: cfg-implicit-finalized import few # Set the log level few.get_config_setter().set_log_level("debug") # Build a FEW object amp = few.amplitude.romannet.RomanAmplitude() .. testoutput:: cfg-implicit-finalized ... ConfigInitialization: final configuration entries are ... log_level=10 (from: ConfigSource.SETTER) ... ```