Monday, October 27, 2014

The cons and cons of testing installations with WiX Lux Extension

On my previous post I presented the challenges of testing installations. I've briefly discussed my reluctance with Lux - the only existing framework for unit-testing installations.

In this post I'll show a simple replacement for Lux that only uses standard WiX markup.
First lets summarize the operational concept of Lux tests:
  • Immediate custom actions set properties
  • Deferred custom actions parse their CustomActionData properties and modify the system accordingly
  • Lux tests properties alone, leaving deferred custom actions for other testing methods.
Which reveal the main problem with Lux - its limited scope: It only tests property values. It doesn't check deferred actions' results, nor does it check table values. It doesn't even check that an installation passes successfully - same goes for removal, upgrade, or any other flow.

An example to a basic Lux test - taken from the extension's manual - looks like this:
<Fragment>
  <lux:UnitTest CustomAction="TestCustomActionSimple" 
           Property="SIMPLE" 
           Value="[INSTALLFOLDER]" 
           Operator="equal" />
</Fragment>
This markup tests that a property named SIMPLE equals INSTALLFOLDER property.

One can replace it with this markup, which makes no use of Lux:
<Fragment>
<?ifdef UNIT_TEST ?> 
  <UI>
    <Error Id="10000">UnitTest failed</Error>
  </UI>
  <CustomAction Id="TestCustomActionSimple" 
           Error="10000" 
           Execute="immediate"  
           Return="check" />
  <CustomActionRef Id="WixExitEarlyWithSuccess" /> 
  <InstallExecuteSequence>
      
      <!-- Error if conditions show failure. -->
      <Custom Action="TestCustomActionSimple" Before="InstallFinalize">
        Not (SIMPLE=INSTALLFOLDER)
      </Custom>
      
      <!-- Terminate Successfully if the test passes -->
      <Custom Action="WixExitEarlyWithSuccess" After="TestCustomActionSimple" />
    </InstallExecuteSequence>
<?endif?> 
</Fragment>

This markup defines an error custom action that will be executed if SIMPLE doesn't equal INSTALLFOLDER. If this condition fails (meaning SIMPLE equals INSTALLFOLDER) then the installation will terminate successfully by executing WixExitEarlyWithSuccess.

Lux defines few more test operators, these however are just shims for Windows Installer conditional operators.

Another major drawback of Lux is in its replacement of Product element: Lux replaces the Product element, ignoring anything within it. That means you have to re-organize your markup to be able to test it.
With the alternative markup I presented you don't need to change or re-organize any markup for unit tests to run.

To summarize Lux pitfalls:
  • Limited scope - Lux only tests property values
  • Lux doesn't check any installation flows
  • Product element is ignored altogether.
On my next post I'll present what I think a Windows Installer unit-testing framework should be able to do.

1 comment:

  1. Thanks to Nir Bar! Here I saw operational concept of Lux tests. Waiting for next post.

    ReplyDelete