Kyle Cordes
BTW, GA4 also hooks the history API to capture SPA navigation without assistance.
(Though GA4 doesn't help with "don't track more than the bare minimum needed for this site" goal.)
A couple years ago, I was hanging out with some guys at my buddy Buddy’s (it's literally his first name) house. We were chatting about typical nerd stuff, when Samuel brought up a self-hostable, privacy-minded analytics service he’s been using: Plausible. I had never heard of it, but he gave a solid pitch, so I told him I'd look into it.
It was good timing. I had been on the hunt for an alternative to Google Analytics for some time. I’ve used it for years, but as time has gone on, there’s been increasing pressure to “de-Google” in this way for a few reasons:
I decided to try out the self-hosted version of Plausible, which I put up on a cheap DigitalOcean droplet. That process was a little cumbersome, but it was extremely satisfying seeing it in action. So satisfying, in fact, that I've made a tool for automating the self-hosting setup.
Looking back, here are the key things I’ve loved about the experience:
For being relatively small in the analytics space, I was very pleased by how thorough Plausible’s documentation is – and extremely SEO-friendly. For the most part, when I needed something, I did a quick Google search with a few keywords, and the first result was exactly what I needed, right in their documentation.
They appear to do a great job at anticipating the questions and concerns of their users, and solid documentation is the result of that. But some of the credit for this might also be due to the fact that the all of the documentation is available on GitHub, open to pull requests, and updated regularly.
This all was a refreshing change from using Google Analytics. I’d often find several different sources that roughly relate to what I’m searching for, often scattered throughout Google’s own resources. I’m sure there are reasons for that (wider feature scope, multiple implementation approaches, maturity & complexity of the product, etc.), but it became a pretty significant point of friction for me, especially when all I wanted to do was get something up & tracking. Since then, GA4's been released, and I imagine the landscape is even worse.
I don’t need any sort of complex event or user action analytics. I have (and probably will for quite some time) pretty simple things I’d like to keep track of:
And some of those events have custom data I’d like to attach to each event (ex: the license type someone purchased).
In GA, it seemed relatively straightforward to send an event, which varied based on the implementation you’re using (Global Site Tag vs. “standard”) — I just need to specify an event action, category, label, and value. But in reality, it was a little more complicated to implement. Among my issues:
value field. Instead, I’d need to set up custom dimensions, which, paired with “metrics,” is an entirely different concept to wrap my head around.Primed with this less-than-pleasant experience, Plausible’s custom event API was a godsend. After installing the core Plausible script, it was as simple as setting up the global window.plausible variable:
<script>window.plausible = window.plausible || function() { (window.plausible.q = window.plausible.q || []).push(arguments) }</script>
… and then sending an event:
window.plausible('Sign Up');
Or, with custom data that pertains certain events.
window.plausible('Purchase', {
  // Optional custom data.
  props: {
    product_slug: 'my-product-slug',
    total: 44
  },
  // Optional callback method fired after event is sent.
  callback: () => {
    console.log('it sent!');
  }
});
After setting up a goal in the Plausible admin, all that data appears in highly-digestible way. Here’s an example of how my “Purchase” event looked at one point:

Good enough for me. And if I need to created some more complex reports with that data, it’s a matter of accessing the database and running with it.
When I moved from Google Analytics to Plausible on TypeIt’s site, I tweeted about my Lighthouse score improving by eight points. At the time, it was a Gatsby site with a lot of JavaScript in play, and Lighthouse doesn’t report on the actual user experience, but even so, it was a little jarring.
Some of the reason for this boost was pretty apparent. According Plausible, its script is many times smaller than Google’s, and that means less time downloading & parsing code. But according to Lighthouse and WebPageTest reports, other factors were due credit as well.
One of those other factors was the amount of work Google Analytics performs on the main thread, choking out any other processes that may actually support the user’s experience. Here’s a shot of the problem scripts on typeitjs.com. GA comes in at the third most thread-obnoxious script (divert your eyes from the top two… remember, it was a Gatsby site).

The other notable factor was the caching policy used by Google to serve its scripts. WebPageTest wasn’t happy about it:

These concerns don’t magically go away by moving from GA to another provider, but the lightweight and self-hosted nature of Plausible sure makes it easier to mitigate them. Addressing the first concern is automatic, and the second is solved by setting the appropriate caching headers when the script is served from my DigitalOcean droplet.
This is a small perk I didn’t at all expect to see by moving to Plausible, but it’s an appreciated one. By default, the version of Google Analytics I used collected its data only by executing after a page is loaded. As such, observing a user’s flow through several different pieces is achieved by stitching together the data from subsequent full page loads.
It’s a fine way to go about it, but also made it a bit of a hassle to track views in single page applications. In SPAs, there’s often a single page load, after which navigation is handled entirely client-side. Google did a decent job equipping people to accurately track page views in SPAs by providing good documentation on it. The short of it is that you need to manually send a pageview event whenever client-side navigation occurs.
This site was previously built on NextJS, and it used the recommended Global Site Tag approach. I had to pull that off by listening for a routeChangeComplete event, which looked something like this (stripped down and a little contrived for simplicity):
const pageView = function (url) {
  gtag("config", GA_TRACKING_ID, {
    page_path: url,
  });
};
export default function App({ Component, pageProps }) {
  const router = useRouter();
	// Track each client-side page navigation.
  useEffect(() => {
    const handleRouteChange = (url) => pageView(url);
    router.events.on("routeChangeComplete", handleRouteChange);
    return () => {
      router.events.off("routeChangeComplete", handleRouteChange);
    };
  }, [router.events]);
	// Track the initial page load.
  useEffect(() => {
    pageView(window.location.pathname);
  }, []);
  return <Component {...pageProps} />;
}
It’s nothing terribly complicated, but it’s also something I didn't really want to worry about myself.
Thankfully, Plausible takes care of this automatically by piggy-backing on client-side routers’ usage of the History API. In order to simulate page views performed in the context of an SPA, libraries like react-router will use history.pushState() to place these views into a user’s browser history. Plausible leverages that integration, meaning I don’t have to do anything to correctly track SPA page views after installing the script (unless I want to). It just worked.
Google Analytics is mature and feature-rich service, but there’s a reason it’s free — all that data it collects is used in a wide variety of ways to benefit the business. I don’t have any particular issue with that model in and of itself, but the scope of its implications also doesn’t sit great with me.
Among Plausible’s selling points, however, are its intentional focuses on privacy and data ownership. Its website lists a wealth of ways they lean into that — things like no cookies, no cross-site tracking, no sharing data with any third party, and being able to self-host the entire thing.
And that also means it’s easy to get my data out of Plausible too, like with its Stats API, which I use to display analytics data on my personal dashboard. One nice thing about self-hosting the tool is that there are virtually no limits to how much of my own data I can request on demand from my droplet. There’s an hourly limit, but even that can be manually increased. That leaves me feeling like my data really is easily accessible, safe, and mine.
Ad blockers and privacy-focused browsers (like Brave) are a growing problem for the client-side analytics space. When using either, it’s increasingly common for third party client-side requests to be blocked by default, impacting the accuracy of your data.
One approach to address this is proxying analytics requests through your domain, which convinces privacy tools to consider the requests to be trustworthy.
Wiring this up with Google Analytics was possible, but it was neither simple nor maintainable in the long term. It required serving the analytics script locally (by either proxying it from Google or maintaining a modified copy yourself), and then passing every event request through your domain as well. Modern tooling like serverless functions make it less cumbersome, but even so, it’s tedious and potentially unreliable if/when Google ever pushed an update to the script.
It’d be remiss of me not to mention the strides Google Analytics has taken more recently to help tackle some of these issues, such as the introduction of server-side tagging. But the amount of time and technical overhead required to take advantage of it is still more than I’m willing to take on for my side projects.
Plausible, on the other hand, has dedicated documentation on how to proxy its script through your own domain, covering a host of platforms, including two I use a lot — Netlify and Vercel. For my Next.js site (at the time), that’s as simple as setting up a couple of rewrites:
// next.config.js 
module.exports = {
  async rewrites() {
    return [
      {
        source: "/js/numbers.js",
        destination: "https://analytics.macarthur.me/js/plausible.js",
      },
      {
        source: "/api/event",
        destination: "https://analytics.macarthur.me/api/event",
      },
    ];
  },
};
And then installing the script using my proxied destination:
<!-- The `<head>` of each page --><script defer data-domain="macarthur.me" src="/js/numbers.js"></script>
With those pieces in place, I’m able to successfully reach my analytics server without issue — even using privacy-focused browsers like Brave. And not to mention, that more reliable data doesn't come at the cost of violating privacy regulations (what a mess).
Plausible is a great example of how attractive and products without a lot of features can be. I’m giving up a large set of tricks and functionality by dumping Google Analytics. But between the simplicity, performance, privacy ease, and everything else in between, the tradeoff has been more than worth it for me. And it’s helped get me one step closer to weaning myself off big tech as much as I reasonably can.
When I first made this migration, I reserved the right to abandon ship if needed. Two years later, I still have no intention of doing so.
(By the way, if you'd like to fast-track trying self-hosted Plausible on your own virtual machine, consider trying out the Plausible Bootstrapper to save you some time and sanity.)
Get blog posts like this in your inbox.
May be irregular. Unsubscribe whenever.BTW, GA4 also hooks the history API to capture SPA navigation without assistance.
(Though GA4 doesn't help with "don't track more than the bare minimum needed for this site" goal.)