Implement complete database schema and Flask application structure for SneakyScan web interface. This establishes the foundation for web-based scan management, scheduling, and visualization. Database & ORM: - Add 11 SQLAlchemy models for comprehensive scan data storage (Scan, ScanSite, ScanIP, ScanPort, ScanService, ScanCertificate, ScanTLSVersion, Schedule, Alert, AlertRule, Setting) - Configure Alembic migrations system with initial schema migration - Add init_db.py script for database initialization and password setup - Support both migration-based and direct table creation Settings System: - Implement SettingsManager with automatic encryption for sensitive values - Add Fernet encryption for SMTP passwords and API tokens - Implement PasswordManager with bcrypt password hashing (work factor 12) - Initialize default settings for SMTP, authentication, and retention Flask Application: - Create Flask app factory pattern with scoped session management - Add 4 API blueprints: scans, schedules, alerts, settings - Implement functional Settings API (GET/PUT/DELETE endpoints) - Add CORS support, error handlers, and request/response logging - Configure development and production logging to file and console Docker & Deployment: - Update Dockerfile to install Flask dependencies - Add docker-compose-web.yml for web application deployment - Configure volume mounts for database, output, and logs persistence - Expose port 5000 for Flask web server Testing & Validation: - Add validate_phase1.py script to verify all deliverables - Validate directory structure, Python syntax, models, and endpoints - All validation checks passing Documentation: - Add PHASE1_COMPLETE.md with comprehensive Phase 1 summary - Update ROADMAP.md with Phase 1 completion status - Update .gitignore to exclude database files and documentation Files changed: 21 files - New: web/ directory with complete Flask app structure - New: migrations/ with Alembic configuration - New: requirements-web.txt with Flask dependencies - Modified: Dockerfile, ROADMAP.md, .gitignore
268 lines
6.6 KiB
Python
268 lines
6.6 KiB
Python
"""
|
|
Settings API blueprint.
|
|
|
|
Handles endpoints for managing application settings including SMTP configuration,
|
|
authentication, and system preferences.
|
|
"""
|
|
|
|
from flask import Blueprint, current_app, jsonify, request
|
|
|
|
from web.utils.settings import PasswordManager, SettingsManager
|
|
|
|
bp = Blueprint('settings', __name__)
|
|
|
|
|
|
def get_settings_manager():
|
|
"""Get SettingsManager instance with current DB session."""
|
|
return SettingsManager(current_app.db_session)
|
|
|
|
|
|
@bp.route('', methods=['GET'])
|
|
def get_settings():
|
|
"""
|
|
Get all settings (sanitized - encrypted values masked).
|
|
|
|
Returns:
|
|
JSON response with all settings
|
|
"""
|
|
try:
|
|
settings_manager = get_settings_manager()
|
|
settings = settings_manager.get_all(decrypt=False, sanitize=True)
|
|
|
|
return jsonify({
|
|
'status': 'success',
|
|
'settings': settings
|
|
})
|
|
except Exception as e:
|
|
current_app.logger.error(f"Failed to retrieve settings: {e}")
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'Failed to retrieve settings'
|
|
}), 500
|
|
|
|
|
|
@bp.route('', methods=['PUT'])
|
|
def update_settings():
|
|
"""
|
|
Update multiple settings at once.
|
|
|
|
Request body:
|
|
settings: Dictionary of setting key-value pairs
|
|
|
|
Returns:
|
|
JSON response with update status
|
|
"""
|
|
# TODO: Add authentication in Phase 2
|
|
data = request.get_json() or {}
|
|
settings_dict = data.get('settings', {})
|
|
|
|
if not settings_dict:
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'No settings provided'
|
|
}), 400
|
|
|
|
try:
|
|
settings_manager = get_settings_manager()
|
|
|
|
# Update each setting
|
|
for key, value in settings_dict.items():
|
|
settings_manager.set(key, value)
|
|
|
|
return jsonify({
|
|
'status': 'success',
|
|
'message': f'Updated {len(settings_dict)} settings'
|
|
})
|
|
except Exception as e:
|
|
current_app.logger.error(f"Failed to update settings: {e}")
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'Failed to update settings'
|
|
}), 500
|
|
|
|
|
|
@bp.route('/<string:key>', methods=['GET'])
|
|
def get_setting(key):
|
|
"""
|
|
Get a specific setting by key.
|
|
|
|
Args:
|
|
key: Setting key
|
|
|
|
Returns:
|
|
JSON response with setting value
|
|
"""
|
|
try:
|
|
settings_manager = get_settings_manager()
|
|
value = settings_manager.get(key)
|
|
|
|
if value is None:
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': f'Setting "{key}" not found'
|
|
}), 404
|
|
|
|
# Sanitize if encrypted key
|
|
if settings_manager._should_encrypt(key):
|
|
value = '***ENCRYPTED***'
|
|
|
|
return jsonify({
|
|
'status': 'success',
|
|
'key': key,
|
|
'value': value
|
|
})
|
|
except Exception as e:
|
|
current_app.logger.error(f"Failed to retrieve setting {key}: {e}")
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'Failed to retrieve setting'
|
|
}), 500
|
|
|
|
|
|
@bp.route('/<string:key>', methods=['PUT'])
|
|
def update_setting(key):
|
|
"""
|
|
Update a specific setting.
|
|
|
|
Args:
|
|
key: Setting key
|
|
|
|
Request body:
|
|
value: New value for the setting
|
|
|
|
Returns:
|
|
JSON response with update status
|
|
"""
|
|
# TODO: Add authentication in Phase 2
|
|
data = request.get_json() or {}
|
|
value = data.get('value')
|
|
|
|
if value is None:
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'No value provided'
|
|
}), 400
|
|
|
|
try:
|
|
settings_manager = get_settings_manager()
|
|
settings_manager.set(key, value)
|
|
|
|
return jsonify({
|
|
'status': 'success',
|
|
'message': f'Setting "{key}" updated'
|
|
})
|
|
except Exception as e:
|
|
current_app.logger.error(f"Failed to update setting {key}: {e}")
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'Failed to update setting'
|
|
}), 500
|
|
|
|
|
|
@bp.route('/<string:key>', methods=['DELETE'])
|
|
def delete_setting(key):
|
|
"""
|
|
Delete a setting.
|
|
|
|
Args:
|
|
key: Setting key to delete
|
|
|
|
Returns:
|
|
JSON response with deletion status
|
|
"""
|
|
# TODO: Add authentication in Phase 2
|
|
try:
|
|
settings_manager = get_settings_manager()
|
|
deleted = settings_manager.delete(key)
|
|
|
|
if not deleted:
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': f'Setting "{key}" not found'
|
|
}), 404
|
|
|
|
return jsonify({
|
|
'status': 'success',
|
|
'message': f'Setting "{key}" deleted'
|
|
})
|
|
except Exception as e:
|
|
current_app.logger.error(f"Failed to delete setting {key}: {e}")
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'Failed to delete setting'
|
|
}), 500
|
|
|
|
|
|
@bp.route('/password', methods=['POST'])
|
|
def set_password():
|
|
"""
|
|
Set the application password.
|
|
|
|
Request body:
|
|
password: New password
|
|
|
|
Returns:
|
|
JSON response with status
|
|
"""
|
|
# TODO: Add current password verification in Phase 2
|
|
data = request.get_json() or {}
|
|
password = data.get('password')
|
|
|
|
if not password:
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'No password provided'
|
|
}), 400
|
|
|
|
if len(password) < 8:
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'Password must be at least 8 characters'
|
|
}), 400
|
|
|
|
try:
|
|
settings_manager = get_settings_manager()
|
|
PasswordManager.set_app_password(settings_manager, password)
|
|
|
|
return jsonify({
|
|
'status': 'success',
|
|
'message': 'Password updated successfully'
|
|
})
|
|
except Exception as e:
|
|
current_app.logger.error(f"Failed to set password: {e}")
|
|
return jsonify({
|
|
'status': 'error',
|
|
'message': 'Failed to set password'
|
|
}), 500
|
|
|
|
|
|
@bp.route('/test-email', methods=['POST'])
|
|
def test_email():
|
|
"""
|
|
Test email configuration by sending a test email.
|
|
|
|
Returns:
|
|
JSON response with test result
|
|
"""
|
|
# TODO: Implement in Phase 4 (email support)
|
|
return jsonify({
|
|
'status': 'not_implemented',
|
|
'message': 'Email testing endpoint - to be implemented in Phase 4'
|
|
}), 501
|
|
|
|
|
|
# Health check endpoint
|
|
@bp.route('/health', methods=['GET'])
|
|
def health_check():
|
|
"""
|
|
Health check endpoint for monitoring.
|
|
|
|
Returns:
|
|
JSON response with API health status
|
|
"""
|
|
return jsonify({
|
|
'status': 'healthy',
|
|
'api': 'settings',
|
|
'version': '1.0.0-phase1'
|
|
})
|