mirror of
https://github.com/langbot-app/LangBot.git
synced 2025-11-25 03:15:06 +08:00
* Initial plan * Add environment variable override support for config.yaml Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Refactor env override code based on review feedback Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Add test for template completion with env overrides Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * Move env override logic to load_config.py as requested Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> * perf: add print log --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: RockChinQ <45992437+RockChinQ@users.noreply.github.com> Co-authored-by: Junyan Qin <rockchinq@gmail.com>
333 lines
10 KiB
Python
333 lines
10 KiB
Python
"""
|
||
Tests for environment variable override functionality in YAML config
|
||
"""
|
||
|
||
import os
|
||
import pytest
|
||
from typing import Any
|
||
|
||
|
||
def _apply_env_overrides_to_config(cfg: dict) -> dict:
|
||
"""Apply environment variable overrides to data/config.yaml
|
||
|
||
Environment variables should be uppercase and use __ (double underscore)
|
||
to represent nested keys. For example:
|
||
- CONCURRENCY__PIPELINE overrides concurrency.pipeline
|
||
- PLUGIN__RUNTIME_WS_URL overrides plugin.runtime_ws_url
|
||
|
||
Arrays and dict types are ignored.
|
||
|
||
Args:
|
||
cfg: Configuration dictionary
|
||
|
||
Returns:
|
||
Updated configuration dictionary
|
||
"""
|
||
def convert_value(value: str, original_value: Any) -> Any:
|
||
"""Convert string value to appropriate type based on original value
|
||
|
||
Args:
|
||
value: String value from environment variable
|
||
original_value: Original value to infer type from
|
||
|
||
Returns:
|
||
Converted value (falls back to string if conversion fails)
|
||
"""
|
||
if isinstance(original_value, bool):
|
||
return value.lower() in ('true', '1', 'yes', 'on')
|
||
elif isinstance(original_value, int):
|
||
try:
|
||
return int(value)
|
||
except ValueError:
|
||
# If conversion fails, keep as string (user error, but non-breaking)
|
||
return value
|
||
elif isinstance(original_value, float):
|
||
try:
|
||
return float(value)
|
||
except ValueError:
|
||
# If conversion fails, keep as string (user error, but non-breaking)
|
||
return value
|
||
else:
|
||
return value
|
||
|
||
# Process environment variables
|
||
for env_key, env_value in os.environ.items():
|
||
# Check if the environment variable is uppercase and contains __
|
||
if not env_key.isupper():
|
||
continue
|
||
if '__' not in env_key:
|
||
continue
|
||
|
||
# Convert environment variable name to config path
|
||
# e.g., CONCURRENCY__PIPELINE -> ['concurrency', 'pipeline']
|
||
keys = [key.lower() for key in env_key.split('__')]
|
||
|
||
# Navigate to the target value and validate the path
|
||
current = cfg
|
||
|
||
for i, key in enumerate(keys):
|
||
if not isinstance(current, dict) or key not in current:
|
||
break
|
||
|
||
if i == len(keys) - 1:
|
||
# At the final key - check if it's a scalar value
|
||
if isinstance(current[key], (dict, list)):
|
||
# Skip dict and list types
|
||
pass
|
||
else:
|
||
# Valid scalar value - convert and set it
|
||
converted_value = convert_value(env_value, current[key])
|
||
current[key] = converted_value
|
||
else:
|
||
# Navigate deeper
|
||
current = current[key]
|
||
|
||
return cfg
|
||
|
||
|
||
class TestEnvOverrides:
|
||
"""Test environment variable override functionality"""
|
||
|
||
def test_simple_string_override(self):
|
||
"""Test overriding a simple string value"""
|
||
cfg = {
|
||
'api': {
|
||
'port': 5300
|
||
}
|
||
}
|
||
|
||
# Set environment variable
|
||
os.environ['API__PORT'] = '8080'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
assert result['api']['port'] == 8080
|
||
|
||
# Cleanup
|
||
del os.environ['API__PORT']
|
||
|
||
def test_nested_key_override(self):
|
||
"""Test overriding nested keys with __ delimiter"""
|
||
cfg = {
|
||
'concurrency': {
|
||
'pipeline': 20,
|
||
'session': 1
|
||
}
|
||
}
|
||
|
||
os.environ['CONCURRENCY__PIPELINE'] = '50'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
assert result['concurrency']['pipeline'] == 50
|
||
assert result['concurrency']['session'] == 1 # Unchanged
|
||
|
||
del os.environ['CONCURRENCY__PIPELINE']
|
||
|
||
def test_deep_nested_override(self):
|
||
"""Test overriding deeply nested keys"""
|
||
cfg = {
|
||
'system': {
|
||
'jwt': {
|
||
'expire': 604800,
|
||
'secret': ''
|
||
}
|
||
}
|
||
}
|
||
|
||
os.environ['SYSTEM__JWT__EXPIRE'] = '86400'
|
||
os.environ['SYSTEM__JWT__SECRET'] = 'my_secret_key'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
assert result['system']['jwt']['expire'] == 86400
|
||
assert result['system']['jwt']['secret'] == 'my_secret_key'
|
||
|
||
del os.environ['SYSTEM__JWT__EXPIRE']
|
||
del os.environ['SYSTEM__JWT__SECRET']
|
||
|
||
def test_underscore_in_key(self):
|
||
"""Test keys with underscores like runtime_ws_url"""
|
||
cfg = {
|
||
'plugin': {
|
||
'enable': True,
|
||
'runtime_ws_url': 'ws://localhost:5400/control/ws'
|
||
}
|
||
}
|
||
|
||
os.environ['PLUGIN__RUNTIME_WS_URL'] = 'ws://newhost:6000/ws'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
assert result['plugin']['runtime_ws_url'] == 'ws://newhost:6000/ws'
|
||
|
||
del os.environ['PLUGIN__RUNTIME_WS_URL']
|
||
|
||
def test_boolean_conversion(self):
|
||
"""Test boolean value conversion"""
|
||
cfg = {
|
||
'plugin': {
|
||
'enable': True,
|
||
'enable_marketplace': False
|
||
}
|
||
}
|
||
|
||
os.environ['PLUGIN__ENABLE'] = 'false'
|
||
os.environ['PLUGIN__ENABLE_MARKETPLACE'] = 'true'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
assert result['plugin']['enable'] is False
|
||
assert result['plugin']['enable_marketplace'] is True
|
||
|
||
del os.environ['PLUGIN__ENABLE']
|
||
del os.environ['PLUGIN__ENABLE_MARKETPLACE']
|
||
|
||
def test_ignore_dict_type(self):
|
||
"""Test that dict types are ignored"""
|
||
cfg = {
|
||
'database': {
|
||
'use': 'sqlite',
|
||
'sqlite': {
|
||
'path': 'data/langbot.db'
|
||
}
|
||
}
|
||
}
|
||
|
||
# Try to override a dict value - should be ignored
|
||
os.environ['DATABASE__SQLITE'] = 'new_value'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
# Should remain a dict, not overridden
|
||
assert isinstance(result['database']['sqlite'], dict)
|
||
assert result['database']['sqlite']['path'] == 'data/langbot.db'
|
||
|
||
del os.environ['DATABASE__SQLITE']
|
||
|
||
def test_ignore_list_type(self):
|
||
"""Test that list/array types are ignored"""
|
||
cfg = {
|
||
'admins': ['admin1', 'admin2'],
|
||
'command': {
|
||
'enable': True,
|
||
'prefix': ['!', '!']
|
||
}
|
||
}
|
||
|
||
# Try to override list values - should be ignored
|
||
os.environ['ADMINS'] = 'admin3'
|
||
os.environ['COMMAND__PREFIX'] = '?'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
# Should remain lists, not overridden
|
||
assert isinstance(result['admins'], list)
|
||
assert result['admins'] == ['admin1', 'admin2']
|
||
assert isinstance(result['command']['prefix'], list)
|
||
assert result['command']['prefix'] == ['!', '!']
|
||
|
||
del os.environ['ADMINS']
|
||
del os.environ['COMMAND__PREFIX']
|
||
|
||
def test_lowercase_env_var_ignored(self):
|
||
"""Test that lowercase environment variables are ignored"""
|
||
cfg = {
|
||
'api': {
|
||
'port': 5300
|
||
}
|
||
}
|
||
|
||
os.environ['api__port'] = '8080'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
# Should not be overridden
|
||
assert result['api']['port'] == 5300
|
||
|
||
del os.environ['api__port']
|
||
|
||
def test_no_double_underscore_ignored(self):
|
||
"""Test that env vars without __ are ignored"""
|
||
cfg = {
|
||
'api': {
|
||
'port': 5300
|
||
}
|
||
}
|
||
|
||
os.environ['APIPORT'] = '8080'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
# Should not be overridden
|
||
assert result['api']['port'] == 5300
|
||
|
||
del os.environ['APIPORT']
|
||
|
||
def test_nonexistent_key_ignored(self):
|
||
"""Test that env vars for non-existent keys are ignored"""
|
||
cfg = {
|
||
'api': {
|
||
'port': 5300
|
||
}
|
||
}
|
||
|
||
os.environ['API__NONEXISTENT'] = 'value'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
# Should not create new key
|
||
assert 'nonexistent' not in result['api']
|
||
|
||
del os.environ['API__NONEXISTENT']
|
||
|
||
def test_integer_conversion(self):
|
||
"""Test integer value conversion"""
|
||
cfg = {
|
||
'concurrency': {
|
||
'pipeline': 20
|
||
}
|
||
}
|
||
|
||
os.environ['CONCURRENCY__PIPELINE'] = '100'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
assert result['concurrency']['pipeline'] == 100
|
||
assert isinstance(result['concurrency']['pipeline'], int)
|
||
|
||
del os.environ['CONCURRENCY__PIPELINE']
|
||
|
||
def test_multiple_overrides(self):
|
||
"""Test multiple environment variable overrides at once"""
|
||
cfg = {
|
||
'api': {
|
||
'port': 5300
|
||
},
|
||
'concurrency': {
|
||
'pipeline': 20,
|
||
'session': 1
|
||
},
|
||
'plugin': {
|
||
'enable': False
|
||
}
|
||
}
|
||
|
||
os.environ['API__PORT'] = '8080'
|
||
os.environ['CONCURRENCY__PIPELINE'] = '50'
|
||
os.environ['PLUGIN__ENABLE'] = 'true'
|
||
|
||
result = _apply_env_overrides_to_config(cfg)
|
||
|
||
assert result['api']['port'] == 8080
|
||
assert result['concurrency']['pipeline'] == 50
|
||
assert result['plugin']['enable'] is True
|
||
|
||
del os.environ['API__PORT']
|
||
del os.environ['CONCURRENCY__PIPELINE']
|
||
del os.environ['PLUGIN__ENABLE']
|
||
|
||
|
||
if __name__ == '__main__':
|
||
pytest.main([__file__, '-v'])
|