mirror of
https://github.com/onionshare/onionshare.git
synced 2025-01-12 07:49:39 -05:00
Install snapcraft before building the snap, and fix some macos cleanup
This commit is contained in:
parent
4e5ad4d0ca
commit
d54b51e7b1
@ -214,6 +214,9 @@ jobs:
|
|||||||
working_directory: ~/repo
|
working_directory: ~/repo
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
|
- run:
|
||||||
|
name: Install snapcraft
|
||||||
|
command: sudo snap install snapcraft --classic
|
||||||
- run:
|
- run:
|
||||||
name: Build the snap
|
name: Build the snap
|
||||||
command: |
|
command: |
|
||||||
|
@ -1,308 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
import os
|
|
||||||
import inspect
|
|
||||||
import subprocess
|
|
||||||
import shutil
|
|
||||||
import itertools
|
|
||||||
import glob
|
|
||||||
|
|
||||||
root = os.path.dirname(
|
|
||||||
os.path.dirname(
|
|
||||||
os.path.dirname(os.path.abspath(inspect.getfile(inspect.currentframe())))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def run(cmd, cwd=None, error_ok=False):
|
|
||||||
print(f"{cmd} # cwd={cwd}")
|
|
||||||
subprocess.run(cmd, cwd=cwd, check=True)
|
|
||||||
|
|
||||||
|
|
||||||
def get_size(dir):
|
|
||||||
size = 0
|
|
||||||
for path, dirs, files in os.walk(dir):
|
|
||||||
for f in files:
|
|
||||||
fp = os.path.join(path, f)
|
|
||||||
size += os.path.getsize(fp)
|
|
||||||
return size
|
|
||||||
|
|
||||||
|
|
||||||
def codesign(path, entitlements, identity):
|
|
||||||
run(
|
|
||||||
[
|
|
||||||
"codesign",
|
|
||||||
"--sign",
|
|
||||||
identity,
|
|
||||||
"--entitlements",
|
|
||||||
str(entitlements),
|
|
||||||
"--timestamp",
|
|
||||||
"--deep",
|
|
||||||
"--force",
|
|
||||||
"--options",
|
|
||||||
"runtime,library",
|
|
||||||
str(path),
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
desktop_dir = f"{root}/desktop"
|
|
||||||
app_dir = f"{desktop_dir}/build/OnionShare.app"
|
|
||||||
|
|
||||||
print("○ Clean up from last build")
|
|
||||||
if os.path.exists(f"{desktop_dir}/build"):
|
|
||||||
shutil.rmtree(f"{desktop_dir}/build")
|
|
||||||
if os.path.exists(f"{desktop_dir}/dist"):
|
|
||||||
shutil.rmtree(f"{desktop_dir}/dist")
|
|
||||||
|
|
||||||
print("○ Building binaries")
|
|
||||||
run(
|
|
||||||
[
|
|
||||||
shutil.which("python"),
|
|
||||||
"setup-freeze.py",
|
|
||||||
"bdist_mac",
|
|
||||||
],
|
|
||||||
desktop_dir,
|
|
||||||
)
|
|
||||||
before_size = get_size(f"{app_dir}")
|
|
||||||
|
|
||||||
print("○ Delete unused Qt Frameworks")
|
|
||||||
for framework in [
|
|
||||||
"Qt3DAnimation",
|
|
||||||
"Qt3DCore",
|
|
||||||
"Qt3DExtras",
|
|
||||||
"Qt3DInput",
|
|
||||||
"Qt3DLogic",
|
|
||||||
"Qt3DQuick",
|
|
||||||
"Qt3DQuickAnimation",
|
|
||||||
"Qt3DQuickExtras",
|
|
||||||
"Qt3DQuickInput",
|
|
||||||
"Qt3DQuickRender",
|
|
||||||
"Qt3DQuickScene2D",
|
|
||||||
"Qt3DRender",
|
|
||||||
"QtBluetooth",
|
|
||||||
"QtBodymovin",
|
|
||||||
"QtCharts",
|
|
||||||
"QtConcurrent",
|
|
||||||
"QtDataVisualization",
|
|
||||||
"QtDesigner",
|
|
||||||
"QtDesignerComponents",
|
|
||||||
"QtGamepad",
|
|
||||||
"QtHelp",
|
|
||||||
"QtLocation",
|
|
||||||
"QtMultimedia",
|
|
||||||
"QtMultimediaQuick",
|
|
||||||
"QtMultimediaWidgets",
|
|
||||||
"QtNetwork",
|
|
||||||
"QtNetworkAuth",
|
|
||||||
"QtNfc",
|
|
||||||
"QtOpenGL",
|
|
||||||
"QtPdf",
|
|
||||||
"QtPdfWidgets",
|
|
||||||
"QtPositioning",
|
|
||||||
"QtPositioningQuick",
|
|
||||||
"QtPrintSupport",
|
|
||||||
"QtPurchasing",
|
|
||||||
"QtQml",
|
|
||||||
"QtQmlModels",
|
|
||||||
"QtQmlWorkerScript",
|
|
||||||
"QtQuick",
|
|
||||||
"QtQuick3D",
|
|
||||||
"QtQuick3DAssetImport",
|
|
||||||
"QtQuick3DRender",
|
|
||||||
"QtQuick3DRuntimeRender",
|
|
||||||
"QtQuick3DUtils",
|
|
||||||
"QtQuickControls2",
|
|
||||||
"QtQuickParticles",
|
|
||||||
"QtQuickShapes",
|
|
||||||
"QtQuickTemplates2",
|
|
||||||
"QtQuickTest",
|
|
||||||
"QtQuickWidgets",
|
|
||||||
"QtRemoteObjects",
|
|
||||||
"QtRepParser",
|
|
||||||
"QtScript",
|
|
||||||
"QtScriptTools",
|
|
||||||
"QtScxml",
|
|
||||||
"QtSensors",
|
|
||||||
"QtSerialBus",
|
|
||||||
"QtSerialPort",
|
|
||||||
"QtSql",
|
|
||||||
"QtSvg",
|
|
||||||
"QtTest",
|
|
||||||
"QtTextToSpeech",
|
|
||||||
"QtUiPlugin",
|
|
||||||
"QtVirtualKeyboard",
|
|
||||||
"QtWebChannel",
|
|
||||||
"QtWebEngine",
|
|
||||||
"QtWebEngineCore",
|
|
||||||
"QtWebEngineWidgets",
|
|
||||||
"QtWebSockets",
|
|
||||||
"QtWebView",
|
|
||||||
"QtXml",
|
|
||||||
"QtXmlPatterns",
|
|
||||||
]:
|
|
||||||
shutil.rmtree(
|
|
||||||
f"{app_dir}/Contents/MacOS/lib/PySide2/Qt/lib/{framework}.framework"
|
|
||||||
)
|
|
||||||
try:
|
|
||||||
os.remove(
|
|
||||||
f"{app_dir}/Contents/MacOS/lib/PySide2/{framework}.abi3.so"
|
|
||||||
)
|
|
||||||
os.remove(
|
|
||||||
f"{app_dir}/Contents/MacOS/lib/PySide2/{framework}.pyi"
|
|
||||||
)
|
|
||||||
except FileNotFoundError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
print("○ Move files around so Apple will notarize")
|
|
||||||
# https://github.com/marcelotduarte/cx_Freeze/issues/594
|
|
||||||
# https://gist.github.com/TechnicalPirate/259a9c24878fcad948452cb148af2a2c#file-custom_bdist_mac-py-L415
|
|
||||||
|
|
||||||
# Move lib from MacOS into Resources
|
|
||||||
os.rename(
|
|
||||||
f"{app_dir}/Contents/MacOS/lib",
|
|
||||||
f"{app_dir}/Contents/Resources/lib",
|
|
||||||
)
|
|
||||||
run(
|
|
||||||
["ln", "-s", "../Resources/lib"],
|
|
||||||
cwd=f"{app_dir}/Contents/MacOS",
|
|
||||||
)
|
|
||||||
|
|
||||||
# Move frameworks from Resources/lib into Frameworks
|
|
||||||
os.makedirs(f"{app_dir}/Contents/Frameworks", exist_ok=True)
|
|
||||||
for framework_filename in glob.glob(
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/lib/Qt*.framework"
|
|
||||||
):
|
|
||||||
basename = os.path.basename(framework_filename)
|
|
||||||
|
|
||||||
os.rename(framework_filename, f"{app_dir}/Contents/Frameworks/{basename}")
|
|
||||||
run(
|
|
||||||
["ln", "-s", f"../../../../../Frameworks/{basename}"],
|
|
||||||
cwd=f"{app_dir}/Contents/Resources/lib/PySide2/Qt/lib",
|
|
||||||
)
|
|
||||||
if os.path.exists(f"{app_dir}/Contents/Frameworks/{basename}/Resources"):
|
|
||||||
os.rename(
|
|
||||||
f"{app_dir}/Contents/Frameworks/{basename}/Resources",
|
|
||||||
f"{app_dir}/Contents/Frameworks/{basename}/Versions/5/Resources"
|
|
||||||
)
|
|
||||||
run(
|
|
||||||
["ln", "-s", "Versions/5/Resources"],
|
|
||||||
cwd=f"{app_dir}/Contents/Frameworks/{basename}",
|
|
||||||
)
|
|
||||||
|
|
||||||
run(
|
|
||||||
["ln", "-s", "5", "Current"],
|
|
||||||
cwd=f"{app_dir}/Contents/Frameworks/{basename}/Versions",
|
|
||||||
)
|
|
||||||
|
|
||||||
# Move Qt plugins
|
|
||||||
os.rename(
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/plugins",
|
|
||||||
f"{app_dir}/Contents/Frameworks/plugins",
|
|
||||||
)
|
|
||||||
run(
|
|
||||||
["ln", "-s", "../../../../Frameworks/plugins"],
|
|
||||||
cwd=f"{app_dir}/Contents/Resources/lib/PySide2/Qt",
|
|
||||||
)
|
|
||||||
|
|
||||||
print("○ Delete more unused PySide2 stuff to save space")
|
|
||||||
for filename in [
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/Designer.app",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/examples",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/glue",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/include",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/pyside2-lupdate",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/qml",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/libpyside2.abi3.5.15.dylib",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/lib/QtRepParser.framework",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/lib/QtUiPlugin.framework",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/lib/QtWebEngineCore.framework/Helpers",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/shiboken2/libshiboken2.abi3.5.15.dylib",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/shiboken2/docs",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/rcc",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/PySide2/uic",
|
|
||||||
]:
|
|
||||||
if os.path.isdir(filename):
|
|
||||||
shutil.rmtree(filename)
|
|
||||||
elif os.path.isfile(filename):
|
|
||||||
os.remove(filename)
|
|
||||||
else:
|
|
||||||
print(f"Cannot delete, filename not found: {filename}")
|
|
||||||
|
|
||||||
after_size = get_size(f"{app_dir}")
|
|
||||||
freed_bytes = before_size - after_size
|
|
||||||
freed_mb = int(freed_bytes / 1024 / 1024)
|
|
||||||
print(f"○ Freed {freed_mb} mb")
|
|
||||||
|
|
||||||
print("○ Sign app bundle")
|
|
||||||
identity_name_application = "Developer ID Application: Micah Lee (N9B95FDWH4)"
|
|
||||||
entitlements_plist_path = f"{desktop_dir}/package/Entitlements.plist"
|
|
||||||
|
|
||||||
for path in itertools.chain(
|
|
||||||
glob.glob(f"{app_dir}/Contents/Resources/lib/**/*.so", recursive=True),
|
|
||||||
glob.glob(f"{app_dir}/Contents/Resources/lib/**/*.dylib", recursive=True),
|
|
||||||
[
|
|
||||||
f"{app_dir}/Contents/Frameworks/QtCore.framework/Versions/5/QtCore",
|
|
||||||
f"{app_dir}/Contents/Frameworks/QtDBus.framework/Versions/5/QtDBus",
|
|
||||||
f"{app_dir}/Contents/Frameworks/QtGui.framework/Versions/5/QtGui",
|
|
||||||
f"{app_dir}/Contents/Frameworks/QtMacExtras.framework/Versions/5/QtMacExtras",
|
|
||||||
f"{app_dir}/Contents/Frameworks/QtWidgets.framework/Versions/5/QtWidgets",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/Python",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/meek-client",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/obfs4proxy",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/snowflake-client",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/tor",
|
|
||||||
f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/libevent-2.1.7.dylib",
|
|
||||||
f"{app_dir}/Contents/MacOS/onionshare",
|
|
||||||
f"{app_dir}/Contents/MacOS/onionshare-cli",
|
|
||||||
f"{app_dir}",
|
|
||||||
],
|
|
||||||
):
|
|
||||||
codesign(path, entitlements_plist_path, identity_name_application)
|
|
||||||
|
|
||||||
print(f"○ Signed app bundle: {app_dir}")
|
|
||||||
|
|
||||||
if not os.path.exists("/usr/local/bin/create-dmg"):
|
|
||||||
print("○ Error: create-dmg is not installed")
|
|
||||||
return
|
|
||||||
|
|
||||||
print("○ Create DMG")
|
|
||||||
version_filename = f"{root}/cli/onionshare_cli/resources/version.txt"
|
|
||||||
with open(version_filename) as f:
|
|
||||||
version = f.read().strip()
|
|
||||||
|
|
||||||
os.makedirs(f"{desktop_dir}/dist", exist_ok=True)
|
|
||||||
dmg_path = f"{desktop_dir}/dist/OnionShare-{version}.dmg"
|
|
||||||
run(
|
|
||||||
[
|
|
||||||
"create-dmg",
|
|
||||||
"--volname",
|
|
||||||
"OnionShare",
|
|
||||||
"--volicon",
|
|
||||||
f"{desktop_dir}/onionshare/resources/onionshare.icns",
|
|
||||||
"--window-size",
|
|
||||||
"400",
|
|
||||||
"200",
|
|
||||||
"--icon-size",
|
|
||||||
"100",
|
|
||||||
"--icon",
|
|
||||||
"OnionShare.app",
|
|
||||||
"100",
|
|
||||||
"70",
|
|
||||||
"--hide-extension",
|
|
||||||
"OnionShare.app",
|
|
||||||
"--app-drop-link",
|
|
||||||
"300",
|
|
||||||
"70",
|
|
||||||
dmg_path,
|
|
||||||
f"{app_dir}",
|
|
||||||
"--identity",
|
|
||||||
identity_name_application,
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
print(f"○ Finished building DMG: {dmg_path}")
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
@ -1,596 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import inspect
|
|
||||||
import subprocess
|
|
||||||
import shutil
|
|
||||||
import uuid
|
|
||||||
import xml.etree.ElementTree as ET
|
|
||||||
|
|
||||||
|
|
||||||
root = os.path.dirname(
|
|
||||||
os.path.dirname(
|
|
||||||
os.path.dirname(os.path.abspath(inspect.getfile(inspect.currentframe())))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def run(cmd, cwd=None, error_ok=False):
|
|
||||||
print(cmd)
|
|
||||||
try:
|
|
||||||
subprocess.run(cmd, cwd=cwd, check=True)
|
|
||||||
except subprocess.CalledProcessError as e:
|
|
||||||
if not error_ok:
|
|
||||||
raise subprocess.CalledProcessError(e)
|
|
||||||
|
|
||||||
|
|
||||||
def sign(filename, cwd=None):
|
|
||||||
run(
|
|
||||||
[
|
|
||||||
shutil.which("signtool"),
|
|
||||||
"sign",
|
|
||||||
"/v",
|
|
||||||
"/d",
|
|
||||||
"OnionShare",
|
|
||||||
"/sha1",
|
|
||||||
"bb1d265ab02272e8fc742f149dcf8751cac63f50",
|
|
||||||
"/fd",
|
|
||||||
"SHA256",
|
|
||||||
"/td",
|
|
||||||
"SHA256",
|
|
||||||
"/tr",
|
|
||||||
"http://timestamp.digicert.com",
|
|
||||||
filename,
|
|
||||||
],
|
|
||||||
cwd,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def get_size(dir):
|
|
||||||
size = 0
|
|
||||||
for path, dirs, files in os.walk(dir):
|
|
||||||
for f in files:
|
|
||||||
fp = os.path.join(path, f)
|
|
||||||
size += os.path.getsize(fp)
|
|
||||||
return size
|
|
||||||
|
|
||||||
|
|
||||||
def wix_build_data(dirname, dir_prefix, id_, name):
|
|
||||||
data = {
|
|
||||||
"id": id_,
|
|
||||||
"name": name,
|
|
||||||
"files": [],
|
|
||||||
"dirs": [],
|
|
||||||
}
|
|
||||||
|
|
||||||
for basename in os.listdir(dirname):
|
|
||||||
filename = os.path.join(dirname, basename)
|
|
||||||
if os.path.isfile(filename):
|
|
||||||
data["files"].append(os.path.join(dir_prefix, basename))
|
|
||||||
elif os.path.isdir(filename):
|
|
||||||
if id_ == "INSTALLDIR":
|
|
||||||
id_prefix = "Folder"
|
|
||||||
else:
|
|
||||||
id_prefix = id_
|
|
||||||
|
|
||||||
# Skip lib/Pyside2/Examples folder
|
|
||||||
if "\\lib\\PySide2\\examples" in dirname:
|
|
||||||
continue
|
|
||||||
|
|
||||||
id_value = f"{id_prefix}{basename.capitalize().replace('-', '_')}"
|
|
||||||
data["dirs"].append(
|
|
||||||
wix_build_data(
|
|
||||||
os.path.join(dirname, basename),
|
|
||||||
os.path.join(dir_prefix, basename),
|
|
||||||
id_value,
|
|
||||||
basename,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
if len(data["files"]) > 0:
|
|
||||||
if id_ == "INSTALLDIR":
|
|
||||||
data["component_id"] = "ApplicationFiles"
|
|
||||||
else:
|
|
||||||
data["component_id"] = "FolderComponent" + id_[len("Folder") :]
|
|
||||||
data["component_guid"] = str(uuid.uuid4())
|
|
||||||
|
|
||||||
return data
|
|
||||||
|
|
||||||
|
|
||||||
def wix_build_dir_xml(root, data):
|
|
||||||
attrs = {}
|
|
||||||
if "id" in data:
|
|
||||||
attrs["Id"] = data["id"]
|
|
||||||
if "name" in data:
|
|
||||||
attrs["Name"] = data["name"]
|
|
||||||
el = ET.SubElement(root, "Directory", attrs)
|
|
||||||
for subdata in data["dirs"]:
|
|
||||||
wix_build_dir_xml(el, subdata)
|
|
||||||
|
|
||||||
# If this is the ProgramMenuFolder, add the menu component
|
|
||||||
if "id" in data and data["id"] == "ProgramMenuFolder":
|
|
||||||
component_el = ET.SubElement(
|
|
||||||
el,
|
|
||||||
"Component",
|
|
||||||
Id="ApplicationShortcuts",
|
|
||||||
Guid="539e7de8-a124-4c09-aa55-0dd516aad7bc",
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
component_el,
|
|
||||||
"Shortcut",
|
|
||||||
Id="ApplicationShortcut1",
|
|
||||||
Name="OnionShare",
|
|
||||||
Description="OnionShare",
|
|
||||||
Target="[INSTALLDIR]onionshare.exe",
|
|
||||||
WorkingDirectory="INSTALLDIR",
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
component_el,
|
|
||||||
"RegistryValue",
|
|
||||||
Root="HKCU",
|
|
||||||
Key="Software\OnionShare",
|
|
||||||
Name="installed",
|
|
||||||
Type="integer",
|
|
||||||
Value="1",
|
|
||||||
KeyPath="yes",
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def wix_build_components_xml(root, data):
|
|
||||||
component_ids = []
|
|
||||||
if "component_id" in data:
|
|
||||||
component_ids.append(data["component_id"])
|
|
||||||
|
|
||||||
for subdata in data["dirs"]:
|
|
||||||
if "component_guid" in subdata:
|
|
||||||
dir_ref_el = ET.SubElement(root, "DirectoryRef", Id=subdata["id"])
|
|
||||||
component_el = ET.SubElement(
|
|
||||||
dir_ref_el,
|
|
||||||
"Component",
|
|
||||||
Id=subdata["component_id"],
|
|
||||||
Guid=subdata["component_guid"],
|
|
||||||
)
|
|
||||||
for filename in subdata["files"]:
|
|
||||||
file_el = ET.SubElement(
|
|
||||||
component_el, "File", Source=filename, Id="file_" + uuid.uuid4().hex
|
|
||||||
)
|
|
||||||
|
|
||||||
component_ids += wix_build_components_xml(root, subdata)
|
|
||||||
|
|
||||||
return component_ids
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
# Figure out the architecture and python path
|
|
||||||
if "64 bit" in sys.version:
|
|
||||||
python_arch = "win-amd64"
|
|
||||||
else:
|
|
||||||
python_arch = "win32"
|
|
||||||
|
|
||||||
if os.getlogin() == "circleci" and python_arch == "win32":
|
|
||||||
python_path = "C:\\Python-32bit\\python"
|
|
||||||
else:
|
|
||||||
python_path = shutil.which("python")
|
|
||||||
|
|
||||||
desktop_dir = os.path.join(root, "desktop")
|
|
||||||
|
|
||||||
print("> Clean up from last build")
|
|
||||||
if os.path.exists(os.path.join(desktop_dir, "build")):
|
|
||||||
shutil.rmtree(os.path.join(desktop_dir, "build"))
|
|
||||||
if os.path.exists(os.path.join(desktop_dir, "dist")):
|
|
||||||
shutil.rmtree(os.path.join(desktop_dir, "dist"))
|
|
||||||
|
|
||||||
print("> Building binaries")
|
|
||||||
run(
|
|
||||||
[
|
|
||||||
python_path,
|
|
||||||
"setup-freeze.py",
|
|
||||||
"build",
|
|
||||||
],
|
|
||||||
desktop_dir,
|
|
||||||
)
|
|
||||||
|
|
||||||
build_path = os.path.join(desktop_dir, "build", f"exe.{python_arch}-3.9")
|
|
||||||
|
|
||||||
before_size = get_size(build_path)
|
|
||||||
|
|
||||||
print("> Delete unused PySide2 stuff to save space")
|
|
||||||
for dirname in ["examples", "qml"]:
|
|
||||||
shutil.rmtree(os.path.join(build_path, "lib", "PySide2", dirname))
|
|
||||||
for filename in [
|
|
||||||
"lconvert.exe",
|
|
||||||
"linguist.exe",
|
|
||||||
"lrelease.exe",
|
|
||||||
"lupdate.exe",
|
|
||||||
"plugins/assetimporters/assimp.dll",
|
|
||||||
"plugins/assetimporters/uip.dll",
|
|
||||||
"plugins/audio/qtaudio_wasapi.dll",
|
|
||||||
"plugins/audio/qtaudio_windows.dll",
|
|
||||||
"plugins/bearer/qgenericbearer.dll",
|
|
||||||
"plugins/canbus/qtpassthrucanbus.dll",
|
|
||||||
"plugins/canbus/qtpeakcanbus.dll",
|
|
||||||
"plugins/canbus/qtsysteccanbus.dll",
|
|
||||||
"plugins/canbus/qttinycanbus.dll",
|
|
||||||
"plugins/canbus/qtvectorcanbus.dll",
|
|
||||||
"plugins/canbus/qtvirtualcanbus.dll",
|
|
||||||
"plugins/gamepads/xinputgamepad.dll",
|
|
||||||
"plugins/generic/qtuiotouchplugin.dll",
|
|
||||||
"plugins/geometryloaders/defaultgeometryloader.dll",
|
|
||||||
"plugins/geometryloaders/gltfgeometryloader.dll",
|
|
||||||
"plugins/geoservices/qtgeoservices_esri.dll",
|
|
||||||
"plugins/geoservices/qtgeoservices_itemsoverlay.dll",
|
|
||||||
"plugins/geoservices/qtgeoservices_mapbox.dll",
|
|
||||||
"plugins/geoservices/qtgeoservices_nokia.dll",
|
|
||||||
"plugins/geoservices/qtgeoservices_osm.dll",
|
|
||||||
"plugins/mediaservice/dsengine.dll",
|
|
||||||
"plugins/mediaservice/qtmedia_audioengine.dll",
|
|
||||||
"plugins/mediaservice/wmfengine.dll",
|
|
||||||
"plugins/platforminputcontexts/qtvirtualkeyboardplugin.dll",
|
|
||||||
"plugins/platforms/qdirect2d.dll",
|
|
||||||
"plugins/platforms/qoffscreen.dll",
|
|
||||||
"plugins/platforms/qwebgl.dll",
|
|
||||||
"plugins/platformthemes/qxdgdesktopportal.dll",
|
|
||||||
"plugins/playlistformats/qtmultimedia_m3u.dll",
|
|
||||||
"plugins/position/qtposition_positionpoll.dll",
|
|
||||||
"plugins/position/qtposition_serialnmea.dll",
|
|
||||||
"plugins/position/qtposition_winrt.dll",
|
|
||||||
"plugins/printsupport/windowsprintersupport.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_debugger.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_inspector.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_local.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_messages.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_native.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_nativedebugger.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_preview.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_profiler.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_quickprofiler.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_server.dll",
|
|
||||||
"plugins/qmltooling/qmldbg_tcp.dll",
|
|
||||||
"plugins/renderers/openglrenderer.dll",
|
|
||||||
"plugins/renderplugins/scene2d.dll",
|
|
||||||
"plugins/scenegraph/qsgd3d12backend.dll",
|
|
||||||
"plugins/sceneparsers/gltfsceneexport.dll",
|
|
||||||
"plugins/sceneparsers/gltfsceneimport.dll",
|
|
||||||
"plugins/sensorgestures/qtsensorgestures_plugin.dll",
|
|
||||||
"plugins/sensorgestures/qtsensorgestures_shakeplugin.dll",
|
|
||||||
"plugins/sensors/qtsensors_generic.dll",
|
|
||||||
"plugins/sqldrivers/qsqlite.dll",
|
|
||||||
"plugins/sqldrivers/qsqlodbc.dll",
|
|
||||||
"plugins/sqldrivers/qsqlpsql.dll",
|
|
||||||
"plugins/styles/qwindowsvistastyle.dll",
|
|
||||||
"plugins/texttospeech/qtexttospeech_sapi.dll",
|
|
||||||
"plugins/virtualkeyboard/qtvirtualkeyboard_hangul.dll",
|
|
||||||
"plugins/virtualkeyboard/qtvirtualkeyboard_openwnn.dll",
|
|
||||||
"plugins/virtualkeyboard/qtvirtualkeyboard_pinyin.dll",
|
|
||||||
"plugins/virtualkeyboard/qtvirtualkeyboard_tcime.dll",
|
|
||||||
"plugins/virtualkeyboard/qtvirtualkeyboard_thai.dll",
|
|
||||||
"plugins/webview/qtwebview_webengine.dll",
|
|
||||||
"pyside2-lupdate.exe",
|
|
||||||
"Qt3DAnimation.pyd",
|
|
||||||
"Qt3DAnimation.pyi",
|
|
||||||
"Qt3DCore.pyd",
|
|
||||||
"Qt3DCore.pyi",
|
|
||||||
"Qt3DExtras.pyd",
|
|
||||||
"Qt3DExtras.pyi",
|
|
||||||
"Qt3DInput.pyd",
|
|
||||||
"Qt3DInput.pyi",
|
|
||||||
"Qt3DLogic.pyd",
|
|
||||||
"Qt3DLogic.pyi",
|
|
||||||
"Qt3DRender.pyd",
|
|
||||||
"Qt3DRender.pyi",
|
|
||||||
"Qt53DAnimation.dll",
|
|
||||||
"Qt53DCore.dll",
|
|
||||||
"Qt53DExtras.dll",
|
|
||||||
"Qt53DInput.dll",
|
|
||||||
"Qt53DLogic.dll",
|
|
||||||
"Qt53DQuick.dll",
|
|
||||||
"Qt53DQuickAnimation.dll",
|
|
||||||
"Qt53DQuickExtras.dll",
|
|
||||||
"Qt53DQuickInput.dll",
|
|
||||||
"Qt53DQuickRender.dll",
|
|
||||||
"Qt53DQuickScene2D.dll",
|
|
||||||
"Qt53DRender.dll",
|
|
||||||
"Qt5Bluetooth.dll",
|
|
||||||
"Qt5Bodymovin.dll",
|
|
||||||
"Qt5Charts.dll",
|
|
||||||
"Qt5Concurrent.dll",
|
|
||||||
"Qt5DataVisualization.dll",
|
|
||||||
"Qt5DBus.dll",
|
|
||||||
"Qt5Designer.dll",
|
|
||||||
"Qt5DesignerComponents.dll",
|
|
||||||
"Qt5Gamepad.dll",
|
|
||||||
"Qt5Help.dll",
|
|
||||||
"Qt5Location.dll",
|
|
||||||
"Qt5Multimedia.dll",
|
|
||||||
"Qt5MultimediaQuick.dll",
|
|
||||||
"Qt5MultimediaWidgets.dll",
|
|
||||||
"Qt5Nfc.dll",
|
|
||||||
"Qt5OpenGL.dll",
|
|
||||||
"Qt5Pdf.dll",
|
|
||||||
"Qt5PdfWidgets.dll",
|
|
||||||
"Qt5Positioning.dll",
|
|
||||||
"Qt5PositioningQuick.dll",
|
|
||||||
"Qt5PrintSupport.dll",
|
|
||||||
"Qt5Purchasing.dll",
|
|
||||||
"Qt5Quick.dll",
|
|
||||||
"Qt5Quick3D.dll",
|
|
||||||
"Qt5Quick3DAssetImport.dll",
|
|
||||||
"Qt5Quick3DRender.dll",
|
|
||||||
"Qt5Quick3DRuntimeRender.dll",
|
|
||||||
"Qt5Quick3DUtils.dll",
|
|
||||||
"Qt5QuickControls2.dll",
|
|
||||||
"Qt5QuickParticles.dll",
|
|
||||||
"Qt5QuickShapes.dll",
|
|
||||||
"Qt5QuickTemplates2.dll",
|
|
||||||
"Qt5QuickTest.dll",
|
|
||||||
"Qt5QuickWidgets.dll",
|
|
||||||
"Qt5RemoteObjects.dll",
|
|
||||||
"Qt5Script.dll",
|
|
||||||
"Qt5ScriptTools.dll",
|
|
||||||
"Qt5Scxml.dll",
|
|
||||||
"Qt5Sensors.dll",
|
|
||||||
"Qt5SerialBus.dll",
|
|
||||||
"Qt5SerialPort.dll",
|
|
||||||
"Qt5Sql.dll",
|
|
||||||
"Qt5Svg.dll",
|
|
||||||
"Qt5Test.dll",
|
|
||||||
"Qt5TextToSpeech.dll",
|
|
||||||
"Qt5VirtualKeyboard.dll",
|
|
||||||
"Qt5WebChannel.dll",
|
|
||||||
"Qt5WebEngine.dll",
|
|
||||||
"Qt5WebEngineCore.dll",
|
|
||||||
"Qt5WebEngineWidgets.dll",
|
|
||||||
"Qt5WebSockets.dll",
|
|
||||||
"Qt5WebView.dll",
|
|
||||||
"Qt5Xml.dll",
|
|
||||||
"Qt5XmlPatterns.dll",
|
|
||||||
"QtAxContainer.pyd",
|
|
||||||
"QtAxContainer.pyi",
|
|
||||||
"QtCharts.pyd",
|
|
||||||
"QtCharts.pyi",
|
|
||||||
"QtConcurrent.pyd",
|
|
||||||
"QtConcurrent.pyi",
|
|
||||||
"QtDataVisualization.pyd",
|
|
||||||
"QtDataVisualization.pyi",
|
|
||||||
"qtdiag.exe",
|
|
||||||
"QtHelp.pyd",
|
|
||||||
"QtHelp.pyi",
|
|
||||||
"QtLocation.pyd",
|
|
||||||
"QtLocation.pyi",
|
|
||||||
"QtMultimedia.pyd",
|
|
||||||
"QtMultimedia.pyi",
|
|
||||||
"QtMultimediaWidgets.pyd",
|
|
||||||
"QtMultimediaWidgets.pyi",
|
|
||||||
"QtNetwork.pyd",
|
|
||||||
"QtNetwork.pyi",
|
|
||||||
"QtOpenGL.pyd",
|
|
||||||
"QtOpenGL.pyi",
|
|
||||||
"QtOpenGLFunctions.pyd",
|
|
||||||
"QtOpenGLFunctions.pyi",
|
|
||||||
"QtPositioning.pyd",
|
|
||||||
"QtPositioning.pyi",
|
|
||||||
"QtPrintSupport.pyd",
|
|
||||||
"QtPrintSupport.pyi",
|
|
||||||
"QtQml.pyd",
|
|
||||||
"QtQml.pyi",
|
|
||||||
"QtQuick.pyd",
|
|
||||||
"QtQuick.pyi",
|
|
||||||
"QtQuickControls2.pyd",
|
|
||||||
"QtQuickControls2.pyi",
|
|
||||||
"QtQuickWidgets.pyd",
|
|
||||||
"QtQuickWidgets.pyi",
|
|
||||||
"QtRemoteObjects.pyd",
|
|
||||||
"QtRemoteObjects.pyi",
|
|
||||||
"QtScript.pyd",
|
|
||||||
"QtScript.pyi",
|
|
||||||
"QtScriptTools.pyd",
|
|
||||||
"QtScriptTools.pyi",
|
|
||||||
"QtScxml.pyd",
|
|
||||||
"QtScxml.pyi",
|
|
||||||
"QtSensors.pyd",
|
|
||||||
"QtSensors.pyi",
|
|
||||||
"QtSerialPort.pyd",
|
|
||||||
"QtSerialPort.pyi",
|
|
||||||
"QtSql.pyd",
|
|
||||||
"QtSql.pyi",
|
|
||||||
"QtSvg.pyd",
|
|
||||||
"QtSvg.pyi",
|
|
||||||
"QtTest.pyd",
|
|
||||||
"QtTest.pyi",
|
|
||||||
"QtTextToSpeech.pyd",
|
|
||||||
"QtTextToSpeech.pyi",
|
|
||||||
"QtUiTools.pyd",
|
|
||||||
"QtUiTools.pyi",
|
|
||||||
"QtWebChannel.pyd",
|
|
||||||
"QtWebChannel.pyi",
|
|
||||||
"QtWebEngine.pyd",
|
|
||||||
"QtWebEngine.pyi",
|
|
||||||
"QtWebEngineCore.pyd",
|
|
||||||
"QtWebEngineCore.pyi",
|
|
||||||
"QtWebEngineProcess.exe",
|
|
||||||
"QtWebEngineWidgets.pyd",
|
|
||||||
"QtWebEngineWidgets.pyi",
|
|
||||||
"QtWebSockets.pyd",
|
|
||||||
"QtWebSockets.pyi",
|
|
||||||
"QtWinExtras.pyd",
|
|
||||||
"QtWinExtras.pyi",
|
|
||||||
"QtXml.pyd",
|
|
||||||
"QtXml.pyi",
|
|
||||||
"QtXmlPatterns.pyd",
|
|
||||||
"QtXmlPatterns.pyi",
|
|
||||||
"rcc.exe",
|
|
||||||
"uic.exe",
|
|
||||||
]:
|
|
||||||
os.remove(
|
|
||||||
os.path.join(
|
|
||||||
build_path,
|
|
||||||
"lib",
|
|
||||||
"PySide2",
|
|
||||||
filename.replace("/", "\\"),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
after_size = get_size(build_path)
|
|
||||||
freed_bytes = before_size - after_size
|
|
||||||
freed_mb = int(freed_bytes / 1024 / 1024)
|
|
||||||
print(f"> Freed {freed_mb} mb")
|
|
||||||
|
|
||||||
print(f"> Signing onionshare.exe")
|
|
||||||
sign(os.path.join(build_path, "onionshare.exe"), desktop_dir)
|
|
||||||
|
|
||||||
print(f"> Signing onionshare-cli.exe")
|
|
||||||
sign(os.path.join(build_path, "onionshare-cli.exe"), desktop_dir)
|
|
||||||
|
|
||||||
print(f"> Build the WiX file")
|
|
||||||
version_filename = os.path.join(
|
|
||||||
root, "cli", "onionshare_cli", "resources", "version.txt"
|
|
||||||
)
|
|
||||||
with open(version_filename) as f:
|
|
||||||
version = f.read().strip()
|
|
||||||
|
|
||||||
dist_dir = os.path.join(root, build_path)
|
|
||||||
|
|
||||||
data = {
|
|
||||||
"id": "TARGETDIR",
|
|
||||||
"name": "SourceDir",
|
|
||||||
"dirs": [
|
|
||||||
{
|
|
||||||
"id": "ProgramFilesFolder",
|
|
||||||
"dirs": [],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "ProgramMenuFolder",
|
|
||||||
"dirs": [],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
}
|
|
||||||
|
|
||||||
data["dirs"][0]["dirs"].append(
|
|
||||||
wix_build_data(
|
|
||||||
build_path,
|
|
||||||
"INSTALLDIR",
|
|
||||||
"OnionShare",
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
root_el = ET.Element("Wix", xmlns="http://schemas.microsoft.com/wix/2006/wi")
|
|
||||||
product_el = ET.SubElement(
|
|
||||||
root_el,
|
|
||||||
"Product",
|
|
||||||
Name="OnionShare",
|
|
||||||
Manufacturer="Micah Lee, et al.",
|
|
||||||
Id="*",
|
|
||||||
UpgradeCode="$(var.ProductUpgradeCode)",
|
|
||||||
Language="1033",
|
|
||||||
Codepage="1252",
|
|
||||||
Version="$(var.ProductVersion)",
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
product_el,
|
|
||||||
"Package",
|
|
||||||
Id="*",
|
|
||||||
Keywords="Installer",
|
|
||||||
Description="OnionShare $(var.ProductVersion) Installer",
|
|
||||||
Manufacturer="Micah Lee, et al.",
|
|
||||||
InstallerVersion="100",
|
|
||||||
Languages="1033",
|
|
||||||
Compressed="yes",
|
|
||||||
SummaryCodepage="1252",
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
product_el, "Media", Id="1", Cabinet="product.cab", EmbedCab="yes"
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
product_el,
|
|
||||||
"Icon",
|
|
||||||
Id="ProductIcon",
|
|
||||||
SourceFile="..\\onionshare\\resources\\onionshare.ico",
|
|
||||||
)
|
|
||||||
ET.SubElement(product_el, "Property", Id="ARPPRODUCTICON", Value="ProductIcon")
|
|
||||||
ET.SubElement(
|
|
||||||
product_el,
|
|
||||||
"Property",
|
|
||||||
Id="ARPHELPLINK",
|
|
||||||
Value="https://docs.onionshare.org",
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
product_el,
|
|
||||||
"Property",
|
|
||||||
Id="ARPURLINFOABOUT",
|
|
||||||
Value="https://onionshare.org",
|
|
||||||
)
|
|
||||||
ET.SubElement(product_el, "UIRef", Id="WixUI_Minimal")
|
|
||||||
ET.SubElement(product_el, "UIRef", Id="WixUI_ErrorProgressText")
|
|
||||||
ET.SubElement(
|
|
||||||
product_el,
|
|
||||||
"WixVariable",
|
|
||||||
Id="WixUILicenseRtf",
|
|
||||||
Value="..\\package\\license.rtf",
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
product_el,
|
|
||||||
"WixVariable",
|
|
||||||
Id="WixUIDialogBmp",
|
|
||||||
Value="..\\package\\dialog.bmp",
|
|
||||||
)
|
|
||||||
ET.SubElement(
|
|
||||||
product_el,
|
|
||||||
"MajorUpgrade",
|
|
||||||
AllowSameVersionUpgrades="yes",
|
|
||||||
DowngradeErrorMessage="A newer version of [ProductName] is already installed. If you are sure you want to downgrade, remove the existing installation via Programs and Features.",
|
|
||||||
)
|
|
||||||
|
|
||||||
wix_build_dir_xml(product_el, data)
|
|
||||||
component_ids = wix_build_components_xml(product_el, data)
|
|
||||||
|
|
||||||
feature_el = ET.SubElement(
|
|
||||||
product_el, "Feature", Id="DefaultFeature", Level="1"
|
|
||||||
)
|
|
||||||
for component_id in component_ids:
|
|
||||||
ET.SubElement(feature_el, "ComponentRef", Id=component_id)
|
|
||||||
ET.SubElement(feature_el, "ComponentRef", Id="ApplicationShortcuts")
|
|
||||||
|
|
||||||
with open(os.path.join(root, "desktop", "build", "OnionShare.wxs"), "w") as f:
|
|
||||||
f.write('<?xml version="1.0" encoding="windows-1252"?>\n')
|
|
||||||
f.write(f'<?define ProductVersion = "{version}"?>\n')
|
|
||||||
f.write(
|
|
||||||
'<?define ProductUpgradeCode = "12b9695c-965b-4be0-bc33-21274e809576"?>\n'
|
|
||||||
)
|
|
||||||
|
|
||||||
ET.indent(root_el)
|
|
||||||
f.write(ET.tostring(root_el).decode())
|
|
||||||
|
|
||||||
print(f"> Build the MSI")
|
|
||||||
run(
|
|
||||||
[shutil.which("candle.exe"), "OnionShare.wxs"],
|
|
||||||
os.path.join(desktop_dir, "build"),
|
|
||||||
)
|
|
||||||
run(
|
|
||||||
[shutil.which("light.exe"), "-ext", "WixUIExtension", "OnionShare.wixobj"],
|
|
||||||
os.path.join(desktop_dir, "build"),
|
|
||||||
)
|
|
||||||
|
|
||||||
print(f"> Prepare OnionShare.msi for signing")
|
|
||||||
run(
|
|
||||||
[
|
|
||||||
shutil.which("insignia.exe"),
|
|
||||||
"-im",
|
|
||||||
os.path.join(desktop_dir, "build", "OnionShare.msi"),
|
|
||||||
],
|
|
||||||
error_ok=True,
|
|
||||||
)
|
|
||||||
sign(os.path.join(desktop_dir, "build", "OnionShare.msi"))
|
|
||||||
|
|
||||||
final_msi_filename = os.path.join(
|
|
||||||
desktop_dir, "dist", f"OnionShare-{version}.msi"
|
|
||||||
)
|
|
||||||
print(f"> Final MSI: {final_msi_filename}")
|
|
||||||
os.makedirs(os.path.join(desktop_dir, "dist"), exist_ok=True)
|
|
||||||
os.rename(
|
|
||||||
os.path.join(desktop_dir, "build", "OnionShare.msi"),
|
|
||||||
final_msi_filename,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
@ -120,7 +120,6 @@ def cleanup_build():
|
|||||||
"QtQuickTest",
|
"QtQuickTest",
|
||||||
"QtQuickWidgets",
|
"QtQuickWidgets",
|
||||||
"QtRemoteObjects",
|
"QtRemoteObjects",
|
||||||
"QtRepParser",
|
|
||||||
"QtScript",
|
"QtScript",
|
||||||
"QtScriptTools",
|
"QtScriptTools",
|
||||||
"QtScxml",
|
"QtScxml",
|
||||||
@ -131,11 +130,9 @@ def cleanup_build():
|
|||||||
"QtSvg",
|
"QtSvg",
|
||||||
"QtTest",
|
"QtTest",
|
||||||
"QtTextToSpeech",
|
"QtTextToSpeech",
|
||||||
"QtUiPlugin",
|
|
||||||
"QtVirtualKeyboard",
|
"QtVirtualKeyboard",
|
||||||
"QtWebChannel",
|
"QtWebChannel",
|
||||||
"QtWebEngine",
|
"QtWebEngine",
|
||||||
"QtWebEngineCore",
|
|
||||||
"QtWebEngineWidgets",
|
"QtWebEngineWidgets",
|
||||||
"QtWebSockets",
|
"QtWebSockets",
|
||||||
"QtWebView",
|
"QtWebView",
|
||||||
|
Loading…
Reference in New Issue
Block a user