A common narrative is that one should always use access tokens to call your APIs, while ID tokens are strictly for identifying users. Some of it has come from this article by Auth0 [1], which makes a strong statement about using ID tokens: However, things are usually more nuanced. In some cases, using ID tokens instead of access tokens is both acceptable and pragmatic. Cognito User Pools might be one of these cases. Cost of using access tokensThe common practice amongst Cognito users is to use ID tokens. Before January 2024, you couldn’t customize access tokens to include the OAuth scopes. So, using access tokens for authorization just wasn’t an option. But now, Cognito lets you customize access tokens through the Pre-Token Generation trigger, like this [2]. With this, you can implement authorization in API Gateway using access tokens and OAuth scopes. However, this requires Cognito’s advanced security features [3], which are charged much higher – starting at $0.050 per MAU and do not have a free tier. You must also pay the standard MAU cost for Cognito User Pools, which starts at $0.0055 per MAU and has a generous free tier of 50,000 MAU. This significantly raises the cost of using Cognito User Pools. Here is how much it’d cost you per month with Advanced Security Features (using access tokens) vs. without (using ID tokens). This might be fine for B2B use cases where you tend to have few high-value users. But it’s practically a death sentence for many B2C businesses, many of whom have thousands of free users. “Why not use something else instead?” For this many MAUs, you’d pay even more with vendors such as Auth0 and Okta. Most of whom require you to sign an enterprise contract before you can reach this scale. Cognito’s greatest strengths are cost efficiency and its integration with other AWS services, such as API Gateway and AppSync. If access tokens are significantly more costly, one must ask, “Is it worth it?” Are access tokens more secure? If so, are they THAT much more secure and worth the extra cost? ID tokens vs. Access tokensWhenever I show an example of using Cognito with ID tokens, someone would tell me, “You should use access tokens instead!”. But, I have yet to hear a compelling argument for why ID tokens are less secure. Let’s quickly debunk the common arguments. “ID tokens are not designed for authorization” Yes, access tokens were intended for APIs and ID tokens for authentication purposes. However, this division is not always necessary or practical. After all, there is more than one way to implement authorization. You can implement the authorization logic in the identity provider and embed the authorization decision in access tokens (in the form of OAuth scopes). However, you can also implement the authorization logic directly in the API. In my last post [4], I demonstrated how you can do this with a Lambda authorizer and Cognito groups. We will also explore other ways to implement authorization with API Gateway in the coming weeks. It’s two routes to the same result – being able to control who can do what in your system. “ID tokens have more information” This is true. ID tokens contain information about the user, such as their name and email. But what can attackers do with this information if they managed to steal your ID tokens? Nothing. Most likely. As an attacker, I have easier ways to acquire names and emails than to steal ID tokens from a system. “ID tokens give you access to the API” So does access tokens. “Access token can only be created by a trusted source” So are ID tokens. “Access tokens have limited lifetime” So can ID tokens. You can configure the validity period for both access and ID tokens in Cognito (and with other vendors). It’s a matter of making sensible architectural decisions. “You can bind access tokens to specific senders to avoid abuse” Yes, the Auth0 article mentioned this. The linked article [5] (which is a good read, btw!) discusses two techniques:
for binding a token to a specific sender. However, nothing about these techniques is specific to access tokens. They will work equally well for ID tokens. SummaryUntil someone can prove otherwise, I believe it’s perfectly safe to use ID tokens with Cognito authorizers. ID tokens are not inherently less secure than access tokens. Furthermore, all the techniques that make access tokens more secure also apply to ID tokens. So, there are no security downsides to using ID tokens with Cognito. On the other hand, there are significant costs to using access tokens. I’m not saying that you shouldn’t use access tokens! In fact, I will show you how to use access tokens to implement authorization in API Gateway in the next post. But you should know the trade-offs and not blindly pick a more costly approach based on hearsay. Links[1] “ID Token and Access Token: What’s the Difference?” by Auth0 [2] How to customize access tokens in Amazon Cognito user pools [3] Cognito User Pool advanced security features [4] Fine-grained access control in API Gateway with Cognito groups & Lambda authorizer |
Join 14K readers and level up you AWS game with just 5 mins a week. Every Monday, I share practical tips, tutorials and best practices for building serverless architectures on AWS.
2024 was the year I got back and amongst the community, and it felt great to be back! Blog I published 33 new blog posts. As a whole, my blog garnered 353k views from 255k visitors. About half of them came through Google search. This is down from 2023... but the decline is offset by more people reading my content through my newsletter nowadays. Most read blog posts: Hit the 6MB Lambda payload limit? Here’s what you can do When to use Step Functions vs. doing it all in a Lambda function How to...
One of my favourite questions from the November cohort of Production-Ready Serverless [1] is, "How do you handle e2e tests involving multiple services across bounded contexts?" In a microservices environment, testing user journeys that span across multiple bounded contexts requires collaboration and a clear delineation of responsibilities. Depending on how your organisation is structured, different teams are responsible for testing parts or the entirety of the user journey. For example... The...
The ability to invalidate a user's session with immediate effect is a common enterprise requirement. For example: If a user's credentials are compromised, we need to immediately revoke the user's access and force the user to change credentials. If an employee is terminated or an external contractor's access is revoked, their session should be invalidated immediately to prevent misuse. Many regulations mandate strict access controls and the ability to prevent unauthorized access in real time....