Testing Certificate Chains in Go
Testing Certificate Chains in Go
After seeing how Go helps test certificate chains, you might just be a Gopher too! Read about the benefits, then check out the open source code on GitHub.
Join the DZone community and get the full member experience.Join For Free
This talk was derived from an opensource.com article I wrote in April 2017, Golang to the rescue: Saving DevOps from TLS turmoil. Presented at GopherCon 2017 as a Lightning Talk. Source for the talk is available on GitHub.
I'm many things. But most importantly, I'm a Gopher.
Not Too Long Ago in a Place of Work Far, Far Away...
- My team of merry DevOps'ers inherited an application.
- A third-party built the app a few years ago.
- The app had long been abandoned.
- Before we could do any re-engineering work, we had to resolve a critical issue.
- The certificates were about to expire!
- Oh! And the only environment this application was in was production.
- And there was no time to implement a new key management system.
- And it was a pet project of someone in senior leadership.
Let's Talk Certificate Chains
2 Chainz (we can talk rap music later).
- Let's talk certificate chains for a minute.
- HTTPS, SSL... it's all TLS, right?
- In my opinion, if you're using TLS, you must have a rock solid configuration.
- This means you have to include the certificate chain in the correct order.
- This is no longer optional in the post-Heartbleed world.
- The Internets are watching.
This Is the Goal
- This is the goal.
- If you are going to bother to encrypt your traffic, you better do it right.
- This is what we're aiming for; an A+.
- At this company, we obtained certs from a preferred vendor that our company was cool with.
- I prefer Let's Encrypt, but some companies aren't comfortable with that yet for various reasons.
- The process goes like this...
- You generate your CSR and private key.
- You send the CSR to the vendor.
- The certificate arrives but usually doesn't have an intermediate key in chain because... vendors be vendoring.
- No big deal.
- Let's go to the vendor's documentation...
- And OMG...
- The vendor docs are terrible.
- This is when you learn...
- Cryptography is hard, but implementing cryptographic best practices might be even harder.
What do we do? Look at statistical probabilities and start shuffling keys around? The series of games you have to play with OpenSSL or Nginx or some other method aren't intuitive. Do you know how hard this is to explain to people?
So What Does Any Good Engineer Do?
- We needed a tool that would fail if the certificate chain provided was incorrect.
- We wanted a lightweight tool that could be publicly accessible.
- Conducting a third-party analysis of the certificates and configuration was also a requirement.
- There were no tools that I could find meeting these needs, so I decided to build my own.
- With Go of course!
Three Go Packages: Log
- The go
logpackage is pretty self-explanatory: a package that enables logging.
- We needed a spectacular failure at the sign of trouble.
loghas three helper functions:
- Output from the package goes to
- We used a
fatalerror to get the web server to stop and log any issue.
- Let me just say, the Go standard library is amazing!
logis designed beautifully.
fatalto break the app and log to
stderrif something isn't right is great!
Three Go Packages: Crypto/TLS
- The Go
crypto/tlspackage partially implements TLS 1.2, as specified in RFC 5246.
- The package configures usable SSL/TLS versions and identifies preferred cipher suites and elliptic curves used during handshakes.
- This is the package that handles connections securely.
- The crypto/tls package is a splendid implementation of the RFC.
- "It just works."
Three Go Packages: Net/HTTP
- This is the Go implementation of HTTP.
net/httphas a function called
- ListenAndServeTLS provides the desired certificate checking functionality.
- "If the certificate is signed by a certificate authority, the certFile should be the concatenation of the server's certificate, any intermediates, and the CA's certificate."
ListenAndServeTLSfunction and it's awesome.
- It fails if your certs aren't up to snuff or ordered properly.
- It helps us Gophers out immensely by enforcing best practices.
Main: Mux, Cfg, Srv
- Code creates a
mux, short for HTTP request multiplexer.
- I ❤️ multiplexers (it's a long story that involves analog signals).
muxhas a function that creates an HTTP server with headers and content (Hello World!).
cfgbrings in all the TLS bits seen in a solid web server config.
srvputs the pieces together and defines what port to listen on.
- I ❤️ DevOps.
- I embrace failure.
log.Fatal(srv.ListenAndServeTLS("/etc/ssl-tester/tls.crt","/etc/ssl-tester/tls.key")defines the path of certificate files to use.
- It also logs a fatal error if the certificate is not valid.
- It fails fast.
- I love DevOps and I embrace failure.
- The code allows us to fail quickly if the certificates aren't in accordance with RFC.
- Stuff in the standard library JUST WORKS.
It's Open Source!
- The code is open-sourced.
- Check it out on github.com/chris-short/ssl-tester.
- Throw a star my way if you feel like it.
- Yes, it works.
- Yes, you can access it right now!
- https://ssl-tester.chrisshort.net redirects you to a reference implementation
No, It Really Works!
- You can even scan it with external tools!
- The Go code does exactly what I need it to do and nothing more.
- About 40 lines of code!!! I ❤️ Go!
- Binary is a self-contained web server.
- Less than 6MB!!! I ❤️ Go!
- It can be safely deployed to any public server.
- External testing runs against it for extra vetting.
- The tool does exactly what I need it to do and nothing more.
- It fails when the certificate chain provided is incorrect.
- It's lightweight and publicly accessible.
- I'm able to test via third-parties.
- It's a tiny, single binary.
Published at DZone with permission of Chris Short . See the original article here.
Opinions expressed by DZone contributors are their own.