2fb067ecbf
checklocks / checklocks (push) Has been cancelled
CodeQL / Analyze (go) (push) Has been cancelled
natlab-integrationtest / natlab-integrationtest (push) Has been cancelled
CI / gomod-cache (push) Has been cancelled
CI / race-root-integration (1/4) (push) Has been cancelled
CI / race-root-integration (2/4) (push) Has been cancelled
CI / race-root-integration (3/4) (push) Has been cancelled
CI / race-root-integration (4/4) (push) Has been cancelled
CI / test (-race, amd64, 1/3) (push) Has been cancelled
CI / test (-race, amd64, 2/3) (push) Has been cancelled
CI / test (-race, amd64, 3/3) (push) Has been cancelled
CI / test (386) (push) Has been cancelled
CI / test (amd64) (push) Has been cancelled
CI / Windows (benchmarks) (push) Has been cancelled
CI / Windows (1/2) (push) Has been cancelled
CI / Windows (2/2) (push) Has been cancelled
CI / macos (push) Has been cancelled
CI / privileged (push) Has been cancelled
CI / vm (push) Has been cancelled
CI / cross (386, linux) (push) Has been cancelled
CI / cross (amd64, darwin) (push) Has been cancelled
CI / cross (amd64, freebsd) (push) Has been cancelled
CI / cross (amd64, openbsd) (push) Has been cancelled
CI / cross (amd64, windows) (push) Has been cancelled
CI / cross (arm, 5, linux) (push) Has been cancelled
CI / cross (arm, 7, linux) (push) Has been cancelled
CI / cross (arm64, darwin) (push) Has been cancelled
CI / cross (arm64, linux) (push) Has been cancelled
CI / cross (arm64, windows) (push) Has been cancelled
CI / cross (loong64, linux) (push) Has been cancelled
CI / ios (push) Has been cancelled
CI / crossmin (amd64, illumos) (push) Has been cancelled
CI / crossmin (amd64, plan9) (push) Has been cancelled
CI / crossmin (amd64, solaris) (push) Has been cancelled
CI / crossmin (ppc64, aix) (push) Has been cancelled
CI / android (push) Has been cancelled
CI / wasm (push) Has been cancelled
CI / tailscale_go (push) Has been cancelled
CI / fuzz (push) Has been cancelled
CI / depaware (push) Has been cancelled
CI / go_generate (push) Has been cancelled
CI / make_tidy (push) Has been cancelled
CI / licenses (push) Has been cancelled
CI / staticcheck (macOS) (push) Has been cancelled
CI / staticcheck (Linux) (push) Has been cancelled
CI / staticcheck (Windows) (push) Has been cancelled
CI / staticcheck (Portable (1/4)) (push) Has been cancelled
CI / staticcheck (Portable (2/4)) (push) Has been cancelled
CI / staticcheck (Portable (3/4)) (push) Has been cancelled
CI / staticcheck (Portable (4/4)) (push) Has been cancelled
CI / notify_slack (push) Has been cancelled
CI / merge_blocker (push) Has been cancelled
CI / check_mergeability_strict (push) Has been cancelled
CI / check_mergeability (push) Has been cancelled
Dockerfile build / deploy (push) Has been cancelled
test installer.sh / test (curl, alpine:3.21) (push) Has been cancelled
test installer.sh / test (curl, alpine:edge) (push) Has been cancelled
test installer.sh / test (curl, alpine:latest) (push) Has been cancelled
test installer.sh / test (curl, amazonlinux:latest) (push) Has been cancelled
test installer.sh / test (curl, archlinux:latest) (push) Has been cancelled
test installer.sh / test (curl, debian:oldstable-slim) (push) Has been cancelled
test installer.sh / test (curl, debian:sid-slim) (push) Has been cancelled
test installer.sh / test (curl, debian:stable-slim, 1.80.0) (push) Has been cancelled
test installer.sh / test (curl, debian:testing-slim) (push) Has been cancelled
test installer.sh / test (curl, elementary/docker:stable) (push) Has been cancelled
test installer.sh / test (curl, elementary/docker:unstable) (push) Has been cancelled
test installer.sh / test (curl, fedora:latest, 1.80.0) (push) Has been cancelled
test installer.sh / test (curl, kalilinux/kali-dev) (push) Has been cancelled
test installer.sh / test (curl, kalilinux/kali-rolling) (push) Has been cancelled
test installer.sh / test (curl, opensuse/leap:latest) (push) Has been cancelled
test installer.sh / test (curl, opensuse/tumbleweed:latest) (push) Has been cancelled
test installer.sh / test (curl, oraclelinux:8) (push) Has been cancelled
test installer.sh / test (curl, oraclelinux:9) (push) Has been cancelled
test installer.sh / test (curl, parrotsec/core:latest) (push) Has been cancelled
test installer.sh / test (curl, rockylinux:8.7) (push) Has been cancelled
test installer.sh / test (curl, rockylinux:9) (push) Has been cancelled
test installer.sh / test (curl, ubuntu:20.04) (push) Has been cancelled
test installer.sh / test (curl, ubuntu:22.04) (push) Has been cancelled
test installer.sh / test (curl, ubuntu:24.04, 1.80.0) (push) Has been cancelled
test installer.sh / test (wget, debian:oldstable-slim) (push) Has been cancelled
test installer.sh / test (wget, debian:sid-slim) (push) Has been cancelled
update-flake / update-flake (push) Has been cancelled
tailscale.com/cmd/vet / vet (push) Has been cancelled
test installer.sh / notify-slack (push) Has been cancelled
Client security fixes (cmd/tailscale-tray/main.go): - SSRF protection in Add Server dialog (validateControlURL): reject private/loopback/link-local/cloud-metadata IPs via DNS resolution - RCE gate on AuthURL/BrowseToURL exec paths (validateAuthURL) - Sanitized URL logging (sanitizeURLForLog drops query auth tokens) - Error handling on exec.Command with user-facing showError() Admin panel security (web-admin): - Bcrypt password hashing (replaces SHA256) - Rate limiting: 5 failed logins → 15-min lockout - Session + login attempt cleanup goroutine (hourly) - url.QueryEscape / encodeURIComponent for all API params - Fail-hard startup when no TLS and non-loopback bind - ADMIN_PASSWORD required (no default), password min 12 chars - Username regex whitelist Installer hardening (Setup.wxs): - util:PermissionEx restricts SCM access: only Administrators + SYSTEM can start/stop/reconfigure service. Authenticated Users limited to QueryStatus/QueryConfig/Interrogate - Vital="yes" on ServiceInstall Docs & roadmap: - PRODUCTION_ROADMAP.md: 5-milestone plan (security + features + distribution + ops) with granular tasks, effort, done-when - CLIENT_SECURITY_AUDIT.md, SECURITY_FIXES.md, DEPLOYMENT.md - AI assistant rules (.cursorrules, .antigravityrules, etc.) Build & distribution: - build-msi.ps1, deploy-and-sign.ps1, sign-release.ps1 - redeploy.ps1, tray-deploy.ps1, test-msi.ps1 - installer/msi/ alternative WXS setup - Restored .github/workflows/ removed in mirror cleanup .gitignore hardened: *.pfx, *.p12, *.key, *.pem, .env*
149 lines
3.9 KiB
Python
149 lines
3.9 KiB
Python
#!/usr/bin/env python3
|
||
"""
|
||
Auto Preview - Antigravity Kit
|
||
==============================
|
||
Manages (start/stop/status) the local development server for previewing the application.
|
||
|
||
Usage:
|
||
python .agent/scripts/auto_preview.py start [port]
|
||
python .agent/scripts/auto_preview.py stop
|
||
python .agent/scripts/auto_preview.py status
|
||
"""
|
||
|
||
import os
|
||
import sys
|
||
import time
|
||
import json
|
||
import signal
|
||
import argparse
|
||
import subprocess
|
||
from pathlib import Path
|
||
|
||
AGENT_DIR = Path(".agent")
|
||
PID_FILE = AGENT_DIR / "preview.pid"
|
||
LOG_FILE = AGENT_DIR / "preview.log"
|
||
|
||
def get_project_root():
|
||
return Path(".").resolve()
|
||
|
||
def is_running(pid):
|
||
try:
|
||
os.kill(pid, 0)
|
||
return True
|
||
except OSError:
|
||
return False
|
||
|
||
def get_start_command(root):
|
||
pkg_file = root / "package.json"
|
||
if not pkg_file.exists():
|
||
return None
|
||
|
||
with open(pkg_file, 'r') as f:
|
||
data = json.load(f)
|
||
|
||
scripts = data.get("scripts", {})
|
||
if "dev" in scripts:
|
||
return ["npm", "run", "dev"]
|
||
elif "start" in scripts:
|
||
return ["npm", "start"]
|
||
return None
|
||
|
||
def start_server(port=3000):
|
||
if PID_FILE.exists():
|
||
try:
|
||
pid = int(PID_FILE.read_text().strip())
|
||
if is_running(pid):
|
||
print(f"⚠️ Preview already running (PID: {pid})")
|
||
return
|
||
except:
|
||
pass # Invalid PID file
|
||
|
||
root = get_project_root()
|
||
cmd = get_start_command(root)
|
||
|
||
if not cmd:
|
||
print("❌ No 'dev' or 'start' script found in package.json")
|
||
sys.exit(1)
|
||
|
||
# Add port env var if needed (simple heuristic)
|
||
env = os.environ.copy()
|
||
env["PORT"] = str(port)
|
||
|
||
print(f"🚀 Starting preview on port {port}...")
|
||
|
||
with open(LOG_FILE, "w") as log:
|
||
process = subprocess.Popen(
|
||
cmd,
|
||
cwd=str(root),
|
||
stdout=log,
|
||
stderr=log,
|
||
env=env,
|
||
shell=True # Required for npm on windows often, or consistent path handling
|
||
)
|
||
|
||
PID_FILE.write_text(str(process.pid))
|
||
print(f"✅ Preview started! (PID: {process.pid})")
|
||
print(f" Logs: {LOG_FILE}")
|
||
print(f" URL: http://localhost:{port}")
|
||
|
||
def stop_server():
|
||
if not PID_FILE.exists():
|
||
print("ℹ️ No preview server found.")
|
||
return
|
||
|
||
try:
|
||
pid = int(PID_FILE.read_text().strip())
|
||
if is_running(pid):
|
||
# Try gentle kill first
|
||
os.kill(pid, signal.SIGTERM) if sys.platform != 'win32' else subprocess.call(['taskkill', '/F', '/T', '/PID', str(pid)])
|
||
print(f"🛑 Preview stopped (PID: {pid})")
|
||
else:
|
||
print("ℹ️ Process was not running.")
|
||
except Exception as e:
|
||
print(f"❌ Error stopping server: {e}")
|
||
finally:
|
||
if PID_FILE.exists():
|
||
PID_FILE.unlink()
|
||
|
||
def status_server():
|
||
running = False
|
||
pid = None
|
||
url = "Unknown"
|
||
|
||
if PID_FILE.exists():
|
||
try:
|
||
pid = int(PID_FILE.read_text().strip())
|
||
if is_running(pid):
|
||
running = True
|
||
# Heuristic for URL, strictly we should save it
|
||
url = "http://localhost:3000"
|
||
except:
|
||
pass
|
||
|
||
print("\n=== Preview Status ===")
|
||
if running:
|
||
print(f"✅ Status: Running")
|
||
print(f"🔢 PID: {pid}")
|
||
print(f"🌐 URL: {url} (Likely)")
|
||
print(f"📝 Logs: {LOG_FILE}")
|
||
else:
|
||
print("⚪ Status: Stopped")
|
||
print("===================\n")
|
||
|
||
def main():
|
||
parser = argparse.ArgumentParser()
|
||
parser.add_argument("action", choices=["start", "stop", "status"])
|
||
parser.add_argument("port", nargs="?", default="3000")
|
||
|
||
args = parser.parse_args()
|
||
|
||
if args.action == "start":
|
||
start_server(int(args.port))
|
||
elif args.action == "stop":
|
||
stop_server()
|
||
elif args.action == "status":
|
||
status_server()
|
||
|
||
if __name__ == "__main__":
|
||
main()
|