NetSparkle
NetSparkle is a software update framework for C# that is compatible with .NET Core 3+ and .NET Framework 4.5.2+, has pre-built UIs for .NET Framework (WinForms, WPF) and .NET Core (WinForms, WPF, Avalonia), uses Ed25519 or other signatures, and even allows for custom UIs or no UI at all! You provide, somewhere on the internet, an app cast with update and version information, along with release notes in Markdown or HTML format. This library then helps you check for an update, show the user the release notes, and offer to download/install the new version of the software.
The develop branch has changed significantly from master and represents a major 2.0 version update. NetSparkle 2.0, currently in beta, brings the ability to customize most of NetSparkle -- custom UIs are easy, you can have custom app cast downloaders and handlers (e.g. for FTP download or JSON appcasts), and more! No more big changes to the API are planned (but we cannot guarantee it before 2.0 RC), but smaller API changes may occur if bugs are found between now and the official 2.0 release.
Built-in supported update download types:
- Windows -- .exe, .msi, .msp
- macOS -- .zip, .pkg, .dmg
- Linux -- .tar.gz, .deb, .rpm
README and other documentation updates for version 2.0 are in progress. If you have specific questions or need help even after looking at the samples, please file an issue or message me on Gitter. You should be able to implement your own handlers and/or UI for most operations, so things like a custom update process, downloading/parsing JSON rather than XML, downloading things from FTP, using your own file signature verification method, etc. are all now possible. Some extra features aren't built-in out of the box yet, such as JSON app cast feeds -- contributions are welcome and benefit the whole community!
Contributing
Contributions are ALWAYS welcome! If you see a new feature you'd like to add, please open an issue to talk about it first, then open a PR for that implementation. If there's a bug you find, please open a PR with the fix or file an issue! Thank you!! :) You can also join us in our Gitter chat room!
Installing NetSparkle
NetSparkle is available via NuGet. To choose a NuGet package to use:
- Reference the core NetSparkle build if you don't care about having a built-in UI and can manage things yourself
- Choose one of the other packages if you want a built-in UI or want to create your UI based on one of the other UIs
| Package | Use Case | Release | Preview | Downloads |
|---|---|---|---|---|
| NetSparkle | Core package; No built-in UI or 100% custom UI | |||
| WinForms UI (.NET Framework) | NetSparkle with built-in WinForms UI | |||
| WinForms UI (.NET Core) | NetSparkle with built-in WinForms UI | |||
| WPF UI (.NET Framework and Core) | NetSparkle with built-in WPF UI | |||
| Avalonia UI | NetSparkle with built-in Avalonia UI | |||
| Command Line Tools | DSA helper; AppCast generator (incl. Ed25519 helpers) |
All notable changes to this project will be documented in the changelog.
- How Updates Work
- Basic Usage
- Appcast
- Updating from 0.x or 1.x
- SparkleUpdater class
- License
- Requirements
- Other Options
How updates work
A typical software update path for a stereotypical piece of software might look like this:
- Compile application so it can be run on other computers (e.g.
dotnet publish) - Programmer puts app in some sort of installer/zip/etc. for distribution (e.g. InnoSetup for Windows)
- Programmer creates app cast file (see the appcast section of this document for more info on how to create this)
- Programmer uploads files for distribution (installer, app cast file, appcast-file.signature file) to their download site.
- Client opens app and is automatically notified of an available update (or the software otherwise detects there is an update)
- Client chooses to update (or update is downloaded if the software downloads it automatically)
- Update is downloaded and sitting on the user's disk
- User is asked to close the software so the update can run. User closes the software.
- Downloaded file/installer is run (or the update is otherwise performed)
Right now, NetSparkleUpdater does not help you with 1., 2., or 4. "Why not?", you might ask:
-
- We can't compile your application for you since we don't know (or care) how you are compiling or packaging your application! :)
-
- A cross-platform installer package/system would be difficult and may not feel normal to end users, although a system that uses Avalonia could maybe work I suppose (might take a lot of work though and make downloads large!). We do not provide support for getting your installer/distribution ready. To generate your installer/distribution, we recommend the following:
- Windows: InnoSetup or NSIS or WiX
- macOS: If you have a .app to distribute, use dotnet-bundle with create-dmg. If you want an installer, create a .pkg installer with macos-installer-builder (tutorial here), Packages, or your terminal. Otherwise, plop things in a zip file.
- Linux: Use dotnet-packaging to create an rpm, deb, or tar.gz file for your users.
-
- We don't know where your files will live on the internet, so you need to be responsible for uploading these files and putting them online somewhere.
To create your app cast file, see the appcast section of this document.
We are open to contributions that might make the overall install/update process easier for the user. Please file an issue first with your idea before starting work so we can talk about it.
Basic Usage
Please look at the sample projects in this repository for basic, runnable usage samples!! There are samples on using each of the built-in UIs as well as a "do it yourself in your own UI" sample!
_sparkle = new SparkleUpdater(
"http://example.com/appcast.xml", // link to your app cast file
new Ed25519Checker(SecurityMode.Strict, // security mode -- use .Unsafe to ignore all signature checking (NOT recommended!!)
"base_64_public_key") // your base 64 public key -- generate this with the NetSparkleUpdater.Tools AppCastGenerator on any OS
) {
UIFactory = new NetSparkleUpdater.UI.WPF.UIFactory(icon) // or null or choose some other UI factory or build your own!
};
_sparkle.StartLoop(true); // `true` to run an initial check online -- only call StartLoop once for a given SparkleUpdater instance!On the first Application.Idle event, your App Cast XML file will be downloaded, read, and compared to the currently running version. If it has a software update inside, the user will be notified with a little toast notification (if supported by the UI and enabled) or with an update dialog containing your release notes. The user can then ignore the update, ask to be reminded later, or download/install it now.
If you want to check for an update in the background without the user seeing anything, use
_sparkle.CheckForUpdatesQuietly();If you want to have a menu item for the user to check for updates so the user can see the UI while NetSparkle looks for updates, use
_sparkle.CheckForUpdatesAtUserRequest();If you have files that need saving, subscribe to the PreparingToExit event:
_sparkle.PreparingToExit += ((x, cancellable) =>
{
// ask the user to save, whatever else is needed to close down gracefully
});Note that if you do not use a UIFactory, you must use the CloseApplication or CloseApplicationAsync events to close your application; otherwise, your downloaded update file will never be executed/read! The only exception to this is if you want to handle all aspects of installing the update package yourself.
The file that launches your downloaded update executable only waits for 90 seconds before giving up! Make sure that your software closes within 90 seconds of CloseApplication/CloseApplicationAsync being called if you implement those events! If you need an event that can be canceled, such as when the user needs to be asked if it's OK to close (e.g. to save their work), use PreparingForExit or PreparingToExitAsync.
App Cast
NetSparkle uses Sparkle-compatible app casts for the most part. NetSparkle uses sparkle:signature rather than sparkle:dsaSignature so that you can choose how to sign your files/app cast. NetSparkle is compatible with and uses Ed25519 signatures by default, but the framework can handle a different implementation of the ISignatureVerifier class to check different kinds of signatures without a major version bump/update.
Note: if your app has DSA signatures, the app cast generator uses Ed25519 signatures by default starting with preview 2.0.0-20200607001. To transition to Ed25519 signatures, create an update where the software has your new Ed25519 public key and a NEW url for a NEW app cast that uses Ed25519 signatures. Upload this update with an app cast that has DSA signatures so your old DSA-enabled app can download the Ed25519-enabled update. Then, future updates and app casts should all use Ed25519.
Here is a sample app cast:
<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:sparkle="http://www.andymatuschak.org/xml-namespaces/sparkle" version="2.0">
<channel>
<title>NetSparkle Test App</title>
<link>https://netsparkleupdater.github.io/NetSparkle/files/sample-app/appcast.xml</link>
<description>Most recent changes with links to updates.</description>
<language>en</language>
<item>
<title>Version 2.0 (2 bugs fixed; 3 new features)</title>
<sparkle:releaseNotesLink>
https://netsparkleupdater.github.io/NetSparkle/files/sample-app/2.0-release-notes.md
</sparkle:releaseNotesLink>
<pubDate>Thu, 27 Oct 2016 10:30:00 +0000</pubDate>
<enclosure url="https://netsparkleupdater.github.io/NetSparkle/files/sample-app/NetSparkleUpdate.exe"
sparkle:version="2.0"
sparkle:os="windows"
length="12288"
type="application/octet-stream"
sparkle:signature="NSG/eKz9BaTJrRDvKSwYEaOumYpPMtMYRq+vjsNlHqRGku/Ual3EoQ==" />
</item>
</channel>
</rss>NetSparkle reads the <item> tags to determine whether updates are available.
The important tags in each <item> are:
<description>- A description of the update in HTML or Markdown.
- Overrides the
<sparkle:releaseNotesLink>tag.
<sparkle:releaseNotesLink>- The URL to an HTML or Markdown document describing the update.
- If the
<description>tag is present, it will be used instead. - Attributes:
sparkle:signature, optional: the DSA signature of the document; NetSparkle does not check this DSA signature for you unless you setReleaseNotesGrabber.ChecksReleaseNotesSignaturetotrue, but you may manually verify changelog DSA signatures if you like or setReleaseNotesGrabber.ChecksReleaseNotesSignature = truein your UI.
<pubDate>- The date this update was published
<enclosure>- This tag describes the update file that NetSparkle will download.
- Attributes:
url: URL of the update filesparkle:version: machine-readable version number of this updatelength, optional: (not validated) size of the update file in bytestype: ignoredsparkle:signature: DSA signature of the update filesparkle:criticalUpdate, optional: if equal totrueor1, the UI will indicate that this is a critical updatesparkle:os: Operating system for the app cast item. Defaults to Windows if not supplied. For Windows, use "win" or "windows"; for macOS, use "macos" or "osx"; for Linux, use "linux".
By default, you need 2 (DSA) signatures (SecurityMode.Strict):
- One in the enclosure tag for the update file (
sparkle:signature="...") - Another on your web server to secure the actual app cast file. This file must be located at [AppCastURL].signature. In other words, if the app cast URL is http://example.com/awesome-software.xml, you need a valid (DSA) signature for that file at http://example.com/awesome-software.xml.signature.
Ed25519 Signatures
You can generate Ed25519 signatures using the AppCastGenerator tool (from this NuGet package or in the source code here). This tool requires the .NET Core 3.1 Desktop Runtime to be installed. Please see below sections for options and examples on generating the Ed25519 keys and for using them when creating an app cast.
DSA Signatures
DSA signatures are not recommended for 2.0+. They are insecure!
You can still generate these signatures, however, using the DSAHelper tool (from this NuGet package or in the source code here). Key generation only works on Windows because .NET Core 3 does not have the proper implementation to generate DSA keys on macOS/Linux; however, you can get DSA signatures for a file on any platform. If you need to generate a DSA public/private key, please use this tool on Windows like this:
NetSparkle.DSAHelper.exe /genkey_pair
You can use the DSAHelper to get a signature like this:
NetSparkle.DSAHelper.exe /sign_update {YourInstallerPackage.msi} {NetSparkle_PrivateKey_DSA.priv}
How can I make the app cast?
- Use the
AppCastGeneratortool (from this NuGet package or in the source code here) to easily create your app cast file. Available options are described below. - Rig up a script that generates the app cast for you in python or some other language (
string.Formator similar is a wonderful thing). - Or you can just copy/paste the above example app cast into your own file and tweak the signatures/download info yourself, then generate the (DSA) signature for the app cast file manually! :)
App Cast Generator Options
Missing some option you'd like to see? File an issue on this repo or add it yourself and send us a pull request!
--show-examples: Print examples of usage to the console.--help: Show all options and their descriptions.
General Options When Generating App Cast
-a/--appcast-output-directory: Directory in which to write the outputappcast.xmlfile. Example use:-a ./Output-e/--ext: When looking for files to add to the app cast, use the given extension. Defaults toexe. Example use:-e exe-b/--binaries: File path to directory that should be searched through when looking for files to add to the app cast. Defaults to.. Example use:-b my/build/directory-f/--file-extract-version: Whether or not to extract the version of the file from the file's name rather than the file itself. Defaults to false. Use when your files that will be downloaded by NetSparkleUpdater will have the version number in the file name, e.g. "My App 1.3.2.exe". Example use:-f true-o/--os: Operating system that the app cast items belong to. Must be one of the following:windows,mac,linux. Defaults towindows. Example use:-o linux-u/--base-url: Beginning portion of the URL to use for downloads. The file name that will be downloaded will be put after this portion of the URL. Example use:-u https://myawesomecompany.com/downloads-l/--change-log-url: Beginning portion of the URL to use for your change log files. The change log file that will be downloaded will be put after this portion of the URL. If this option is not specified, then the change log data will be put into the app cast itself. Example use:-l https://myawesomecompany.com/changes-p/--change-log-path: Path to the change log files for your software. These are expected to be in markdown format with an extension of.md. The file name of the change log files must contain the version of the software, e.g.MyApp 1.3.2.md. Example use:-p path/to/change/logs-n/--product-name: Product name for your software. Used when setting the title for your app cast and its items. Defaults toApplication. Example use:-n "My Awesome App"-x/--url-prefix-version: Add the version number as a prefix to the file name for the download URL. Defaults to false. For example, if--base-urliswww.example.com/downloads, your version is1.4.2, and your app name isMyApp.exe, your download URL will becomewww.example.com/downloads/1.4.2/MyApp.exe. Example use:-x true.--key-path: Path toNetSparkle_Ed25519.privandNetSparkle_Ed25519.pubfiles, which are your private and public Ed25519 keys for your software updates, respectively. Example use:--key-path my/path/to/keys- If you want to use keys dynamically, you can set the
SPARKLE_PRIVATE_KEYandSPARKLE_PUBLIC_KEYenvironment variables before runninggenerate_appcast. The tool prioritizes environment keys over keys sitting on disk!
- If you want to use keys dynamically, you can set the
Options for Key Generation
--generate-keys: If set, will attempt to generate NEW Ed25519 keys for you. Can be used in conjunction with--key-path. Once keys are successfully (or unsuccessfully) generated, the program ends without generating an app cast. By default, existing keys are not overwritten. This option defaults tofalse.--force: If set totrue, will overwrite existing keys on disk. WARNING: THIS COULD RESULT IN A LOSS OF YOUR PUBLIC AND PRIVATE KEYS. USE WITH CAUTION. DO NOT USE IF YOU DO NOT KNOW WHAT YOU ARE DOING! THIS WILL MAKE NO ATTEMPT TO BACK UP YOUR DATA. This option defaults tofalse. Example use:--generate-keys --force true.--export: Export keys as base 64 strings to the console. Defaults tofalse. Example use:--export true. Output format:
Private Key:
2o34usledjfs0
Public Key:
sdljflase;ru2u3
Options for Generating Signatures Without App Cast
--generate-signature: Generate a signature for a file and output it to the console. Example use:--generate-signature path/to/app/MyApp.exe. Outputs in format:Signature: seljr13412zpdfj.
Options for Verifying Signatures
Note that these options are only for verifying Ed25519 signatures. For DSA signatures, please use the DSAHelper tool. Both of the following options must be used together. You must have keys already generated in order to verify file signatures.
--verify: Path to the file that has a signature you want to verify.--signature: Base 64 signature of the file.
Example use: --verify my/path/MyApp.exe --signature 123l4ijsdfzderu23.
This will return either Signature valid (signature is good!) or Signature invalid (signature does not match file).
App Cast Generator Examples
#### Key Generation
# Generate Ed25519 keys for the first time
generate_appcast.exe --generate-keys
# Store keys in a custom location
generate_appcast.exe --key-path path/to/store/keys
# By default, your Ed25519 signatures are stored on disk in your local
# application data folder in a subdirectory called `netsparkle`.
# If you want to export your keys to the console, you can do:
generate_appcast.exe --export
#### Generate a signature for a binary without creating an app cast:
generate_appcast.exe --generate-signature path/to/binary.exe
#### Verifying Binaries
generate_appcast.exe --verify path/to/binary.exe --signature base_64_signature
#### Using a custom key location:
# If your keys are sitting on disk somewhere
# (`NetSparkle_Ed25519.priv` and `NetSparkle_Ed25519.pub` -- both
# in base 64 and both on disk in the same folder!), you can pass in
# the path to these keys like this:
generate_appcast.exe --key-path path/to/keys/
#### Generating an app cast
# Generate an app cast for Windows executables that are sitting in a
# specific directory
generate_appcast.exe -a directory/for/appcast/output/ -e exe -b directory/with/binaries/ -o windows
# Add change log info to your app cast
generate_appcast.exe -b binary/folder -p change/log/folder
# Customize download URL for binaries and change logs
generate_appcast.exe -b binary/folder -p change/log/folder -u https://example.com/downloads -p https://example.com/downloads/changelogs
# Set your application name for the app cast
generate_appcast.exe -n "My Awesome App" -b binary/folder
# Use file versions in file names, e.g. for apps like "My App 1.2.1.dmg"
generate_appcast.exe -n "macOS version" -o macos -f true -b binary/folder -e dmgUpdating from 0.X or 1.X
This section is still WIP, but major changes include:
- Minimum .NET requirement is now .NET Framework 4.5.2 instead of 4.5.1
- Change of base namespace from
NetSparkletoNetSparkleUpdater Sparklerenamed toSparkleUpdaterfor clarity- The default NetSparkleUpdater/NetSparkle.New package has no built-in UI. Please use one of the NetSparkleUpdater packages with a UI if you want a built-in UI that is provided for you.
- Note that if you do not use a
UIFactory, you must use theCloseApplicationorCloseApplicationAsyncevents to close your application; otherwise, your downloaded update file will never be executed/read! The only exception to this is if you want to handle all aspects of installing the update package yourself.
- Note that if you do not use a
- XML docs are now properly shipped with the code for all public and protected methods rather than being here in this README file
- Enabled build time warnings for functions that need documentation that don't have it
SparkleUpdaterconstructors now require anISignatureVerifierin order to "force" you to choose your signature verification methodSecurityModeis a new enum that defines which signatures are required and which signatures are not required- Added
SecurityMode.OnlyVerifySoftwareDownloadsif you want to verify only software download signatures and don't care about verifying your app cast or release notes
- Added
- UIs are now in different namespaces. If you want to use a UI, you must pass in a
UIFactorythat implementsIUIFactoryand handles showing/handling all user interface elementsSparkleUpdaterno longer holds its ownIconHideReleaseNotes,HideRemindMeLaterButton, andHideSkipButtonare all handled by theUIFactoryobjects
- Added built-in UIs for WPF and Avalonia 0.9
- Localization capabilities are non-functional and are expected to come back in a later version. See this issue. (Contributions are welcome!)
- Most
SparkleUpdaterelements are now configurable. For example, you can implementIAppCastHandlerto implement your own app cast parsing and checking.IAppCastDataDownloaderto implement downloading of your app cast fileIAppCastHandlerto implement your own app cast parsingISignatureVerifierto implement your own download/app cast signature checking. NetSparkleUpdater has built-in DSA and Ed25519 signature verifiers.IUIFactoryto implement your own UIILoggerto implement your own logger class (rather than being forced to subclassLogWriter)Configurationsubclasses now take anIAssemblyAccessorin their constructor(s) in order to define where assembly information is loaded from- Many
SparkleUpdaterfunctions are now virtual and thus more easily overridden for your specific use case
- Samples have been updated and improved
- Sample apps for Avalonia, WinForms, and WPF UIs
- Sample app to demonstrate how to handle events yourself with your own UI
- Many delegates, events, and functions have been renamed, removed, and/or tweaked for clarity and better use
DownloadEventnow has theAppCastItemthat is being downloaded rather than being just the download pathAboutToExitForInstallerRun/AboutToExitForInstallerRunAsynchas been renamed toPreparingToExit/PreparingToExitAsync, respectively- The
UserSkippedVersionevent has been removed. UseUserRespondedToUpdateinstead. - The
RemindMeLaterSelectedevent has been removed. UseUserRespondedToUpdateinstead. - The
FinishedDownloading/DownloadedFileReadyevents have been removed. UseDownloadFinishedinstead. - The
StartedDownloadingevent has been removed and replaced withDownloadStarted - The
DownloadErrorevent has been removed and replaced withDownloadHadError Sparkle.RunUpdateno longer exists. UseSparkleUpdater.InstallUpdateinstead.Sparkle.DownloadPathForAppCastItem->SparkleUpdater.GetDownloadPathForAppCastItemAppCastItem.DownloadDSASignature->AppCastItem.DownloadSignatureSilentModeTypesenum renamed toUserInteractionModeSparkle.SilentModerenamed toSparkle.UserInteractionMode
- By default, the app cast signature file now has a
.signatureextension. The app cast downloader will look for a file with the old.dsasignature if data is not available or found in aappcast.xml.signatureon your server. sparkle:dsaSignatureis nowsparkle:signatureinstead. If nosparkle:signatureis found,sparkle:dsaSignaturewill be used (if available). Ifsparkle:dsaSignatureis not found,sparkle:edSignaturewill be used (if available). This is to give us as much compatibility with old versions ofNetSparkleas well as the macOSSparklelibrary.- An entirely new app cast generator tool is now available for use.
- By default, the app cast generator tool now uses Ed25519 signatures. If you don't want to use files on disk to store your keys, set the
SPARKLE_PRIVATE_KEYandSPARKLE_PUBLIC_KEYenvironment variables before running the app cast generator tool. You can also store these signatures in a custom location with the--key-pathflag.- You can still use DSA signatures via the DSAHelper tool and the
DSACheckerclass. This is not recommended.
- You can still use DSA signatures via the DSAHelper tool and the
- Removed
AssemblyAccessorclass in lieu ofIAssemblyAccessorimplementors - We now rely on Portable.BouncyCastle (BouncyCastle.Crypto.dll) for the ed25519 implementation. This means there is another DLL to reference when you use NetSparkle!
- We now rely on System.Text.Json (netstandard2.0) OR Newtonsoft.Json (.NET Framework 4.5.2) for the JSON items. This means there is another DLL to reference when you use NetSparkle, and it will change depending on if the
System.Text.JsonorNewtonsoft.Jsonitem is used!
FAQ
Am I required to use a UI with NetSparkleUpdater?
Nope. You can just reference the core library and handle everything yourself, including any custom UI. Check out the code samples for an example of doing that!
How can I use NetSparkleUpdater with AppCenter?
- Make sure you've read over the documentation here
- Decide if you want to generate signatures for your files. If so, make sure that works, and then use NetSparkleUpdater as normal.
- If you don't want to generate signatures because you trust your AppCenter builds, use
SecurityMode.Unsafeor the followingIAppCastHandleroverride:
public bool DownloadAndParse()
{
try
{
_logWriter.PrintMessage("Downloading app cast data...");
var appcast = _dataDownloader.DownloadAndGetAppCastData(_castUrl);
if (!string.IsNullOrEmpty(appcast))
{
ParseAppCast(appcast);
return true;
}
}
catch (Exception e)
{
_logWriter.PrintMessage("Error reading app cast {0}: {1} ", _castUrl, e.Message);
}
return false;
}Is reverting your application version supported?
The answer is both yes and no. No, because that is not the default behavior. Yes, because if you use installers for each of your versions, you can use your app cast to see which previous versions are available and download those versions. If your installers are standalone, they should install an old version just fine. Just keep in mind that if you install an old version and then there is a newer version in your app cast, after opening the older software, it will ask them if they want to update to the newer version!
Here's a summary of what you can do:
- Setup your
SparkleUpdaterobject - Call
_updateInfo = await _sparkle.CheckForUpdatesQuietly();(no UI) or_sparkle.CheckForUpdatesAtUserRequest()(shows UI). I would recommend checking quietly because the UI method will always show the latest version. You can always show your own UI. - Look in
_updateInfo.Updatesfor the available versions in your app cast. You can compare it with your currently installed version to see which ones are new and which ones are old. - Call
await _sparkle.InitAndBeginDownload(update);with the update you want to download. The download path is provided in theDownloadFinishedevent. - When it's done downloading, call
_sparkle.InstallUpdate(update, _downloadPath);
The "Handle Events Yourself" sample will be very helpful to you: https://github.com/NetSparkleUpdater/NetSparkle/tree/develop/src/NetSparkle.Samples.HandleEventsYourself
Does this work with Avalonia 0.10?
Right now, the Avalonia UI is compatible with Avalonia 0.9. Please see #122 for details on this issue -- basically, when 0.10 is officially released, we'll update the Avalonia build. For now, you can use your own IUIFactory implementation to fix any issues that come up.
Things aren't working. Help!
Here are some things you can do to figure out how to get your app running:
- Make sure you have enabled and debugged your application thoroughly. A great way to do this is to set
SparkleUpdater.LogWriter = new LogWriter(true)and then watch your console output while debugging. - Look at the NetSparkleUpdater samples by downloading this repo and running the samples. You can even try putting your app cast URL in there and using your public key to debug with the source code!
- Ask for help in our Gitter
- Post an issue and wait for someone to respond with assistance
Are you accepting contributions?
Yes! Please help us make this library awesome!
License
NetSparkle is available under the MIT License.
Requirements
- .NET Framework 4.5.2+ OR .NET Core 3+
Acknowledgements
- The original NetSparkle library, found at dei79/netsparkle
- A function for finding the base directory was taken from MIT-licensed WalletWasabi
- MarkdownSharp is from here
Other Options
An incomplete list of other projects related to software updating: