* Brighter red * Move openssl to vcpkg.exe * Revert "Move openssl to vcpkg.exe" This reverts commit 36f39d916f3cc2c8ce779442bd964bd6af7edd69. * Add missing copyright * Fix copyright dates * Remove openssl from choco * Install openssl using vcpkg * Add deps for vcpkg * Add missing vcpkg manifest * Revert "Add deps for vcpkg" This reverts commit c266d29c2cec4474a181b89c2f759b62ed67db10. * Make vcpkg Windows only * Improve comment about vcpkg * Remove unused var * Add caching for vcpkg * Reorg launch.json * Remove static env var for openssl on Windows * Add openssl dep to vcpkg * Update ChangeLog * Add OpenSSL include dir * Remove pointless choco cache * Remove vcpkg downloads * Remove wixtoolset for CI already installed on GitHub runners * Use `Remove-Item` instead of `rmdir` * Move cmake and ninja out of choco * Revert "Move cmake and ninja out of choco (winget not supported on GitHub runner)" This reverts commit a65c02d275e58705b8cd86fac72629284191d737. * Move cmake and ninja to winget and don't use choco on CI * Remove winget action * Use scoop on CI and winget locally * Use `seanmiddleditch/gha-setup-ninja` for Ninja * Improve comments * Install Ninja before deps * Use system vcpkg if installed * Revert "Use system vcpkg if installed" This reverts commit 4ddee1c66e8cace458c047285a70d2f98cf9d82c. * Add comment about why we're using local vcpkg * Fixed comment about VC++ * Improve config comments * Delete dead code * Improve comment about vcpkg/buildtrees
213 lines
6.5 KiB
Python
213 lines
6.5 KiB
Python
#!/usr/bin/env python3
|
|
|
|
# Synergy -- mouse and keyboard sharing utility
|
|
# Copyright (C) 2024 Symless Ltd.
|
|
#
|
|
# This package is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License
|
|
# found in the file LICENSE that should have accompanied this file.
|
|
#
|
|
# This package is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
import lib.env as env
|
|
|
|
env.ensure_in_venv(__file__)
|
|
|
|
import os, sys, time, subprocess, argparse
|
|
import lib.windows as windows
|
|
import psutil # type: ignore
|
|
import lib.colors as colors
|
|
import lib.file_utils as file_utils
|
|
|
|
DEFAULT_BIN_NAME = "synergyd"
|
|
DEFAULT_SOURCE_DIR = os.path.join("build", "temp", "bin")
|
|
DEFAULT_TARGET_DIR = os.path.join("build", "bin")
|
|
SERVICE_NOT_RUNNING_ERROR = 2
|
|
ERROR_ACCESS_VIOLATION = 0xC0000005
|
|
IGNORE_PROCESSES = ["synergy.exe"]
|
|
|
|
|
|
class Context:
|
|
def __init__(self, verbose):
|
|
self.verbose = verbose
|
|
|
|
|
|
def main():
|
|
"""Entry point for the script."""
|
|
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("--reinstall", action="store_true")
|
|
parser.add_argument("--stop", action="store_true")
|
|
parser.add_argument("--restart", action="store_true")
|
|
parser.add_argument("--pause-on-exit", action="store_true")
|
|
parser.add_argument("--source-dir", default=DEFAULT_SOURCE_DIR)
|
|
parser.add_argument("--target-dir", default=DEFAULT_TARGET_DIR)
|
|
parser.add_argument("--bin-name", default=DEFAULT_BIN_NAME)
|
|
parser.add_argument("--verbose", action="store_true")
|
|
args = parser.parse_args()
|
|
|
|
if not env.is_windows():
|
|
print(
|
|
f"{colors.ERROR_TEXT} This script is only supported on Windows",
|
|
file=sys.stderr,
|
|
)
|
|
sys.exit(1)
|
|
|
|
context = Context(args.verbose)
|
|
|
|
try:
|
|
if args.reinstall:
|
|
reinstall(context, args.source_dir, args.target_dir, args.bin_name)
|
|
elif args.stop:
|
|
stop(context, args.target_dir)
|
|
elif args.restart:
|
|
restart(context, args.source_dir, args.target_dir)
|
|
else:
|
|
print("No action specified", file=sys.stderr)
|
|
exit(1)
|
|
except Exception as e:
|
|
print(f"{colors.ERROR_TEXT} {e}", file=sys.stderr)
|
|
|
|
if args.pause_on_exit:
|
|
input("Press enter to continue...")
|
|
|
|
|
|
def print_verbose(context, message):
|
|
if context.verbose:
|
|
print(message)
|
|
|
|
|
|
def ensure_admin():
|
|
if not windows.is_admin():
|
|
windows.run_elevated(__file__)
|
|
sys.exit()
|
|
|
|
|
|
def restart(context, source_dir, target_dir):
|
|
"""Stops the daemon service, copies files, and restarts the daemon service."""
|
|
|
|
ensure_admin()
|
|
stop(context, target_dir)
|
|
copy_files(source_dir, target_dir)
|
|
start()
|
|
|
|
|
|
def reinstall(context, source_dir, target_dir, bin_name):
|
|
"""Stops and uninstalls daemon service, copies files, and reinstalls the daemon service."""
|
|
|
|
ensure_admin()
|
|
stop(context, target_dir)
|
|
|
|
source_bin_path = f"{os.path.join(source_dir, bin_name)}.exe"
|
|
|
|
copy_files(source_dir, target_dir)
|
|
|
|
print("Removing old daemon service")
|
|
try:
|
|
subprocess.run([source_bin_path, "/uninstall"], shell=True, check=True)
|
|
except subprocess.CalledProcessError as e:
|
|
check_access_violation(e.returncode, source_bin_path)
|
|
if e.returncode != 0:
|
|
print(
|
|
f"{colors.WARNING_TEXT} Uninstall failed, return code: {e.returncode}",
|
|
file=sys.stderr,
|
|
)
|
|
|
|
target_bin_path = os.path.join(target_dir, bin_name + ".exe")
|
|
|
|
try:
|
|
print("Installing daemon service")
|
|
subprocess.run([target_bin_path, "/install"], shell=True, check=True)
|
|
except subprocess.CalledProcessError as e:
|
|
check_access_violation(e.returncode, target_bin_path)
|
|
if e.returncode != 0:
|
|
print(f"{colors.WARNING_TEXT} Install failed, return code: {e.returncode}")
|
|
|
|
|
|
def copy_files(source_dir, target_dir):
|
|
options = file_utils.CopyOptions(ignore_errors=True, verbose=False)
|
|
print(f"Copying files from {source_dir} to {target_dir}")
|
|
file_utils.copy(f"{source_dir}/*", target_dir, options)
|
|
|
|
|
|
def stop(context, target_dir):
|
|
ensure_admin()
|
|
print("Stopping daemon service")
|
|
try:
|
|
subprocess.run(["net", "stop", "synergy"], shell=True, check=True)
|
|
except subprocess.CalledProcessError as e:
|
|
if e.returncode == SERVICE_NOT_RUNNING_ERROR:
|
|
print_verbose(context, "Daemon service not running")
|
|
else:
|
|
raise e
|
|
|
|
# Wait for Windows to release the file handles after process termination.
|
|
wait_for_stop(context, target_dir)
|
|
|
|
|
|
def start():
|
|
ensure_admin()
|
|
print("Starting daemon service")
|
|
subprocess.run(["net", "start", "synergy"], shell=True, check=True)
|
|
|
|
|
|
def wait_for_stop(context, target_dir):
|
|
if is_any_process_running(context, target_dir):
|
|
print("Waiting for file handles to release...", end="", flush=True)
|
|
while is_any_process_running(context, target_dir):
|
|
if not context.verbose:
|
|
print(".", end="", flush=True)
|
|
time.sleep(1)
|
|
if not context.verbose:
|
|
print()
|
|
|
|
|
|
def check_access_violation(return_code, bin_path):
|
|
if return_code == ERROR_ACCESS_VIOLATION:
|
|
print(
|
|
f"{colors.WARNING_TEXT} Process crashed with memory access violation: {bin_path}",
|
|
file=sys.stderr,
|
|
)
|
|
|
|
|
|
def is_ignored_process(exe):
|
|
for ignore_process in IGNORE_PROCESSES:
|
|
if exe.endswith(ignore_process):
|
|
return True
|
|
return False
|
|
|
|
|
|
def is_any_process_running(context, dir):
|
|
"""Check if there is any running process that contains the given directory."""
|
|
|
|
print_verbose(context, f"Checking if any process is running in: {dir}")
|
|
|
|
for proc in psutil.process_iter(attrs=["name", "exe"]):
|
|
exe = proc.info["exe"]
|
|
|
|
if not exe:
|
|
print_verbose(context, f"Skipping process with no exe: {proc}")
|
|
continue
|
|
|
|
if is_ignored_process(exe):
|
|
print_verbose(context, f"Ignoring process: {exe}")
|
|
continue
|
|
|
|
try:
|
|
if dir.lower() in exe.lower():
|
|
print_verbose(context, f"Process found: {exe}")
|
|
return True
|
|
except (psutil.NoSuchProcess, psutil.AccessDenied):
|
|
pass
|
|
|
|
return False
|
|
|
|
|
|
main()
|