Compare commits
	
		
			84 Commits
		
	
	
		
			090842ab28
			...
			27-04-2025
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 4eb9d6acd4 | |||
| 97ea07d34b | |||
| cd625d173d | |||
| 950e04a246 | |||
| 1a853d4ff1 | |||
| f6204c9071 | |||
| f9ecd54e0a | |||
| 67ff24f0b6 | |||
| af702f297b | |||
| d2a57b329b | |||
| 87943b2489 | |||
| c43c12f961 | |||
| 3bab0710a4 | |||
| aa66f4dc68 | |||
| c220951781 | |||
| 6c0662c62c | |||
| 5753a35d6c | |||
| 47bdfbfb17 | |||
| 2a3bf2bbcb | |||
| 8896f59efd | |||
| c58a630e25 | |||
| dc6f8cb094 | |||
| dcda647e90 | |||
| 2a995eabd5 | |||
| ea750b0cfc | |||
| 980ce3c7a7 | |||
| 28cc423138 | |||
| 39550b392d | |||
| 582ef21042 | |||
| ca58ac86a4 | |||
| 177fa1cc34 | |||
| ca1a7a6b94 | |||
| a49c5b00e1 | |||
| 75a247797d | |||
| 4b019b0c1f | |||
| ab78eb4f59 | |||
| 9f3f0246b4 | |||
| 5302aae807 | |||
| 13832d916f | |||
| 1667682c9d | |||
| 8771be760d | |||
| 97bf9df041 | |||
| 1bba45a6c1 | |||
| a5eb6293c6 | |||
| 6e54529c0f | |||
| e28235af4b | |||
| 862cc91fa1 | |||
| 5fe6fd3f29 | |||
| 48a48ffd4c | |||
| 1278b02a95 | |||
| 22abe9f8b9 | |||
| 52f22beec6 | |||
| 8754e7ecda | |||
| a6742a0c0e | |||
| 6b75713121 | |||
| 762694aa94 | |||
| cc54f329fe | |||
| 8c01c1b984 | |||
| ebafdbdcb3 | |||
| ce4f6c84dc | |||
| b68908eaef | |||
| c4565ed169 | |||
| 485cfff457 | |||
| 1a0fc74a80 | |||
| 971c86c97f | |||
| 66dd27cc64 | |||
| cf71eae7d3 | |||
| 1f80d5342c | |||
| e719a1f23f | |||
| a0da356cec | |||
| ac87e0a4a7 | |||
| 6a3a982057 | |||
| aab90eec70 | |||
| 511d5f72df | |||
| ed93e99026 | |||
| 10c3b07053 | |||
| e9814f2b60 | |||
| c470783ff2 | |||
| 2fac03a927 | |||
| 03a7e48018 | |||
| 8a32065676 | |||
| bc0d46d633 | |||
| 80c63eaf78 | |||
| 4bfc8c318e | 
							
								
								
									
										3
									
								
								.idea/dictionaries/project.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <component name="ProjectDictionaryState"> | ||||||
|  |   <dictionary name="project" /> | ||||||
|  | </component> | ||||||
							
								
								
									
										2
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
								
							
							
						
						| @@ -3,5 +3,5 @@ | |||||||
|   <component name="Black"> |   <component name="Black"> | ||||||
|     <option name="sdkName" value="Python 3.12 (wire-py)" /> |     <option name="sdkName" value="Python 3.12 (wire-py)" /> | ||||||
|   </component> |   </component> | ||||||
|   <component name="ProjectRootManager" version="2" project-jdk-name="Python 3.12 (wire-py)" project-jdk-type="Python SDK" /> |   <component name="ProjectRootManager" version="2" project-jdk-name="Python 3.12" project-jdk-type="Python SDK" /> | ||||||
| </project> | </project> | ||||||
							
								
								
									
										2
									
								
								.idea/wire-py.iml
									
									
									
										generated
									
									
									
								
							
							
						
						| @@ -4,7 +4,7 @@ | |||||||
|     <content url="file://$MODULE_DIR$"> |     <content url="file://$MODULE_DIR$"> | ||||||
|       <excludeFolder url="file://$MODULE_DIR$/.venv" /> |       <excludeFolder url="file://$MODULE_DIR$/.venv" /> | ||||||
|     </content> |     </content> | ||||||
|     <orderEntry type="inheritedJdk" /> |     <orderEntry type="jdk" jdkName="Python 3.12" jdkType="Python SDK" /> | ||||||
|     <orderEntry type="sourceFolder" forTests="false" /> |     <orderEntry type="sourceFolder" forTests="false" /> | ||||||
|   </component> |   </component> | ||||||
| </module> | </module> | ||||||
							
								
								
									
										280
									
								
								.idea/workspace.xml
									
									
									
										generated
									
									
									
								
							
							
						
						| @@ -4,16 +4,9 @@ | |||||||
|     <option name="autoReloadType" value="SELECTIVE" /> |     <option name="autoReloadType" value="SELECTIVE" /> | ||||||
|   </component> |   </component> | ||||||
|   <component name="ChangeListManager"> |   <component name="ChangeListManager"> | ||||||
|     <list default="true" id="940e1630-c825-4d4c-be80-bc11f543c122" name="Changes" comment="- Optimize Class and Tooltip"> |     <list default="true" id="940e1630-c825-4d4c-be80-bc11f543c122" name="Changes" comment=" - Update Translate Files"> | ||||||
|       <change afterPath="$PROJECT_DIR$/run_as" afterDir="false" /> |       <change beforePath="$PROJECT_DIR$/cls_mth_fc.py" beforeDir="false" afterPath="$PROJECT_DIR$/cls_mth_fc.py" afterDir="false" /> | ||||||
|       <change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" /> |       <change beforePath="$PROJECT_DIR$/wirepy.py" beforeDir="false" afterPath="$PROJECT_DIR$/wirepy.py" afterDir="false" /> | ||||||
|       <change beforePath="$PROJECT_DIR$/Changelog" beforeDir="false" afterPath="$PROJECT_DIR$/Changelog" afterDir="false" /> |  | ||||||
|       <change beforePath="$PROJECT_DIR$/install" beforeDir="false" afterPath="$PROJECT_DIR$/install" afterDir="false" /> |  | ||||||
|       <change beforePath="$PROJECT_DIR$/settings" beforeDir="false" /> |  | ||||||
|       <change beforePath="$PROJECT_DIR$/start_wg.py" beforeDir="false" afterPath="$PROJECT_DIR$/start_wg.py" afterDir="false" /> |  | ||||||
|       <change beforePath="$PROJECT_DIR$/theme" beforeDir="false" /> |  | ||||||
|       <change beforePath="$PROJECT_DIR$/wg_func.py" beforeDir="false" afterPath="$PROJECT_DIR$/wg_func.py" afterDir="false" /> |  | ||||||
|       <change beforePath="$PROJECT_DIR$/wg_main.py" beforeDir="false" afterPath="$PROJECT_DIR$/wg_main.py" afterDir="false" /> |  | ||||||
|     </list> |     </list> | ||||||
|     <option name="SHOW_DIALOG" value="false" /> |     <option name="SHOW_DIALOG" value="false" /> | ||||||
|     <option name="HIGHLIGHT_CONFLICTS" value="true" /> |     <option name="HIGHLIGHT_CONFLICTS" value="true" /> | ||||||
| @@ -30,21 +23,20 @@ | |||||||
|   <component name="Git.Settings"> |   <component name="Git.Settings"> | ||||||
|     <option name="RECENT_BRANCH_BY_REPOSITORY"> |     <option name="RECENT_BRANCH_BY_REPOSITORY"> | ||||||
|       <map> |       <map> | ||||||
|         <entry key="$PROJECT_DIR$" value="main" /> |         <entry key="$PROJECT_DIR$" value="1.11.0824" /> | ||||||
|       </map> |       </map> | ||||||
|     </option> |     </option> | ||||||
|     <option name="RECENT_GIT_ROOT_PATH" value="$PROJECT_DIR$" /> |     <option name="RECENT_GIT_ROOT_PATH" value="$PROJECT_DIR$" /> | ||||||
|     <option name="UPDATE_TYPE" value="REBASE" /> |     <option name="UPDATE_TYPE" value="REBASE" /> | ||||||
|   </component> |   </component> | ||||||
|   <component name="HighlightingSettingsPerFile"> |   <component name="HighlightingSettingsPerFile"> | ||||||
|     <setting file="file://$PROJECT_DIR$/wg_func.py" root0="SKIP_INSPECTION" /> |     <setting file="file:///usr/local/bin/ssl_decrypt.py" root0="SKIP_INSPECTION" /> | ||||||
|     <setting file="file://$PROJECT_DIR$/wg_main.py" root0="FORCE_HIGHLIGHTING" /> |  | ||||||
|   </component> |   </component> | ||||||
|   <component name="ProjectColorInfo">{ |   <component name="ProjectColorInfo">{ | ||||||
|   "associatedIndex": 3 |   "associatedIndex": 3 | ||||||
| }</component> | }</component> | ||||||
|   <component name="ProjectId" id="2kSbZdjOvr0wsVJSNcaMwSfVaxR" /> |   <component name="ProjectId" id="2kSbZdjOvr0wsVJSNcaMwSfVaxR" /> | ||||||
|   <component name="ProjectLevelVcsManager" settingsEditedManually="true"> |   <component name="ProjectLevelVcsManager"> | ||||||
|     <ConfirmationsSetting value="2" id="Add" /> |     <ConfirmationsSetting value="2" id="Add" /> | ||||||
|   </component> |   </component> | ||||||
|   <component name="ProjectViewState"> |   <component name="ProjectViewState"> | ||||||
| @@ -55,6 +47,7 @@ | |||||||
|   "keyToString": { |   "keyToString": { | ||||||
|     "ASKED_ADD_EXTERNAL_FILES": "true", |     "ASKED_ADD_EXTERNAL_FILES": "true", | ||||||
|     "Python.INSTALL.executor": "Run", |     "Python.INSTALL.executor": "Run", | ||||||
|  |     "Python.cls_mth_fc.executor": "Run", | ||||||
|     "Python.install.executor": "Run", |     "Python.install.executor": "Run", | ||||||
|     "Python.main.executor": "Run", |     "Python.main.executor": "Run", | ||||||
|     "Python.messagebox.executor": "Run", |     "Python.messagebox.executor": "Run", | ||||||
| @@ -62,16 +55,19 @@ | |||||||
|     "Python.testtheme.executor": "Run", |     "Python.testtheme.executor": "Run", | ||||||
|     "Python.wg_func.executor": "Run", |     "Python.wg_func.executor": "Run", | ||||||
|     "Python.wg_main.executor": "Run", |     "Python.wg_main.executor": "Run", | ||||||
|  |     "Python.wirepy.executor": "Run", | ||||||
|     "RunOnceActivity.ShowReadmeOnStart": "true", |     "RunOnceActivity.ShowReadmeOnStart": "true", | ||||||
|  |     "RunOnceActivity.git.unshallow": "true", | ||||||
|     "Shell Script.install.executor": "Run", |     "Shell Script.install.executor": "Run", | ||||||
|     "Shell Script.run_as.executor": "Run", |     "Shell Script.run_as.executor": "Run", | ||||||
|     "git-widget-placeholder": "main", |     "git-widget-placeholder": "21-04-2025-new-tooltip", | ||||||
|     "last_opened_file_path": "/home/punix/Pyapps/wire-py", |     "last_opened_file_path": "/home/punix/Pyapps/wire-py", | ||||||
|     "settings.editor.selected.configurable": "reference.settingsdialog.IDE.editor.colors" |     "settings.editor.selected.configurable": "ml.llm.LLMConfigurable" | ||||||
|   } |   } | ||||||
| }</component> | }</component> | ||||||
|   <component name="RecentsManager"> |   <component name="RecentsManager"> | ||||||
|     <key name="CopyFile.RECENT_KEYS"> |     <key name="CopyFile.RECENT_KEYS"> | ||||||
|  |       <recent name="$PROJECT_DIR$/lx-icons" /> | ||||||
|       <recent name="$PROJECT_DIR$" /> |       <recent name="$PROJECT_DIR$" /> | ||||||
|     </key> |     </key> | ||||||
|     <key name="MoveFile.RECENT_KEYS"> |     <key name="MoveFile.RECENT_KEYS"> | ||||||
| @@ -81,7 +77,7 @@ | |||||||
|       <recent name="$PROJECT_DIR$/wire-py" /> |       <recent name="$PROJECT_DIR$/wire-py" /> | ||||||
|     </key> |     </key> | ||||||
|   </component> |   </component> | ||||||
|   <component name="RunManager" selected="Python.wg_main"> |   <component name="RunManager" selected="Python.wirepy"> | ||||||
|     <configuration name="start_wg" type="PythonConfigurationType" factoryName="Python" temporary="true" nameIsGenerated="true"> |     <configuration name="start_wg" type="PythonConfigurationType" factoryName="Python" temporary="true" nameIsGenerated="true"> | ||||||
|       <module name="wire-py" /> |       <module name="wire-py" /> | ||||||
|       <option name="ENV_FILES" value="" /> |       <option name="ENV_FILES" value="" /> | ||||||
| @@ -126,8 +122,31 @@ | |||||||
|       <option name="INPUT_FILE" value="" /> |       <option name="INPUT_FILE" value="" /> | ||||||
|       <method v="2" /> |       <method v="2" /> | ||||||
|     </configuration> |     </configuration> | ||||||
|  |     <configuration name="wirepy" type="PythonConfigurationType" factoryName="Python" temporary="true" nameIsGenerated="true"> | ||||||
|  |       <module name="wire-py" /> | ||||||
|  |       <option name="ENV_FILES" value="" /> | ||||||
|  |       <option name="INTERPRETER_OPTIONS" value="" /> | ||||||
|  |       <option name="PARENT_ENVS" value="true" /> | ||||||
|  |       <envs> | ||||||
|  |         <env name="PYTHONUNBUFFERED" value="1" /> | ||||||
|  |       </envs> | ||||||
|  |       <option name="SDK_HOME" value="" /> | ||||||
|  |       <option name="WORKING_DIRECTORY" value="$PROJECT_DIR$" /> | ||||||
|  |       <option name="IS_MODULE_SDK" value="true" /> | ||||||
|  |       <option name="ADD_CONTENT_ROOTS" value="true" /> | ||||||
|  |       <option name="ADD_SOURCE_ROOTS" value="true" /> | ||||||
|  |       <option name="SCRIPT_NAME" value="$PROJECT_DIR$/wirepy.py" /> | ||||||
|  |       <option name="PARAMETERS" value="" /> | ||||||
|  |       <option name="SHOW_COMMAND_LINE" value="false" /> | ||||||
|  |       <option name="EMULATE_TERMINAL" value="false" /> | ||||||
|  |       <option name="MODULE_MODE" value="false" /> | ||||||
|  |       <option name="REDIRECT_INPUT" value="false" /> | ||||||
|  |       <option name="INPUT_FILE" value="" /> | ||||||
|  |       <method v="2" /> | ||||||
|  |     </configuration> | ||||||
|     <recent_temporary> |     <recent_temporary> | ||||||
|       <list> |       <list> | ||||||
|  |         <item itemvalue="Python.wirepy" /> | ||||||
|         <item itemvalue="Python.start_wg" /> |         <item itemvalue="Python.start_wg" /> | ||||||
|       </list> |       </list> | ||||||
|     </recent_temporary> |     </recent_temporary> | ||||||
| @@ -135,7 +154,7 @@ | |||||||
|   <component name="SharedIndexes"> |   <component name="SharedIndexes"> | ||||||
|     <attachedChunks> |     <attachedChunks> | ||||||
|       <set> |       <set> | ||||||
|         <option value="bundled-python-sdk-0e3be3396995-c546a90a8094-com.jetbrains.pycharm.community.sharedIndexes.bundled-PC-242.23339.19" /> |         <option value="bundled-python-sdk-348a24fa61fa-5312c7369657-com.jetbrains.pycharm.community.sharedIndexes.bundled-PC-251.23774.444" /> | ||||||
|       </set> |       </set> | ||||||
|     </attachedChunks> |     </attachedChunks> | ||||||
|   </component> |   </component> | ||||||
| @@ -148,102 +167,6 @@ | |||||||
|       <option name="presentableId" value="Default" /> |       <option name="presentableId" value="Default" /> | ||||||
|       <updated>1723279982210</updated> |       <updated>1723279982210</updated> | ||||||
|     </task> |     </task> | ||||||
|     <task id="LOCAL-00017" summary="remove a ',' in DNS Name"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1724576599289</created> |  | ||||||
|       <option name="number" value="00017" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00017" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1724576599289</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00018" summary="fix when Filname > 17 first copy file after rename"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1724593165879</created> |  | ||||||
|       <option name="number" value="00018" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00018" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1724593165879</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00019" summary="add export Tunnel as zip"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1724610514657</created> |  | ||||||
|       <option name="number" value="00019" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00019" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1724610514658</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00020" summary="columnconfigure on all widgets set"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1724778892233</created> |  | ||||||
|       <option name="number" value="00020" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00020" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1724778892233</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00021" summary="little fixes a " " to ' '"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725119445803</created> |  | ||||||
|       <option name="number" value="00021" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00021" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725119445803</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00022" summary="add class FileHandle,
add Label to show autoconnect Tunnel
disable checkbox when Listbox is empty or no select Tunnel"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725391658456</created> |  | ||||||
|       <option name="number" value="00022" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00022" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725391658456</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00023" summary="add if question and add autoconnect, autoconnect_var to class Filehandle in box_set no finish!"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725434328731</created> |  | ||||||
|       <option name="number" value="00023" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00023" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725434328731</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00024" summary="fixes on empty Listbox now is disable and now works autoconnect label with read and write, delete works now with read and write"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725475967338</created> |  | ||||||
|       <option name="number" value="00024" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00024" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725475967338</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00025" summary="add Frame widget 3
for Buttons and Listbox with Scrollbar.
all Widgets new format
delete works now of disable checkbox when Listbox empty (part two)"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725567453540</created> |  | ||||||
|       <option name="number" value="00025" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00025" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725567453540</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00026" summary="little fixes"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725639633873</created> |  | ||||||
|       <option name="number" value="00026" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00026" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725639633873</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00027" summary="little fixes"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725746027132</created> |  | ||||||
|       <option name="number" value="00027" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00027" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725746027132</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00028" summary="little fixes replace os.system with check_call
first steps in install Script add wg_start.service file"> |  | ||||||
|       <option name="closed" value="true" /> |  | ||||||
|       <created>1725820337669</created> |  | ||||||
|       <option name="number" value="00028" /> |  | ||||||
|       <option name="presentableId" value="LOCAL-00028" /> |  | ||||||
|       <option name="project" value="LOCAL" /> |  | ||||||
|       <updated>1725820337669</updated> |  | ||||||
|     </task> |  | ||||||
|     <task id="LOCAL-00029" summary="little fixes a labels when stop and start, installer first functions works"> |     <task id="LOCAL-00029" summary="little fixes a labels when stop and start, installer first functions works"> | ||||||
|       <option name="closed" value="true" /> |       <option name="closed" value="true" /> | ||||||
|       <created>1725991610908</created> |       <created>1725991610908</created> | ||||||
| @@ -540,18 +463,109 @@ | |||||||
|       <option name="project" value="LOCAL" /> |       <option name="project" value="LOCAL" /> | ||||||
|       <updated>1729938941027</updated> |       <updated>1729938941027</updated> | ||||||
|     </task> |     </task> | ||||||
|     <option name="localTasksCounter" value="66" /> |     <task id="LOCAL-00066" summary="- methods from class MainWindow move to class FrameWidgets for active color_label when theme change
- optimize columnconfigure, rowconfigure in class MainWindow and FrameWidgets
- add new Frame for Widgets on Bottom
- optimize from tkinter * to from tkinter import filedialog, ttk, TclError"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731097309468</created> | ||||||
|  |       <option name="number" value="00066" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00066" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731097309468</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00067" summary="- methods from class MainWindow move to class FrameWidgets for active color_label when theme change
- optimize columnconfigure, rowconfigure in class MainWindow and FrameWidgets
- add new Frame for Widgets on Bottom
- optimize from tkinter * to from tkinter import filedialog, ttk, TclError"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731097969343</created> | ||||||
|  |       <option name="number" value="00067" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00067" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731097969344</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00068" summary="- methods from class MainWindow move to class FrameWidgets for active color_label when theme change
- optimize columnconfigure, rowconfigure in class MainWindow and FrameWidgets
- add new Frame for Widgets on Bottom
- optimize from tkinter * to from tkinter import filedialog, ttk, TclError"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731098372497</created> | ||||||
|  |       <option name="number" value="00068" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00068" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731098372497</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00069" summary=" - - Fix Checkbutton Autostart when first install Wire-Py"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731690583059</created> | ||||||
|  |       <option name="number" value="00069" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00069" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731690583060</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00070" summary=" - - Fix Checkbutton Autostart when first install Wire-Py"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731836942211</created> | ||||||
|  |       <option name="number" value="00070" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00070" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731836942212</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00071" summary=" - Update Translate Files"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731840048762</created> | ||||||
|  |       <option name="number" value="00071" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00071" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731840048763</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00072" summary=" - Update Translate Files"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731840089956</created> | ||||||
|  |       <option name="number" value="00072" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00072" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731840089956</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00073" summary=" - Update Translate Files"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731840188277</created> | ||||||
|  |       <option name="number" value="00073" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00073" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731840188278</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00074" summary=" - Update Translate Files"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731840383592</created> | ||||||
|  |       <option name="number" value="00074" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00074" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731840383592</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00075" summary=" - Update Translate Files"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731841930614</created> | ||||||
|  |       <option name="number" value="00075" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00075" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731841930615</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00076" summary=" - Update Translate Files"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731844213239</created> | ||||||
|  |       <option name="number" value="00076" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00076" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731844213239</updated> | ||||||
|  |     </task> | ||||||
|  |     <task id="LOCAL-00077" summary=" - Update Translate Files"> | ||||||
|  |       <option name="closed" value="true" /> | ||||||
|  |       <created>1731844339039</created> | ||||||
|  |       <option name="number" value="00077" /> | ||||||
|  |       <option name="presentableId" value="LOCAL-00077" /> | ||||||
|  |       <option name="project" value="LOCAL" /> | ||||||
|  |       <updated>1731844339039</updated> | ||||||
|  |     </task> | ||||||
|  |     <option name="localTasksCounter" value="78" /> | ||||||
|     <servers /> |     <servers /> | ||||||
|   </component> |   </component> | ||||||
|   <component name="UnknownFeatures"> |   <component name="UnknownFeatures"> | ||||||
|     <option featureType="com.intellij.fileTypeFactory" implementationName="*.policy" /> |     <option featureType="com.intellij.fileTypeFactory" implementationName="*.policy" /> | ||||||
|   </component> |   </component> | ||||||
|   <component name="Vcs.Log.Tabs.Properties"> |   <component name="Vcs.Log.Tabs.Properties"> | ||||||
|     <option name="OPEN_GENERIC_TABS"> |  | ||||||
|       <map> |  | ||||||
|         <entry key="f9756e7d-0c5c-4b84-84e4-660314c46ea4" value="TOOL_WINDOW" /> |  | ||||||
|       </map> |  | ||||||
|     </option> |  | ||||||
|     <option name="TAB_STATES"> |     <option name="TAB_STATES"> | ||||||
|       <map> |       <map> | ||||||
|         <entry key="MAIN"> |         <entry key="MAIN"> | ||||||
| @@ -562,7 +576,7 @@ | |||||||
|                   <entry key="branch"> |                   <entry key="branch"> | ||||||
|                     <value> |                     <value> | ||||||
|                       <list> |                       <list> | ||||||
|                         <option value="origin/wire-py-reformat-14-08-2024" /> |                         <option value="1.11.0824" /> | ||||||
|                       </list> |                       </list> | ||||||
|                     </value> |                     </value> | ||||||
|                   </entry> |                   </entry> | ||||||
| @@ -571,19 +585,11 @@ | |||||||
|             </State> |             </State> | ||||||
|           </value> |           </value> | ||||||
|         </entry> |         </entry> | ||||||
|         <entry key="f9756e7d-0c5c-4b84-84e4-660314c46ea4"> |  | ||||||
|           <value> |  | ||||||
|             <State /> |  | ||||||
|           </value> |  | ||||||
|         </entry> |  | ||||||
|       </map> |       </map> | ||||||
|     </option> |     </option> | ||||||
|   </component> |   </component> | ||||||
|   <component name="VcsManagerConfiguration"> |   <component name="VcsManagerConfiguration"> | ||||||
|     <option name="ADD_EXTERNAL_FILES_SILENTLY" value="true" /> |     <option name="ADD_EXTERNAL_FILES_SILENTLY" value="true" /> | ||||||
|     <MESSAGE value="fix a filedialog for hidden Files work
install rollback to bash for start wirepy and wirepy rollback to bash" /> |  | ||||||
|     <MESSAGE value="install rollback bash to py wirepy and wirepy rollback to py" /> |  | ||||||
|     <MESSAGE value="fix install and .desktop File Tar works now for user home and filebrowser.askfilebrowser start now in user home" /> |  | ||||||
|     <MESSAGE value="replace tar with zip and Check if Zip file is empty" /> |     <MESSAGE value="replace tar with zip and Check if Zip file is empty" /> | ||||||
|     <MESSAGE value="Create your own message boxes for export" /> |     <MESSAGE value="Create your own message boxes for export" /> | ||||||
|     <MESSAGE value="chown Export File to 1000:1000" /> |     <MESSAGE value="chown Export File to 1000:1000" /> | ||||||
| @@ -606,15 +612,23 @@ | |||||||
|     <MESSAGE value=" - Add Options, Help, Update Label and Update Menubutton 
 - Theme now separate Light and Dark" /> |     <MESSAGE value=" - Add Options, Help, Update Label and Update Menubutton 
 - Theme now separate Light and Dark" /> | ||||||
