-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathguiunstable.py
414 lines (377 loc) · 15.5 KB
/
guiunstable.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
import sys
import subprocess
import os
import re
import json
import atexit
import tempfile
import urllib.request # Needed for Downloading TWRP
import requests # Needed for Downloading TWRP
from bs4 import BeautifulSoup # Needed for Downloading TWRP
from PySide6.QtWidgets import (
QApplication,
QMainWindow,
QLabel,
QLineEdit,
QPushButton,
QComboBox,
QWidget,
QVBoxLayout,
QHBoxLayout,
QFileDialog,
QMessageBox,
QPlainTextEdit,
QProgressBar,
)
from PySide6.QtGui import QAction
from PySide6.QtCore import QProcess
if os.name == "nt":
print("Running on Windows!")
import ctypes
cwd = os.path.dirname(os.path.abspath(__file__))
os.chdir(cwd)
class Form(QMainWindow):
def __init__(self, parent=None):
super(Form, self).__init__(parent)
self.setWindowTitle("SamsungFlashGUI")
atexit.register(self.on_exit)
self.w = None
self.i = None
global local_filename
local_filename = None
# Create widgets
self.label = QLabel(self)
self.label.setText("Please select your IMG to Flash & Partiton.")
self.filenameentry = QLineEdit("twrp.img")
self.chooseimage = QPushButton("Choose Image")
self.flash = QPushButton("Flash")
self.buttoncheck = QComboBox()
self.buttoncheck.addItems(["BOOT", "RECOVERY", "DATA", "SYSTEM"])
layout = QVBoxLayout()
hbox = QHBoxLayout()
buttonbox = QHBoxLayout()
# Set up Menu Buttons
if os.name == "nt":
driverinstallb = QAction("Install Drivers (Windows)", self)
driverinstallb.setStatusTip("For Heimdall (Zadig)")
driverinstallb.triggered.connect(self.driverinstall)
driverinstallb.setCheckable(False)
aboutbutton = QAction("About", self)
aboutbutton.setStatusTip("About this project")
aboutbutton.triggered.connect(self.aboutdialog)
aboutbutton.setCheckable(False)
TWRPFlash = QAction("TWRP Flash", self)
TWRPFlash.setStatusTip("Download / Flash TWRP")
TWRPFlash.triggered.connect(self.TWRP_window)
TWRPFlash.setCheckable(False)
quitbutton = QAction("Quit", self)
quitbutton.setStatusTip("Exit this program")
quitbutton.triggered.connect(lambda: quit())
quitbutton.setCheckable(False)
menu = self.menuBar()
file_menu = menu.addMenu("File")
file_menu.addAction(aboutbutton)
file_menu.addAction(quitbutton)
file_menu.addAction(TWRPFlash)
if os.name == "nt":
file_menu.addAction(driverinstallb)
widget = QWidget()
widget.setLayout(layout)
self.setCentralWidget(widget)
layout.addWidget(self.label)
hbox.addWidget(self.filenameentry)
hbox.addWidget(self.chooseimage)
hbox.addWidget(self.buttoncheck)
layout.addLayout(buttonbox)
layout.addLayout(hbox)
layout.addWidget(self.flash)
self.chooseimage.clicked.connect(self.Image)
self.flash.clicked.connect(self.flash_window)
def on_exit(self):
if local_filename is not None:
print("Deleting Temp File...")
os.remove(local_filename)
def Image(self):
global filename
file_dialog = QFileDialog()
file_dialog.setFileMode(QFileDialog.ExistingFile)
file_dialog.setNameFilter("Image Files (*.img)")
file_dialog.setViewMode(QFileDialog.List)
if file_dialog.exec():
filenames = file_dialog.selectedFiles()
if filenames:
filename = filenames[0]
print(f"File {filename} chosen")
self.filenameentry.setText(filename)
def flash_window(self, checked):
if self.w is None:
partition = self.buttoncheck.currentText()
image = self.filenameentry.text()
self.w = FlashWindow()
self.w.start_process(partition, image)
self.w.show()
def TWRP_window(self, checked):
if self.i is None:
self.i = TWRPWindow()
self.i.show()
else:
self.i.close() # Close window.
self.i = None # Discard reference.
def aboutdialog(self):
logo = os.path.abspath(os.path.join(cwd, "python-logo-only.svg"))
text = f"""
<p><center>SamsungFlashGUI<center><p>
<p><center><img src="{logo}" alt=""><center></p>
Made By <a href="https://github.com/justaCasualCoder/SamsungFlashGUI"><i>justaCasualCoder<i></a></p>
<p>This project would not be possible without:</p>
<p><a href="https://github.com/Benjamin-Dobell/Heimdall">Heimdall</a> (Used to Flash Images)</p>
<p><a href="https://doc.qt.io/qtforpython-6/quickstart.html">Pyside6</a> (Used for the GUI)</p>
<p><a href="https://www.python.org/">Python</a> (What it is coded in)</p>
"""
dlg = QMessageBox(self)
dlg.setWindowTitle("About")
dlg.setText(text)
dlg.exec()
def driverinstall(self):
dlg = QMessageBox(self)
dlg.setIcon(QMessageBox.Question)
yes_button = dlg.addButton(QMessageBox.Yes)
no_button = dlg.addButton(QMessageBox.No)
dlg.setWindowTitle("Driver Install Directions")
dlg.setText(
"""
<p>Driver Installation Instructions:</p>
<ol>
<li><p>Put your device into download mode and plug it in.</p>
</li>
<li><p>Run <code>zadig.exe</code> included in the Drivers subdirectory.</p>
</li>
<li><p>From the menu choose Options -> List All Devices.</p>
</li>
<li><p>From the USB Device list pick "Samsung USB Composite Device".</p>
</li>
<li><p>Press "Install Driver", click "Yes" to the prompt, and if you receive</p>
<pre><code><span class="hljs-keyword">a</span> message about being unable <span class="hljs-built_in">to</span> verify <span class="hljs-keyword">the</span> publisher <span class="hljs-keyword">of</span> <span class="hljs-keyword">the</span> driver.
Click <span class="hljs-string">"Install this driver software anyway"</span>.
</code></pre></li>
<li><p>Done</p>
See <a href="https://github.com/justaCasualCoder/SamsungFlashGUI#windows">This</a> For the guide.
</li>
</ol>
<p>Would you like to continue?</p>
"""
)
result = dlg.exec()
if result == QMessageBox.Yes:
file = os.path.abspath(os.path.join(cwd, "heimdall\Drivers\zadig.exe"))
command = ["powershell.exe", "-command", file]
subprocess.Popen(command)
class FlashWindow(QWidget):
"""
This "window" is a QWidget. If it has no parent, it
will appear as a free-floating window as we want.
"""
def simple_percent_parser(output):
"""
Matches lines using the progress_re regex,
returning a single integer for the % progress.
"""
progress_re = re.compile("(\d+)%")
m = progress_re.search(output)
if m:
pc_complete = m.group(1)
return int(pc_complete)
def message(self, s):
self.text.appendPlainText(s)
def start_process(self, partition, image):
if self.p is None: # No process running.
self.message("Running Heimdall!")
self.image = image # Needed for function handle_stderr
self.p = (
QProcess()
) # Keep a reference to the QProcess (e.g. on self) while it's running.
self.p.readyReadStandardOutput.connect(self.handle_stdout)
self.p.readyReadStandardError.connect(self.handle_stderr)
self.p.stateChanged.connect(self.handle_state)
self.p.finished.connect(self.process_finished) # Clean up once complete.
heimdallbin = os.path.abspath(os.path.join(cwd, "heimdall/heimdall_linux"))
if os.name == "nt":
print("Running on Windows!")
print("Running heimdall.exe")
heimdallbin = "./heimdall/heimdall.exe"
else:
if os.path.exists(heimdallbin):
print("Using Bundled Heimdall")
heimdallbin = "./heimdall/heimdall_linux"
else:
heimdallbin = "/bin/heimdall"
command = [f"{heimdallbin}", "flash", f"--{partition}", image]
self.p.start(command[0], command[1:])
def handle_stdout(self):
data = self.p.readAllStandardOutput()
stdout = bytes(data).decode("utf8")
# Extract progress if it is in the data.
progress = self.simple_percent_parser(stdout)
if progress:
self.progress.setValue(progress)
self.message(stdout)
def simple_percent_parser(self, output):
"""
Matches lines using the progress_re regex,
returning a single integer for the % progress.
"""
progress_re = re.compile("(\d+)%")
m = progress_re.search(output)
if m:
pc_complete = m.group(1)
return int(pc_complete)
def handle_stderr(self):
data = self.p.readAllStandardError()
stderr = bytes(data).decode("utf8")
self.message(stderr)
if "ERROR: Failed to detect compatible download-mode device." in stderr:
print("No Device Attached!")
info = "Failed to detect compatible download-mode device."
self.setWindowTitle("SamsungFlashGUI: Error!")
button = QMessageBox.critical(self, "Error", f"{info}")
self.close()
if "Failed to open file" in stderr:
print("Can't find file!")
info = f"Failed to find file: {self.image}"
self.setWindowTitle("SamsungFlashGUI: Error!")
button = QMessageBox.critical(self, "Error", f"{info}")
self.close()
def handle_state(self, state):
states = {
QProcess.NotRunning: "Not running",
QProcess.Starting: "Starting",
QProcess.Running: "Running",
}
state_name = states[state]
self.message(f"State changed: {state_name}")
def process_finished(self):
self.message("Done Flashing!")
self.p = None
def __init__(self):
super().__init__()
self.p = None
layout = QVBoxLayout()
self.setWindowTitle("SamsungFlashGUI: Flashing...")
self.text = QPlainTextEdit()
self.text.setReadOnly(True)
self.label = QLabel("Running Heimdall...")
self.progress = QProgressBar()
self.progress.setRange(0, 100)
layout.addWidget(self.label)
layout.addWidget(self.progress)
layout.addWidget(self.text)
self.setLayout(layout)
class TWRPWindow(QWidget):
def fetch_devices(self):
jsonfile = os.path.abspath(os.path.join(cwd, "devices.json"))
if os.path.exists(jsonfile):
with open(jsonfile, "r") as file:
self.devices = json.load(file)
self.entry.addItems(list(self.devices.keys()))
return
devices = {}
request = requests.get("https://twrp.me/Devices/Samsung")
parsed_request = BeautifulSoup(request.content, "html.parser")
device_elements = parsed_request.select("ul#post-list > p > strong > a")
# Iterate through the 'a' tags and extract the device name and corresponding code
for element in device_elements:
# Extract the device name (text within the 'a' tag)
device_name = element.text.strip()
# Split the href and get the last part, then remove the file extension
code = element["href"].split("/")[-1].split(".")[0]
# Extract all content within parentheses
parentheses_content = [
content.strip() for content in device_name.split("(")[1:]
]
# Use the second set of parentheses if it exists; otherwise, use the first set of parentheses
if len(parentheses_content) >= 2:
code = parentheses_content[1].split(")")[0]
else:
code = parentheses_content[0].split(")")[0]
# Remove spaces and convert to lowercase
code = code.replace(" ", "").lower()
# Add the device name and customized code to the dictionary
devices[device_name] = code
self.devices = devices
self.entry.addItems(list(self.devices.keys()))
# Dump JSON to file
with open("devices.json", "w") as file:
json.dump(devices, file)
return devices
def download_flash(self):
# Huge thanks to the orginal creator - JBBgameich
# https://github.com/JBBgameich/halium-install
dlpagerequest = requests.get("https://dl.twrp.me/" + self.current_code)
dlpage = BeautifulSoup(dlpagerequest.content, "html.parser")
try:
dllinks = dlpage.table.find_all("a")
except:
print("E: Couldn't find a TWRP image for " + self.current_code)
sys.exit(1)
url = "https://dl.twrp.me" + dllinks[1]["href"].replace(".html", "")
print("I: Downloading " + url)
local_filename = tempfile.NamedTemporaryFile(suffix=".img", delete=False).name
referer_url = url + ".html"
headers = {"Referer": referer_url}
request = urllib.request.Request(url, headers=headers)
with urllib.request.urlopen(request) as response, open(
local_filename, "wb"
) as f:
total_size = int(response.info().get("Content-Length", 0))
block_size = 1024
current_size = 0
while True:
buffer = response.read(block_size)
if not buffer:
break
f.write(buffer)
current_size += len(buffer)
percent = (current_size / total_size) * 100
print(f"\rDownloaded: {percent:.2f}%", end="", flush=True)
self.progressb.setValue(percent)
self.close()
self.w = FlashWindow()
self.w.start_process("RECOVERY", local_filename)
self.w.show()
def __init__(self):
super().__init__()
self.p = None
self.setWindowTitle("SamsungFlashGUI: Flash TWRP")
layoutv = QVBoxLayout()
self.entry = QComboBox()
# self.entry.addItems(list(self.devices.keys()))
self.entry.currentIndexChanged.connect(self.device_selected)
info1 = QLabel("Device")
button = QPushButton("Download / Flash")
self.progressb = QProgressBar()
button.clicked.connect(self.download_flash)
layoutv.addWidget(info1)
layoutv.addWidget(self.entry)
layoutv.addWidget(button)
layoutv.addWidget(self.progressb)
self.setLayout(layoutv)
self.fetch_devices()
def device_selected(self, index):
# Retrieve the selected device from the combobox
selected_device = self.sender().itemText(index)
# Find the corresponding custom code for the selected device from the stored devices dictionary
selected_device_code = self.devices.get(selected_device)
if selected_device_code is not None:
print(f"Selected device: {selected_device}, Code: {selected_device_code}")
else:
print(f"Code not found for {selected_device}")
self.current_code = selected_device_code
if __name__ == "__main__":
# Create the Qt Application
app = QApplication(sys.argv)
app.setStyle('fusion')
# Create and show the form
form = Form()
form.show()
# Run the main Qt loop
sys.exit(app.exec())