Почему значение ErrorLevel не обновляется?

Я пытаюсь написать подпрограмму в пакетном режиме, чтобы определить, установлен ли на компьютере Deep Freeze или безопасно ли он заморожен. (Для тех, кто не знает, Deep Freeze - это программа, которая обычно используется для восстановления / предотвращения и изменения диска в операционной системе.) Dfc.exe - это файл, устанавливаемый методом глубокой заморозки, который можно использовать для определения. Возвращает 0 в случае размораживания или 1 в случае замораживания

По сути, я думал, что бессмысленно запускать сценарии, которые устанавливают определенные вещи, если компьютер завис. Сначала я проверяю, установлен ли dfc.exe, а затем пытаюсь проверить, нет ли оттаявшего / замороженного состояния, но проблема в том, что по какой-то причине возвращаемое значение dfc не обновляется, когда я проверяю уровень ошибки во второй раз.

Кто-нибудь знает, почему я не вижу возвращаемого значения второй проверки ErrorLevel ( в строке 41), которую я также включил в код ниже.

РЕДАКТИРОВАТЬ: Добавлен Psuedocode для моего мыслительного процесса до кода.

::Point1
IF Dfc.exe is present then (
    ::Point2
    If "dfc get /ISFROZEN" returns FROZEN then (
        ::Point3
        Write out to file so we can know that something was skipped, 
        goto EOF to close out of script and avoid wasting cycles installing
    )
    ::Point4

::Deep freeze is installed, but thawed or it would have gotten caught in the "FROZEN" IF
::Fall through out of outer IF without running anything else
)
::Point5

GOTO (Return to where we left to check if we were wasting time with a label) 

Код ниже здесь

@ECHO Off
::CheckForDF here
ECHO We will now test for DeepFreeze ether not installed or is thawed
Pause 
ECHO.

set flagfile=C:\testjava.txt
::Flagfile variable should already be defined before calling this function
Goto :CheckForDF
:DFNotFrozen
ECHO DeepFreeze wasn't installed or is currently thawed
ECHO **Continue with the rest of the script** 
ECHO.
PAUSE

GOTO:eof







::***************************
::********Subroutines********
::***************************


:CheckForDF
WHERE dfc >nul 2>&1


::ErrorLEvel 0 means file was found, which means DF is installed

IF %ERRORLEVEL% EQU 0 (
   dfc get /ISFROZEN
   ECHO %errorlevel%
   ::ErrorLevel 0 - THAWED and ready to install 
   ::ErrorLevel 1 - FROZEN and pointless to try

   IF %errorlevel% EQU 1 (

      ::Echo out what WOULD have been installed to a file so we could check that the
      ::   script still ran (and get an idea of how bad we need to unfreeze and log into each computer) 
      ECHO %flagfile% %date%%time% >> C:\BRCCIT\ScriptsSkippedByDeepFreeze.txt
      ECHO ****DeepFreeze is currently frozen****

      ECHO.
      PAUSE   
      ::Else - DeepFreeze is thawed. return to normal script
      goto :EOF

   )

   ::DeepFreeze is thawed, but is installed. We'll just fall through to the not installed result
)

::DeepFreeze Installation not found. Okay to return to normal script
ECHO DeepFreeze is not installed
goto :DFNotFrozen

ОБНОВЛЕНИЕ: я отказался от вложенных IF и вернулся к GOTOs и Labels. Это не так красиво, но этот код на самом деле работает и буквально за десять минут. Я отступил, чтобы создать визуальный эффект искусственного "вложения"

@ECHO Off
::CheckForDF here
ECHO We will now test for DeepFreeze ether not installed or is thawed
Pause 
ECHO.

set flagfile=C:\testjava.txt
::Flagfile variable should already be defined before calling this function
Goto :CheckForDF
:DFNotFrozen
ECHO DeepFreeze wasn't installed or is currently thawed
ECHO **Continue with the rest of the script** 
ECHO.
PAUSE

GOTO:eof



::***************************
::********Subroutines********
::***************************



:CheckForDF
WHERE dfc >nul 2>&1

