Skip to main content

How to automate GUI tests for macOS '.dmg' installer images

If you have used Squish before, you probably already test your application thoroughly. However for a new user, there are a few steps to take until they even run the application in the first place: they need to install it. That procedure is part of the first impression you make, so it would be nice to get it right, wouldn't it? Let's take a look!

Just in case you are (also) on Windows, we outlined a procedure for MSI installers in an earlier blog post.

Since we are on macOS, applications are packaged inside disk image (.dmg) files. They usually contain either the app bundle itself or an installer package (.pkg) that performs additional setup besides extracting an app bundle to /Applications. The easiest way to get to grips with the contents of a disk image (from Squish) is to open it via the command line. But before we get our hands dirty let's start by looking at the general idea in pseudo code:
[code language="python" title="The general idea how to access the installer image"]openDmg("my_application_installer.dmg")
registerInstallerAsAUT("application_installer")
startApplication("application_installer")
closeDmg()[/code]
As you can see, there are three functions. You might recognize the next-to-last one, it is the built-in Squish function to start an application under test. The others are helpers we are going to implement in the next paragraphs to access the dmg installer image.

Opening the DMG Installer Image

Let's start with openDmg() so that we can see what's inside the image...
[code language="python" title="Mounting a dmg image"]def openDmg():
test.startSection("Open/mount .dmg file")
args = ["open", os.path.join(dmg_file_path, dmg_file_name)]
test.log("Executing: %s" % args)

try:
output = subprocess.check_output(args)
except subprocess.CalledProcessError:
test.fatal("Opening/mounting failed:")
test.fatal("File: %s" % os.path.join(dmg_file_path, dmg_file_name))
test.fatal("Output: %s" % output)
return False
finally:
test.endSection()
return True[/code]
This snippet essentially just executes the command open ~/path/to/Application.dmg and makes use of macOS's automatic dmg mounting mechanism. (The open command performs the same action as clicking an item in a Finder window would.) You'll notice we use this code pattern several times to execute the commands we need.

Registering the Installer as a Squish AUT

The disk image is mounted and macOS should have opened a window as if we had clicked it manually. The next step is to execute the installer so that we can record and replay actions on it as usual. For that to work the installer has to be registered with Squish as a separate application under test (AUT) - which boils down to another command line call:
[code language="python" title="Registering the installer as AUT"]def registerInstallerAsAUT():
test.startSection("Register installer as AUT")
args = [os.path.join(os.getenv("SQUISH_PREFIX"), "bin", "squishserver"), "--config", "addAUT", installer_file_name, installer_file_path]
test.log("Executing: %s" % args)

try:
output = subprocess.check_output(args)
except subprocess.CalledProcessError:
test.fatal("Registering installer as AUT failed:")
test.fatal("Output: %s" % output)
return False
finally:
test.endSection()
return True[/code]
The command line this time: $SQUISH_PREFIX/bin/squishserver --config addAUT your_installer_name /Volumes/Disk_Image/. As you can see: no magic involved. We have completed the two helper functions from the pseudo code we started out with. Let's get them to do something helpful!

Putting Everything Together

If we put everything together it could look like this:
[code language="python" highlight="16" title="The main() function"]def main():
if not openDmg():
return
if not registerInstallerAsAUT():
return

# Start running the installer as the AUT from here
test.startSection("Start installer as AUT")
try:
startApplication('"%s"' % installer_file_name)
finally:
test.endSection()

# Set breakpoint on the next line, execute to it, start
# recording a snippet (Run > Record Snippet):
snooze(1)[/code]
You can now teach Squish to complete your installer by recording the necessary steps. At the snooze(1) in the last line, the installer is up and running. Place a breakpoint there and start recording!

Concerning cleaning up after ourselves, you might remember the closeDmg() function in the pseudo code far above. That one just calls umount /Volumes/Your_Image to eject the image after the installer has terminated. A working implementation as well as all the source code from this post can be found in the Knowledge Base article that covers this topic as well.

If You Have an App Bundle

If the AUT just consists of an app bundle (Application.app) as outlined in the beginning, we could simply copy it to a temporary location, register it with Squish (like we did with the installer above) and run it right away.

If you have a .pkg based installer

.pkg files are not executable, and they are handled by a system application called Installer.app. Hooking this application via Squish is not currently possible, so Squish can not be used to automate the installation of apps that are packaged in this way.

Comments

?
M.
0 points
91 months ago

I've been waiting for this locator feature for years! Great!

?
cesar
0 points
91 months ago

QT IS THE BEST BROOOO

?
nocpes
0 points
91 months ago

I see a bug that qt creator cannot understand the std::map::iterator. that it does not show hint for std::map::iterator->.
we can see that qt creator is using its own front end parser. Why doesn't it use clang for front end while it has clang-code-model plugin?

Netbeans and Kdevelop are handling very good for the cases like this.

?
Eike Ziller
0 points
91 months ago

If you enable the ClangCodeModel plugin (and restart Qt Creator afterwards), then Qt Creator uses Clang for semantic highlighting and completion.

For me, completion on e.g. "std::map<int, int>::iterator it; it->" seems to work correctly (after including , of course). Do you see any errors from the code model regarding your code? (E.g. as annotation after some code line, or a red circle icon on the left of the line numbers.)

?
tr3w
0 points
91 months ago

CMake grouping is nice, but can you delete a variable? (Or is it just me who wants this?)

?
Kai Pastor
0 points
91 months ago

QTCREATORBUG-16241. This is really annoying...

?
Konstantin Podsvirov
0 points
91 months ago

+1

?
Tobias Hunger
0 points
91 months ago

https://codereview.qt-proje... implements that, but unfortunately that got stuck in review and did not make the cut before feature freeze.

Next on my CMake configuration ToDo list is a bulk-edit feature, where you get a text editor where you can edit your project configuration in a CMake commandline compatible fashion and that can be cut/pasted.

?
tr3w
0 points
91 months ago

Good to hear, thanks.

?
Nick
0 points
91 months ago

I suppose there's no way to uninstall Qt Creator 4.4?

Also, after installing it in the MaintenanceTool (on Linux), trying to start it results in an error message:

"Could not find the program '/home/realnc/opt/Qt/Tools/Preview/Qt'"

The installed .desktop file should quote the executable path in the "Exec" entry.

?
Eike Ziller
0 points
91 months ago

"I suppose there's no way to uninstall Qt Creator 4.4?"

Still not, no. It's still used by the installer to actually register the predefined kits etc.

"The installed .desktop file should quote the executable path in the "Exec" entry."

Sorry for that and thanks for the report. I've prepared a patch.

?
Eike Ziller
0 points
91 months ago

Fix is now available through online update (might take some more time to appear depending on your region / mirror)

?
Tom Pollok
0 points
91 months ago

Is there a roadmap for QtCreator for the next 12 monts or so? Im curious.

?
Fabrice Mousset
0 points
91 months ago

I've seen for Android project, Ant is no more supported since this release.
I am not an Android expert, but have started a Qt/Android project last year using Ant (I have followed a tutorial found somewhere on internet).

What does it mean to me?
What do I have to change to continue this Android project with Qt Creator 4.5 and later?
I think now Qt Creator will use graddle, but how to switch from Ant to grabble?
Is there something I have to take care?
Is it done automatically?

?
rudi
0 points
91 months ago

Just updated to Qt 5.9.2 on Windows7 64 bit VS2015 which contains QtCreator 4.4.1.

I Just noticed a smal but very annoying bug:

In the To-Do-Entries Tab, the items are soo light that they seem to be transparent
and I cant read them.

But if I switch to a dark theme the items are readable.

Is this a new problem with the light themes??

Is this bug fixed in QtCreator 4.5??

?
Eike Ziller
0 points
91 months ago

There is an issue with settings upgrade (https://bugreports.qt.io/br...), solvable by resetting the colors in the TODO settings.

?
rudi
0 points
91 months ago

OK thanks for the info.

Would be nice to have a changelog with informations
"incopatible changes" or "howto migrate" so such questions
would not/less happen.

But again thanks for the info :-)

?
Roman
0 points
91 months ago

Fuzzy locator is fantastic, now QtCreator is on par with Sublime text or Visual Studio Code here. I hope that multiple cursors support is also in developers short list.

?
AlterX
0 points
91 months ago

Please fix all bug in the designer...4.4.1 is fully bugged!

?
Andre
0 points
91 months ago

Could you be a bit more specific what is buggy?

I assume you refer to the QtQuick Designer. Where do you exactly have problems? Best is, to report them to https://bugreports.qt.io

?
Rich
0 points
91 months ago

Great to see fuzzy search in Locator, one of those things you don't realise you need and then can't live without.

Also love the inline annotations added in 4.4. One thing that would make annotation support even better, would be a keyboard shortcut to display the annotation under the cursor in a pop-up. Basically allow the current mouseover behaviour to be triggered from the keyboard. For warnings that are too long to fit on the end of a code line, it would save having to take your hands off the keys.

On a related anti-rodent note, while the navigation pane is being worked on, any chance of enabling fakevim support in there?

?
Kha Nguyen
0 points
91 months ago

Qt Creator 4.4 seems to have problem with the latest Android build tool (26). Would this be addressed in Qt Creator 4.5?

?
Keith
0 points
91 months ago

Qt Creator is getting better and better, but it still needs a few things:

  1. Multiple 'Locals and expressions' windows in the debugger - preferably 4 of them, tabbed, like VC++
  2. The ability to just type in a variable name in the above windows, rather than right click/enter into dialog/ok, that's just too tedious.
  3. Speedups for large projects, once you get to a dozen or so subprojects each with maybe 50-100 files it is painful.

The Qt Company acquired froglogic GmbH in order to bring the functionality of their market-leading automated testing suite of tools to our comprehensive quality assurance offering.