Files
aqtinstall/aqt/updater.py
Hiroshi Miura 13a83b9f14 PEP8
Signed-off-by: Hiroshi Miura <miurahr@linux.com>
2021-05-02 12:03:44 +09:00

239 lines
9.4 KiB
Python

#!/usr/bin/env python
#
# Copyright (C) 2019-2021 Hiroshi Miura <miurahr@linux.com>
#
# Permission is hereby granted, free of charge, to any person obtaining a copy of
# this software and associated documentation files (the "Software"), to deal in
# the Software without restriction, including without limitation the rights to
# use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
# the Software, and to permit persons to whom the Software is furnished to do so,
# subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
# FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
# COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
# IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
import os
import pathlib
import subprocess
class Updater:
def __init__(self, prefix: pathlib.Path, logger):
self.logger = logger
self.prefix = prefix
self.qmake_path = None
self.qconfigs = {}
def _patch_binfile(self, file: pathlib.Path, key: bytes, newpath: bytes):
"""Patch binary file with key/value"""
st = file.stat()
data = file.read_bytes()
idx = data.find(key)
if idx > 0:
return
assert len(newpath) < 256, "Qt Prefix path is too long(255)."
data = data[:idx] + key + newpath + data[idx + len(newpath) :]
file.write_bytes(data)
os.chmod(str(file), st.st_mode)
def _append_string(self, file: pathlib.Path, val: str):
"""Append string to file"""
st = file.stat()
data = file.read_text("UTF-8")
data += val
file.write_text(data, "UTF-8")
os.chmod(str(file), st.st_mode)
def _patch_textfile(self, file: pathlib.Path, old: str, new: str):
st = file.stat()
data = file.read_text("UTF-8")
data = data.replace(old, new)
file.write_text(data, "UTF-8")
os.chmod(str(file), st.st_mode)
def _detect_qmake(self) -> bool:
"""detect Qt configurations from qmake."""
for qmake_path in [
self.prefix.joinpath("bin", "qmake"),
self.prefix.joinpath("bin", "qmake.exe"),
]:
if not qmake_path.exists():
continue
try:
result = subprocess.run(
[str(qmake_path), "-query"], stdout=subprocess.PIPE
)
except subprocess.SubprocessError:
return False
if result.returncode == 0:
self.qmake_path = qmake_path
for line in result.stdout.splitlines():
vals = line.decode("UTF-8").split(":")
self.qconfigs[vals[0]] = vals[1]
return True
return False
def patch_pkgconfig(self):
for pcfile in self.prefix.joinpath("lib", "pkgconfig").glob("*.pc"):
self.logger.info("Patching {}".format(pcfile))
self._patch_textfile(
pcfile,
"prefix=/home/qt/work/install",
"prefix={}".format(str(self.prefix)),
)
def patch_qmake(self):
"""Patch to qmake binary"""
if self._detect_qmake():
self.logger.info("Patching {}".format(str(self.qmake_path)))
self._patch_binfile(
self.qmake_path,
key=b"qt_prfxpath=",
newpath=bytes(str(self.prefix), "UTF-8"),
)
def patch_qmake_script(self, base_dir, qt_version, os_name):
if os_name == "linux":
self.logger.info("Patching {}/bin/qmake".format(self.prefix))
self._patch_textfile(
self.prefix / "bin" / "qmake",
"/home/qt/work/install/bin",
"{}/{}/{}/bin".format(base_dir, qt_version, "gcc_64"),
)
elif os_name == "mac":
self.logger.info("Patching {}/bin/qmake".format(self.prefix))
self._patch_textfile(
self.prefix / "bin" / "qmake",
"/Users/qt/work/install/bin",
"{}/{}/{}/bin".format(base_dir, qt_version, "clang_64"),
)
elif os_name == "windows":
self.logger.info("Patching {}/bin/qmake.bat".format(self.prefix))
self._patch_textfile(
self.prefix / "bin" / "qmake.bat",
"/Users/qt/work/install/bin",
"{}\\{}\\{}\\bin".format(base_dir, qt_version, "mingw81_64"),
)
else:
pass
def patch_qtcore(self, target):
"""patch to QtCore"""
if target.os_name == "mac":
lib_dir = self.prefix.joinpath("lib", "QtCore.framework")
components = ["QtCore", "QtCore_debug"]
elif target.os_name == "linux":
lib_dir = self.prefix.joinpath("lib")
components = ["libQt5Core.so"]
elif target.os_name == "windows":
lib_dir = self.prefix.joinpath("bin")
components = ["Qt5Cored.dll", "Qt5Core.dll"]
else:
return
for component in components:
if lib_dir.joinpath(component).exists():
qtcore_path = lib_dir.joinpath(component).resolve()
self.logger.info("Patching {}".format(qtcore_path))
newpath = bytes(str(self.prefix), "UTF-8")
self._patch_binfile(qtcore_path, b"qt_prfxpath=", newpath)
def make_qtconf(self, base_dir, qt_version, arch_dir):
"""Prepare qt.conf"""
with open(
os.path.join(base_dir, qt_version, arch_dir, "bin", "qt.conf"), "w"
) as f:
f.write("[Paths]\n")
f.write("Prefix=..\n")
def set_license(self, base_dir, qt_version, arch_dir):
"""Update qtconfig.pri as OpenSource"""
with open(
os.path.join(base_dir, qt_version, arch_dir, "mkspecs", "qconfig.pri"), "r+"
) as f:
lines = f.readlines()
f.seek(0)
f.truncate()
for line in lines:
if line.startswith("QT_EDITION ="):
line = "QT_EDITION = OpenSource\n"
if line.startswith("QT_LICHECK ="):
line = "QT_LICHECK =\n"
f.write(line)
def patch_target_qt_conf(self, base_dir, qt_version, arch_dir, os_name):
target_qt_conf = self.prefix / "bin" / "target_qt.conf"
if os_name == "linux":
old_targetprefix = "Prefix=/home/qt/work/install/target"
new_hostprefix = "HostPrefix=../../gcc_64"
elif os_name == "mac":
old_targetprefix = "Prefix=/Users/qt/work/install/target"
new_hostprefix = "HostPrefix=../../clang_64"
else:
old_targetprefix = "Prefix=/Users/qt/work/install/target"
new_hostprefix = "HostPrefix=../../mingw81_64"
new_targetprefix = "Prefix={}".format(
str(pathlib.Path(base_dir).joinpath(qt_version, arch_dir, "target"))
)
new_hostdata = "HostData=../{}".format(arch_dir)
self._patch_textfile(target_qt_conf, old_targetprefix, new_targetprefix)
self._patch_textfile(target_qt_conf, "HostPrefix=../../", new_hostprefix)
self._patch_textfile(target_qt_conf, "HostData=target", new_hostdata)
@classmethod
def update(cls, target, base_dir: str, logger):
"""
Make Qt configuration files, qt.conf and qtconfig.pri.
And update pkgconfig and patch Qt5Core and qmake
"""
qt_version = target.version
arch = target.arch
if arch is None:
arch_dir = ""
elif arch.startswith("win64_mingw"):
arch_dir = arch[6:] + "_64"
elif arch.startswith("win32_mingw"):
arch_dir = arch[6:] + "_32"
elif arch.startswith("win"):
arch_dir = arch[6:]
else:
arch_dir = arch
try:
prefix = pathlib.Path(base_dir) / target.version / arch_dir
updater = Updater(prefix, logger)
updater.set_license(base_dir, qt_version, arch_dir)
if target.arch not in [
"ios",
"android",
"wasm_32",
"android_x86_64",
"android_arm64_v8a",
"android_x86",
"android_armv7",
]: # desktop version
updater.make_qtconf(base_dir, qt_version, arch_dir)
updater.patch_qmake()
if target.os_name == "linux":
updater.patch_pkgconfig()
if versiontuple(target.version) < (5, 14, 0):
updater.patch_qtcore(target)
elif qt_version.startswith("5."): # qt5 non-desktop
updater.patch_qmake()
else: # qt6 non-desktop
updater.patch_qmake_script(base_dir, qt_version, target.os_name)
updater.patch_target_qt_conf(
base_dir, qt_version, arch_dir, target.os_name
)
except IOError as e:
raise e
def versiontuple(v: str):
return tuple(map(int, (v.split("."))))