So you’re ready to release a new version of your app. How do you go about doing that?
Put a copy of your .app (with the same name as the version it’s replacing) in a .dmg, .zip, .tar.*, or .aar.
Make sure symlinks are preserved when you create the archive. macOS frameworks use symlinks, and their code signature will be broken if your archival tool follows symlinks instead of archiving them.
For creating dmg archives, APFS formatted images that use lzfse compression are recommended for decent decompression speed.
For creating zip archives, use ditto
(behaves similar to Finder’s Compress option):
ditto -c -k --sequesterRsrc --keepParent MyApp.app MyApp.zip
For creating tar archives, stripping extended attributes is recommended for backwards compatibility from older systems. This example creates a LZMA compressed tar archive with high compression (but slower decompression).
tar --no-xattrs -cJf MyApp.tar.xz MyApp.app
Note --no-xattrs
assumes your application and its code signature does not rely on extended attributes and places code and data into their proper places.
For creating Apple Archives (.aar
), check man aa
. Sparkle 2.7 (beta) / macOS 10.15+ support this format. Enabling SUVerifyUpdateBeforeExtraction is required for using this archive type.
Please see notes for Installer packages if you are not updating a regular bundle.
In order to prevent corruption and man-in-the-middle attacks against your users, you must cryptographically sign your updates.
Signatures are automatically generated when you make an appcast using generate_appcast
tool. This is the recommended method.
To manually generate signatures for your updates, Sparkle includes a tool to help you make a EdDSA signature of the archive. From the Sparkle distribution:
./bin/sign_update path_to_your_update.(zip|dmg|tar.*)
The output will be an XML fragment with your update’s EdDSA signature and (optional) file size like so:
sparkle:edSignature="7cLALFUHSwvEJWSkV8aMreoBe4fhRa4FncC5NoThKxwThL6FDR7hTiPJh1fo2uagnPogisnQsgFgq6mGkt2RBw==" length="1623481"
You’ll add these attributes to your enclosure in the next step.
Since 10.11, macOS has App Transport Security policy which blocks apps from using insecure HTTP connections. This restriction applies to Sparkle as well, so you will need to serve your appcast and the update files over HTTPS.
You need to create an <item>
for your update in your appcast. See the sample appcast for an example. Here’s a template you might use:
<item>
<title>Version 2.0 (2 bugs fixed; 3 new features)</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.0</sparkle:version>
<sparkle:releaseNotesLink>
https://example.com/release_notes/app_2.0.html
</sparkle:releaseNotesLink>
<pubDate>Mon, 05 Oct 2015 19:20:11 +0000</pubDate>
<enclosure url="https://example.com/downloads/app.zip.or.dmg.or.tar.etc"
sparkle:edSignature="7cLALFUHSwvEJWSkV8aMreoBe4fhRa4FncC5NoThKxwThL6FDR7hTiPJh1fo2uagnPogisnQsgFgq6mGkt2RBw=="
length="1623481"
type="application/octet-stream" />
</item>
Test your update, and you’re done!
Note on sparkle:version
: Our previous documentation used to recommend specifying sparkle:version
(and sparkle:shortVersionString
) as part of the enclosure
item. While this works fine, for overall consistency we now recommend specifying them as top level items instead as shown here. This is supported across all versions of Sparkle.
If your app is large, or if you’re updating primarily only a small part of it, you may find delta updates useful: they allow your users to only download the bits that have changed. The generate_appcast
tool automatically generates delta updates.
If you use internal build numbers for your CFBundleVersion
key and a human-readable CFBundleShortVersionString
, you can make Sparkle hide the internal version from your users.
Set the sparkle:version
element in your item to the internal, machine-readable version (ie: “1248”). Then set a sparkle:shortVersionString
element on in your item to the human-readable version (ie: “1.5.1”). For example:
<item>
<title>Version 1.5.1 (bug fixes)</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>1248</sparkle:version>
<sparkle:shortVersionString>1.5.1</sparkle:shortVersionString>
<sparkle:releaseNotesLink>
https://example.com/release_notes/app_2.0.html
</sparkle:releaseNotesLink>
<pubDate>Mon, 05 Oct 2015 19:20:11 +0000</pubDate>
<enclosure url="https://example.com/downloads/app.zip.or.dmg.or.tar.etc"
sparkle:edSignature="7cLALFUHSwvEJWSkV8aMreoBe4fhRa4FncC5NoThKxwThL6FDR7hTiPJh1fo2uagnPogisnQsgFgq6mGkt2RBw=="
length="1623481"
type="application/octet-stream" />
</item>
Note that the internal version number (CFBundleVersion
and sparkle:version
) is intended to be machine-readable and is not generally suitable for formatted text or git changeset IDs.
If an update to your application raises the required version of macOS, you can restrict that update to qualified users.
Add a sparkle:minimumSystemVersion
child to the <item>
in question specifying the required system version, such as “10.13.0” (be sure to specify a three-part version in form of major.minor.patch):
<item>
<title>Version 2.0 (2 bugs fixed; 3 new features)</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.0</sparkle:version>
<sparkle:minimumSystemVersion>10.13.0</sparkle:minimumSystemVersion>
</item>
Note that Sparkle 2.3 or later only works with macOS 10.13 or later (macOS 10.11 or later for Sparkle 2.2.2 and macOS 10.9 or later for Sparkle 1), so that’s the lowest minimum version you can use.
Sparkle also supports a sparkle:maximumSystemVersion
element that can limit the maximum system version similarly.
Please note that if your application is built using the macOS 10.15 SDK or earlier, the system may report its operating system as 10.16.0 for compatibility reasons. To minimize issues, we recommend building your application with an up to date Xcode and SDK.
Additionally in Sparkle 2, if the user checks for new updates manually and the cannot update because of an operating system requirement, the standard updater alert will inform the user their operating system is incompatible and provide them an option to visit your website specified by the <link>
element.
If an update to your application is a major or paid upgrade, you may want to prevent the update from being installed automatically.
Add a sparkle:minimumAutoupdateVersion
child to the <item>
in question specifying the major update’s <CFBundleVersion>
, such as “2.0”:
<item>
<title>Version 2.1 (2 bugs fixed; 3 new features)</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.1</sparkle:version>
<sparkle:minimumAutoupdateVersion>2.0</sparkle:minimumAutoupdateVersion>
</item>
If this value is set, it indicates the lowest version that can automatically update to the version referenced by the appcast (i.e. without showing the update available GUI). Apps with a lower CFBundleVersion
will always see the update available GUI, regardless of their SUAutomaticallyUpdate
user defaults setting.
Additionally in Sparkle 2:
sparkle:minimumAutoupdateVersion
set to 2.0), but 1.9.4 is available then 1.9.4 will be offered first.From Sparkle 2.1 onwards, a developer can publish a new update specifying sparkle:ignoreSkippedUpgradesBelowVersion
to ignore skipped major upgrades below a specific version. For example, if a user skips a major upgrade for version 2.0, this may also skip version 2.1 in the future. However version 2.2 may be a special release that may contain new noteworthy features, so a developer may want to re-notify users that have skipped 2.0 through 2.1 like so (read from bottom to top):
<!-- If the user skips 2.0 or 2.1, alerts for this update will not be skipped. However if the user skips 2.2, alerts for this 2.2.1 update will be skipped -->
<item>
<title>Version 2.2.1</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.2.1</sparkle:version>
<sparkle:minimumAutoupdateVersion>2.0</sparkle:minimumAutoupdateVersion>
<sparkle:ignoreSkippedUpgradesBelowVersion>2.2</sparkle:ignoreSkippedUpgradesBelowVersion>
</item>
<!-- If the user skipped 2.0 or 2.1, alerts for 2.2 will not be skipped due to sparkle:ignoreSkippedUpgradesBelowVersion requirement -->
<item>
<title>Version 2.2</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.2</sparkle:version>
<sparkle:minimumAutoupdateVersion>2.0</sparkle:minimumAutoupdateVersion>
<sparkle:ignoreSkippedUpgradesBelowVersion>2.2</sparkle:ignoreSkippedUpgradesBelowVersion>
</item>
<item>
<title>Version 2.1</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.1</sparkle:version>
<sparkle:minimumAutoupdateVersion>2.0</sparkle:minimumAutoupdateVersion>
</item>
<!-- Skipping this upgrade will also skip alerts for 2.1 above because it has the same sparkle:minimumAutoupdateVersion -->
<item>
<title>Version 2.0</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.0</sparkle:version>
<sparkle:minimumAutoupdateVersion>2.0</sparkle:minimumAutoupdateVersion>
</item>
If you want to provide a download link, instead of having Sparkle download and install the update itself, you can omit the <enclosure>
tag and add <sparkle:version>
and <link>
tags. For example:
<item>
<title>Version 1.2.4</title>
<sparkle:version>1.2.4</sparkle:version>
<sparkle:releaseNotesLink>https://example.com/release_notes_test.html</sparkle:releaseNotesLink>
<pubDate>Mon, 28 Jan 2013 14:30:00 +0500</pubDate>
<link>https://myproductwebsite.com</link>
</item>
Apps that use Sparkle 2 can use the new <sparkle:informationalUpdate>
tag instead of omitting the <enclosure>
tag. This also allows specifying which application versions should see an update as only informational:
<item>
<title>Version 1.2.4</title>
<sparkle:version>1.2.4</sparkle:version>
<sparkle:informationalUpdate>
<sparkle:version>1.2.3</sparkle:version>
</sparkle:informationalUpdate>
<sparkle:releaseNotesLink>https://example.com/release_notes_test.html</sparkle:releaseNotesLink>
<pubDate>Mon, 28 Jan 2013 14:30:00 +0500</pubDate>
<link>https://myproductwebsite.com</link>
<enclosure url="https://example.com/downloads/app.zip.or.dmg.or.tar.etc"
sparkle:edSignature="7cLALFUHSwvEJWSkV8aMreoBe4fhRa4FncC5NoThKxwThL6FDR7hTiPJh1fo2uagnPogisnQsgFgq6mGkt2RBw=="
length="1623481"
type="application/octet-stream" />
</item>
In this example, because <sparkle:version>1.2.3</sparkle:version>
is specified in <sparkle:informationalUpdate>
, only version 1.2.3
will see this update as an informational one with a download link. Other versions of the application will see this as an update they can install from inside the application. You can add more children to specify, for example, that version 1.2.2 should also see the update as informational. If you do not specify any children to <sparkle:informationalUpdate>
, then the update is informational to all versions.
In Sparkle 2.1 onwards, <sparkle:belowVersion>
can be used to specify that versions below a specific version should see the update as an informational one. For example, the below snippet says that all versions below 1.0 should treat this update as informational:
<sparkle:informationalUpdate>
<sparkle:belowVersion>1.0</sparkle:belowVersion>
</sparkle:informationalUpdate>
Updates that are marked critical are shown to the user more promptly and do not let the user skip them. You can use the <sparkle:criticalUpdate>
tag. For example, if version 1.2.4 is a critical update:
<item>
<title>Version 1.2.4</title>
<sparkle:version>1.2.4</sparkle:version>
<sparkle:releaseNotesLink>https://example.com/release_notes_test.html</sparkle:releaseNotesLink>
<pubDate>Mon, 28 Jan 2013 14:30:00 +0500</pubDate>
<link>https://myproductwebsite.com</link>
<sparkle:tags>
<sparkle:criticalUpdate></sparkle:criticalUpdate>
</sparkle:tags>
<enclosure url="https://example.com/downloads/app.zip.or.dmg.or.tar.etc"
sparkle:edSignature="7cLALFUHSwvEJWSkV8aMreoBe4fhRa4FncC5NoThKxwThL6FDR7hTiPJh1fo2uagnPogisnQsgFgq6mGkt2RBw=="
length="1623481"
type="application/octet-stream" />
</item>
Apps that use Sparkle 2 can use the newer <sparkle:criticalUpdate>
tag that is a top-level element and not placed within <sparkle:tags>
. Additionally, the version that was last critical can be specified. For example, when 1.2.5 is released you can specify that only versions less than 1.2.4 should treat this as a critical update:
<item>
<title>Version 1.2.5</title>
<sparkle:version>1.2.5</sparkle:version>
<sparkle:releaseNotesLink>https://example.com/release_notes_test.html</sparkle:releaseNotesLink>
<pubDate>Mon, 28 Jan 2013 14:30:00 +0500</pubDate>
<link>https://myproductwebsite.com</link>
<sparkle:criticalUpdate sparkle:version="1.2.4"></sparkle:criticalUpdate>
<enclosure url="https://example.com/downloads/app.zip.or.dmg.or.tar.etc"
sparkle:edSignature="7cLALFUHSwvEJWSkV8aMreoBe4fhRa4FncC5NoThKxwThL6FDR7hTiPJh1fo2uagnPogisnQsgFgq6mGkt2RBw=="
length="1623481"
type="application/octet-stream" />
</item>
Phased group rollouts allows distributing your update to users in different groups and phases. For example on the first day an update is posted, one group of your users may be notified of a new update, but the second group will only be notified of the update on the second day. And so on. This allows posting updates out in the field more gradually.
This feature requires:
<pubDate>
tag in your update item. The E, dd MMM yyyy HH:mm:ss Z
date format will work.<sparkle:phasedRolloutInterval>
tag with the specified phased rollout interval between groups in seconds.For example:
<item>
<title>Version 1.2.5</title>
<sparkle:version>1.2.5</sparkle:version>
<sparkle:releaseNotesLink>https://example.com/release_notes_test.html</sparkle:releaseNotesLink>
<pubDate>Mon, 28 Jan 2013 14:30:00 +0500</pubDate>
<link>https://myproductwebsite.com</link>
<sparkle:phasedRolloutInterval>86400</sparkle:phasedRolloutInterval>
<enclosure url="https://example.com/downloads/app.zip.or.dmg.or.tar.etc"
sparkle:edSignature="7cLALFUHSwvEJWSkV8aMreoBe4fhRa4FncC5NoThKxwThL6FDR7hTiPJh1fo2uagnPogisnQsgFgq6mGkt2RBw=="
length="1623481"
type="application/octet-stream" />
</item>
This update item specifies the phasedRolloutInterval
as 86400 seconds, or 1 day. This means that there is 1 day of interval between groups. Sparkle hardcodes the number of groups to 7, so this means that the update will be rolled out completely in 7 days and an update will be rolled out to a new group each day after the pubDate
.
Sparkle generates a random group ID in the application’s user defaults using the SUUpdateGroupIdentifier
key. This ID is sometimes re-generated (upon downloading an update) and is not transmitted to servers.
Phased group rollouts do not take effect for updates that are marked critical or for when the user manually checks for new updates.
Sparkle 2 provides specifying what channel an update is on. Examples of channels may include beta updates or updates that are staged and not ready for production. By default, updaters only look for updates that are on the default channel.
Updates are posted on the default channel unless specified otherwise. This example shows an update posted on the “beta” channel:
<item>
<title>Version 2.0 (Beta 1)</title>
<sparkle:channel>beta</sparkle:channel>
<sparkle:version>20001</sparkle:version>
<sparkle:shortVersionString>2.0b1</sparkle:shortVersionString>
</item>
Only updaters that are allowed to look in the beta channel can find this update. An updater may use -[SPUUpdaterDelegate allowedChannelsForUpdater:] to find this update in addition to updates that are on the default channel:
func allowedChannels(for updater: SPUUpdater) -> Set<String> {
return Set(["beta"])
}
Note that an updater cannot exclude itself from the default channel. Channels are intended to be a way to branch off updates to your application until newer updates are ready to come back to the default channel. On the other hand, channels are not intended to be used for parallel releases (such as distributing “1.0” and “1.0 demo” for example).
Channels can only be used when all of your users downloading your appcast are running a version of Sparkle that supports them. Sparkle 2 added them as of June 27, 2021. You can expedite this process by switching to a new appcast. Otherwise older versions of Sparkle may need to set the feed url programmatically below.
If you were using -setFeedURL:
for alternate feeds and want to migrate to using channels, please see important notes below in setting the feed url programmatically on how to migrate away from using that API.
The appcast feed URL can be changed programmatically at runtime. If the feed URL is always static and the same, please set it in the Info.plist with the SUFeedURL
key instead. Even if you have a secondary appcast feed, we recommend keeping a default one specified in the application’s Info.plist.
If you want to set the feed programmatically to provide beta/nightly updates, please try to adopt channels in the future instead. If you are supporting older Sparkle versions, continue reading on.
The recommended way to change the feed URL programmatically is using -[SUUpdaterDelegate feedURLStringForUpdater:]
or -[SPUUpdaterDelegate feedURLStringForUpdater:] in Sparkle 2.
Here is an example:
func feedURLString(for updater: SPUUpdater) -> String? {
return UserDefaults.standard.bool(forKey: "beta") ? BETA_FEED_URL_STRING : STABLE_FEED_URL_STRING
}
Sparkle also has -setFeedURL:
or feedURL
property on the updater which is deprecated and not recommended using due to race conditions and user defaults permanence. To migrate away from using this API, you should use -[SPUUpdater clearFeedURLFromUserDefaults] immediately after starting the updater in Sparkle 2.4 or later. This will clear out any custom feed you have previously set in the bundles’s user defaults, so Sparkle doesn’t automatically read it later. For older versions of Sparkle, you should instead set the feed URL to nil
using -setFeedURL:
.
To make use of Sparkle’s newest features (such as channels or dropping DSA signatures), you may not be able to adopt these features unless all your users are using a recent version of Sparkle.
One approach is just waiting until the majority of your users are running a recent version of Sparkle and your application.
A second approach is migrating to a new SUFeedURL
in your new application’s Info.plist
. This ensures that the application versions using your newer appcast have access to newer features.
Instead of linking external release notes using the <sparkle:releaseNotesLink>
element, you can also embed the release notes directly in the appcast item, inside a <description>
element. If you wrap it in <![CDATA[ ... ]]>
, you can use unescaped HTML.
<item>
<title>Version 2.0 (2 bugs fixed; 3 new features)</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.0</sparkle:version>
<description><![CDATA[
<h2>New Features</h2>
...
]]>
</description>
...
</item>
You can embed just marked up text (it’ll be displayed using standard system font), or a full document with <!DOCTYPE html><style>
, etc.
In Sparkle 2.4 or later, you can also embed plain text release notes using <description sparkle:format="plain-text">
.
In Sparkle 2, the <sparkle:fullReleaseNotesLink>
element may be used to specify the full release notes, or version history, link to your product. When the user checks for updates and no new updates are available, Sparkle may let the user open this link in their web browser. If this element is not specified, Sparkle will default to using the <sparkle:releaseNotesLink>
element instead if present, which may be version specific. Full release notes can also be used if your application uses embedded release notes.
<item>
<title>Version 2.0 (2 bugs fixed; 3 new features)</title>
<link>https://myproductwebsite.com</link>
<sparkle:version>2.0</sparkle:version>
<sparkle:releaseNotesLink>https://myproductwebsite.com/app/2.0.html</sparkle:releaseNotesLink>
<sparkle:fullReleaseNotesLink>https://myproductwebsite.com/app/full-history/</sparkle:fullReleaseNotesLink>
</item>
Alternatively, an application that uses Sparkle’s standard user interface may implement -[SPUStandardUserDriverDelegate standardUserDriverShowVersionHistoryForAppcastItem:] to show full offline or in-app release notes to the user.
When displaying HTML release notes, Sparkle 2.5 and later automatically adds a sparkle-installed-version
class to certain elements, based on the user’s currently installed app version. This is useful for highlighting changes relevant to the user: you can use CSS to de-emphasize already-installed releases.
Sparkle looks for elements with a sparkle-version
data attribute whose value exactly matches the installed app’s CFBundleVersion
. All matching elements receive the sparkle-installed-version
class.
For example, these three div elements are eligible:
<div data-sparkle-version="3">
Version 1.2: Update icon
</div>
<div data-sparkle-version="2">
Version 1.1: Fix list
</div>
<div data-sparkle-version="1">
Version 1.0: Everything is new
</div>
You can make use of this injected class by styling the marked element, or elements around it.
For instance, the following CSS will make the installed version <div>
semi-transparent, and completely hide previous versions:
div.sparkle-installed-version {
opacity: 0.5;
}
div.sparkle-installed-version ~ div {
display: none;
}
Note this feature is not supported if your application uses the Downloader XPC Service.
You can provide additional release notes for localization purposes. For instance:
<sparkle:releaseNotesLink xml:lang="en">https://example.com/app/2.0.en.html</sparkle:releaseNotesLink>
<sparkle:releaseNotesLink xml:lang="de">https://example.com/app/2.0.de.html</sparkle:releaseNotesLink>
Use the xml:lang
attribute with the appropriate two-letter country code for each localization. You can also use this attribute with the <description>
tag.
You can define your own top level elements in the appcast item using your own custom XML namespace (specified by xmlns
). Here is an example defining xmlns:myapp
namespace and creating a myapp:messageOfTheDay
element:
<?xml version="1.0" encoding="utf-8"?>
<rss version="2.0" xmlns:sparkle="http://www.andymatuschak.org/xml-namespaces/sparkle" xmlns:myapp="https://myproductpage.com/" xmlns:dc="http://purl.org/dc/elements/1.1/">
<channel>
<title>My App Changelog</title>
<language>en</language>
<item>
<title>Version 2.0</title>
<myapp:messageOfTheDay>Tip: you can do X by using Y</myapp:messageOfTheDay>
<!-- The rest of the feed item -->
</item>
</channel>
</rss>
In Sparkle 2, one place to parse this custom property is in -[SPUUpdaterDelegate updaterDidNotFindUpdate:error:]. Here’s an example:
func updaterDidNotFindUpdate(_ updater: SPUUpdater, error: Error) {
let userInfo = (error as NSError).userInfo
guard
let latestUpdateItem = userInfo[SPULatestAppcastItemFoundKey] as? SUAppcastItem,
let userInitiated = userInfo[SPUNoUpdateFoundUserInitiatedKey] as? Bool,
let reasonValue = userInfo[SPUNoUpdateFoundReasonKey] as? OSStatus,
let reason = SPUNoUpdateFoundReason(rawValue: reasonValue),
let messageOfTheDay = latestUpdateItem.propertiesDictionary["myapp:messageOfTheDay"] as? String else {
return
}
if !userInitiated &&
!latestUpdateItem.isMajorUpgrade &&
!latestUpdateItem.isCriticalUpdate &&
(reason == .onLatestVersion || reason == .onNewerThanLatestVersion) {
print(messageOfTheDay)
}
}
Sparkle is available for other platforms and operating systems. Check out WinSparkle and NetSparkle.
We recommend using separate appcast feeds for macOS and Windows versions of your app.
While there is partial support for having macOS and non-macOS items in the same appcast feed (using a sparkle:os
attribute in a sparkle:enclosure
element), it is not recommended.
A more formal specification of Sparkle’s appcast item properties can be found in the SUAppcastItem API Reference.