From b653bcbb80497a8fa1c7694f8039d8712c3f70ff Mon Sep 17 00:00:00 2001
From: gfdgd_xi <3025613752@qq.com>
Date: Sat, 26 Nov 2022 22:08:06 +0800
Subject: [PATCH] =?UTF-8?q?=E7=AE=80=E6=98=93=E6=89=93=E5=8C=85=E5=99=A8?=
 =?UTF-8?q?=E9=9B=8F=E5=BD=A2?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

---
 deepin-wine-easy-packager.py | 451 +++++++++++++++++++++++++++++++++++
 1 file changed, 451 insertions(+)
 create mode 100644 deepin-wine-easy-packager.py

diff --git a/deepin-wine-easy-packager.py b/deepin-wine-easy-packager.py
new file mode 100644
index 0000000..b337710
--- /dev/null
+++ b/deepin-wine-easy-packager.py
@@ -0,0 +1,451 @@
+import os
+import sys
+import json
+import random
+import xpinyin
+import traceback
+import subprocess
+import PyQt5.QtGui as QtGui
+import PyQt5.QtCore as QtCore
+import PyQt5.QtWidgets as QtWidgets
+
+def ShowText(text: str):
+    if text.replace(" ", "").replace("\n", "") == "":
+        return
+    logText.append(text)
+    
+def ErrorMessage(text: str):
+    QtWidgets.QMessageBox.critical(window, "错误", text)
+
+def InformationMessage(text: str):
+    QtWidgets.QMessageBox.information(window, "提示", text)
+
+def QuestionMessage(text: str):
+    if QtWidgets.QMessageBox.question(window, "提示", text) == QtWidgets.QMessageBox.Yes:
+        return True
+    return False
+
+def DisbledAndEnabledAll(choose: bool):
+    pass
+
+# 获取用户主目录
+def get_home():
+    return os.path.expanduser('~')
+
+def get_desktop_path():
+    try:
+        for line in open(get_home() + "/.config/user-dirs.dirs"):  # 以行来读取配置文件
+            desktop_index = line.find("XDG_DESKTOP_DIR=\"")  # 寻找是否有对应项,有返回 0,没有返回 -1
+            if desktop_index != -1:  # 如果有对应项
+                break  # 结束循环
+        if desktop_index == -1:  # 如果是提前结束,值一定≠-1,如果是没有提前结束,值一定=-1
+            return -1
+        else:
+            get = line[17:-2]  # 截取桌面目录路径
+            get_index = get.find("$HOME")  # 寻找是否有对应的项,需要替换内容
+            if get != -1:  # 如果有
+                get = get.replace("$HOME", get_home())  # 则把其替换为用户目录(~)
+            return get  # 返回目录
+    except:
+        traceback.print_exc()
+        return get_home()
+
+# 读取 lnk 文件
+def GetLnkDesktop(path):
+    lnkList = []
+    for i in os.listdir(path):
+        filePath = f"{path}/{i}"
+        if os.path.islink(filePath):
+            # 忽略 link 链接
+            continue
+        if os.path.isdir(filePath):
+            lists = GetLnkDesktop(filePath)
+            for k in lists:
+                lnkList.append(k)
+            continue
+        if os.path.isfile(filePath) and os.path.splitext(filePath)[1] == ".lnk":
+            with open(filePath, "rb") as file:
+                while True:
+                    things = file.readline().lower()
+                    if things == b"":
+                        break
+                    for k in str(things[1: -2]).split("\\x00"):
+                        things = k
+                        if things == b"":
+                            break
+                        if "c:" in things:
+                            lnkList.append([filePath, things])
+                            print(things)
+    return lnkList
+
+def ReplaceText(string: str, lists: list):
+    for i in lists:
+        string = string.replace(i[0], i[1])
+    return string
+
+control = '''Package: @@@Package@@@
+Version: @@@Version@@@
+Architecture: i386
+Maintainer: @@@Maintainer@@@
+Depends: @@@Depends@@@
+Section: non-free/otherosfs
+Priority: optional
+Multi-Arch: foreign
+Installed-Size: @@@Installed-Size@@@
+Description: @@@Description@@@
+'''
+
+info = f'''{{
+    "appid": "@@@Package@@@",
+    "name": "@@@Name@@@",
+    "version": "@@@Version@@@",
+    "arch": ["i386"],
+    "permissions": {{
+        "autostart": false,
+        "notification": false,
+        "trayicon": true,
+        "clipboard": true,
+        "account": false,
+        "bluetooth": false,
+        "camera": true,
+        "audio_record": true,
+        "installed_apps": false
+    }}
+}}'''
+
+postrm = f"""#!/bin/bash
+if [ "$1" = "remove" ] || [ "$1" = "purge" ];then
+
+echo "清理卸载残留"
+CONTAINER_NAME="@@@Package@@@"
+
+if [ -z $CONTAINER_NAME ];then
+echo "W: 没有指定容器,跳过清理容器。请手动前往 ~/.deepinwine/ 下删除"
+exit
+fi
+
+/opt/deepinwine/tools/kill.sh $CONTAINER_NAME
+###这里注意,如果没写CONTAINER_NAME,会把QQ杀了
+
+for username in $(ls /home)  
+    do
+      echo /home/$username
+        if [ -d /home/$username/.deepinwine/$CONTAINER_NAME ]  
+        then
+        rm -rf /home/$username/.deepinwine/$CONTAINER_NAME
+        fi
+    done
+else
+echo "非卸载,跳过清理"
+fi"""
+
+runsh = f'''#!/bin/sh
+
+#   Copyright (C) 2016 Deepin, Inc.
+#
+#   Author:     Li LongYu <lilongyu@linuxdeepin.com>
+#               Peng Hao <penghao@linuxdeepin.com>
+#
+#
+#   Copyright (C) 2022 The Spark Project
+#
+#
+#   Modifier    shenmo <shenmo@spark-app.store>
+#
+#
+#
+
+#######################函数段。下文调用的额外功能会在此处声明
+
+Get_Dist_Name()
+{{
+    if grep -Eqii "Deepin" /etc/issue || grep -Eq "Deepin" /etc/*-release; then
+        DISTRO='Deepin'
+    elif grep -Eqi "UnionTech" /etc/issue || grep -Eq "UnionTech" /etc/*-release; then
+        DISTRO='UniontechOS'
+    else
+	 DISTRO='OtherOS'
+	fi
+}}
+
+
+####获得发行版名称
+
+#########################预设值段
+
+version_gt() {{ test "$(echo "$@" | tr " " "\n" | sort -V | head -n 1)" != "$1"; }}
+####用于比较版本?未实装
+BOTTLENAME="@@@Package@@@"
+APPVER="@@@Version@@@"
+EXEC_PATH="@@@EXEC_PATH@@@"
+##### 软件在wine中的启动路径
+START_SHELL_PATH="/opt/deepinwine/tools/spark_run_v4.sh"
+export MIME_TYPE=""
+#####没什么用
+export DEB_PACKAGE_NAME="@@@Package@@@"
+####这里写包名才能在启动的时候正确找到files.7z,似乎也和杀残留进程有关
+export APPRUN_CMD="@@@Package@@@"
+#####wine启动指令,建议
+EXPORT_ENVS=""
+
+export SPECIFY_SHELL_DIR=`dirname $START_SHELL_PATH`
+
+ARCHIVE_FILE_DIR="/opt/apps/$DEB_PACKAGE_NAME/files"
+
+export WINEDLLPATH=/opt/$APPRUN_CMD/lib:/opt/$APPRUN_CMD/lib64
+
+export WINEPREDLL="$ARCHIVE_FILE_DIR/dlls"
+
+DISABLE_ATTACH_FILE_DIALOG=""
+##默认为空。若为1,则不使用系统自带的文件选择,而是使用wine的
+##对于deepin/UOS,大部分的应用都不需要使用wine的,如果有需求(比如wine应用选择的限定种类文件系统的文管不支持)
+##请填1。
+##注意:因为非DDE的环境不确定,所以默认会在非Deepin/UOS发行版上禁用这个功能。如果你确认在适配的发行版上可以正常启动,请注释或者删除下面这段
+
+##############<<<<<<<<<禁用文件选择工具开始
+Get_Dist_Name
+#此功能实现参见结尾函数段
+if [ "$DISTRO" != "Deepin" ] && [ "$DISTRO" != "UniontechOS" ];then
+DISABLE_ATTACH_FILE_DIALOG="1"
+echo "非deepin/UOS,默认关闭系统自带的文件选择工具,使用Wine的"
+echo "如果你想改变这个行为,请到/opt/apps/$DEB_PACKAGE_NAME/files/$0处修改"
+echo "To打包者:如果你要打开自带请注意在适配的发行版上进行测试"
+echo "To用户:打包者没有打开这个功能,这证明启用这个功能可能造成运行问题。如果你要修改这个行为,请确保你有一定的动手能力"
+fi
+##############>>>>>>>>>禁用文件选择工具结束
+
+##############<<<<<<<<<屏蔽mono和gecko安装器开始
+##默认屏蔽mono和gecko安装器
+#if [ "$APPRUN_CMD" = "spark-wine7-devel" ];then
+
+#export WINEDLLOVERRIDES="mscoree,mshtml="
+#echo "为了降低打包体积,默认关闭gecko和momo,如有需要,注释此行(仅对spark-wine7-devel有效)"
+
+#fi
+##############>>>>>>>>>屏蔽mono和gecko安装器结束
+
+#########################执行段
+
+
+
+
+if [ -z "$DISABLE_ATTACH_FILE_DIALOG" ];then
+    export ATTACH_FILE_DIALOG=1
+fi
+
+if [ -n "$EXPORT_ENVS" ];then
+    export $EXPORT_ENVS
+fi
+
+if [ -n "$EXEC_PATH" ];then
+    if [ -z "${{EXEC_PATH##*.lnk*}}" ];then
+        $START_SHELL_PATH $BOTTLENAME $APPVER "C:/windows/command/start.exe" "/Unix" "$EXEC_PATH" "$@"
+    else
+        $START_SHELL_PATH $BOTTLENAME $APPVER "$EXEC_PATH" "$@"
+    fi
+else
+    $START_SHELL_PATH $BOTTLENAME $APPVER "uninstaller.exe" "$@"
+fi'''
+
+desktopFile = f'''#!/usr/bin/env xdg-open
+[Desktop Entry]
+Encoding=UTF-8
+Type=Application
+X-Created-By=@@@Maintainer@@@
+Icon=@@@Icon@@@
+Exec="/opt/apps/@@@Package@@@/files/run.sh"
+Name=@@@Name@@@
+Comment=@@@Description@@@
+MimeType=
+GenericName=@@@Package@@@
+Terminal=false
+StartupNotify=false'''
+
+def WriteTxt(path, things):
+    with open(path, "w") as file:
+        file.write(things)
+
+def ReadTxt(path):
+    things = ""
+    with open(path, "r") as file:
+        things = file.read()
+    return things
+
+class RunThread(QtCore.QThread):
+    showLogText = QtCore.pyqtBoundSignal(str)
+    error = QtCore.pyqtBoundSignal(str)
+    info = QtCore.pyqtBoundSignal(str)
+    question = QtCore.pyqtBoundSignal(str)
+    disbledAll = QtCore.pyqtBoundSignal(bool)
+
+    def RunCommand(self, command):
+        res = subprocess.Popen([command], shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
+        while res.poll() is None:
+            try:
+                text = res.stdout.readline().decode("utf8")
+            except:
+                text = ""
+            self.showLogText.emit(text)
+            print(text, end="")
+
+    def __init__(self) -> None:
+        super().__init__()
+    
+    def run(self):
+        try:
+            if not self.question.emit("在此过程中,需要回答一系列的问题以进行打包,点击确定继续"):
+                self.disbledAll.emit(False)
+                return
+            bottlePath = f"/tmp/deepin-wine-runner-bottle{random.randint(0, 10000)}"
+            # 清空容器以保证能正常使用
+            if os.path.exists(bottlePath):
+                self.RunCommand(f"rm -rfv '{bottlePath}'")
+            ############# 后面将全部调用 deepin wine6 stable 进行操作
+            exeName = os.path.basename(exePath.text())
+            # 暂定
+            debPackageName = "spark-" + xpinyin.Pinyin().get_pinyin(os.path.splitext(exeName)[0].replace(" ", "")).lower().replace("--", "-").replace(" ", "")
+            debPackageVersion = "1.0.0"
+            debMaintainer = os.getlogin()
+            debBuildPath = f"/tmp/deepin-wine-packager-builder-{debPackageName}"
+            bottlePackagePath = f"{debBuildPath}/opt/apps/{debPackageName}/files/files.7z"
+            desktopPath = get_desktop_path()
+            self.RunCommand(f"mkdir -pv '{debBuildPath}/DEBIAN'")
+            self.RunCommand(f"mkdir -pv '{debBuildPath}/opt/apps/{debPackageName}/files'")
+            self.RunCommand(f"mkdir -pv '{debBuildPath}/opt/apps/{debPackageName}/entries/applications'")
+            self.RunCommand(f"mkdir -pv '{debBuildPath}/opt/apps/{debPackageName}/entries/icons'")
+            ############## 运行 EXE
+            if self.question.emit("请问此可执行文件是安装包还是绿色软件?是安装包请按 Yes,绿色软件按 No"):
+                # 清空无益处的 lnk 文件
+                lnkPath = f"{bottlePath}/drive_c/ProgramData/Microsoft/Windows/Start Menu/Programs"
+                self.RunCommand(f"rm -rfv '{lnkPath}'")
+                # 安装包
+                self.RunCommand(f"WINEPREFIX='{bottlePath}' deepin-wine6-stable '{exePath.text()}'")
+                # 安装锁,先不写
+                # 识别 lnk
+                lnkList = GetLnkDesktop(lnkPath)
+                if len(lnkList) <= 0:
+                    self.info.error("无法识别到任何 lnk 快捷方式")
+                    self.disbledAll.emit(False)
+                    return
+                # 选择最优 lnk
+                secondChooseList = []
+                for k in lnkList:
+                    lnkPath = k[0].lower()
+                    if "卸载" in lnkPath or "uninstall" in lnkPath or "update" in lnkPath:
+                        continue
+                    secondChooseList.append(k)
+                if len(secondChooseList) <= 0:
+                    secondChooseList = lnkList
+                rightLnk = secondChooseList[0]
+                miniLenge = len(rightLnk[1])
+                for k in secondChooseList:
+                    # 择优选择路径最短一项
+                    if len(k[1]) < miniLenge:
+                        rightLnk = k
+                        miniLenge = len(rightLnk[1])
+                folderExePath = os.path.dirname(rightLnk[1].replace("\\", "/").replace("c:/", bottlePath))
+                exePathInBottle = rightLnk[1]
+                exeName = os.path.splitext(os.path.basename(folderExePath))[0]
+                programIconPath = ""
+            else:
+                # 绿色软件
+                folderExePath = os.path.dirname(exePath.text())               
+                exePathInBottle = f"c:/Program Files/{os.path.basename(folderExePath)}/{exeName}"
+                exeName = os.path.splitext(os.path.basename(os.path.basename(exePath.text())))[0]
+                programIconPath = ""
+                # 拷贝文件到容器
+                self.RunCommand(f"cp -rv '{folderExePath}/..' '{bottlePath}/Program Files'")
+            debDescription = f"{exeName} By Deepin Wine 6 Stable And Build By Wine Runner"
+            debDepends = "deepin-wine6-stable, spark-dwine-helper | store.spark-app.spark-dwine-helper, fonts-wqy-microhei, fonts-wqy-zenhei"
+            ############ 处理容器
+            # 对用户目录进行处理
+            os.chdir(bottlePath)
+            self.RunCommand("sed -i \"s#$USER#@current_user@#\" ./*.reg")
+            os.chdir(f"{bottlePath}/drive_c/users")
+            # 如果缩放文件 scale.txt 存在,需要移除以便用户自行调节缩放设置
+            if os.path.exists(f"{bottlePath}/scale.txt"):
+                self.RunCommand(f"rm -rfv '{bottlePath}/scale.txt'")
+            # 删除因为脚本失误导致用户目录嵌套(如果存在)
+            if os.path.exists(f"{bottlePath}/drive_c/users/@current_user@/@current_user@"):
+                self.RunCommand(f"rm -rfv '{bottlePath}/drive_c/users/@current_user@/@current_user@'")
+            self.RunCommand(f"mv -fv '{os.getlogin()}' @current_user@")
+            self.RunCommand(f"rm -fv '{bottlePath}/drive_c/users/@current_user@/我的'*")
+            self.RunCommand(f"rm -fv '{bottlePath}/drive_c/users/@current_user@/My '*")
+            self.RunCommand(f"rm -fv '{bottlePath}/drive_c/users/@current_user@/Desktop'")
+            self.RunCommand(f"rm -fv '{bottlePath}/drive_c/users/@current_user@/Downloads'")
+            self.RunCommand(f"rm -fv '{bottlePath}/drive_c/users/@current_user@/Templates'")
+            ########### 打包容器
+            self.RunCommand(f"7z a '{bottlePackagePath}' '{bottlePath}/'*")
+            ########### 生成文件内容
+            buildProgramSize = 0
+            replaceMap = [
+                ["@@@Package@@@", debPackageName],
+                ["@@@Version@@@", debPackageVersion],
+                ["@@@Maintainer@@@", debMaintainer],
+                ["@@@Depends@@@", debDepends],
+                ["@@@Description@@@", debDescription],
+                ["@@@Installed-Size@@@", buildProgramSize],
+                ["@@@Name@@@", exeName],
+                ["@@@EXEC_PATH@@@", exePathInBottle],
+                ["@@@Icon@@@", programIconPath]
+            ]
+            debControl = ReplaceText(control, replaceMap)
+            debPostrm = ReplaceText(postrm, replaceMap)
+            debInfo = ReplaceText(info, replaceMap)
+            debRunSh = ReplaceText(runsh, replaceMap)
+            debDesktop = ReplaceText(desktopFile, replaceMap)
+            ########### 写入文件
+            WriteTxt(f"{debBuildPath}/opt/apps/{debPackageName}/entries/applications/{debPackageName}.desktop", debDesktop)
+            WriteTxt(f"{debBuildPath}/opt/apps/{debPackageName}/files/run.sh", debRunSh)
+            WriteTxt(f"{debBuildPath}/opt/apps/{debPackageName}/info", debInfo)
+            WriteTxt(f"{debBuildPath}/DEBIAN/control", debControl)
+            WriteTxt(f"{debBuildPath}/DEBIAN/postrm", debPostrm)
+            ########### 打包 deb
+            self.RunCommand(f"dpkg -b '{debBuildPath}' '{desktopPath}/{debPackageName}_{debPackageVersion}_i386.deb'")
+            self.info.emit("打包完成!")
+            self.disbledAll.emit(False)
+        except:
+            # 若打包出现任何错误
+            traceback.print_exc()
+            self.error.emit(f"打包错误,详细详细如下:{traceback.format_exc()}")
+            self.showLogText.emit(traceback.format_exc())
+            self.disbledAll.emit(False)
+
+def RunBuildThread():
+    global buildThread
+    buildThread = RunThread()
+    buildThread.showLogText.connect(ShowText)
+    buildThread.error.connect(ErrorMessage)
+    buildThread.info.connect(InformationMessage)
+    buildThread.question.connect(QuestionMessage)
+    buildThread.disbledAll.connect(DisbledAndEnabledAll)
+    buildThread.start()
+
+if __name__ == "__main__":
+    programPath = os.path.split(os.path.realpath(__file__))[0]  # 返回 string
+    iconPath = "{}/deepin-wine-runner.svg".format(programPath)
+    information = json.loads(ReadTxt(f"{programPath}/information.json"))
+    app = QtWidgets.QApplication(sys.argv)
+    version = information["Version"]
+    window = QtWidgets.QMainWindow()
+    widget = QtWidgets.QWidget()
+    layout = QtWidgets.QGridLayout()
+    exePath = QtWidgets.QLineEdit()
+    browserExeButton = QtWidgets.QPushButton("浏览……")
+    logText = QtWidgets.QTextBrowser()
+    logText.setStyleSheet("""
+        background-color: black;
+        color: white;
+    """)
+    buildButton = QtWidgets.QPushButton("现在打包……")
+    buildButton.clicked.connect(RunBuildThread)
+    layout.addWidget(QtWidgets.QLabel("选择 EXE:"), 0, 0)
+    layout.addWidget(exePath, 0, 1)
+    layout.addWidget(browserExeButton, 0, 2)
+    layout.addWidget(buildButton, 1, 1)
+    layout.addWidget(logText, 2, 0, 1, 3)
+    widget.setLayout(layout)
+    window.setCentralWidget(widget)
+    window.setWindowTitle(f"Wine 运行器 {version}——简易打包器")
+    window.show()
+    sys.exit(app.exec_())
\ No newline at end of file