The browser can store gigabytes of data

Until recently, I thought browsers could only store a few megabytes of data. I was wrong!

Modern browsers can store gigabytes of data.

Read the "Why this matters: local-first apps" section Why this matters: local-first apps

I've been looking into local-first architectures for web applications lately. I'll define it here as having a database in the browser and syncing data between server and client using a sync engine. In its simplest form, you send the client all the data it might ever need and store it in a browser database like SQLite.
Famous apps in the local-first world include the issue tracker Linear and the design app Figma.

As I read about this, I wondered: can we really store everything in the browser? How big is too big?

Read the "Investigating browser storage" section Investigating browser storage

As a complement to reading the docs, I enlisted Claude (the AI) to write a little tool that writes random data to IndexedDB. You can try it here.
Good for people who don't trust documentation, and to test the quota on a variety of devices and browsers.

The storage docs explain how the allotted quota is computed by each browser. The rules are a bit of a mouthful and vary between browsers.

The best way is to ask the API:

// Example: 24MB of data used out of ~600GB quota in Chrome.
await navigator.storage.estimate()
{
    "quota": 598791846297,
    "usage": 24525537,
    "usageDetails": {
        "indexedDB": 24525537
    }
}

This same info is also in the devtools under Application > Storage.

Note: for navigator.storage to exist, we need to be in a secure context. Basically https or localhost, 127.0.0.1.

A proxy rule I use as mental model: Firefox is the least generous right now, and allows 10GB per origin1.

Extra notes:

  • If you write too much data you'll get a QuotaExceededError: Chrome devtools (Application > Storage) allow simulating a storage quota. Convenient to trigger QuotaExceededError.
  • In Chrome incognito the quota is only 2GB, and the data is cleared when the window closes. Firefox is 10GB, private window or not.
  • I looked at IndexedDB, but the storage quota rules apply the same way to the OPFS (Origin Private File System), the other large browser storage contender.

Read the "Best-effort vs persistent storage" section Best-effort vs persistent storage

A caveat: by default, storage is best-effort.
The browser might decide to remove the data, for instance if the user runs low on disk space.

If you want stronger guarantees, you can ask for persistent storage:

const granted = await navigator.storage.persist();
console.log(granted);
  • Firefox asks the user with a permission prompt (like for geolocation).
  • Chrome just decides on its own based on vague rules2.
  • Safari also decides without prompting the user.

In practice, eviction of best-effort storage is very rare3. My conclusion is that best-effort is fine for any app that also stores data on a server.

Persistent storage might enable true local, serverless apps with the feel of native software - a slightly different use case from the typical web app.

Read the "Conclusion" section Conclusion

So how big is too big? Well, a few gigabytes of storage could be fine. Hundreds of megabytes is not pushing boundaries.

This is a lot of data! For a typical business application, it might mean you can sync all the user data to the client. I think this is true in many, many cases4.

For these applications, sync engines could provide an amazing development experience. I'm pretty excited about that.

Read the "Footnotes" section Footnotes

  1. Technically it's 10GB per eTLD+1 domain, but that doesn't matter much for most apps.

  2. I didn't try very hard but I was not able to get my localhost:9001 app persistent storage. I bookmarked it, allowed push notifications... But that was not the right incantation.

  3. Mentioned on this doc and on this excellent article that describes how IndexedDB can be used as storage for SQLite in the browser.

  4. This is just my current feeling. As a counterpoint, Aaron Boodman makes the case that it's often not possible. But he uses GitHub and Vercel's review commenting feature (the one open to the whole internet) as examples. There is a lot of software that does not deal with as much shared data.