feat(api): MCP, OpenAPI & Dynamic Introspection

New Features:
- API endpoints now support comprehensive input validation with detailed error responses via Pydantic models.
- OpenAPI specification endpoint (/openapi.json) and interactive Swagger UI documentation (/docs) now available for API discovery.
- Enhanced MCP session lifecycle management with create, retrieve, and delete operations.
- Network diagnostic tools: traceroute, nslookup, NMAP scanning, and network topology viewing exposed via API.
- Device search, filtering by status (including 'offline'), and bulk operations (copy, delete, update).
- Wake-on-LAN functionality for remote device management.
- Added dynamic tool disablement and status reporting.

Bug Fixes:
- Fixed get_tools_status in registry to correctly return boolean values instead of None for enabled tools.
- Improved error handling for invalid API inputs with standardized validation responses.
- Fixed OPTIONS request handling for cross-origin requests.

Refactoring:
- Significant refactoring of api_server_start.py to use decorator-based validation (@validate_request).
This commit is contained in:
Adam Outler
2026-01-18 18:16:18 +00:00
parent cea3369b5e
commit ecea1d1fbd
46 changed files with 5195 additions and 1053 deletions

View File

@@ -9,12 +9,8 @@ import os
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
import sys
# Add test directory to path
sys.path.insert(0, os.path.dirname(__file__))
from test_helpers import BASE_URL # noqa: E402 [flake8 lint suppression]
from .test_helpers import BASE_URL, wait_for_page_load
def test_settings_page_loads(driver):
@@ -23,14 +19,14 @@ def test_settings_page_loads(driver):
WebDriverWait(driver, 10).until(
EC.presence_of_element_located((By.TAG_NAME, "body"))
)
time.sleep(2)
wait_for_page_load(driver, timeout=10)
assert "setting" in driver.page_source.lower(), "Page should contain settings content"
def test_settings_groups_present(driver):
"""Test: Settings groups/sections are rendered"""
driver.get(f"{BASE_URL}/settings.php")
time.sleep(2)
wait_for_page_load(driver, timeout=10)
groups = driver.find_elements(By.CSS_SELECTOR, ".settings-group, .panel, .card, fieldset")
assert len(groups) > 0, "Settings groups should be present"
@@ -38,7 +34,7 @@ def test_settings_groups_present(driver):
def test_settings_inputs_present(driver):
"""Test: Settings input fields are rendered"""
driver.get(f"{BASE_URL}/settings.php")
time.sleep(2)
wait_for_page_load(driver, timeout=10)
inputs = driver.find_elements(By.CSS_SELECTOR, "input, select, textarea")
assert len(inputs) > 0, "Settings input fields should be present"
@@ -46,7 +42,7 @@ def test_settings_inputs_present(driver):
def test_save_button_present(driver):
"""Test: Save button is visible"""
driver.get(f"{BASE_URL}/settings.php")
time.sleep(2)
wait_for_page_load(driver, timeout=10)
save_btn = driver.find_elements(By.CSS_SELECTOR, "button[type='submit'], button#save, .btn-save")
assert len(save_btn) > 0, "Save button should be present"
@@ -63,7 +59,7 @@ def test_save_settings_with_form_submission(driver):
6. Verifies the config file was updated
"""
driver.get(f"{BASE_URL}/settings.php")
time.sleep(3)
wait_for_page_load(driver, timeout=10)
# Wait for the save button to be present and clickable
save_btn = WebDriverWait(driver, 10).until(
@@ -161,7 +157,7 @@ def test_save_settings_no_loss_of_data(driver):
4. Check API endpoint that the setting is updated correctly
"""
driver.get(f"{BASE_URL}/settings.php")
time.sleep(3)
wait_for_page_load(driver, timeout=10)
# Find the PLUGINS_KEEP_HIST input field
plugins_keep_hist_input = None
@@ -181,12 +177,12 @@ def test_save_settings_no_loss_of_data(driver):
new_value = "333"
plugins_keep_hist_input.clear()
plugins_keep_hist_input.send_keys(new_value)
time.sleep(1)
wait_for_page_load(driver, timeout=10)
# Click save
save_btn = driver.find_element(By.CSS_SELECTOR, "button#save")
driver.execute_script("arguments[0].click();", save_btn)
time.sleep(3)
wait_for_page_load(driver, timeout=10)
# Check for errors after save
error_elements = driver.find_elements(By.CSS_SELECTOR, ".alert-danger, .error-message, .callout-danger")