-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathmain.py
453 lines (412 loc) · 15.9 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
# -*- coding: utf-8 -*-
"""本脚本用于打包后的程序瘦身"""
import json
import os
import sys
import threading
from _tkinter import TclError
from pathlib import Path
from tkinter import *
from tkinter import filedialog, messagebox
from tkinter.ttk import *
from typing import List
# 记录白名单文件路径
WHITE_FILE = Path(__file__).parent.joinpath("white_files.json")
# 默认的白名单文件内容,用于第一次程序运行生成json文件
WHITE_DICT = {
"python": [
"api-ms-win-core",
"base_library.zip",
".tcl",
"tclIndex",
"MSVCP140.dll",
"cacert.pem",
"cp936.enc",
"__init__",
"python.exe",
"pythonw.exe",
"VCRUNTIME140_1.dll",
],
"matplotlib": ["matplotlibrc", ".load-order", "matplotlib.svg"],
"request": ["msgcat-1.6.1.tm"],
"plotly": ["plotly.json", "plotly.min.js", "package_data\\templates",'favicon.ico'],
"kaleido": ["kaleido\\executable"],
"pyecharts": ["pyecharts"],
"pyqtwebengine": [
"QtWebEngineProcess.exe",
"icudtl.dat",
"qtwebengine_devtools_resources.pak",
"qtwebengine_resources",
"qt.conf",
],
"streamlit": ["streamlit\\static"],
"trame_vtk": ["static_viewer.html"],
"python-docx": ["docx\\templates"],
"python-pptx": ["pptx\\templates"],
"setuptools": ["setuptools\\_vendor"]
}
WHITE_FILE_TYPE = [".pyi", ".py", ".pyc", ".pth", "._pth"]
class WinGUI(Tk):
"""窗口类"""
def __init__(self):
super().__init__()
self._win()
self.tk_button_file = self._tk_button_file(self)
self.tk_button_start = self._tk_button_start(self)
self.tk_input_file = self._tk_input_file(self)
self.tk_button_info = self._tk_button_info(self)
self.tk_check_button_safe = self._tk_check_button_mode(self)
self.tk_label_progressbar = self._tk_label_progressbar(self)
self.tk_progressbar_progress = self._tk_progressbar_progress(self)
def _win(self):
"""设置窗口属性"""
self.title("打包瘦身小程序")
width = 500
height = 150
screenwidth = self.winfo_screenwidth()
screenheight = self.winfo_screenheight()
geometry = f"{width}x{height}+{int((screenwidth - width) / 2)}+{int((screenheight - height) / 2)}"
self.geometry(geometry)
self.resizable(width=False, height=False)
@staticmethod
def _tk_button_file(parent) -> Button:
"""
创建文件选择button
:param parent:父组件对象
:return: button组件对象
"""
btn = Button(
parent,
text="目标文件夹",
takefocus=False,
)
btn.place(x=5, y=10, width=100, height=30)
return btn
@staticmethod
def _tk_button_start(parent) -> Button:
"""
创建开始button
:param parent: 父组件对象
:return: button组件对象
"""
btn = Button(
parent,
text="开始处理",
takefocus=False,
)
btn.place(x=5, y=100, width=100, height=30)
return btn
@staticmethod
def _tk_input_file(parent) -> Entry:
"""
创建文件夹路径输入框
:param parent: 父组件对象
:return: entry组件对象
"""
ipt = Entry(
parent,
)
ipt.place(x=110, y=10, width=370, height=30)
return ipt
@staticmethod
def _tk_button_info(parent) -> Button:
"""
创建程序说明button
:param parent:父组件对象
:return: button对象
"""
btn = Button(
parent,
text="使用说明",
takefocus=False,
)
btn.place(x=380, y=100, width=100, height=30)
return btn
@staticmethod
def _tk_check_button_mode(parent) -> Checkbutton:
"""
创建模式选择
:param parent: 父组件对象
:return: checkbutton对象
"""
cb = Checkbutton(
parent,
text="当前:安全模式",
)
cb.place(x=180, y=100, width=120, height=30)
return cb
@staticmethod
def _tk_label_progressbar(parent) -> Label:
"""
创建进度条标签
:param parent: 父组件对象
:return: label对象
"""
label = Label(parent, text="处理进度", anchor="center")
label.place(x=5, y=55, width=100, height=30)
return label
@staticmethod
def _tk_progressbar_progress(parent) -> Progressbar:
"""
创建进度条对象
:param parent:父组件对象
:return: 进度条对象
"""
progressbar = Progressbar(parent, orient=HORIZONTAL)
progressbar.place(x=110, y=55, width=370, height=30)
return progressbar
class MasterGui(WinGUI):
"""程序主窗口"""
def __init__(self):
super().__init__()
icon = Path(__file__).parent.joinpath("file.ico")
try:
self.wm_iconbitmap(icon)
except FileNotFoundError:
pass
except TclError:
pass
self.mode_var = BooleanVar()
self.progress_var = DoubleVar()
self.tk_button_file.config(command=self.select_file)
self.tk_button_start.config(command=self.start_work)
self.tk_button_info.config(command=self.sys_info)
self.tk_check_button_safe.config(variable=self.mode_var)
self.mode_var.set(True)
self.mode_var.trace_add("write", self.mode_change)
self.tk_progressbar_progress.config(
variable=self.progress_var,
length=100,
)
self.file_name = ""
self.white_dict = None
self.white_list = []
self.ini_window()
self.protocol("WM_DELETE_WINDOW", self.close_window)
self.mainloop()
def ini_window(self) -> None:
"""程序初始化白名单列表,通过读取白名单json文件或读取默认白名单字典"""
if Path(WHITE_FILE).exists():
print("读取白名单配置文件", f"{WHITE_FILE}")
with open(WHITE_FILE, "r", encoding="utf-8") as file:
self.white_dict = json.load(file)
try:
assert isinstance(self.white_dict, dict)
except AssertionError:
print("读取的白名单配置文件不合法,请检查!")
input("按回车键继续...")
sys.exit()
else:
self.white_dict = WHITE_DICT
for value in self.white_dict.values():
self.white_list.extend(value)
@staticmethod
def close_window():
"""设置窗口关闭事件,保存默认白名单文件列表"""
if not Path(WHITE_FILE).exists():
json_data = json.dumps(WHITE_DICT, indent=4)
with open(WHITE_FILE, "w", encoding="utf-8") as json_file:
json_file.write(json_data)
print(f"已写入默认白名单配置文件,{WHITE_FILE},下次运行程序自动读取")
messagebox.showinfo(
title="保存", message=f"已写入默认白名单配置文件{WHITE_FILE}"
)
sys.exit()
def update_progress_safe(self, value: float) -> None:
"""
安全地更新进度条的值
:param value: 进度
:return: None
"""
self.after(0, lambda: self.update_progress(value))
def update_progress(self, value: float) -> None:
"""
更新进度条的值
:param value: 进度
:return: None
"""
self.progress_var.set(value)
self.update_idletasks()
def select_file(self) -> None:
"""设置需要处理的目标文件夹"""
selected_path = filedialog.askdirectory()
if selected_path == "":
return
selected_path = Path(selected_path)
self.tk_input_file.delete(0, "end")
self.tk_input_file.insert(0, str(selected_path))
def work_thread_func(self, white_list: List[str], file_path: Path) -> None:
"""
在额外线程进行文件夹操作
:param white_list: 包含白名单文件的列表
:param file_path: 需要操作的目标文件夹路径
:return: None
"""
file_dir_new = Path(str(file_path) + "_new")
only_binary_file = self.mode_var.get()
file_list = FileRemove.get_file_paths(file_path, only_binary_file)
file_move_lst = []
file_trans_lst = Path(file_path).parent.joinpath(
f"{file_path.name}_文件移动清单.txt"
)
file_count = len(file_list)
for idx, filename in enumerate(file_list):
if FileRemove.check_file(white_list, filename) and (
not FileRemove.is_file_in_use(filename)
): # 检查白名单和占用
relative_path = filename.relative_to(file_path)
filename_new = file_dir_new.joinpath(relative_path)
FileRemove.move_file_to_folder(filename, filename_new)
file_move_lst.append(str(filename))
print(f"移动 {filename}")
else:
print(f"跳过 {filename}")
self.update_progress_safe((idx + 1) / file_count * 100)
if file_trans_lst.exists():
os.remove(str(file_trans_lst))
try:
with open(file_trans_lst, "w", encoding="utf-8") as file:
for temp_file in file_move_lst:
file.write(f"{temp_file}\n")
except PermissionError:
messagebox.showerror(
title="错误", message="权限不足,考虑用管理员权限重新运行程序"
)
return
messagebox.showinfo("提示", "文件移动结束!")
self.tk_button_start.config(state=ACTIVE)
def start_work(self):
"""开始处理"""
file_path = self.tk_input_file.get()
if file_path == "":
return
file_path = Path(file_path)
if not file_path.exists():
return
# 为了避免误操作C盘重要文件,本程序不允许在C盘进行操作
if "C" in file_path.drive:
result = messagebox.askokcancel(
title="警告",
message=f"检测到当前需要处理的路径为C盘路径,\n程序可能因为权限不足无法正常工作。请确认是否执行\n待处理文件夹:{file_path}",
)
if not result:
return
self.tk_progressbar_progress.config(value=0)
file_move_thread = threading.Thread(
target=self.work_thread_func,
args=(
self.white_list,
file_path,
),
daemon=True,
)
self.tk_button_start.config(state=DISABLED)
file_move_thread.start()
def mode_change(self, *args) -> None:
"""
模式切换点击事件
:param args: 无
:return: None
"""
value = self.mode_var.get()
if value:
self.tk_check_button_safe.config(text="当前:安全模式")
else:
self.tk_check_button_safe.config(text="当前:极限模式")
@staticmethod
def sys_info() -> None:
"""程序说明"""
message = (
"运行原理:\n"
"本程序仅在win7/win11测试过。"
'程序通过指定文件目录,来移动该目录下未被程序占用的文件,移动的文件会在指定文件目录外的"目录_new"里面,同时生成文件移动清单.txt。\n\n'
"使用方法:\n1,运行自己打包好的程序,尽量将功能全开,实现程序完全加载;\n2,使用本程序,指定程序目录,程序开始移动未占用的文件;\n"
"3,移动结束后,请再运行一遍自己的程序,如果提示文件缺少,可以从《文件移动清单.txt》去找,还原即可。\n\n"
"瘦身效果:\n根据我的历史使用记录,本程序针对pyinstaller打包的程序能缩小体积50%左右,针对nuitka打包的程序能缩小体积30%左右。\n"
"注意事项:程序目前还不够完善,对于有些文件,是程序必须的,但是又在程序运行时不占用的,就需要将文件名加入白名单列表中。\n\n"
"安全模式:程序只针对pyd,dll文件进行操作,瘦身成功率较高;\n"
"极限模式:程序针对所有类型文件进行操作,瘦身效果较好,默认使用安全模式。\n\n"
"程序第一次运行,会自动生成white_files.json文件,该文件记录了程序运行时不移动的文件,请根据使用情况酌情添加。"
)
messagebox.showinfo(title="使用说明", message=message)
class FileRemove:
"""文件移动函数类"""
@staticmethod
def get_file_paths(directory: Path, only_binary: bool = True) -> List[Path]:
"""
获取指定路径所有文件名,only_binary设置是否只检测pyd,dll文件
:param directory: 目标路径
:param only_binary: 是否只检测二进制文件,即dll,pyd文件
:return:
"""
file_list = []
if only_binary:
files_paths = [
path
for pattern in ("*.pyd", "*.dll")
for path in directory.glob("**/" + pattern)
]
else:
files_paths = directory.glob("**/*.*")
for path in files_paths:
if path.is_file():
file_list.append(path)
return file_list
@staticmethod
def is_file_in_use(file_name: Path) -> bool:
"""
判断该文件是否被占用,被占用即返回True
:param file_name: 文件路径
:return: 是否是被占用状态
"""
file_name = str(file_name)
try:
# 尝试以读写模式打开文件
with open(file_name, "r+", encoding="utf-8") as _:
# 如果文件成功打开,说明没有被占用
return False
except OSError:
# 如果抛出OSError异常,说明文件被占用
return True
except Exception as e:
# 处理其他可能的异常
print(f"An error occurred: {e}")
return True
@staticmethod
def move_file_to_folder(file_old: Path, file_new: Path) -> None:
"""
使用Path.rename方法移动文件
:param file_old: 文件原路径
:param file_new: 文件新路径
:return: None
"""
if not file_new.parent.exists():
try:
os.makedirs(file_new.parent)
except (PermissionError, FileNotFoundError) as e:
print(e)
try:
file_old.rename(file_new)
except Exception as e:
print(e)
@staticmethod
def check_file(white_list: List[str], file: Path) -> bool:
"""
通过检测字符串包含的方法,判断该文件是否在白名单中,或在白名单文件类型中,如果在即返回False
:param white_list: 白名单文件列表
:param file: 文件路径
:return:
"""
file = str(file)
for data in white_list:
if data in file:
return False
for file_type in WHITE_FILE_TYPE:
if Path(file).suffix == file_type:
return False
return True
def start():
"""本模块外部接口,便于打包"""
win = MasterGui()
if __name__ == "__main__":
win = MasterGui()