|     <MESSAGE value="- Optimize Class. Move to wg_main Import Start/StopBTN and Tooltip" /> |     <MESSAGE value="- Optimize Class. Move to wg_main Import Start/StopBTN and Tooltip" /> | ||||||
|     <MESSAGE value="- Optimize Class and Tooltip" /> |     <MESSAGE value="- Optimize Class and Tooltip" /> | ||||||
|     <option name="LAST_COMMIT_MESSAGE" value="- Optimize Class and Tooltip" /> |     <MESSAGE value="- methods from class MainWindow move to class FrameWidgets for active color_label when theme change
- optimize columnconfigure, rowconfigure in class MainWindow and FrameWidgets
- add new Frame for Widgets on Bottom
- optimize from tkinter * to from tkinter import filedialog, ttk, TclError" /> | ||||||
|  |     <MESSAGE value=" - - Fix Checkbutton Autostart when first install Wire-Py" /> | ||||||
|  |     <MESSAGE value=" - Update Translate Files" /> | ||||||
|  |     <option name="LAST_COMMIT_MESSAGE" value=" - Update Translate Files" /> | ||||||
|   </component> |   </component> | ||||||
|   <component name="XDebuggerManager"> |   <component name="XDebuggerManager"> | ||||||
|     <breakpoint-manager> |     <breakpoint-manager> | ||||||
|       <breakpoints> |       <breakpoints> | ||||||
|         <line-breakpoint enabled="true" suspend="THREAD" type="python-line"> |         <line-breakpoint enabled="true" suspend="THREAD" type="python-line"> | ||||||
|           <url>file://$PROJECT_DIR$/start_wg.py</url> |           <url>file://$PROJECT_DIR$/wg_main.py</url> | ||||||
|           <line>1</line> |           <line>1128</line> | ||||||
|           <option name="timeStamp" value="2" /> |           <option name="timeStamp" value="3" /> | ||||||
|  |         </line-breakpoint> | ||||||
|  |         <line-breakpoint enabled="true" suspend="THREAD" type="python-line"> | ||||||
|  |           <url>file://$PROJECT_DIR$/ssl_decrypt.py</url> | ||||||
|  |           <line>3</line> | ||||||
|  |           <option name="timeStamp" value="4" /> | ||||||
|         </line-breakpoint> |         </line-breakpoint> | ||||||
|       </breakpoints> |       </breakpoints> | ||||||
|     </breakpoint-manager> |     </breakpoint-manager> | ||||||
|   | |||||||
							
								
								
									
										95
									
								
								Changelog
									
									
									
									
									
								
							
							
						
						| @@ -3,41 +3,118 @@ My standard System: Linux Mint 22 Cinnamon | |||||||
|  |  | ||||||
| ## [Unreleased] | ## [Unreleased] | ||||||
|  |  | ||||||
|  - os import in wg_func replaced by other methods |  - os import in cls_mth_fc.py replaced by other methods | ||||||
|  - If Wire-Py already runs, prevent further start |  - If Wire-Py already runs, prevent further start | ||||||
|  - for loops with lists replaced by List Comprehensions |  - for loops with lists replaced by List Comprehensions | ||||||
|  - Update search after start of Wire-Py |  | ||||||
|  |  | ||||||
|  |    ### Added | ||||||
|  | 13-04-0725 | ||||||
|  |  | ||||||
|  |  - Installer update for Open Suse Tumbleweed and Leap | ||||||
|  |  - add symbolic link wirepy.py  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    ### Added | ||||||
|  | 09-04-0725 | ||||||
|  |  | ||||||
|  |  - Installer now with query and remove | ||||||
|  |  - Icons merged | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    ### Added | ||||||
|  | 07-04-0725 | ||||||
|  |  | ||||||
|  |  - Installers will support other systems again | ||||||
|  |  - Installer is now finished clean with wrong password | ||||||
|  |  - Rename wg_main to wirepy  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    ### Added | ||||||
|  | 03-03-2025 | ||||||
|  |  | ||||||
|  |  - Fixes a new user files create | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   ### Added | ||||||
|  | 02-03-2025 | ||||||
|  |  | ||||||
|  |  - Fix ipv6 in Config File on import | ||||||
|  |  - Wirepy run now as user | ||||||
|  |  - settings, keys and Config Files now in ~/.config/wire_py | ||||||
|  |  - For new users, the required files are created and autostart service is started.  | ||||||
|  |  - Tunnels are now read from the directory to view them in the list.  | ||||||
|  |    To display only own tunnels, and read errors are minimized. | ||||||
|  |  | ||||||
|  |   ### Added | ||||||
|  | 10-11-2024 | ||||||
|  |  | ||||||
|  |  - Fix Checkbutton Autostart when first install Wire-Py | ||||||
|  |  - Update Translate Files | ||||||
|  |  | ||||||
|  |   ### Added | ||||||
|  | 10-11-2024 | ||||||
|  |  | ||||||
|  |  - Translate file de_De complete | ||||||
|  |  - new FrameWidget for rename Entry and rename Button | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   ### Added | ||||||
|  | 09-11-2024 | ||||||
|  |  | ||||||
|  |  - Move Tips Method in separate class for Tooltips in another Apps | ||||||
|  |  - Move Version Variable in main script | ||||||
|  |  - Edit Class GiteaUpdate for requests in api_down and download | ||||||
|  |  - Description on Class GiteaUpdate | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   ### Added | ||||||
|  | 08-11-2024 | ||||||
|  |  | ||||||
|  |  - Preparation for language translation part 1 | ||||||
|  |  - separate folder for general icons | ||||||
|  |  - install edit for new dir lx-icons | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   ### Added | ||||||
|  | 08-11-2024 | ||||||
|  |  | ||||||
|  |  - methods from class MainWindow move to class FrameWidgets for active color_label when theme change | ||||||
|  |  - optimize columnconfigure, rowconfigure in class MainWindow and FrameWidgets | ||||||
|  |  - add new Frame for Widgets on Bottom | ||||||
|  |  - optimize from tkinter * to from tkinter import filedialog, ttk, TclError | ||||||
|  |  | ||||||
|  |  | ||||||
|   ### Added |   ### Added | ||||||
| 07-11-2024 | 07-11-2024 | ||||||
|  |  | ||||||
| - remove classes and add methods to class FrameWidgets (removed self errors) |  - remove classes and add methods to class FrameWidgets (removed self errors) | ||||||
|  |  | ||||||
|  |  | ||||||
|   ### Added |   ### Added | ||||||
| 27-10-2024 | 27-10-2024 | ||||||
|  |  | ||||||
| - Add Autoconnect settings to settings.conf |  - Add Autoconnect settings to settings | ||||||
|  |  | ||||||
|  |  | ||||||
|   ### Added |   ### Added | ||||||
| 26-10-2024 | 26-10-2024 | ||||||
|  |  | ||||||
| - Add run_as Bash script and open_gitea.py python script |  - Add run_as Bash script and open_gitea.py python script | ||||||
| - Add Tooltip disable/enable |  - Add Tooltip disable/enable | ||||||
| - Rename settings to settings.conf for theme, updates and tooltip enable in one file |  - Rename settings to settings for theme, updates and tooltip enable in one file | ||||||
|  |  | ||||||
|  |  | ||||||
|   ### Added |   ### Added | ||||||
| 25-10-2024 | 25-10-2024 | ||||||
|  |  | ||||||
| - Optimize Class and Tooltip |  - Optimize Class and Tooltip | ||||||
|  |  | ||||||
|  |  | ||||||
|   ### Added |   ### Added | ||||||
| 21-10-2024 | 21-10-2024 | ||||||
|  |  | ||||||
| - Optimize Class. Move to wg_main Import Start/StopBTN and Tooltip |  - Optimize Class. Move to wg_main Import Start/StopBTN and Tooltip | ||||||
|  |  | ||||||
|  |  | ||||||
|  ### Added |  ### Added | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								Wire-Py.desktop
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| @@ -1,7 +1,7 @@ | |||||||
| [Desktop Entry] | [Desktop Entry] | ||||||
| Type=Application | Type=Application | ||||||
| Name=Wire-Py | Name=Wire-Py | ||||||
| Exec=/usr/bin/wirepy.py | Exec=/usr/local/bin/wirepy.py | ||||||
| Terminal=false | Terminal=false | ||||||
| Categories=Network; | Categories=Network; | ||||||
| Icon=/usr/share/icons/wp-icons/128/wg_vpn.png | Icon=/usr/share/icons/lx-icons/128/wg_vpn.png | ||||||
							
								
								
									
										
											BIN
										
									
								
								__pycache__/cls_mth_fc.cpython-312.pyc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								__pycache__/message.cpython-312.pyc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								__pycache__/start_wg.cpython-312.pyc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								__pycache__/wg_main.cpython-312.pyc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										496
									
								
								cls_mth_fc.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,496 @@ | |||||||
