So you want to sign for Windows?

By Kyle Wenholz
January 4, 2023
By Kyle Wenholz
January 4, 2023
When you package executables for Windows, you have two options: Code Signing or Wishful Thinking. If you go the Wishful Thinking route, you get this warning:
At best, it’s an intimidating message your end user can bypass. At worst, corporate IT policy blocks it completely and they can’t run your program. We don’t want that.
ngrok publishes its agent (get it here!) to a number of platforms, Windows included. One of the nice security features on Windows is code signing. Code signing uses certificates distributed by reputable businesses to establish trust of the software you install and run on your computer. We recently started signing the ngrok agent to improve trust in our platform.
Unfortunately, the documentation around Code Signing is scattered, vague, and often out of date (even on those reputable business’ websites). You’ll probably find some articles or Stack Overflow posts with details, but they’ll be from a decade back and sound like black magic at first. We celebrated when we figured it out and automated the code signing process but we also want to make it easier for the next teams. These resources are particularly geared toward *nix users publishing for Windows.
We’re signing an executable built with go, but these instructions will generally work with other languages. We will
A number of companies exist to verify you/your business and provide a code signing certificate. This certificate is one of part of a chain where the root is owned by the certificate authority. Often, that root is trusted by your end users’ machines already, so when they go to use your signed software, the transitive trust provides the experience you want: no security popups.
We chose DigiCert Code Signing Certificates. Some of our folks had past experiences using them, and a quick survey of installed software on a few of our computers pointed to DigiCert as a common root authority. Given we’re developers and our customers are developers, we figured it likely our users also had the DigiCert root trusted on their machines.
Purchasing a signing certificate from DigiCert is pretty straightforward as you follow their instructions.
Three notes:
NOTE: This step is optional but needs to be done before you sign your executables. Skip to Signing if you don’t want to add metadata.
We use go-winres to add metadata, like a nice favicon, to our executable. If doing this step, do it while building your executable (varies depending on language toolchain). There are versions of this command using a configuration file, but our build pipeline prefers the full command for clarity:
go-winres simply \\
--arch “amd64,386” \\
--product-version yourversion \\
--manifest none \\
--file-description "A description of your executable." \\
--product-name "yourname” \\
--icon youricon.ico \\
// The rsrc extension gets picked up by go’s build tooling but is also a standard format many toolchains work with \\
--out yourexecutable.rsrc
As long as that rsrc file is where your go source code is, go build will automatically include the metadata in your executable.
Before:
After:
Once you’ve purchased your code signing certificate, you’ll be able to download it in a number of formats. On your certificate page, something like https://www.digicert.com/secure/orders/SOME_ID, grab the one marked .crt (best for Apache / Linux) and put it on the machine you’ll be using for signing. You’ll want the file titled yourname.crt, rather than the ones with DigiCert in the name. With that file and yourname.key from the CSR above, we’ll use osslsigncode to sign our executable. From the osslsigncode docs:
osslsigncode is based on OpenSSL and cURL, and thus should be able to compile on most platforms where these exist.
We use the following set of arguments to sign, timestamp, and lightly brand:
osslsigncode sign \\
// Signing cert from trusted authority \\
-certs yourname.crt \\
// Private key we created a CSR with for said authority and intend to sign code with \\
-key yourname.key \\
// The display name of the executable \\
-n "yourname" \\
-i yourdomain.com \\
// DigiCert’s time servers \\
-t <http://timestamp.digicert.com> \\
-in yourexecutable.unsigned.exe \\
-out yourexecutable.exe \\
// Hash algorithm used by DigiCert, defaults to sha256 \\
-h sha256
After that, place your executable on a Windows Machine, right click from Explorer, and check out your security settings!
We have just
You should now have a nicely signed and branded executable for use on Windows. Because all of the above was done on the command line, you can easily move it to the release process of your choice.
We strongly suggest your team automates this whole process. We put it in our developer commands for simplicity and making the right incantations the default. Next up for us is to add it to CI and automate releases for nightly, beta, and stable releases.