Your API Authentication is insecure, and we'll tell you why
3 days ago I reported Spree Commerce critical JSONP+CSRF vulnerability on all API endpoints. Instagram API was vulnerable to CSRF. Disqus, Stripe and Shopify APIs were leaking private data via JSONP. All that happened because they were not using Hybrid API Authentication properly.
This post is a must read for every API developer. I will keep it short and concise though. Seriously, give your friends the link, because I’m going to explain essential basics of API Authentication and current state-of-the-art.
So, you have Application Program Interface that authenticates by api_key
:
Then someone asks you to implement CORS because they want to use your API with JS:
And you obviously have skip_before_action :verify_authenticity_token
in your ApiController. Why would you need CSRF verification for API requests coming from, say, your Android app?
Yet another customer asked for JSONP support because CORS is not supported in old browsers. Sure thing!
Everything is fine so far. But eventually your developers decide to follow the trend Backend-As-API and use your own api.example.com on the client side. There are two options:
Append api_token manually
For example Soundcloud sends Authorization:OAuth 1-16343-15233329-796b6b695d2c7c1
header with every API request, Foursquare adds oauth_token=YXIAC4Y254HGZBNPQW6S0UFBGGSU57RBP
.
Disadvantadge #1: XSS. OAuth tokens are accessible with Javascript and the attacker can leak victim’s credentials. There’s HttpOnly
flag for cookies to prevent that. Nothing like that can be created for OAuth tokens.
Disadvantadge #2: For every request there will be an OPTIONS
request, doubling the latency. By the way I wrote about CORS without preflights trick.
Despite some high profile use this approach I do not recommend it.
Authenticate the user by cookies
The fix is short and you are all set: @current_api_user = (try_spree_current_user || Spree.user_class.find_by(spree_api_key: api_key.to_s))
. try_spree_current_user
parses _spree_session cookie, extracts user_id and returns User.find(session[:user_id])
. So what can be wrong with this line of code?
Cookies is also a header like “Authorization”, but very tricky to understand even for mature developers. I call it “sticky credentials”, because they are attached automatically, even to requests from 3rd party domains (evil.com).
The fact that absolute majority of web developers don’t understand this simple concept made Cross Site Request Forgery the most wide spread security issue, and I’m not exaggerating. That’s why every cookie-based authentication must be “double-authenticated” with extra csrf_token nonce. This nonce helps you to make sure the request comes from your domain.
-
Since you skipped CSRF protection for API requests, all your API endpoints are now vulnerable to request forgery. Example changing admin’s password on Spree.
-
JSONP leaks any GET response via cross-site with
<script src="https://api.example.com/orders.json?callback=leakMe"></script>
-
CORS is even worse, because every kind of request is leaking.
Doing it right: Hybrid API Authentication
This Hybrid approach allows you to use your api.example.com with both frontend (JS/HTML app) and 3rd party application, keeps your credentials secure from XSS (HttpOnly) and doesn’t generate pointless OPTIONS requests. This is state-of-the-art and if your approach is different, it’s wrong.