3.7 KiB
date | title | description | tags | ||||
---|---|---|---|---|---|---|---|
2023-06-08 | From ICS to JSON: surfacing anticipated albums | I use MusicHarbor by Marcos Tanaka to track upcoming albums from my favorite artists (typically by syncing my last.fm data with the app.) When I see something new that I want to add to my collection I throw it on a calendar creatively titled Albums. |
|
I use MusicHarbor by Marcos Tanaka to track upcoming albums from my favorite artists (typically by syncing my last.fm data with the app.) When I see something new that I want to add to my collection I throw it on a calendar creatively titled Albums
.
My calendar sits over at Proton which lets you share calendars with other users or as a syncable/downloadable ics
file. As another entry into a list of experiments done because they can be I decided to surface these album events on my now page1.
To do this I installed ics-to-json-extended
and created a data file:
const { AssetCache } = require('@11ty/eleventy-fetch')
const ics = require('ics-to-json-extended')
const { DateTime } = require('luxon')
module.exports = async function () {
const URL = process.env.SECRET_FEED_ALBUM_RELEASES
const icsToJson = ics.default
const asset = new AssetCache('album_release_data')
if (asset.isCacheValid('1h')) return await asset.getCachedValue()
const icsRes = await fetch(URL)
const icsData = await icsRes.text()
const data = icsToJson(icsData)
const albumReleases = data.filter((d) => DateTime.fromISO(d.startDate) > DateTime.now())
return albumReleases.sort((a, b) => new Date(a.startDate) - new Date(b.startDate))
}
We surface the url, require the ICS conversion library, cache and convert the response using Luxon's DateTime
interface to compare the current time and dates in the object returned from the calendar, which look like this:
{
"startDate": "20180505T020000Z",
"endDate": "20180505T060000Z",
"location": "url",
"summary": "Artist - Album"
}
Rendering the output is as simple as: {% raw %}
{% if albumReleases.size > 0 %}
<h2 class="flex-centered">
{% tablericon "calendar-time" "Anticipated albums" %}
Anticipated albums
</h2>
<ul>
{% for album in albumReleases %}
<li>
<strong>{{ album.date | readableDate }}: </strong>
<a href="https://{{album.url}}" title="{{album.title | escape}}">
{{album.title}}
</a>
</li>
{% endfor %}
</ul>
{% endif %}
{% endraw %}
Leaving us with:
-
At this point, a dev playground. ↩︎