1072 lines
46 KiB
Python
Executable File
1072 lines
46 KiB
Python
Executable File
#!/usr/bin/python3
|
|
import os
|
|
import shutil
|
|
import subprocess
|
|
import tkinter as tk
|
|
from pathlib import Path
|
|
from subprocess import check_call
|
|
from tkinter import *
|
|
from tkinter import filedialog, ttk
|
|
|
|
from wg_func import (Tunnel, FileHandle, OnOff, msg_window, WirePyUpdate, res, _u, version, path_to_file2, tips)
|
|
|
|
tcl_path = Path('/usr/share/TK-Themes')
|
|
|
|
|
|
class MainWindow(tk.Tk):
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.my_tool_tip = None
|
|
self.switch_on = None
|
|
self.switch_off = None
|
|
self.x_width = 600
|
|
self.y_height = 400
|
|
self.monitor_center_x = self.winfo_screenwidth() / 2 - (self.x_width / 2)
|
|
self.monitor_center_y = self.winfo_screenheight() / 2 - (self.y_height / 2)
|
|
self.resizable(width=False, height=False)
|
|
self.title('Wire-Py')
|
|
self.geometry('%dx%d+%d+%d' % (self.x_width, self.y_height, self.monitor_center_x, self.monitor_center_y))
|
|
self.style = ttk.Style(self)
|
|
self.tk.call('source', str(tcl_path) + '/water.tcl')
|
|
''' self.tk.call('source', 'TK-Themes/water.tcl') '''
|
|
with open(path_to_file2, 'r') as read_file:
|
|
lines = read_file.readlines()
|
|
if 'light\n' in lines:
|
|
self.tk.call('set_theme', 'light')
|
|
else:
|
|
self.tk.call('set_theme', 'dark')
|
|
|
|
''' Load the image file from disk. '''
|
|
self.wg_icon = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_vpn.png')
|
|
|
|
''' Set it as the window icon '''
|
|
self.iconphoto(True, self.wg_icon)
|
|
|
|
''' Set on or off in file '''
|
|
def update():
|
|
if set_update.get() == 1:
|
|
with open(path_to_file2, 'r') as set_file2:
|
|
lines2 = set_file2.readlines()
|
|
lines2[1] = 'off\n'
|
|
with open(path_to_file2, 'w') as set_file2:
|
|
set_file2.writelines(lines2)
|
|
|
|
if set_update.get() == 0:
|
|
with open(path_to_file2, 'r') as set_file2:
|
|
lines2 = set_file2.readlines()
|
|
lines2[1] = 'on\n'
|
|
with open(path_to_file2, 'w') as set_file2:
|
|
set_file2.writelines(lines2)
|
|
|
|
''' Set on or off in file '''
|
|
def tooltip():
|
|
if set_tip.get():
|
|
with open(path_to_file2, 'r') as set_file2:
|
|
lines2 = set_file2.readlines()
|
|
lines2[5] = 'False\n'
|
|
with open(path_to_file2, 'w') as set_file2:
|
|
set_file2.writelines(lines2)
|
|
|
|
else:
|
|
with open(path_to_file2, 'r') as set_file2:
|
|
lines2 = set_file2.readlines()
|
|
lines2[5] = 'True\n'
|
|
with open(path_to_file2, 'w') as set_file2:
|
|
set_file2.writelines(lines2)
|
|
|
|
''' Set dark or light '''
|
|
def theme_change_light():
|
|
if self.tk.call("ttk::style", "theme", "use") == "water-dark":
|
|
''' Set light theme '''
|
|
self.tk.call('set_theme', 'light')
|
|
with open(path_to_file2, 'r') as theme_set2:
|
|
lines3 = theme_set2.readlines()
|
|
lines3[3] = 'light\n'
|
|
with open(path_to_file2, 'w') as theme_set2:
|
|
theme_set2.writelines(lines3)
|
|
|
|
def theme_change_dark():
|
|
if not self.tk.call("ttk::style", "theme", "use") == "water-dark":
|
|
''' Set dark theme '''
|
|
self.tk.call('set_theme', 'dark')
|
|
with open(path_to_file2, 'r') as theme_set2:
|
|
lines4 = theme_set2.readlines()
|
|
lines4[3] = 'dark\n'
|
|
with open(path_to_file2, 'w') as theme_set2:
|
|
theme_set2.writelines(lines4)
|
|
|
|
def info():
|
|
def link_btn():
|
|
_u1 = str(_u[6:])
|
|
path_to_file = Path('/usr/bin/./run_as')
|
|
check_call(['su', _u1, path_to_file])
|
|
|
|
"""img_w, img_i, w_title, w_txt , txt2, com hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/48/wg_vpn.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Info'
|
|
msg_t = ('Wire-Py a simple Wireguard Gui for Linux systems.\n\n'
|
|
'Wire-Py is open source software written in Python.\n\n'
|
|
'Email: polunga40@unity-mail.de also likes for donation.\n\n'
|
|
'Use without warranty!\n')
|
|
txt2 = 'Go to Wire-Py git'
|
|
com = link_btn
|
|
|
|
msg_window(iw, ii, wt, msg_t, txt2, com)
|
|
|
|
''' Frame for Menu '''
|
|
self.menu_frame = ttk.Frame(self)
|
|
self.menu_frame.configure(relief='flat')
|
|
self.menu_frame.grid(column=0, row=0, sticky='w')
|
|
self.columnconfigure(0, weight=1)
|
|
self.rowconfigure(0, weight=1)
|
|
|
|
''' App Menu '''
|
|
self.version_lb = ttk.Label(self.menu_frame, text=version)
|
|
self.version_lb.config(font=('Ubuntu', 11), foreground='#00c4ff')
|
|
self.version_lb.grid(column=0, row=0, padx=10)
|
|
|
|
def version_enter(event):
|
|
""" The mouse moves into the Version widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root, f'Version: {version[2:]}')
|
|
|
|
def version_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.options_btn = ttk.Menubutton(self.menu_frame, text='Options')
|
|
self.options_btn.grid(column=1, row=0)
|
|
|
|
def sets_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root, 'Click for Settings')
|
|
|
|
def sets_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.version_lb.bind('<Enter>', version_enter)
|
|
self.version_lb.bind('<Leave>', version_leave)
|
|
self.options_btn.bind('<Enter>', sets_enter)
|
|
self.options_btn.bind('<Leave>', sets_leave)
|
|
|
|
set_update = tk.IntVar()
|
|
set_tip = tk.BooleanVar()
|
|
self.settings = tk.Menu(self, relief='flat')
|
|
self.options_btn.configure(menu=self.settings, style='Toolbutton')
|
|
self.settings.add_checkbutton(label='Disable Updates', command=update, variable=set_update)
|
|
self.settings.add_checkbutton(label='Disable Tooltips', command=tooltip, variable=set_tip)
|
|
self.settings.add_command(label='Light', command=theme_change_light)
|
|
self.settings.add_command(label='Dark', command=theme_change_dark)
|
|
|
|
''' About BTN Menu / Label '''
|
|
self.about_btn = ttk.Button(self.menu_frame, text='About', style='Toolbutton', command=info)
|
|
self.about_btn.grid(column=2, row=0)
|
|
|
|
self.readme = tk.Menu(self)
|
|
|
|
''' Update and Tooltip Label '''
|
|
self.updates_lb = ttk.Label(self.menu_frame)
|
|
self.updates_lb.grid(column=3, row=0, padx=10)
|
|
|
|
'''View Checkbox for enable or disable Tooltip '''
|
|
if tips:
|
|
set_tip.set(value=False)
|
|
else:
|
|
set_tip.set(value=True)
|
|
|
|
'''View Checkbox for enable or disable Updates '''
|
|
if res == 'False':
|
|
set_update.set(value=1)
|
|
self.updates_lb.configure(text='Update search off')
|
|
|
|
def disable_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root, 'Updates you have disabled')
|
|
|
|
def disable_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.updates_lb.bind('<Enter>', disable_enter)
|
|
self.updates_lb.bind('<Leave>', disable_leave)
|
|
elif res == 'No Internet Connection!':
|
|
self.updates_lb.configure(text='No Server Connection!', foreground='red')
|
|
elif res == 'No Updates':
|
|
self.updates_lb.configure(text='No Updates')
|
|
|
|
def congratulations_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'Congratulations! Wire-Py is up to date')
|
|
|
|
def congratulations_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.updates_lb.bind('<Enter>', congratulations_enter)
|
|
self.updates_lb.bind('<Leave>', congratulations_leave)
|
|
else:
|
|
set_update.set(value=0)
|
|
text = f'Update {res} available!'
|
|
|
|
''' Update BTN Menu'''
|
|
self.update_btn = ttk.Menubutton(self.menu_frame, text=text)
|
|
self.update_btn.grid(column=4, row=0, padx=0)
|
|
|
|
def download_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'Click to download new version')
|
|
|
|
def download_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.update_btn.bind('<Enter>', download_enter)
|
|
self.update_btn.bind('<Leave>', download_leave)
|
|
|
|
self.download = tk.Menu(self, relief='flat')
|
|
|
|
self.update_btn.configure(menu=self.download, style='Toolbutton')
|
|
self.download.add_command(label='Download', command=WirePyUpdate.download)
|
|
|
|
FrameWidgets(self).grid()
|
|
|
|
|
|
class StSt:
|
|
"""
|
|
Show Start and Stop Button in Label
|
|
"""
|
|
|
|
def __init__(self):
|
|
self.lb_frame_btn_lbox = None
|
|
self.wg_switch = None
|
|
self.btn_stst = None
|
|
self.wg_vpn_start = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_vpn-start.png')
|
|
self.wg_vpn_stop = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_vpn-stop.png')
|
|
|
|
def stop(self):
|
|
self.btn_stst = ttk.Button(self.lb_frame_btn_lbox, image=self.wg_vpn_stop, command=self.wg_switch, padding=0)
|
|
self.btn_stst.grid(column=0, row=0, padx=5, pady=8)
|
|
|
|
def stop_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'Click to stop selected Wireguard Tunnel')
|
|
|
|
def stop_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.btn_stst.bind('<Enter>', stop_enter)
|
|
self.btn_stst.bind('<Leave>', stop_leave)
|
|
|
|
def start(self):
|
|
self.btn_stst = ttk.Button(self.lb_frame_btn_lbox, image=self.wg_vpn_start, command=self.wg_switch, padding=0)
|
|
self.btn_stst.grid(column=0, row=0, padx=5, pady=8)
|
|
|
|
def empty_list_start_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'No tunnels to start in the list')
|
|
|
|
def empty_list_start_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def start_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'Click to start selected Wireguard Tunnel')
|
|
|
|
def start_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
tl = Tunnel.list()
|
|
if len(tl) == 0:
|
|
self.btn_stst.bind('<Enter>', empty_list_start_enter)
|
|
self.btn_stst.bind('<Leave>', empty_list_start_leave)
|
|
else:
|
|
self.btn_stst.bind('<Enter>', start_enter)
|
|
self.btn_stst.bind('<Leave>', start_leave)
|
|
|
|
|
|
class TLImp:
|
|
"""
|
|
Import Class for Wireguard config Files.
|
|
Before importing, it is checked whether PrivateKey and PublicKey are in the file.
|
|
If True then it is checked whether the PreSharedKey is already in the key file
|
|
to avoid an import error so that no double wgconf are imported.
|
|
Thus, tunnels can be renamed without the problems arise. If False then the key is written into the file.
|
|
Furthermore, it is checked whether the name is longer than 12 characters.
|
|
If True then the name is automatically shortened to 12 characters and then imported.
|
|
If in each case false comes out, a corresponding window comes to inform the user that something is wrong.
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
self.lb_rename = None
|
|
self.btn_tr = None
|
|
self.btn_exp = None
|
|
self.wg_autostart = None
|
|
self.select_tunnel = None
|
|
self.wg_switch = None
|
|
self.btn_stst = None
|
|
self.lb_tunnel = None
|
|
self.StrVar = None
|
|
self.a = None
|
|
self.l_box = None
|
|
|
|
def import_sl(self):
|
|
|
|
try:
|
|
filepath = filedialog.askopenfilename(initialdir=str(_u), title='Select Wireguard config File',
|
|
filetypes=[('WG config files', '*.conf')], )
|
|
|
|
with open(filepath, 'r') as file:
|
|
read = file.read()
|
|
path_split = filepath.split('/')
|
|
path_split1 = path_split[-1]
|
|
self.a = Tunnel.active()
|
|
|
|
if 'PrivateKey = ' in read and 'PublicKey = ' in read and 'Endpoint =' in read:
|
|
with open(filepath, 'r') as file:
|
|
key = Tunnel.con_to_dict(file)
|
|
pre_key = key[3]
|
|
if len(pre_key) != 0:
|
|
with open('/etc/wire_py/.keys', 'r') as readfile:
|
|
p_key = readfile.readlines()
|
|
if pre_key in p_key or pre_key + '\n' in p_key:
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/error.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Import Error'
|
|
msg_t = 'Tunnel already available!\nPlease use another file for import'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
else:
|
|
|
|
with open('/etc/wire_py/.keys', 'a') as keyfile:
|
|
keyfile.write(pre_key + '\r')
|
|
if len(path_split1) > 17:
|
|
p1 = shutil.copy(filepath, Path('/etc/wire_py/'))
|
|
path_split = path_split1[len(path_split1) - 17:]
|
|
os.rename(p1, Path('/etc/wire_py') / str(path_split))
|
|
new_conf = '/etc/wire_py/' + path_split
|
|
if self.a != '':
|
|
check_call(['nmcli', 'connection', 'down', Tunnel.active()])
|
|
Tunnel.label_empty(self)
|
|
|
|
subprocess.check_output(['nmcli', 'connection', 'import', 'type',
|
|
'wireguard', 'file', new_conf], text=True)
|
|
|
|
else:
|
|
shutil.copy(filepath, Path('/etc/wire_py/'))
|
|
if self.a != '':
|
|
check_call(['nmcli', 'connection', 'down', Tunnel.active()])
|
|
Tunnel.label_empty(self)
|
|
|
|
subprocess.check_output(['nmcli', 'connection', 'import', 'type',
|
|
'wireguard', 'file', filepath], text=True)
|
|
|
|
self.StrVar.set('')
|
|
self.a = Tunnel.active()
|
|
self.l_box.insert(0, self.a)
|
|
self.wg_autostart.configure(state='normal')
|
|
self.l_box.selection_clear(0, tk.END)
|
|
self.l_box.update()
|
|
self.l_box.selection_set(0)
|
|
|
|
def chk_a_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'To use the autostart, enable this '
|
|
'Checkbox')
|
|
|
|
def chk_a_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def list_info_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'List of available tunnels')
|
|
|
|
def list_info_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def del_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'Click to delete a Wireguard '
|
|
'Tunnel\nSelect from the list!')
|
|
|
|
def del_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def exp_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
' Click to export '
|
|
'all\nWireguard Tunnel to Zipfile')
|
|
|
|
def exp_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def rename_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'To rename a tunnel, you need to\n'
|
|
'select a tunnel from the list')
|
|
|
|
def rename_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.wg_autostart.bind('<Enter>', chk_a_enter)
|
|
self.wg_autostart.bind('<Leave>', chk_a_leave)
|
|
self.l_box.bind('<Enter>', list_info_enter)
|
|
self.l_box.bind('<Leave>', list_info_leave)
|
|
self.btn_tr.bind('<Enter>', del_enter)
|
|
self.btn_tr.bind('<Leave>', del_leave)
|
|
self.btn_exp.bind('<Enter>', exp_enter)
|
|
self.btn_exp.bind('<Leave>', exp_leave)
|
|
self.lb_rename.bind('<Enter>', rename_enter)
|
|
self.lb_rename.bind('<Leave>', rename_leave)
|
|
self.lb_rename.insert(0, 'Max. 12 characters!')
|
|
self.StrVar = tk.StringVar()
|
|
self.StrVar.set(self.a)
|
|
Tunnel.color(self)
|
|
StSt.stop(self)
|
|
wg_read = Path('/etc/wire_py') / str(self.a + '.conf')
|
|
with open(wg_read, 'r') as file_for_key:
|
|
data = Tunnel.con_to_dict(file_for_key)
|
|
|
|
''' Address Label '''
|
|
Tunnel.init_and_report(self, data)
|
|
Tunnel.show_data(self)
|
|
check_call(['nmcli', 'con', 'mod', self.a, 'connection.autoconnect', 'no'])
|
|
Path.chmod(wg_read, 0o600)
|
|
|
|
if 'PrivateKey = ' and 'Endpoint = ' not in read:
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/error.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Import Error'
|
|
msg_t = 'Oh... no valid Wireguard File!\nPlease select a valid Wireguard File'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
except EOFError:
|
|
pass
|
|
except TypeError:
|
|
pass
|
|
except FileNotFoundError:
|
|
pass
|
|
except subprocess.CalledProcessError:
|
|
|
|
print('Tunnel exist!')
|
|
|
|
|
|
class FrameWidgets(ttk.Frame):
|
|
def __init__(self, container, **kwargs):
|
|
super().__init__(container, **kwargs)
|
|
|
|
self.btn_stst = None
|
|
self.auto_con = None
|
|
self.enp = None
|
|
self.DNS = None
|
|
self.add = None
|
|
self.data = None
|
|
self.peer = None
|
|
self.lb_tunnel = None
|
|
self.wg_read = None
|
|
self.wg_vpn_start = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_vpn-start.png')
|
|
self.wg_vpn_stop = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_vpn-stop.png')
|
|
self.imp_pic = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_import.png')
|
|
self.tr_pic = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_trash.png')
|
|
self.exp_pic = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/48/wg_export.png')
|
|
self.warning_pic = tk.PhotoImage(file=r'/usr/share/icons/wp-icons/64/error.png')
|
|
|
|
''' Show active Tunnel '''
|
|
self.a = Tunnel.active()
|
|
|
|
''' Label Frame 1 '''
|
|
self.lb_frame_btn_lbox = ttk.Frame(self)
|
|
self.lb_frame_btn_lbox.configure(relief='flat')
|
|
self.lb_frame_btn_lbox.grid(column=0, rowspan=3, row=1)
|
|
self.columnconfigure(0, weight=1)
|
|
self.rowconfigure(1, weight=1)
|
|
|
|
''' Label Frame 2 '''
|
|
self.lb_frame = ttk.Frame(self)
|
|
self.lb_frame.configure(relief='solid')
|
|
self.lb_frame.grid(column=2, row=2, sticky='snew', padx=20, pady=5)
|
|
self.columnconfigure(2, weight=1)
|
|
self.rowconfigure(2, weight=1)
|
|
|
|
''' Label Frame 3 '''
|
|
self.lb_frame2 = ttk.Frame(self)
|
|
self.lb_frame2.configure(relief='solid')
|
|
self.lb_frame2.grid(column=2, row=3, sticky='snew', padx=20, pady=5)
|
|
self.columnconfigure(2, weight=1)
|
|
self.rowconfigure(3, weight=1)
|
|
|
|
''' Show active Label '''
|
|
self.select_tunnel = None
|
|
self.lb = ttk.Label(self, text='Active: ')
|
|
self.lb.config(font=('Ubuntu', 11, 'bold'))
|
|
self.lb.grid(column=2, row=1, padx=15, pady=4, sticky='w')
|
|
self.columnconfigure(2, weight=1)
|
|
self.rowconfigure(0, weight=1)
|
|
|
|
''' Label to Show active Tunnel '''
|
|
self.StrVar = tk.StringVar(value=self.a)
|
|
Tunnel.color(self)
|
|
|
|
''' Interface Label '''
|
|
self.interface = ttk.Label(self.lb_frame, text='Interface')
|
|
self.interface.grid(column=0, row=3, sticky='we', padx=120)
|
|
self.interface.config(font=('Ubuntu', 9))
|
|
|
|
''' Peer Label '''
|
|
self.peer = ttk.Label(self.lb_frame2, text='Peer')
|
|
self.peer.config(font=('Ubuntu', 9))
|
|
self.peer.grid(column=0, row=4, sticky='we', padx=130)
|
|
|
|
''' Listbox with Scrollbar '''
|
|
|
|
def enable_check_box(_):
|
|
|
|
if self.l_box.size() != 0:
|
|
self.wg_autostart.configure(state='normal')
|
|
self.lb_rename.config(state='normal')
|
|
self.lb_rename.delete(0, tk.END)
|
|
self.btn_rename.config(state='normal')
|
|
|
|
self.l_box = tk.Listbox(self.lb_frame_btn_lbox, selectmode='single')
|
|
self.l_box.config(relief='ridge', font=('Ubuntu', 12, 'bold'))
|
|
self.l_box.grid(column=1, rowspan=4, row=0, sticky='ns')
|
|
self.l_box.event_add('<<ClickEvent>>', '<Button-1>')
|
|
self.l_box.bind('<<ClickEvent>>', enable_check_box)
|
|
self.scrollbar = ttk.Scrollbar(self.lb_frame_btn_lbox, orient='vertical', command=self.l_box.yview)
|
|
self.scrollbar.grid(column=1, rowspan=4, row=0, sticky='nse')
|
|
self.l_box.configure(yscrollcommand=self.scrollbar.set)
|
|
self.rowconfigure(0, weight=1)
|
|
|
|
''' Tunnel List '''
|
|
self.tl = Tunnel.list()
|
|
for tunnels in self.tl:
|
|
self.l_box.insert("end", tunnels)
|
|
self.l_box.update()
|
|
|
|
def list_empty_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'You must first import\na Wireguard tunnel')
|
|
|
|
def list_empty_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def list_not_empty_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root, 'Select a Tunnel')
|
|
|
|
def list_not_empty_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
if self.l_box.size() == 0:
|
|
self.l_box.bind('<Enter>', list_empty_enter)
|
|
self.l_box.bind('<Leave>', list_empty_leave)
|
|
else:
|
|
self.l_box.bind('<Enter>', list_not_empty_enter)
|
|
self.l_box.bind('<Leave>', list_not_empty_leave)
|
|
|
|
''' Button Vpn '''
|
|
if self.a != '':
|
|
StSt.stop(self)
|
|
wg_read = Path('/etc/wire_py') / str(self.a + '.conf')
|
|
with open(wg_read, 'r') as file:
|
|
data = Tunnel.con_to_dict(file)
|
|
|
|
''' Address Label '''
|
|
Tunnel.init_and_report(self, data)
|
|
Tunnel.show_data(self)
|
|
else:
|
|
StSt.start(self)
|
|
|
|
''' Address Label '''
|
|
self.add = tk.StringVar()
|
|
self.DNS = tk.StringVar()
|
|
self.enp = tk.StringVar()
|
|
Tunnel.label_empty(self)
|
|
Tunnel.show_data(self)
|
|
|
|
''' Button Import '''
|
|
self.btn_i = ttk.Button(self.lb_frame_btn_lbox,
|
|
image=self.imp_pic, command=lambda: TLImp.import_sl(self),
|
|
padding=0)
|
|
self.btn_i.grid(column=0, row=1, padx=15, pady=8)
|
|
|
|
def imp_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'Click to import a Wireguard Tunnel')
|
|
|
|
def imp_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.btn_i.bind('<Enter>', imp_enter)
|
|
self.btn_i.bind('<Leave>', imp_leave)
|
|
|
|
def delete():
|
|
|
|
try:
|
|
self.select_tunnel = self.l_box.curselection()
|
|
select_tl = self.l_box.get(self.select_tunnel[0])
|
|
with open('/etc/wire_py/' + select_tl + '.conf', 'r+') as file2:
|
|
key = Tunnel.con_to_dict(file2)
|
|
pre_key = key[3]
|
|
check_call(['nmcli', 'connection', 'delete', select_tl])
|
|
self.l_box.delete(self.select_tunnel[0])
|
|
with open(path_to_file2, 'r') as set_file6:
|
|
lines6 = set_file6.readlines()
|
|
if select_tl == lines6[7].strip() and 'off' not in lines6[7].strip():
|
|
lines6[7] = 'off'
|
|
with open(path_to_file2, 'w') as set_file7:
|
|
set_file7.writelines(lines6)
|
|
self.selected_option.set(0)
|
|
self.autoconnect_var.set('no Autoconnect')
|
|
|
|
Path.unlink(Path('/etc/wire_py') / str(select_tl + '.conf'))
|
|
with open('/etc/wire_py/.keys', 'r') as readfile:
|
|
with open('/etc/wire_py/.keys2', 'w') as writefile:
|
|
for line in readfile:
|
|
if pre_key not in line.strip("\n"):
|
|
writefile.write(line)
|
|
file_one = Path('/etc/wire_py/.keys2')
|
|
file_two = file_one.with_name('.keys')
|
|
file_one.replace(file_two)
|
|
self.wg_autostart.configure(state='disabled')
|
|
|
|
''' for disable checkbox when Listbox empty '''
|
|
|
|
def empty_list_start_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'No tunnels to start in the list')
|
|
|
|
def empty_list_start_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
if self.l_box.size() == 0:
|
|
self.wg_autostart.configure(state='disabled')
|
|
self.lb_rename.configure(state='disabled')
|
|
self.l_box.bind('<Enter>', list_empty_enter)
|
|
self.l_box.bind('<Leave>', list_empty_leave)
|
|
self.wg_autostart.bind('<Enter>', chk_enter)
|
|
self.wg_autostart.bind('<Leave>', chk_leave)
|
|
self.btn_tr.bind('<Enter>', empty_list_enter)
|
|
self.btn_tr.bind('<Leave>', empty_list_leave)
|
|
self.btn_exp.bind('<Enter>', empty_list_enter)
|
|
self.btn_exp.bind('<Leave>', empty_list_leave)
|
|
self.btn_stst.bind('<Enter>', empty_list_start_enter)
|
|
self.btn_stst.bind('<Leave>', empty_list_start_leave)
|
|
self.lb_rename.bind('<Enter>', rename_no_active_enter)
|
|
self.lb_rename.bind('<Leave>', rename_no_active_leave)
|
|
self.lb_rename.insert(0, 'Max. 12 characters!')
|
|
|
|
if self.a != '' and self.a == select_tl:
|
|
self.StrVar.set(value='')
|
|
StSt.start(self)
|
|
self.l_box.update()
|
|
|
|
''' Address Label '''
|
|
self.add.set('')
|
|
self.DNS.set('')
|
|
self.enp.set('')
|
|
return select_tl
|
|
|
|
except IndexError:
|
|
|
|
if self.l_box.size() != 0:
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Select tunnel'
|
|
msg_t = 'Please select a tunnel from the list.'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
else:
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Select tunnel'
|
|
msg_t = 'Please first import tunnel.'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
''' Button Trash '''
|
|
self.btn_tr = ttk.Button(self.lb_frame_btn_lbox, image=self.tr_pic, command=delete, padding=0,
|
|
style='CButton.TButton')
|
|
self.btn_tr.grid(column=0, row=2, padx=15, pady=8)
|
|
|
|
def empty_list_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'No tunnels to delete in the list')
|
|
|
|
def empty_list_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def del_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'Click to delete a Wireguard Tunnel\nSelect from the list!')
|
|
|
|
def del_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
if self.l_box.size() == 0:
|
|
self.btn_tr.bind('<Enter>', empty_list_enter)
|
|
self.btn_tr.bind('<Leave>', empty_list_leave)
|
|
else:
|
|
self.btn_tr.bind('<Enter>', del_enter)
|
|
self.btn_tr.bind('<Leave>', del_leave)
|
|
|
|
''' Button Export '''
|
|
self.btn_exp = ttk.Button(self.lb_frame_btn_lbox, image=self.exp_pic, command=Tunnel.export,
|
|
padding=0)
|
|
self.btn_exp.grid(column=0, row=3, padx=15, pady=8)
|
|
|
|
def empty_list_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'No Tunnels in List for Export')
|
|
|
|
def empty_list_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def exp_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
' Click to export all\nWireguard Tunnel to Zipfile')
|
|
|
|
def exp_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
if self.l_box.size() == 0:
|
|
self.btn_exp.bind('<Enter>', empty_list_enter)
|
|
self.btn_exp.bind('<Leave>', empty_list_leave)
|
|
else:
|
|
self.btn_exp.bind('<Enter>', exp_enter)
|
|
self.btn_exp.bind('<Leave>', exp_leave)
|
|
|
|
''' Label Entry '''
|
|
self.lb_rename = ttk.Entry(self, width=20)
|
|
self.lb_rename.grid(column=2, row=4, padx=30, pady=15, sticky='nw')
|
|
self.lb_rename.insert(0, 'Max. 12 characters!')
|
|
self.lb_rename.config(state='disable')
|
|
|
|
def rename_no_active_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
def rename_no_active_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'To rename a tunnel, at least one must be in the list')
|
|
|
|
def rename_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'To rename a tunnel, you need to\nselect a tunnel from the list')
|
|
|
|
def rename_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
if self.l_box.size() != 0:
|
|
self.lb_rename.bind('<Enter>', rename_enter)
|
|
self.lb_rename.bind('<Leave>', rename_leave)
|
|
else:
|
|
self.lb_rename.bind('<Enter>', rename_no_active_enter)
|
|
self.lb_rename.bind('<Leave>', rename_no_active_leave)
|
|
|
|
def tl_rename():
|
|
special_characters = ['\\', '/', '{', '}', ' ']
|
|
|
|
if len(self.lb_rename.get()) > 12:
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Renaming not possible'
|
|
msg_t = 'The new name may contain only 12 characters.'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
elif len(self.lb_rename.get()) == 0:
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Renaming not possible'
|
|
msg_t = 'At least one character must be entered.'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
elif any(ch in special_characters for ch in self.lb_rename.get()):
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Renaming not possible'
|
|
msg_t = 'No valid sign. These must not be used.\nBlank, Slash, Backslash and { }\n'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
else:
|
|
|
|
try:
|
|
self.select_tunnel = self.l_box.curselection()
|
|
select_tl = self.l_box.get(self.select_tunnel[0])
|
|
|
|
''' nmcli connection modify old connection.id iphone '''
|
|
check_call(['nmcli', 'connection', 'modify', select_tl, 'connection.id', self.lb_rename.get()])
|
|
source = Path('/etc/wire_py') / str(select_tl + '.conf')
|
|
destination = source.with_name(str(self.lb_rename.get() + '.conf'))
|
|
source.replace(destination)
|
|
self.l_box.delete(self.select_tunnel[0])
|
|
self.l_box.insert("end", self.lb_rename.get())
|
|
self.l_box.update()
|
|
new_a_connect = self.lb_rename.get()
|
|
self.lb_rename.delete(0, tk.END)
|
|
if self.a != '' and self.a == select_tl:
|
|
self.a = Tunnel.active()
|
|
self.StrVar.set(value=self.a)
|
|
with open(path_to_file2, 'r') as set_file5:
|
|
lines5 = set_file5.readlines()
|
|
if select_tl == lines5[7].strip() and 'off' not in lines5[7].strip():
|
|
lines5[7] = new_a_connect
|
|
with open(path_to_file2, 'w') as theme_set5:
|
|
theme_set5.writelines(lines5)
|
|
self.autoconnect_var.set(value=new_a_connect)
|
|
|
|
return select_tl
|
|
|
|
except IndexError:
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Renaming not possible'
|
|
msg_t = 'Please select a tunnel from the list.'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
''' Button Rename '''
|
|
self.btn_rename = ttk.Button(self, text='Rename', state='disable', command=tl_rename, padding=4,
|
|
style='RnButton.TButton')
|
|
self.btn_rename.grid(column=2, row=4, padx=20, pady=15, sticky='ne')
|
|
|
|
''' Check Buttons '''
|
|
self.selected_option = tk.IntVar()
|
|
self.autoconnect_var = tk.StringVar()
|
|
self.autoconnect_var.set(self.auto_con)
|
|
''' Frame for Labels, Entry and Button'''
|
|
self.autoconnect = ttk.Label(self, textvariable=self.autoconnect_var)
|
|
self.autoconnect.config(font=('Ubuntu', 11))
|
|
self.autoconnect.grid(column=0, row=4, sticky='ne', padx=10, pady=15)
|
|
self.wg_autostart = ttk.Checkbutton(self,
|
|
text='Autoconnect on:',
|
|
variable=self.selected_option,
|
|
command=lambda: FileHandle.box_set(self))
|
|
self.wg_autostart.grid(column=0, row=4, pady=15, padx=15, sticky='nw')
|
|
|
|
def chk_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'You must have at least one\n'
|
|
'tunnel in the list,to use the autostart')
|
|
|
|
def chk_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
if self.l_box.size() >= 1 and len(self.l_box.curselection()) >= 1:
|
|
def chk_a_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'To use the autostart, enable this Checkbox')
|
|
|
|
def chk_a_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.wg_autostart.bind('<Enter>', chk_a_enter)
|
|
self.wg_autostart.bind('<Leave>', chk_a_leave)
|
|
|
|
if self.l_box.size() == 0:
|
|
self.wg_autostart.bind('<Enter>', chk_enter)
|
|
self.wg_autostart.bind('<Leave>', chk_leave)
|
|
else:
|
|
def chk_a_enter(event):
|
|
""" The mouse moves into the entry widget """
|
|
window.my_tool_tip = MyToolTip(event.x_root, event.y_root,
|
|
'To use the autostart, a tunnel must be selected from the list')
|
|
|
|
def chk_a_leave(_):
|
|
""" The mouse moves from the entry widget """
|
|
''' Remove Tool-Tip '''
|
|
window.my_tool_tip.destroy()
|
|
|
|
self.wg_autostart.bind('<Enter>', chk_a_enter)
|
|
self.wg_autostart.bind('<Leave>', chk_a_leave)
|
|
|
|
OnOff.on_off(self)
|
|
|
|
def wg_switch(self):
|
|
self.a = Tunnel.active()
|
|
try:
|
|
if self.a == '':
|
|
|
|
StSt.start(self)
|
|
self.select_tunnel = self.l_box.curselection()
|
|
select_tl = self.l_box.get(self.select_tunnel[0])
|
|
check_call(['nmcli', 'connection', 'up', select_tl])
|
|
wg_read = Path('/etc/wire_py') / str(select_tl + '.conf')
|
|
with open(wg_read, 'r') as file:
|
|
data = Tunnel.con_to_dict(file)
|
|
|
|
''' Address Label '''
|
|
Tunnel.init_and_report(self, data)
|
|
Tunnel.show_data(self)
|
|
|
|
''' Button Start/Stop '''
|
|
StSt.stop(self)
|
|
self.a = Tunnel.active()
|
|
self.StrVar = tk.StringVar()
|
|
self.StrVar.set(self.a)
|
|
Tunnel.color(self)
|
|
|
|
elif self.a != '':
|
|
|
|
''' Button Start/Stop '''
|
|
StSt.stop(self)
|
|
check_call(['nmcli', 'connection', 'down', self.a])
|
|
|
|
''' Button Start/Stop '''
|
|
StSt.start(self)
|
|
self.a = Tunnel.active()
|
|
self.StrVar.set('')
|
|
Tunnel.color(self)
|
|
|
|
''' Address Label '''
|
|
self.add.set('')
|
|
self.DNS.set('')
|
|
self.enp.set('')
|
|
Tunnel.show_data(self)
|
|
|
|
except IndexError:
|
|
|
|
if self.l_box.size() != 0:
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Select tunnel'
|
|
msg_t = 'Please select a tunnel from the list.'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
else:
|
|
|
|
"""img_w, img_i, w_title, w_txt hand over"""
|
|
iw = r'/usr/share/icons/wp-icons/64/info.png'
|
|
ii = r'/usr/share/icons/wp-icons/48/wg_msg.png'
|
|
wt = 'Select tunnel'
|
|
msg_t = 'Please first import tunnel.'
|
|
msg_window(iw, ii, wt, msg_t)
|
|
|
|
|
|
class MyToolTip(tk.Toplevel):
|
|
TIP_X_OFFSET = 20
|
|
TIP_Y_OFFSET = 20
|
|
if not tips:
|
|
AUTO_CLEAR_TIME = 0
|
|
else:
|
|
AUTO_CLEAR_TIME = 900 # Millisecond. (1/90 sec.)
|
|
|
|
def __init__(self, x_pos, y_pos, message=None, auto_clear=True):
|
|
self.x_pos = x_pos
|
|
self.y_pos = y_pos
|
|
self.message = message
|
|
self.auto_clear = auto_clear
|
|
|
|
tk.Toplevel.__init__(self)
|
|
self.overrideredirect(True)
|
|
|
|
self.message_label = ttk.Label(self, compound='left', text=self.message, padding=6)
|
|
self.message_label.pack()
|
|
|
|
self.geometry("+%d+%d" % (self.x_pos + self.TIP_X_OFFSET,
|
|
self.y_pos + self.TIP_X_OFFSET))
|
|
|
|
if self.auto_clear:
|
|
self.after(self.AUTO_CLEAR_TIME, self.clear_tip)
|
|
|
|
def clear_tip(self):
|
|
"""Remove Tool-Tip"""
|
|
self.destroy()
|
|
|
|
|
|
if __name__ == '__main__':
|
|
window = MainWindow()
|
|
"""
|
|
the hidden files are hidden in Filedialog
|
|
"""
|
|
try:
|
|
window.tk.call('tk_getOpenFile', '-foobarbaz')
|
|
except TclError:
|
|
pass
|
|
|
|
window.tk.call('set', '::tk::dialog::file::showHiddenBtn', '0')
|
|
window.tk.call('set', '::tk::dialog::file::showHiddenVar', '0')
|
|
window.mainloop()
|