feat: Shutdown on lan
This commit is contained in:
parent
9023767f90
commit
fa31227cff
@ -13,5 +13,6 @@
|
||||
./vpn.nix
|
||||
./printing.nix
|
||||
./adb.nix
|
||||
./shutdown-on-lan.nix
|
||||
];
|
||||
}
|
||||
|
70
nixos/modules/shutdown-on-lan.nix
Normal file
70
nixos/modules/shutdown-on-lan.nix
Normal file
@ -0,0 +1,70 @@
|
||||
{ pkgs, lib, ... }:
|
||||
let
|
||||
sol = pkgs.writers.writePython3 "shutdown-on-lan.py" {
|
||||
libraries = [ pkgs.python312Packages.psutil ];
|
||||
flakeIgnore = [ "E501" "E302" "E305" ];
|
||||
} /*py*/ ''
|
||||
# https://habr.com/ru/articles/816765/
|
||||
|
||||
import socket
|
||||
import os
|
||||
import logging
|
||||
import psutil
|
||||
from time import sleep
|
||||
|
||||
WOL_PORT = 9
|
||||
|
||||
logging.basicConfig(format='%(levelname)s: %(asctime)s %(message)s', level=logging.INFO)
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
def get_ip_mac_address() -> tuple:
|
||||
ip_addr = mac_addr = None
|
||||
|
||||
while not ip_addr or not mac_addr or ip_addr == '127.0.0.1':
|
||||
net = psutil.net_if_addrs()
|
||||
for item in net[list(net.keys())[-1]]:
|
||||
addr = item.address
|
||||
# В IPv4-адресах разделители - точки
|
||||
if '.' in addr:
|
||||
ip_addr = addr
|
||||
# В MAC-адресах разделители либо тире, либо одинарное двоеточие.
|
||||
# Двойное двоеточие - это разделители для адресов IPv6
|
||||
elif ('-' in addr or ':' in addr) and '::' not in addr:
|
||||
# Приводим MAC-адрес к одному формату. Формат может меняться в зависимости от ОС
|
||||
mac_addr = addr.replace(':', '-').upper()
|
||||
if not ip_addr or not mac_addr or ip_addr == '127.0.0.1':
|
||||
logger.debug('Не удалось получить IP или MAC-адрес сетевого интерфейса')
|
||||
sleep(10)
|
||||
return ip_addr, mac_addr
|
||||
|
||||
def assemble_wol_packet(mac_address: str) -> str:
|
||||
return f'{"FF-" * 6}{(mac_address + "-") * 16}'
|
||||
|
||||
def check_is_wol_packet(raw_bytes: bytes, assembled_wol_packet: str) -> int:
|
||||
return '-'.join(f'{byte:02x}' for byte in raw_bytes).upper() + '-' == assembled_wol_packet
|
||||
|
||||
def run_udp_port_listener(port: int):
|
||||
ip_addr, mac_addr = get_ip_mac_address()
|
||||
server_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
||||
server_socket.bind((ip_addr, port))
|
||||
logger.info(f'Listening on {ip_addr}:{port}')
|
||||
assembled_wol_packet = assemble_wol_packet(mac_addr)
|
||||
while True:
|
||||
data, _ = server_socket.recvfrom(1024)
|
||||
if check_is_wol_packet(data, assembled_wol_packet):
|
||||
if os.name == 'posix':
|
||||
os.system('shutdown -h now')
|
||||
elif os.name == 'nt':
|
||||
os.system('shutdown -s -t 0 -f')
|
||||
run_udp_port_listener(WOL_PORT)
|
||||
'';
|
||||
in {
|
||||
systemd.services.shutdown-on-lan = {
|
||||
enable = true;
|
||||
after = [ "network.target" ];
|
||||
wantedBy = [ "default.target" ];
|
||||
serviceConfig.ExecStart = sol;
|
||||
};
|
||||
|
||||
networking.firewall.allowedUDPPorts = [ 9 ];
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user