nwup is an updater for packaged NW.js/node-webkit applications. It allows the application code to be updated separately from the NW.js runtime, leading to much smaller update file sizes. There's also a grunt plugin that helps create updates that can be used with nwup.
This is still in early development, currently only supports Windows applications, though cross-platform OSX/Linux support is planned.
nwup reverses the application packaging process to extract the original NW.js runtime from the packaged application. It then merges the update and runtime, repackaging into a new updated executable.
nw.exe + app.zip -> app.exe
app.exe -> nw.exe + app.zip
nw.exe + update.zip -> app.updated.exe
For more details see the usage section.
nwup assumes the packaged executable was created by concatenating the nw.exe runtime with the zipped application as described in the NW.js wiki: either by using node-webkit-builder or manually.
nwup needs some configuration information to function, and it looks for them in the application's package.json
manifest. It checks for an nwup
field, and extracts the configuration from the following subfields:
(string) [required] URL pointing to the update manifest JSON.
(int) [required] the size of the NW.js runtime before packaging.
(string) [optional] flag to determine if running in a development or production enviroment. Set to DEVEL
or PROD
. If mode is set to DEVEL
then the update process will exit without updating to avoid modifying the unpackaged NW.js runtime.
{
"name": "myapp",
"version": "0.0.2",
"main": "index.html",
...
"nwup": {
"updateManifest": "https://www.example.com/updates/latest.json",
"mode": "PROD",
"runtimesize": 61049344
}
}
When checking for updates nwup will make a request to the URL defined in the application's manifest [under the updateManifest
field]. It expects a list of fields in JSON format, with version
and update
being required. Any extra fields provided there will be made available to the application [for use as a changelog, file size, additional info, etc].The following fields are required:
(string) [required] the latest version available, if this is newer than the application's current version then the application needs to be updated. Versioning should follow the semver format.
(string) [required] URL pointing to the latest update zip file.
{
"version": "1.1.0",
"update": "https://www.example.com/updates/2.1.0.zip",
"changelog": "Some additional info here that will be ignored by nwup, but passed along to the application."
}
npm install nwup --save
The update lifecycle is as follows:
- Check if an update is available.
- Download the update.
- Restart the application to apply the update.
- Check if an update is in progress.
- Restart the application to complete the update.
- Check if an update was completed and the temporary files need to be deleted.
- Delete temporary update files.
nwup provides a method that takes care of each of these steps:
-
checkForUpdate()
Makes a request to the remote URL defined in the app manifest under
nwup.updateManifest
. Compares the version number in the remote manifest with the app's current version. -
downloadUpdate()
Extracts the NW.js runtime from the application [this is done by reading out the first n bytes where n is the size of the NW.js runtime determined before packaging]. Downloads the update zip from the remote server at the location provided in updateLocation. Merges the update with the NW.js runtime to create a new executable [update.exe] which contains the updated version.
-
applyUpdate(updatePath)
Flags the application into update mode [isUpdating() will return true], stores the path to the the current executable, closes the current application, and opens the new application determined by updatePath.
-
isUpdating()
When nwup downloads an update and starts updating, it will need to restart the app to apply those updates. On startup apps should call isUpdating() to check if the update has not yet finished, and call completeUpdate() to finish the process and apply the updates.
-
completeUpdate()
Completes the update by overwriting the old app with the new updated executable. Flags the application into clean mode [needsCleaning() will return true], closes the current application [updater], and opens the new updated application in the original path.
-
needsCleaning()
On startup apps should call needsCleaning() to check if the temp files still need to be deleted, and call clean() to delete them and end the update lifecycle.
-
clean()
Deletes the temporary update files that are no longer needed.
A simplified example:
var nwup = require('nwup');
var updater = new nwup();
// Check if an update is in progress.
if (updater.isUpdating()) {
// Restart the application to complete the update.
setTimeout(function() {updater.completeUpdate();}, 2000);
} else {
// Check if an update was completed and the temporary files need to be deleted.
if (updater.needsCleaning()) {
// Delete temporary update files.
updater.clean()
}
// Check if an update is available.
updater.checkForUpdate()
.then(function(payload) {
if (payload.updateAvailable) {
// Download the update.
updater.downloadUpdate(payload.updateInformation.update)
.then(function (updatePath) {
// Restart the application to apply the update.
updater.applyUpdate(updatePath);
}, function(error) {
throw(error);
}, function(progress) {
// Track the download progress
if (progress.totalSize) {
console.log('update size', progress.totalSize);
}
if (progress.receivedSize) {
console.log('amount dled', progress.receivedSize);
}
})
.done();
}
});
Typically an application would allow for user input in the process, mainly:
- after step 1: Prompt the user to download the update.
- after setp 2: Prompt the user to apply the update.
- after step 4: Display 'updating' text to the user.
- after step 7: Display 'update complete' notice.
To fulfill all the requirements listed above, every time a new version is released you would need the following:
- The update files in a zipped archive [all your application files zipped up together, without the NW.js runtime/dlls]
-
nwup
fields in the application's manifest. - An update manifest at the nwup.updateManifest location.
The grunt-nwup plugin can be integrated in your project to automate the process.