|  | """ Classes Method and Functions for lx Apps """ | ||||||
|  |  | ||||||
|  | import gettext | ||||||
|  | import locale | ||||||
|  | import os | ||||||
|  | import shutil | ||||||
|  | import signal | ||||||
|  | import subprocess | ||||||
|  | import sys | ||||||
|  | import tkinter as tk | ||||||
|  | from typing import Optional, Dict, Any, NoReturn, TextIO, Tuple, List | ||||||
|  | import zipfile | ||||||
|  | from datetime import datetime | ||||||
|  | from pathlib import Path | ||||||
|  | from subprocess import check_call, CompletedProcess | ||||||
|  | from tkinter import ttk, Toplevel | ||||||
|  |  | ||||||
|  | import requests | ||||||
|  |  | ||||||
|  | APP = "wirepy" | ||||||
|  | LOCALE_DIR = "/usr/share/locale/" | ||||||
|  | locale.bindtextdomain(APP, LOCALE_DIR) | ||||||
|  | gettext.bindtextdomain(APP, LOCALE_DIR) | ||||||
|  | gettext.textdomain(APP) | ||||||
|  | _ = gettext.gettext | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Create: | ||||||
|  |     """ | ||||||
|  |     This class is for the creation of the folders and files | ||||||
|  |     required by Wire-Py, as well as for decryption | ||||||
|  |     the tunnel from the user's home directory | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def dir_and_files() -> None: | ||||||
|  |         """ | ||||||
|  |         check and create folders and files if not present | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         pth: Path = Path.home() / ".config/wire_py" | ||||||
|  |         pth.mkdir(parents=True, exist_ok=True) | ||||||
|  |         sett: Path = Path.home() / ".config/wire_py/settings" | ||||||
|  |         ks: Path = Path.home() / ".config/wire_py/keys" | ||||||
|  |  | ||||||
|  |         if sett.exists(): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |             sett.touch() | ||||||
|  |             sett.write_text("[UPDATES]\non\n[THEME]\nlight\n[TOOLTIP]\nTrue\n[AUTOSTART ON]\noff\n") | ||||||
|  |  | ||||||
|  |         if ks.exists(): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |             ks.touch() | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def files_for_autostart() -> None: | ||||||
|  |         """ | ||||||
|  |         check and create a file for auto start if not present and enable the service | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         pth2: Path = Path.home() / ".config/systemd/user" | ||||||
|  |         pth2.mkdir(parents=True, exist_ok=True) | ||||||
|  |         wg_ser: Path = Path.home() / ".config/systemd/user/wg_start.service" | ||||||
|  |  | ||||||
|  |         if wg_ser.exists(): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |             wg_ser.touch() | ||||||
|  |             wg_ser.write_text("[Unit]\nDescription=Automatic Tunnel Start\nAfter=network-online.target\n\n[Service]\n" | ||||||
|  |                               "Type=oneshot\nExecStartPre=/bin/sleep 5\nExecStart=/usr/local/bin/start_wg.py\n[Install]" | ||||||
|  |                               "\nWantedBy=default.target") | ||||||
|  |             check_call(["systemctl", "--user", "enable", "wg_start.service"]) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def make_dir() -> None: | ||||||
|  |         """Folder Name "tlecdewg" = Tunnel Encrypt Decrypt Wireguard""" | ||||||
|  |  | ||||||
|  |         folder_path: Path = Path("/tmp/tlecdcwg/") | ||||||
|  |         if folder_path.exists(): | ||||||
|  |             pass | ||||||
|  |         else: | ||||||
|  |             folder_path.mkdir() | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def decrypt() -> None: | ||||||
|  |         """ | ||||||
|  |         Starts SSL dencrypt | ||||||
|  |         """ | ||||||
|  |         process: CompletedProcess[str] = subprocess.run(["pkexec", "/usr/local/bin/ssl_decrypt.py"], | ||||||
|  |                                                         stdout=subprocess.PIPE, text=True, check=True) | ||||||
|  |         path: Path = Path.home() / ".config/wire_py/" | ||||||
|  |         file_in_path: list[Path] = list(path.rglob("*.dat")) | ||||||
|  |         if file_in_path: | ||||||
|  |             if process.returncode == 0: | ||||||
|  |                 print("File successfully decrypted...") | ||||||
|  |             else: | ||||||
|  |                 print(f"Error with the following code... {process.returncode}") | ||||||
|  |         else: | ||||||
|  |             print(_("Ready for import")) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def encrypt() -> None: | ||||||
|  |         """ | ||||||
|  |         Starts SSL encryption | ||||||
|  |         """ | ||||||
|  |         process: CompletedProcess[str] = subprocess.run(["pkexec", "/usr/local/bin/ssl_encrypt.py"], | ||||||
|  |                                                         stdout=subprocess.PIPE, text=True, check=True) | ||||||
|  |         print(process.stdout) | ||||||
|  |         if process.returncode == 0: | ||||||
|  |             print("All Files successfully encrypted...") | ||||||
|  |         else: | ||||||
|  |             print(f"Error with the following code... {process.returncode}") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class LxTools(tk.Tk): | ||||||
|  |     """ | ||||||
|  |     Class LinuxTools methods that can also be used for other apps | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, *args: Any, **kwargs: Any) -> None: | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def uos() -> None: | ||||||
|  |         """ | ||||||
|  |         uos = LOGIN USERNAME | ||||||
|  |  | ||||||
|  |         This method displays the username of the logged-in user, | ||||||
|  |         even if you are rooted in a shell | ||||||
|  |         """ | ||||||
|  |         log_name: str = f"{Path.home()}"[6:] | ||||||
|  |         file: Path = Path.home() / "/tmp/.log_user" | ||||||
|  |         with open(file, "w", encoding="utf-8") as f: | ||||||
|  |             f.write(log_name) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def clean_files(folder_path: Path = None, file: Path = None) -> None: | ||||||
|  |         """ | ||||||
|  |         method that can be added after need to delete a folder and a file when quitting. | ||||||
|  |         Args: | ||||||
|  |             :param file: default None | ||||||
|  |             :param folder_path: default None | ||||||
|  |         """ | ||||||
|  |         if folder_path is not None: | ||||||
|  |             shutil.rmtree(folder_path) | ||||||
|  |         if file is not None: | ||||||
|  |             Path.unlink(file) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def if_tip(path: Path) -> bool: | ||||||
|  |         """ | ||||||
|  |         method that writes in file whether tooltip is displayed or not | ||||||
|  |         """ | ||||||
|  |         with open(path, "r", encoding="utf-8") as set_f2: | ||||||
|  |             lines2 = set_f2.readlines() | ||||||
|  |             if "False\n" in lines2: | ||||||
|  |                 tip = False | ||||||
|  |             else: | ||||||
|  |                 tip = True | ||||||
|  |             return tip | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def msg_window(img_w: str, img_i: str, w_title: str, w_txt: str, txt2: Optional[str] = None, | ||||||
|  |                    com: Optional[str] = None) -> None: | ||||||
|  |         """ | ||||||
|  |         Creates message windows | ||||||
|  |  | ||||||
|  |         :argument img_w = Image for TK window which is displayed to the left of the text | ||||||
|  |         :argument img_i = Image for Task Icon | ||||||
|  |         :argument w_title = Windows Title | ||||||
|  |         :argument w_txt = Text for Tk Window | ||||||
|  |         :argument txt2 = Text for Button two | ||||||
|  |         :argument com = function for Button command | ||||||
|  |         """ | ||||||
|  |         msg: tk.Toplevel = tk.Toplevel() | ||||||
|  |         msg.resizable(width=False, height=False) | ||||||
|  |         msg.title(w_title) | ||||||
|  |         msg.configure(pady=15, padx=15) | ||||||
|  |         msg.img = tk.PhotoImage(file=img_w) | ||||||
|  |         msg.i_window = tk.Label(msg, image=msg.img) | ||||||
|  |  | ||||||
|  |         label: tk.Label = tk.Label(msg, text=w_txt) | ||||||
|  |  | ||||||
|  |         label.grid(column=1, row=0) | ||||||
|  |  | ||||||
|  |         if txt2 is not None and com is not None: | ||||||
|  |             label.config(font=("Ubuntu", 11), padx=15, justify="left") | ||||||
|  |             msg.i_window.grid(column=0, row=0, sticky="nw") | ||||||
|  |             button2: ttk.Button = ttk.Button(msg, text=f"{txt2}", command=com, padding=4) | ||||||
|  |             button2.grid(column=0, row=1, sticky="e", columnspan=2) | ||||||
|  |             button: ttk.Button = ttk.Button(msg, text="OK", command=msg.destroy, padding=4) | ||||||
|  |             button.grid(column=0, row=1, sticky="w", columnspan=2) | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |             label.config(font=("Ubuntu", 11), padx=15) | ||||||
|  |             msg.i_window.grid(column=0, row=0) | ||||||
|  |             button: ttk.Button = ttk.Button(msg, text="OK", command=msg.destroy, padding=4) | ||||||
|  |             button.grid(column=0, columnspan=2, row=1) | ||||||
|  |  | ||||||
|  |         img_i: tk.PhotoImage = tk.PhotoImage(file=img_i) | ||||||
|  |         msg.iconphoto(True, img_i) | ||||||
|  |         msg.columnconfigure(0, weight=1) | ||||||
|  |         msg.rowconfigure(0, weight=1) | ||||||
|  |         msg.winfo_toplevel() | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def sigi(file_path: Optional[Path] = None, file: Optional[Path] = None) -> None: | ||||||
|  |         """ | ||||||
|  |         Function for cleanup after a program interruption | ||||||
|  |  | ||||||
|  |         :param file: Optional - File to be deleted | ||||||
|  |         :param file_path: Optional - Directory to be deleted | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         def signal_handler(signum: int, frame: Any) -> NoReturn: | ||||||
|  |             """ | ||||||
|  |             Determines clear text names for signal numbers and handles signals | ||||||
|  |  | ||||||
|  |             Args: | ||||||
|  |                 signum: The signal number | ||||||
|  |                 frame: The current stack frame | ||||||
|  |  | ||||||
|  |             Returns: | ||||||
|  |                 NoReturn since the function either exits the program or continues execution | ||||||
|  |             """ | ||||||
|  |  | ||||||
|  |             signals_to_names_dict: Dict[int, str] = dict((getattr(signal, n), n) for n in dir(signal) | ||||||
|  |                                                          if n.startswith("SIG") and "_" not in n) | ||||||
|  |  | ||||||
|  |             signal_name: str = signals_to_names_dict.get(signum, f"Unnamed signal: {signum}") | ||||||
|  |  | ||||||
|  |             # End program for certain signals, report to others only reception | ||||||
|  |             if signum in (signal.SIGINT, signal.SIGTERM): | ||||||
|  |                 exit_code: int = 1 | ||||||
|  |                 print(f"\nSignal {signal_name} {signum} received. => Aborting with exit code {exit_code}.") | ||||||
|  |                 LxTools.clean_files(file_path, file) | ||||||
|  |                 print("Breakdown by user...") | ||||||
|  |                 sys.exit(exit_code) | ||||||
|  |             else: | ||||||
|  |                 print(f"Signal {signum} received and ignored.") | ||||||
|  |                 LxTools.clean_files(file_path, file) | ||||||
|  |                 print("Process unexpectedly ended...") | ||||||
|  |  | ||||||
|  |         # Register signal handlers for various signals | ||||||
|  |         signal.signal(signal.SIGINT, signal_handler) | ||||||
|  |         signal.signal(signal.SIGTERM, signal_handler) | ||||||
|  |         signal.signal(signal.SIGHUP, signal_handler) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GiteaUpdate: | ||||||
|  |     """ | ||||||
|  |     Calling download requests the download URL of the running script, | ||||||
|  |     the taskbar image for the “Download OK” window, the taskbar image for the | ||||||
|  |     “Download error” window and the variable res | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def api_down(update_api_url: str, version: str, file: Optional[Path] = None) -> str: | ||||||
|  |         """ | ||||||
|  |         Checks for updates via API | ||||||
|  |          | ||||||
|  |         Args: | ||||||
|  |             update_api_url: Update API URL | ||||||
|  |             version: Current version | ||||||
|  |             file: Optional - Configuration file | ||||||
|  |              | ||||||
|  |         Returns: | ||||||
|  |             New version or status message | ||||||
|  |         """ | ||||||
|  |         try: | ||||||
|  |             response: requests.Response = requests.get(update_api_url, timeout=10) | ||||||
|  |             response_dict: Any = response.json() | ||||||
|  |             response_dict: Dict[str, Any] = response_dict[0] | ||||||
|  |             with open(file, "r", encoding="utf-8") as set_f: | ||||||
|  |                 set_f = set_f.read() | ||||||
|  |                 if "on\n" in set_f: | ||||||
|  |                     if version[3:] != response_dict["tag_name"]: | ||||||
|  |                         req: str = response_dict["tag_name"] | ||||||
|  |                     else: | ||||||
|  |                         req: str = "No Updates" | ||||||
|  |                 else: | ||||||
|  |                     req: str = "False" | ||||||
|  |                 return req | ||||||
|  |         except requests.exceptions.RequestException: | ||||||
|  |             req: str = "No Internet Connection!" | ||||||
|  |             return req | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def download(urld: str, res: str, img_w: str = None, img_i: str = None, img_w2: str = None, img_i2: str = None) -> None: | ||||||
|  |         """ | ||||||
|  |         Downloads new version of wirepy | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             urld: Download URL | ||||||
|  |             res: Result filename | ||||||
|  |             img_w: Image for TK window which is displayed to the left of the text | ||||||
|  |             img_i: Image for Task Icon | ||||||
|  |             img_w2: Image for TK window which is displayed to the left of the text | ||||||
|  |             img_i2: Image for Task Icon | ||||||
|  |         """ | ||||||
|  |         try: | ||||||
|  |             to_down: str = f"wget -qP {Path.home()} {" "} {urld}" | ||||||
|  |             result: int = subprocess.call(to_down, shell=True) | ||||||
|  |             if result == 0: | ||||||
|  |                 shutil.chown(f"{Path.home()}/{res}.zip", 1000, 1000) | ||||||
|  |  | ||||||
|  |                 wt: str = _("Download Successful") | ||||||
|  |                 msg_t: str = _("Your zip file is in home directory") | ||||||
|  |                 LxTools.msg_window(img_w, img_i, wt, msg_t) | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |  | ||||||
|  |                 wt: str = _("Download error") | ||||||
|  |                 msg_t: str = _("Download failed! Please try again") | ||||||
|  |                 LxTools.msg_window(img_w2, img_i2, wt, msg_t) | ||||||
|  |  | ||||||
|  |         except subprocess.CalledProcessError: | ||||||
|  |  | ||||||
|  |             wt: str = _("Download error") | ||||||
|  |             msg_t: str = _("Download failed! No internet connection!") | ||||||
|  |             LxTools.msg_window(img_w2, img_i2, wt, msg_t) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Tunnel: | ||||||
|  |     """ | ||||||
|  |     Class of Methods for Wire-Py | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def con_to_dict(cls, file: TextIO) -> Tuple[str, str, str, Optional[str]]: | ||||||
|  |         """ | ||||||
|  |         Returns tuple of (address, dns, endpoint, pre_key) | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         dictlist: List[str] = [] | ||||||
|  |         for lines in file.readlines(): | ||||||
|  |             line_plit: List[str] = lines.split() | ||||||
|  |             dictlist = dictlist + line_plit | ||||||
|  |         dictlist.remove("[Interface]") | ||||||
|  |         dictlist.remove("[Peer]") | ||||||
|  |         for items in dictlist: | ||||||
|  |             if items == "=": | ||||||
|  |                 dictlist.remove(items) | ||||||
|  |             if items == "::/0": | ||||||
|  |                 dictlist.remove(items) | ||||||
|  |  | ||||||
|  |         # Here is the beginning (Loop) of convert List to Dictionary | ||||||
|  |         for _ in dictlist: | ||||||
|  |             a: List[str] = [dictlist[0], dictlist[1]] | ||||||
|  |             b: List[str] = [dictlist[2], dictlist[3]] | ||||||
|  |             c: List[str] = [dictlist[4], dictlist[5]] | ||||||
|  |             d: List[str] = [dictlist[6], dictlist[7]] | ||||||
|  |             e: List[str] = [dictlist[8], dictlist[9]] | ||||||
|  |             f: List[str] = [dictlist[10], dictlist[11]] | ||||||
|  |             g: List[str] = [dictlist[12], dictlist[13]] | ||||||
|  |             h: List[str] = [dictlist[14], dictlist[15]] | ||||||
|  |             new_list: List[List[str]] = [a, b, c, d, e, f, g, h] | ||||||
|  |             final_dict: Dict[str, str] = {} | ||||||
|  |             for elements in new_list: | ||||||
|  |                 final_dict[elements[0]] = elements[1] | ||||||
|  |  | ||||||
|  |             # end... result a Dictionary | ||||||
|  |  | ||||||
|  |         address: str = final_dict["Address"] | ||||||
|  |         dns: str = final_dict["DNS"] | ||||||
|  |         if "," in dns: | ||||||
|  |             dns = dns[:-1] | ||||||
|  |         endpoint: str = final_dict["Endpoint"] | ||||||
|  |         pre_key: Optional[str] = final_dict.get("PresharedKey") | ||||||
|  |         if pre_key is None: | ||||||
|  |             pre_key: Optional[str] = final_dict.get("PreSharedKey") | ||||||
|  |         return address, dns, endpoint, pre_key | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def active() -> str: | ||||||
|  |         """ | ||||||
|  |         Shows the Active Tunnel | ||||||
|  |         """ | ||||||
|  |         active = (os.popen('nmcli con show --active | grep -iPo "(.*)(wireguard)"').read().split()) | ||||||
|  |         if not active: | ||||||
|  |             active = "" | ||||||
|  |         else: | ||||||
|  |             active = active[0] | ||||||
|  |  | ||||||
|  |         return active | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def list() -> List[str]: | ||||||
|  |         """ | ||||||
|  |         Returns a list of Wireguard tunnel names | ||||||
|  |         """ | ||||||
|  |         folder_path: Path = Path("/tmp/tlecdcwg/") | ||||||
|  |         wg_s: List[str] = os.listdir(folder_path) | ||||||
|  |  | ||||||
|  |         return wg_s | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def export(img_w: str = None, img_i: str = None, img_w2: str = None, img_i2: str = None, sl: str = None, pfit:str = None) -> None: | ||||||
|  |         """ | ||||||
|  |         This will export the tunnels. | ||||||
|  |         A zipfile with the current date and time is created | ||||||
|  |         in the user's home directory with the correct right | ||||||
|  |         Args: | ||||||
|  |             img_w: Image for TK window which is displayed to the left of the text | ||||||
|  |             img_i: Image for Task Icon | ||||||
|  |             img_w2: Image for TK window which is displayed to the left of the text | ||||||
|  |             img_i2: Image for Task Icon | ||||||
|  |         """ | ||||||
|  |         now_time: datetime = datetime.now() | ||||||
|  |         now_datetime: str = now_time.strftime("wg-exp-%m-%d-%Y-%H:%M") | ||||||
|  |         tl: List[str] = Tunnel.list() | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             if len(tl) != 0: | ||||||
|  |                 wg_tar: str = f"{Path.home()}/{now_datetime}" | ||||||
|  |                 shutil.copytree("/tmp/tlecdcwg/", "/tmp/wire_py", dirs_exist_ok=True) | ||||||
|  |                 source: Path = Path("/tmp/wire_py") | ||||||
|  |                 shutil.make_archive(wg_tar, "zip", source) | ||||||
|  |                 shutil.rmtree(source) | ||||||
|  |                 with zipfile.ZipFile(f"{wg_tar}.zip", "r") as zf: | ||||||
|  |                     if len(zf.namelist()) != 0: | ||||||
|  |  | ||||||
|  |                         msg_t: str = _("Your zip file is in home directory") | ||||||
|  |                         LxTools.msg_window(img_w, img_i, _("Export Successful"), msg_t) | ||||||
|  |  | ||||||
|  |                     else: | ||||||
|  |  | ||||||
|  |                         msg_t: str = _("Export failed! Please try again") | ||||||
|  |                         LxTools.msg_window(img_w2, img_i2, _("Export error"), msg_t) | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |  | ||||||
|  |                 LxTools.msg_window(img_w, img_i2, sl, pfit) | ||||||
|  |  | ||||||
|  |         except TypeError: | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Tooltip: | ||||||
|  |     """ | ||||||
|  |     class for Tooltip | ||||||
|  |  | ||||||
|  |     import Tooltip | ||||||
|  |  | ||||||
|  |     example: Tooltip(label, "Show tooltip on label") | ||||||
|  |     example: Tooltip(button, "Show tooltip on button") | ||||||
|  |     info: label and button are parent. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, widget: Any, text: str, tips: Optional[bool] = None) -> None: | ||||||
|  |         """ | ||||||
|  |         Tooltip Class | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         self.widget: Any = widget | ||||||
|  |         self.text: str = text | ||||||
|  |         self.tooltip_window: Optional[Toplevel] = None | ||||||
|  |         if tips: | ||||||
|  |             self.widget.bind("<Enter>", self.show_tooltip) | ||||||
|  |             self.widget.bind("<Leave>", self.hide_tooltip) | ||||||
|  |  | ||||||
|  |     def show_tooltip(self, event: Optional[Any] = None) -> None: | ||||||
|  |         """ | ||||||
|  |         Shows the tooltip | ||||||
|  |         """ | ||||||
|  |         if self.tooltip_window or not self.text: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         x: int | ||||||
|  |         y: int | ||||||
|  |         cx: int | ||||||
|  |         cy: int | ||||||
|  |         x, y, cx, cy = self.widget.bbox("insert") | ||||||
|  |         x += self.widget.winfo_rootx() + 65 | ||||||
|  |         y += self.widget.winfo_rooty() + 40 | ||||||
|  |         self.tooltip_window = tw = tk.Toplevel(self.widget) | ||||||
|  |         tw.wm_overrideredirect(True) | ||||||
|  |         tw.wm_geometry(f"+{x}+{y}") | ||||||
|  |  | ||||||
|  |         label: tk.Label = tk.Label(tw, text=self.text, background="lightgreen", foreground="black", relief="solid", | ||||||
|  |                                    borderwidth=1, padx=5, pady=5) | ||||||
|  |         label.grid() | ||||||
|  |  | ||||||
|  |     def hide_tooltip(self, event: Optional[Any] = None) -> None: | ||||||
|  |         """ | ||||||
|  |         Hides the tooltip | ||||||
|  |         """ | ||||||
|  |         if self.tooltip_window: | ||||||
|  |             self.tooltip_window.destroy() | ||||||
|  |             self.tooltip_window = None | ||||||
|  |                         | ||||||
							
								
								
									
										307
									
								
								install
									
									
									
									
									
								
							
							
						
						| @@ -6,126 +6,223 @@ BLUE='\033[30;1;34m' | |||||||
