Learn Phoenix LiveView is the comprehensive tutorial that teaches you everything you need to build a complex, realistic, fully-featured web app with Phoenix LiveView. Click here to learn more!
I recently made the mistake of getting embroiled in a Twitter[1] drama. A popular Javascript “starter kit” framework was found to be riddled with embarrassingly basic security flaws, and I joined the resulting pile-on with some threads that went quite viral. I may have made some enemies.
I’m not going to relitigate the argument about that one particular framework - I don’t care. Instead I want to make a more general point, because I got a lot of replies that, frankly, disturbed me.
A lot of “indie hackers” apparently think that security isn’t important. Sure, it’s better to be secure than insecure, but you have a business to launch - security is a cost you can live without, so get your MVP out the door asap then you can worry about security later. Right? People actually think like this.
To me this is like a car manufacturer saying “just make it driveable, we’ll add safety later”. It’s more than reckless; it speaks to something fundamentally wrong about how you approach engineering. Safety and security aren’t something you slap on later as an afterthought. They should be woven into your design from day one - and they don’t have to cost you anything!
A well-designed technology is secure by default. You shouldn’t have to spend much effort if any to keep your app secure. Just use secure tooling, and most of the work is done for you. If anything a secure app should take less time to build than an insecure one, because you’re simply working with the framework’s defaults, not against them.
We were all beginners once. We can’t expect every developer to know about and protect against every possible security issue. A healthy software industry is one where we make things as easy as possible for juniors as well as seniors; let’s use tooling that automates the “solved problems” so we can focus on the interesting stuff that hasn’t been done already a million times before.
To see what I mean, let’s explore some of the ways in which Phoenix keeps you secure by default. None of this is specific to Phoenix - everything I list below is also a feature of Rails and many other frameworks too (although, apparently, not all of them.)
No spam. Unsubscribe any time.
Phoenix automatically HTML-escapes any text that you output between <%=
and %>
. So if a user tries to maliciously insert a <script>
tag:
…then it’ll be rendered harmlessly as text, rather than parsed as valid HTML:
This protects you against Cross-Site Scripting (XSS) attacks.
Phoenix automatically applies CSRF protection to all non-GET endpoints. Submit a request without a valid CSRF token and you’ll get an error:
You don’t normally need to think about this. Phoenix’s <.form>
component outputs a hidden input containing the _csrf_token
, so the CSRF check will pass and the form will submit successfully.
The default Phoenix router calls the put_secure_browser_headers
plug in its :browser
pipeline. This sets some headers on the HTTP response that enhance security, as detailed in the docs:
Ecto, Phoenix’s data mapping library, automatically parameterizes its inputs, protecting you against SQL injection attacks:
If only Little Bobby Tables’s school had used Ecto, they would have been safe.
Practically every web app needs an authentication system by which users can create accounts and log in. There’s no excuse in <%= current_year %>
for getting this wrong. Login systems are as old as the web and you don’t need to innovate. You can just use someone else’s battle-tested tooling.
Phoenix makes it super easy: just run mix phx.gen.auth
and you get user registrations, login, email confirmation and password resets, for only 2 seconds’ work:
This contains everything you need to keep your login system secure, including some less well-known details, such as protecting against timing attacks:
(For more on how this login system works and what makes it secure, see this thread I wrote.)
These attacks and their mitigations are all worth knowing about. But my point is that you don’t need to know about any of them. Phoenix gives you all this security for free - even if you’re a total beginner who’s never heard of XSS, CSRF, SQL injection etc., you’re still protected.
You can bypass these protections, but you have to do so explicitly. For example, you can render unescaped HTML (bypassing XSS protection) using Phoenix.HTML.raw/1. So the option is there if you need it - but it’s not the kind of thing you might do accidentally.
This protects us against ourselves. Security failures often result from ignorance: we don’t guard against a particular type of attack, e.g. XSS, because we didn’t know we needed to. Or they can result from sloppiness: everyone makes mistakes, and we only need to be wrong once for an attacker to get his shot. In any case, using secure-by-default tooling means that our software is secure even when we’re ignorant and sloppy.
Even better, secure-by-default tooling teaches you about security. Use Phoenix (or Rails) for long enough and eventually you’ll bump into its safety barriers. This is a learning opportunity!
For example, I don’t remember how I originally learned about XSS, but it probably involved me trying to render some HTML in Rails and noticing that the “<”s and “>”s were being changed to the escaped versions. Then I looked it up, and learned something new about web security.
This is far better than the alternative - learning about XSS because your tooling didn’t protect you, and now you’ve accidentally shipped something insecure and a hacker has ruined you.
This shouldn’t be difficult, but recent events suggest otherwise. As I said on Twitter, a big part of the problem, I think, is the proliferation of shitty tools. Devs have been trained to neglect security because they’ve only ever used awkward, sloppy frameworks that give you a million ways to shoot yourself in the foot. They don’t realise there’s a better, easier way.
If anything I’ve written here has piqued your interest in Phoenix, allow me to shamelessly plug my LiveView course.
(This blog post is adapted from some threads I wrote on Twitter here and here.)
No spam. Unsubscribe any time.