security
https
hsts
The first time I came upon HSTS or whatever it was supposed to mean was when I was trying to be a developer like everyone else - improving our product’s Lighthouse score. We then later concluded that Lighthouse only gives us the basic few requirements and we needed something more muscled up to check for other vulnerabilities along with basic performance. This is also how we ended up updating our third-party CSS and JS libraries but that is not the point we are discussing today.
We had a little warning that we did not have a specific HSTS header set. Well I hadn’t heard of it before (a point to note here would be that it is because of my lack of exposure to security headers in general). And then I did, again, wha any other developer would - Google the hell out of it!
This was quite some time back, but I’m revisiting this to oil the information on it. I tend to forget stuff if I don’t write it down. And also, who knows, it might help someone understand something! That’s always a good thing to have.
🛡️ It all starts with HTTPS
Alright alright, I know the protagonist of this blog is HSTS, and not HTTPS. But it is impossible to talk about HSTS wihout tagging its buddy HTTPS along in the conversation. We will not go into the detais of how it works, but instead of why it matters and its importance in the way the web works. If you’re familiar with https, the next section might be your starting point.
There’s http and there’s https. Both are the same protocols with the difference of an ’s’ between them. Literally, and also conceptually. It is a what-you-see-is-what-you-get kinda thing if you want to be funny about it. The “s” is for security. HTTPS is a secure version of HTTP. Secure in what sense, you ask?
While on the internet, we transfer A LOT of data. We keep on sending requests, receiving responses for every information, for every page. And it would be much easier if the only data we were trading were cat pictures (which ultimately we’d want to share with as many people as possible), but it’s not. We trade details like login details, bank informaion, and so much more. These are sensitive, something we don’t want to share with people, but something that still has to flow through the wires and be communicaed between you and the server. HTTPS is what helps us send this over securely.
With https, the communicaion between the client (browser) and the server is encrypted such that the information being sent over the wire cannot be intercepted by someone other than the server for which the information is intended to. If anyone tries to look at the encrypted data, it will seem like a string of random characters. But because of the way encryption works, the server has some additional knowledge which lets it decrupt the information and consume the actual information. This additional knowledge that the server has needs to be kept secret and private.
ecause of what we discussed before, the use of https becomes especially important when dealing with wesites that log you in your account. Since HTTP is stateless (ie, everytime you request a page, it knows nothing about you other than what you send in the current request), the server has to somehow recognize you and determine your identitt from the information you send with every request. Which also means that with every request, you’re sending some sensitive data over to the server which you do not want anyone else to get hold of. (if someone gets hold of what you send with each request to authenticate/identify yourself, the attacker could also easily use that in their request to pose as you).
🛡️ Present Day: HSTS
Let’s continue HTTPS’ story. We now know how important it is for websites to serve and receive content over https. But what if users access the site over http?
One way to get around it and enforce that communications take place over https would be to redirect any request on http to https. We can permanently redirect all http requests to https and it would work for us most times. But what could be the problems?
-
Redirection affects performance. We first request http which redirects us to https, after which our request is processed. Let’s say we care more about security than performance, and we don’t mind the redirection. Any problems then?
-
Yes, there still can be problems - security problems. Even if we set up a redirection, the initial request goes over http. And if you’re logged in or you’re trying to submit a form with sensitive data, it is still available for someone in the middle to intercept or attack it.
There is another way to enforce HTTPS, and that is (finally!) with HSTS.
HSTS (HTTP Strict Transport Security) is a security-enhancement achieved with the use of a response header (headers that are sent by the server in response to a client’s request).
Suppose you’re trying to access a website example.com, you can do it either via http or https. Let’s assume it does it over http, so this initial request is not secure. Now, the server redirects to the https endpoint like we discussed previously.
At this stage, HSTS comes into picture as the response includes the HSTS header. This header instructs the browser to use HTTPS for every connection to the site and its subdomains (if includeSubDomains specified in the header) from now on. This rule is valid for a period of time which is specified n the response header as well.
What this means is that after receiving this header, the browser will absolutely not load any resource over http for the particular domain. If the browser is asked to load a resurce over HTTP, it will try using HTTPS instead. In case HTTPS fails, the connection must be terminated.
There only needs to be one request to the domain, and with the help of redirection + HSTS, we can ensure that any more connections to the domain for that browser happen via https.
Note from the MDN Docs:
The Strict-Transport-Security header is ignored by the browser when your site is accessed using HTTP; this is because an attacker may intercept HTTP connections and inject the header or remove it. When your site is accessed over HTTPS with no certificate errors, the browser knows your site is HTTPS capable and will honor the Strict-Transport-Security header.
Is there a caveat here as well? Hmm, we’ll see. ut first let’s have the last piece of HSTS together, - how to set it up, and its componenets.
🛡️ HSTS! Dis-assemble!
The HSTS hader can be set in a response like you’d set u any other header. For more technical how-to, visit your preferred backend tech documentation. The syntax for the header is:
Strict-Transport-Security: max-age=<expiry_time>; [includeSubDomains] [preload]
In the above syntax, the directives enclosed in square brackets [] are optional.
-
max-age: This is what tells the browser the duration for which it has to remember to exclusvely access a domain over https. This is specified in seconds and the maximum value is 2 years.
We usually specify the HSTS header for every response delivered over https. Thus, whenever the header is sent in the response, the rwser will update the expiration time for that ste, and this duration is refreshed which prevents it from timing out and expiring.
When the specified time elapses, the next attemt to request resource over http will proceed normally without automatically using https.
-
includeSubDomains: This is optional. If this parameter is included in the header value, the security enforcement will be applied to all the subdomains of the site.
-
preload: This is optonal as well. This is an additional layer of protection that helps to tackle the issue that still remains after using the HSTS header. Adding this directive indicates that you want to be preloaded. We will discuss about it in detail in the next section.
🛡️ HSTS Preload List
Let’s talk about the caveat that remains. We now know that with the help of redirection and HSTS, the user only has to visit the site over https once before it expires to enforce communications over https. But it leaves open a small window of opportunity for attackers to sniff data and modify it if they want to.
Suppose you visit a site for the first time using http, and then the website redirects you to the https version and sends you the HSTS header in response. You are safe for any subsequent connections, but that first connection that you had made insecurely, is what might ultimately prove to be a point of failure.
HSTS preload is not part of the HSTS standard spec though. The HSTS preload list is a list maintained by the Chromium project which is used by most major browsers.
The browser has this internal preload list which is checked when making a connection. So suppose you have ever visited a website before but you have the domain added in the preload list, and you try to access the website over http for the intial request - it will not happen. The website then is never accessed via HTTP, and that includes the first connection which was the tiny window for an attack.
In short, if your domain is added into the HSTS preload list, most browsers will never connect to your domain without https.
🛡️ How to add to the HSTS Preload List
Theere are certain requirements that need to be fulfilled before you can request your domain to be added to the HSTS Preload List.
- Your site must serve a valid certificate
- If you are listending for HTTP requests, your site should redirect all requests to HTTPS
- All the subdomains must be served over HTTPS.
- For your base domain, serve the HSTS header with: 1) max-age of at least 1 year; 2) includeSubDomains directive; 3) preload directive.
- Make sure the site continues to satisfy the above requirements at all times. Removing the
preload
directive will make your site available for the HSTS preload removal.
So after you have made sure that all your subdomains are served over HTTPS with a valid certificate, and that nothing breaks with the addition of HSTS headers (try it out with max-age of a few minutes first), add the preload
directive to the header.
Then, head over to hstspreload.org and submit your domain. You can also find more cautions listed on the page and the step for removal if anything goes wrong, which is not a process they recommend. So try to be sure and test HSTS without the preload before you request to add a domain to the preload list.
This list is not immediately updated by the browser by downloading it over everyday or at intervals. It is a hard-coded list which is distriuted with new browser versions, which means that it might take some time for your site to appear/remove in one of the browser’s list.
🛡️ Tip for tiny rewinds
When testing HSTS with smaller max-age
(and without the preload coming into the picture), you might occassionally want to remove the HTTPS-only restriction of your site. appuals has a very helpful section on how to remove a domain from HSTS cache in browser. I will brief it here ut you can find more information for more browsers in the given link. Note that this will not be effective if the domain is included in the preload list.
Chrome
Visit chrome://net-internals/#hsts from your browser. Enter the domain name in the “Delete domain security policies” section and hit Enter. Restart Chrome.
Firefox
One way to delete HSTS cache for a site would be to “Forget ABout this Site” but that would remove a lot more data than just the HSTS restriction. For more methods, visit the above link.