|  |  | ||||||
| install_file_with(){ | install_file_with(){ | ||||||
|   clear |   clear | ||||||
|   sudo apt install python3-tk && \ |   mkdir -p ~/.config/wire_py && touch ~/.config/wire_py/keys && cp -u settings ~/.config/wire_py/ && \ | ||||||
|   sudo cp -u wg_main.py start_wg.py wg_func.py wirepy.py run_as open_gitea.py /usr/bin/ && \ |   mkdir -p ~/.config/systemd/user && cp -u wg_start.service ~/.config/systemd/user/ && \ | ||||||
|   sudo mkdir -p /etc/wire_py && sudo touch /etc/wire_py/.keys && sudo cp -u settings.conf /etc/wire_py/ && \ |   systemctl --user enable wg_start.service >/dev/null 2>&1 | ||||||
|   sudo cp -uR wp-icons /usr/share/icons/ &&  sudo cp -uR TK-Themes /usr/share/ && \ |   sudo cp -f org.sslcrypt.policy /usr/share/polkit-1/actions/  | ||||||
|   sudo chown -R root:root /etc/wire_py && sudo chmod 755 /etc/wire_py && \ |   if [ $? -ne 0 ] | ||||||
|   sudo ln -sf /usr/bin/wirepy.py /usr/local/bin/wirepy && \ |     then  | ||||||
|   sudo cp -u org.wirepy.policy /usr/share/polkit-1/actions/ && \ |       systemctl --user disable wg_start.service | ||||||
|   sudo cp -u Wire-Py.desktop /usr/share/applications/ && \ |       rm -r ~/.config/wire_py && rm -r ~/.config/systemd | ||||||
|   sudo cp -u wg_start.service /lib/systemd/system/ && \ |       exit 0 | ||||||
|   sudo systemctl enable wg_start.service |   else  | ||||||
|  |     sudo apt install python3-tk && \ | ||||||
|  |     sudo cp -fv wirepy.py start_wg.py cls_mth_fc.py ssl_encrypt.py ssl_decrypt.py /usr/local/bin/ && \ | ||||||
|  |     sudo cp -uR lx-icons /usr/share/icons/ &&  sudo cp -uR TK-Themes /usr/share/ && \ | ||||||
|  |     sudo cp -u languages/de/*.mo /usr/share/locale/de/LC_MESSAGES/ && \ | ||||||
|  |     sudo cp -fv Wire-Py.desktop /usr/share/applications/ && \ | ||||||
|  |     sudo ln -sf /usr/local/bin/wirepy.py /usr/local/bin/wirepy | ||||||
|  |     sudo mkdir -p /usr/local/etc/ssl | ||||||
|  |     if [ ! -f /usr/local/etc/ssl/pwgk.pem ] | ||||||
|  |       then | ||||||
|  |         sudo openssl genrsa -out /usr/local/etc/ssl/pwgk.pem 4096 | ||||||
|  |     fi | ||||||
|  |   fi   | ||||||
|   } |   } | ||||||
|  |  | ||||||
| install_arch_d(){ | install_arch_d(){ | ||||||
|   clear |   clear | ||||||
|   sudo pacman -S --noconfirm tk python3 python-requests && \ |   mkdir -p ~/.config/wire_py && touch ~/.config/wire_py/keys && cp -u settings ~/.config/wire_py/ && \ | ||||||
|   sudo cp -u wg_main.py start_wg.py wg_func.py wirepy.py run_as open_gitea.py /usr/bin/ && \ |   mkdir -p ~/.config/systemd/user && cp -u wg_start.service ~/.config/systemd/user/ && \ | ||||||
|   sudo mkdir -p /etc/wire_py && sudo touch /etc/wire_py/.keys && sudo cp -u settings.conf  /etc/wire_py/ && \ |   systemctl --user enable wg_start.service >/dev/null 2>&1 | ||||||
|   sudo cp -uR wp-icons /usr/share/icons/ && sudo cp -uR TK-Themes /usr/share/ && \ |   sudo cp -f org.sslcrypt.policy /usr/share/polkit-1/actions/ | ||||||
|   sudo chown -R root:root /etc/wire_py && sudo chmod 755 /etc/wire_py && \ |   if [ $? -ne 0 ] | ||||||
|   sudo ln -sf /usr/bin/wirepy.py /usr/local/bin/wirepy && \ |     then  | ||||||
|   sudo cp -u org.wirepy.policy /usr/share/polkit-1/actions/ && \ |         systemctl --user disable wg_start.service | ||||||
|   sudo cp -u Wire-Py.desktop /usr/share/applications/ && \ |         rm -r ~/.config/wire_py && rm -r ~/.config/systemd | ||||||
|   sudo cp -u wg_start.service /lib/systemd/system/ && \ |         exit 0 | ||||||
|   sudo systemctl enable wg_start.service |   else  | ||||||
|   } |     sudo pacman -S --noconfirm tk python3 python-requests && \ | ||||||
|  |     sudo cp -fv wirepy.py start_wg.py cls_mth_fc.py ssl_encrypt.py ssl_decrypt.py /usr/local/bin/ && \ | ||||||
| if grep -i 'debian' /etc/os-release > /dev/null 2>&1 |     sudo cp -uR lx-icons /usr/share/icons/ &&  sudo cp -uR TK-Themes /usr/share/ && \ | ||||||
|   then |     sudo cp -u languages/de/*.mo /usr/share/locale/de/LC_MESSAGES/ && \ | ||||||
|     groups > /tmp/isgroup |     sudo cp -fv Wire-Py.desktop /usr/share/applications/ && \ | ||||||
|     if grep 'sudo' /tmp/isgroup |     sudo ln -sf /usr/local/bin/wirepy.py /usr/local/bin/wirepy  | ||||||
|  |     sudo mkdir -p /usr/local/etc/ssl | ||||||
|  |     if [ ! -f /usr/local/etc/ssl/pwgk.pem ] | ||||||
|       then |       then | ||||||
|           install_file_with |         sudo openssl genrsa -out /usr/local/etc/ssl/pwgk.pem 4096 | ||||||
|  |  | ||||||
|     else |  | ||||||
|  |  | ||||||
|     echo -e "$BLUE"The installer found that they are not in the group sudo."" |  | ||||||
|     echo -e "with "$RED"su -"$BLUE" "they can enter the root shell in which they then"" |  | ||||||
|     echo -e "enter "$GREEN""usermod -aG sudo $USER.""$BLUE"" |  | ||||||
|     echo -e ""after logging in from the system, they can then run Wire-Py install again." $NORMAL" |  | ||||||
|     read -n 1 -s -r -p $"To close the Window press a button" |  | ||||||
|     clear |  | ||||||
|     exit 0 |  | ||||||
|     fi    |     fi    | ||||||
|     |     | ||||||
|  |   fi  | ||||||
|  |   } | ||||||
|  |  | ||||||
| elif grep -i 'mint\|ubuntu\|pop|' /etc/os-release > /dev/null 2>&1 | install(){ | ||||||
|   then |   if grep -i 'debian' /etc/os-release > /dev/null 2>&1 | ||||||
|       install_file_with |  | ||||||
|  |  | ||||||
|  |  | ||||||
| elif grep -i 'arch' /etc/os-release > /dev/null 2>&1 |  | ||||||
|     then |     then | ||||||
|         groups > /tmp/isgroup |       groups > /tmp/isgroup | ||||||
|  |       if grep 'sudo' /tmp/isgroup | ||||||
|  |         then | ||||||
|  |           install_file_with | ||||||
|  |       else | ||||||
|  |         echo -e "$BLUE"The installer found that they are not in the group sudo."" | ||||||
|  |         echo -e "with "$RED"su -"$BLUE" "they can enter the root shell in which they then"" | ||||||
|  |         echo -e "enter "$GREEN""usermod -aG sudo $USER.""$BLUE"" | ||||||
|  |         echo -e ""after logging in from the system, they can then run Wire-Py install again." $NORMAL" | ||||||
|  |         read -n 1 -s -r -p $"Press Enter to exit" | ||||||
|         clear |         clear | ||||||
|         if grep 'wheel' /tmp/isgroup |         exit 0 | ||||||
|           then |  | ||||||
|               install_arch_d |       fi | ||||||
|         else |  | ||||||
|             echo "The installer found that they are not in the group sudo." |   elif grep -i 'mint\|ubuntu\|pop|' /etc/os-release > /dev/null 2>&1 | ||||||
|             echo "The sudoers file must be edited with" |     then | ||||||
|             echo -e "$RED""su -""$NORMAL" |         install_file_with | ||||||
|             echo -e "$GREEN"""EDITOR=nano visudo"""$NORMAL" |  | ||||||
|             echo "Find the line:" |   elif grep -i 'arch' /etc/os-release > /dev/null 2>&1 | ||||||
|             echo "## Uncomment to allow members of group wheel to execute any command" |     then | ||||||
|             echo "remove '#' on  # %wheel ALL=(ALL) ALL and save the file" |       groups > /tmp/isgroup | ||||||
|             echo -e "then enter "$GREEN"gpasswd -a $USER wheel.""$NORMAL" |       clear | ||||||
|             echo "after logging in from the system, they can then run Wire-Py install again." |       if grep 'wheel' /tmp/isgroup | ||||||
|             read -n 1 -s -r -p $"To close the Window press a button" |         then | ||||||
|             clear |             install_arch_d | ||||||
|  |       else | ||||||
|  |           echo "The installer found that they are not in the group sudo." | ||||||
|  |           echo "The sudoers file must be edited with" | ||||||
|  |           echo -e "$RED""su -""$NORMAL" | ||||||
|  |           echo -e "$GREEN"""EDITOR=nano visudo"""$NORMAL" | ||||||
|  |           echo "Find the line:" | ||||||
|  |           echo "## Uncomment to allow members of group wheel to execute any command" | ||||||
|  |           echo "remove '#' on  # %wheel ALL=(ALL) ALL and save the file" | ||||||
|  |           echo -e "then enter "$GREEN"gpasswd -a $USER wheel.""$NORMAL" | ||||||
|  |           echo "after logging in from the system, they can then run Wire-Py install again." | ||||||
|  |           read -n 1 -s -r -p $"Press Enter to exit" | ||||||
|  |           clear | ||||||
|  |           exit 0 | ||||||
|  |  | ||||||
|  |       fi | ||||||
|  |  | ||||||
|  |   elif grep -i '|manjaro\|garuda\|endeavour|' /etc/os-release > /dev/null 2>&1 | ||||||
|  |       then | ||||||
|  |           install_arch_d | ||||||
|  |  | ||||||
|  |   elif grep -i 'fedora' /etc/os-release > /dev/null 2>&1 | ||||||
|  |     then | ||||||
|  |       clear | ||||||
|  |       mkdir -p ~/.config/wire_py && touch ~/.config/wire_py/keys && cp -u settings ~/.config/wire_py/ && \ | ||||||
|  |       mkdir -p ~/.config/systemd/user && cp -u wg_start.service ~/.config/systemd/user/ && \ | ||||||
|  |       systemctl --user enable wg_start.service >/dev/null 2>&1 | ||||||
|  |       sudo cp -f org.sslcrypt.policy /usr/share/polkit-1/actions/ | ||||||
|  |       if [ $? -ne 0 ] | ||||||
|  |         then  | ||||||
|  |             systemctl --user disable wg_start.service | ||||||
|  |             rm -r ~/.config/wire_py && rm -r ~/.config/systemd | ||||||
|             exit 0 |             exit 0 | ||||||
|  |       else | ||||||
|  |         sudo dnf install python3-tkinter -y | ||||||
|  |         sudo cp -fv wirepy.py start_wg.py cls_mth_fc.py ssl_encrypt.py ssl_decrypt.py /usr/local/bin/ && \ | ||||||
|  |         sudo cp -uR lx-icons /usr/share/icons/ &&  sudo cp -uR TK-Themes /usr/share/ && \ | ||||||
|  |         sudo cp -u languages/de/*.mo /usr/share/locale/de/LC_MESSAGES/ && \ | ||||||
|  |         sudo cp -fv Wire-Py.desktop /usr/share/applications/  && \ | ||||||
|  |         sudo ln -sf /usr/local/bin/wirepy.py /usr/local/bin/wirepy | ||||||
|  |         sudo mkdir -p /usr/local/etc/ssl | ||||||
|  |         if [ ! -f /usr/local/etc/ssl/pwgk.pem ] | ||||||
|  |           then | ||||||
|  |             sudo openssl genrsa -out /usr/local/etc/ssl/pwgk.pem 4096 | ||||||
|         fi |         fi | ||||||
|  |  | ||||||
| elif grep -i '|manjaro\|garuda\|endeavour|' /etc/os-release > /dev/null 2>&1 |       fi | ||||||
|  |   elif grep -i 'suse' /etc/os-release > /dev/null 2>&1 | ||||||
|     then |     then | ||||||
|         install_arch_d |       clear | ||||||
|  |       mkdir -p ~/.config/wire_py && touch ~/.config/wire_py/keys && cp -u settings ~/.config/wire_py/ && \ | ||||||
|  |       mkdir -p ~/.config/systemd/user && cp -u wg_start.service ~/.config/systemd/user/ && \ | ||||||
| elif grep -i 'fedora' /etc/os-release > /dev/null 2>&1 |       systemctl --user enable wg_start.service >/dev/null 2>&1 | ||||||
|     then |       sudo cp -f org.sslcrypt.policy /usr/share/polkit-1/actions/ | ||||||
|         if ! which python3-tkinter &> /dev/null |       if [ $? -ne 0 ] | ||||||
|             then sudo dnf install python3-tkinter -y |         then  | ||||||
|  |           systemctl --user disable wg_start.service | ||||||
|                 sudo cp -u wg_main.py start_wg.py wg_func.py wirepy.py run_as open_gitea.py /usr/bin/ && \ |           rm -r ~/.config/wire_py && rm -r ~/.config/systemd | ||||||
|                 sudo mkdir -p /etc/wire_py && sudo touch /etc/wire_py/.keys && \ |           exit 0 | ||||||
|                 sudo cp -u settings.conf /etc/wire_py/ && \ |       else  | ||||||
|                 sudo cp -uR wp-icons /usr/share/icons/ && sudo cp -uR TK-Themes /usr/share/ && \ |         sudo cp -fv wirepy.py start_wg.py cls_mth_fc.py ssl_encrypt.py ssl_decrypt.py /usr/local/bin/ && \ | ||||||
|                 sudo chown -R root:root /etc/wire_py && sudo chmod 755 /etc/wire_py && \ |         sudo cp -uR lx-icons /usr/share/icons/ &&  sudo cp -uR TK-Themes /usr/share/ && \ | ||||||
|                 sudo ln -sf /usr/bin/wirepy.py /usr/local/bin/wirepy && \ |         sudo cp -u languages/de/*.mo /usr/share/locale/de/LC_MESSAGES/ && \ | ||||||
|                 sudo cp -u org.wirepy.policy /usr/share/polkit-1/actions/ && \ |         sudo cp -fv Wire-Py.desktop /usr/share/applications/ && \ | ||||||
|                 sudo cp -u Wire-Py.desktop /usr/share/applications/ && \ |         sudo ln -sf /usr/local/bin/wirepy.py /usr/local/bin/wirepy  | ||||||
|                 sudo cp -u wg_start.service /lib/systemd/system/ && \ |         sudo mkdir -p /usr/local/etc/ssl | ||||||
|                 sudo systemctl enable wg_start.service |         if [ ! -f /usr/local/etc/ssl/pwgk.pem ] | ||||||
|  |           then | ||||||
|  |             sudo openssl genrsa -out /usr/local/etc/ssl/pwgk.pem 4096 | ||||||
|         fi |         fi | ||||||
| elif grep -i 'suse' /etc/os-release > /dev/null 2>&1 |         if grep -i 'Tumbleweed' /etc/os-release > /dev/null 2>&1 | ||||||
|     then |           then | ||||||
|         if ! which python311-tk &> /dev/null |             sudo zypper install python313-tk | ||||||
|             then sudo zypper install python311-tk |         else | ||||||
|                  sudo cp -u wg_main.py start_wg.py wg_func.py wirepy.py run_as open_gitea.py /usr/bin/ && \ |           sudo zypper install python36-tk | ||||||
|                  sudo mkdir -p /etc/wire_py && sudo touch /etc/wire_py/.keys && \ |  | ||||||
|                  sudo cp -u settings.conf /etc/wire_py/ && \ |  | ||||||
|                  sudo cp -uR wp-icons /usr/share/icons/ && sudo cp -uR TK-Themes /usr/share/ && \ |  | ||||||
|                  sudo chown -R root:root /etc/wire_py && sudo chmod 755 /etc/wire_py && \ |  | ||||||
|                  sudo ln -sf /usr/bin/wirepy.py /usr/local/bin/wirepy && \ |  | ||||||
|                  sudo cp -u org.wirepy.policy /usr/share/polkit-1/actions/ && \ |  | ||||||
|                  sudo cp -u Wire-Py.desktop /usr/share/applications/ && \ |  | ||||||
|                  sudo cp -u wg_start.service /lib/systemd/system/ && \ |  | ||||||
|                  sudo systemctl enable wg_start.service |  | ||||||
|         fi |         fi | ||||||
|  |                | ||||||
|  |       fi | ||||||
|  |        | ||||||
|  |   else | ||||||
|  |       clear | ||||||
|  |       echo $"Your System could not be determined." | ||||||
|  |       echo | ||||||
|  |       read -n 1 -s -r -p $"Press Enter to exit" | ||||||
|  |       clear | ||||||
|  |       exit 0 | ||||||
|  |  | ||||||
|  |   fi | ||||||
|  |   #clear | ||||||
|  |   read -n 1 -s -r -p $"Press Enter to exit" | ||||||
|  |   clear | ||||||
|  |  | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | remove(){ | ||||||
|  |     sudo rm -f /usr/local/bin/wirepy /usr/local/bin/wirepy.py /usr/local/bin/start_wg.py \ | ||||||
|  |     /usr/local/bin/cls_mth_fc.py /usr/local/bin/ssl_encrypt.py /usr/local/bin/ssl_decrypt.py  | ||||||
|  |     if [ $? -ne 0 ] | ||||||
|  |       then | ||||||
|  |         exit 0 | ||||||
|  |     else  | ||||||
|  |       systemctl --user disable wg_start.service | ||||||
|  |       rm -r ~/.config/wire_py && rm -r ~/.config/systemd | ||||||
|  |       sudo rm /usr/share/applications/Wire-Py.desktop | ||||||
|  |       sudo rm /usr/share/locale/de/LC_MESSAGES/languages/de/wirepy.mo | ||||||
|  |       sudo rm -r /usr/local/etc/ssl | ||||||
|  |       which syncpy >/dev/null | ||||||
|  |       if [ $? -ne 0 ] | ||||||
|  |         then  | ||||||
|  |           sudo rm -r /usr/share/icons/lx-icons &&  sudo rm -r /usr/share/TK-Themes   | ||||||
|  |  | ||||||
|  |       fi | ||||||
|  |  | ||||||
|  |       echo | ||||||
|  |       read -p "Press Enter to exit..." | ||||||
|  |  | ||||||
|  |     fi | ||||||
|  |  | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | which wirepy >/dev/null | ||||||
|  | if [ $? -eq 0 ] | ||||||
|  |     then | ||||||
|  |         echo "Do you want to update/reinstall or uninstall wirepy?" | ||||||
|  |         echo | ||||||
|  |         echo "Update/reinstall: press y, uninstall press r" | ||||||
|  |         echo | ||||||
|  |         read -n 1 -s -r -p "Cancel with any other key..."  result | ||||||
|  |         case $result in | ||||||
|  |             [y]* ) clear; install; exit;; | ||||||
|  |             [Y]* ) clear; install; exit;; | ||||||
|  |             [j]* ) clear; install; exit;; | ||||||
|  |             [J]* ) clear; install; exit;; | ||||||
|  |             [r]* ) clear; remove; exit;; | ||||||
|  |             [R]* ) clear; remove; exit;; | ||||||
|  |         esac | ||||||
|  |         clear | ||||||
| else | else | ||||||
|     clear |   install        | ||||||
|     echo $"Your System could not be determined." |     | ||||||
|     echo |  | ||||||
|     read -n 1 -s -r -p $"To close the window press a button" |  | ||||||
|     clear |  | ||||||
|     exit 0 |  | ||||||
| fi         | fi         | ||||||
| clear |  | ||||||
| read -n 1 -s -r -p $"To close the Window press a button" |  | ||||||
| clear |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								languages/de/wirepy.mo
									
									
									
									
									
										Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 5.7 KiB After Width: | Height: | Size: 5.7 KiB | 
| Before Width: | Height: | Size: 5.7 KiB After Width: | Height: | Size: 5.7 KiB | 
| Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 16 KiB | 
| Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 13 KiB | 
| Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 14 KiB | 
| Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 14 KiB | 
| Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 16 KiB | 
| Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB | 
| Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 14 KiB | 
| Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 12 KiB | 
| Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 10 KiB | 
| Before Width: | Height: | Size: 28 KiB After Width: | Height: | Size: 28 KiB | 
| Before Width: | Height: | Size: 32 KiB After Width: | Height: | Size: 32 KiB | 
| Before Width: | Height: | Size: 18 KiB After Width: | Height: | Size: 18 KiB | 
| Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 24 KiB | 
| Before Width: | Height: | Size: 26 KiB After Width: | Height: | Size: 26 KiB | 
| Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 24 KiB | 
| Before Width: | Height: | Size: 18 KiB After Width: | Height: | Size: 18 KiB | 
| Before Width: | Height: | Size: 5.1 KiB After Width: | Height: | Size: 5.1 KiB | 
| Before Width: | Height: | Size: 1.6 KiB After Width: | Height: | Size: 1.6 KiB | 
| Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 6.1 KiB | 
| Before Width: | Height: | Size: 6.0 KiB After Width: | Height: | Size: 6.0 KiB | 
| Before Width: | Height: | Size: 5.7 KiB After Width: | Height: | Size: 5.7 KiB | 
| Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 6.1 KiB | 
| Before Width: | Height: | Size: 6.2 KiB After Width: | Height: | Size: 6.2 KiB | 
| Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 6.1 KiB | 
| Before Width: | Height: | Size: 5.8 KiB After Width: | Height: | Size: 5.8 KiB | 
| Before Width: | Height: | Size: 5.7 KiB After Width: | Height: | Size: 5.7 KiB | 
| Before Width: | Height: | Size: 2.2 KiB After Width: | Height: | Size: 2.2 KiB | 
| Before Width: | Height: | Size: 7.3 KiB After Width: | Height: | Size: 7.3 KiB | 
| Before Width: | Height: | Size: 17 KiB After Width: | Height: | Size: 17 KiB | 
| Before Width: | Height: | Size: 7.0 KiB After Width: | Height: | Size: 7.0 KiB | 
| Before Width: | Height: | Size: 7.3 KiB After Width: | Height: | Size: 7.3 KiB | 
| Before Width: | Height: | Size: 7.3 KiB After Width: | Height: | Size: 7.3 KiB | 
| Before Width: | Height: | Size: 7.1 KiB After Width: | Height: | Size: 7.1 KiB | 
| Before Width: | Height: | Size: 17 KiB After Width: | Height: | Size: 17 KiB | 
| Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 2.7 KiB | 
| Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 2.7 KiB | 
| Before Width: | Height: | Size: 9.0 KiB After Width: | Height: | Size: 9.0 KiB | 
| Before Width: | Height: | Size: 8.2 KiB After Width: | Height: | Size: 8.2 KiB | 
| Before Width: | Height: | Size: 8.2 KiB After Width: | Height: | Size: 8.2 KiB | 
| Before Width: | Height: | Size: 8.3 KiB After Width: | Height: | Size: 8.3 KiB | 
| Before Width: | Height: | Size: 9.0 KiB After Width: | Height: | Size: 9.0 KiB | 
| Before Width: | Height: | Size: 8.7 KiB After Width: | Height: | Size: 8.7 KiB | 
| Before Width: | Height: | Size: 8.4 KiB After Width: | Height: | Size: 8.4 KiB | 
| @@ -1,5 +0,0 @@ | |||||||
| #!/usr/bin/python3 |  | ||||||
|  |  | ||||||
| import webbrowser |  | ||||||
|  |  | ||||||
| webbrowser.open('https://git.ilunix.de/punix/Wire-Py') |  | ||||||
							
								
								
									
										42
									
								
								org.sslcrypt.policy
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <!DOCTYPE policyconfig PUBLIC "-//freedesktop//DTD PolicyKit Policy Configuration 1.0//EN" "http://www.freedesktop.org/standards/PolicyKit/1/policyconfig.dtd"> | ||||||
|  |  | ||||||
|  | <!-- | ||||||
|  | Policy definitions for ssl_encrypt and ssl_decrypt | ||||||
|  |  | ||||||
|  | Copyright (C) 2025 Désiré Werner Menrath <polunga40@unity-mail.de> | ||||||
|  |  | ||||||
|  | This library is free software; you can redistribute it and/or | ||||||
|  | modify it under the terms of the GNU Lesser General Public | ||||||
|  | License as published by the Free Software Foundation; either | ||||||
|  | version 2.1 of the License, or (at your option) any later version. | ||||||
|  |  | ||||||
|  | This library is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | ||||||
|  | Lesser General Public License for more details. | ||||||
|  |  | ||||||
|  | You should have received a copy of the GNU Lesser General Public | ||||||
|  | License along with this library.  If not, see | ||||||
|  | <http://www.gnu.org/licenses/>. | ||||||
|  | --> | ||||||
|  |  | ||||||
|  | <policyconfig> | ||||||
|  |   <action id="org.ssl_encrypt"> | ||||||
|  |     <defaults> | ||||||
|  |       <allow_any>auth_admin_keep</allow_any> | ||||||
|  |       <allow_active>yes</allow_active> | ||||||
|  |     </defaults> | ||||||
|  |     <annotate key="org.freedesktop.policykit.exec.path">/usr/local/bin/ssl_encrypt.py</annotate> | ||||||
|  |   </action> | ||||||
|  |  | ||||||
|  |   <action id="org.ssl_decrypt"> | ||||||
|  |     <defaults> | ||||||
|  |       <allow_any>auth_admin_keep</allow_any> | ||||||
|  |       <allow_inactive>auth_admin_keep</allow_inactive> | ||||||
|  |       <allow_active>yes</allow_active> | ||||||
|  |     </defaults> | ||||||
|  |     <annotate key="org.freedesktop.policykit.exec.path">/usr/local/bin/ssl_decrypt.py</annotate> | ||||||
|  |      | ||||||
|  |   </action> | ||||||
|  | </policyconfig> | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="UTF-8"?> |  | ||||||
| <!DOCTYPE policyconfig PUBLIC "-//freedesktop//DTD PolicyKit Policy Configuration 1.0//EN" "http://www.freedesktop.org/standards/PolicyKit/1/policyconfig.dtd"> |  | ||||||
| <policyconfig> |  | ||||||
|   <vendor>Project Wire-Py</vendor> |  | ||||||
|   <vendor_url>https://git.ilunix.de/punix/Wire-Py</vendor_url> |  | ||||||
|   <icon_name>wg-vpn</icon_name> |  | ||||||
|   <action id="org.wirepy"> |  | ||||||
|     <defaults> |  | ||||||
|       <allow_any>auth_admin_keep</allow_any> |  | ||||||
|       <allow_inactive>auth_admin_keep</allow_inactive> |  | ||||||
|       <allow_active>yes</allow_active> |  | ||||||
|     </defaults> |  | ||||||
|     <annotate key="org.freedesktop.policykit.exec.path">/usr/bin/wg_main.py</annotate> |  | ||||||
|     <annotate key="org.freedesktop.policykit.exec.allow_gui">true</annotate> |  | ||||||
|   </action> |  | ||||||
| </policyconfig> |  | ||||||
							
								
								
									
										35
									
								
								ssl_decrypt.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | |||||||
|  | #!/usr/bin/python3 | ||||||
|  | """ This Script decrypt Wireguard files for Wirepy users """ | ||||||
|  |  | ||||||
|  | import os | ||||||
|  | import shutil | ||||||
|  | from pathlib import Path | ||||||
|  | from subprocess import check_call | ||||||
|  |  | ||||||
|  | uname: Path = Path("/tmp/.log_user") | ||||||
|  |  | ||||||
|  | with open(uname, "r", encoding="utf-8") as f: | ||||||
|  |     log_name = f.readline() | ||||||
|  |  | ||||||
|  | # Folder Name "tlecdewg" = Tunnel Encrypt Decrypt Wireguard | ||||||
|  | folder_path: Path = Path("/tmp/tlecdcwg/") | ||||||
|  | keyfile: Path = Path(f"/home/{log_name}/.config/wire_py/pbwgk.pem") | ||||||
|  | PKEYFILE: Path = "/usr/local/etc/ssl/pwgk.pem" | ||||||
|  |  | ||||||
|  | if not keyfile.is_file(): | ||||||
|  |  | ||||||
|  |     check_call(["openssl", "rsa", "-in", PKEYFILE, "-out", keyfile, "-outform", "PEM", "-pubout"]) | ||||||
|  |     shutil.chown(keyfile, 1000, 1000) | ||||||
|  |  | ||||||
|  | folder_path2 = f"/home/{log_name}/.config/wire_py/" | ||||||
|  | detl: list[str] = os.listdir(folder_path2) | ||||||
|  | os.chdir(folder_path2) | ||||||
|  | detl.remove("keys") | ||||||
|  | detl.remove("settings") | ||||||
|  | if os.path.exists(f"{folder_path2}pbwgk.pem"): | ||||||
|  |     detl.remove("pbwgk.pem") | ||||||
|  |     for detunnels in detl: | ||||||
|  |         tlname2 = f"{detunnels[:-4]}.conf" | ||||||
|  |         extpath = f"{folder_path}/{tlname2}" | ||||||
|  |         check_call(["openssl", "pkeyutl", "-decrypt", "-inkey", PKEYFILE, "-in", detunnels, "-out", extpath]) | ||||||
|  |         shutil.chown(extpath, 1000, 1000) | ||||||
							
								
								
									
										47
									
								
								ssl_encrypt.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,47 @@ | |||||||
|  | #!/usr/bin/python3 | ||||||
|  | """ This Script encrypt Wireguardfiles for Wirepy users for more Security """ | ||||||
|  |  | ||||||
|  | import os | ||||||
|  | import shutil | ||||||
|  | from pathlib import Path | ||||||
|  | from subprocess import check_call | ||||||
|  |  | ||||||
|  | uname: Path = Path("/tmp/.log_user") | ||||||
|  |  | ||||||
|  | with open(uname, "r", encoding="utf-8") as f: | ||||||
|  |     log_name: str = f.readline() | ||||||
|  |  | ||||||
|  | keyfile: Path = Path(f"/home/{log_name}/.config/wire_py/pbwgk.pem") | ||||||
|  | folder_path: Path = Path("/tmp/tlecdcwg/") | ||||||
|  | PKEYFILE = "/usr/local/etc/ssl/pwgk.pem" | ||||||
|  |  | ||||||
|  | if not keyfile.is_file(): | ||||||
|  |  | ||||||
|  |     check_call(["openssl", "rsa", "-in", PKEYFILE, "-out", keyfile, "-outform", "PEM", "-pubout"]) | ||||||
|  |     shutil.chown(keyfile, 1000, 1000) | ||||||
|  |  | ||||||
|  |     if folder_path.exists(): | ||||||
|  |         tl = os.listdir(f"{folder_path}") | ||||||
|  |         CPTH: str = f"{keyfile}" | ||||||
|  |         CRYPTFILES: str = CPTH[:-9] | ||||||
|  |  | ||||||
|  |         if keyfile.exists() and len(tl) != 0: | ||||||
|  |             for tunnels in tl: | ||||||
|  |                 sourcetl: str = f"{folder_path}/{tunnels}" | ||||||
|  |                 tlname: str = f"{CRYPTFILES}{tunnels[:-5]}.dat" | ||||||
|  |                 check_call(["openssl", "pkeyutl", "-encrypt", "-inkey", keyfile, "-pubin", "-in", sourcetl, "-out", | ||||||
|  |                             tlname,]) | ||||||
|  |  | ||||||
|  | else: | ||||||
|  |  | ||||||
|  |     if folder_path.exists(): | ||||||
|  |         tl: list[str] = os.listdir(f"{folder_path}") | ||||||
|  |         CPTH: str = f"{keyfile}" | ||||||
|  |         CRYPTFILES: str = CPTH[:-9] | ||||||
|  |  | ||||||
|  |         if keyfile.exists() and len(tl) != 0: | ||||||
|  |             for tunnels in tl: | ||||||
|  |                 sourcetl: str = f"{folder_path}/{tunnels}" | ||||||
|  |                 tlname: str = f"{CRYPTFILES}{tunnels[:-5]}.dat" | ||||||
|  |                 check_call(["openssl", "pkeyutl", "-encrypt", "-inkey", keyfile, "-pubin", "-in", sourcetl, "-out", | ||||||
|  |                             tlname]) | ||||||
							
								
								
									
										16
									
								
								start_wg.py
									
									
									
									
									
								
							
							
						
						| @@ -1,14 +1,18 @@ | |||||||
| #!/usr/bin/python3 | #!/usr/bin/python3 | ||||||
| from subprocess import check_call | """ | ||||||
|  |  This script belongs to wirepy and is for the auto start of the tunnel   | ||||||
|  | """ | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  | from subprocess import check_call | ||||||
|  |  | ||||||
| path_to_file = Path('/etc/wire_py/settings.conf') | path_to_file = Path(Path.home() / ".config/wire_py/settings") | ||||||
|  |  | ||||||
| with open(path_to_file, 'r') as a_con: | with open(path_to_file, "r", encoding="utf-8") as a_con: | ||||||
|  |  | ||||||
|  |     # This function is for the independent autostarted of the previously selected tunnel | ||||||
|     lines = a_con.readlines() |     lines = a_con.readlines() | ||||||
|     a_con = lines[7].strip() |     a_con = lines[7].strip() | ||||||
|     if a_con != 'off': |     if a_con != "off": | ||||||
|         check_call(['nmcli', 'connection', 'up', a_con]) |         check_call(["nmcli", "connection", "up", a_con]) | ||||||
|     else: |     else: | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										267
									
								
								wg_func.py
									
									
									
									
									
								
							
							
						
						| @@ -1,267 +0,0 @@ | |||||||
| """ Wireguard Classes and Method for Wire-Py """ |  | ||||||
|  |  | ||||||
| import os |  | ||||||
| import shutil |  | ||||||
| import subprocess |  | ||||||
| import time |  | ||||||
| import tkinter as tk |  | ||||||
| import zipfile |  | ||||||
| from datetime import datetime |  | ||||||
| from pathlib import Path |  | ||||||
| from subprocess import check_call |  | ||||||
| from tkinter import ttk |  | ||||||
| import requests |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ''' 1 = 1. Year, 09 = Month of the Year, 2924 = Day and Year of the Year ''' |  | ||||||
| version = 'v. 1.11.0824' |  | ||||||
|  |  | ||||||
| path_to_file2 = Path('/etc/wire_py/settings.conf') |  | ||||||
| _u = Path.read_text(Path('/tmp/_u')) |  | ||||||
|  |  | ||||||
| UPDATE_API_URL = 'https://git.ilunix.de/api/v1/repos/punix/Wire-Py/releases' |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WirePyUpdate: |  | ||||||
|     @staticmethod |  | ||||||
|     def api_down(): |  | ||||||
|         try: |  | ||||||
|             response = requests.get(UPDATE_API_URL) |  | ||||||
|             response_dict = response.json() |  | ||||||
|             response_dict = response_dict[0] |  | ||||||
|             with open(path_to_file2, 'r') as set_file: |  | ||||||
|                 set_file = set_file.read() |  | ||||||
|                 if 'on\n' in set_file: |  | ||||||
|                     if version[3:] != response_dict['tag_name']: |  | ||||||
|                         return response_dict['tag_name'] |  | ||||||
|                     else: |  | ||||||
|                         return 'No Updates' |  | ||||||
|                 else: |  | ||||||
|                     return 'False' |  | ||||||
|         except requests.exceptions.ConnectionError: |  | ||||||
|             return 'No Internet Connection!' |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def download(): |  | ||||||
|         try: |  | ||||||
|             url = f'https://git.ilunix.de/punix/Wire-Py/archive/{res}.zip' |  | ||||||
|             to_down = 'wget -qP ' + str(_u) + ' ' + url |  | ||||||
|             result = subprocess.call(to_down, shell=True) |  | ||||||
|             if result == 0: |  | ||||||
|                 shutil.chown(str(_u) + f'/{res}.zip', 1000, 1000) |  | ||||||
|                 """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_vpn.png' |  | ||||||
|                 wt = 'Download Successful' |  | ||||||
|                 msg_t = 'Your zip file is in home directory' |  | ||||||
|                 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/error.png' |  | ||||||
|                 ii = r'/usr/share/icons/wp-icons/48/wg_msg.png' |  | ||||||
|                 wt = 'Download error' |  | ||||||
|                 msg_t = 'Download failed! Please try again' |  | ||||||
|                 msg_window(iw, ii, wt, msg_t) |  | ||||||
|         except subprocess.CalledProcessError: |  | ||||||
|             """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 = 'Download error' |  | ||||||
|             msg_t = 'Download failed! No internet connection!' |  | ||||||
|             msg_window(iw, ii, wt, msg_t) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| res = WirePyUpdate.api_down() |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def msg_window(img_w, img_i, w_title, w_txt, txt2=None, com=None): |  | ||||||
|     """ |  | ||||||
|     Function for different message windows for the user. with 4 arguments to be passed. |  | ||||||
|     To create messages with your own images, icons, and titles. As an alternative to Python Messagebox. |  | ||||||
|     Paths to images must be specified: r'/usr/share/icons/wp-icons/64/info.png' |  | ||||||
|     img_w = Image for Tk Window |  | ||||||
|     img_i = Image for Icon |  | ||||||
|     w_title = Windows Title |  | ||||||
|     w_txt = Text for Tk Window |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     msg = tk.Toplevel() |  | ||||||
|     msg.resizable(width=False, height=False) |  | ||||||
|     msg.title(w_title) |  | ||||||
|     msg.configure(pady=15, padx=15) |  | ||||||
|     msg.img = tk.PhotoImage(file=img_w) |  | ||||||
|     msg.i_window = tk.Label(msg, image=msg.img) |  | ||||||
|  |  | ||||||
|     label = tk.Label(msg, text=w_txt) |  | ||||||
|  |  | ||||||
|     label.grid(column=1, row=0) |  | ||||||
|  |  | ||||||
|     if txt2 != None and com != None: |  | ||||||
|         label.config(font=('Ubuntu', 11), padx=15, justify='left') |  | ||||||
|         msg.i_window.grid(column=0, row=0, sticky='nw') |  | ||||||
|         button2 = ttk.Button(msg, text=f'{txt2}', command=com, padding=4) |  | ||||||
|         button2.grid(column=0, row=1, sticky='e', columnspan=2) |  | ||||||
|         button = ttk.Button(msg, text='OK', command=msg.destroy, padding=4) |  | ||||||
|         button.grid(column=0, row=1, sticky='w', columnspan=2) |  | ||||||
|  |  | ||||||
|     else: |  | ||||||
|         label.config(font=('Ubuntu', 11), padx=15) |  | ||||||
|         msg.i_window.grid(column=0, row=0) |  | ||||||
|         button = ttk.Button(msg, text='OK', command=msg.destroy, padding=4) |  | ||||||
|         button.grid(column=0, columnspan=2, row=1) |  | ||||||
|  |  | ||||||
|     img_i = tk.PhotoImage(file=img_i) |  | ||||||
|     msg.iconphoto(True, img_i) |  | ||||||
|     msg.columnconfigure(0, weight=1) |  | ||||||
|     msg.rowconfigure(0, weight=1) |  | ||||||
|     msg.winfo_toplevel() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Tunnel: |  | ||||||
|  |  | ||||||
|     """ |  | ||||||
|     Class of Methods for Wire-Py |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     """ |  | ||||||
|     The config file is packed into a dictionary, |  | ||||||
|     to display the values Address , DNS and Peer in the labels |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def con_to_dict(cls, file): |  | ||||||
|  |  | ||||||
|         dictlist = [] |  | ||||||
|         for lines in file.readlines(): |  | ||||||
|             line_plit = lines.split() |  | ||||||
|             dictlist = dictlist + line_plit |  | ||||||
|         dictlist.remove('[Interface]') |  | ||||||
|         dictlist.remove('[Peer]') |  | ||||||
|         for items in dictlist: |  | ||||||
|             if items == '=': |  | ||||||
|                 dictlist.remove(items) |  | ||||||
|  |  | ||||||
|         ''' Here is the beginning (Loop) of convert List to Dictionary ''' |  | ||||||
|         for _ in dictlist: |  | ||||||
|             a = [dictlist[0], dictlist[1]] |  | ||||||
|             b = [dictlist[2], dictlist[3]] |  | ||||||
|             c = [dictlist[4], dictlist[5]] |  | ||||||
|             d = [dictlist[6], dictlist[7]] |  | ||||||
|             e = [dictlist[8], dictlist[9]] |  | ||||||
|             f = [dictlist[10], dictlist[11]] |  | ||||||
|             g = [dictlist[12], dictlist[13]] |  | ||||||
|             h = [dictlist[14], dictlist[15]] |  | ||||||
|             new_list = [a, b, c, d, e, f, g, h] |  | ||||||
|             final_dict = {} |  | ||||||
|             for elements in new_list: |  | ||||||
|                 final_dict[elements[0]] = elements[1] |  | ||||||
|  |  | ||||||
|             ''' end... result a Dictionary ''' |  | ||||||
|  |  | ||||||
|         address = final_dict['Address'] |  | ||||||
|         dns = final_dict['DNS'] |  | ||||||
|         if ',' in dns: |  | ||||||
|             dns = dns[:-1] |  | ||||||
|         endpoint = final_dict['Endpoint'] |  | ||||||
|         if 'PresharedKey' in final_dict: |  | ||||||
|             pre_key = final_dict['PresharedKey'] |  | ||||||
|         else: |  | ||||||
|             pre_key = final_dict['PreSharedKey'] |  | ||||||
|         return address, dns, endpoint, pre_key |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     """ |  | ||||||
|     Shows the Active Tunnel |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def active(): |  | ||||||
|  |  | ||||||
|         active = os.popen('nmcli con show --active | grep -iPo "(.*)(wireguard)"').read().split() |  | ||||||
|         if not active: |  | ||||||
|             active = '' |  | ||||||
|         else: |  | ||||||
|             active = active[0] |  | ||||||
|  |  | ||||||
|         return active |  | ||||||
|  |  | ||||||
|     """ |  | ||||||
|     Shows all existing Wireguard tunnels |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def list(): |  | ||||||
|         wg_s = os.popen('nmcli con show | grep -iPo "(.*)(wireguard)"').read().split() |  | ||||||
|  |  | ||||||
|         ''' tl = Tunnel list # Show of 4.Element in list ''' |  | ||||||
|         tl = wg_s[::3] |  | ||||||
|         return tl |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     """ |  | ||||||
|     This will export the tunnels. |  | ||||||
|     A zipfile with current date and time is created |  | ||||||
|     in the user's home directory with correct right |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def export(): |  | ||||||
|         _u1 = str(_u[6:]) |  | ||||||
|         now_time = datetime.now() |  | ||||||
|         now_datetime = now_time.strftime('wg-exp-' + '%m-%d-%Y' + '-' + '%H:%M') |  | ||||||
|         tl = Tunnel.list() |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             if len(tl) != 0: |  | ||||||
|                 wg_tar = str(_u) + '/' + now_datetime |  | ||||||
|                 shutil.copytree('/etc/wire_py', '/tmp/wire_py', dirs_exist_ok=True) |  | ||||||
|                 source = Path('/tmp/wire_py') |  | ||||||
|                 Path.unlink(Path(source) / 'wg_py', missing_ok=True) |  | ||||||
|                 Path.unlink(Path(source) / '.keys', missing_ok=True) |  | ||||||
|                 Path.unlink(Path(source) / 'settings.conf', missing_ok=True) |  | ||||||
|                 shutil.make_archive(wg_tar, 'zip', source) |  | ||||||
|                 shutil.chown(wg_tar + '.zip', 1000, 1000) |  | ||||||
|                 shutil.rmtree(source) |  | ||||||
|                 with zipfile.ZipFile((wg_tar + '.zip'), 'r') as zf: |  | ||||||
|                     if len(zf.namelist()) != 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_vpn.png' |  | ||||||
|                         wt = 'Export Successful' |  | ||||||
|                         msg_t = 'Your zip file is in home directory' |  | ||||||
|                         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/error.png' |  | ||||||
|                         ii = r'/usr/share/icons/wp-icons/48/wg_msg.png' |  | ||||||
|                         wt = 'Export error' |  | ||||||
|                         msg_t = 'Export failed! Please try again' |  | ||||||
|                         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) |  | ||||||
|  |  | ||||||
|         except TypeError: |  | ||||||
|             pass |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def if_tip(): |  | ||||||
|         with open(path_to_file2, 'r') as set_file2: |  | ||||||
|                 lines2 = set_file2.readlines() |  | ||||||
|                 if 'False\n' in lines2: |  | ||||||
|                     return False |  | ||||||
|                 else: |  | ||||||
|                     return True |  | ||||||
|  |  | ||||||
| tips = Tunnel.if_tip() |  | ||||||
							
								
								
									
										1158
									
								
								wg_main.py
									
									
									
									
									
								
							
							
						
						| @@ -5,6 +5,6 @@ After=network-online.target | |||||||
| [Service] | [Service] | ||||||
| Type=oneshot | Type=oneshot | ||||||
| ExecStartPre=/bin/sleep 5 | ExecStartPre=/bin/sleep 5 | ||||||
| ExecStart=/usr/bin/start_wg.py | ExecStart=/usr/local/bin/start_wg.py | ||||||
| [Install] | [Install] | ||||||
| WantedBy=multi-user.target | WantedBy=default.target | ||||||
|   | |||||||
							
								
								
									
										875
									
								
								wirepy.py
									
									
									
									
									
								
							
							
						
						| @@ -1,6 +1,875 @@ | |||||||
| #!/usr/bin/python3 | #!/usr/bin/python3 | ||||||
| from subprocess import check_call | """ | ||||||
|  | this script is a simple GUI for managing Wireguard Tunnels | ||||||
|  | """ | ||||||
|  | import gettext | ||||||
|  | import locale | ||||||
|  | import os | ||||||
|  | import shutil | ||||||
|  | import subprocess | ||||||
|  | import sys | ||||||
|  | import tkinter as tk | ||||||
|  | import webbrowser | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  | from subprocess import check_call | ||||||
|  | from tkinter import TclError, filedialog, ttk | ||||||
|  |  | ||||||
| Path.write_text(Path('/tmp/_u'), str(Path.home())) | from cls_mth_fc import (Create, GiteaUpdate, Tooltip, Tunnel, LxTools) | ||||||
| check_call(['pkexec', '/usr/bin/wg_main.py']) |  | ||||||
|  | LxTools.uos() | ||||||
|  | Create.dir_and_files() | ||||||
|  | Create.make_dir() | ||||||
|  | Create.decrypt() | ||||||
|  |  | ||||||
|  | tcl_path: Path = Path("/usr/share/TK-Themes") | ||||||
|  | set_file: Path = Path(Path.home() / ".config/wire_py/settings") | ||||||
|  | tips = LxTools.if_tip(set_file) | ||||||
|  | folder_path: Path = Path("/tmp/tlecdcwg/") | ||||||
|  | user_file = Path("/tmp/.log_user") | ||||||
|  |  | ||||||
|  | # 1 = 1. Year, 09 = Month of the Year, 2924 = Day and Year of the Year | ||||||
|  | VERSION: str = "v. 2.04.1725" | ||||||
|  |  | ||||||
|  | res = GiteaUpdate.api_down("https://git.ilunix.de/api/v1/repos/punix/Wire-Py/releases", VERSION, set_file) | ||||||
|  |  | ||||||
|  | # Translate | ||||||
|  | APP = "wirepy" | ||||||
|  | LOCALE_DIR = "/usr/share/locale/" | ||||||
|  | locale.bindtextdomain(APP, LOCALE_DIR) | ||||||
|  | gettext.bindtextdomain(APP, LOCALE_DIR) | ||||||
|  | gettext.textdomain(APP) | ||||||
|  | _ = gettext.gettext | ||||||
|  |  | ||||||
|  | img_w: str = r"/usr/share/icons/lx-icons/64/info.png" | ||||||
|  | img_i: str = r"/usr/share/icons/lx-icons/48/wg_vpn.png" | ||||||
|  | img_w2: str = r"/usr/share/icons/lx-icons/64/error.png" | ||||||
|  | img_i2: str = r"/usr/share/icons/lx-icons/48/wg_msg.png" | ||||||
|  | sl: str = _("Select tunnel") | ||||||
|  | rnp: str = _("Renaming not possible") | ||||||
|  | ie:str = _("Import Error") | ||||||
|  | pfit: str = _("Please first import tunnel") | ||||||
|  | pstl: str = _("Please select a tunnel from the list") | ||||||
|  |  | ||||||
|  | LxTools.sigi(folder_path, user_file) | ||||||
|  |  | ||||||
|  | class Wirepy(tk.Tk): | ||||||
|  |     """ | ||||||
|  |     Class Wirepy this is the Main Window of wirepy | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |         self.my_tool_tip = None | ||||||
|  |         self.x_width = 600 | ||||||
|  |         self.y_height = 383 | ||||||
|  |         self.monitor_center_x = int(self.winfo_screenwidth() / 2 - (self.x_width / 2)) | ||||||
|  |         self.monitor_center_y = int(self.winfo_screenheight() / 2 - (self.y_height / 2)) | ||||||
|  |         self.resizable(width=False, height=False) | ||||||
|  |         self.title("Wire-Py") | ||||||
|  |         self.geometry(f"{self.x_width}x{self.y_height}+{self.monitor_center_x}+{self.monitor_center_y}") | ||||||
|  |         self.columnconfigure(0, weight=1) | ||||||
|  |         self.rowconfigure(0, weight=1) | ||||||
|  |  | ||||||
|  |         self.style = ttk.Style(self) | ||||||
|  |         self.tk.call("source", f"{tcl_path}/water.tcl") | ||||||
|  |         with open(set_file, "r", encoding="utf-8") 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 the disk | ||||||
|  |         self.wg_icon = tk.PhotoImage(file=r"/usr/share/icons/lx-icons/48/wg_vpn.png") | ||||||
|  |  | ||||||
|  |         # Set it as the window icon | ||||||
|  |         self.iconphoto(True, self.wg_icon) | ||||||
|  |  | ||||||
|  |         FrameWidgets(self).grid() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FrameWidgets(ttk.Frame): | ||||||
|  |     """ | ||||||
|  |     ttk frame class for better structure | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, container, **kwargs): | ||||||
|  |         super().__init__(container, **kwargs) | ||||||
|  |  | ||||||
|  |         self.lb_tunnel = None | ||||||
|  |         self.btn_stst = None | ||||||
|  |         self.endpoint = None | ||||||
|  |         self.dns = None | ||||||
|  |         self.address = None | ||||||
|  |         self.auto_con = None | ||||||
|  |         self.wg_vpn_start = tk.PhotoImage(file=r"/usr/share/icons/lx-icons/48/wg_vpn-start.png") | ||||||
|  |         self.wg_vpn_stop = tk.PhotoImage(file=r"/usr/share/icons/lx-icons/48/wg_vpn-stop.png") | ||||||
|  |         self.imp_pic = tk.PhotoImage(file=r"/usr/share/icons/lx-icons/48/wg_import.png") | ||||||
|  |         self.tr_pic = tk.PhotoImage(file=r"/usr/share/icons/lx-icons/48/wg_trash.png") | ||||||
|  |         self.exp_pic = tk.PhotoImage(file=r"/usr/share/icons/lx-icons/48/wg_export.png") | ||||||
|  |         self.warning_pic = tk.PhotoImage(file=r"/usr/share/icons/lx-icons/64/error.png") | ||||||
|  |  | ||||||
|  |         def update(): | ||||||
|  |             """ | ||||||
|  |             Set on or off in file | ||||||
|  |             """ | ||||||
|  |             if set_update.get() == 1: | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as set_f2: | ||||||
|  |                     lines2 = set_f2.readlines() | ||||||
|  |                     lines2[1] = "off\n" | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as set_f2: | ||||||
|  |                     set_f2.writelines(lines2) | ||||||
|  |  | ||||||
|  |             if set_update.get() == 0: | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as set_f2: | ||||||
|  |                     lines2 = set_f2.readlines() | ||||||
|  |                     lines2[1] = "on\n" | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as set_f2: | ||||||
|  |                     set_f2.writelines(lines2) | ||||||
|  |  | ||||||
|  |         def theme_change_light(): | ||||||
|  |             """ | ||||||
|  |             Set a light theme | ||||||
|  |             """ | ||||||
|  |             if self.tk.call("ttk::style", "theme", "use") == "water-dark": | ||||||
|  |                 self.tk.call("set_theme", "light") | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as theme_set2: | ||||||
|  |                     lines3 = theme_set2.readlines() | ||||||
|  |                     lines3[3] = "light\n" | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as theme_set2: | ||||||
|  |                     theme_set2.writelines(lines3) | ||||||
|  |                 self.color_label() | ||||||
|  |  | ||||||
|  |         def theme_change_dark(): | ||||||
|  |             """ | ||||||
|  |             Set a dark theme | ||||||
|  |             """ | ||||||
|  |             if not self.tk.call("ttk::style", "theme", "use") == "water-dark": | ||||||
|  |                 self.tk.call("set_theme", "dark") | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as theme_set2: | ||||||
|  |                     lines4 = theme_set2.readlines() | ||||||
|  |                     lines4[3] = "dark\n" | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as theme_set2: | ||||||
|  |                     theme_set2.writelines(lines4) | ||||||
|  |                 self.color_label() | ||||||
|  |  | ||||||
|  |         def tooltip(): | ||||||
|  |             """ | ||||||
|  |             Set True or False in a file | ||||||
|  |             """ | ||||||
|  |             if set_tip.get(): | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as set_f2: | ||||||
|  |                     lines2 = set_f2.readlines() | ||||||
|  |                     lines2[5] = "False\n" | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as set_f2: | ||||||
|  |                     set_f2.writelines(lines2) | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as set_f2: | ||||||
|  |                     lines2 = set_f2.readlines() | ||||||
|  |                     lines2[5] = "True\n" | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as set_f2: | ||||||
|  |                     set_f2.writelines(lines2) | ||||||
|  |  | ||||||
|  |         def info(): | ||||||
|  |             def link_btn(): | ||||||
|  |                 webbrowser.open("https://git.ilunix.de/punix/Wire-Py") | ||||||
|  |  | ||||||
|  |             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") | ||||||
|  |  | ||||||
|  |             LxTools.msg_window(img_i, img_i, _("Info"), msg_t, _("Go to Wire-Py git"), link_btn) | ||||||
|  |  | ||||||
|  |         # Frame for Menu | ||||||
|  |         self.menu_frame = ttk.Frame(self) | ||||||
|  |         self.menu_frame.configure(relief="flat") | ||||||
|  |         self.menu_frame.grid(column=0, row=0, columnspan=4, sticky="w") | ||||||
|  |  | ||||||
|  |         # 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, rowspan=4, padx=10) | ||||||
|  |  | ||||||
|  |         Tooltip(self.version_lb, f"Version: {VERSION[2:]}", tips) | ||||||
|  |  | ||||||
|  |         self.options_btn = ttk.Menubutton(self.menu_frame, text=_("Options")) | ||||||
|  |         self.options_btn.grid(column=1, columnspan=1, row=0) | ||||||
|  |  | ||||||
|  |         Tooltip(self.options_btn, _("Click for Settings"), tips) | ||||||
|  |  | ||||||
|  |         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, columnspan=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=4, columnspan=3, row=0, padx=10) | ||||||
|  |  | ||||||
|  |         # View Checkbox to 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")) | ||||||
|  |  | ||||||
|  |             Tooltip(self.updates_lb, _("Updates you have disabled"), tips) | ||||||
|  |  | ||||||
|  |         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")) | ||||||
|  |  | ||||||
|  |             Tooltip(self.updates_lb, _("Congratulations! Wire-Py is up to date"), tips) | ||||||
|  |  | ||||||
|  |         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, columnspan=3, row=0, padx=0) | ||||||
|  |  | ||||||
|  |             Tooltip(self.update_btn, _("Click to download new version"), tips) | ||||||
|  |  | ||||||
|  |             self.download = tk.Menu(self, relief="flat") | ||||||
|  |  | ||||||
|  |             self.update_btn.configure(menu=self.download, style="Toolbutton") | ||||||
|  |             self.download.add_command( | ||||||
|  |                 label=_("Download"), | ||||||
|  |                 command=lambda: GiteaUpdate.download(f"https://git.ilunix.de/punix/Wire-Py/archive/{res}.zip",  | ||||||
|  |                                                     res, img_w, img_i, img_w2, img_i2)) | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |         # Bottom Frame 4 | ||||||
|  |         self.lb_frame3 = ttk.Frame(self) | ||||||
|  |         self.lb_frame3.configure(relief="flat") | ||||||
|  |         self.lb_frame3.grid(column=0, row=5, columnspan=4, sticky="snew", padx=2, pady=2) | ||||||
|  |  | ||||||
|  |         # Bottom Frame 5 | ||||||
|  |         self.lb_frame4 = ttk.Frame(self) | ||||||
|  |         self.lb_frame4.configure(relief="flat") | ||||||
|  |         self.lb_frame4.grid(column=2, row=5, columnspan=3, sticky="e", padx=15) | ||||||
|  |  | ||||||
|  |         # 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") | ||||||
|  |  | ||||||
|  |         # Label to Show active Tunnel | ||||||
|  |         self.str_var = tk.StringVar(value=self.a) | ||||||
|  |         self.color_label() | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |         def enable_check_box(_): | ||||||
|  |             """ | ||||||
|  |             checkbox for enable autostart Tunnel | ||||||
|  |             """ | ||||||
|  |             Create.files_for_autostart() | ||||||
|  |             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") | ||||||
|  |  | ||||||
|  |         # Listbox with Scrollbar | ||||||
|  |         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) | ||||||
|  |  | ||||||
|  |         # Tunnel List | ||||||
|  |         self.tl = Tunnel.list() | ||||||
|  |         for tunnels in self.tl: | ||||||
|  |             self.l_box.insert("end", tunnels[:-5]) | ||||||
|  |             self.l_box.update() | ||||||
|  |  | ||||||
|  |         # Button Vpn | ||||||
|  |         if self.a != "": | ||||||
|  |             self.stop() | ||||||
|  |             wg_read = f"/tmp/tlecdcwg/{self.a}.conf" | ||||||
|  |             with open(wg_read, "r", encoding="utf-8") as file: | ||||||
|  |                 data = Tunnel.con_to_dict(file) | ||||||
|  |  | ||||||
|  |             # Address Label | ||||||
|  |             self.init_and_report(data) | ||||||
|  |             self.show_data() | ||||||
|  |         else: | ||||||
|  |             self.start() | ||||||
|  |  | ||||||
|  |             # Address Label | ||||||
|  |             self.add = tk.StringVar() | ||||||
|  |             self.DNS = tk.StringVar() | ||||||
|  |             self.enp = tk.StringVar() | ||||||
|  |             self.label_empty() | ||||||
|  |             self.show_data() | ||||||
|  |  | ||||||
|  |         # Button Import | ||||||
|  |         self.btn_i = ttk.Button(self.lb_frame_btn_lbox, image=self.imp_pic, command=self.import_sl, padding=0) | ||||||
|  |         self.btn_i.grid(column=0, row=1, padx=15, pady=8) | ||||||
|  |  | ||||||
|  |         Tooltip(self.btn_i, _("Click to import a Wireguard Tunnel"), tips) | ||||||
|  |  | ||||||
|  |         def delete(): | ||||||
|  |             """ | ||||||
|  |             delete Wireguard Tunnel | ||||||
|  |             """ | ||||||
|  |             try: | ||||||
|  |                 self.select_tunnel = self.l_box.curselection() | ||||||
|  |                 select_tl = self.l_box.get(self.select_tunnel[0]) | ||||||
|  |                 with open(f"/tmp/tlecdcwg/{select_tl}.conf", "r+", encoding="utf-8") 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(set_file, "r", encoding="utf-8") as set_f6: | ||||||
|  |                     lines6 = set_f6.readlines() | ||||||
|  |                     if ( | ||||||
|  |                             select_tl == lines6[7].strip() | ||||||
|  |                             and "off\n" not in lines6[7].strip() | ||||||
|  |                     ): | ||||||
|  |                         lines6[7] = "off\n" | ||||||
|  |                         with open(set_file, "w", encoding="utf-8") as set_f7: | ||||||
|  |                             set_f7.writelines(lines6) | ||||||
|  |                         self.selected_option.set(0) | ||||||
|  |                         self.autoconnect_var.set(_("no Autoconnect")) | ||||||
|  |                 is_encrypt = Path.home() / f".config/wire_py/{select_tl}.dat" | ||||||
|  |                 if is_encrypt.is_file(): | ||||||
|  |                     Path.unlink(f"{Path.home()}/.config/wire_py/{select_tl}.dat") | ||||||
|  |                 Path.unlink(f"/tmp/tlecdcwg/{select_tl}.conf") | ||||||
|  |                 with open(f"{Path.home()}/.config/wire_py/keys", "r", encoding="utf-8") as readfile: | ||||||
|  |                     with open(f"{Path.home()}/.config/wire_py/keys2", "w", encoding="utf-8") as writefile: | ||||||
|  |                         for line in readfile: | ||||||
|  |                             if pre_key not in line.strip("\n"): | ||||||
|  |                                 writefile.write(line) | ||||||
|  |                 file_one = Path(f"{Path.home()}/.config/wire_py/keys2") | ||||||
|  |                 file_two = file_one.with_name("keys") | ||||||
|  |                 file_one.replace(file_two) | ||||||
|  |                 self.wg_autostart.configure(state="disabled") | ||||||
|  |  | ||||||
|  |                 # for disabling checkbox when Listbox empty | ||||||
|  |                 if self.l_box.size() == 0: | ||||||
|  |                     self.wg_autostart.configure(state="disabled") | ||||||
|  |                     self.lb_rename.configure(state="disabled") | ||||||
|  |                     Tooltip(self.wg_autostart, _("You must have at least one\ntunnel in the list,to use the autostart") | ||||||
|  |                             , tips) | ||||||
|  |  | ||||||
|  |                     Tooltip(self.btn_exp, _("No Tunnels in List for Export"), tips) | ||||||
|  |                     Tooltip(self.btn_stst, _("No tunnels to start in the list"), tips) | ||||||
|  |                     Tooltip(self.lb_rename, _("To rename a tunnel, at least one must be in the list"), tips, ) | ||||||
|  |                     self.lb_rename.insert(0, _("Max. 12 characters!")) | ||||||
|  |  | ||||||
|  |                 if self.a != "" and self.a == select_tl: | ||||||
|  |                     self.str_var.set(value="") | ||||||
|  |                     self.start() | ||||||
|  |                     self.l_box.update() | ||||||
|  |  | ||||||
|  |                     # Address Label | ||||||
|  |                     self.add.set("") | ||||||
|  |                     self.DNS.set("") | ||||||
|  |                     self.enp.set("") | ||||||
|  |  | ||||||
|  |             except IndexError: | ||||||
|  |  | ||||||
|  |                 if self.l_box.size() != 0: | ||||||
|  |  | ||||||
|  |                     LxTools.msg_window(img_w, img_i2, sl, pstl) | ||||||
|  |  | ||||||
|  |                 else: | ||||||
|  |  | ||||||
|  |                     LxTools.msg_window(img_w, img_i2, sl, pfit) | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |         if self.l_box.size() == 0: | ||||||
|  |             Tooltip(self.btn_tr, _("No tunnels to delete in the list"), tips) | ||||||
|  |         else: | ||||||
|  |             Tooltip(self.btn_tr, _("Click to delete a Wireguard Tunnel\nSelect from the list!"), tips) | ||||||
|  |  | ||||||
|  |         # Button Export | ||||||
|  |         self.btn_exp = ttk.Button(self.lb_frame_btn_lbox, image=self.exp_pic, | ||||||
|  |                                   command=lambda: Tunnel.export(img_w, img_i, img_w2, img_i2, sl, pfit), padding=0) | ||||||
|  |         self.btn_exp.grid(column=0, row=3, padx=15, pady=8) | ||||||
|  |  | ||||||
|  |         if self.l_box.size() == 0: | ||||||
|  |             Tooltip(self.btn_exp, _("No Tunnels in List for Export"), tips) | ||||||
|  |         else: | ||||||
|  |             Tooltip(self.btn_exp, _("Click to export all\nWireguard Tunnel to Zipfile"), tips) | ||||||
|  |  | ||||||
|  |         # Label Entry | ||||||
|  |         self.lb_rename = ttk.Entry(self.lb_frame4, width=20) | ||||||
|  |         self.lb_rename.grid(column=2, row=0, padx=8, pady=10, sticky="ne") | ||||||
|  |         self.lb_rename.insert(0, _("Max. 12 characters!")) | ||||||
|  |         self.lb_rename.config(state="disable") | ||||||
|  |  | ||||||
|  |         if self.l_box.size() != 0: | ||||||
|  |             Tooltip(self.lb_rename, _("To rename a tunnel, you need to\nselect a tunnel from the list"), tips) | ||||||
|  |         else: | ||||||
|  |             Tooltip(self.lb_rename, _("To rename a tunnel, at least one must be in the list"), tips) | ||||||
|  |  | ||||||
|  |         def tl_rename(): | ||||||
|  |  | ||||||
|  |             special_characters = ["\\", "/", "{", "}", " "] | ||||||
|  |  | ||||||
|  |             if len(self.lb_rename.get()) > 12: | ||||||
|  |  | ||||||
|  |                 LxTools.msg_window(img_w, img_i2, rnp, _("The new name may contain only 12 characters")) | ||||||
|  |  | ||||||
|  |             elif len(self.lb_rename.get()) == 0: | ||||||
|  |  | ||||||
|  |                 LxTools.msg_window(img_w, img_i2, rnp, _("At least one character must be entered")) | ||||||
|  |  | ||||||
|  |             elif any(ch in special_characters for ch in self.lb_rename.get()): | ||||||
|  |  | ||||||
|  |                 msg_t = _("No valid sign. These must not be used.\nBlank, Slash, Backslash and { }\n") | ||||||
|  |                 LxTools.msg_window(img_w, img_i2, rnp, 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(f"/tmp/tlecdcwg/{select_tl}.conf") | ||||||
|  |                     destination = source.with_name(f"{self.lb_rename.get()}.conf") | ||||||
|  |                     source.replace(destination) | ||||||
|  |                     Path.unlink(f"{Path.home()}/.config/wire_py/{select_tl}.dat") | ||||||
|  |  | ||||||
|  |                     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.str_var.set(value=self.a) | ||||||
|  |                         with open(set_file, "r", encoding="utf-8") as set_f5: | ||||||
|  |                             lines5 = set_f5.readlines() | ||||||
|  |                         if select_tl == lines5[7].strip() and "off\n" not in lines5[7].strip(): | ||||||
|  |                             lines5[7] = new_a_connect | ||||||
|  |                             with open(set_file, "w", encoding="utf-8") as theme_set5: | ||||||
|  |                                 theme_set5.writelines(lines5) | ||||||
|  |                             self.autoconnect_var.set(value=new_a_connect) | ||||||
|  |  | ||||||
|  |                     Create.encrypt() | ||||||
|  |  | ||||||
|  |                 except IndexError: | ||||||
|  |  | ||||||
|  |                     LxTools.msg_window(img_w, img_i2, rnp, pstl) | ||||||
|  |  | ||||||
|  |         # Button Rename | ||||||
|  |         self.btn_rename = ttk.Button(self.lb_frame4, text=_("Rename"), state="disable", command=tl_rename, padding=4, | ||||||
|  |                                      style="RnButton.TButton") | ||||||
|  |         self.btn_rename.grid(column=3, row=0, padx=5, pady=10, 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.lb_frame3, textvariable=self.autoconnect_var, width=15) | ||||||
|  |         self.autoconnect.config(font=("Ubuntu", 11)) | ||||||
|  |         self.autoconnect.grid(column=1, row=0, sticky="e", pady=19) | ||||||
|  |         self.wg_autostart = ttk.Checkbutton(self.lb_frame3, text=_("Autoconnect on:"), variable=self.selected_option, | ||||||
|  |                                             command=self.box_set) | ||||||
|  |         self.wg_autostart.grid(column=0, row=0, pady=15, padx=15, sticky="nw") | ||||||
|  |  | ||||||
|  |         if self.l_box.size() >= 1 and len(self.l_box.curselection()) >= 1: | ||||||
|  |             Tooltip(self.wg_autostart, _("To use the autostart, enable this Checkbox"), tips) | ||||||
|  |  | ||||||
|  |         if self.l_box.size() == 0: | ||||||
|  |             Tooltip(self.wg_autostart, _("You must have at least one\ntunnel in the list,to use the autostart"), tips) | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |  | ||||||
|  |             Tooltip(self.wg_autostart, _("To use the autostart, a tunnel must be selected from the list"), tips) | ||||||
|  |  | ||||||
|  |         self.on_off() | ||||||
|  |  | ||||||
|  |     def import_sl(self): | ||||||
|  |         """ | ||||||
|  |         Import Methode 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 arising. | ||||||
|  |         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. | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         Create.dir_and_files() | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             filepath = filedialog.askopenfilename(initialdir=f"{Path.home()}", title=_("Select Wireguard config File"), | ||||||
|  |                                                   filetypes=[(_("WG config files"), "*.conf")]) | ||||||
|  |  | ||||||
|  |             with open(filepath, "r", encoding="utf-8") 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", encoding="utf-8") as file: | ||||||
|  |                     key = Tunnel.con_to_dict(file) | ||||||
|  |                     pre_key = key[3] | ||||||
|  |                     if len(pre_key) != 0: | ||||||
|  |                         with open(f"{Path.home()}/.config/wire_py/keys", "r", encoding="utf-8") as readfile: | ||||||
|  |                             p_key = readfile.readlines() | ||||||
|  |                             if pre_key in p_key or f"{pre_key}\n" in p_key: | ||||||
|  |  | ||||||
|  |                                 msg_t = _("Tunnel already available!\nPlease use another file for import") | ||||||
|  |                                 LxTools.msg_window(img_w2, img_i2, ie, msg_t) | ||||||
|  |  | ||||||
|  |                             else: | ||||||
|  |  | ||||||
|  |                                 with open(f"{Path.home()}/.config/wire_py/keys", "a", encoding="utf-8") as keyfile: | ||||||
|  |                                     keyfile.write(f"{pre_key}\r") | ||||||
|  |                                     if len(path_split1) > 17: | ||||||
|  |                                         p1 = shutil.copy(filepath, "/tmp/tlecdcwg/") | ||||||
|  |                                         path_split = path_split1[len(path_split1) - 17:] | ||||||
|  |                                         os.rename(p1, f"/tmp/tlecdcwg/{path_split}") | ||||||
|  |                                         new_conf = f"/tmp/tlecdcwg/{path_split}" | ||||||
|  |                                         if self.a != "": | ||||||
|  |                                             check_call(["nmcli", "connection", "down", Tunnel.active()]) | ||||||
|  |                                             self.label_empty() | ||||||
|  |  | ||||||
|  |                                         subprocess.check_output(["nmcli", "connection", "import", "type", | ||||||
|  |                                                                  "wireguard", "file", new_conf], text=True) | ||||||
|  |  | ||||||
|  |                                         Create.encrypt() | ||||||
|  |  | ||||||
|  |                                     else: | ||||||
|  |                                         shutil.copy(filepath, "/tmp/tlecdcwg/") | ||||||
|  |                                         if self.a != "": | ||||||
|  |                                             check_call(["nmcli", "connection", "down", Tunnel.active()]) | ||||||
|  |                                             self.label_empty() | ||||||
|  |  | ||||||
|  |                                         subprocess.check_output(["nmcli", "connection", "import", "type", | ||||||
|  |                                                                  "wireguard", "file", filepath], text=True) | ||||||
|  |  | ||||||
|  |                                         Create.encrypt() | ||||||
|  |  | ||||||
|  |                                     self.str_var.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) | ||||||
|  |  | ||||||
|  |                                     Tooltip(self.wg_autostart, _("To use the autostart, enable this Checkbox"), tips) | ||||||
|  |  | ||||||
|  |                                     # Tooltip(self.l_box, _("List of available tunnels")) | ||||||
|  |  | ||||||
|  |                                     Tooltip(self.btn_tr, _("Click to delete a Wireguard Tunnel\nSelect from the list!") | ||||||
|  |                                             , tips,) | ||||||
|  |  | ||||||
|  |                                     Tooltip(self.btn_exp, _("         Click to export all\nWireguard Tunnel to Zipfile") | ||||||
|  |                                             , tips) | ||||||
|  |  | ||||||
|  |                                     Tooltip(self.btn_rename, _("To rename a tunnel, you need to\nselect a tunnel from" | ||||||
|  |                                                                " the list"), tips) | ||||||
|  |  | ||||||
|  |                                     self.lb_rename.insert(0, "Max. 12 characters!") | ||||||
|  |                                     self.str_var = tk.StringVar() | ||||||
|  |                                     self.str_var.set(self.a) | ||||||
|  |                                     self.color_label() | ||||||
|  |                                     self.stop() | ||||||
|  |                                     wg_read = f"/tmp/tlecdcwg/{self.a}.conf" | ||||||
|  |                                     with open(wg_read, "r", encoding="utf-8") as file_for_key: | ||||||
|  |                                         data = Tunnel.con_to_dict(file_for_key) | ||||||
|  |  | ||||||
|  |                                     # Address Label | ||||||
|  |                                     self.init_and_report(data) | ||||||
|  |                                     self.show_data() | ||||||
|  |                                     check_call(["nmcli", "con", "mod", self.a, "connection.autoconnect", "no"]) | ||||||
|  |                                     Path.chmod(wg_read, 0o600) | ||||||
|  |  | ||||||
|  |             if ("PrivateKey = " in read) and ("Endpoint = " in read): | ||||||
|  |                 pass | ||||||
|  |             else: | ||||||
|  |  | ||||||
|  |                 msg_t = _("Oh... no valid Wireguard File!\nPlease select a valid Wireguard File") | ||||||
|  |                 LxTools.msg_window(img_w2, img_i2, ie, msg_t) | ||||||
|  |  | ||||||
|  |         except EOFError as e: | ||||||
|  |             print(e) | ||||||
|  |         except TypeError: | ||||||
|  |             print("File import: abort by user...") | ||||||
|  |         except FileNotFoundError: | ||||||
|  |             print("File import: abort by user...") | ||||||
|  |         except subprocess.CalledProcessError: | ||||||
|  |             print("Tunnel exist!") | ||||||
|  |  | ||||||
|  |     def box_set(self): | ||||||
|  |         """ | ||||||
|  |         This Method will display the autostart label which | ||||||
|  |         Tunnel is automatically started regardless of the active tunnel. | ||||||
|  |         The selected tunnel is written into a file to read it after the start of the system. | ||||||
|  |         """ | ||||||
|  |         try: | ||||||
|  |             select_tunnel = self.l_box.curselection() | ||||||
|  |             select_tl = self.l_box.get(select_tunnel[0]) | ||||||
|  |  | ||||||
|  |             if self.selected_option.get() == 0: | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as set_f3: | ||||||
|  |                     lines3 = set_f3.readlines() | ||||||
|  |                     lines3[7] = "off\n" | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as set_f3: | ||||||
|  |                     set_f3.writelines(lines3) | ||||||
|  |  | ||||||
|  |                 tl = Tunnel.list() | ||||||
|  |  | ||||||
|  |                 if len(tl) == 0: | ||||||
|  |                     self.wg_autostart.configure(state="disabled") | ||||||
|  |  | ||||||
|  |             if self.selected_option.get() >= 1: | ||||||
|  |                 with open(set_file, "r", encoding="utf-8") as set_f3: | ||||||
|  |                     lines3 = set_f3.readlines() | ||||||
|  |                     lines3[7] = select_tl | ||||||
|  |                 with open(set_file, "w", encoding="utf-8") as set_f3: | ||||||
|  |                     set_f3.writelines(lines3) | ||||||
|  |  | ||||||
|  |         except IndexError: | ||||||
|  |             self.selected_option.set(1) | ||||||
|  |  | ||||||
|  |         self.on_off() | ||||||
|  |  | ||||||
|  |     def on_off(self): | ||||||
|  |         """ | ||||||
|  |         Here it is checked whether the path to the file is there, if not, it is created. | ||||||
|  |         Set (on), the selected tunnel is displayed in the label. | ||||||
|  |         At (off) the label is first emptied then filled with No Autoconnect | ||||||
|  |         """ | ||||||
|  |         with open(set_file, "r", encoding="utf-8") as set_f4: | ||||||
|  |             lines4 = set_f4.readlines() | ||||||
|  |  | ||||||
|  |         if lines4[7] != "off\n": | ||||||
|  |             print(f"{lines4[7]} starts automatically when the system starts.") | ||||||
|  |             self.selected_option.set(1) | ||||||
|  |             self.autoconnect_var.set("") | ||||||
|  |             self.auto_con = lines4[7] | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |             self.selected_option.set(0) | ||||||
|  |             self.auto_con = _("no Autoconnect") | ||||||
|  |             print("Autostart disabled.") | ||||||
|  |         self.autoconnect_var.set("") | ||||||
|  |         self.autoconnect_var = tk.StringVar() | ||||||
|  |         self.autoconnect_var.set(self.auto_con) | ||||||
|  |  | ||||||
|  |         self.autoconnect = ttk.Label(self.lb_frame3, textvariable=self.autoconnect_var, foreground="#0071ff", width=15) | ||||||
|  |         self.autoconnect.config(font=("Ubuntu", 11)) | ||||||
|  |         self.autoconnect.grid(column=1, row=0, sticky="e", pady=19) | ||||||
|  |  | ||||||
|  |     def init_and_report(self, data=None): | ||||||
|  |         """ | ||||||
|  |         Displays the value address, DNS and peer in the labels | ||||||
|  |         or empty it again | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         # Address Label | ||||||
|  |         self.add = tk.StringVar() | ||||||
|  |         self.add.set(f"{_("Address:    ")}{data[0]}") | ||||||
|  |         self.DNS = tk.StringVar() | ||||||
|  |         self.DNS.set(f"       DNS:   {data[1]}") | ||||||
|  |         self.enp = tk.StringVar() | ||||||
|  |         self.enp.set(f"{_("Endpoint:    ")}{data[2]}") | ||||||
|  |  | ||||||
|  |     def label_empty(self): | ||||||
|  |         """ | ||||||
|  |         empty labels | ||||||
|  |         """ | ||||||
|  |         self.add.set("") | ||||||
|  |         self.DNS.set("") | ||||||
|  |         self.enp.set("") | ||||||
|  |  | ||||||
|  |     def show_data(self): | ||||||
|  |         """ | ||||||
|  |         shows data in the label | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         # Address Label | ||||||
|  |         self.address = ttk.Label(self.lb_frame, textvariable=self.add, foreground="#0071ff") | ||||||
|  |         self.address.grid(column=0, row=5, sticky="w", padx=10, pady=6) | ||||||
|  |         self.address.config(font=("Ubuntu", 9)) | ||||||
|  |  | ||||||
|  |         # DNS Label | ||||||
|  |         self.dns = ttk.Label(self.lb_frame, textvariable=self.DNS, foreground="#0071ff") | ||||||
|  |         self.dns.grid(column=0, row=7, sticky="w", padx=10, pady=6) | ||||||
|  |         self.dns.config(font=("Ubuntu", 9)) | ||||||
|  |  | ||||||
|  |         # Endpoint Label | ||||||
|  |         self.endpoint = ttk.Label(self.lb_frame2, textvariable=self.enp, foreground="#0071ff") | ||||||
|  |         self.endpoint.grid(column=0, row=8, sticky="w", padx=10, pady=20) | ||||||
|  |         self.endpoint.config(font=("Ubuntu", 9)) | ||||||
|  |  | ||||||
|  |     def stop(self): | ||||||
|  |         """ | ||||||
|  |         Stop Button | ||||||
|  |         """ | ||||||
|  |         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) | ||||||
|  |  | ||||||
|  |         Tooltip(self.btn_stst, _("Click to stop selected Wireguard Tunnel"), tips) | ||||||
|  |  | ||||||
|  |     def start(self): | ||||||
|  |         """ | ||||||
|  |         Start Button | ||||||
|  |         """ | ||||||
|  |         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) | ||||||
|  |  | ||||||
|  |         tl = Tunnel.list() | ||||||
|  |         if len(tl) == 0: | ||||||
|  |             Tooltip(self.btn_stst, _("No tunnels to start in the list"), tips) | ||||||
|  |         else: | ||||||
|  |             Tooltip(self.btn_stst, _("Click to start selected Wireguard Tunnel"), tips) | ||||||
|  |  | ||||||
|  |     def color_label(self): | ||||||
|  |         """ | ||||||
|  |         View activ Tunnel in the color green or yellow | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         with open(set_file, "r", encoding="utf-8") as read_file: | ||||||
|  |             lines = read_file.readlines() | ||||||
|  |             if "light\n" in lines: | ||||||
|  |                 self.lb_tunnel = ttk.Label(self, textvariable=self.str_var, foreground="green") | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |                 self.lb_tunnel = ttk.Label(self, textvariable=self.str_var, foreground="yellow") | ||||||
|  |  | ||||||
|  |         self.lb_tunnel.config(font=("Ubuntu", 11, "bold")) | ||||||
|  |         self.lb_tunnel.grid(column=2, padx=10, row=1) | ||||||
|  |  | ||||||
|  |     def wg_switch(self): | ||||||
|  |         """ | ||||||
|  |         switch Tunnel method change from labels and buttons | ||||||
|  |         """ | ||||||
|  |         self.a = Tunnel.active() | ||||||
|  |         try: | ||||||
|  |             if self.a == "": | ||||||
|  |  | ||||||
|  |                 self.start() | ||||||
|  |                 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 = f"/tmp/tlecdcwg/{select_tl}.conf" | ||||||
|  |                 with open(wg_read, "r", encoding="utf-8") as file: | ||||||
|  |                     data = Tunnel.con_to_dict(file) | ||||||
|  |  | ||||||
|  |                 # Address Label | ||||||
|  |                 self.init_and_report(data) | ||||||
|  |                 self.show_data() | ||||||
|  |  | ||||||
|  |                 # Button Start/Stop | ||||||
|  |                 self.stop() | ||||||
|  |                 self.a = Tunnel.active() | ||||||
|  |                 self.str_var = tk.StringVar() | ||||||
|  |                 self.str_var.set(self.a) | ||||||
|  |                 self.color_label() | ||||||
|  |  | ||||||
|  |             elif self.a != "": | ||||||
|  |  | ||||||
|  |                 # Button Start/Stop | ||||||
|  |                 self.stop() | ||||||
|  |                 check_call(["nmcli", "connection", "down", self.a]) | ||||||
|  |  | ||||||
|  |                 # Button Start/Stop | ||||||
|  |                 self.start() | ||||||
|  |                 self.a = Tunnel.active() | ||||||
|  |                 self.str_var.set("") | ||||||
|  |                 self.color_label() | ||||||
|  |  | ||||||
|  |                 # Address Label | ||||||
|  |                 self.add.set("") | ||||||
|  |                 self.DNS.set("") | ||||||
|  |                 self.enp.set("") | ||||||
|  |                 self.show_data() | ||||||
|  |  | ||||||
|  |         except IndexError: | ||||||
|  |  | ||||||
|  |             if self.l_box.size() != 0: | ||||||
|  |  | ||||||
|  |                 LxTools.msg_window(img_w, img_i2, sl, pstl) | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |  | ||||||
|  |                 LxTools.msg_window(img_w, img_i2, sl, pfit) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == "__main__": | ||||||
|  |     window = Wirepy() | ||||||
|  |     """ | ||||||
|  |     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() | ||||||
|  |  | ||||||
|  | LxTools.clean_files(folder_path, user_file) | ||||||
|  | sys.exit(0) | ||||||
|   | |||||||