我正在将我们的NSI安装程序移植到Linux和Mac而不是Windows,以便更好地与我们的Maven构建系统集成。
我们需要签署安装程序和卸载程序。这是按照http://nsis.sourceforge.net/Signing_an_Uninstaller的建议完成的,但我刚刚意识到它会尝试运行tempinstaller来强制它生成uninstaller.exe然后可以签名。
显然,这个技巧在* Nix系统上效果不佳,并使这部分过程不可移植。
有没有人有更好的解决方案。我不是NSIS的专家,想知道是否有一种聪明的方法来获取uninstall.exe以便签名?
答案 0 :(得分:3)
我认为没有真正的解决方案。
安装程序和卸载程序使用相同的exe代码,并且仅在启动时检查标记(FH_FLAGS_UNINSTALL
中的firstheader
)以查看它是否是卸载程序。仅仅翻转这个位是不够的,程序将无法通过CRC检查,即使您绕过了卸载程序数据的压缩,因此您必须将其解压缩到文件中的正确位置。要实际完成此操作,您必须编写自定义工具。如果搜索EW_WRITEUNINSTALLER
,则可以在exec.c的NSIS源中看到此操作。
答案 1 :(得分:0)
我们需要签署我们的安装程序和卸载程序。这是按照http://nsis.sourceforge.net/Signing_an_Uninstaller的建议完成的,但是我只是意识到它试图运行tempinstaller来强制它生成可以进行签名的uninstaller.exe。 [...]这个技巧在* Nix系统上效果不佳,并且使该过程的这一部分不可移植。
如果您使用存根安装程序进行卸载操作(没有有效负载),那么这似乎是可能的。
它将从uninstall.exe
文件夹中产生一个$TEMP
进程,然后可以删除$INSTDIR
。
此脚本将创建一个存根(un)安装程序,然后可以对其进行Authenticode签名。它将在Windows,MacOS和Linux上编译。
$INSTDIR
,第二次从$TEMP
)。这是一个子进程,它允许uninstall.exe
删除自身,类似于NSIS does it in the Section "Uninstall"
的方式。.nsi
脚本,如果您在安装/卸载部分之间有很多共享逻辑,那么就很麻烦。
"Uninstall"
部分标题,因为在生成该字节码时,您将遇到与OP相同的问题。$TEMP
显式运行时,某些相对文件逻辑将不正确。该示例分别将这些作为$DELETE_DIR
和$DELETE_EXE
传回。!include MUI2.nsh
!include x64.nsh
!include LogicLib.nsh
!include FileFunc.nsh
!include WinMessages.nsh
!define MUI_PRODUCT "My App"
!define MUI_VERSION "1.0.0"
; Masquerade the title
!define MUI_PAGE_HEADER_TEXT "Uninstall My App"
!define MUI_PAGE_HEADER_SUBTEXT "Remove My App from your computer"
!define MUI_INSTFILESPAGE_FINISHHEADER_TEXT "Uninstallation Complete"
!define MUI_INSTFILESPAGE_FINISHHEADER_SUBTEXT "Uninstall was completed successfully."
!insertmacro MUI_PAGE_INSTFILES
!insertmacro MUI_LANGUAGE "English"
!insertmacro GetParameters
RequestExecutionLevel admin
CRCCheck On
OutFile "uninstall.exe"
Name "Uninstall"
Var /GLOBAL RESPAWN
Var /GLOBAL DELETE_DIR
Var /GLOBAL DELETE_EXE
Section
; Masquerade as uninstall
SendMessage $HWNDPARENT ${WM_SETTEXT} 0 "STR:Uninstall"
${GetParameters} $0
${GetOptions} "$0" "/RESPAWN=" $RESPAWN
${GetOptions} "$0" "/DELETE_DIR=" $DELETE_DIR
${GetOptions} "$0" "/DELETE_EXE=" $DELETE_EXE
${If} $RESPAWN != ""
; We're running from $TEMP; Perform the uninstall
!define yay "We're running from $EXEPATH, yay, we can remove the install directory!$\n$\n"
!define myvars "$\tRESPAWN$\t$RESPAWN$\n$\tDELETE_EXE$\t$DELETE_EXE$\n$\tDELETE_DIR$\t$DELETE_DIR"
MessageBox MB_OK "${yay}${myvars}"
; Your uninstall code goes here
; RMDir /r $DELETE_DIR\*.*
; Delete "$DESKTOP\${MUI_PRODUCT}.lnk"
; Delete "$SMPROGRAMS\${MUI_PRODUCT}\*.*"
; RmDir "$SMPROGRAMS\${MUI_PRODUCT}"
; Delete Uninstaller And Unistall Registry Entries
; DeleteRegKey HKEY_LOCAL_MACHINE "SOFTWARE\${MUI_PRODUCT}"
; DeleteRegKey HKEY_LOCAL_MACHINE "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\${MUI_PRODUCT}"
; Remove the old version of ourself
ClearErrors
Delete $DELETE_EXE
IfErrors 0 +3
MessageBox MB_OK "File could NOT be deleted: $DELETE_EXE"
Goto +2
MessageBox MB_OK "File was successfully deleted: $DELETE_EXE"
; Remove ourself from $TEMP after reboot
Delete /REBOOTOK $EXEPATH
; ${If} ${RunningX64}
; ${EnableX64FSRedirection}
; ${EndIf}
SetDetailsPrint textonly
DetailPrint "Completed"
${Else}
; We're NOT running from $TEMP, copy to temp and respawn ourself
GetTempFileName $0
CopyFiles "$EXEPATH" "$0"
Exec '"$0" /RESPAWN=1 /DELETE_DIR="$EXEDIR" /DELETE_EXE="$EXEPATH"'
Quit
${EndIf}
SectionEnd
Function .onInit
; ${If} ${RunningX64}
; SetRegView 64
; ${DisableX64FSRedirection}
; ${EndIf}
FunctionEnd