Stop Using JWTs

(gist.github.com)

182 points | by dzonga 6 hours ago

33 comments

  • solatic 4 hours ago
    Necessary qualifier: for browser-based user sessions.

    Plenty of good uses for JWTs for service-to-service communication.

    edit: I read some of the linked stuff, e.g. https://paragonie.com/blog/2017/03/jwt-json-web-tokens-is-ba... . Please, if JWTs are such a horrifically insecure standard, go ahead and publish your means for hacking AWS STS's AssumeRoleWithWebIdentity , or don't publish and just exploit it by launching cryptominers in every Fortune 500 production AWS account. Let me know when you inevitably succeed, because JWTs are so insecure, right? /sarcasm

    • RagingCactus 4 hours ago
      > Necessary qualifier: for browser-based user sessions.

      > Plenty of good uses for JWTs for service-to-service communication.

      This is the sensible conclusion right there. I agree JWTs are the wrong tool for the use case of user sessions in the browser.

      To give some more arguments:

      All the signature and encryption stuff in JWTs is complex. While common JWT libraries have now mostly got their stuff together, this has not always been the case. There were plenty of libraries accepting the "none" algorithm [1] or allowing attackers to forge tokens by using a public key as a shared secret [2]. This is the direct result of the complexity criticized in the linked blog post.

      JWTs also cannot do some stuff you want for user sessions. You can't invalidate them without keeping a revocation list somewhere. But if you have to check an identifier for revocation on every request you could just use an opaque session ID and look that up on every request instead! Sure, you can use short-lived tokens and refresh them all the time, but why bother with that for a typical application that has to keep some state anyway?

      All that being said, I wholeheartedly agree that there are use cases in distributed systems and machine-to-machine communication where signed tokens can be useful. Just please don't confuse the two cases.

      [1] https://nvd.nist.gov/vuln/detail/cve-2022-23540

      [2] https://nvd.nist.gov/vuln/detail/CVE-2024-54150 (just a random example from googling, I don't know what library made this one infamous)

      • 0x696C6961 0 minutes ago
        The design I've landed over the years is to use both. The cookie is a session token and that's where you handle refresh tokens. Then there's an endpoint where you can mint a short-lived tenant-sepecific JWT. This holds the scopes & tenant id.
      • nine_k 3 hours ago
        > if you have to check an identifier for revocation on every request you could just use an opaque session ID and look that up on every request instead!

        One reason could be the size. A revocation list only needs to keep session IDs of recently logged-out sessions, for which the token's TTL hasn't yet expired. It may be a much smaller list than a list of every active session.

        Also, a JWT (or a Macaroon, etc) can store a large amount of details about the session in a cryptographically secure, unforgeable way. This rids you of the necessity to store all that in your active session database, again cutting the size.

        • saganus 32 minutes ago
          I am still waiting for Macaroons to be used widely. I think they are a fantastic invention.

          It seems they were not of very much use in the past, with the agentic-everything now, I see this as a great way of delegating permissions to subagents, third-party agents, etc.

          Working on something along these lines but unfortunately I cannot dedicate as much time as I'd like.

          Still, if anyone is reading, give Macaroons a try!

        • agwa 3 hours ago
          As someone who operates a PostgreSQL database containing 27 billion SSL certificates, each 1-2kb each, with a bunch of secondary indexes that get inserted in random order, I find it pretty incredible that people see the need to optimize their session database. At what scale does the size of the session database actually matter?

          Those stateless tokens may be "unforgeable", but they are replayable, and if you're not mindful of that you can have security vulnerabilities.

          • mewpmewp2 1 hour ago
            I think one meaningful case is when you have services in very different locations and you would rather than having to make a request to a session store in a single location, replicate the data to each location for better latency, so in this case a revocation list.
          • hparadiz 3 hours ago
            You should do some basic optimizations. Fixed length table and indexes on the unique string for fast lookups. I also like to do a rolling delete for old sessions after 30 days unless mobile session that is logged in. Those get to live forever.
            • agwa 3 hours ago
              Fair enough, but those optimizations are basically free. People think stateless tokens are free but they really are not.
              • hparadiz 3 hours ago
                The cost of the stateless token is basically the CPU usage for signing the message and checking the signature with the public key on the client. Example: Google Compute Instance asks metadata server for OIDC token (which is a JWT). The metadata server respond with the token that basically says "here's the machine service account, here's the machines ID, this token is proof that I am service account abc123 and it's valid for 20 seconds". This is one of the most common uses of JWTs in enterprise. You don't store them. They actually are free.

                Lots of web devs get tricked into using them as primary session tokens and it's a huge anti pattern. I see it all the time and people get aggressive about it.

                • agwa 2 hours ago
                  The cost is the vigilance required to use them safely. It's not just compute/storage costs.
                  • hparadiz 1 hour ago
                    I didn't downvote you. You're absolutely right. Implementation of anything is work.
      • robertlagrant 2 hours ago
        > While common JWT libraries have now mostly got their stuff together, this has not always been the case. There were plenty of libraries accepting the "none" algorithm [1] or allowing attackers to forge tokens by using a public key as a shared secret [2]. This is the direct result of the complexity criticized in the linked blog post.

        I'm a bit surprised at this. These are extremely simple to solve - the first time I ever did a JWT-reading implementation I specified the right defaults, which are very simple, even for a mid-level backend person I would say, and they haven't needed changing in 8 years or whatever it's been. It really isn't very complex.

      • figassis 44 minutes ago
        • dchest 35 minutes ago
          WTF:

          > Each user has a secret: Stored securely in the database.

          > Stateless Validation: The core validation remains stateless. We only need to consult the database for the user's secret, which we'd likely do anyway for authorization checks.

          Is "stateless" the same as "serverless" now? Is author's brain stateless?

        • throwaway7783 31 minutes ago
          "We only need to consult the database for the user's secret..." , which kinda defeats the purpose.
        • RagingCactus 25 minutes ago
          > First, we need to add a token_secret column to our users table:

          > ALTER TABLE users ADD COLUMN token_secret;

          So it's "stateless" but we have to query the users database on every request? How is that more stateless than SELECT * FROM session WHERE id = cookie?

          Ignoring that and taking the mechanism as given: Why the obsession with cryptography, in this case HMAC? I don't see any reason why another signature is needed here when I believe the same outcome could be accomplished with a token_epoch field in both the signed JWT and the users table. Just increment the epoch to revome old tokens. Or even better, drop the epoch field and have an iat_not_before field per user. The field in the JWT is signed, the whole point is that you can trust it.

          Do let me know if I miss anything here please. Assuming I haven't: it's always puzzling to me to see people being so eager to sprinkle more cryptography on anything that is supposed to be secure. For me, I've become more afraid of cryptography the more I learned about it. Cryptography is hard. It's not a magic ingredient for security. At best, it's dangerous black magic -- very potent, but pronounce a single syllable of your magic spell wrong and it _will_ blow up in your face.

      • chuckadams 1 hour ago
        A revocation list defeats the purpose of JWTs. If you find yourself needing one, JWTs were probably the wrong choice to begin with.
      • LgWoodenBadger 2 hours ago
        Come on, it’s not like the two are even within the same magnitude or three

        “But if you have to check an identifier for revocation on every request you could just use an opaque session ID and look that up on every request instead!”

      • hparadiz 3 hours ago
        If you don't understand conceptually how to verify a signature with a public key the very first thing you should do is get that working and then work from there. It's completely unacceptable to ship without this.
    • kyrra 4 hours ago
      JWT used to be bad due to libraries with poor defaults. Downgrade attacks were fairly common a number of years ago.

      Since most of the common libraries across all languages have gotten more sane defaults, it actually is pretty secure nowadays.

      • tptacek 2 hours ago
        If we stipulate that, we're still left wondering what the utility is of a standard that creates affordances for the insecure defaults, as opposed to just designing it right from the beginning.
        • ForHackernews 36 minutes ago
          Spec writers and library authors are human? Who knew
          • tptacek 32 minutes ago
            I don't understand what this is meant to communicate. The standard is either good or it isn't. "Good effort" is not an engineering assessment.
            • ForHackernews 29 minutes ago
              Your objection is that they should be "designing it right from the beginning" but that applies to all realms of endeavour. The reason they didn't is human frailty.

              If everyone simply designed everything right from the beginning we would live in nirvana.

              • tptacek 27 minutes ago
                You've completely missed my point. I don't even accept the premise of the JWT standard. But the eventual migration to safer default settings, in a format that continues to expend implementation effort to support settings nobody should use, is in fact a practical engineering problem with the standard.
                • ForHackernews 18 minutes ago
                  And they've published updates[0] and libraries have hardened their defaults and removed support for insecure values (e.g. alg='none'). I'm not sure what more you want?

                  I'd rather use a refined, battle-tested standard with lots of eyes on it than some new untested contender produced by a handful of upstarts ("look, we just designed it right from the beginning! This time it's perfect!") PASETO reeks of second-system syndrome.

                  [0] https://www.rfc-editor.org/info/rfc8725/

    • jkrejcha 1 hour ago
      JOSE can still have problems if it's secure when implemented properly. A lot of API surfaces for them can kinda suck. If secure when held right was equivalent to good, then that would apply also to stuff like X.509

      There are better alternatives for a lot of cases, standard session tokens or API keys are a popular one in use in most major websites online and work pretty much perfectly for most use cases.

      I'm not gonna say those standards are completely without merit. The best thing about them is that it is some basic standard on passing stuff around that isn't like ASN.1 encoded or whatever, to which the tooling seems incredibly brittle and bug-prone.

    • jeltz 3 hours ago
      I agree with your first part but your edit is a logic fallacy. I don't need to be able to hack something to say that it is insecure.

      For example: I don't know how to exploit SAML but I know it is a terrible standard dur to making all of the XML parser an attack surface. I am not a security researcher so I dont know how to find exploits in XML parsers but I know having a huge attack surface is bad.

    • tptacek 2 hours ago
      There is in fact a long lineage of vulnerabilities caused by JWTs in real applications.
  • littlecranky67 3 hours ago
    In sessions vs. JWT revocation lists, there is an argument in favor of JWT revocation lists. JWTs have a limited expiry timestamp, so you only ever need to maintain a revocation list for tokens not expired yet. Given that you probably only have a fraction of JWTs revoked compare to valid JWTs in circulation, you only need to query a very small dataset for each request.

    When using sessions, your list of valid sessions is probably orders of magnitudes higher that the revocation list - thus the data lookup costs and the storage cost of that statefulness is higher.

    Plus, the article mentions JWTs are stateless but that is usually not true. You mostly not only validate the JWT, but also obtain a matching identity object (i.e. user details) for each request to see if the user is still enabled/authorized to do whatever he does. You can leverage stuff such as per-user revocation lists, or a minimum_issued_at that will validate any JWT iat field. This allows the "Logout from all devices" pattern, where that action will simply set a user's minimum_issued_at field to $NOW. All previous tokens will thus be revoked, without individuall revocation list checks.

    • vidarh 3 hours ago
      The moment you have to look up the user object, you've lost the primary advantage of JWT, and might as well ditch it.
      • joshmarlow 2 hours ago
        It definitely violates DRY but if you keep passing the JWT down the call chain, you can do redundant permission checking in your business layer.

        Now the reasonable response to the above is that this should be happening in a dedicated authn/z concern - and that is correct! But when paranoia is called for, it's not unreasonable to have redundant checks in logic where authz is critical.

      • littlecranky67 3 hours ago
        Depends on the system. If you use JWTs for authentication only, they still serve a purpose. Sessions also only serve as authentication, not authorization. Authorization is independent of the both systems, and it depends how you implement that.

        There are systems where the authorization is done in the JWT too (i.e. scopes/permissions in the token) - in that case you are right.

    • jpalomaki 3 hours ago
      Session data lookup is one select to database that gives 0..1 rows and uses index. In most cases this is not something you need to worry about.
      • littlecranky67 3 hours ago
        I agree. The storage space, however, is a different story. Your session DB can grow huge, depending on your session lifetime and your users logout behaviour. Plus, it is a concern in a distributed system (i.e. a token can be validated on every node, vs. a session lookup must be globally in sync)
        • 10000truths 9 minutes ago
          1. For the vast majority of CRUD apps, active sessions will be a very small fraction of the actual storage requirements. A SaaS with 100K MAU may have only 100 or so active users at any given time.

          2. Sessions by definition are ephemeral. A database should not be necessary at all, an in-memory cache should suffice.

          3. If you really need to distribute session data across multiple nodes, just propagate them asynchronously. Authentication and authorization are semantically idempotent operations. Having to possibly re-auth when making a cross-region request within milliseconds of logging in might be mildly annoying for the user, but consistency isn't a deal breaker here.

          • littlecranky67 5 minutes ago
            > 3. If you really need to distribute session data across multiple nodes

            What you mean, "if" - you will need that once you are international. You can't afford to verify every http request against a centralized session store when you have users in Australia, US, Europe, Japan etc. You can't beat the speed of light. My point is, replicating revocation lists that are append-only, only a small megabytes, and can be publicly known, is always easier than syncing session databases for a complexity standpoint.

      • mewpmewp2 1 hour ago
        If you have multiple services in multiple locations however, you may want to replicate this data, so in this case revocation list as it's much smaller would be far easier to replicate for much less latency overhead.
      • matt-p 2 hours ago
        Can even put it in redis too, if you have performance issues from looking for it in memory then you have probably have more users than google.
        • littlecranky67 2 hours ago
          What if you have two servers, one in japan and one in central europe? Where do the sessions live?

          With JWTs, you would only need to replicate your revocation list of the last X hours (X being your JWT default lifetime) and probably be in the megabytes for the total list. Easy to replicate that ever 5-10seconds to all your locations.

    • zsoltkacsandi 3 hours ago
      > JWTs have a limited expiry timestamp, so you only ever need to maintain a revocation list for tokens not expired yet.

      Sessions have expiration timestamps too, and you can configure them however you like.

      • littlecranky67 3 hours ago
        Yeah of course, but how does that relate to my point? With JWTs you don'T have a list of valid tokens as state, but only a list of invalid ones (revoked). But the list of revoked tokens in the last X hours (where X is your token lifetime) is always going to be smaller than the list of active sessions given a large enough user base. Hence my original point stands, that the lookup and storage costs are lower than on sessions. Whether or not sessions have session lifetimes does not change the fact at all.
        • zsoltkacsandi 2 hours ago
          > With JWTs you don'T have a list of valid tokens as state, but only a list of invalid ones (revoked).

          Yes, and a lookup operation is a lookup operation.

          Your database or data structure used for storing the sessions/JWT revocation entries won't really care whether you look for things that are active or things that are inactive/revoked. If you store it in the right database, both lookups will be O(1), so it is the same (or at least the difference is negligible), regardless of the size.

          • littlecranky67 2 hours ago
            The story changes if you have a distributed database. replicating a smaller revocation list (that is append only) that will never be more than a couple of MB, is easier to do accross distributed nodes around the world than keeping a larger, session state db replicated. Heck, your revocation list can even be public (it contains only a list of substring of a few bytes of hashes).

            Syncing sessins can be done, no question, I would just think JWT+revocation db is easier to implement, yet robust.

            • conradludgate 55 minutes ago
              It can also be encoded as a bloom filter for very fast checks. Then you can defer to the replicated LSMTree that's stored replicated on your local node
        • doctorpangloss 2 hours ago
          isn't it that you must have a revocation list in many cases? if you cannot get from N to 1 or 1 to 0 states, if you're just going from N to N-1>1, you haven't materially decreased your statefulness
          • littlecranky67 2 hours ago
            But the revokation list is always going to be orders of magnitude smaller than the list of active sessions.
            • doctorpangloss 1 hour ago
              that may be, but who cares how big or small it is really

              if you are facebook sized, with 1b+ active sessions versus an alternative with 10m+ revocations... the kind of applications that reach this scale, they have enormous amounts of state anyway.

              • littlecranky67 1 hour ago
                Everybody thinks Mag7 scale, but actually it is more relevant if you are a tiny webservice - but available world wide. If you need to match each and every http request from a user half way around the world against a central db, you can't beat the speed of light. If you can do authentication on each downstream server directly using crypto and JWT validation, you at least save that roundtrip to the session db. The revocation list is tiny (a few megabytes tops), append-only, can be public to the world, and thus easy to replicate to your downstream nodes.

                If you are a smaller gig, you won't have to bother with replicating your sessions and keeping them in sync globally.

  • tracker1 4 hours ago
    JWTs are insecure... even when using trusted, rsa/ppk based signing methods? not shared secrets.

    JWTs are too long lived... Nothing is stopping you from limiting the JWT lifetime and having a refresh model against an authentication authority... I mean, even if you use cookie based sessions, you're storing somewhere... you can have a jwt valid for 5-15min. 15minutes is roughly the cache timing for many authorization systems including Entra... and even a 5min token with a refresh system can be used fine from a browser.

    Lastly, I prefer to have identity/auth separated from the application/api services... it externalizes context and JWT per request is easier to deal with than some shared cache/state system that may intermittently fail as opposed to a signed token that you can verify the signature against known authorities.

    • hparadiz 3 hours ago
      You can make a JWT invalid after 30 seconds or even 1 second. You should set an aud (audience) when creating the JWT. Otherwise the signature is crypto-graphically sound. Validate every single JWT every single time with a short lifetime.

      OIDC tokens are all JWTs btw.

      • tracker1 3 hours ago
        If your talking about a browser context, where the authority is separate from the requesting body, then expiring even at 30s is excessive for user context, let alone every 1s or every request... you're effectively then inflating every single API request into 2 requests... one for a new token, then another to the API being called. This is irresponsible for not much gain in a user-facing context.
        • hparadiz 3 hours ago
          You should not be using them for user contexts at all. The cookie should be the session token and the sessions should be stored on the server side where you can simply delete them and the user's login becomes invalid. Using JWTs for this use case is just plain wrong.
          • tracker1 26 minutes ago
            I disagree with you and the article on this... I thought that was pretty clear.

            You can use a revocation list with JWT if necessary, and if your JWTs never last more than 15m you'll be fine.. and if your security window is tighter than that, you probably have bigger issues to deal with.

  • rdegges 1 hour ago
    =0 I stumbled across this post and was thinking that it's interesting to see this topic trending now, since I've done a lot of work on it in the past. Then I clicked through and realized the author is linking to some of my stuff! What a blast from the past.

    Anyhow, there are way smarter people than myself who have covered this topic extensively over the years, but I still think that, even in 2026, JWTs are the wrong tools for web auth. They're fine to use for service-to-service stuff, but if you have the option, just use PASETO -- it solves a lot of the issues!

  • ApolloFortyNine 4 hours ago
    This links to some other blog post for the bulk of it's 'why', and that blog post mostly seems to be annoyed about "You cannot invalidate individual JWT tokens". Which every time I've implemented, the general guideline is to check for invalidated nonces somewhere. Which resolves that random blog posts second point too.

    >The JWT specification itself is not trusted by security experts.

    This feels like it needs more evidence than just one blog post. And that blog post seems to just largely blame bad implementations? Something that will plague any standard.

    Overall, I don't know what I expected clicking a random gist link.

    • tracker1 4 hours ago
      Yeah... some early implementations just allowed for any authority to be set in the header and trusted it... that's of course wrong from the start... if you only allow for trusted or "known" authorities a lot of the contextual concerns become non-concerns.

      Beyond this, you can make shorter lived JWTs just fine in the browser and have the agents self-update. If you use Azure Entra or a number of other providers it works this way in practice... you keep your JWTs relatively short lived (5-15m) and can even check for jti revokation.

      JWTs are incredibly useful for separating/reusing an access authority from your applications/api systems. You shift the attack surface and do it in a way that can be trusted. We use PPK for lots of things, including SSH all over the world. No, I wouldn't use shared secrets and I wouldn't use long lived tokens... but short lived, ppk signed tokens from verified/known sources are generally fine.

      For that matter, it's often API keys that are really problematic. Just had to implement them... for me, the API key presents as a Bearer token as well, but there's a short "sak." prefix then an identity part (base64url uuid bytes) followed by a secret as base64url bytes... in the database is the uuid and a passphrase level salt+hash from the secret.. so the api key generated should be treated as a secret and is one-way to the database, so a db breach doesn't breach auth.

      Even then, an API key leak is far mroe likely than a problem with a well implemented JWT solution.

    • jotato 4 hours ago
      > "You cannot invalidate individual JWT tokens". Which every time I've implemented, the general guideline is to check for invalidated nonces somewhere. Which resolves that random blog posts second point too.

      100% agree. This is common sense to me and I'm always surprised to re-learn people don't do this

      • hparadiz 3 hours ago
        Not checking the signature on every single JWT is the same as storing a password in plain text.
  • himata4113 55 minutes ago
    Hmh, the way I usually use JWTs is as an authentication cache. You obtain your authentication token from the auth service which grants you permission to other services.

    This has several advantages, the main one being that sub-services do not have to interact with the authentication database or have access to the capability to mint tokens (this assumes you use RS256 not HMAC). So if a sub-service gets compromised it's not as devastating as a service which has access to the authentication database.

    If you have sensitive data inside the token you should use JWEs, although they're not as good because you have to ask an internal service (which has the private key) to decode the token each time you want to use it.

    My typical layout is {"id": (uuid), "scopes": ["scope:read/write"]}.

    Also they're really neat for SPA's as you can have your static site server validate that the JWE with the public key before serving any resources. The way I use this is that I have my static site compiled to /(scope)/path and the static service will not serve pages that you cannot access anyway. This is very useful in cases where you have administrative panels where you don't want to expose to users what capabilities your backend has or/and expose the internal service paths that can be attacked.

    My lifetime for JWT's is around 5 minutes for "backend access", things like /me are cached in localStorage unless explicitely instructed in /refresh to drop localStorage cache. My request handler in my SPA applications detects "refresh required" and refreshes the token.

    I think most of the blame here belongs to node/next and python libraries. I write my backends in strongly typed languages and my frontend is always made out of precompiled static pages. My current setup for the frontend is using VITE with prerendered pages for landing and normal SPA for applications.

    With all of that said I strongly disagree with this entire gist. JWT is as secure as you want it to be.

    • iririririr 51 minutes ago
      irrevocable* cache
      • himata4113 47 minutes ago
        I've adapted dynamic public-key hotswapping whenever there is a need to revoke tokens as it would simply force all tokens to go to /refresh endpoint instead of the standard 5m cache. Never had to use it though. I've experimented deriving the public key from the uuid so I could broadcast that "keys with this id and this revision should no longer be accepted and should be refreshed", but as I said never ran into a situation where 5 minute expiration wasn't fast enough. That said if you're dealing with critical infrastructure JWEs are the way, you just lose the speed benefits of JWEs as you have to make a request to an internal service to validate and decode, but for everything else JWTs are completely fine.
  • bastawhiz 1 hour ago
    One of the linked posts explaining why you shouldn't use JWTs is bizarre at best:

    https://paragonie.com/blog/2017/03/jwt-json-web-tokens-is-ba...

    It boils down to "there were bugs in some of the libraries" and then goes on to recommend you...pull in libsodium and do it yourself??? This is ludicrous advice that I simply can't take seriously. All software has bugs. The whole Internet lost its shit with Heartbleed, but we still use TLS and OpenSSL.

    > The JWT specification is specifically designed only for very short-live tokens (~5 minute or less).

    I've never heard this before and can't find any evidence to back this claim up. RFC 7519 doesn't make any such claim.

  • Grollicus 4 hours ago
    I'm right now adding rabbitmq for notification pushing to a website. Using JWT authentication to control where and what clients are allowed to read, with short lifetimes and regular token refresh.

    I don't see another setup that comes close to the ease of setting this up - add an endpoint that provides jwt tokens to valid sessions, done. With user-individual permissions.

  • Aeolun 18 minutes ago
    At this point this just feels like an old man shouting at the clouds.
  • blixt 2 hours ago
    JWTs are fine, seems a bit sensationalist title...

    Some nice topics to talk about instead:

    - When to use an encrypted value (and symmetric or asymmetric), vs. a random (but secret) value, vs. a signed value (readable but not tamperable)

    - Where to put these values (memory, localStorage, cookies)

    - How to make sure these values don't last forever, and whether you need to be able to revoke them (make them invalid before their natural expiration timestamp)

  • BlackFly 1 hour ago
    If I understand the point being made here then the idea is that a stateless session via a cryptographically verified bearer token needs a stateful revocation list to eliminate hijacking (a user logout should completely invalidate the login but a bearer token would otherwise continue to be valid) and if you are maintaining state then you can just use a complete stateful session and avoid the complexity of the cryptography.

    This point is not made very clearly and is buried by overemphasising JWTs instead of just quickly pointing them out as an example of a stateless session. But yeah, it is a good point.

    • bastawhiz 1 hour ago
      Revocation lists can simply be replaced with a "tokens not valid before" field per user. When a user logs out, set the field to now(). Reject JWTs that have an iat less than that value. Am I missing something?
      • Sohcahtoa82 58 minutes ago
        What you're missing is that you're still creating state. You're still having to check a database to determine what the "tokens not valid before" value is for that user.

        And what if the user is logged in from multiple devices, but only wants to log out from ONE of them? Your solution logs them out from all of them.

        The entire point is that it is not possible to have authentication that is both: 1. stateless. 2. secure.

        And so if authN is going to be stateful anyways, you might as well just use an opaque token in a database and eliminate all the complexities and foot-guns of JWTs.

      • dchest 45 minutes ago
        Yeah, you made a revocation list but with time value instead of the token value.
  • adamddev1 2 hours ago
    I remember learning to make sites back around 2019 and seeing so many blog posts and hype around JWTs. It seemed like "this was the way to do it!" But I couldn't understand why session cookies weren't the better, simpler solution. I just used session cookies. Nice to be vindicated in retrospect.
  • miiiiiike 3 hours ago
    Security doesn't start or end with JWTs.

    A user wants to access a read-only resource with an invalid JWT? Envoy bounces it without passing the request through to the backend. Valid JWT? Let the request through without having to look up any session information. No DB, no cache, no session server hit. Fast.

    A user wants to change a password, email address, or add an authenticator? First, require a password, second, require a second factor. If all of that checks out, look for the JWT access token in a revocation list that is only accessed during sensitive, infrequent, requests like these. If the token has been revoked, 403.

    Tokens are dropped from the revocation list once the original access token's TTL has passed. Which should be low. I use 5 minutes. Most sessions on my site last 4-10 minutes.

    Worst case scenario, a malicious user is able to access certain read-only resources for a few minutes.

    • zdragnar 2 hours ago
      > look for the JWT access token in a revocation list that is only accessed during sensitive, infrequent, requests

      I've clearly spent too much time working with data covered by HIPAA because this sentence gave me a brief bit of panic. The vagueness and extent of what it technically covers means it's far safer to just assume literally everything about your users needs maximum security.

      • miiiiiike 11 minutes ago
        This is the eternal conversation around auth. “The thing you do doesn’t work for the thing I do.” OK. Use something else.
  • lucassz 2 hours ago
    > You can't securely have truly stateless authentication without having massive resources, see the cryto.net link above.

    I don't think the cryto.net post really explains why this is true (at least in a way that would be made different by "massive resources").

  • feelingsonice 3 hours ago
    PASETO is great but there's not enough ecosystem support
  • vova_hn2 4 hours ago
    One of the articles that TFA links to [0] contains the following paragraphs:

    > And there are more security problems. Unlike sessions - which can be invalidated by the server whenever it feels like it - individual stateless JWT tokens cannot be invalidated. By design, they will be valid until they expire, no matter what happens. This means that you cannot, for example, invalidate the session of an attacker after detecting a compromise. You also cannot invalidate old sessions when a user changes their password.

    > You are essentially powerless, and cannot 'kill' a session without building complex (and stateful!) infrastructure to explicitly detect and reject them, defeating the entire point of using stateless JWT tokens to begin with.

    I'm not sure that this is entirely true. Typically, the total number of non-expired issued tokens is much higher than the number of invalidated unexpired tokens. Therefore, if you store only invalidated tokens and delete them when they get expired, you can significantly reduce the amount of required storage and the cost of lookup.

    Although, in any real application the performance gains will be minuscule (compared to the cost of, you know, everything else. Auth is just a small part) and probably not worth the extra complexity.

    [0] "Stop using JWT for sessions" - http://cryto.net/~joepie91/blog/2016/06/13/stop-using-jwt-fo...

    • littlecranky67 3 hours ago
      >> You are essentially powerless, and cannot 'kill' a session without building complex (and stateful!) infrastructure to explicitly detect and reject them, defeating the entire point of using stateless JWT tokens to begin with.

      > I'm not sure that this is entirely true.

      You can be sure it is not true, because it is utter BS. JWTs have an "iat" timestamp field (issued at) and in the described case that an attacker has a leaked token, your validation logic simply should refuse any token with iat < $NOW for that identity.

      I have JWTs implemented for a site and in my case, users cannot individually revoke tokens - but they have a "Signout from all devices" option. That will basically just set a field "minimum_issued_at" to $NOW in the database for their user, and any tokens will always be validated against the minimum iat timestamp. That is a good compromise in security and simplicity.

      Revocation lists have their purpose, though, in systems with heightened security requirements.

      • vova_hn2 3 hours ago
        > your validation logic simply should refuse any token before $NOW.

        Well, this approach throws out a lot of babies with the bathwater. You invalidate tons of legitimate tokens along with the one that you wanted to invalidate and get a thundering herd [0] of clients wishing to re-authenticate.

        This is probably not good in case of a really high load.

        And if you don't have a really high load, then there is no good reason not to have a stateful session storage.

        [0] https://en.wikipedia.org/wiki/Thundering_herd_problem

        • littlecranky67 3 hours ago
          I edited my comment after I posted it to clearify you do this on a per-identity basis. I.e. every user/identity has a minimum_issued_at field. A user can "sign out from all devices", and that will simply update minimum_issued_at with $NOW.

          You are not throwing out a lot of babies with the bathwater if you would do it in a case of a known attack. You would invalidate ALL tokens of a user, which is a sane default especially since usually you wouldn't be able to rule out what other tokens were compromised. And yes, if it later turned out ALL your users and all their token were possibly compromised because you had some kind of security flaw, setting a global minimum_issued_at is exactly what you would do after you fixed the flaw. And yes, that means all your users must reauthenticate.

          • vova_hn2 2 hours ago
            Thanks for the correction, I didn't think about this approach and it sounds like it should work.

            The only comment that I have that if you are already querying users table (or collection in case of NoSQL or whatever), you might as well have a sessions table/collection in the same database/storage and query them together. It seems that difference is not that big.

            The purported advantage of stateless sessions is that you can check the auth without querying the main db/storage (maybe only querying a smaller/faster axillary storage).

            • littlecranky67 1 hour ago
              Think a small (as in client base), but distributed system - i.e. Asia/EU/US locations of a webshop. You can easily replicate/cache your products from a central server, and reuse the cache from the localized ones. But each and every web request would have to be authenticated against a central db somewhere around the world. It is just easier if each node can just validate the JWT themselves by using crypto. All they need to do is maintain a revocation list locally. Now, your revocation list is append-only, can be publicy available and never going to be more than a couple MB. Very easy to replicate/cache this. I can't say the same for a session database.
      • megous 1 hour ago
        > your validation logic simply should refuse any token with iat < $NOW for that identity.

        makes no sense

        ... ok now it does :) your now is not now, but a stored value

    • elcritch 3 hours ago
      It really doesn’t seem very hard to have a small invalidation list. Just a redis cache or a simple broadcaster, etc.

      Does anyone have an example of how they built a JWT revocation service?

      • littlecranky67 3 hours ago
        See my sibling comment about the "signout from all devices / iat" pattern. This is only a few lines of code.

        If you want to be more fancy and fast, you can use bloom filters to check if a token is in a revocation list.

  • gabrielsroka 4 hours ago
    2019
  • cjoelrun 3 hours ago
    I ain't never gonna stop!
    • InsideOutSanta 3 hours ago
      I might stop if somebody linked to an article pointing out an actual problem, rather than making vague and/or incorrect/misleading assertions.
      • andreyvit 2 hours ago
        Let me bite, as someone who usually hates JWT but sometimes uses it, including for browser auth.

        Why JWT is bad: it's a cargo cult solving a non-existent issue in a more complicated way than necessary. An HTTPOnly session cookie containing just a random ID is shorter and easier to handle.

        Why JWT is also bad: a typical way to use it exposes too much attack surface. Almost every JWT library has way too much functionality, supports multiple algorithms, and many people are too sloppy with their dependencies, so you probably haven't read every line of code that runs in your auth.

        How to use JWT safely:

        1. Have a use case that cannot be easier solved with just a random session identifier. For example, one party creates tokens and another unrelated party verifies them. If same party issues and validates tokens, you better have a super high load, unique use case -- but then you're senior enough to not take random advice from strangers.

        2. Write your own JWT handling code. It's literally a few lines of code to create tokens and a few dozen to validate. Only implement the exact algorithms and claims you use.

        3. In a typical scenario, JWT should still carry something like a user ID which you should immediately verify against a database. Stateless sessions doesn't mean no DB lookups on validation. If you DO authenticate based on the token alone, the token should be super short lived (seconds or single digit minutes).

      • hparadiz 3 hours ago
        HN is full of people who don't actually fully understand the subject matter speaking confidentiality. And lots of arguing even when they are clearly wrong.
  • hparadiz 3 hours ago
    JWTs are for authenticating an already trusted system with another system.

    Using them as the primary source of truth is an anti-pattern like the blog post is actually saying.

    • NetOpWibby 2 hours ago
      It's not a blog post, it's a post on Github Gist.
  • ForHackernews 37 minutes ago
    I'm not convinced. All the PASETO objections[0] are things that have long since been addressed by every major JOSE implementation.

    [0] https://paragonie.com/blog/2017/03/jwt-json-web-tokens-is-ba...

  • EGreg 57 minutes ago
    Over the past 15 years of developing secure web apps for various organizations, I ended up architecting solutions which had to deal correctly with ITP, webauthn, CHIPS, iframes and more.

    This year, I ended up publishing an open protocol that can be used for everything from secure authentication to API requests to micropayments, and it works with the existing Web stack and P256 curve with JSON serialization curves as well as EVM blockchains via K256 curve and EIP712 serialization.

    I encourage everyone to take a look at it and consider using it, so we can stop reinventing the wheel. Everything has already been deployed, it’s an extremely simple protocol, much simpler than JWT and requires no global registry.

    https://openclaiming.org

    https://github.com/OpenClaiming

    It is also used in stuff like https://safebots.github.io/Safecloud/

  • misano 2 hours ago
    imo instead of random cookie u can carry some data in it and avoid more joins in database why not ?
  • mgaunard 3 hours ago
    A whole lot of nonsense from a web guy.

    Please, keep using JWTs, they do their job well: giving you an access or ID token that you can pass between applications and trust based on cryptographic signatures from an identity provider.

  • kobalsky 2 hours ago
    I agree that using cookies is better for web sessions but I absolutely despise those using the boogeyman to shoo people away from stuff they don't like, instead of asking them to use their brains.

    > they are not secure.

    They are secure if they fit your risk profile, a blanket statement like this is just disinformation.

    Don't treat your peers like idiots.

  • _el1s7 43 minutes ago
    I hate these type of clickbait articles
  • szmarczak 4 hours ago
    No need to stop. The XSS argument also applies when using cookies.

    JWTs are just tokens like session data but in JSON format. What format you choose to go with doesn't matter.

    You can keep storing JWTs in local storage and still be secure. Discord removes it on page load and restores it when the tab is closed.

    Also if your website is susceptible to XSS, skill issue, exactly like in the case of SQL injections. That wouldn't have happened had people used the right tools and not played with fire.

    • adithyassekhar 15 minutes ago
      > Discord removes it on page load and restores it when the tab is closed.

      How does this work? You have no real control over what the browser does when it closes a tab.

    • wccrawford 4 hours ago
      I think anything can be abused, and too many people don't have a security-first mindset.

      One of the advantages of JWTs is that you don't have to check your database or filesystem to make sure the the user is valid and logged in. All that data is in the JWT. If it's just a static page, it doesn't need to hit any data.

      The problem then comes that some developers think that makes it secure, and don't check the database for revocation before doing anything with the account. Especially not for giving out private data. They might check before changing any data.

      I think it's a really neat idea that is far too easy to mishandle and create a bad situation. It can save a lot of bandwidth and CPU cycles if you have a lot of non-interactive pages and all you need to know is whether to show that the user is logged in or not. But for actually doing anything, it's practically no better than a session cookie, and it's got a lot of foot-guns.

    • jkrejcha 1 hour ago
      A lot of times local storage is much less secure than using cookies. Cookies have about 20 years of infra built around it (HttpOnly, SameSite, Secure, etc). There's some weird parts about cookies, but local storage really shouldn't be used for anything security sensitive
      • megous 35 minutes ago
        20 years of security:

        sqlite3 cookies.sqlite 'SELECT name, value FROM moz_cookies WHERE isSecure AND isHttpOnly'

        And that's a supposedly a master password protected browser. They can't even bother encrypting cookies. Don't be ridiculous.

      • szmarczak 24 minutes ago
        > A lot of times local storage is much less secure than using cookies.

        Is it? If an attacker can't do XSS then it's as strong as cookies.

        Supply chain attacks aren't an argument here because they can also happen with cookies. CSRF as well. The same can happen in actual executable binaries.

        I don't get the 20 yr age argument:

        - HttpOnly fights XSS which is impossible to execute with modern frontend frameworks.

        - SameSite fights CSRF but the real solution is to disable loading the website in iframes (remember clickjacking?).

        - Secure fights MITM which is already fixed by default when using local storage and HSTS is the real deal.

        Having said that, I'd say that local storage is more secure than cookies (no need to remember whether you put Secure on or not). Unless you're still using PHP, which means touch grass.

    • dariosalvi78 3 hours ago
      with cookies you can restrict them to HttpOnly so that they are not exposed to client-side scripts. This reduces the chances of XSS to access the long-lived access tokens (JWT or session ids).
      • Sohcahtoa82 44 minutes ago
        HttpOnly makes it so XSS can't steal your token, but that won't stop XSS from using your token.
      • littlecranky67 2 hours ago
        This. I store my JWT in a cookie, and the cookie is of course set to HttpOnly,Secure and SameSite=strict. That basically kills XSS. I do not use openid connect, and one of my pet peeves with OIDC is that the access/refresh tokens are always exposed to the JS side (not in a cookie using HttpOnly) in any impl. i've seen.
  • hoppp 3 hours ago
    What if I put a jwt in a session cookie?

    The post is not descriptive enough

    It should explain how to not store JWT instead of just saying JWT is bad.

  • jheriko 3 hours ago
    [dead]
  • well_ackshually 4 hours ago
    [dead]
  • dzonga 6 hours ago
    due to the recent FIFA hack - just a reminder - stop using JWTs
    • dgrin91 4 hours ago
      The Fifa hack had nothing to do with JWTs, it was because FIFA was doing auth on the client side. They would have had the same issue if they used cookie auth.
      • mycall 4 hours ago
        h4ckernews also accessed an Azure Function App that provided direct download URLs for internal FIFA files, including transfer reports and board level data, due to a lack of RBAC access checks.
    • tancop 2 hours ago
      if you are fifa please keep using them in the most insecure way possible. release the infantino files
  • bijowo1676 38 minutes ago
    JWTs are in every bit superior to cookies