Using Windows Installer (WIX) to schedule a task as the system user

Goal: use WIX to schedule a task via the task scheduler. The task must run every 30 minutes after the computer starts, it must also run as the SYSTEM user.

Answer:

<Product Id="*" 
  Name="FooBar" 
  Language="1033" 
  Version="1.0.0.0" Manufacturer="Foo" 
  UpgradeCode="GID">
<Package Id="*" 
  InstallerVersion="200" 
  Compressed="yes" 
  InstallScope="perMachine" 
  InstallPrivileges="elevated"/>
     

<InstallExecuteSequence>
      <Custom Action="CreateScheduledTask" After="InstallFiles">
        NOT Installed
      </Custom>
</InstallExecuteSequence>

<CustomAction Id="CreateScheduledTask" 
    Return="check" 
    Impersonate="no" 
    Execute="deferred"
    Directory="TARGETDIR" 
    ExeCommand="&quot;[SystemFolder]SCHTASKS.EXE&quot; /CREATE /SC MINUTE /MO 20 /TN &quot;Foobar&quot;  /TR &quot;&quot;[INSTALLFOLDER]\Foobar.exe&quot;&quot; /RU &quot;NT Authority\System&quot; /RP /RL HIGHEST" />
</Product>

Explanation:

To schedule a task you need to create a custom action which calls the command line version of the Windows Task Scheduler (schtasks.exe). In the example above the task is being scheduled to run every 20 minutes starting from when the computer boots (/SC MINUTE /MO 20).

The tricky part is making the new scheduled task run as the SYSTEM user with the highest permissions possible (/RU “NT Authority\System” /RP /RL HIGHEST”). To do this the installer itself must run with elevated privileges AND the CustomAction must run with elevated privileges. To run the installer with elevated privileges I added InstallPrivileges=”elevated” to the Package element. To run the CustomAction with elevated privileges I added Impersonate=”no” and Execute=”deferred” to the CustomAction element. Aaron Stebner explains why deferring the action is necessary. Also note that the use of ‘&quot;’ instead of ‘”‘ is a feature of WIX and not an error in web formatting.

This solution works, but it is limited by the number of parameters schtasks will take in. This prevents you from doing things like disabling the AC Power requirement. Here is an explanation of how to do that.

Getting rid of the black boxes

Another issue is that a command prompt pops up briefly making the install look messy. To solve the issue you have to use a Quiet Execution Custom Action.

<Product Id="*" 
  Name="FooBar" 
  Language="1033" 
  Version="1.0.0.0" Manufacturer="Foo" 
  UpgradeCode="GID">
<Package Id="*" 
  InstallerVersion="200" 
  Compressed="yes" 
  InstallScope="perMachine" 
  InstallPrivileges="elevated"/>
     

<InstallExecuteSequence>
      <Custom Action="CreateScheduledTask" After="InstallFiles">
        NOT Installed
      </Custom>
      <Custom Action="CreateScheduledTaskId" After="CostFinalize">
        NOT Installed
      </Custom>
</InstallExecuteSequence>

<CustomAction Id="CreateScheduledTaskId" 
    Property="CreateScheduledTask"
    Execute="immediate"
    Value="&quot;[SystemFolder]SCHTASKS.EXE&quot; /CREATE /SC MINUTE /MO 20 /TN &quot;Foobar&quot; /TR &quot;&quot;[INSTALLFOLDER]\Foobar.exe&quot;&quot; /RU &quot;NT Authority\System&quot; /RP /RL HIGHEST" />

<CustomAction Id="CreateScheduledTask" 
    Return="check" 
    Impersonate="no" 
    Execute="deferred"
    BinaryKey="WixCA"
    DllEntry="CAQuietExec" />
</Product>