nwup v0.0.2
nwup
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.
Note:
This is still in early development, currently only supports Windows applications, though cross-platform OSX/Linux support is planned.
How Does it Work?
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.
Packaging
nw.exe + app.zip -> app.exe
Extracting Runtime
app.exe -> nw.exe + app.zip
Updating
nw.exe + update.zip -> app.updated.exe
For more details see the usage section.
Requirements
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.
Manifest
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:
updateManifest
(string) required URL pointing to the update manifest JSON.
runtimesize
(int) required the size of the NW.js runtime before packaging.
mode
(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.
Example:
{
  "name": "myapp",
  "version": "0.0.2",
  "main": "index.html",
  ...
  "nwup": {
    "updateManifest": "https://www.example.com/updates/latest.json",
    "mode": "PROD",
    "runtimesize": 61049344
  }
}Update Manifest
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:
version
(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.
update
(string) required URL pointing to the latest update zip file.
Example:
{
    "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."
}Installation
npm install nwup --saveUsage
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.
Creating Updates
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
- nwupfields 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.