mirror of
https://github.com/deskflow/deskflow.git
synced 2026-07-01 21:02:39 +08:00
* Reorganize CMake Packaging module
* Match if statements to function order
* Cleanup root, res, and doc dirs
* Move deps to requirements.txt file
* Reorganize and format CMake files
* Rename changelog lint
* Add reccomended extension
* Workflow to lint CMake files
* Move CMake lint to script
* Try lighter dep
* Use venv
* Add --format arg
* Format all CMake files
* Convert bash script to Python
* Set CMake line ending format
* Restore formatting
* Add pyyaml dep
* Remove unused arg
* Rename config file
* Remove comment
* Repair copyrights (broken by defualt cmake-format)
* Restore 3rd party copyright
* Break up libs config into smaller macros
* Better macro name
* Load config after venv
* Make intentional noop clearer
* Only use upload step if required (make skip clearer)
* Use CPack for deb and rpm packaging
* Add upload step for Linux
* Remove cpack dep, doesn't exist
* Roll back presets version
* Fixed distro like match
* Update ChangeLog
* Legacy checkout for some distros
* All distros support v4
* Trying out newer Linux distros
* Install Git on Docker images
* Install without actions (not available before checkout)
* Delete useless action
* Install Python
* Support for Arch and OpenSUSE
* Add Arch and OpenSUSE to deps
* Name steps
* Full OpenSUSE names
* Mark Git dir safe
* Add pkgconf
* Legacy CMake for Debian 11
* Add OpenSSL to OpenSUSE
* Drop OpenSUSE Leap (no C++20 support)
* Skip packaging for Arch and OpenSUSE (for now)
* Shorten Arch/OpenSUSE names
* Clearer step name
* SImpler bootstrap
* Shell not needed
* Update apt
* Don't check return code
* Simplify python deps commands
* Add STGZ/.sh package type
* Prevent input prompt
* Only config git safe dir when needed
* Try cache v4
* Safe dir for Ubuntu
* Safe dir for Arch
* All Docker images seem to need safe dir config
* Refactor env var getters
* Make Ubuntu build extra packages
* Condense bootstep to single step
* Fixed var name
* Fixed bootstrap logic
* Simplify logic for upload condition (Windows and macOS)
* Make package/upload condition easier to understand
* Add Manjaro
* Generic names for Linux .tar.gz and .sh packages
* Add Manjaro deps
* Swap macOS matrix entries
* Add Red Hat UBI
* Remove RHEL subscription manager
* Throw on unsupported package distro
* Conditionally install pip and venv
* Remove extra pip arg
* Add config for RHEL
* Install EPEL for RHEL
* Back-out RHEL as EPEL requires subscription
* Restore Python deps logic
* Fixed bug: Packacking run twice
* Testing arm32v7 and arm64v8
* Revert "Testing arm32v7 and arm64v8"
This reverts commit cb3caf188d.
* Re-add icon and shortcut file for Linux to package
* Support OpenSUSE RPM build
* Check return code
* Add `rpm-build` for OpenSUSE
* Reorg packages
* Remove busybox-which
* Add --non-interactive
* Move --non-interactive to correct position
* Experiment with makepkg
* Check and print package commands
* Make distro version optional
* Use 8 cores to build
* Default to distro name only
* Fixed bad PKGBUILD filename
* Use 4-part version for Arch
* Remove comma from conflicts
* Use .tar.gz from cwd
* Generate checksum for Arch
* Fixed file extension
* Use shell to print output
* Don't use shell
* Gaurd against bad cmd_utils.run
* Fixed bad import
* Use list command
* Fixed unable to run list commands
* Use source file name
* Simplify PKGBUILD to use make install
* Change install prefix
* Use DESTDIR
* Copy .desktop and .png to build dir
* Restore original `install(FILES...`
* Improving comments
* Fixed: makepkg runs from `src` by default
* Move error after command print
* Remove shell arg
* Package as a user instead of root (makepkg can't run as root)
* Fixed codesign runs in shell
* Allow list commands in shell on windows
* Don't use sudo on arch
* Install sudo on Arch
* Fixed typo
* Fix ownership of build directory for package user
* Improve example .env
* Change to depend on libstdc++6
* Add TODO
* Fixed Fedora version
* Remove libstdc++ deps (names vary between distros)
* Roll back to Fedora 40 and 39
* Improve comment
* Remove unneccesary default
231 lines
6.7 KiB
Python
231 lines
6.7 KiB
Python
import ctypes, sys, os, shutil
|
|
import xml.etree.ElementTree as ET
|
|
import lib.cmd_utils as cmd_utils
|
|
import lib.env as env
|
|
from lib.certificate import Certificate
|
|
|
|
msbuild_cmd = "msbuild"
|
|
signtool_cmd = "signtool"
|
|
certutil_cmd = "certutil"
|
|
cmake_env_var = "CMAKE_PREFIX_PATH"
|
|
runner_temp_env_var = "RUNNER_TEMP"
|
|
qt_base_dir_env_var = "QT_BASE_DIR"
|
|
dist_dir = "dist"
|
|
build_dir = "build"
|
|
wix_solution_file = f"{build_dir}/installer/Synergy.sln"
|
|
installer_file = f"{build_dir}/installer/bin/Release/Synergy.msi"
|
|
|
|
|
|
def relaunch_as_admin(script):
|
|
args = " ".join(sys.argv[1:])
|
|
command = f"{script} --pause-on-exit {args}"
|
|
print(f"Re-launching script as admin: {command}")
|
|
ctypes.windll.shell32.ShellExecuteW(None, "runas", sys.executable, command, None, 1)
|
|
|
|
|
|
def is_admin():
|
|
"""Returns True if the current process has admin privileges."""
|
|
try:
|
|
return ctypes.windll.shell32.IsUserAnAdmin()
|
|
except ctypes.WinError:
|
|
return False
|
|
|
|
|
|
def set_env_var(name, value):
|
|
"""
|
|
Sets or updates an environment variable. Appends the value if it doesn't already exist.
|
|
|
|
Args:
|
|
name (str): The name of the environment variable.
|
|
value (str): The value of the environment variable.
|
|
"""
|
|
|
|
current_value = os.getenv(name, "")
|
|
|
|
if value not in current_value:
|
|
new_value = f"{current_value}{os.pathsep}{value}" if current_value else value
|
|
os.environ[name] = new_value
|
|
print(f"Setting environment variable: {name}={value}")
|
|
cmd_utils.run(["setx", name, new_value], check=True, shell=True, print_cmd=True)
|
|
|
|
|
|
def package(filename_base):
|
|
cert_base64 = env.get_env("WINDOWS_PFX_CERTIFICATE")
|
|
cert_password = env.get_env("WINDOWS_PFX_PASSWORD")
|
|
|
|
sign_binaries(cert_base64, cert_password)
|
|
build_msi(filename_base)
|
|
sign_msi(filename_base, cert_base64, cert_password)
|
|
|
|
|
|
def assert_vs_cmd(cmd):
|
|
has_cmd = cmd_utils.has_command(cmd)
|
|
if not has_cmd:
|
|
raise RuntimeError(
|
|
f"The '{cmd}' command was not found, "
|
|
"re-run from 'Developer Command Prompt for VS'"
|
|
)
|
|
|
|
|
|
def build_msi(filename_base):
|
|
print("Building MSI installer...")
|
|
configuration = "Release"
|
|
platform = "x64"
|
|
|
|
assert_vs_cmd(msbuild_cmd)
|
|
cmd_utils.run(
|
|
[
|
|
msbuild_cmd,
|
|
wix_solution_file,
|
|
f"/p:Configuration={configuration}",
|
|
f"/p:Platform={platform}",
|
|
],
|
|
shell=True,
|
|
print_cmd=True,
|
|
)
|
|
|
|
path = get_package_path(filename_base)
|
|
print(f"Copying MSI installer to {dist_dir}")
|
|
os.makedirs(dist_dir, exist_ok=True)
|
|
shutil.copy(installer_file, path)
|
|
|
|
|
|
def get_package_path(filename_base):
|
|
return f"{dist_dir}/{filename_base}.msi"
|
|
|
|
|
|
def sign_binaries(cert_base64, cert_password):
|
|
exe_pattern = f"{build_dir}/bin/*.exe"
|
|
run_codesign(exe_pattern, cert_base64, cert_password)
|
|
|
|
|
|
def sign_msi(filename_base, cert_base64, cert_password):
|
|
path = get_package_path(filename_base)
|
|
run_codesign(path, cert_base64, cert_password)
|
|
|
|
|
|
def run_codesign(path, cert_base64, cert_password):
|
|
time_server = "http://timestamp.digicert.com"
|
|
hashing_algorithm = "SHA256"
|
|
|
|
with Certificate(cert_base64, "pfx") as cert_path:
|
|
print("Signing MSI installer...")
|
|
assert_vs_cmd(signtool_cmd)
|
|
|
|
# WARNING: contains private key password, never print this command
|
|
cmd_utils.run(
|
|
[
|
|
signtool_cmd,
|
|
"sign",
|
|
"/f",
|
|
cert_path,
|
|
"/p",
|
|
cert_password,
|
|
"/t",
|
|
time_server,
|
|
"/fd",
|
|
hashing_algorithm,
|
|
path,
|
|
]
|
|
)
|
|
|
|
|
|
class WindowsChoco:
|
|
"""Chocolatey for Windows."""
|
|
|
|
def install(self, command, ci_env):
|
|
"""Installs packages using Chocolatey."""
|
|
if ci_env:
|
|
# don't show noisy choco progress bars in ci env
|
|
cmd_utils.run(
|
|
f"{command} --no-progress",
|
|
shell=True,
|
|
print_cmd=True,
|
|
)
|
|
else:
|
|
cmd_utils.run(
|
|
"winget install chocolatey",
|
|
check=False,
|
|
shell=True,
|
|
print_cmd=True,
|
|
)
|
|
cmd_utils.run(
|
|
command,
|
|
shell=True,
|
|
print_cmd=True,
|
|
)
|
|
|
|
def config_ci_cache(self):
|
|
"""Configures Chocolatey cache for CI."""
|
|
|
|
runner_temp = os.environ.get(runner_temp_env_var)
|
|
if runner_temp:
|
|
# sets the choco cache dir, which should match the dir in the ci cache action.
|
|
key_arg = '--name="cacheLocation"'
|
|
value_arg = f'--value="{runner_temp}/choco"'
|
|
cmd_utils.run(
|
|
["choco", "config", "set", key_arg, value_arg],
|
|
print_cmd=True,
|
|
)
|
|
else:
|
|
print(f"Warning: CI environment variable {runner_temp_env_var} not set")
|
|
|
|
def remove_from_config(self, choco_config_file, remove_packages):
|
|
"""Removes a package from the Chocolatey configuration."""
|
|
|
|
tree = ET.parse(choco_config_file)
|
|
root = tree.getroot()
|
|
for remove in remove_packages:
|
|
for package in root.findall("package"):
|
|
if package.get("id") == remove:
|
|
root.remove(package)
|
|
print(f"Removed package from choco config: {remove}")
|
|
|
|
tree.write(choco_config_file)
|
|
|
|
|
|
class WindowsQt:
|
|
"""Qt for Windows."""
|
|
|
|
def __init__(self, mirror_url, version, base_dir):
|
|
self.mirror_url = mirror_url
|
|
self.version = version
|
|
|
|
# allows ci to override the qt base dir path
|
|
self.base_dir = os.environ.get(qt_base_dir_env_var)
|
|
if not self.base_dir:
|
|
print(f"QT_BASE_DIR not set, using: {base_dir}")
|
|
self.base_dir = base_dir
|
|
|
|
self.install_dir = f"{self.base_dir}\\{self.version}"
|
|
|
|
def get_install_dir(self):
|
|
if os.path.isdir(self.install_dir):
|
|
return self.install_dir
|
|
|
|
def install(self):
|
|
"""Installs Qt on Windows."""
|
|
|
|
cmd_utils.run(
|
|
["pip", "install", "aqtinstall"],
|
|
shell=True,
|
|
print_cmd=True,
|
|
)
|
|
|
|
args = ["python", "-m", "aqt", "install-qt"]
|
|
args.extend(["--outputdir", self.base_dir])
|
|
args.extend(["--base", self.mirror_url])
|
|
args.extend(["windows", "desktop", self.version, "win64_msvc2019_64"])
|
|
cmd_utils.run(
|
|
args,
|
|
shell=True,
|
|
print_cmd=True,
|
|
)
|
|
|
|
install_dir = self.get_install_dir()
|
|
if not install_dir:
|
|
raise RuntimeError(f"Qt not installed, path not found: {install_dir}")
|
|
|
|
def set_env_vars(self):
|
|
set_env_var(cmake_env_var, f"{self.get_install_dir()}\\msvc2019_64")
|