从Linux或Mac签署NSIS卸载程序

时间:2015-08-02 21:38:56

标签: nsis

我正在将我们的NSI安装程序移植到Linux和Mac而不是Windows,以便更好地与我们的Maven构建系统集成。

我们需要签署安装程序和卸载程序。这是按照http://nsis.sourceforge.net/Signing_an_Uninstaller的建议完成的,但我刚刚意识到它会尝试运行tempinstaller来强制它生成uninstaller.exe然后可以签名。

显然,这个技巧在* Nix系统上效果不佳,并使这部分过程不可移植。

有没有人有更好的解决方案。我不是NSIS的专家,想知道是否有一种聪明的方法来获取uninstall.exe以便签名?

2 个答案:

答案 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上编译。

install masqueraded screenshot

注意事项:

  • 您必须手动将其捆绑到安装程序中(简单)
  • 您必须管理自己的卸载注册表项(简单)
  • 外观可能与NSIS对于卸载程序的默认设置不符
  • 您将看到安装程序打开两次(第一次从$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