IF %ErrorLevel% EQU 0 Goto :DFInstalled
::ELSE - DF Not  found 
GOTO :ReturnToScript


    :DFInstalled
    ::DFC.exe get /ISFROZEN
    Call ExitWithSpecifiedCode.bat 1
    ::If Frozen
    IF %ErrorLevel% EQU 1 GOTO DFFrozen
    ::If Thawed
    IF %ErrorLevel% EQU 0 GOTO ReturnToScript


        :DFFrozen
        ECHO %flagfile% %date%%time% >> C:\BRCCIT\ScriptsSkippedByDeepFreeze.txt
        ECHO ****DeepFreeze is currently frozen****

        ECHO.
        PAUSE
        ::Exit Script Execution
        goto :EOF

:ReturnToScript
ECHO ReturnToScript
PAUSE
GOTO (Return to script execution)

4 ответа

ERRORLEVEL не обновляет внутренние блоки управления, такие как операторы IF, если вы не используете!ERRORLEVEL! вместо%ERRORLEVEL% и используйте эту команду в начале вашего кода: setlocal ENABLEDELAYEDEXPANSION

см. http://batcheero.blogspot.ca/2007/06/how-to-enabledelayedexpansion.html

У меня нет машины с Windows, но если бы это был я, я бы проверил, ECHO утверждение не меняется ERRORLEVEL,

Вероятно, это не связано, но, возможно, вы используете интерпретатор команд, который обрабатывает прописные и строчные буквы ERRORLEVEL по-другому? Даже если это не имеет никакого эффекта, это свойственно быть непоследовательным.

Следующий пакетный файл даст вам статус, который вы ищете. Вы можете легко настроить gotos и этикетки в соответствии с вашими потребностями.

Проверяется, существует ли файл, и ERRORLEVEL обновляется соответствующим образом.

@echo off

REM ---------------------------------------------------------------------------------
REM -- Check the status of Deep Freeze
set "DFC=C:\Program Files (x86)\Faronics\Deep Freeze Enterprise\DFC.exe"
if not exist "%DFC%" goto alwaysRun
"%DFC%" get /ISFROZEN >nul 2>&1
if [9009] == [%ERRORLEVEL%] goto alwaysRun
if ERRORLEVEL 1 goto isFrozen
if ERRORLEVEL 0 goto isThawed
REM ---------------------------------------------------------------------------------





REM ---------------------------------------------------------------------------------
REM -- Run all the commands when the workstation is frozen
REM -- (these commands are skipped when the workstation is thawed)
goto alwaysRun
:isFrozen

echo The workstation is frozen
REM ---------------------------------------------------------------------------------





REM ---------------------------------------------------------------------------------
REM -- Run all the commands when the workstation is thawed
REM -- (these commands are skipped when the workstation is frozen)
goto alwaysRun
:isThawed

echo The workstation is thawed
REM ---------------------------------------------------------------------------------





REM ---------------------------------------------------------------------------------
:alwaysRun

echo Always run this part
REM ---------------------------------------------------------------------------------





REM ---------------------------------------------------------------------------------
:end
pause

Ваша проблема в том, что ваша вторая проверка выполняется в блоке кода, который используется ТОЛЬКО, если ваша первая проверка равна 0. Вы похожи на:

IF (we want to go out) THEN (
  IF (it rains) THEN (
    take umbrella
  )
)

Смотрите, вторая проверка, чтобы увидеть, идет ли дождь, если, и ТОЛЬКО если, мы хотим выйти. Если мы не хотим выходить на улицу, мы не проверяем дождь.

Решение: закройте блок после первого IF, прежде чем перейти ко второму if, так что вы получите:

IF %ERRORLEVEL% EQU 0 (
  do stuff
)
IF %errorlevel% EQU 1 (
  do other stuff
)

Или используйте ELSE:

IF %ERRORLEVEL% EQU 0 (
  do stuff
)
ELSE (
  :: errorlevel was anything other than 0, i.e. NEQ 0
  do other stuff
)
Другие вопросы по тегам