Core: - paths.py: centralized ~/.soosef/ path constants - config.py: JSON config loader with dataclass defaults - exceptions.py: SoosefError hierarchy - cli.py: unified Click CLI wrapping stegasoo + verisoo + native commands Keystore: - manager.py: unified key management (Ed25519 identity + channel keys) - models.py: IdentityInfo, KeystoreStatus dataclasses - export.py: encrypted key bundle export/import for USB transfer Fieldkit: - killswitch.py: ordered emergency data destruction (keys first) - deadman.py: dead man's switch with check-in timer - tamper.py: SHA-256 file integrity baseline + checking - usb_monitor.py: pyudev USB whitelist enforcement - geofence.py: haversine-based GPS boundary checking Web frontend (Flask app factory + blueprints): - app.py: create_app() factory with context processor - blueprints: stego, attest, fieldkit, keys, admin - templates: base.html (dark theme, unified nav), dashboard, all section pages - static: CSS, favicon Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
50 lines
1.6 KiB
Python
50 lines
1.6 KiB
Python
"""
|
|
Fieldkit blueprint — killswitch, dead man's switch, status dashboard.
|
|
"""
|
|
|
|
from flask import Blueprint, flash, redirect, render_template, request, url_for
|
|
|
|
bp = Blueprint("fieldkit", __name__, url_prefix="/fieldkit")
|
|
|
|
|
|
@bp.route("/")
|
|
def status():
|
|
"""Fieldkit status dashboard — all monitors and system health."""
|
|
from soosef.fieldkit.deadman import DeadmanSwitch
|
|
|
|
deadman = DeadmanSwitch()
|
|
return render_template(
|
|
"fieldkit/status.html",
|
|
deadman_status=deadman.status(),
|
|
)
|
|
|
|
|
|
@bp.route("/killswitch", methods=["GET", "POST"])
|
|
def killswitch():
|
|
"""Killswitch arming and firing UI."""
|
|
if request.method == "POST":
|
|
action = request.form.get("action")
|
|
if action == "fire" and request.form.get("confirm") == "CONFIRM-PURGE":
|
|
from soosef.fieldkit.killswitch import PurgeScope, execute_purge
|
|
|
|
result = execute_purge(PurgeScope.ALL, reason="web_ui")
|
|
flash(
|
|
f"Purge executed: {len(result.steps_completed)} steps completed, "
|
|
f"{len(result.steps_failed)} failed",
|
|
"warning" if result.steps_failed else "success",
|
|
)
|
|
return redirect(url_for("fieldkit.status"))
|
|
|
|
return render_template("fieldkit/killswitch.html")
|
|
|
|
|
|
@bp.route("/deadman/checkin", methods=["POST"])
|
|
def deadman_checkin():
|
|
"""Record a dead man's switch check-in."""
|
|
from soosef.fieldkit.deadman import DeadmanSwitch
|
|
|
|
deadman = DeadmanSwitch()
|
|
deadman.checkin()
|
|
flash("Check-in recorded.", "success")
|
|
return redirect(url_for("fieldkit.status"))
|