Как сохранить сообщение команды DetailPrint в файл в NSIS Script? - PullRequest
4 голосов
/ 11 февраля 2012

Идея состоит в том, чтобы запустить сценарий NSIS в режиме без вывода сообщений на удаленном компьютере и после успешного завершения установки получить обратно файл журнала на хост-компьютер.

У меня есть множество Detailprint командных сообщений, которые показываютпрогресс сценария.Теперь вопрос в том, как мне сохранить эти сообщения в лог-файл.Я смотрел на это http://nsis.sourceforge.net/Dump_log_to_file, но он говорит, что он не будет работать в беззвучном режиме.

Ответы [ 2 ]

4 голосов
/ 14 февраля 2012

Я использую этот метод для сохранения логов. Я не знаю, является ли это лучшим методом, но он хорошо работает для меня.

!ifndef DEBUG_MODE
  !define DEBUG_MODE true
!endif

!define LOGFILE "$TEMP\my_logfile.log"

!if ${DEBUG_MODE} == true
  !define DetailPrint '!insertmacro _debugMsg'
!else
  !define DetailPrint '!insertmacro _nodebugMsg'
!endif

!macro _debugMsg MSG
  push $3
  push $2
  push $1
  push $0
  push $R0

  strcpy $R0 "${MSG}"

  ClearErrors
  FileOpen $1 ${LOGFILE} a
  FileSeek $1 0 END
  IfErrors +8
  ${GetTime} "" "L" $0 $0 $0 $0 $0 $2 $3
  FileWrite $1 "$0:$2:$3$\t"
  FileWrite $1 "${__FUNCTION__}$\t"
  FileWrite $1 "$R0"
  FileWrite $1 "$\t(${__FILE__},${__FUNCTION__},${__LINE__})"
  FileWrite $1 "$\n"
  FileClose $1

  pop $R0
  pop $0
  pop $1
  pop $2
  pop $3
!macroend

!macro _nodebugMsg _MSG
  ;you can put here nothing or you can put the regular DetailPrint
  DetailPrint "${MSG}"
!macroend

После этого вам просто нужно найти и заменить DetailPrint на $ {DetailPrint}

1 голос
/ 11 февраля 2012

Это верно, окно списка не существует в режиме без вывода сообщений.

Я полагаю, у вас есть два варианта, прокрутите свою собственную запись в лог:

var hFileLog
!macro Log_Init logfile
FileOpen $hFileLog "${logfile}" w ;Or "a" for append
!macroend
!macro Log_String msg
DetailPrint "${msg}"
FileWrite $hFileLog "${msg}$\r$\n"
!macroend
!macro Log_Close
FileWrite $hFileLog 'Done.'
FileClose $hFileLog
!macroend
!macro File src
File "${src}"
FileWrite $hFileLog 'Extracting "${src}" to $outdir$\r$\n'
!macroend

Section "Main Program"
!insertmacro Log_Init "$instdir\install.log"
!insertmacro Log_String "Starting install..."

SetOutPath $instdir
!insertmacro File "${__FILE__}"
SectionEnd

Section "Optional Foo Component"
!insertmacro Log_String "Installing Foo"
SectionEnd

Section
!insertmacro Log_Close
SectionEnd

или добавьте больше хаков, чтобы DumpLog работал:

Function .onInit
IfSilent 0 +2
StrCpy $0 1 ;We need to know if we are in "real silent" mode
SetSilent normal ;Turn off real silent mode so we can fake it
FunctionEnd

Function .onGuiInit
StrCmp $0 1 0 +3
HideWindow
SetSilent silent ;The docs say this is only supported in .onInit but all we care about is the internal flag used by IfSilent
FunctionEnd

LicenseData "${__FILE__}"
page license skipifsilent ; All pages except instfiles needs this hack
page directory skipifsilent
page instfiles

Function skipifsilent
IfSilent 0 +3
HideWindow
Abort
FunctionEnd

Section
IfSilent 0 +2
HideWindow

SetOutPath $instdir
File "${__FILE__}"
Sleep 2222 ;So we can see that it is hidden and not just finishing quickly


DetailPrint "$(^Completed)" ; Fake the completed message show it shows in the log
Push "$EXEDIR\install.log"
Call DumpLog

IfSilent 0 +2
Quit
SetDetailsPrint textonly
SectionEnd

!define LVM_GETITEMCOUNT 0x1004
!define LVM_GETITEMTEXT 0x102D

Function DumpLog
  Exch $5
  Push $0
  Push $1
  Push $2
  Push $3
  Push $4
  Push $6

  FindWindow $0 "#32770" "" $HWNDPARENT
  GetDlgItem $0 $0 1016
  StrCmp $0 0 exit
  FileOpen $5 $5 "w"
  StrCmp $5 "" exit
    SendMessage $0 ${LVM_GETITEMCOUNT} 0 0 $6
    System::Alloc ${NSIS_MAX_STRLEN}
    Pop $3
    StrCpy $2 0
    System::Call "*(i, i, i, i, i, i, i, i, i) i \
      (0, 0, 0, 0, 0, r3, ${NSIS_MAX_STRLEN}) .r1"
    loop: StrCmp $2 $6 done
      System::Call "User32::SendMessageA(i, i, i, i) i \
        ($0, ${LVM_GETITEMTEXT}, $2, r1)"
      System::Call "*$3(&t${NSIS_MAX_STRLEN} .r4)"
      FileWrite $5 "$4$\r$\n"
      IntOp $2 $2 + 1
      Goto loop
    done:
      FileClose $5
      System::Free $1
      System::Free $3
  exit:
    Pop $6
    Pop $4
    Pop $3
    Pop $2
    Pop $1
    Pop $0
    Exch $5
FunctionEnd
...