ConfigHandler

The ConfigHandler enables flexible configuration management for WizLib applications, providing users with multiple ways to configure the application.

Overview

The ConfigHandler allows commands to retrieve configuration values from:

  1. Environment variables
  2. YAML configuration files
  3. Command-line arguments

This flexibility gives users options for how to provide configuration information based on their preferences and needs.

Accessing Configuration Values

Commands can request values from the configuration using the get() method:

# In a command's execute() or handle_vals() method
value = self.app.config.get('myapp-host')

The argument to get() is a hyphen-separated set of words that indicate a path through a hierarchy of possible values. Typically, the first word in the argument is the name of the application.

Configuration Lookup Order

When retrieving a configuration value, the handler follows this order:

  1. Cache: First checks if the value has already been retrieved
  2. Environment Variables: Looks for an environment variable by converting the hyphens to underscores and the words to all caps
  3. YAML Configuration File: Searches for a YAML file in several locations

Environment Variables

For a configuration key like myapp-host, the handler will look for an environment variable named MYAPP_HOST.

# Setting a configuration value via environment variable
export MYAPP_HOST=api.example.com

YAML Configuration Files

If no environment variable is found, the handler looks for a YAML configuration file in the following order:

  1. Path specified by the --config / -c command-line option
  2. Path in the MYAPP_CONFIG environment variable
  3. .myapp.yml in the current working directory
  4. ~/.myapp.yml in the user's home directory

YAML Configuration Format

Configuration files use YAML format. The structure should mirror the hyphen-separated path used in the get() method:

myapp:
  host: api.example.com
  database:
    username: dbuser
    password: secret
  features:
    enable_logging: true

With this configuration, you could access values like:

host = self.app.config.get('myapp-host')                    # "api.example.com"
username = self.app.config.get('myapp-database-username')   # "dbuser"
logging = self.app.config.get('myapp-features-enable_logging')  # True

Dynamic Values with Command Execution

The YAML configuration file can include dynamic values by executing OS commands using the $(command) syntax. This is particularly useful for retrieving secrets from password managers:

myapp:
  # Static value
  host: api.example.com
  
  # Dynamic value from command execution
  token: $(op read "op://Private/example/api-key")

When the configuration value is requested, the command inside $() will be executed and its output will be used as the value.

Example Usage in a Command

Here's how you might use the ConfigHandler in a command:

def handle_vals(self):
    super().handle_vals()
    
    # Get API host from config, with a default if not found
    if not hasattr(self, 'host'):
        self.host = self.app.config.get('myapp-host') or 'api.default.com'
    
    # Get API token from config, prompt if not found
    self.token = self.app.config.get('myapp-token')
    if not self.token:
        self.token = self.app.ui.get_input('Enter API token: ')

Testing with Fake ConfigHandler

For testing, you can create a fake ConfigHandler with predefined values:

from wizlib.config_handler import ConfigHandler

# Create a fake config handler with test values
config = ConfigHandler.fake(
    myapp_host='test.example.com',
    myapp_token='test-token'
)

# Use in tests
assert config.get('myapp-host') == 'test.example.com'