chore: analytics

This commit is contained in:
Cory Dransfeldt 2024-03-24 08:15:21 -07:00
parent 6e5c5c5725
commit 7e0259aafd
No known key found for this signature in database
14 changed files with 40 additions and 34 deletions

3
.env
View file

@ -2,8 +2,7 @@ ACCESS_KEY_B2=
SECRET_KEY_B2= SECRET_KEY_B2=
BUCKET_B2= BUCKET_B2=
API_KEY_LASTFM= API_KEY_LASTFM=
SITE_ID= API_KEY_FATHOM=
API_KEY_NETLIFY=
API_KEY_TRAKT= API_KEY_TRAKT=
API_KEY_MOVIEDB= API_KEY_MOVIEDB=
API_TOKEN_READWISE= API_TOKEN_READWISE=

View file

@ -47,11 +47,11 @@ export default {
getPopularPosts: (posts, analytics) => { getPopularPosts: (posts, analytics) => {
return posts return posts
.filter((post) => { .filter((post) => {
if (analytics.find((p) => p.resource === post.url)) return true if (analytics.find((p) => p.pathname === post.url)) return true
}) })
.sort((a, b) => { .sort((a, b) => {
const count = (page) => analytics.filter((p) => p.resource === page.url).pop().count const visitors = (page) => analytics.filter((p) => p.pathname === page.url).pop().visitors
return count(b) - count(a) return visitors(b) - visitors(a)
}) })
}, },

View file

@ -1,6 +1,6 @@
{ {
"name": "coryd.dev", "name": "coryd.dev",
"version": "8.5.1", "version": "8.6.1",
"description": "The source for my personal site. Built using 11ty.", "description": "The source for my personal site. Built using 11ty.",
"type": "module", "type": "module",
"scripts": { "scripts": {

View file

@ -1,23 +1,18 @@
import EleventyFetch from '@11ty/eleventy-fetch' import EleventyFetch from '@11ty/eleventy-fetch'
export default async function () { export default async function () {
const API_KEY_NETLIFY = process.env.API_KEY_NETLIFY const API_KEY_FATHOM = process.env.API_KEY_FATHOM
const SITE_ID = process.env.SITE_ID
const endDate = new Date();
const startDate = new Date(new Date().setDate(endDate.getDate() - 30));
const timezoneOffset = new Date().getTimezoneOffset() / 60;
const url = const url =
`https://analytics.services.netlify.com/v2/${SITE_ID}/ranking/pages?from=${startDate.getTime()}&to=${endDate.getTime()}&timezone=-0${timezoneOffset}00&limit=50` 'https://api.usefathom.com/v1/aggregations?entity=pageview&entity_id=CWSVCDJC&aggregates=pageviews&field_grouping=pathname&sort_by=pageviews:desc&limit=10'
const res = EleventyFetch(url, { const res = EleventyFetch(url, {
duration: '1h', duration: '1h',
type: 'json', type: 'json',
fetchOptions: { fetchOptions: {
headers: { headers: {
Authorization: `Bearer ${API_KEY_NETLIFY}`, Authorization: `Bearer ${API_KEY_FATHOM}`,
}, },
}, },
}).catch() }).catch()
const responseObject = await res const pages = await res
const pages = responseObject['data'] return pages.filter((p) => p.pathname.includes('posts'))
return pages.filter((p) => p.resource.includes('posts')) }
}

View file

@ -15,6 +15,7 @@
{% render "../assets/styles/index.css" %} {% render "../assets/styles/index.css" %}
{% endcapture %} {% endcapture %}
<style>{{ css }}</style> <style>{{ css }}</style>
<script src="https://cdn.usefathom.com/script.js" data-site="CWSVCDJC" defer></script>
<link rel="canonical" href="{{ fullUrl }}" /> <link rel="canonical" href="{{ fullUrl }}" />
<meta property="og:title" content="{{ pageTitle }}" /> <meta property="og:title" content="{{ pageTitle }}" />
<meta name="description" content="{% if description %}{{ description }}{% else %}{{ meta.siteDescription }}{% endif %}" /> <meta name="description" content="{% if description %}{{ description }}{% else %}{{ meta.siteDescription }}{% endif %}" />

View file

@ -3,6 +3,10 @@
{% endcapture %} {% endcapture %}
<style>{{ css }}</style> <style>{{ css }}</style>
<div class="addon-links__wrapper"> <div class="addon-links__wrapper">
{%- if analytics.size > 0 -%}
<div>{% render "partials/popular-posts.liquid", posts:posts, analytics:analytics %}</div> <div>{% render "partials/popular-posts.liquid", posts:posts, analytics:analytics %}</div>
{%- endif -%}
{%- if links.size > 0 -%}
<div>{% render "partials/recent-links.liquid", links:links %}</div> <div>{% render "partials/recent-links.liquid", links:links %}</div>
{%- endif -%}
</div> </div>

View file

@ -1,4 +1,5 @@
{% assign posts = posts | getPopularPosts: analytics %} {% assign posts = posts | getPopularPosts: analytics %}
{% if posts.size > 0 %}
<h2 class="flex--centered"> <h2 class="flex--centered">
{% tablericon "flame" "Popular" %} {% tablericon "flame" "Popular" %}
Popular posts Popular posts
@ -11,4 +12,5 @@
</a> </a>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
{% endif %}

View file

@ -29,6 +29,7 @@
const query = $input.value const query = $input.value
const results = (query.length > 1) ? getSearchResults(query) : [] const results = (query.length > 1) ? getSearchResults(query) : []
if (query === '') renderSearchResults([]) if (query === '') renderSearchResults([])
if (results && fathom) fathom?.trackEvent(`Search query: ${query}`)
renderSearchResults(results) renderSearchResults(results)
}) })

View file

@ -9,7 +9,7 @@ description: 'These are awesome blogs that I enjoy and you may enjoy too.'
{% endcapture %} {% endcapture %}
<style>{{ css }}</style> <style>{{ css }}</style>
<h2 class="page__header">{{ title }}</h2> <h2 class="page__header">{{ title }}</h2>
<p>You can <a href="/blogroll.opml">download an OPML file</a> containing all of these feeds and import them into your RSS reader.</p> <p>You can <a onclick="fathom?.trackEvent('OPML download')" href="/blogroll.opml">download an OPML file</a> containing all of these feeds and import them into your RSS reader.</p>
<div class="blog__banner--grid"> <div class="blog__banner--grid">
{% for blog in blogroll %} {% for blog in blogroll %}
<div class="blog__banner"> <div class="blog__banner">

View file

@ -8,9 +8,12 @@ description: 'Referral links for services that I use.'
Referral links for services I use. I save some money, and you do as well if you choose to use them. Referral links for services I use. I save some money, and you do as well if you choose to use them.
- [NextDNS](https://nextdns.io/?from=m56mt3z6) <ul>
- [DNSimple](https://dnsimple.com/r/3a7cbb9e15df8f) <li><a onclick="fathom?.trackEvent('Fathom referral')" href="https://usefathom.com/ref/TFFKEZ">Fathom Analytics</a></li>
- [Bunny.net](https://bunny.net?ref=revw3mehej) <li><a onclick="fathom?.trackEvent('NextDNS referral')" href="https://nextdns.io/?from=m56mt3z6">NextDNS</a></li>
- [Feedpress](https://feedpress.com/?affid=34370) <li><a onclick="fathom?.trackEvent('DNSimple referral')" href="https://dnsimple.com/r/3a7cbb9e15df8f">DNSimple</a></li>
- [Proton](https://pr.tn/ref/X775YX40Z50G) <li><a onclick="fathom?.trackEvent('Bunny.net referral')" href="https://bunny.net?ref=revw3mehej">Bunny.net</a></li>
- [DigitalOcean](https://m.do.co/c/3635bf99aee2) <li><a onclick="fathom?.trackEvent('Feedpress referral')" href="https://feedpress.com/?affid=34370">Feedpress</a></li>
<li><a onclick="fathom?.trackEvent('Proton referral')" href="https://pr.tn/ref/X775YX40Z50G">Proton</a></li>
<li><a onclick="fathom?.trackEvent('DigitalOcean referral')" href="https://m.do.co/c/3635bf99aee2">DigitalOcean</a></li>
</ul>

View file

@ -60,11 +60,12 @@ Software and services that I use for work and my own enjoyment.
<h3>Services</h3> <h3>Services</h3>
- [HEY](https://hey.com): it's the email service that works best for me and I quite enjoy their calendar. - [HEY](https://hey.com): it's the email service that works best for me and I quite enjoy their calendar.
- [NextDNS](https://nextdns.io/?from=m56mt3z6): a privacy-focused, set it and forget it DNS service. I use their security features on my home network and a profile with strict ad-blocking rules on all of my devices. - <a onclick="fathom?.trackEvent('Fathom referral')" href="https://usefathom.com/ref/TFFKEZ">Fathom Analytics</a>: privacy-friendly, lightweight and nicely designed analytics.
- [DNSimple](https://dnsimple.com/r/3a7cbb9e15df8f): a robust, user-friendly DNS provider and registrar. I moved my domains here after my old provider was acquired. - <a onclick="fathom?.trackEvent('NextDNS referral')" href="https://nextdns.io/?from=m56mt3z6">NextDNS</a>: a privacy-focused, set it and forget it DNS service. I use their security features on my home network and a profile with strict ad-blocking rules on all of my devices.
- [Bunny.net](https://bunny.net?ref=revw3mehej): a powerful and affordable CDN. I use it to serve assets on my personal site (most notable my now page — raw assets are fetched by Eleventy and then optimized to webp). - <a onclick="fathom?.trackEvent('DNSimple referral')" href="https://dnsimple.com/r/3a7cbb9e15df8f">DNSimple</a>: a robust, user-friendly DNS provider and registrar. I moved my domains here after my old provider was acquired.
- [Netlify](https://netlify.com): I use their hosting, forms, webhooks and analytics. It's a reliable and feature-rich platform. - <a onclick="fathom?.trackEvent('Bunny.net referral')" href="https://bunny.net?ref=revw3mehej">Bunny.net</a>: a powerful and affordable CDN. I use it to serve assets on my personal site (most notable my now page — raw assets are fetched by Eleventy and then optimized to webp).
- [Feedpress](https://feedpress.com/?affid=34370): they've been around for a while now and don't change much (nor do they need to), but look no further for reliable, helpful feed analytics. - [Netlify](https://netlify.com): I use their hosting, forms and webhooks. It's a reliable and feature-rich platform.
- <a onclick="fathom?.trackEvent('Feedpress referral')" href="https://feedpress.com/?affid=34370">Feedpress</a>: they've been around for a while now and don't change much (nor do they need to), but look no further for reliable, helpful feed analytics.
- [Kagi](https://kagi.com): the only search engine at this point (or so it seems) that's interested in surfacing quality results. - [Kagi](https://kagi.com): the only search engine at this point (or so it seems) that's interested in surfacing quality results.
- [Readwise Reader](https://readwise.io/read): an impressive RSS reader that handles all the functionality you'd expect from a read it later app beautifully. It's as powerful as it is flexible. - [Readwise Reader](https://readwise.io/read): an impressive RSS reader that handles all the functionality you'd expect from a read it later app beautifully. It's as powerful as it is flexible.
- [Todoist](https://todoist.com): the task manager I always return to for it's flexibility and endless integrations. - [Todoist](https://todoist.com): the task manager I always return to for it's flexibility and endless integrations.
@ -75,6 +76,6 @@ Software and services that I use for work and my own enjoyment.
- [Slack](http://slack.com): I have a family Slack set up to avoid group text messages and am in a few other community Slacks. - [Slack](http://slack.com): I have a family Slack set up to avoid group text messages and am in a few other community Slacks.
- [Discord](http://discord.com): I don't _like_ Discord but, for better or worse, it's where some communities I frequent are. - [Discord](http://discord.com): I don't _like_ Discord but, for better or worse, it's where some communities I frequent are.
- [Backblaze](https://backblaze.com): It backs up my MacBook Air and attached storage drive and I don't have to think about it. - [Backblaze](https://backblaze.com): It backs up my MacBook Air and attached storage drive and I don't have to think about it.
- [Proton VPN](https://pr.tn/ref/X775YX40Z50G): a reliable, trustworthy VPN with all of the features you'd expect from such a service. - <a onclick="fathom?.trackEvent('Proton referral')" href="https://pr.tn/ref/X775YX40Z50G">Proton VPN</a>: a reliable, trustworthy VPN with all of the features you'd expect from such a service.
Check out [uses.tech](https://uses.tech) for more lists like this one. Check out [uses.tech](https://uses.tech) for more lists like this one.

View file

@ -11,7 +11,7 @@ A rundown of privacy tools that work well with Apple's technology ecosystem.<!--
Ubiquitous free email providers profit by mining user data (whether humans are involved or not). Your inbox acts as a key to your digital life, and you should avoid using any provider that monetizes its contents. Ubiquitous free email providers profit by mining user data (whether humans are involved or not). Your inbox acts as a key to your digital life, and you should avoid using any provider that monetizes its contents.
- [Fastmail](https://ref.fm/u30190984)[^2]: based in Melbourne, Australia Fastmail offers a range of affordably priced plans with a focus on support for open standards (including active development support for [JMAP](https://jmap.io) and the [Cyrus IMAP email server](https://fastmail.blog/open-technologies/why-we-contribute/)). They also [articulate a clear commitment to protecting and respecting your privacy](https://www.fastmail.com/values/) and offer an extensive [rundown of the privacy and security measures they employ on their site](https://www.fastmail.com/privacy-and-security/). - <a onclick="fathom?.trackEvent('Fastmail referral') href="https://ref.fm/u30190984">Fastmail</a>[^2]: based in Melbourne, Australia Fastmail offers a range of affordably priced plans with a focus on support for open standards (including active development support for [JMAP](https://jmap.io) and the [Cyrus IMAP email server](https://fastmail.blog/open-technologies/why-we-contribute/)). They also [articulate a clear commitment to protecting and respecting your privacy](https://www.fastmail.com/values/) and offer an extensive [rundown of the privacy and security measures they employ on their site](https://www.fastmail.com/privacy-and-security/).
- I would also recommend exploring their [masked email implementation](https://www.fastmail.help/hc/en-us/articles/4406536368911-Masked-Email), which integrates seamlessly with [1Password](https://1password.com) (though using 1Password isn't required). - I would also recommend exploring their [masked email implementation](https://www.fastmail.help/hc/en-us/articles/4406536368911-Masked-Email), which integrates seamlessly with [1Password](https://1password.com) (though using 1Password isn't required).
- [mailbox.org](https://mailbox.org): based in Germany, [mailbox.org](http://mailbox.org) also has [a long history](https://mailbox.org/en/company#our-history) and [commitment to privacy](https://mailbox.org/en/company#our-mission). Their service is reliable, straightforward and fully featured (it's based off of a customized implementation [Open-Xchange](https://www.open-xchange.com)) and supports features like incoming address blocking, PGP support and so forth. - [mailbox.org](https://mailbox.org): based in Germany, [mailbox.org](http://mailbox.org) also has [a long history](https://mailbox.org/en/company#our-history) and [commitment to privacy](https://mailbox.org/en/company#our-mission). Their service is reliable, straightforward and fully featured (it's based off of a customized implementation [Open-Xchange](https://www.open-xchange.com)) and supports features like incoming address blocking, PGP support and so forth.
- [Proton Mail](http://protonmail.com): Proton offers a host of encrypted tools, ranging from mail to drive, calendaring and VPN services. They're also the only option in this list that includes end-to-end encryption. The service is extremely polished and reliable but, it's worth noting, doesn't support access to your email via open standards like IMAP/SMTP without the use of a cumbersome, desktop-only, bridge application. - [Proton Mail](http://protonmail.com): Proton offers a host of encrypted tools, ranging from mail to drive, calendaring and VPN services. They're also the only option in this list that includes end-to-end encryption. The service is extremely polished and reliable but, it's worth noting, doesn't support access to your email via open standards like IMAP/SMTP without the use of a cumbersome, desktop-only, bridge application.

View file

@ -11,7 +11,7 @@ I've used (and/or tried) nearly every email service I've heard of and have stuck
### Kicking things off ### Kicking things off
Register for an account at [Fastmail](https://ref.fm/u30190984)[^3] — you'll be run through their lightweight onboarding process which allows you to select an address at a domain they own or use your own. If you use your own, they'll guide you through configuring the DNS records for it, often with registrar specific instructions. Register for an account at <a onclick="fathom?.trackEvent('Fastmail referral') href="https://ref.fm/u30190984">Fastmail</a>[^3] — you'll be run through their lightweight onboarding process which allows you to select an address at a domain they own or use your own. If you use your own, they'll guide you through configuring the DNS records for it, often with registrar specific instructions.
They also offer [extensive documentation](https://www.fastmail.com/help/domain_management_custom_dns.html) on this process and offer a UI that validates that the records you have set are correct. For example, your finalized records would look like the following: They also offer [extensive documentation](https://www.fastmail.com/help/domain_management_custom_dns.html) on this process and offer a UI that validates that the records you have set are correct. For example, your finalized records would look like the following:

View file

@ -134,7 +134,7 @@ This is most impactful on [my now page](https://coryd.dev/now) which is populate
``` ```
{% endraw %} {% endraw %}
For this page in particular, the images that are rendered above the fold are set to load as `eager` to mitigate performance impacts related to [too much lazy loading](https://web.dev/lcp-lazy-loading/). These images are fetched from caches hosted at [Bunny.net](https://bunny.net?ref=revw3mehej) when the site is built. For this page in particular, the images that are rendered above the fold are set to load as `eager` to mitigate performance impacts related to [too much lazy loading](https://web.dev/lcp-lazy-loading/). These images are fetched from caches hosted at <a onclick="fathom?.trackEvent('Bunny.net referral')" href="https://bunny.net?ref=revw3mehej">Bunny.net</a> when the site is built.
All of these boilerplate steps leave us with a quick to load, accessible and resilient site: All of these boilerplate steps leave us with a quick to load, accessible and resilient site: