mirror of
				https://github.com/bunny-lab-io/Borealis.git
				synced 2025-10-26 17:41:58 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			2251 lines
		
	
	
		
			87 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			2251 lines
		
	
	
		
			87 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #////////// PROJECT FILE SEPARATION LINE ////////// CODE AFTER THIS LINE ARE FROM: <ProjectRoot>/Data/Agent/borealis-agent.py
 | |
| import sys
 | |
| import uuid
 | |
| import socket
 | |
| import os
 | |
| import json
 | |
| import asyncio
 | |
| import concurrent.futures
 | |
| from functools import partial
 | |
| from io import BytesIO
 | |
| import base64
 | |
| import traceback
 | |
| import random  # Macro Randomization
 | |
| import platform  # OS Detection
 | |
| import importlib.util
 | |
| import time  # Heartbeat timestamps
 | |
| import subprocess
 | |
| import getpass
 | |
| import datetime
 | |
| import shutil
 | |
| import string
 | |
| import ssl
 | |
| import threading
 | |
| from typing import Any, Dict, Optional, List
 | |
| 
 | |
| import requests
 | |
| try:
 | |
|     import psutil
 | |
| except Exception:
 | |
|     psutil = None
 | |
| import aiohttp
 | |
| 
 | |
| import socketio
 | |
| from security import AgentKeyStore
 | |
| 
 | |
| from cryptography.hazmat.primitives import serialization
 | |
| from cryptography.hazmat.primitives.asymmetric import ed25519
 | |
| 
 | |
| # Centralized logging helpers (Agent)
 | |
| def _agent_logs_root() -> str:
 | |
|     try:
 | |
|         root = _find_project_root()
 | |
|         return os.path.abspath(os.path.join(root, 'Logs', 'Agent'))
 | |
|     except Exception:
 | |
|         return os.path.abspath(os.path.join(os.path.dirname(__file__), 'Logs', 'Agent'))
 | |
| 
 | |
| 
 | |
| def _rotate_daily(path: str):
 | |
|     try:
 | |
|         import datetime as _dt
 | |
|         if os.path.isfile(path):
 | |
|             mtime = os.path.getmtime(path)
 | |
|             dt = _dt.datetime.fromtimestamp(mtime)
 | |
|             today = _dt.datetime.now().date()
 | |
|             if dt.date() != today:
 | |
|                 suffix = dt.strftime('%Y-%m-%d')
 | |
|                 newp = f"{path}.{suffix}"
 | |
|                 try:
 | |
|                     os.replace(path, newp)
 | |
|                 except Exception:
 | |
|                     pass
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
| 
 | |
| # Early bootstrap logging (goes to agent.log)
 | |
| def _bootstrap_log(msg: str):
 | |
|     try:
 | |
|         base = _agent_logs_root()
 | |
|         os.makedirs(base, exist_ok=True)
 | |
|         path = os.path.join(base, 'agent.log')
 | |
|         _rotate_daily(path)
 | |
|         ts = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')
 | |
|         with open(path, 'a', encoding='utf-8') as fh:
 | |
|             fh.write(f'[{ts}] {msg}\n')
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
| # Headless/service mode flag (skip Qt and interactive UI)
 | |
| SYSTEM_SERVICE_MODE = ('--system-service' in sys.argv) or (os.environ.get('BOREALIS_AGENT_MODE') == 'system')
 | |
| SERVICE_MODE = 'system' if SYSTEM_SERVICE_MODE else 'currentuser'
 | |
| _bootstrap_log(f'agent.py loaded; SYSTEM_SERVICE_MODE={SYSTEM_SERVICE_MODE}; argv={sys.argv!r}')
 | |
| def _argv_get(flag: str, default: str = None):
 | |
|     try:
 | |
|         if flag in sys.argv:
 | |
|             idx = sys.argv.index(flag)
 | |
|             if idx >= 0 and idx + 1 < len(sys.argv):
 | |
|                 return sys.argv[idx + 1]
 | |
|     except Exception:
 | |
|         pass
 | |
|     return default
 | |
| CONFIG_NAME_SUFFIX = _argv_get('--config', None)
 | |
| 
 | |
| 
 | |
| def _canonical_config_suffix(raw_suffix: str) -> str:
 | |
|     try:
 | |
|         if not raw_suffix:
 | |
|             return ''
 | |
|         value = str(raw_suffix).strip()
 | |
|         if not value:
 | |
|             return ''
 | |
|         normalized = value.lower()
 | |
|         if normalized in ('svc', 'system', 'system_service', 'service'):
 | |
|             return 'SYSTEM'
 | |
|         if normalized in ('user', 'currentuser', 'interactive'):
 | |
|             return 'CURRENTUSER'
 | |
|         sanitized = ''.join(ch for ch in value if ch.isalnum() or ch in ('_', '-')).strip('_-')
 | |
|         return sanitized
 | |
|     except Exception:
 | |
|         return ''
 | |
| 
 | |
| 
 | |
| CONFIG_SUFFIX_CANONICAL = _canonical_config_suffix(CONFIG_NAME_SUFFIX)
 | |
| 
 | |
| INSTALLER_CODE_OVERRIDE = (
 | |
|     (_argv_get('--installer-code') or os.environ.get('BOREALIS_INSTALLER_CODE') or '')
 | |
|     .strip()
 | |
| )
 | |
| 
 | |
| 
 | |
| def _agent_guid_path() -> str:
 | |
|     try:
 | |
|         root = _find_project_root()
 | |
|         return os.path.join(root, 'Agent', 'Borealis', 'agent_GUID')
 | |
|     except Exception:
 | |
|         return os.path.abspath(os.path.join(os.path.dirname(__file__), 'agent_GUID'))
 | |
| 
 | |
| 
 | |
| def _settings_dir():
 | |
|     try:
 | |
|         return os.path.join(_find_project_root(), 'Agent', 'Borealis', 'Settings')
 | |
|     except Exception:
 | |
|         return os.path.abspath(os.path.join(os.path.dirname(__file__), 'Settings'))
 | |
| 
 | |
| 
 | |
| _KEY_STORE_INSTANCE = None
 | |
| 
 | |
| 
 | |
| def _key_store() -> AgentKeyStore:
 | |
|     global _KEY_STORE_INSTANCE
 | |
|     if _KEY_STORE_INSTANCE is None:
 | |
|         scope = 'SYSTEM' if SYSTEM_SERVICE_MODE else 'CURRENTUSER'
 | |
|         _KEY_STORE_INSTANCE = AgentKeyStore(_settings_dir(), scope=scope)
 | |
|     return _KEY_STORE_INSTANCE
 | |
| 
 | |
| 
 | |
| def _persist_agent_guid_local(guid: str):
 | |
|     guid = _normalize_agent_guid(guid)
 | |
|     if not guid:
 | |
|         return
 | |
|     try:
 | |
|         _key_store().save_guid(guid)
 | |
|     except Exception as exc:
 | |
|         _log_agent(f'Unable to persist guid via key store: {exc}', fname='agent.error.log')
 | |
|     path = _agent_guid_path()
 | |
|     try:
 | |
|         directory = os.path.dirname(path)
 | |
|         if directory:
 | |
|             os.makedirs(directory, exist_ok=True)
 | |
|         existing = ''
 | |
|         if os.path.isfile(path):
 | |
|             try:
 | |
|                 with open(path, 'r', encoding='utf-8') as fh:
 | |
|                     existing = fh.read().strip()
 | |
|             except Exception:
 | |
|                 existing = ''
 | |
|         if existing != guid:
 | |
|             with open(path, 'w', encoding='utf-8') as fh:
 | |
|                 fh.write(guid)
 | |
|     except Exception as exc:
 | |
|         _log_agent(f'Failed to persist agent GUID locally: {exc}', fname='agent.error.log')
 | |
| 
 | |
| if not SYSTEM_SERVICE_MODE:
 | |
|     # Reduce noisy Qt output and attempt to avoid Windows OleInitialize warnings
 | |
|     os.environ.setdefault("QT_LOGGING_RULES", "qt.qpa.*=false;*.debug=false")
 | |
|     from qasync import QEventLoop
 | |
|     from PyQt5 import QtCore, QtGui, QtWidgets
 | |
|     try:
 | |
|         # Swallow Qt framework messages to keep console clean
 | |
|         def _qt_msg_handler(mode, context, message):
 | |
|             return
 | |
|         QtCore.qInstallMessageHandler(_qt_msg_handler)
 | |
|     except Exception:
 | |
|         pass
 | |
|     from PIL import ImageGrab
 | |
| 
 | |
| # New modularized components
 | |
| from role_manager import RoleManager
 | |
| 
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| # CORE SECTION: CONFIG MANAGER
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| def _user_config_default_path():
 | |
|     """Return the prior per-user config file path (used for migration)."""
 | |
|     try:
 | |
|         plat = sys.platform
 | |
|         if plat.startswith("win"):
 | |
|             base = os.environ.get("APPDATA") or os.path.expanduser(r"~\AppData\Roaming")
 | |
|             return os.path.join(base, "Borealis", "agent_settings.json")
 | |
|         elif plat == "darwin":
 | |
|             base = os.path.expanduser("~/Library/Application Support")
 | |
|             return os.path.join(base, "Borealis", "agent_settings.json")
 | |
|         else:
 | |
|             base = os.environ.get("XDG_CONFIG_HOME") or os.path.expanduser("~/.config")
 | |
|             return os.path.join(base, "borealis", "agent_settings.json")
 | |
|     except Exception:
 | |
|         return os.path.join(os.path.dirname(__file__), "agent_settings.json")
 | |
| 
 | |
| def _find_project_root():
 | |
|     """Attempt to locate the Borealis project root (folder with Borealis.ps1 or users.json)."""
 | |
|     # Allow explicit override
 | |
|     override_root = os.environ.get("BOREALIS_ROOT") or os.environ.get("BOREALIS_PROJECT_ROOT")
 | |
|     if override_root and os.path.isdir(override_root):
 | |
|         return os.path.abspath(override_root)
 | |
| 
 | |
|     cur = os.path.abspath(os.path.dirname(__file__))
 | |
|     for _ in range(8):
 | |
|         if (
 | |
|             os.path.exists(os.path.join(cur, "Borealis.ps1"))
 | |
|             or os.path.exists(os.path.join(cur, "users.json"))
 | |
|             or os.path.isdir(os.path.join(cur, ".git"))
 | |
|         ):
 | |
|             return cur
 | |
|         parent = os.path.dirname(cur)
 | |
|         if parent == cur:
 | |
|             break
 | |
|         cur = parent
 | |
|     # Heuristic fallback: two levels up from Agent/Borealis
 | |
|     return os.path.abspath(os.path.join(os.path.dirname(__file__), "..", ".."))
 | |
| 
 | |
| # Simple file logger under Logs/Agent
 | |
| def _log_agent(message: str, fname: str = 'agent.log'):
 | |
|     try:
 | |
|         log_dir = _agent_logs_root()
 | |
|         os.makedirs(log_dir, exist_ok=True)
 | |
|         ts = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')
 | |
|         path = os.path.join(log_dir, fname)
 | |
|         _rotate_daily(path)
 | |
|         with open(path, 'a', encoding='utf-8') as fh:
 | |
|             fh.write(f'[{ts}] {message}\n')
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
| 
 | |
| def _decode_base64_text(value):
 | |
|     if not isinstance(value, str):
 | |
|         return None
 | |
|     stripped = value.strip()
 | |
|     if not stripped:
 | |
|         return ""
 | |
|     cleaned = ''.join(stripped.split())
 | |
|     if not cleaned:
 | |
|         return ""
 | |
|     try:
 | |
|         decoded = base64.b64decode(cleaned, validate=True)
 | |
|     except Exception:
 | |
|         return None
 | |
|     try:
 | |
|         return decoded.decode('utf-8')
 | |
|     except Exception:
 | |
|         return decoded.decode('utf-8', errors='replace')
 | |
| 
 | |
| 
 | |
| def _decode_base64_bytes(value):
 | |
|     if not isinstance(value, str):
 | |
|         return None
 | |
|     stripped = value.strip()
 | |
|     if not stripped:
 | |
|         return b""
 | |
|     cleaned = ''.join(stripped.split())
 | |
|     if not cleaned:
 | |
|         return b""
 | |
|     try:
 | |
|         return base64.b64decode(cleaned, validate=True)
 | |
|     except Exception:
 | |
|         return None
 | |
| 
 | |
| 
 | |
| def _decode_script_payload(content, encoding_hint):
 | |
|     if isinstance(content, str):
 | |
|         encoding = str(encoding_hint or '').strip().lower()
 | |
|         if encoding in ('base64', 'b64', 'base-64'):
 | |
|             decoded = _decode_base64_text(content)
 | |
|             if decoded is not None:
 | |
|                 return decoded
 | |
|         decoded = _decode_base64_text(content)
 | |
|         if decoded is not None:
 | |
|             return decoded
 | |
|         return content
 | |
|     return ''
 | |
| 
 | |
| def _resolve_config_path():
 | |
|     """
 | |
|     Resolve the path for agent settings json in the centralized location:
 | |
|     <ProjectRoot>/Agent/Borealis/Settings/agent_settings[_{suffix}].json
 | |
| 
 | |
|     Precedence/order:
 | |
|     - If BOREALIS_AGENT_CONFIG is set (full path), use it.
 | |
|     - Else if BOREALIS_AGENT_CONFIG_DIR is set (dir), use agent_settings.json under it.
 | |
|     - Else use <ProjectRoot>/Agent/Borealis/Settings and migrate any legacy files into it.
 | |
|     - If suffix is provided, seed from base if present.
 | |
|     """
 | |
|     # Full file path override
 | |
|     override_file = os.environ.get("BOREALIS_AGENT_CONFIG")
 | |
|     if override_file:
 | |
|         cfg_dir = os.path.dirname(override_file)
 | |
|         if cfg_dir and not os.path.exists(cfg_dir):
 | |
|             os.makedirs(cfg_dir, exist_ok=True)
 | |
|         return override_file
 | |
| 
 | |
|     # Optional directory override
 | |
|     override_dir = os.environ.get("BOREALIS_AGENT_CONFIG_DIR")
 | |
|     if override_dir:
 | |
|         os.makedirs(override_dir, exist_ok=True)
 | |
|         return os.path.join(override_dir, "agent_settings.json")
 | |
| 
 | |
|     project_root = _find_project_root()
 | |
|     settings_dir = os.path.join(project_root, 'Agent', 'Borealis', 'Settings')
 | |
|     try:
 | |
|         os.makedirs(settings_dir, exist_ok=True)
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
|     # Determine filename with optional suffix
 | |
|     cfg_basename = 'agent_settings.json'
 | |
|     suffix = CONFIG_SUFFIX_CANONICAL
 | |
|     if suffix:
 | |
|         cfg_basename = f"agent_settings_{suffix}.json"
 | |
| 
 | |
|     cfg_path = os.path.join(settings_dir, cfg_basename)
 | |
|     if os.path.exists(cfg_path):
 | |
|         return cfg_path
 | |
| 
 | |
|     # Migrate legacy suffixed config names to the new canonical form
 | |
|     legacy_map = {
 | |
|         'SYSTEM': ['agent_settings_svc.json'],
 | |
|         'CURRENTUSER': ['agent_settings_user.json'],
 | |
|     }
 | |
|     try:
 | |
|         legacy_candidates = []
 | |
|         if suffix:
 | |
|             for legacy_name in legacy_map.get(suffix.upper(), []):
 | |
|                 legacy_candidates.extend([
 | |
|                     os.path.join(settings_dir, legacy_name),
 | |
|                     os.path.join(project_root, legacy_name),
 | |
|                     os.path.join(project_root, 'Agent', 'Settings', legacy_name),
 | |
|                 ])
 | |
|         for legacy in legacy_candidates:
 | |
|             if os.path.exists(legacy):
 | |
|                 try:
 | |
|                     shutil.move(legacy, cfg_path)
 | |
|                 except Exception:
 | |
|                     shutil.copy2(legacy, cfg_path)
 | |
|                 return cfg_path
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
|     # If using a suffixed config and there is a base config (new or legacy), seed from it
 | |
|     try:
 | |
|         if suffix:
 | |
|             base_new = os.path.join(settings_dir, 'agent_settings.json')
 | |
|             base_old_settings = os.path.join(project_root, 'Agent', 'Settings', 'agent_settings.json')
 | |
|             base_legacy = os.path.join(project_root, 'agent_settings.json')
 | |
|             seed_from = None
 | |
|             if os.path.exists(base_new):
 | |
|                 seed_from = base_new
 | |
|             elif os.path.exists(base_old_settings):
 | |
|                 seed_from = base_old_settings
 | |
|             elif os.path.exists(base_legacy):
 | |
|                 seed_from = base_legacy
 | |
|             if seed_from:
 | |
|                 try:
 | |
|                     shutil.copy2(seed_from, cfg_path)
 | |
|                     return cfg_path
 | |
|                 except Exception:
 | |
|                     pass
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
|     # Migrate legacy root configs or prior Agent/Settings into Agent/Borealis/Settings
 | |
|     try:
 | |
|         legacy_names = [cfg_basename]
 | |
|         try:
 | |
|             if suffix:
 | |
|                 legacy_names.extend(legacy_map.get(suffix.upper(), []))
 | |
|         except Exception:
 | |
|             pass
 | |
|         old_settings_dir = os.path.join(project_root, 'Agent', 'Settings')
 | |
|         for legacy_name in legacy_names:
 | |
|             legacy_root = os.path.join(project_root, legacy_name)
 | |
|             if os.path.exists(legacy_root):
 | |
|                 try:
 | |
|                     shutil.move(legacy_root, cfg_path)
 | |
|                 except Exception:
 | |
|                     shutil.copy2(legacy_root, cfg_path)
 | |
|                 return cfg_path
 | |
|             legacy_old_settings = os.path.join(old_settings_dir, legacy_name)
 | |
|             if os.path.exists(legacy_old_settings):
 | |
|                 try:
 | |
|                     shutil.move(legacy_old_settings, cfg_path)
 | |
|                 except Exception:
 | |
|                     shutil.copy2(legacy_old_settings, cfg_path)
 | |
|                 return cfg_path
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
|     # Migration: from legacy user dir or script dir
 | |
|     legacy_user = _user_config_default_path()
 | |
|     legacy_script_dir = os.path.join(os.path.dirname(__file__), "agent_settings.json")
 | |
|     for legacy in (legacy_user, legacy_script_dir):
 | |
|         try:
 | |
|             if legacy and os.path.exists(legacy):
 | |
|                 try:
 | |
|                     shutil.move(legacy, cfg_path)
 | |
|                 except Exception:
 | |
|                     shutil.copy2(legacy, cfg_path)
 | |
|                 return cfg_path
 | |
|         except Exception:
 | |
|             pass
 | |
| 
 | |
|     # Nothing to migrate; return desired path in new Settings dir
 | |
|     return cfg_path
 | |
| 
 | |
| CONFIG_PATH = _resolve_config_path()
 | |
| DEFAULT_CONFIG = {
 | |
|     "config_file_watcher_interval": 2,
 | |
|     "agent_id": "",
 | |
|     "regions": {},
 | |
|     "installer_code": ""
 | |
| }
 | |
| 
 | |
| class ConfigManager:
 | |
|     def __init__(self, path):
 | |
|         self.path = path
 | |
|         self._last_mtime = None
 | |
|         self.data = {}
 | |
|         self.load()
 | |
| 
 | |
|     def load(self):
 | |
|         if not os.path.exists(self.path):
 | |
|             print("[INFO] agent_settings.json not found - Creating...")
 | |
|             self.data = DEFAULT_CONFIG.copy()
 | |
|             self._write()
 | |
|         else:
 | |
|             try:
 | |
|                 with open(self.path, 'r') as f:
 | |
|                     loaded = json.load(f)
 | |
|                 self.data = {**DEFAULT_CONFIG, **loaded}
 | |
|                 # Strip deprecated/relocated fields
 | |
|                 for k in ('borealis_server_url','max_task_workers','agent_operating_system','created'):
 | |
|                     if k in self.data:
 | |
|                         self.data.pop(k, None)
 | |
|                         # persist cleanup best-effort
 | |
|                         try:
 | |
|                             self._write()
 | |
|                         except Exception:
 | |
|                             pass
 | |
|             except Exception as e:
 | |
|                 print(f"[WARN] Failed to parse config: {e}")
 | |
|                 self.data = DEFAULT_CONFIG.copy()
 | |
|         try:
 | |
|             self._last_mtime = os.path.getmtime(self.path)
 | |
|         except Exception:
 | |
|             self._last_mtime = None
 | |
| 
 | |
|     def _write(self):
 | |
|         try:
 | |
|             with open(self.path, 'w') as f:
 | |
|                 json.dump(self.data, f, indent=2)
 | |
|         except Exception as e:
 | |
|             print(f"[ERROR] Could not write config: {e}")
 | |
| 
 | |
|     def watch(self):
 | |
|         try:
 | |
|             mtime = os.path.getmtime(self.path)
 | |
|             if self._last_mtime is None or mtime != self._last_mtime:
 | |
|                 self.load()
 | |
|                 return True
 | |
|         except Exception:
 | |
|             pass
 | |
|         return False
 | |
| 
 | |
| CONFIG = ConfigManager(CONFIG_PATH)
 | |
| CONFIG.load()
 | |
| 
 | |
| 
 | |
| class AgentHttpClient:
 | |
|     def __init__(self):
 | |
|         self.key_store = _key_store()
 | |
|         self.identity = IDENTITY
 | |
|         self.session = requests.Session()
 | |
|         self.base_url: Optional[str] = None
 | |
|         self.guid: Optional[str] = self.key_store.load_guid()
 | |
|         self.access_token: Optional[str] = self.key_store.load_access_token()
 | |
|         self.refresh_token: Optional[str] = self.key_store.load_refresh_token()
 | |
|         self.access_expires_at: Optional[int] = self.key_store.get_access_expiry()
 | |
|         self._auth_lock = threading.RLock()
 | |
|         self.refresh_base_url()
 | |
|         self._configure_verify()
 | |
|         if self.access_token:
 | |
|             self.session.headers.update({"Authorization": f"Bearer {self.access_token}"})
 | |
|         self.session.headers.setdefault("User-Agent", "Borealis-Agent/secure")
 | |
| 
 | |
|     # ------------------------------------------------------------------
 | |
|     # Session helpers
 | |
|     # ------------------------------------------------------------------
 | |
|     def refresh_base_url(self) -> None:
 | |
|         try:
 | |
|             url = (get_server_url() or "").strip()
 | |
|         except Exception:
 | |
|             url = ""
 | |
|         if not url:
 | |
|             url = "https://localhost:5000"
 | |
|         if url.endswith("/"):
 | |
|             url = url[:-1]
 | |
|         if url != self.base_url:
 | |
|             self.base_url = url
 | |
| 
 | |
|     def _configure_verify(self) -> None:
 | |
|         cert_path = self.key_store.server_certificate_path()
 | |
|         if cert_path and os.path.isfile(cert_path):
 | |
|             self.session.verify = cert_path
 | |
|         else:
 | |
|             self.session.verify = False
 | |
|             try:
 | |
|                 import urllib3  # type: ignore
 | |
| 
 | |
|                 urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)  # type: ignore[attr-defined]
 | |
|             except Exception:
 | |
|                 pass
 | |
| 
 | |
|     def auth_headers(self) -> Dict[str, str]:
 | |
|         if self.access_token:
 | |
|             return {"Authorization": f"Bearer {self.access_token}"}
 | |
|         return {}
 | |
| 
 | |
|     def configure_socketio(self, client: "socketio.AsyncClient") -> None:
 | |
|         """Align the Socket.IO engine's TLS verification with the REST client."""
 | |
|         try:
 | |
|             verify = getattr(self.session, "verify", True)
 | |
|             engine = getattr(client, "eio", None)
 | |
|             if engine is None:
 | |
|                 return
 | |
| 
 | |
|             # python-engineio accepts either a boolean or an ``ssl.SSLContext``
 | |
|             # for TLS verification.  When we have a pinned certificate bundle
 | |
|             # on disk, prefer constructing a dedicated context that trusts that
 | |
|             # bundle so WebSocket connections succeed even with private CAs.
 | |
|             if isinstance(verify, str) and os.path.isfile(verify):
 | |
|                 try:
 | |
|                     context = ssl.create_default_context(cafile=verify)
 | |
|                     context.check_hostname = False
 | |
|                 except Exception:
 | |
|                     context = None
 | |
|                 if context is not None:
 | |
|                     engine.ssl_context = context
 | |
|                     engine.ssl_verify = True
 | |
|                 else:
 | |
|                     engine.ssl_context = None
 | |
|                     engine.ssl_verify = verify
 | |
|             elif verify is False:
 | |
|                 engine.ssl_context = None
 | |
|                 engine.ssl_verify = False
 | |
|             else:
 | |
|                 engine.ssl_context = None
 | |
|                 engine.ssl_verify = True
 | |
|         except Exception:
 | |
|             pass
 | |
| 
 | |
|     # ------------------------------------------------------------------
 | |
|     # Enrollment & token management
 | |
|     # ------------------------------------------------------------------
 | |
|     def ensure_authenticated(self) -> None:
 | |
|         with self._auth_lock:
 | |
|             self._ensure_authenticated_locked()
 | |
| 
 | |
|     def _ensure_authenticated_locked(self) -> None:
 | |
|         self.refresh_base_url()
 | |
|         if not self.guid or not self.refresh_token:
 | |
|             self._perform_enrollment_locked()
 | |
|         if not self.access_token or self._token_expiring_soon():
 | |
|             self._refresh_access_token_locked()
 | |
| 
 | |
|     def _token_expiring_soon(self) -> bool:
 | |
|         if not self.access_token:
 | |
|             return True
 | |
|         if not self.access_expires_at:
 | |
|             return True
 | |
|         return (self.access_expires_at - time.time()) < 60
 | |
| 
 | |
|     def perform_enrollment(self) -> None:
 | |
|         with self._auth_lock:
 | |
|             self._perform_enrollment_locked()
 | |
| 
 | |
|     def _perform_enrollment_locked(self) -> None:
 | |
|         if self.guid and self.refresh_token:
 | |
|             return
 | |
|         code = self._resolve_installer_code()
 | |
|         if not code:
 | |
|             raise RuntimeError(
 | |
|                 "Installer code is required for enrollment. "
 | |
|                 "Set BOREALIS_INSTALLER_CODE, pass --installer-code, or update agent_settings.json."
 | |
|             )
 | |
|         self.refresh_base_url()
 | |
|         client_nonce = os.urandom(32)
 | |
|         payload = {
 | |
|             "hostname": socket.gethostname(),
 | |
|             "enrollment_code": code,
 | |
|             "agent_pubkey": PUBLIC_KEY_B64,
 | |
|             "client_nonce": base64.b64encode(client_nonce).decode("ascii"),
 | |
|         }
 | |
|         request_url = f"{self.base_url}/api/agent/enroll/request"
 | |
|         _log_agent("Starting enrollment request...", fname="agent.log")
 | |
|         resp = self.session.post(request_url, json=payload, timeout=30)
 | |
|         resp.raise_for_status()
 | |
|         data = resp.json()
 | |
|         if data.get("server_certificate"):
 | |
|             self.key_store.save_server_certificate(data["server_certificate"])
 | |
|             self._configure_verify()
 | |
|         signing_key = data.get("signing_key")
 | |
|         if signing_key:
 | |
|             try:
 | |
|                 self.store_server_signing_key(signing_key)
 | |
|             except Exception as exc:
 | |
|                 _log_agent(f'Unable to persist signing key from enrollment handshake: {exc}', fname='agent.error.log')
 | |
|         if data.get("status") != "pending":
 | |
|             raise RuntimeError(f"Unexpected enrollment status: {data}")
 | |
|         approval_reference = data.get("approval_reference")
 | |
|         server_nonce_b64 = data.get("server_nonce")
 | |
|         if not approval_reference or not server_nonce_b64:
 | |
|             raise RuntimeError("Enrollment response missing approval_reference or server_nonce")
 | |
|         server_nonce = base64.b64decode(server_nonce_b64)
 | |
|         poll_delay = max(int(data.get("poll_after_ms", 3000)) / 1000, 1)
 | |
|         while True:
 | |
|             time.sleep(min(poll_delay, 15))
 | |
|             signature = self.identity.sign(server_nonce + approval_reference.encode("utf-8") + client_nonce)
 | |
|             poll_payload = {
 | |
|                 "approval_reference": approval_reference,
 | |
|                 "client_nonce": base64.b64encode(client_nonce).decode("ascii"),
 | |
|                 "proof_sig": base64.b64encode(signature).decode("ascii"),
 | |
|             }
 | |
|             poll_resp = self.session.post(
 | |
|                 f"{self.base_url}/api/agent/enroll/poll",
 | |
|                 json=poll_payload,
 | |
|                 timeout=30,
 | |
|             )
 | |
|             poll_resp.raise_for_status()
 | |
|             poll_data = poll_resp.json()
 | |
|             status = poll_data.get("status")
 | |
|             if status == "pending":
 | |
|                 poll_delay = max(int(poll_data.get("poll_after_ms", 5000)) / 1000, 1)
 | |
|                 continue
 | |
|             if status == "denied":
 | |
|                 raise RuntimeError("Enrollment denied by operator")
 | |
|             if status in ("expired", "unknown"):
 | |
|                 raise RuntimeError(f"Enrollment failed with status={status}")
 | |
|             if status in ("approved", "completed"):
 | |
|                 self._finalize_enrollment(poll_data)
 | |
|                 break
 | |
|             raise RuntimeError(f"Unexpected enrollment poll response: {poll_data}")
 | |
| 
 | |
|     def _finalize_enrollment(self, payload: Dict[str, Any]) -> None:
 | |
|         server_cert = payload.get("server_certificate")
 | |
|         if server_cert:
 | |
|             self.key_store.save_server_certificate(server_cert)
 | |
|             self._configure_verify()
 | |
|         signing_key = payload.get("signing_key")
 | |
|         if signing_key:
 | |
|             try:
 | |
|                 self.store_server_signing_key(signing_key)
 | |
|             except Exception as exc:
 | |
|                 _log_agent(f'Unable to persist signing key from enrollment approval: {exc}', fname='agent.error.log')
 | |
|         guid = payload.get("guid")
 | |
|         access_token = payload.get("access_token")
 | |
|         refresh_token = payload.get("refresh_token")
 | |
|         expires_in = int(payload.get("expires_in") or 900)
 | |
|         if not (guid and access_token and refresh_token):
 | |
|             raise RuntimeError("Enrollment approval response missing tokens or guid")
 | |
|         self.guid = str(guid).strip()
 | |
|         self.access_token = access_token.strip()
 | |
|         self.refresh_token = refresh_token.strip()
 | |
|         expiry = int(time.time()) + max(expires_in - 5, 0)
 | |
|         self.access_expires_at = expiry
 | |
|         self.key_store.save_guid(self.guid)
 | |
|         self.key_store.save_refresh_token(self.refresh_token)
 | |
|         self.key_store.save_access_token(self.access_token, expires_at=expiry)
 | |
|         self.key_store.set_access_binding(SSL_KEY_FINGERPRINT)
 | |
|         self.session.headers.update({"Authorization": f"Bearer {self.access_token}"})
 | |
|         try:
 | |
|             _update_agent_id_for_guid(self.guid)
 | |
|         except Exception as exc:
 | |
|             _log_agent(f"Failed to update agent id after enrollment: {exc}", fname="agent.error.log")
 | |
|         _log_agent(f"Enrollment finalized for guid={self.guid}", fname="agent.log")
 | |
| 
 | |
|     def refresh_access_token(self) -> None:
 | |
|         with self._auth_lock:
 | |
|             self._refresh_access_token_locked()
 | |
| 
 | |
|     def _refresh_access_token_locked(self) -> None:
 | |
|         if not self.refresh_token or not self.guid:
 | |
|             self._clear_tokens_locked()
 | |
|             self._perform_enrollment_locked()
 | |
|             return
 | |
|         payload = {"guid": self.guid, "refresh_token": self.refresh_token}
 | |
|         resp = self.session.post(
 | |
|             f"{self.base_url}/api/agent/token/refresh",
 | |
|             json=payload,
 | |
|             headers=self.auth_headers(),
 | |
|             timeout=20,
 | |
|         )
 | |
|         if resp.status_code in (401, 403):
 | |
|             _log_agent("Refresh token rejected; re-enrolling", fname="agent.error.log")
 | |
|             self._clear_tokens_locked()
 | |
|             self._perform_enrollment_locked()
 | |
|             return
 | |
|         resp.raise_for_status()
 | |
|         data = resp.json()
 | |
|         access_token = data.get("access_token")
 | |
|         expires_in = int(data.get("expires_in") or 900)
 | |
|         if not access_token:
 | |
|             raise RuntimeError("Token refresh response missing access_token")
 | |
|         self.access_token = access_token.strip()
 | |
|         expiry = int(time.time()) + max(expires_in - 5, 0)
 | |
|         self.access_expires_at = expiry
 | |
|         self.key_store.save_access_token(self.access_token, expires_at=expiry)
 | |
|         self.key_store.set_access_binding(SSL_KEY_FINGERPRINT)
 | |
|         self.session.headers.update({"Authorization": f"Bearer {self.access_token}"})
 | |
| 
 | |
|     def clear_tokens(self) -> None:
 | |
|         with self._auth_lock:
 | |
|             self._clear_tokens_locked()
 | |
| 
 | |
|     def _clear_tokens_locked(self) -> None:
 | |
|         self.key_store.clear_tokens()
 | |
|         self.access_token = None
 | |
|         self.refresh_token = None
 | |
|         self.access_expires_at = None
 | |
|         self.guid = self.key_store.load_guid()
 | |
|         self.session.headers.pop("Authorization", None)
 | |
| 
 | |
|     def _resolve_installer_code(self) -> str:
 | |
|         if INSTALLER_CODE_OVERRIDE:
 | |
|             return INSTALLER_CODE_OVERRIDE
 | |
|         try:
 | |
|             code = (CONFIG.data.get("installer_code") or "").strip()
 | |
|             return code
 | |
|         except Exception:
 | |
|             return ""
 | |
| 
 | |
|     # ------------------------------------------------------------------
 | |
|     # HTTP helpers
 | |
|     # ------------------------------------------------------------------
 | |
|     def post_json(self, path: str, payload: Optional[Dict[str, Any]] = None, *, require_auth: bool = True) -> Any:
 | |
|         if require_auth:
 | |
|             self.ensure_authenticated()
 | |
|         url = f"{self.base_url}{path}"
 | |
|         headers = self.auth_headers()
 | |
|         response = self.session.post(url, json=payload, headers=headers, timeout=30)
 | |
|         if response.status_code in (401, 403) and require_auth:
 | |
|             self.clear_tokens()
 | |
|             self.ensure_authenticated()
 | |
|             headers = self.auth_headers()
 | |
|             response = self.session.post(url, json=payload, headers=headers, timeout=30)
 | |
|         response.raise_for_status()
 | |
|         if response.headers.get("Content-Type", "").lower().startswith("application/json"):
 | |
|             return response.json()
 | |
|         return response.text
 | |
| 
 | |
|     async def async_post_json(
 | |
|         self,
 | |
|         path: str,
 | |
|         payload: Optional[Dict[str, Any]] = None,
 | |
|         *,
 | |
|         require_auth: bool = True,
 | |
|     ) -> Any:
 | |
|         loop = asyncio.get_running_loop()
 | |
|         task = partial(self.post_json, path, payload, require_auth=require_auth)
 | |
|         return await loop.run_in_executor(None, task)
 | |
| 
 | |
|     def websocket_base_url(self) -> str:
 | |
|         self.refresh_base_url()
 | |
|         return self.base_url or "https://localhost:5000"
 | |
| 
 | |
|     def store_server_signing_key(self, value: str) -> None:
 | |
|         try:
 | |
|             self.key_store.save_server_signing_key(value)
 | |
|         except Exception as exc:
 | |
|             _log_agent(f"Unable to store server signing key: {exc}", fname="agent.error.log")
 | |
| 
 | |
|     def load_server_signing_key(self) -> Optional[str]:
 | |
|         try:
 | |
|             return self.key_store.load_server_signing_key()
 | |
|         except Exception:
 | |
|             return None
 | |
| 
 | |
| 
 | |
| HTTP_CLIENT: Optional[AgentHttpClient] = None
 | |
| 
 | |
| 
 | |
| def http_client() -> AgentHttpClient:
 | |
|     global HTTP_CLIENT
 | |
|     if HTTP_CLIENT is None:
 | |
|         HTTP_CLIENT = AgentHttpClient()
 | |
|     return HTTP_CLIENT
 | |
| 
 | |
| def _get_context_label() -> str:
 | |
|     return 'SYSTEM' if SYSTEM_SERVICE_MODE else 'CURRENTUSER'
 | |
| 
 | |
| 
 | |
| def _normalize_agent_guid(guid: str) -> str:
 | |
|     try:
 | |
|         if not guid:
 | |
|             return ''
 | |
|         value = str(guid).strip().replace('\ufeff', '')
 | |
|         if not value:
 | |
|             return ''
 | |
|         value = value.strip('{}')
 | |
|         try:
 | |
|             return str(uuid.UUID(value)).upper()
 | |
|         except Exception:
 | |
|             cleaned = ''.join(ch for ch in value if ch in string.hexdigits or ch == '-')
 | |
|             cleaned = cleaned.strip('-')
 | |
|             if cleaned:
 | |
|                 try:
 | |
|                     return str(uuid.UUID(cleaned)).upper()
 | |
|                 except Exception:
 | |
|                     pass
 | |
|             return value.upper()
 | |
|     except Exception:
 | |
|         return ''
 | |
| 
 | |
| 
 | |
| def _read_agent_guid_from_disk() -> str:
 | |
|     try:
 | |
|         ks_guid = _key_store().load_guid()
 | |
|         if ks_guid:
 | |
|             return _normalize_agent_guid(ks_guid)
 | |
|         path = _agent_guid_path()
 | |
|         if os.path.isfile(path):
 | |
|             with open(path, 'r', encoding='utf-8') as fh:
 | |
|                 value = fh.read()
 | |
|             return _normalize_agent_guid(value)
 | |
|     except Exception:
 | |
|         pass
 | |
|     return ''
 | |
| 
 | |
| 
 | |
| def _ensure_agent_guid() -> str:
 | |
|     guid = _read_agent_guid_from_disk()
 | |
|     if guid:
 | |
|         return guid
 | |
|     new_guid = str(uuid.uuid4()).upper()
 | |
|     _persist_agent_guid_local(new_guid)
 | |
|     return new_guid
 | |
| 
 | |
| 
 | |
| def _compose_agent_id(hostname: str, guid: str, context: str) -> str:
 | |
|     host = (hostname or '').strip()
 | |
|     if not host:
 | |
|         host = 'UNKNOWN-HOST'
 | |
|     host = host.replace(' ', '-').upper()
 | |
|     normalized_guid = _normalize_agent_guid(guid) or guid or 'UNKNOWN-GUID'
 | |
|     context_label = (context or '').strip().upper() or _get_context_label()
 | |
|     return f"{host}_{normalized_guid}_{context_label}"
 | |
| 
 | |
| 
 | |
| def _update_agent_id_for_guid(guid: str):
 | |
|     normalized_guid = _normalize_agent_guid(guid)
 | |
|     if not normalized_guid:
 | |
|         return
 | |
|     desired = _compose_agent_id(socket.gethostname(), normalized_guid, _get_context_label())
 | |
|     existing = (CONFIG.data.get('agent_id') or '').strip()
 | |
|     if existing != desired:
 | |
|         CONFIG.data['agent_id'] = desired
 | |
|         CONFIG._write()
 | |
|     global AGENT_ID
 | |
|     AGENT_ID = CONFIG.data['agent_id']
 | |
| 
 | |
| 
 | |
| def init_agent_id():
 | |
|     guid = _ensure_agent_guid()
 | |
|     _update_agent_id_for_guid(guid)
 | |
|     return CONFIG.data['agent_id']
 | |
| 
 | |
| AGENT_ID = init_agent_id()
 | |
| 
 | |
| def clear_regions_only():
 | |
|     CONFIG.data['regions'] = CONFIG.data.get('regions', {})
 | |
|     CONFIG._write()
 | |
| 
 | |
| clear_regions_only()
 | |
| 
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| # CORE SECTION: OPERATING SYSTEM DETECTION
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| def detect_agent_os():
 | |
|     """
 | |
|     Detects the full, user-friendly operating system name and version.
 | |
|     Examples:
 | |
|       - "Windows 11"
 | |
|       - "Windows 10"
 | |
|       - "Fedora Workstation 42"
 | |
|       - "Ubuntu 22.04 LTS"
 | |
|       - "macOS Sonoma"
 | |
|     Falls back to a generic name if detection fails.
 | |
|     """
 | |
|     try:
 | |
|         plat = platform.system().lower()
 | |
| 
 | |
|         if plat.startswith('win'):
 | |
|             # Aim for: "Microsoft Windows 11 Pro 24H2 Build 26100.5074"
 | |
|             # Pull details from the registry when available and correct
 | |
|             # historical quirks like CurrentVersion reporting 6.3.
 | |
|             try:
 | |
|                 import winreg  # Only available on Windows
 | |
| 
 | |
|                 reg_path = r"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion"
 | |
|                 access = winreg.KEY_READ
 | |
|                 try:
 | |
|                     access |= winreg.KEY_WOW64_64KEY
 | |
|                 except Exception:
 | |
|                     pass
 | |
| 
 | |
|                 try:
 | |
|                     key = winreg.OpenKey(winreg.HKEY_LOCAL_MACHINE, reg_path, 0, access)
 | |
|                 except OSError:
 | |
|                     key = winreg.OpenKey(winreg.HKEY_LOCAL_MACHINE, reg_path, 0, winreg.KEY_READ)
 | |
| 
 | |
|                 def _get(name, default=None):
 | |
|                     try:
 | |
|                         return winreg.QueryValueEx(key, name)[0]
 | |
|                     except Exception:
 | |
|                         return default
 | |
| 
 | |
|                 product_name = _get("ProductName", "")  # e.g., "Windows 11 Pro"
 | |
|                 edition_id = _get("EditionID", "")       # e.g., "Professional"
 | |
|                 display_version = _get("DisplayVersion", "")  # e.g., "24H2" / "22H2"
 | |
|                 release_id = _get("ReleaseId", "")            # e.g., "2004" on older Windows 10
 | |
|                 build_number = _get("CurrentBuildNumber", "") or _get("CurrentBuild", "")
 | |
|                 ubr = _get("UBR", None)  # Update Build Revision (int)
 | |
| 
 | |
|                 # Determine Windows major (10 vs 11) from build number to avoid relying
 | |
|                 # on inconsistent registry values like CurrentVersion (which may say 6.3).
 | |
|                 try:
 | |
|                     build_int = int(str(build_number).split(".")[0]) if build_number else 0
 | |
|                 except Exception:
 | |
|                     build_int = 0
 | |
|                 if build_int >= 22000:
 | |
|                     major_label = "11"
 | |
|                 elif build_int >= 10240:
 | |
|                     major_label = "10"
 | |
|                 else:
 | |
|                     major_label = platform.release()
 | |
| 
 | |
|                 # Derive friendly edition name, prefer parsing from ProductName
 | |
|                 edition = ""
 | |
|                 pn = product_name or ""
 | |
|                 if pn.lower().startswith("windows "):
 | |
|                     tokens = pn.split()
 | |
|                     # tokens like ["Windows", "11", "Pro", ...]
 | |
|                     if len(tokens) >= 3:
 | |
|                         edition = " ".join(tokens[2:])
 | |
|                 if not edition and edition_id:
 | |
|                     eid_map = {
 | |
|                         "Professional": "Pro",
 | |
|                         "ProfessionalN": "Pro N",
 | |
|                         "ProfessionalEducation": "Pro Education",
 | |
|                         "ProfessionalWorkstation": "Pro for Workstations",
 | |
|                         "Enterprise": "Enterprise",
 | |
|                         "EnterpriseN": "Enterprise N",
 | |
|                         "EnterpriseS": "Enterprise LTSC",
 | |
|                         "Education": "Education",
 | |
|                         "EducationN": "Education N",
 | |
|                         "Core": "Home",
 | |
|                         "CoreN": "Home N",
 | |
|                         "CoreSingleLanguage": "Home Single Language",
 | |
|                         "IoTEnterprise": "IoT Enterprise",
 | |
|                     }
 | |
|                     edition = eid_map.get(edition_id, edition_id)
 | |
| 
 | |
|                 os_name = f"Windows {major_label}"
 | |
| 
 | |
|                 # Choose version label: DisplayVersion (preferred) then ReleaseId
 | |
|                 version_label = display_version or release_id or ""
 | |
| 
 | |
|                 # Build string with UBR if present
 | |
|                 if isinstance(ubr, int):
 | |
|                     build_str = f"{build_number}.{ubr}" if build_number else str(ubr)
 | |
|                 else:
 | |
|                     try:
 | |
|                         build_str = f"{build_number}.{int(ubr)}" if build_number and ubr is not None else build_number
 | |
|                     except Exception:
 | |
|                         build_str = build_number
 | |
| 
 | |
|                 parts = ["Microsoft", os_name]
 | |
|                 if edition:
 | |
|                     parts.append(edition)
 | |
|                 if version_label:
 | |
|                     parts.append(version_label)
 | |
|                 if build_str:
 | |
|                     parts.append(f"Build {build_str}")
 | |
| 
 | |
|                 # Correct possible mislabeling in ProductName (e.g., says Windows 10 on Win 11)
 | |
|                 # by trusting build-based major_label.
 | |
|                 return " ".join(p for p in parts if p).strip()
 | |
| 
 | |
|             except Exception:
 | |
|                 # Safe fallback if registry lookups fail
 | |
|                 return f"Windows {platform.release()}"
 | |
| 
 | |
|         elif plat.startswith('linux'):
 | |
|             try:
 | |
|                 import distro  # External package, better for Linux OS detection
 | |
|                 name = distro.name(pretty=True)  # e.g., "Fedora Workstation 42"
 | |
|                 if name:
 | |
|                     return name
 | |
|                 else:
 | |
|                     # Fallback if pretty name not found
 | |
|                     return f"{platform.system()} {platform.release()}"
 | |
|             except ImportError:
 | |
|                 # Fallback to basic info if distro not installed
 | |
|                 return f"{platform.system()} {platform.release()}"
 | |
| 
 | |
|         elif plat.startswith('darwin'):
 | |
|             # macOS — platform.mac_ver()[0] returns version number
 | |
|             version = platform.mac_ver()[0]
 | |
|             # Optional: map version numbers to marketing names
 | |
|             macos_names = {
 | |
|                 "14": "Sonoma",
 | |
|                 "13": "Ventura",
 | |
|                 "12": "Monterey",
 | |
|                 "11": "Big Sur",
 | |
|                 "10.15": "Catalina"
 | |
|             }
 | |
|             pretty_name = macos_names.get(".".join(version.split(".")[:2]), "")
 | |
|             return f"macOS {pretty_name or version}"
 | |
| 
 | |
|         else:
 | |
|             return f"Unknown OS ({platform.system()} {platform.release()})"
 | |
| 
 | |
|     except Exception as e:
 | |
|         print(f"[WARN] OS detection failed: {e}")
 | |
|         return "Unknown"
 | |
| 
 | |
| 
 | |
| def _system_uptime_seconds() -> Optional[int]:
 | |
|     try:
 | |
|         if psutil and hasattr(psutil, "boot_time"):
 | |
|             return int(time.time() - psutil.boot_time())
 | |
|     except Exception:
 | |
|         pass
 | |
|     return None
 | |
| 
 | |
| 
 | |
| def _collect_heartbeat_metrics() -> Dict[str, Any]:
 | |
|     metrics: Dict[str, Any] = {
 | |
|         "operating_system": detect_agent_os(),
 | |
|         "service_mode": SERVICE_MODE,
 | |
|     }
 | |
|     uptime = _system_uptime_seconds()
 | |
|     if uptime is not None:
 | |
|         metrics["uptime"] = uptime
 | |
|     try:
 | |
|         metrics["hostname"] = socket.gethostname()
 | |
|     except Exception:
 | |
|         pass
 | |
|     try:
 | |
|         metrics["username"] = getpass.getuser()
 | |
|     except Exception:
 | |
|         pass
 | |
|     if psutil:
 | |
|         try:
 | |
|             cpu = psutil.cpu_percent(interval=None)
 | |
|             metrics["cpu_percent"] = cpu
 | |
|         except Exception:
 | |
|             pass
 | |
|         try:
 | |
|             mem = psutil.virtual_memory()
 | |
|             metrics["memory_percent"] = getattr(mem, "percent", None)
 | |
|         except Exception:
 | |
|             pass
 | |
|     return metrics
 | |
| 
 | |
| 
 | |
| SERVER_CERT_PATH = _key_store().server_certificate_path()
 | |
| 
 | |
| 
 | |
| IDENTITY = _key_store().load_or_create_identity()
 | |
| SSL_KEY_FINGERPRINT = IDENTITY.fingerprint
 | |
| PUBLIC_KEY_B64 = IDENTITY.public_key_b64
 | |
| 
 | |
| 
 | |
| def get_server_url() -> str:
 | |
|     """Return the Borealis server URL from env or Agent/Borealis/Settings/server_url.txt.
 | |
|     - Strips UTF-8 BOM and whitespace
 | |
|     - Adds http:// if scheme is missing
 | |
|     - Falls back to http://localhost:5000 when missing/invalid
 | |
|     """
 | |
|     def _sanitize(val: str) -> str:
 | |
|         try:
 | |
|             s = (val or '').strip().replace('\ufeff', '')
 | |
|             if not s:
 | |
|                 return ''
 | |
|             if not (s.lower().startswith('http://') or s.lower().startswith('https://') or s.lower().startswith('ws://') or s.lower().startswith('wss://')):
 | |
|                 s = 'http://' + s
 | |
|             # Remove trailing slash for consistency
 | |
|             return s.rstrip('/')
 | |
|         except Exception:
 | |
|             return ''
 | |
| 
 | |
|     try:
 | |
|         env_url = os.environ.get('BOREALIS_SERVER_URL')
 | |
|         if env_url and _sanitize(env_url):
 | |
|             return _sanitize(env_url)
 | |
|         # New location
 | |
|         path = os.path.join(_settings_dir(), 'server_url.txt')
 | |
|         if os.path.isfile(path):
 | |
|             try:
 | |
|                 with open(path, 'r', encoding='utf-8-sig') as f:
 | |
|                     txt = f.read()
 | |
|                 s = _sanitize(txt)
 | |
|                 if s:
 | |
|                     return s
 | |
|             except Exception:
 | |
|                 pass
 | |
|         # Prior interim location (Agent/Settings) migration support
 | |
|         try:
 | |
|             project_root = _find_project_root()
 | |
|             old_path = os.path.join(project_root, 'Agent', 'Settings', 'server_url.txt')
 | |
|             if os.path.isfile(old_path):
 | |
|                 with open(old_path, 'r', encoding='utf-8-sig') as f:
 | |
|                     txt = f.read()
 | |
|                 s = _sanitize(txt)
 | |
|                 if s:
 | |
|                     # Best-effort copy forward to new location so future reads use it
 | |
|                     try:
 | |
|                         os.makedirs(_settings_dir(), exist_ok=True)
 | |
|                         with open(path, 'w', encoding='utf-8') as wf:
 | |
|                             wf.write(s)
 | |
|                     except Exception:
 | |
|                         pass
 | |
|                     return s
 | |
|         except Exception:
 | |
|             pass
 | |
|     except Exception:
 | |
|         pass
 | |
|     return 'http://localhost:5000'
 | |
| 
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| # CORE SECTION: ASYNC TASK / WEBSOCKET
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| sio = socketio.AsyncClient(reconnection=True, reconnection_attempts=0, reconnection_delay=5)
 | |
| role_tasks = {}
 | |
| background_tasks = []
 | |
| AGENT_LOOP = None
 | |
| ROLE_MANAGER = None
 | |
| ROLE_MANAGER_SYS = None
 | |
| 
 | |
| # ---------------- Local IPC Bridge (Service -> Agent) ----------------
 | |
| def start_agent_bridge_pipe(loop_ref):
 | |
|     import threading
 | |
|     import win32pipe, win32file, win32con, pywintypes
 | |
| 
 | |
|     pipe_name = r"\\.\pipe\Borealis_Agent_Bridge"
 | |
| 
 | |
|     def forward_to_server(msg: dict):
 | |
|         try:
 | |
|             evt = msg.get('type')
 | |
|             if evt == 'screenshot':
 | |
|                 payload = {
 | |
|                     'agent_id': AGENT_ID,
 | |
|                     'node_id': msg.get('node_id') or 'user_session',
 | |
|                     'image_base64': msg.get('image_base64') or '',
 | |
|                     'timestamp': msg.get('timestamp') or int(time.time())
 | |
|                 }
 | |
|                 asyncio.run_coroutine_threadsafe(sio.emit('agent_screenshot_task', payload), loop_ref)
 | |
|         except Exception:
 | |
|             pass
 | |
| 
 | |
|     def server_thread():
 | |
|         while True:
 | |
|             try:
 | |
|                 handle = win32pipe.CreateNamedPipe(
 | |
|                     pipe_name,
 | |
|                     win32con.PIPE_ACCESS_DUPLEX,
 | |
|                     win32con.PIPE_TYPE_MESSAGE | win32con.PIPE_READMODE_MESSAGE | win32con.PIPE_WAIT,
 | |
|                     1, 65536, 65536, 0, None)
 | |
|             except pywintypes.error:
 | |
|                 time.sleep(1.0)
 | |
|                 continue
 | |
| 
 | |
|             try:
 | |
|                 win32pipe.ConnectNamedPipe(handle, None)
 | |
|             except pywintypes.error:
 | |
|                 try:
 | |
|                     win32file.CloseHandle(handle)
 | |
|                 except Exception:
 | |
|                     pass
 | |
|                 time.sleep(0.5)
 | |
|                 continue
 | |
| 
 | |
|             # Read loop per connection
 | |
|             try:
 | |
|                 while True:
 | |
|                     try:
 | |
|                         hr, data = win32file.ReadFile(handle, 65536)
 | |
|                         if not data:
 | |
|                             break
 | |
|                         try:
 | |
|                             obj = json.loads(data.decode('utf-8', errors='ignore'))
 | |
|                             forward_to_server(obj)
 | |
|                         except Exception:
 | |
|                             pass
 | |
|                     except pywintypes.error:
 | |
|                         break
 | |
|             finally:
 | |
|                 try:
 | |
|                     win32file.CloseHandle(handle)
 | |
|                 except Exception:
 | |
|                     pass
 | |
|             time.sleep(0.2)
 | |
| 
 | |
|     t = threading.Thread(target=server_thread, daemon=True)
 | |
|     t.start()
 | |
| 
 | |
| def send_service_control(msg: dict):
 | |
|     try:
 | |
|         import win32file
 | |
|         pipe = r"\\.\pipe\Borealis_Service_Control"
 | |
|         h = win32file.CreateFile(
 | |
|             pipe,
 | |
|             win32file.GENERIC_WRITE,
 | |
|             0,
 | |
|             None,
 | |
|             win32file.OPEN_EXISTING,
 | |
|             0,
 | |
|             None,
 | |
|         )
 | |
|         try:
 | |
|             win32file.WriteFile(h, json.dumps(msg).encode('utf-8'))
 | |
|         finally:
 | |
|             win32file.CloseHandle(h)
 | |
|     except Exception:
 | |
|         pass
 | |
| IS_WINDOWS = sys.platform.startswith('win')
 | |
| 
 | |
| def _is_admin_windows():
 | |
|     if not IS_WINDOWS:
 | |
|         return False
 | |
|     try:
 | |
|         import ctypes
 | |
|         return ctypes.windll.shell32.IsUserAnAdmin() != 0
 | |
|     except Exception:
 | |
|         return False
 | |
| 
 | |
| def _write_temp_script(content: str, suffix: str):
 | |
|     import tempfile
 | |
|     temp_dir = os.path.join(tempfile.gettempdir(), "Borealis", "quick_jobs")
 | |
|     os.makedirs(temp_dir, exist_ok=True)
 | |
|     fd, path = tempfile.mkstemp(prefix="bj_", suffix=suffix, dir=temp_dir, text=True)
 | |
|     with os.fdopen(fd, 'w', encoding='utf-8', newline='\n') as fh:
 | |
|         fh.write(content or "")
 | |
|     return path
 | |
| 
 | |
| async def _run_powershell_local(path: str):
 | |
|     """Run powershell script as current user hidden window and capture output."""
 | |
|     ps = None
 | |
|     if IS_WINDOWS:
 | |
|         ps = os.path.expandvars(r"%SystemRoot%\\System32\\WindowsPowerShell\\v1.0\\powershell.exe")
 | |
|         if not os.path.isfile(ps):
 | |
|             ps = "powershell.exe"
 | |
|     else:
 | |
|         ps = "pwsh"
 | |
|     try:
 | |
|         proc = await asyncio.create_subprocess_exec(
 | |
|             ps,
 | |
|             "-ExecutionPolicy", "Bypass" if IS_WINDOWS else "Bypass",
 | |
|             "-NoProfile",
 | |
|             "-File", path,
 | |
|             stdout=asyncio.subprocess.PIPE,
 | |
|             stderr=asyncio.subprocess.PIPE,
 | |
|             creationflags=(0x08000000 if IS_WINDOWS else 0)  # CREATE_NO_WINDOW
 | |
|         )
 | |
|         out_b, err_b = await proc.communicate()
 | |
|         rc = proc.returncode
 | |
|         out = (out_b or b"").decode(errors='replace')
 | |
|         err = (err_b or b"").decode(errors='replace')
 | |
|         return rc, out, err
 | |
|     except Exception as e:
 | |
|         return -1, "", str(e)
 | |
| 
 | |
| async def _run_powershell_as_system(path: str):
 | |
|     """Attempt to run as SYSTEM using schtasks; requires admin."""
 | |
|     if not IS_WINDOWS:
 | |
|         return -1, "", "SYSTEM run not supported on this OS"
 | |
|     # Name with timestamp to avoid collisions
 | |
|     name = f"Borealis_QuickJob_{int(time.time())}_{random.randint(1000,9999)}"
 | |
|     # Create scheduled task
 | |
|     # Start time: 1 minute from now (HH:MM)
 | |
|     t = time.localtime(time.time() + 60)
 | |
|     st = f"{t.tm_hour:02d}:{t.tm_min:02d}"
 | |
|     create_cmd = [
 | |
|         "schtasks", "/Create", "/TN", name,
 | |
|         "/TR", f"\"powershell.exe -ExecutionPolicy Bypass -NoProfile -WindowStyle Hidden -File \"\"{path}\"\"\"",
 | |
|         "/SC", "ONCE", "/ST", st, "/RL", "HIGHEST", "/RU", "SYSTEM", "/F"
 | |
|     ]
 | |
|     try:
 | |
|         p1 = await asyncio.create_subprocess_exec(*create_cmd, stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE)
 | |
|         c_out, c_err = await p1.communicate()
 | |
|         if p1.returncode != 0:
 | |
|             return p1.returncode, "", (c_err or b"").decode(errors='replace')
 | |
|         # Run immediately
 | |
|         run_cmd = ["schtasks", "/Run", "/TN", name]
 | |
|         p2 = await asyncio.create_subprocess_exec(*run_cmd, stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE)
 | |
|         r_out, r_err = await p2.communicate()
 | |
|         # Give some time for task to run and finish (best-effort)
 | |
|         await asyncio.sleep(5)
 | |
|         # We cannot reliably capture stdout from scheduled task directly; advise writing output to file in script if needed.
 | |
|         # Return status of scheduling; actual script result unknown. We will try to check last run result.
 | |
|         query_cmd = ["schtasks", "/Query", "/TN", name, "/V", "/FO", "LIST"]
 | |
|         p3 = await asyncio.create_subprocess_exec(*query_cmd, stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE)
 | |
|         q_out, q_err = await p3.communicate()
 | |
|         status_txt = (q_out or b"").decode(errors='replace')
 | |
|         # Cleanup
 | |
|         await asyncio.create_subprocess_exec("schtasks", "/Delete", "/TN", name, "/F")
 | |
|         # We cannot get stdout/stderr; return status text to stderr and treat success based on return codes
 | |
|         status = "Success" if p2.returncode == 0 else "Failed"
 | |
|         return 0 if status == "Success" else 1, "", status_txt
 | |
|     except Exception as e:
 | |
|         return -1, "", str(e)
 | |
| 
 | |
| async def _run_powershell_with_credentials(path: str, username: str, password: str):
 | |
|     if not IS_WINDOWS:
 | |
|         return -1, "", "Credentialed run not supported on this OS"
 | |
|     # Build a one-liner to convert plaintext password to SecureString and run Start-Process -Credential
 | |
|     ps_cmd = (
 | |
|         f"$user=\"{username}\"; "
 | |
|         f"$pass=\"{password}\"; "
 | |
|         f"$sec=ConvertTo-SecureString $pass -AsPlainText -Force; "
 | |
|         f"$cred=New-Object System.Management.Automation.PSCredential($user,$sec); "
 | |
|         f"Start-Process powershell -ArgumentList '-ExecutionPolicy Bypass -NoProfile -File \"{path}\"' -Credential $cred -WindowStyle Hidden -PassThru | Wait-Process;"
 | |
|     )
 | |
|     try:
 | |
|         proc = await asyncio.create_subprocess_exec(
 | |
|             "powershell.exe", "-NoProfile", "-Command", ps_cmd,
 | |
|             stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE,
 | |
|             creationflags=(0x08000000 if IS_WINDOWS else 0)
 | |
|         )
 | |
|         out_b, err_b = await proc.communicate()
 | |
|         out = (out_b or b"").decode(errors='replace')
 | |
|         err = (err_b or b"").decode(errors='replace')
 | |
|         return proc.returncode, out, err
 | |
|     except Exception as e:
 | |
|         return -1, "", str(e)
 | |
| 
 | |
| async def stop_all_roles():
 | |
|     print("[DEBUG] Stopping all roles.")
 | |
|     try:
 | |
|         if ROLE_MANAGER is not None:
 | |
|             ROLE_MANAGER.stop_all()
 | |
|     except Exception:
 | |
|         pass
 | |
|     try:
 | |
|         if ROLE_MANAGER_SYS is not None:
 | |
|             ROLE_MANAGER_SYS.stop_all()
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
| # ---------------- Heartbeat ----------------
 | |
| async def send_heartbeat():
 | |
|     """
 | |
|     Periodically send agent heartbeat to the server so the Devices page can
 | |
|     show hostname, OS, and last_seen.
 | |
|     """
 | |
|     await asyncio.sleep(15)
 | |
|     client = http_client()
 | |
|     while True:
 | |
|         try:
 | |
|             client.ensure_authenticated()
 | |
|             payload = {
 | |
|                 "guid": client.guid or _read_agent_guid_from_disk(),
 | |
|                 "hostname": socket.gethostname(),
 | |
|                 "inventory": {},
 | |
|                 "metrics": _collect_heartbeat_metrics(),
 | |
|             }
 | |
|             await client.async_post_json("/api/agent/heartbeat", payload, require_auth=True)
 | |
|         except Exception as exc:
 | |
|             _log_agent(f'Heartbeat post failed: {exc}', fname='agent.error.log')
 | |
|         await asyncio.sleep(60)
 | |
| 
 | |
| 
 | |
| def _verify_and_store_script_signature(
 | |
|     client: AgentHttpClient,
 | |
|     script_bytes: bytes,
 | |
|     signature_b64: str,
 | |
|     signing_key_hint: Optional[str] = None,
 | |
| ) -> bool:
 | |
|     candidates: List[str] = []
 | |
|     if isinstance(signing_key_hint, str) and signing_key_hint.strip():
 | |
|         candidates.append(signing_key_hint.strip())
 | |
|     stored_key = client.load_server_signing_key()
 | |
|     if stored_key:
 | |
|         key_text = stored_key.strip()
 | |
|         if key_text and key_text not in candidates:
 | |
|             candidates.append(key_text)
 | |
|     for key_b64 in candidates:
 | |
|         try:
 | |
|             key_der = base64.b64decode(key_b64, validate=True)
 | |
|         except Exception:
 | |
|             continue
 | |
|         try:
 | |
|             public_key = serialization.load_der_public_key(key_der)
 | |
|         except Exception:
 | |
|             continue
 | |
|         if not isinstance(public_key, ed25519.Ed25519PublicKey):
 | |
|             continue
 | |
|         try:
 | |
|             signature = base64.b64decode(signature_b64, validate=True)
 | |
|         except Exception:
 | |
|             return False
 | |
|         try:
 | |
|             public_key.verify(signature, script_bytes)
 | |
|             if stored_key and stored_key.strip() != key_b64:
 | |
|                 client.store_server_signing_key(key_b64)
 | |
|             elif not stored_key:
 | |
|                 client.store_server_signing_key(key_b64)
 | |
|             return True
 | |
|         except Exception:
 | |
|             continue
 | |
|     return False
 | |
| 
 | |
| 
 | |
| async def poll_script_requests():
 | |
|     await asyncio.sleep(20)
 | |
|     client = http_client()
 | |
|     while True:
 | |
|         try:
 | |
|             client.ensure_authenticated()
 | |
|             payload = {"guid": client.guid or _read_agent_guid_from_disk()}
 | |
|             response = await client.async_post_json("/api/agent/script/request", payload, require_auth=True)
 | |
|             if isinstance(response, dict):
 | |
|                 signing_key = response.get("signing_key")
 | |
|                 script_b64 = response.get("script")
 | |
|                 signature_b64 = response.get("signature")
 | |
|                 sig_alg = (response.get("sig_alg") or "").lower()
 | |
|                 if script_b64 and signature_b64:
 | |
|                     script_bytes = _decode_base64_bytes(script_b64)
 | |
|                     if script_bytes is None:
 | |
|                         _log_agent('received script payload with invalid base64 encoding', fname='agent.error.log')
 | |
|                     elif sig_alg and sig_alg not in ("ed25519", "eddsa"):
 | |
|                         _log_agent(f'unsupported script signature algorithm: {sig_alg}', fname='agent.error.log')
 | |
|                     else:
 | |
|                         existing_key = client.load_server_signing_key()
 | |
|                         key_available = bool(
 | |
|                             (isinstance(signing_key, str) and signing_key.strip())
 | |
|                             or (isinstance(existing_key, str) and existing_key.strip())
 | |
|                         )
 | |
|                         if not key_available:
 | |
|                             _log_agent('no server signing key available to verify script payload', fname='agent.error.log')
 | |
|                         elif _verify_and_store_script_signature(client, script_bytes, signature_b64, signing_key):
 | |
|                             _log_agent('received signed script payload (verification succeeded); awaiting executor integration')
 | |
|                         else:
 | |
|                             _log_agent('rejected script payload due to invalid signature', fname='agent.error.log')
 | |
|                 elif signing_key:
 | |
|                     # No script content, but we may need to persist updated signing key.
 | |
|                     try:
 | |
|                         client.store_server_signing_key(signing_key)
 | |
|                     except Exception as exc:
 | |
|                         _log_agent(f'failed to persist server signing key: {exc}', fname='agent.error.log')
 | |
|         except Exception as exc:
 | |
|             _log_agent(f'script request poll failed: {exc}', fname='agent.error.log')
 | |
|         await asyncio.sleep(30)
 | |
| 
 | |
| # ---------------- Detailed Agent Data ----------------
 | |
| ## Moved to agent_info module
 | |
| 
 | |
| def collect_summary():
 | |
|     # migrated to role_DeviceInventory
 | |
|     return {}
 | |
| 
 | |
| def collect_software():
 | |
|     # migrated to role_DeviceInventory
 | |
|     return []
 | |
| 
 | |
| def collect_memory():
 | |
|     # migrated to role_DeviceInventory
 | |
|     entries = []
 | |
|     plat = platform.system().lower()
 | |
|     try:
 | |
|         if plat == "windows":
 | |
|             try:
 | |
|                 out = subprocess.run(
 | |
|                     ["wmic", "memorychip", "get", "BankLabel,Speed,SerialNumber,Capacity"],
 | |
|                     capture_output=True,
 | |
|                     text=True,
 | |
|                     timeout=60,
 | |
|                 )
 | |
|                 lines = [l for l in out.stdout.splitlines() if l.strip() and "BankLabel" not in l]
 | |
|                 for line in lines:
 | |
|                     parts = [p for p in line.split() if p]
 | |
|                     if len(parts) >= 4:
 | |
|                         entries.append({
 | |
|                             "slot": parts[0],
 | |
|                             "speed": parts[1],
 | |
|                             "serial": parts[2],
 | |
|                             "capacity": parts[3],
 | |
|                         })
 | |
|             except FileNotFoundError:
 | |
|                 ps_cmd = (
 | |
|                     "Get-CimInstance Win32_PhysicalMemory | "
 | |
|                     "Select-Object BankLabel,Speed,SerialNumber,Capacity | ConvertTo-Json"
 | |
|                 )
 | |
|                 out = subprocess.run(
 | |
|                     ["powershell", "-NoProfile", "-Command", ps_cmd],
 | |
|                     capture_output=True,
 | |
|                     text=True,
 | |
|                     timeout=60,
 | |
|                 )
 | |
|                 data = json.loads(out.stdout or "[]")
 | |
|                 if isinstance(data, dict):
 | |
|                     data = [data]
 | |
|                 for stick in data:
 | |
|                     entries.append({
 | |
|                         "slot": stick.get("BankLabel", "unknown"),
 | |
|                         "speed": str(stick.get("Speed", "unknown")),
 | |
|                         "serial": stick.get("SerialNumber", "unknown"),
 | |
|                         "capacity": stick.get("Capacity", "unknown"),
 | |
|                     })
 | |
|         elif plat == "linux":
 | |
|             out = subprocess.run(["dmidecode", "-t", "17"], capture_output=True, text=True)
 | |
|             slot = speed = serial = capacity = None
 | |
|             for line in out.stdout.splitlines():
 | |
|                 line = line.strip()
 | |
|                 if line.startswith("Locator:"):
 | |
|                     slot = line.split(":", 1)[1].strip()
 | |
|                 elif line.startswith("Speed:"):
 | |
|                     speed = line.split(":", 1)[1].strip()
 | |
|                 elif line.startswith("Serial Number:"):
 | |
|                     serial = line.split(":", 1)[1].strip()
 | |
|                 elif line.startswith("Size:"):
 | |
|                     capacity = line.split(":", 1)[1].strip()
 | |
|                 elif not line and slot:
 | |
|                     entries.append({
 | |
|                         "slot": slot,
 | |
|                         "speed": speed or "unknown",
 | |
|                         "serial": serial or "unknown",
 | |
|                         "capacity": capacity or "unknown",
 | |
|                     })
 | |
|                     slot = speed = serial = capacity = None
 | |
|             if slot:
 | |
|                 entries.append({
 | |
|                     "slot": slot,
 | |
|                     "speed": speed or "unknown",
 | |
|                     "serial": serial or "unknown",
 | |
|                     "capacity": capacity or "unknown",
 | |
|                 })
 | |
|     except Exception as e:
 | |
|         print(f"[WARN] collect_memory failed: {e}")
 | |
| 
 | |
|     if not entries:
 | |
|         try:
 | |
|             if psutil:
 | |
|                 vm = psutil.virtual_memory()
 | |
|                 entries.append({
 | |
|                     "slot": "physical",
 | |
|                     "speed": "unknown",
 | |
|                     "serial": "unknown",
 | |
|                     "capacity": vm.total,
 | |
|                 })
 | |
|         except Exception:
 | |
|             pass
 | |
|     return entries
 | |
| 
 | |
| def collect_storage():
 | |
|     # migrated to role_DeviceInventory
 | |
|     disks = []
 | |
|     plat = platform.system().lower()
 | |
|     try:
 | |
|         if psutil:
 | |
|             for part in psutil.disk_partitions():
 | |
|                 try:
 | |
|                     usage = psutil.disk_usage(part.mountpoint)
 | |
|                 except Exception:
 | |
|                     continue
 | |
|                 disks.append({
 | |
|                     "drive": part.device,
 | |
|                     "disk_type": "Removable" if "removable" in part.opts.lower() else "Fixed Disk",
 | |
|                     "usage": usage.percent,
 | |
|                     "total": usage.total,
 | |
|                     "free": usage.free,
 | |
|                     "used": usage.used,
 | |
|                 })
 | |
|         elif plat == "windows":
 | |
|             found = False
 | |
|             for letter in string.ascii_uppercase:
 | |
|                 drive = f"{letter}:\\"
 | |
|                 if os.path.exists(drive):
 | |
|                     try:
 | |
|                         usage = shutil.disk_usage(drive)
 | |
|                     except Exception:
 | |
|                         continue
 | |
|                     disks.append({
 | |
|                         "drive": drive,
 | |
|                         "disk_type": "Fixed Disk",
 | |
|                         "usage": (usage.used / usage.total * 100) if usage.total else 0,
 | |
|                         "total": usage.total,
 | |
|                         "free": usage.free,
 | |
|                         "used": usage.used,
 | |
|                     })
 | |
|                     found = True
 | |
|             if not found:
 | |
|                 try:
 | |
|                     out = subprocess.run(
 | |
|                         ["wmic", "logicaldisk", "get", "DeviceID,Size,FreeSpace"],
 | |
|                         capture_output=True,
 | |
|                         text=True,
 | |
|                         timeout=60,
 | |
|                     )
 | |
|                     lines = [l for l in out.stdout.splitlines() if l.strip()][1:]
 | |
|                     for line in lines:
 | |
|                         parts = line.split()
 | |
|                         if len(parts) >= 3:
 | |
|                             drive, free, size = parts[0], parts[1], parts[2]
 | |
|                             try:
 | |
|                                 total = float(size)
 | |
|                                 free_bytes = float(free)
 | |
|                                 used = total - free_bytes
 | |
|                                 usage = (used / total * 100) if total else 0
 | |
|                                 disks.append({
 | |
|                                     "drive": drive,
 | |
|                                     "disk_type": "Fixed Disk",
 | |
|                                     "usage": usage,
 | |
|                                     "total": total,
 | |
|                                     "free": free_bytes,
 | |
|                                     "used": used,
 | |
|                                 })
 | |
|                             except Exception:
 | |
|                                 pass
 | |
|                 except FileNotFoundError:
 | |
|                     ps_cmd = (
 | |
|                         "Get-PSDrive -PSProvider FileSystem | "
 | |
|                         "Select-Object Name,Free,Used,Capacity,Root | ConvertTo-Json"
 | |
|                     )
 | |
|                     out = subprocess.run(
 | |
|                         ["powershell", "-NoProfile", "-Command", ps_cmd],
 | |
|                         capture_output=True,
 | |
|                         text=True,
 | |
|                         timeout=60,
 | |
|                     )
 | |
|                     data = json.loads(out.stdout or "[]")
 | |
|                     if isinstance(data, dict):
 | |
|                         data = [data]
 | |
|                     for d in data:
 | |
|                         total = d.get("Capacity") or 0
 | |
|                         used = d.get("Used") or 0
 | |
|                         free_bytes = d.get("Free") or max(total - used, 0)
 | |
|                         usage = (used / total * 100) if total else 0
 | |
|                         drive = d.get("Root") or f"{d.get('Name','')}:"
 | |
|                         disks.append({
 | |
|                             "drive": drive,
 | |
|                             "disk_type": "Fixed Disk",
 | |
|                             "usage": usage,
 | |
|                             "total": total,
 | |
|                             "free": free_bytes,
 | |
|                             "used": used,
 | |
|                         })
 | |
|         else:
 | |
|             out = subprocess.run(
 | |
|                 ["df", "-kP"], capture_output=True, text=True, timeout=60
 | |
|             )
 | |
|             lines = out.stdout.strip().splitlines()[1:]
 | |
|             for line in lines:
 | |
|                 parts = line.split()
 | |
|                 if len(parts) >= 6:
 | |
|                     total = int(parts[1]) * 1024
 | |
|                     used = int(parts[2]) * 1024
 | |
|                     free_bytes = int(parts[3]) * 1024
 | |
|                     usage = float(parts[4].rstrip("%"))
 | |
|                     disks.append({
 | |
|                         "drive": parts[5],
 | |
|                         "disk_type": "Fixed Disk",
 | |
|                         "usage": usage,
 | |
|                         "total": total,
 | |
|                         "free": free_bytes,
 | |
|                         "used": used,
 | |
|                     })
 | |
|     except Exception as e:
 | |
|         print(f"[WARN] collect_storage failed: {e}")
 | |
|     return disks
 | |
| 
 | |
| def collect_network():
 | |
|     # migrated to role_DeviceInventory
 | |
|     adapters = []
 | |
|     plat = platform.system().lower()
 | |
|     try:
 | |
|         if psutil:
 | |
|             for name, addrs in psutil.net_if_addrs().items():
 | |
|                 ips = [a.address for a in addrs if getattr(a, "family", None) == socket.AF_INET]
 | |
|                 mac = next((a.address for a in addrs if getattr(a, "family", None) == getattr(psutil, "AF_LINK", object)), "unknown")
 | |
|                 adapters.append({"adapter": name, "ips": ips, "mac": mac})
 | |
|         elif plat == "windows":
 | |
|             ps_cmd = (
 | |
|                 "Get-NetIPConfiguration | "
 | |
|                 "Select-Object InterfaceAlias,@{Name='IPv4';Expression={$_.IPv4Address.IPAddress}},"
 | |
|                 "@{Name='MAC';Expression={$_.NetAdapter.MacAddress}} | ConvertTo-Json"
 | |
|             )
 | |
|             out = subprocess.run(
 | |
|                 ["powershell", "-NoProfile", "-Command", ps_cmd],
 | |
|                 capture_output=True,
 | |
|                 text=True,
 | |
|                 timeout=60,
 | |
|             )
 | |
|             data = json.loads(out.stdout or "[]")
 | |
|             if isinstance(data, dict):
 | |
|                 data = [data]
 | |
|             for a in data:
 | |
|                 ip = a.get("IPv4")
 | |
|                 adapters.append({
 | |
|                     "adapter": a.get("InterfaceAlias", "unknown"),
 | |
|                     "ips": [ip] if ip else [],
 | |
|                     "mac": a.get("MAC", "unknown"),
 | |
|                 })
 | |
|         else:
 | |
|             out = subprocess.run(
 | |
|                 ["ip", "-o", "-4", "addr", "show"],
 | |
|                 capture_output=True,
 | |
|                 text=True,
 | |
|                 timeout=60,
 | |
|             )
 | |
|             for line in out.stdout.splitlines():
 | |
|                 parts = line.split()
 | |
|                 if len(parts) >= 4:
 | |
|                     name = parts[1]
 | |
|                     ip = parts[3].split("/")[0]
 | |
|                     adapters.append({"adapter": name, "ips": [ip], "mac": "unknown"})
 | |
|     except Exception as e:
 | |
|         print(f"[WARN] collect_network failed: {e}")
 | |
|     return adapters
 | |
| 
 | |
| async def send_agent_details():
 | |
|     """Collect detailed agent data and send to server periodically."""
 | |
|     while True:
 | |
|         try:
 | |
|             details = {
 | |
|                 "summary": collect_summary(),
 | |
|                 "software": collect_software(),
 | |
|                 "memory": collect_memory(),
 | |
|                 "storage": collect_storage(),
 | |
|                 "network": collect_network(),
 | |
|             }
 | |
|             payload = {
 | |
|                 "agent_id": AGENT_ID,
 | |
|                 "hostname": details.get("summary", {}).get("hostname", socket.gethostname()),
 | |
|                 "details": details,
 | |
|             }
 | |
|             client = http_client()
 | |
|             await client.async_post_json("/api/agent/details", payload, require_auth=True)
 | |
|             _log_agent('Posted agent details to server.')
 | |
|         except Exception as e:
 | |
|             print(f"[WARN] Failed to send agent details: {e}")
 | |
|             _log_agent(f'Failed to send agent details: {e}', fname='agent.error.log')
 | |
|         await asyncio.sleep(300)
 | |
| 
 | |
| async def send_agent_details_once():
 | |
|     try:
 | |
|         details = {
 | |
|             "summary": collect_summary(),
 | |
|             "software": collect_software(),
 | |
|             "memory": collect_memory(),
 | |
|             "storage": collect_storage(),
 | |
|             "network": collect_network(),
 | |
|         }
 | |
|         payload = {
 | |
|             "agent_id": AGENT_ID,
 | |
|             "hostname": details.get("summary", {}).get("hostname", socket.gethostname()),
 | |
|             "details": details,
 | |
|         }
 | |
|         client = http_client()
 | |
|         await client.async_post_json("/api/agent/details", payload, require_auth=True)
 | |
|         _log_agent('Posted agent details (once) to server.')
 | |
|     except Exception as e:
 | |
|         _log_agent(f'Failed to post agent details once: {e}', fname='agent.error.log')
 | |
| 
 | |
| @sio.event
 | |
| async def connect():
 | |
|     print(f"[INFO] Successfully Connected to Borealis Server!")
 | |
|     _log_agent('Connected to server.')
 | |
|     await sio.emit('connect_agent', {"agent_id": AGENT_ID, "service_mode": SERVICE_MODE})
 | |
| 
 | |
|     # Send an immediate heartbeat via authenticated REST call.
 | |
|     try:
 | |
|         client = http_client()
 | |
|         client.ensure_authenticated()
 | |
|         payload = {
 | |
|             "guid": client.guid or _read_agent_guid_from_disk(),
 | |
|             "hostname": socket.gethostname(),
 | |
|             "inventory": {},
 | |
|             "metrics": _collect_heartbeat_metrics(),
 | |
|         }
 | |
|         await client.async_post_json("/api/agent/heartbeat", payload, require_auth=True)
 | |
|     except Exception as exc:
 | |
|         _log_agent(f'Initial REST heartbeat failed: {exc}', fname='agent.error.log')
 | |
| 
 | |
|     await sio.emit('request_config', {"agent_id": AGENT_ID})
 | |
|     # Inventory details posting is managed by the DeviceAudit role (SYSTEM). No one-shot post here.
 | |
|     # Fire-and-forget service account check-in so server can provision WinRM credentials
 | |
|     try:
 | |
|         async def _svc_checkin_once():
 | |
|             try:
 | |
|                 payload = {"agent_id": AGENT_ID, "hostname": socket.gethostname(), "username": ".\\svcBorealis"}
 | |
|                 client = http_client()
 | |
|                 response = await client.async_post_json("/api/agent/checkin", payload, require_auth=True)
 | |
|                 if isinstance(response, dict):
 | |
|                     guid_value = (response.get('agent_guid') or '').strip()
 | |
|                     if guid_value:
 | |
|                         _persist_agent_guid_local(guid_value)
 | |
|                         _update_agent_id_for_guid(guid_value)
 | |
|             except Exception:
 | |
|                 pass
 | |
|         asyncio.create_task(_svc_checkin_once())
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
| @sio.event
 | |
| async def disconnect():
 | |
|     print("[WebSocket] Disconnected from Borealis server.")
 | |
|     await stop_all_roles()
 | |
|     CONFIG.data['regions'].clear()
 | |
|     CONFIG._write()
 | |
| 
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| # CORE SECTION: AGENT CONFIG MANAGEMENT / WINDOW MANAGEMENT
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| @sio.on('agent_config')
 | |
| async def on_agent_config(cfg):
 | |
|     print("[DEBUG] agent_config event received.")
 | |
|     roles = cfg.get('roles', [])
 | |
|     if not roles:
 | |
|         print("[CONFIG] Config Reset by Borealis Server Operator - Awaiting New Config...")
 | |
|         await stop_all_roles()
 | |
|         return
 | |
| 
 | |
|     print(f"[CONFIG] Received New Agent Config with {len(roles)} Role(s).")
 | |
| 
 | |
|     # Forward screenshot config to service helper (interval only)
 | |
|     try:
 | |
|         for role_cfg in roles:
 | |
|             if role_cfg.get('role') == 'screenshot':
 | |
|                 interval_ms = int(role_cfg.get('interval', 1000))
 | |
|                 send_service_control({'type': 'screenshot_config', 'interval_ms': interval_ms})
 | |
|                 break
 | |
|     except Exception:
 | |
|         pass
 | |
| 
 | |
|     try:
 | |
|         if ROLE_MANAGER is not None:
 | |
|             ROLE_MANAGER.on_config(roles)
 | |
|     except Exception as e:
 | |
|         print(f"[WARN] role manager (interactive) apply config failed: {e}")
 | |
|     try:
 | |
|         if ROLE_MANAGER_SYS is not None:
 | |
|             ROLE_MANAGER_SYS.on_config(roles)
 | |
|     except Exception as e:
 | |
|         print(f"[WARN] role manager (system) apply config failed: {e}")
 | |
| 
 | |
| ## Script execution and list windows handlers are registered by roles
 | |
| 
 | |
| # ---------------- Config Watcher ----------------
 | |
| async def config_watcher():
 | |
|     while True:
 | |
|         CONFIG.watch()
 | |
|         await asyncio.sleep(CONFIG.data.get('config_file_watcher_interval',2))
 | |
| 
 | |
| # ---------------- Persistent Idle Task ----------------
 | |
| async def idle_task():
 | |
|     try:
 | |
|         while True:
 | |
|             await asyncio.sleep(60)
 | |
|     except asyncio.CancelledError:
 | |
|         print("[FATAL] Idle task was cancelled!")
 | |
|     except Exception as e:
 | |
|         print(f"[FATAL] Idle task crashed: {e}")
 | |
|         traceback.print_exc()
 | |
| 
 | |
| # ---------------- Quick Job Helpers (System/Local) ----------------
 | |
| def _project_root_for_temp():
 | |
|     try:
 | |
|         return os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..'))
 | |
|     except Exception:
 | |
|         return os.path.abspath(os.path.dirname(__file__))
 | |
| 
 | |
| def _run_powershell_script_content_local(content: str):
 | |
|     try:
 | |
|         temp_dir = os.path.join(_project_root_for_temp(), "Temp")
 | |
|         os.makedirs(temp_dir, exist_ok=True)
 | |
|         import tempfile as _tf
 | |
|         fd, path = _tf.mkstemp(prefix="sj_", suffix=".ps1", dir=temp_dir, text=True)
 | |
|         with os.fdopen(fd, 'w', encoding='utf-8', newline='\n') as fh:
 | |
|             fh.write(content or "")
 | |
|         ps = os.path.expandvars(r"%SystemRoot%\\System32\\WindowsPowerShell\\v1.0\\powershell.exe")
 | |
|         if not os.path.isfile(ps):
 | |
|             ps = "powershell.exe"
 | |
|         flags = 0x08000000 if os.name == 'nt' else 0
 | |
|         proc = subprocess.run([ps, "-ExecutionPolicy", "Bypass", "-NoProfile", "-File", path], capture_output=True, text=True, timeout=60*60, creationflags=flags)
 | |
|         return proc.returncode, proc.stdout or "", proc.stderr or ""
 | |
|     except Exception as e:
 | |
|         return -1, "", str(e)
 | |
|     finally:
 | |
|         try:
 | |
|             if 'path' in locals() and os.path.isfile(path):
 | |
|                 os.remove(path)
 | |
|         except Exception:
 | |
|             pass
 | |
| 
 | |
| def _run_powershell_via_system_task(content: str):
 | |
|     ps_exe = os.path.expandvars(r"%SystemRoot%\\System32\\WindowsPowerShell\\v1.0\\powershell.exe")
 | |
|     if not os.path.isfile(ps_exe):
 | |
|         ps_exe = 'powershell.exe'
 | |
|     script_path = None
 | |
|     out_path = None
 | |
|     try:
 | |
|         temp_root = os.path.join(_project_root_for_temp(), 'Temp')
 | |
|         os.makedirs(temp_root, exist_ok=True)
 | |
|         import tempfile as _tf
 | |
|         fd, script_path = _tf.mkstemp(prefix='sys_task_', suffix='.ps1', dir=temp_root, text=True)
 | |
|         with os.fdopen(fd, 'w', encoding='utf-8', newline='\n') as f:
 | |
|             f.write(content or '')
 | |
|         try:
 | |
|             log_dir = os.path.join(_project_root_for_temp(), 'Logs', 'Agent')
 | |
|             os.makedirs(log_dir, exist_ok=True)
 | |
|             debug_copy = os.path.join(log_dir, 'system_last.ps1')
 | |
|             with open(debug_copy, 'w', encoding='utf-8', newline='\n') as df:
 | |
|                 df.write(content or '')
 | |
|         except Exception:
 | |
|             pass
 | |
|         out_path = os.path.join(temp_root, f'out_{uuid.uuid4().hex}.txt')
 | |
|         task_name = f"Borealis Agent - Task - {uuid.uuid4().hex} @ SYSTEM"
 | |
|         task_ps = f"""
 | |
| $ErrorActionPreference='Continue'
 | |
| $task = "{task_name}"
 | |
| $ps   = "{ps_exe}"
 | |
| $scr  = "{script_path}"
 | |
| $out  = "{out_path}"
 | |
| try {{ Unregister-ScheduledTask -TaskName $task -Confirm:$false -ErrorAction SilentlyContinue }} catch {{}}
 | |
| $action   = New-ScheduledTaskAction -Execute $ps -Argument ('-NoProfile -ExecutionPolicy Bypass -WindowStyle Hidden -File "' + $scr + '" *> "' + $out + '"') -WorkingDirectory (Split-Path -Parent $scr)
 | |
| $settings = New-ScheduledTaskSettingsSet -DeleteExpiredTaskAfter (New-TimeSpan -Minutes 5) -AllowStartIfOnBatteries -DontStopIfGoingOnBatteries
 | |
| $principal= New-ScheduledTaskPrincipal -UserId 'SYSTEM' -LogonType ServiceAccount -RunLevel Highest
 | |
| Register-ScheduledTask -TaskName $task -Action $action -Settings $settings -Principal $principal -Force | Out-Null
 | |
| Start-ScheduledTask -TaskName $task | Out-Null
 | |
| Start-Sleep -Seconds 2
 | |
| Get-ScheduledTask -TaskName $task | Out-Null
 | |
| """
 | |
|         proc = subprocess.run([ps_exe, '-NoProfile', '-ExecutionPolicy', 'Bypass', '-Command', task_ps], capture_output=True, text=True)
 | |
|         if proc.returncode != 0:
 | |
|             return -999, '', (proc.stderr or proc.stdout or 'scheduled task creation failed')
 | |
|         deadline = time.time() + 60
 | |
|         out_data = ''
 | |
|         while time.time() < deadline:
 | |
|             try:
 | |
|                 if os.path.isfile(out_path) and os.path.getsize(out_path) > 0:
 | |
|                     with open(out_path, 'r', encoding='utf-8', errors='replace') as f:
 | |
|                         out_data = f.read()
 | |
|                     break
 | |
|             except Exception:
 | |
|                 pass
 | |
|             time.sleep(1)
 | |
|         # Cleanup best-effort
 | |
|         try:
 | |
|             cleanup_ps = f"try {{ Unregister-ScheduledTask -TaskName '{task_name}' -Confirm:$false }} catch {{}}"
 | |
|             subprocess.run([ps_exe, '-NoProfile', '-ExecutionPolicy', 'Bypass', '-Command', cleanup_ps], capture_output=True, text=True)
 | |
|         except Exception:
 | |
|             pass
 | |
|         try:
 | |
|             if script_path and os.path.isfile(script_path):
 | |
|                 os.remove(script_path)
 | |
|         except Exception:
 | |
|             pass
 | |
|         try:
 | |
|             if out_path and os.path.isfile(out_path):
 | |
|                 os.remove(out_path)
 | |
|         except Exception:
 | |
|             pass
 | |
|         return 0, out_data or '', ''
 | |
|     except Exception as e:
 | |
|         return -999, '', str(e)
 | |
| 
 | |
| async def _run_powershell_via_user_task(content: str):
 | |
|     ps = None
 | |
|     if IS_WINDOWS:
 | |
|         ps = os.path.expandvars(r"%SystemRoot%\System32\WindowsPowerShell\v1.0\powershell.exe")
 | |
|         if not os.path.isfile(ps):
 | |
|             ps = "powershell.exe"
 | |
|     else:
 | |
|         return -999, '', 'Windows only'
 | |
|     path = None
 | |
|     out_path = None
 | |
|     try:
 | |
|         temp_dir = os.path.join(os.path.dirname(__file__), '..', '..', 'Temp')
 | |
|         temp_dir = os.path.abspath(temp_dir)
 | |
|         os.makedirs(temp_dir, exist_ok=True)
 | |
|         fd, path = tempfile.mkstemp(prefix='usr_task_', suffix='.ps1', dir=temp_dir, text=True)
 | |
