mirror of
https://github.com/corpnewt/gibMacOS.git
synced 2025-01-22 21:01:22 +01:00
Auto-install basic Clover
Add OS type/version check Utilize BOOTICEx64.exe for mbr/pbr adjustments (might change in the future) Add flags to only install Clover, set the first partition as EFI/Basic Data, and force GPT
This commit is contained in:
parent
20488d435b
commit
9f0e914912
374
MakeInstall.py
374
MakeInstall.py
@ -1,16 +1,15 @@
|
|||||||
from Scripts import *
|
from Scripts import *
|
||||||
import os, sys, tempfile, shutil, zipfile
|
import os, sys, tempfile, shutil, zipfile, platform, json, time
|
||||||
|
|
||||||
class WinUSB:
|
class WinUSB:
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
self.u = utils.Utils("MakeInstall")
|
||||||
|
self.min_plat = 9600
|
||||||
# Make sure we're on windows
|
# Make sure we're on windows
|
||||||
if not os.name=="nt":
|
self.verify_os()
|
||||||
print("This script is only for Windows!")
|
|
||||||
exit(1)
|
|
||||||
# Setup initial vars
|
# Setup initial vars
|
||||||
self.d = diskwin.Disk()
|
self.d = diskwin.Disk()
|
||||||
self.u = utils.Utils("BinaryDestructionPls")
|
|
||||||
self.dl = downloader.Downloader()
|
self.dl = downloader.Downloader()
|
||||||
self.r = run.Run()
|
self.r = run.Run()
|
||||||
self.scripts = "Scripts"
|
self.scripts = "Scripts"
|
||||||
@ -19,12 +18,54 @@ class WinUSB:
|
|||||||
self.dd_name = os.path.basename(self.dd_url)
|
self.dd_name = os.path.basename(self.dd_url)
|
||||||
self.z_url = "https://www.7-zip.org/a/7z1805-x64.msi"
|
self.z_url = "https://www.7-zip.org/a/7z1805-x64.msi"
|
||||||
self.z_name = "7z.exe"
|
self.z_name = "7z.exe"
|
||||||
|
self.bi_url = "https://sites.google.com/site/gbrtools/home/software/bootice-portable/files/BOOTICE_v1.3.3.2.zip"
|
||||||
|
self.bi_name = "BOOTICEx64.exe"
|
||||||
|
self.clover_url = "https://api.github.com/repos/dids/clover-builder/releases/latest"
|
||||||
# From Tim Sutton's brigadier: https://github.com/timsutton/brigadier/blob/master/brigadier
|
# From Tim Sutton's brigadier: https://github.com/timsutton/brigadier/blob/master/brigadier
|
||||||
self.z_path = os.path.join(os.environ['SYSTEMDRIVE'] + "\\", "Program Files", "7-Zip", "7z.exe")
|
self.z_path = os.path.join(os.environ['SYSTEMDRIVE'] + "\\", "Program Files", "7-Zip", "7z.exe")
|
||||||
self.recovery_suffixes = (
|
self.recovery_suffixes = (
|
||||||
"recoveryhdupdate.pkg",
|
"recoveryhdupdate.pkg",
|
||||||
"recoveryhdmetadmg.pkg"
|
"recoveryhdmetadmg.pkg"
|
||||||
)
|
)
|
||||||
|
self.dd_bootsector = True
|
||||||
|
self.boot0 = "boot0af"
|
||||||
|
self.boot1 = "boot1f32alt"
|
||||||
|
self.boot = "boot6"
|
||||||
|
self.efi_id = "c12a7328-f81f-11d2-ba4b-00a0c93ec93b" # EFI
|
||||||
|
self.bas_id = "ebd0a0a2-b9e5-4433-87c0-68b6b72699c7" # Microsoft Basic Data
|
||||||
|
self.hfs_id = "48465300-0000-11AA-AA11-00306543ECAC" # HFS+
|
||||||
|
self.rec_id = "426F6F74-0000-11AA-AA11-00306543ECAC" # Apple Boot partition (Recovery HD)
|
||||||
|
|
||||||
|
def verify_os(self):
|
||||||
|
self.u.head("Verifying OS")
|
||||||
|
print("")
|
||||||
|
print("Verifying OS name...")
|
||||||
|
if not os.name=="nt":
|
||||||
|
print("")
|
||||||
|
print("This script is only for Windows!")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to exit...")
|
||||||
|
exit(1)
|
||||||
|
print(" - Name = NT")
|
||||||
|
print("Verifying OS version...")
|
||||||
|
# Verify we're at version 9600 or greater
|
||||||
|
try:
|
||||||
|
# Set plat to the last item of the output split by . - looks like:
|
||||||
|
# Windows-8.1-6.3.9600
|
||||||
|
# or this:
|
||||||
|
# Windows-10-10.0.17134-SP0
|
||||||
|
plat = int(platform.platform().split(".")[-1].split("-")[0])
|
||||||
|
except:
|
||||||
|
plat = 0
|
||||||
|
if plat < self.min_plat:
|
||||||
|
print("")
|
||||||
|
print("Currently running {}, this script requires version {} or newer.".format(platform.platform(), self.min_plat))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to exit...")
|
||||||
|
exit(1)
|
||||||
|
print(" - Version = {}".format(plat))
|
||||||
|
print("")
|
||||||
|
print("{} >= {}, continuing...".format(plat, self.min_plat))
|
||||||
|
|
||||||
def get_disks_of_type(self, disk_list, disk_type=(0,2)):
|
def get_disks_of_type(self, disk_list, disk_type=(0,2)):
|
||||||
disks = {}
|
disks = {}
|
||||||
@ -41,7 +82,7 @@ class WinUSB:
|
|||||||
# or False if issues.
|
# or False if issues.
|
||||||
# Check for dd.exe in the current dir
|
# Check for dd.exe in the current dir
|
||||||
if os.path.exists(os.path.join(self.s_path, self.dd_name)):
|
if os.path.exists(os.path.join(self.s_path, self.dd_name)):
|
||||||
print("Located {}!".format(self.dd_name))
|
# print("Located {}!".format(self.dd_name))
|
||||||
# Got it
|
# Got it
|
||||||
return True
|
return True
|
||||||
print("Couldn't locate {} - downloading...".format(self.dd_name))
|
print("Couldn't locate {} - downloading...".format(self.dd_name))
|
||||||
@ -55,7 +96,7 @@ class WinUSB:
|
|||||||
#
|
#
|
||||||
# Returns True if found, False if not
|
# Returns True if found, False if not
|
||||||
if os.path.exists(self.z_path):
|
if os.path.exists(self.z_path):
|
||||||
print("Located {}!".format(self.z_name))
|
# print("Located {}!".format(self.z_name))
|
||||||
# Got it
|
# Got it
|
||||||
return True
|
return True
|
||||||
print("Didn't locate {} - downloading...".format(self.z_name))
|
print("Didn't locate {} - downloading...".format(self.z_name))
|
||||||
@ -69,11 +110,56 @@ class WinUSB:
|
|||||||
print("")
|
print("")
|
||||||
return os.path.exists(self.z_path)
|
return os.path.exists(self.z_path)
|
||||||
|
|
||||||
|
def check_bi(self):
|
||||||
|
# Checks for BOOTICEx64.exe in our scripts dir
|
||||||
|
# and downloads it if need be
|
||||||
|
if os.path.exists(os.path.join(self.s_path, self.bi_name)):
|
||||||
|
# print("Located {}!".format(self.bi_name))
|
||||||
|
# Got it
|
||||||
|
return True
|
||||||
|
print("Couldn't locate {} - downloading...".format(self.bi_name))
|
||||||
|
temp = tempfile.mkdtemp()
|
||||||
|
zfile = os.path.basename(self.bi_url)
|
||||||
|
print("Downloading {}...".format(os.path.basename(self.bi_url)))
|
||||||
|
self.dl.stream_to_file(self.bi_url, os.path.join(temp,zfile))
|
||||||
|
print("")
|
||||||
|
print(" - Extracting...")
|
||||||
|
# Extract with built-in tools \o/
|
||||||
|
os.chdir(temp)
|
||||||
|
with zipfile.ZipFile(os.path.join(temp,zfile)) as z:
|
||||||
|
z.extractall(temp)
|
||||||
|
for x in os.listdir(temp):
|
||||||
|
if self.bi_name.lower() in x.lower():
|
||||||
|
# Found it
|
||||||
|
print(" - Found {}".format(x))
|
||||||
|
print(" - Copying to {} directory...".format(self.scripts))
|
||||||
|
shutil.copy(os.path.join(temp,x), os.path.join(self.s_path,x))
|
||||||
|
# Return the cd to the local script dir
|
||||||
|
os.chdir(os.path.dirname(os.path.realpath(__file__)))
|
||||||
|
# Remove the temp folder
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
return os.path.exists(os.path.join(self.s_path,self.bi_name))
|
||||||
|
|
||||||
|
def get_dl_info(self):
|
||||||
|
# Returns the latest download package and info in a
|
||||||
|
# dictionary: { "url" : dl_url, "name" : name, "info" : update_info }
|
||||||
|
json_data = self.dl.get_string(self.clover_url, False)
|
||||||
|
if not json_data or not len(json_data):
|
||||||
|
return None
|
||||||
|
try:
|
||||||
|
j = json.loads(json_data)
|
||||||
|
except:
|
||||||
|
return None
|
||||||
|
dl_link = next((x.get("browser_download_url", None) for x in j.get("assets", []) if x.get("browser_download_url", "").lower().endswith(".lzma")), None)
|
||||||
|
if not dl_link:
|
||||||
|
return None
|
||||||
|
return { "url" : dl_link, "name" : os.path.basename(dl_link), "info" : j.get("body", None) }
|
||||||
|
|
||||||
def get_size(self, size):
|
def get_size(self, size):
|
||||||
# Returns the size passed as human readable
|
# Returns the size passed as human readable
|
||||||
if size == -1:
|
if size == -1:
|
||||||
return "Unknown"
|
return "Unknown"
|
||||||
ext = ["B","KB","MB","GB","PB"]
|
ext = ["B","KB","MB","GB","TB","PB"]
|
||||||
s = float(size)
|
s = float(size)
|
||||||
s_dict = {}
|
s_dict = {}
|
||||||
# Iterate the ext list, and divide by 1000 each time
|
# Iterate the ext list, and divide by 1000 each time
|
||||||
@ -99,23 +185,73 @@ class WinUSB:
|
|||||||
else:
|
else:
|
||||||
return "{} {}".format(str(bval).split(".")[0], biggest)
|
return "{} {}".format(str(bval).split(".")[0], biggest)
|
||||||
|
|
||||||
def diskpart_erase(self, disk):
|
def diskpart_flag(self, disk, as_efi=False):
|
||||||
|
# Sets and unsets the GUID needed for a GPT EFI partition ID
|
||||||
|
self.u.head("Changing ID With DiskPart")
|
||||||
|
print("")
|
||||||
|
print("Setting type as {}...".format("EFI" if as_efi else "Basic Data"))
|
||||||
|
print("")
|
||||||
|
# - EFI system partition: c12a7328-f81f-11d2-ba4b-00a0c93ec93b
|
||||||
|
# - Basic data partition: ebd0a0a2-b9e5-4433-87c0-68b6b72699c7
|
||||||
|
dp_script = "\n".join([
|
||||||
|
"select disk {}".format(disk.get("index",-1)),
|
||||||
|
"sel part 1",
|
||||||
|
"set id={}".format(self.efi_id if as_efi else self.bas_id)
|
||||||
|
])
|
||||||
|
temp = tempfile.mkdtemp()
|
||||||
|
script = os.path.join(temp, "diskpart.txt")
|
||||||
|
try:
|
||||||
|
with open(script,"w") as f:
|
||||||
|
f.write(dp_script)
|
||||||
|
except:
|
||||||
|
shutil.rmtree(temp)
|
||||||
|
print("Error creating script!")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# Let's try to run it!
|
||||||
|
out = self.r.run({"args":["diskpart","/s",script],"stream":True})
|
||||||
|
# Ditch our script regardless of whether diskpart worked or not
|
||||||
|
shutil.rmtree(temp)
|
||||||
|
print("")
|
||||||
|
if out[2] != 0:
|
||||||
|
# Error city!
|
||||||
|
print("DiskPart exited with non-zero status ({}). Aborting.".format(out[2]))
|
||||||
|
else:
|
||||||
|
print("Done - You may need to replug your drive for the")
|
||||||
|
print("changes to take effect.")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
|
||||||
|
def diskpart_erase(self, disk, gpt=False):
|
||||||
# Generate a script that we can pipe to diskpart to erase our disk
|
# Generate a script that we can pipe to diskpart to erase our disk
|
||||||
self.u.head("Creating DiskPart Script")
|
self.u.head("Erasing With DiskPart")
|
||||||
print("")
|
print("")
|
||||||
# Then we'll re-gather our disk info on success and move forward
|
# Then we'll re-gather our disk info on success and move forward
|
||||||
# Using MBR to effectively set the individual partition types
|
# Using MBR to effectively set the individual partition types
|
||||||
# Keeps us from having issues mounting the EFI on Windows -
|
# Keeps us from having issues mounting the EFI on Windows -
|
||||||
# and also lets us explicitly set the partition id for the main
|
# and also lets us explicitly set the partition id for the main
|
||||||
# data partition.
|
# data partition.
|
||||||
|
if not gpt:
|
||||||
|
print("Using MBR...")
|
||||||
dp_script = "\n".join([
|
dp_script = "\n".join([
|
||||||
"select disk {}".format(disk.get("index",-1)),
|
"select disk {}".format(disk.get("index",-1)),
|
||||||
"clean",
|
"clean",
|
||||||
"convert mbr",
|
"convert mbr",
|
||||||
"create partition primary size=200",
|
"create partition primary size=200",
|
||||||
"format quick fs=fat32 label='CLOVER'",
|
"format quick fs=fat32 label='CLOVER'",
|
||||||
"create partition primary id=AB", # AF = HFS, AB = Recovery
|
"active",
|
||||||
"active"
|
"create partition primary id=AB" # AF = HFS, AB = Recovery
|
||||||
|
])
|
||||||
|
else:
|
||||||
|
print("Using GPT...")
|
||||||
|
dp_script = "\n".join([
|
||||||
|
"select disk {}".format(disk.get("index",-1)),
|
||||||
|
"clean",
|
||||||
|
"convert gpt",
|
||||||
|
"create partition primary size=200",
|
||||||
|
"format quick fs=fat32 label='CLOVER'",
|
||||||
|
"create partition primary id={}".format(self.hfs_id)
|
||||||
])
|
])
|
||||||
temp = tempfile.mkdtemp()
|
temp = tempfile.mkdtemp()
|
||||||
script = os.path.join(temp, "diskpart.txt")
|
script = os.path.join(temp, "diskpart.txt")
|
||||||
@ -193,20 +329,23 @@ class WinUSB:
|
|||||||
self.u.grab("Press [enter] to return to package selection...")
|
self.u.grab("Press [enter] to return to package selection...")
|
||||||
self.select_package(disk)
|
self.select_package(disk)
|
||||||
return
|
return
|
||||||
|
self.u.head("Extracting Package")
|
||||||
|
print("")
|
||||||
temp = tempfile.mkdtemp()
|
temp = tempfile.mkdtemp()
|
||||||
print(temp)
|
|
||||||
cwd = os.getcwd()
|
cwd = os.getcwd()
|
||||||
os.chdir(temp)
|
os.chdir(temp)
|
||||||
# Extract in sections and remove any files we run into
|
# Extract in sections and remove any files we run into
|
||||||
out = self.r.run({"args":[self.z_path, "e", "-txar", path, "*.dmg"],"stream":True})
|
print("Extracting Recovery dmg...")
|
||||||
|
out = self.r.run({"args":[self.z_path, "e", "-txar", path, "*.dmg"]})
|
||||||
if out[2] != 0:
|
if out[2] != 0:
|
||||||
shutil.rmtree(temp,ignore_errors=True)
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
print("An error occurred extracting: {}".format(out[2]))
|
print("An error occurred extracting: {}".format(out[2]))
|
||||||
print("")
|
print("")
|
||||||
self.u.grab("Press [enter] to return...")
|
self.u.grab("Press [enter] to return...")
|
||||||
return
|
return
|
||||||
|
print("Extracting BaseSystem.dmg...")
|
||||||
# No files to delete here - let's extract the next part
|
# No files to delete here - let's extract the next part
|
||||||
out = self.r.run({"args":[self.z_path, "e", "*.dmg", "*/Base*.dmg"],"stream":True})
|
out = self.r.run({"args":[self.z_path, "e", "*.dmg", "*/Base*.dmg"]})
|
||||||
if out[2] != 0:
|
if out[2] != 0:
|
||||||
shutil.rmtree(temp,ignore_errors=True)
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
print("An error occurred extracting: {}".format(out[2]))
|
print("An error occurred extracting: {}".format(out[2]))
|
||||||
@ -218,9 +357,9 @@ class WinUSB:
|
|||||||
del_list = [x for x in os.listdir(temp) if not (x.lower().startswith("base") and x.lower().endswith(".dmg"))]
|
del_list = [x for x in os.listdir(temp) if not (x.lower().startswith("base") and x.lower().endswith(".dmg"))]
|
||||||
for d in del_list:
|
for d in del_list:
|
||||||
os.remove(os.path.join(temp, d))
|
os.remove(os.path.join(temp, d))
|
||||||
# shutil.rmtree(os.path.join(temp, d),ignore_errors=True)
|
|
||||||
# Onto the last command
|
# Onto the last command
|
||||||
out = self.r.run({"args":[self.z_path, "e", "-tdmg", "Base*.dmg", "*.hfs"],"stream":True})
|
print("Extracting hfs...")
|
||||||
|
out = self.r.run({"args":[self.z_path, "e", "-tdmg", "Base*.dmg", "*.hfs"]})
|
||||||
if out[2] != 0:
|
if out[2] != 0:
|
||||||
shutil.rmtree(temp,ignore_errors=True)
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
print("An error occurred extracting: {}".format(out[2]))
|
print("An error occurred extracting: {}".format(out[2]))
|
||||||
@ -232,7 +371,6 @@ class WinUSB:
|
|||||||
del_list = [x for x in os.listdir(temp) if not x.lower().endswith(".hfs")]
|
del_list = [x for x in os.listdir(temp) if not x.lower().endswith(".hfs")]
|
||||||
for d in del_list:
|
for d in del_list:
|
||||||
os.remove(os.path.join(temp, d))
|
os.remove(os.path.join(temp, d))
|
||||||
# shutil.rmtree(os.path.join(temp, d),ignore_errors=True)
|
|
||||||
print("Extracted successfully!")
|
print("Extracted successfully!")
|
||||||
hfs = next((x for x in os.listdir(temp) if x.lower().endswith(".hfs")),None)
|
hfs = next((x for x in os.listdir(temp) if x.lower().endswith(".hfs")),None)
|
||||||
# Now to dd our image - if it exists
|
# Now to dd our image - if it exists
|
||||||
@ -275,12 +413,168 @@ class WinUSB:
|
|||||||
# that for "Error" then split by that to skip the extra fluff and show only
|
# that for "Error" then split by that to skip the extra fluff and show only
|
||||||
# the error.
|
# the error.
|
||||||
print("An error occurred:\n\n{}".format("Error"+out[1].split("Error")[1]))
|
print("An error occurred:\n\n{}".format("Error"+out[1].split("Error")[1]))
|
||||||
else:
|
print("")
|
||||||
print("Done!")
|
self.u.grab("Press [enter] to return to the main menu...")
|
||||||
|
return
|
||||||
|
# Install Clover to the target drive
|
||||||
|
self.install_clover(disk)
|
||||||
|
|
||||||
|
def install_clover(self, disk):
|
||||||
|
self.u.head("Installing Clover")
|
||||||
|
print("")
|
||||||
|
print("Gathering info...")
|
||||||
|
c = self.get_dl_info()
|
||||||
|
if c == None:
|
||||||
|
print(" - Error communicating with github!")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
print(" - Got {}".format(c.get("name","Unknown Version")))
|
||||||
|
print("Downloading...")
|
||||||
|
temp = tempfile.mkdtemp()
|
||||||
|
os.chdir(temp)
|
||||||
|
clover_lzma = c["name"]
|
||||||
|
self.dl.stream_to_file(c["url"], os.path.join(temp, c["name"]))
|
||||||
|
print("") # Empty space to clear the download progress
|
||||||
|
if not os.path.exists(os.path.join(temp, c["name"])):
|
||||||
|
print(" - Download failed. Aborting...")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# Got a valid file in our temp dir
|
||||||
|
print("Extracting {}...".format(clover_lzma))
|
||||||
|
out = self.r.run({"args":[self.z_path, "e", os.path.join(temp,clover_lzma)]})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred extracting: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# Should result in a .tar file
|
||||||
|
clover_tar = next((x for x in os.listdir(temp) if x.lower().endswith(".tar")),None)
|
||||||
|
if not clover_tar:
|
||||||
|
print(" - No .tar archive found - aborting...")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# Got the .tar archive - get the .iso
|
||||||
|
print("Extracting {}...".format(clover_tar))
|
||||||
|
out = self.r.run({"args":[self.z_path, "e", os.path.join(temp,clover_tar)]})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred extracting: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# Should result in a .iso file
|
||||||
|
clover_iso = next((x for x in os.listdir(temp) if x.lower().endswith(".iso")),None)
|
||||||
|
if not clover_tar:
|
||||||
|
print(" - No .iso found - aborting...")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# Got the .iso - let's extract the needed parts
|
||||||
|
print("Extracting EFI from {}...".format(clover_iso))
|
||||||
|
out = self.r.run({"args":[self.z_path, "x", os.path.join(temp,clover_iso), "EFI*"]})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred extracting: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
print("Extracting {} from {}...".format(self.boot0,clover_iso))
|
||||||
|
out = self.r.run({"args":[self.z_path, "e", os.path.join(temp,clover_iso), self.boot0, "-r"]})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred extracting: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
print("Extracting {} from {}...".format(self.boot1,clover_iso))
|
||||||
|
out = self.r.run({"args":[self.z_path, "e", os.path.join(temp,clover_iso), self.boot1, "-r"]})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred extracting: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
print("Extracting {} from {}...".format(self.boot,clover_iso))
|
||||||
|
out = self.r.run({"args":[self.z_path, "e", os.path.join(temp,clover_iso), self.boot, "-r"]})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred extracting: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# At this point, we should have a boot0xx file and an EFI folder in the temp dir
|
||||||
|
# We need to udpate the disk list though - to reflect the current file system on part 1
|
||||||
|
# of our current disk
|
||||||
|
self.d.update() # assumes our disk number stays the same
|
||||||
|
part = self.d.disks[str(disk["index"])]["partitions"].get("0",{}).get("letter",None) # get the first partition's letter
|
||||||
|
if part == None:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print("Lost original disk - or formatting failed!")
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
# Here we have our disk and partitions and such - the CLOVER partition
|
||||||
|
# will be the first partition
|
||||||
|
# Let's copy over the EFI folder and then dd the boot0xx file
|
||||||
|
print("Copying EFI folder to {}/EFI...".format(part))
|
||||||
|
if os.path.exists("{}/EFI".format(part)):
|
||||||
|
print(" - EFI exists - removing...")
|
||||||
|
shutil.rmtree("{}/EFI".format(part),ignore_errors=True)
|
||||||
|
time.sleep(1) # Added because windows is dumb
|
||||||
|
shutil.copytree(os.path.join(temp,"EFI"), "{}/EFI".format(part))
|
||||||
|
# Copy boot6 over to the root of the EFI volume - and rename it to boot
|
||||||
|
print("Copying {} to {}/boot...".format(self.boot,part))
|
||||||
|
shutil.copy(os.path.join(temp,self.boot),"{}/boot".format(part))
|
||||||
|
# Use bootice to update the MBR and PBR - always on the first
|
||||||
|
# partition (which is 0 in bootice)
|
||||||
|
print("Updating the MBR with {}...".format(self.boot0))
|
||||||
|
args = [
|
||||||
|
os.path.join(self.s_path,self.bi_name),
|
||||||
|
"/device={}".format(disk.get("index",-1)),
|
||||||
|
"/mbr",
|
||||||
|
"/restore",
|
||||||
|
"/file={}".format(os.path.join(temp,self.boot0)),
|
||||||
|
"/keep_dpt",
|
||||||
|
"/quiet"
|
||||||
|
]
|
||||||
|
out = self.r.run({"args":args})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred updating the MBR: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
print("Updating the PBR with {}...".format(self.boot1))
|
||||||
|
args = [
|
||||||
|
os.path.join(self.s_path,self.bi_name),
|
||||||
|
"/device={}:0".format(disk.get("index",-1)),
|
||||||
|
"/pbr",
|
||||||
|
"/restore",
|
||||||
|
"/file={}".format(os.path.join(temp,self.boot1)),
|
||||||
|
"/keep_bpb",
|
||||||
|
"/quiet"
|
||||||
|
]
|
||||||
|
out = self.r.run({"args":args})
|
||||||
|
if out[2] != 0:
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print(" - An error occurred updating the PBR: {}".format(out[2]))
|
||||||
|
print("")
|
||||||
|
self.u.grab("Press [enter] to return...")
|
||||||
|
return
|
||||||
|
print("Cleaning up...")
|
||||||
|
shutil.rmtree(temp,ignore_errors=True)
|
||||||
|
print("")
|
||||||
|
print("Done.")
|
||||||
print("")
|
print("")
|
||||||
self.u.grab("Press [enter] to return to the main menu...")
|
self.u.grab("Press [enter] to return to the main menu...")
|
||||||
|
|
||||||
def main(self):
|
def main(self):
|
||||||
|
# Start out with our cd in the right spot.
|
||||||
|
os.chdir(os.path.dirname(os.path.realpath(__file__)))
|
||||||
# Let's make sure we have the required files needed
|
# Let's make sure we have the required files needed
|
||||||
self.u.head("Checking Required Tools")
|
self.u.head("Checking Required Tools")
|
||||||
print("")
|
print("")
|
||||||
@ -290,6 +584,9 @@ class WinUSB:
|
|||||||
if not self.check_7z():
|
if not self.check_7z():
|
||||||
print("Couldn't find or install {} - aborting!".format(self.z_name))
|
print("Couldn't find or install {} - aborting!".format(self.z_name))
|
||||||
exit(1)
|
exit(1)
|
||||||
|
if not self.check_bi():
|
||||||
|
print("Couldn't find or install {} - aborting!".format(self.bi_name))
|
||||||
|
exit(1)
|
||||||
# Let's just setup a real simple interface and try to write some data
|
# Let's just setup a real simple interface and try to write some data
|
||||||
self.u.head("Gathering Disk Info")
|
self.u.head("Gathering Disk Info")
|
||||||
print("")
|
print("")
|
||||||
@ -300,10 +597,10 @@ class WinUSB:
|
|||||||
# Let's serve up a list of *only* removable media
|
# Let's serve up a list of *only* removable media
|
||||||
self.u.head("Potential Removable Media")
|
self.u.head("Potential Removable Media")
|
||||||
print("")
|
print("")
|
||||||
# rem_disks = self.get_disks_of_type(self.d.disks)
|
rem_disks = self.get_disks_of_type(self.d.disks)
|
||||||
|
|
||||||
# SHOWING ALL DISKS CURRENTLY - CHANGE THIS FOR RELEASE!!!!
|
# SHOWING ALL DISKS CURRENTLY - CHANGE THIS FOR RELEASE!!!!
|
||||||
rem_disks = self.d.disks
|
# rem_disks = self.d.disks
|
||||||
|
|
||||||
# Types: 0 = Unknown, 1 = No Root Dir, 2 = Removable, 3 = Local, 4 = Network, 5 = Disc, 6 = RAM disk
|
# Types: 0 = Unknown, 1 = No Root Dir, 2 = Removable, 3 = Local, 4 = Network, 5 = Disc, 6 = RAM disk
|
||||||
|
|
||||||
@ -333,12 +630,33 @@ class WinUSB:
|
|||||||
print("")
|
print("")
|
||||||
print("Q. Quit")
|
print("Q. Quit")
|
||||||
print("")
|
print("")
|
||||||
menu = self.u.grab("Please select a disk: ")
|
print("Usage: [drive][option (only one allowed)] (eg. 1C)")
|
||||||
|
print(" Options are as follows with precedence C > E > U > G:")
|
||||||
|
print(" C = Only install Clover to the drive's first partition.")
|
||||||
|
print(" E = Sets the type of the drive's first partition to EFI.")
|
||||||
|
print(" U = Similar to E, but sets the type to Basic Data (useful for editing).")
|
||||||
|
print(" G = Format as GPT (default is MBR).")
|
||||||
|
print("")
|
||||||
|
menu = self.u.grab("Please select a disk or press [enter] with no options to refresh: ")
|
||||||
if not len(menu):
|
if not len(menu):
|
||||||
self.main()
|
self.main()
|
||||||
return
|
return
|
||||||
if menu.lower() == "q":
|
if menu.lower() == "q":
|
||||||
self.u.custom_quit()
|
self.u.custom_quit()
|
||||||
|
only_clover = set_efi = unset_efi = use_gpt = False
|
||||||
|
if "c" in menu.lower():
|
||||||
|
only_clover = True
|
||||||
|
menu = menu.lower().replace("c","")
|
||||||
|
if "e" in menu.lower():
|
||||||
|
set_efi = True
|
||||||
|
menu = menu.lower().replace("e","")
|
||||||
|
if "u" in menu.lower():
|
||||||
|
unset_efi = True
|
||||||
|
menu = menu.lower().replace("u","")
|
||||||
|
if "g" in menu.lower():
|
||||||
|
use_gpt = True
|
||||||
|
menu = menu.lower().replace("g","")
|
||||||
|
|
||||||
selected_disk = rem_disks.get(menu,None)
|
selected_disk = rem_disks.get(menu,None)
|
||||||
if not selected_disk:
|
if not selected_disk:
|
||||||
self.u.head("Invalid Choice")
|
self.u.head("Invalid Choice")
|
||||||
@ -349,6 +667,14 @@ class WinUSB:
|
|||||||
self.main()
|
self.main()
|
||||||
return
|
return
|
||||||
# Got a disk!
|
# Got a disk!
|
||||||
|
if only_clover:
|
||||||
|
self.install_clover(selected_disk)
|
||||||
|
elif set_efi:
|
||||||
|
self.diskpart_flag(selected_disk, True)
|
||||||
|
elif unset_efi:
|
||||||
|
self.diskpart_flag(selected_disk, False)
|
||||||
|
else:
|
||||||
|
# Check erase
|
||||||
while True:
|
while True:
|
||||||
self.u.head("Erase {}".format(selected_disk.get("model","Unknown")))
|
self.u.head("Erase {}".format(selected_disk.get("model","Unknown")))
|
||||||
print("")
|
print("")
|
||||||
@ -370,7 +696,7 @@ class WinUSB:
|
|||||||
if yn.lower() == "y":
|
if yn.lower() == "y":
|
||||||
break
|
break
|
||||||
# Got the OK to erase! Let's format a diskpart script!
|
# Got the OK to erase! Let's format a diskpart script!
|
||||||
self.diskpart_erase(selected_disk)
|
self.diskpart_erase(selected_disk, use_gpt)
|
||||||
self.main()
|
self.main()
|
||||||
|
|
||||||
w = WinUSB()
|
w = WinUSB()
|
||||||
|
Loading…
Reference in New Issue
Block a user