from flask_sqlalchemy import SQLAlchemy from flask_login import UserMixin from werkzeug.security import generate_password_hash, check_password_hash db = SQLAlchemy() class User(UserMixin, db.Model): __tablename__ = "users" id = db.Column(db.Integer, primary_key=True) username = db.Column(db.String(80), unique=True, nullable=False, index=True) password_hash = db.Column(db.String(256), nullable=False) def set_password(self, password: str) -> None: self.password_hash = generate_password_hash(password) def check_password(self, password: str) -> bool: return check_password_hash(self.password_hash, password) class ServiceGroup(db.Model): __tablename__ = "service_groups" id = db.Column(db.Integer, primary_key=True) name = db.Column(db.String(120), nullable=False) sort_order = db.Column(db.Integer, nullable=False, default=0) services = db.relationship( "Service", backref="group", lazy="dynamic", cascade="all, delete-orphan", ) class Service(db.Model): __tablename__ = "services" id = db.Column(db.Integer, primary_key=True) name = db.Column(db.String(120), nullable=False) scheme = db.Column(db.String(8), nullable=False, default="http") host = db.Column(db.String(255), nullable=False) port = db.Column(db.Integer, nullable=False) path = db.Column(db.String(512), nullable=False, default="/") sort_order = db.Column(db.Integer, nullable=False, default=0) group_id = db.Column( db.Integer, db.ForeignKey("service_groups.id"), nullable=False, index=True ) # hub=复盘中控(iframe 嵌入需 /embed-auth);留空=普通内嵌 embed_kind = db.Column(db.String(16), nullable=False, default="", server_default="") def build_origin(self) -> str: proto = (self.scheme or "http").strip().lower() if proto not in ("http", "https"): proto = "http" return f"{proto}://{self.host}:{self.port}" def build_url(self) -> str: p = (self.path or "/").strip() if not p.startswith("/"): p = "/" + p return f"{self.build_origin()}{p}" def build_open_url(self) -> str: """导航 iframe 首次打开的地址(中控走 login?embed=1 以便写入会话)。""" kind = (self.embed_kind or "").strip().lower() next_path = (self.path or "/monitor").strip() or "/monitor" if not next_path.startswith("/"): next_path = "/" + next_path if kind == "hub": from urllib.parse import urlencode q = urlencode({"embed": "1", "next": next_path}) return f"{self.build_origin()}/login?{q}" return self.build_url() def is_hub_embed(self) -> bool: return (self.embed_kind or "").strip().lower() == "hub"