|         with os.fdopen(fd, 'w', encoding='utf-8', newline='\n') as f:
 | |
|             f.write(content or '')
 | |
|         out_path = os.path.join(temp_dir, f'out_{uuid.uuid4().hex}.txt')
 | |
|         name = f"Borealis Agent - Task - {uuid.uuid4().hex} @ CurrentUser"
 | |
|         task_ps = f"""
 | |
| $ErrorActionPreference='Continue'
 | |
| $task = "{name}"
 | |
| $ps   = "{ps}"
 | |
| $scr  = "{path}"
 | |
| $out  = "{out_path}"
 | |
| try {{ Unregister-ScheduledTask -TaskName $task -Confirm:$false -ErrorAction SilentlyContinue }} catch {{}}
 | |
| $action   = New-ScheduledTaskAction -Execute $ps -Argument ('-NoProfile -ExecutionPolicy Bypass -WindowStyle Hidden -File "' + $scr + '" *> "' + $out + '"')
 | |
| $settings = New-ScheduledTaskSettingsSet -DeleteExpiredTaskAfter (New-TimeSpan -Minutes 5) -AllowStartIfOnBatteries -DontStopIfGoingOnBatteries
 | |
| $principal= New-ScheduledTaskPrincipal -UserId ([System.Security.Principal.WindowsIdentity]::GetCurrent().Name) -LogonType Interactive -RunLevel Limited
 | |
| Register-ScheduledTask -TaskName $task -Action $action -Settings $settings -Principal $principal -Force | Out-Null
 | |
| Start-ScheduledTask -TaskName $task | Out-Null
 | |
| Start-Sleep -Seconds 2
 | |
| Get-ScheduledTask -TaskName $task | Out-Null
 | |
| """
 | |
|         proc = await asyncio.create_subprocess_exec(ps, '-NoProfile', '-ExecutionPolicy', 'Bypass', '-Command', task_ps, stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE)
 | |
|         await proc.communicate()
 | |
|         if proc.returncode != 0:
 | |
|             return -999, '', 'failed to create user task'
 | |
|         # Wait for output
 | |
|         deadline = time.time() + 60
 | |
|         out_data = ''
 | |
|         while time.time() < deadline:
 | |
|             try:
 | |
|                 if os.path.isfile(out_path) and os.path.getsize(out_path) > 0:
 | |
|                     with open(out_path, 'r', encoding='utf-8', errors='replace') as f:
 | |
|                         out_data = f.read()
 | |
|                     break
 | |
|             except Exception:
 | |
|                 pass
 | |
|             await asyncio.sleep(1)
 | |
|         cleanup = f"try {{ Unregister-ScheduledTask -TaskName '{name}' -Confirm:$false }} catch {{}}"
 | |
|         await asyncio.create_subprocess_exec(ps, '-NoProfile', '-ExecutionPolicy', 'Bypass', '-Command', cleanup)
 | |
|         return 0, out_data or '', ''
 | |
|     except Exception as e:
 | |
|         return -999, '', str(e)
 | |
|     finally:
 | |
|         # Best-effort cleanup of temp script and output files
 | |
|         try:
 | |
|             if path and os.path.isfile(path):
 | |
|                 os.remove(path)
 | |
|         except Exception:
 | |
|             pass
 | |
|         try:
 | |
|             if out_path and os.path.isfile(out_path):
 | |
|                 os.remove(out_path)
 | |
|         except Exception:
 | |
|             pass
 | |
| 
 | |
| # ---------------- Dummy Qt Widget to Prevent Exit ----------------
 | |
| if not SYSTEM_SERVICE_MODE:
 | |
|     class PersistentWindow(QtWidgets.QWidget):
 | |
|         def __init__(self):
 | |
|             super().__init__()
 | |
|             self.setWindowTitle("KeepAlive")
 | |
|             self.setGeometry(-1000,-1000,1,1)
 | |
|             self.setAttribute(QtCore.Qt.WA_DontShowOnScreen)
 | |
|             self.hide()
 | |
| 
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| # MAIN & EVENT LOOP
 | |
| # //////////////////////////////////////////////////////////////////////////
 | |
| async def connect_loop():
 | |
|     retry = 5
 | |
|     client = http_client()
 | |
|     while True:
 | |
|         try:
 | |
|             client.ensure_authenticated()
 | |
|             client.configure_socketio(sio)
 | |
|             url = client.websocket_base_url()
 | |
|             print(f"[INFO] Connecting Agent to {url}...")
 | |
|             _log_agent(f'Connecting to {url}...')
 | |
|             await sio.connect(
 | |
|                 url,
 | |
|                 transports=['websocket'],
 | |
|                 headers=client.auth_headers(),
 | |
|             )
 | |
|             break
 | |
|         except Exception as e:
 | |
|             print(f"[WebSocket] Server unavailable: {e}. Retrying in {retry}s...")
 | |
|             _log_agent(f'Server unavailable: {e}', fname='agent.error.log')
 | |
|             await asyncio.sleep(retry)
 | |
| 
 | |
| if __name__=='__main__':
 | |
|     try:
 | |
|         _bootstrap_log('enter __main__')
 | |
|     except Exception:
 | |
|         pass
 | |
|     # Ansible logs are rotated daily on write; no explicit clearing on startup
 | |
|     if SYSTEM_SERVICE_MODE:
 | |
|         loop = asyncio.new_event_loop(); asyncio.set_event_loop(loop)
 | |
|     else:
 | |
|         app=QtWidgets.QApplication(sys.argv)
 | |
|         loop=QEventLoop(app); asyncio.set_event_loop(loop)
 | |
|     AGENT_LOOP = loop
 | |
|     try:
 | |
|         if not SYSTEM_SERVICE_MODE:
 | |
|             start_agent_bridge_pipe(loop)
 | |
|     except Exception:
 | |
|         pass
 | |
|     if not SYSTEM_SERVICE_MODE:
 | |
|         dummy_window=PersistentWindow(); dummy_window.show()
 | |
|     # Initialize roles context for role tasks
 | |
|     # Initialize role manager and hot-load roles from Roles/
 | |
|     client = http_client()
 | |
|     try:
 | |
|         client.ensure_authenticated()
 | |
|     except Exception as exc:
 | |
|         _log_agent(f'Authentication bootstrap failed: {exc}', fname='agent.error.log')
 | |
|         print(f"[WARN] Authentication bootstrap failed: {exc}")
 | |
|     try:
 | |
|         base_hooks = {'send_service_control': send_service_control, 'get_server_url': get_server_url}
 | |
|         if not SYSTEM_SERVICE_MODE:
 | |
|             # Load interactive-context roles (tray/UI, current-user execution, screenshot, etc.)
 | |
|             hooks_interactive = {**base_hooks, 'service_mode': 'currentuser'}
 | |
|             ROLE_MANAGER = RoleManager(
 | |
|                 base_dir=os.path.dirname(__file__),
 | |
|                 context='interactive',
 | |
|                 sio=sio,
 | |
|                 agent_id=AGENT_ID,
 | |
|                 config=CONFIG,
 | |
|                 loop=loop,
 | |
|                 hooks=hooks_interactive,
 | |
|             )
 | |
|             ROLE_MANAGER.load()
 | |
|         # Load system roles only when running in SYSTEM service mode
 | |
|         ROLE_MANAGER_SYS = None
 | |
|         if SYSTEM_SERVICE_MODE:
 | |
|             hooks_system = {**base_hooks, 'service_mode': 'system'}
 | |
|             ROLE_MANAGER_SYS = RoleManager(
 | |
|                 base_dir=os.path.dirname(__file__),
 | |
|                 context='system',
 | |
|                 sio=sio,
 | |
|                 agent_id=AGENT_ID,
 | |
|                 config=CONFIG,
 | |
|                 loop=loop,
 | |
|                 hooks=hooks_system,
 | |
|             )
 | |
|             ROLE_MANAGER_SYS.load()
 | |
|     except Exception as e:
 | |
|         try:
 | |
|             _bootstrap_log(f'role load init failed: {e}')
 | |
|         except Exception:
 | |
|             pass
 | |
|     try:
 | |
|         background_tasks.append(loop.create_task(config_watcher()))
 | |
|         background_tasks.append(loop.create_task(connect_loop()))
 | |
|         background_tasks.append(loop.create_task(idle_task()))
 | |
|         # Start periodic heartbeats
 | |
|         background_tasks.append(loop.create_task(send_heartbeat()))
 | |
|         background_tasks.append(loop.create_task(poll_script_requests()))
 | |
|         # Inventory upload is handled by the DeviceAudit role running in SYSTEM context.
 | |
|         # Do not schedule the legacy agent-level details poster to avoid duplicates.
 | |
|         
 | |
|         # Register unified Quick Job handler last to avoid role override issues
 | |
|         @sio.on('quick_job_run')
 | |
|         async def _quick_job_dispatch(payload):
 | |
|             try:
 | |
|                 _log_agent(f"quick_job_run received: mode={payload.get('run_mode')} type={payload.get('script_type')} job_id={payload.get('job_id')}")
 | |
|                 import socket as _s
 | |
|                 hostname = _s.gethostname()
 | |
|                 target = (payload.get('target_hostname') or '').strip().lower()
 | |
|                 if target and target not in ('unknown', '*', '(unknown)') and target != hostname.lower():
 | |
|                     return
 | |
|                 job_id = payload.get('job_id')
 | |
|                 script_type = (payload.get('script_type') or '').lower()
 | |
|                 content = _decode_script_payload(payload.get('script_content'), payload.get('script_encoding'))
 | |
|                 run_mode = (payload.get('run_mode') or 'current_user').lower()
 | |
|                 if script_type != 'powershell':
 | |
|                     await sio.emit('quick_job_result', { 'job_id': job_id, 'status': 'Failed', 'stdout': '', 'stderr': f"Unsupported type: {script_type}" })
 | |
|                     return
 | |
|                 rc = -1; out = ''; err = ''
 | |
|                 if run_mode == 'system':
 | |
|                     if not SYSTEM_SERVICE_MODE:
 | |
|                         # Let the SYSTEM service handle these exclusively
 | |
|                         return
 | |
|                     try:
 | |
|                         # Save last SYSTEM script for debugging
 | |
|                         dbg_dir = os.path.join(_find_project_root(), 'Logs', 'Agent')
 | |
|                         os.makedirs(dbg_dir, exist_ok=True)
 | |
|                         with open(os.path.join(dbg_dir, 'system_last.ps1'), 'w', encoding='utf-8', newline='\n') as df:
 | |
|                             df.write(content or '')
 | |
|                     except Exception:
 | |
|                         pass
 | |
|                     rc, out, err = _run_powershell_script_content_local(content)
 | |
|                     if rc == -999:
 | |
|                         # Fallback: attempt scheduled task (should not be needed in service mode)
 | |
|                         rc, out, err = _run_powershell_via_system_task(content)
 | |
|                 elif run_mode == 'admin':
 | |
|                     rc, out, err = -1, '', 'Admin credentialed runs are disabled; use SYSTEM or Current User.'
 | |
|                 else:
 | |
|                     rc, out, err = await _run_powershell_via_user_task(content)
 | |
|                     if rc == -999:
 | |
|                         # Fallback to plain local run
 | |
|                         rc, out, err = _run_powershell_script_content_local(content)
 | |
|                 status = 'Success' if rc == 0 else 'Failed'
 | |
|                 await sio.emit('quick_job_result', {
 | |
|                     'job_id': job_id,
 | |
|                     'status': status,
 | |
|                     'stdout': out or '',
 | |
|                     'stderr': err or '',
 | |
|                 })
 | |
|                 _log_agent(f"quick_job_result sent: job_id={job_id} status={status}")
 | |
|             except Exception as e:
 | |
|                 try:
 | |
|                     await sio.emit('quick_job_result', {
 | |
|                         'job_id': payload.get('job_id') if isinstance(payload, dict) else None,
 | |
|                         'status': 'Failed',
 | |
|                         'stdout': '',
 | |
|                         'stderr': str(e),
 | |
|                     })
 | |
|                 except Exception:
 | |
|                     pass
 | |
|                 _log_agent(f"quick_job_run handler error: {e}", fname='agent.error.log')
 | |
|         _bootstrap_log('starting event loop...')
 | |
|         loop.run_forever()
 | |
|     except Exception as e:
 | |
|         try:
 | |
|             _bootstrap_log(f'FATAL: event loop crashed: {e}')
 | |
|         except Exception:
 | |
|             pass
 | |
|         print(f"[FATAL] Event loop crashed: {e}")
 | |
|         traceback.print_exc()
 | |
|     finally:
 | |
|         try:
 | |
|             _bootstrap_log('Agent exited unexpectedly.')
 | |
|         except Exception:
 | |
|             pass
 | |
|         print("[FATAL] Agent exited unexpectedly.")
 | |
| 
 | |
| # (moved earlier so async tasks can log immediately)
 | |
| # ---- Ansible log helpers (Agent) ----
 | |
| def _ansible_log_agent(msg: str):
 | |
|     try:
 | |
|         d = _agent_logs_root()
 | |
|         os.makedirs(d, exist_ok=True)
 | |
|         path = os.path.join(d, 'ansible.log')
 | |
|         _rotate_daily(path)
 | |
|         ts = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')
 | |
|         with open(path, 'a', encoding='utf-8') as fh:
 | |
|             fh.write(f'[{ts}] {msg}\n')
 | |
|     except Exception:
 | |
|         pass
 |