Netdata.cloud bot for Zulip
1"""Configuration management for the Netdata Zulip bot.""" 2 3import os 4from pathlib import Path 5from typing import Optional 6 7import structlog 8from dotenv import load_dotenv 9 10from .models import ZulipConfig, ServerConfig 11 12logger = structlog.get_logger() 13 14 15def load_config() -> tuple[ZulipConfig, ServerConfig]: 16 """Load configuration from environment variables and .env files.""" 17 18 # Load .env file if present 19 env_file = Path(".env") 20 if env_file.exists(): 21 load_dotenv(env_file) 22 logger.info("Loaded configuration from .env file") 23 24 # Load Zulip configuration 25 zulip_config = ZulipConfig( 26 site=os.getenv("ZULIP_SITE", ""), 27 email=os.getenv("ZULIP_EMAIL", ""), 28 api_key=os.getenv("ZULIP_API_KEY", ""), 29 stream=os.getenv("ZULIP_STREAM", "netdata-alerts"), 30 ) 31 32 # Validate required Zulip settings 33 if not all([zulip_config.site, zulip_config.email, zulip_config.api_key]): 34 raise ValueError( 35 "Missing required Zulip configuration. Please set ZULIP_SITE, " 36 "ZULIP_EMAIL, and ZULIP_API_KEY environment variables." 37 ) 38 39 # Load server configuration 40 server_config = ServerConfig( 41 host=os.getenv("SERVER_HOST", "0.0.0.0"), 42 port=int(os.getenv("SERVER_PORT", "8443")), 43 domain=os.getenv("SERVER_DOMAIN", ""), 44 cert_path=os.getenv("SERVER_CERT_PATH", "/etc/letsencrypt/live"), 45 enable_mtls=os.getenv("SERVER_ENABLE_MTLS", "true").lower() == "true", 46 ) 47 48 # Validate required server settings 49 if not server_config.domain: 50 raise ValueError( 51 "Missing required server configuration. Please set SERVER_DOMAIN " 52 "environment variable." 53 ) 54 55 logger.info( 56 "Configuration loaded", 57 zulip_site=zulip_config.site, 58 zulip_email=zulip_config.email, 59 zulip_stream=zulip_config.stream, 60 server_host=server_config.host, 61 server_port=server_config.port, 62 server_domain=server_config.domain, 63 mtls_enabled=server_config.enable_mtls, 64 ) 65 66 return zulip_config, server_config 67 68 69def load_zuliprc_config(zuliprc_path: Optional[str] = None) -> ZulipConfig: 70 """Load Zulip configuration from a zuliprc file. 71 72 Args: 73 zuliprc_path: Path to zuliprc file. If None, looks for ~/.zuliprc 74 75 Returns: 76 ZulipConfig instance 77 """ 78 if zuliprc_path is None: 79 zuliprc_path = Path.home() / ".zuliprc" 80 else: 81 zuliprc_path = Path(zuliprc_path) 82 83 if not zuliprc_path.exists(): 84 raise FileNotFoundError(f"Zuliprc file not found: {zuliprc_path}") 85 86 config = {} 87 with open(zuliprc_path, 'r') as f: 88 for line in f: 89 line = line.strip() 90 if line and not line.startswith('#') and '=' in line: 91 key, value = line.split('=', 1) 92 config[key.strip()] = value.strip() 93 94 # Map zuliprc keys to our config 95 zulip_config = ZulipConfig( 96 site=config.get('site', ''), 97 email=config.get('email', ''), 98 api_key=config.get('key', ''), 99 stream=config.get('stream', 'netdata-alerts'), 100 ) 101 102 logger.info( 103 "Loaded Zulip configuration from zuliprc", 104 path=str(zuliprc_path), 105 site=zulip_config.site, 106 email=zulip_config.email, 107 stream=zulip_config.stream 108 ) 109 110 return zulip_config