1
0
Fork 0
mirror of https://github.com/foxlet/macOS-Simple-KVM.git synced 2025-01-14 13:09:45 +01:00
macOS-Simple-KVM/tools/FetchMacOS/fetch-macos.py

139 lines
5.8 KiB
Python
Raw Normal View History

2019-04-23 05:20:12 +02:00
#!/usr/bin/python
"""fetch-macos.py: Fetches macOS products from Apple's SoftwareUpdate service."""
import logging
import plistlib
import os
import errno
import click
import requests
import sys
2019-04-23 05:20:12 +02:00
__author__ = "Foxlet"
2019-06-07 01:36:28 +02:00
__copyright__ = "Copyright 2019, FurCode Project"
2019-04-23 05:20:12 +02:00
__license__ = "GPLv3"
2019-06-07 01:36:28 +02:00
__version__ = "1.3"
2019-04-23 05:20:12 +02:00
logging.basicConfig(format='%(asctime)-15s %(message)s', level=logging.INFO)
logger = logging.getLogger('webactivity')
class ClientMeta:
# Client used to connect to the Software CDN
osinstall = {"User-Agent":"osinstallersetupplaind (unknown version) CFNetwork/720.5.7 Darwin/14.5.0 (x86_64)"}
# Client used to connect to the Software Distribution service
swupdate = {"User-Agent":"Software%20Update (unknown version) CFNetwork/807.0.1 Darwin/16.0.0 (x86_64)"}
class Filesystem:
@staticmethod
def download_file(url, size, path):
label = url.split('/')[-1]
filename = os.path.join(path, label)
# Set to stream mode for large files
remote = requests.get(url, stream=True, headers=ClientMeta.osinstall)
with open(filename, 'wb') as f:
with click.progressbar(remote.iter_content(1024), length=size/1024, label="Fetching {} ...".format(filename)) as stream:
for data in stream:
f.write(data)
return filename
@staticmethod
def check_directory(path):
try:
os.makedirs(path)
except OSError as exception:
if exception.errno != errno.EEXIST:
raise
class SoftwareService:
2019-06-07 01:36:28 +02:00
# macOS 10.15 is available in 4 different catalogs from SoftwareScan
2019-06-03 21:58:00 +02:00
catalogs = {"CustomerSeed":"https://swscan.apple.com/content/catalogs/others/index-10.15customerseed-10.15-10.14-10.13-10.12-10.11-10.10-10.9-mountainlion-lion-snowleopard-leopard.merged-1.sucatalog",
"DeveloperSeed":"https://swscan.apple.com/content/catalogs/others/index-10.15seed-10.15-10.14-10.13-10.12-10.11-10.10-10.9-mountainlion-lion-snowleopard-leopard.merged-1.sucatalog",
"PublicSeed":"https://swscan.apple.com/content/catalogs/others/index-10.15beta-10.15-10.14-10.13-10.12-10.11-10.10-10.9-mountainlion-lion-snowleopard-leopard.merged-1.sucatalog",
2019-06-07 01:36:28 +02:00
"PublicRelease":"https://swscan.apple.com/content/catalogs/others/index-10.15-10.14-10.13-10.12-10.11-10.10-10.9-mountainlion-lion-snowleopard-leopard.merged-1.sucatalog",
"PublicRelease14":"https://swscan.apple.com/content/catalogs/others/index-10.14-10.13-10.12-10.11-10.10-10.9-mountainlion-lion-snowleopard-leopard.merged-1.sucatalog",
"PublicRelease13":"https://swscan.apple.com/content/catalogs/others/index-10.13-10.12-10.11-10.10-10.9-mountainlion-lion-snowleopard-leopard.merged-1.sucatalog"}
2019-04-23 05:20:12 +02:00
def __init__(self, catalog_id):
self.catalog_url = self.catalogs.get(catalog_id, self.catalogs["PublicRelease"])
self.catalog_data = ""
def getcatalog(self):
logging.info("Network Request: %s", "Fetching {}".format(self.catalog_url))
catalog_raw = requests.get(self.catalog_url, headers=ClientMeta.swupdate)
self.catalog_data = catalog_raw.text.encode('UTF-8')
return catalog_raw.text.encode('UTF-8')
def getosinstall(self):
2019-06-07 01:36:28 +02:00
# Load catalogs based on Py3/2 lib
if sys.version_info > (3, 0):
root = plistlib.loads(self.catalog_data)
else:
root = plistlib.readPlistFromString(self.catalog_data)
2019-06-07 01:36:28 +02:00
# Iterate to find valid OSInstall packages
ospackages = []
2019-04-23 05:20:12 +02:00
products = root['Products']
for product in products:
2019-06-07 01:36:28 +02:00
if products.get(product, {}).get('ExtendedMetaInfo', {}).get('InstallAssistantPackageIdentifiers', {}).get('OSInstall', {}) == 'com.apple.mpkg.OSInstall':
ospackages.append(product)
return ospackages
2019-04-23 05:20:12 +02:00
class MacOSProduct:
def __init__(self, catalog, product_id):
2019-06-07 01:36:28 +02:00
if sys.version_info > (3, 0):
root = plistlib.loads(catalog)
else:
root = plistlib.readPlistFromString(catalog)
2019-04-23 05:20:12 +02:00
products = root['Products']
self.date = root['IndexDate']
self.product = products[product_id]
2019-06-07 01:36:28 +02:00
def fetchpackages(self, path, keyword=None):
2019-04-23 05:20:12 +02:00
Filesystem.check_directory(path)
packages = self.product['Packages']
2019-06-07 01:36:28 +02:00
if keyword:
for item in packages:
if keyword in item.get("URL"):
Filesystem.download_file(item.get("URL"), item.get("Size"), path)
else:
for item in packages:
2019-04-23 05:20:12 +02:00
Filesystem.download_file(item.get("URL"), item.get("Size"), path)
@click.command()
@click.option('-o', '--output-dir', default="BaseSystem/", help="Target directory for package output.")
@click.option('-c', '--catalog-id', default="PublicRelease", help="Name of catalog.")
2019-04-23 05:20:12 +02:00
@click.option('-p', '--product-id', default="", help="Product ID (as seen in SoftwareUpdate).")
@click.option('-l', '--latest', is_flag=True, help="Get latest available macOS package.")
def fetchmacos(output_dir="BaseSystem/", catalog_id="PublicRelease", product_id="", latest=False):
2019-04-23 05:20:12 +02:00
# Get the remote catalog data
remote = SoftwareService(catalog_id)
catalog = remote.getcatalog()
2019-06-07 01:36:28 +02:00
# If latest is used, find the latest OSInstall package
2019-04-23 05:20:12 +02:00
if latest:
2019-06-07 01:36:28 +02:00
product_id = remote.getosinstall()[-1]
2019-04-23 05:20:12 +02:00
else:
if product_id == "":
print("You must provide a Product ID (or pass the -l flag) to continue.")
2019-04-23 05:20:12 +02:00
exit(1)
2019-06-07 01:36:28 +02:00
# Fetch the given Product ID
2019-04-23 05:20:12 +02:00
try:
2019-06-07 01:36:28 +02:00
product = MacOSProduct(catalog, product_id)
2019-04-23 05:20:12 +02:00
except KeyError:
print("Product ID {} could not be found.".format(product_id))
2019-04-23 05:20:12 +02:00
exit(1)
logging.info("Selected macOS Product: {}".format(product_id))
# Download package to disk
2019-06-07 01:36:28 +02:00
product.fetchpackages(output_dir, keyword="BaseSystem")
2019-04-23 05:20:12 +02:00
if __name__ == "__main__":
fetchmacos()