Загрузка данных


import os
import sys
import ctypes
import subprocess
import json
import socket
import customtkinter as ctk

# Конфиг
CONFIG_FILE = "dns_config.json"

class FerrariDNS(ctk.CTk):
    def __init__(self):
        super().__init__()
        
        self.title("Ferrari Private DNS")
        self.geometry("450x500")
        ctk.set_appearance_mode("dark")

        # Загрузка конфига
        self.config = self.load_config()

        # Интерфейс
        self.label = ctk.CTkLabel(self, text="PRIVATE DNS SETTINGS", font=("Arial", 20, "bold"))
        self.label.pack(pady=20)

        self.dns_input = ctk.CTkEntry(self, width=350)
        self.dns_input.pack(pady=10)
        self.dns_input.insert(0, self.config.get("dns_url", "dns.comss.one"))

        self.status_label = ctk.CTkLabel(self, text=f"Статус: {self.config.get('status', 'Ожидание')}", text_color="gray")
        self.status_label.pack(pady=5)

        self.btn_on = ctk.CTkButton(self, text="ВКЛЮЧИТЬ", fg_color="green", command=self.activate)
        self.btn_on.pack(pady=10)

        self.btn_off = ctk.CTkButton(self, text="СБРОСИТЬ (АВТО)", fg_color="#d32f2f", command=self.reset)
        self.btn_off.pack(pady=10)

        self.auto_run_var = ctk.BooleanVar(value=self.config.get("auto_run", False))
        self.check_auto = ctk.CTkCheckBox(self, text="Автозагрузка при старте", variable=self.auto_run_var, command=self.toggle_autorun)
        self.check_auto.pack(pady=20)

    def load_config(self):
        if os.path.exists(CONFIG_FILE):
            try:
                with open(CONFIG_FILE, "r") as f:
                    return json.load(f)
            except:
                pass
        return {"dns_url": "dns.comss.one", "status": "Отключено", "auto_run": False}

    def save_config(self, status):
        config_data = {
            "dns_url": self.dns_input.get(),
            "status": status,
            "auto_run": self.auto_run_var.get()
        }
        with open(CONFIG_FILE, "w") as f:
            json.dump(config_data, f)

    def get_adapter(self):
        try:
            # Берем первый активный адаптер
            cmd = 'powershell "(Get-NetAdapter | Where-Object {$_.Status -eq \'Up\'} | Select-Object -First 1).Name"'
            name = subprocess.check_output(cmd, shell=True).decode('cp866').strip()
            return name if name else "Wi-Fi"
        except:
            return "Wi-Fi"

    def activate(self):
        raw_input = self.dns_input.get().strip()
        adapter = self.get_adapter()
        self.status_label.configure(text="Настройка...", text_color="yellow")
        
        try:
            # Формируем хост и ссылку
            host = raw_input.replace("https://", "").split("/")[0]
            ip = socket.gethostbyname(host)
            doh_template = raw_input if "://" in raw_input else f"https://{raw_input}/dns-query"

            # Команды PowerShell с экранированием кавычек
            # 1. Регистрация сервера в базе DoH
            cmd_reg = f'powershell "Add-DnsClientDohServerAddress -ServerAddress {ip} -DohTemplate \'{doh_template}\' -AllowFallbackToUdp $false -ErrorAction SilentlyContinue"'
            # 2. Установка DNS для адаптера
            cmd_set = f'powershell "Set-DnsClientServerAddress -InterfaceAlias \'{adapter}\' -ServerAddresses {ip}"'
            # 3. Принудительный DoH
            cmd_force = f'powershell "Set-DnsClient -InterfaceAlias \'{adapter}\' -UseDnsOverHttps Only"'

            subprocess.run(cmd_reg, shell=True)
            subprocess.run(cmd_set, shell=True)
            subprocess.run(cmd_force, shell=True)
            subprocess.run('ipconfig /flushdns', shell=True)
            
            self.status_label.configure(text="Статус: АКТИВНО", text_color="green")
            self.save_config("Активно")
        except Exception as e:
            self.status_label.configure(text=f"Ошибка: {str(e)}", text_color="red")

    def reset(self):
        adapter = self.get_adapter()
        try:
            subprocess.run(f'powershell "Set-DnsClientServerAddress -InterfaceAlias \'{adapter}\' -ResetServerAddresses"', shell=True)
            subprocess.run(f'powershell "Set-DnsClient -InterfaceAlias \'{adapter}\' -UseDnsOverHttps Prohibit"', shell=True)
            subprocess.run('ipconfig /flushdns', shell=True)
            self.status_label.configure(text="Статус: Сброшено", text_color="white")
            self.save_config("Отключено")
        except:
            self.status_label.configure(text="Ошибка сброса", text_color="red")

    def toggle_autorun(self):
        path = os.path.realpath(sys.argv[0])
        key = r"Software\Microsoft\Windows\CurrentVersion\Run"
        if self.auto_run_var.get():
            subprocess.run(f'reg add "HKEY_CURRENT_USER\\{key}" /v "FerrariDNS" /t REG_SZ /d "{path}" /f', shell=True)
        else:
            subprocess.run(f'reg delete "HKEY_CURRENT_USER\\{key}" /v "FerrariDNS" /f', shell=True)
        self.save_config(self.config.get("status", "Отключено"))

if __name__ == "__main__":
    if not ctypes.windll.shell32.IsUserAnAdmin():
        # Запуск с правами админа
        ctypes.windll.shell32.ShellExecuteW(None, "runas", sys.executable, f'"{__file__}"', None, 1)
    else:
        app = FerrariDNS()
        app.mainloop()