feat: mock when running locally

This commit is contained in:
Cory Dransfeldt 2024-02-08 16:42:17 -08:00
parent 12f4929dd3
commit 6757a60134
No known key found for this signature in database
22 changed files with 1342 additions and 300 deletions

View file

@ -18,7 +18,7 @@ export const img = async (
sizes = '90vw', sizes = '90vw',
formats = ['avif', 'webp', 'jpeg'] formats = ['avif', 'webp', 'jpeg']
) => { ) => {
const isLocal = src.includes('src/assets'); const isLocal = src?.includes('src/assets');
const imageExists = async () => { const imageExists = async () => {
try { try {
return await fetch(src, { method: 'HEAD' }).then((res) => res.ok); return await fetch(src, { method: 'HEAD' }).then((res) => res.ok);
@ -69,5 +69,9 @@ export const img = async (
); );
}; };
return isLocal ? await generateImage() : await imageExists().then(async (exists) => (exists ? await generateImage() : await generatePlaceholder())); if (process.env.ELEVENTY_PRODUCTION) {
}; return isLocal ? await generateImage() : await imageExists().then(async (exists) => (exists ? await generateImage() : await generatePlaceholder()));
} else {
return await generatePlaceholder();
}
};

View file

@ -1,10 +1,10 @@
{ {
"name": "coryd.dev", "name": "coryd.dev",
"version": "5.4.13", "version": "5.5.0",
"description": "The source for my personal site. Built using 11ty and hosted on Netlify.", "description": "The source for my personal site. Built using 11ty and hosted on Netlify.",
"type": "module", "type": "module",
"scripts": { "scripts": {
"start": "ELEVENTY_PRODUCTION=false eleventy --serve", "start": "eleventy --serve",
"start:search": "run-s build:11ty index:local", "start:search": "run-s build:11ty index:local",
"start:quick": "eleventy --serve --incremental --ignore-initial", "start:quick": "eleventy --serve --incremental --ignore-initial",
"build": "ELEVENTY_PRODUCTION=true eleventy && npm run build:index", "build": "ELEVENTY_PRODUCTION=true eleventy && npm run build:index",

View file

@ -5,20 +5,25 @@ import { DateTime } from 'luxon'
export default async function () { export default async function () {
const URL = process.env.SECRET_FEED_ALBUM_RELEASES const URL = process.env.SECRET_FEED_ALBUM_RELEASES
const icsToJson = ics.default const icsToJson = ics.default
const asset = new AssetCache('album_release_data') if (process.env.ELEVENTY_PRODUCTION) {
if (asset.isCacheValid('1h')) return await asset.getCachedValue() const asset = new AssetCache('album_release_data')
const icsRes = await fetch(URL) if (asset.isCacheValid('1h')) return await asset.getCachedValue()
const icsData = await icsRes.text() const icsRes = await fetch(URL)
const data = icsToJson(icsData) const icsData = await icsRes.text()
const albumReleases = data const data = icsToJson(icsData)
.filter((d) => DateTime.fromISO(d.startDate) > DateTime.now()) const albumReleases = data
.sort((a, b) => new Date(a.startDate) - new Date(b.startDate)) .filter((d) => DateTime.fromISO(d.startDate) > DateTime.now())
.map((release) => { .sort((a, b) => new Date(a.startDate) - new Date(b.startDate))
return { .map((release) => {
date: release.startDate, return {
url: release.location, date: release.startDate,
title: release.summary.replace(/\\/g, ''), url: release.location,
} title: release.summary.replace(/\\/g, ''),
}) }
return albumReleases })
await asset.save(albumReleases, 'json')
return albumReleases
} else {
return {}
}
} }

View file

@ -1,15 +1,12 @@
import EleventyFetch from '@11ty/eleventy-fetch' import EleventyFetch from '@11ty/eleventy-fetch'
import AlbumsMock from './json/mocks/albums.js'
const ALBUM_DENYLIST = ['no-love-deep-web', 'unremittance'] const ALBUM_DENYLIST = ['no-love-deep-web', 'unremittance']
export default async function () { export default async function () {
const MUSIC_KEY = process.env.API_KEY_LASTFM const MUSIC_KEY = process.env.API_KEY_LASTFM
const url = `https://ws.audioscrobbler.com/2.0/?method=user.gettopalbums&user=coryd_&api_key=${MUSIC_KEY}&limit=8&format=json&period=7day` const url = `https://ws.audioscrobbler.com/2.0/?method=user.gettopalbums&user=coryd_&api_key=${MUSIC_KEY}&limit=8&format=json&period=7day`
const res = EleventyFetch(url, { const formatAlbumData = (albums) => albums.map((album) => {
duration: '1h',
type: 'json',
}).catch()
const data = await res
return data['topalbums']['album'].map((album) => {
return { return {
title: album['name'], title: album['name'],
artist: album['artist']['name'], artist: album['artist']['name'],
@ -31,4 +28,15 @@ export default async function () {
type: 'album', type: 'album',
} }
}) })
if (process.env.ELEVENTY_PRODUCTION) {
const res = EleventyFetch(url, {
duration: '1h',
type: 'json',
}).catch()
const data = await res
return formatAlbumData(data['topalbums']['album'])
} else {
return formatAlbumData(AlbumsMock);
}
} }

View file

@ -1,18 +1,24 @@
import EleventyFetch from '@11ty/eleventy-fetch' import EleventyFetch from '@11ty/eleventy-fetch'
import AnalyticsMock from './json/mocks/analytics.js'
export default async function () { export default async function () {
const API_KEY_PLAUSIBLE = process.env.API_KEY_PLAUSIBLE const API_KEY_PLAUSIBLE = process.env.API_KEY_PLAUSIBLE
const url = const url =
'https://plausible.io/api/v1/stats/breakdown?site_id=coryd.dev&period=6mo&property=event:page&limit=30' 'https://plausible.io/api/v1/stats/breakdown?site_id=coryd.dev&period=6mo&property=event:page&limit=30'
const res = EleventyFetch(url, { let pages;
duration: '1h', if (process.env.ELEVENTY_PRODUCTION) {
type: 'json', const res = EleventyFetch(url, {
fetchOptions: { duration: '1h',
headers: { type: 'json',
Authorization: `Bearer ${API_KEY_PLAUSIBLE}`, fetchOptions: {
headers: {
Authorization: `Bearer ${API_KEY_PLAUSIBLE}`,
},
}, },
}, }).catch()
}).catch() pages = await res
const pages = await res } else {
pages = AnalyticsMock
}
return pages.results.filter((p) => p.page.includes('posts')) return pages.results.filter((p) => p.page.includes('posts'))
} }

View file

@ -1,5 +1,6 @@
import EleventyFetch from '@11ty/eleventy-fetch'; import EleventyFetch from '@11ty/eleventy-fetch';
import mbidPatches from './json/mbid-patches.js'; import mbidPatches from './json/mbid-patches.js';
import ArtistsMock from './json/mocks/artists.js'
const mbidMap = (artist) => { const mbidMap = (artist) => {
return mbidPatches[artist.toLowerCase()] || ''; return mbidPatches[artist.toLowerCase()] || '';
@ -13,12 +14,7 @@ const removeAccents = (inputStr) => {
export default async function () { export default async function () {
const MUSIC_KEY = process.env.API_KEY_LASTFM; const MUSIC_KEY = process.env.API_KEY_LASTFM;
const url = `https://ws.audioscrobbler.com/2.0/?method=user.gettopartists&user=coryd_&api_key=${MUSIC_KEY}&limit=8&format=json&period=7day`; const url = `https://ws.audioscrobbler.com/2.0/?method=user.gettopartists&user=coryd_&api_key=${MUSIC_KEY}&limit=8&format=json&period=7day`;
const res = EleventyFetch(url, { const formatArtistData = (artists) => artists.map((artist) => {
duration: '1h',
type: 'json',
}).catch();
const data = await res;
return data['topartists']['artist'].map((artist) => {
let mbid = artist['mbid']; let mbid = artist['mbid'];
// mbid mismatches // mbid mismatches
@ -40,4 +36,15 @@ export default async function () {
type: 'artist', type: 'artist',
}; };
}); });
if (process.env.ELEVENTY_PRODUCTION) {
const res = EleventyFetch(url, {
duration: '1h',
type: 'json',
}).catch();
const data = await res;
return formatArtistData(data['topartists']['artist'])
} else {
return formatArtistData(ArtistsMock);
}
} }

View file

@ -1,70 +1,75 @@
import jsdom from 'jsdom' import jsdom from 'jsdom'
import { AssetCache } from '@11ty/eleventy-fetch' import { AssetCache } from '@11ty/eleventy-fetch'
import BooksMock from './json/mocks/books.js'
const { JSDOM } = jsdom const { JSDOM } = jsdom
export default async function () { export default async function () {
const COOKIE = process.env.COOKIE_STORYGRAPH const COOKIE = process.env.COOKIE_STORYGRAPH
const url = 'https://app.thestorygraph.com/currently-reading/coryd' const url = 'https://app.thestorygraph.com/currently-reading/coryd'
const asset = new AssetCache('books_data') if (process.env.ELEVENTY_PRODUCTION) {
if (asset.isCacheValid('1h')) return await asset.getCachedValue() const asset = new AssetCache('books_data')
const data = [] if (asset.isCacheValid('1h')) return await asset.getCachedValue()
await fetch(url, { const data = []
headers: { await fetch(url, {
Cookie: COOKIE, headers: {
}, Cookie: COOKIE,
}) },
.then((res) => res.text()) })
.then((html) => { .then((res) => res.text())
const DOM = new JSDOM(html) .then((html) => {
const doc = DOM.window.document const DOM = new JSDOM(html)
const bookCount = doc.querySelectorAll('.book-pane-content').length const doc = DOM.window.document
const titles = doc.querySelectorAll('.book-title-author-and-series h3 > a') const bookCount = doc.querySelectorAll('.book-pane-content').length
const authors = doc.querySelectorAll('.book-title-author-and-series h3 p:last-of-type > a') const titles = doc.querySelectorAll('.book-title-author-and-series h3 > a')
const images = doc.querySelectorAll('.md\\:block .book-cover img') const authors = doc.querySelectorAll('.book-title-author-and-series h3 p:last-of-type > a')
const urls = doc.querySelectorAll('.md\\:block .book-cover a') const images = doc.querySelectorAll('.md\\:block .book-cover img')
const percentages = doc.querySelectorAll('.md\\:block .progress-tracker-pane .font-semibold') const urls = doc.querySelectorAll('.md\\:block .book-cover a')
const dates = doc.querySelectorAll('.md\\:block .action-menu a p') const percentages = doc.querySelectorAll('.md\\:block .progress-tracker-pane .font-semibold')
const dates = doc.querySelectorAll('.md\\:block .action-menu a p')
for (let i = 0; i < bookCount; i++) { for (let i = 0; i < bookCount; i++) {
const date = new Date( const date = new Date(
dates[i]?.textContent.replace('Started ', '').split('\n')[0] dates[i]?.textContent.replace('Started ', '').split('\n')[0]
).toLocaleString('en-US', { ).toLocaleString('en-US', {
timeZone: 'America/Los_Angeles', timeZone: 'America/Los_Angeles',
}) })
if (!data[i]) { if (!data[i]) {
data.push({ title: titles[i]?.textContent }) data.push({ title: titles[i]?.textContent })
data.push({ author: authors[i]?.textContent }) data.push({ author: authors[i]?.textContent })
data.push({ data.push({
image: images[i].src.replace( image: images[i].src.replace(
'https://cdn.thestorygraph.com',
'https://cd-books.b-cdn.net'
),
})
data.push({ url: `https://app.thestorygraph.com${urls[i].href}` })
data.push({ percentage: percentages[i]?.textContent })
data.push({
dateAdded: date,
})
data.push({ type: 'book' })
}
if (data[i]) {
data[i]['title'] = titles[i]?.textContent
data[i]['author'] = authors[i]?.textContent
data[i]['image'] = images[i]?.src.replace(
'https://cdn.thestorygraph.com', 'https://cdn.thestorygraph.com',
'https://cd-books.b-cdn.net' 'https://cd-books.b-cdn.net'
), )
}) data[i]['url'] = `https://app.thestorygraph.com${urls[i]?.href}`
data.push({ url: `https://app.thestorygraph.com${urls[i].href}` }) data[i]['percentage'] = percentages[i]?.textContent
data.push({ percentage: percentages[i]?.textContent }) data[i]['dateAdded'] = date
data.push({ data[i]['type'] = 'book'
dateAdded: date, }
})
data.push({ type: 'book' })
} }
})
if (data[i]) { const books = data.filter((book) => book.title)
data[i]['title'] = titles[i]?.textContent await asset.save(books, 'json')
data[i]['author'] = authors[i]?.textContent return books
data[i]['image'] = images[i]?.src.replace( } else {
'https://cdn.thestorygraph.com', return BooksMock
'https://cd-books.b-cdn.net' }
)
data[i]['url'] = `https://app.thestorygraph.com${urls[i]?.href}`
data[i]['percentage'] = percentages[i]?.textContent
data[i]['dateAdded'] = date
data[i]['type'] = 'book'
}
}
})
const books = data.filter((book) => book.title)
await asset.save(books, 'json')
return books
} }

View file

@ -0,0 +1,258 @@
export default [
{
"artist": {
"url": "https://www.last.fm/music/Hauntologist",
"name": "Hauntologist",
"mbid": "3cb2b7a7-53e6-4b0c-9143-16227fc5d1c3"
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/5846746d971a039ac66e120f96a73d1b.jpg"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/5846746d971a039ac66e120f96a73d1b.jpg"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/5846746d971a039ac66e120f96a73d1b.jpg"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/5846746d971a039ac66e120f96a73d1b.jpg"
}
],
"mbid": "c7a88e0d-681c-46b7-be5a-8af7c660a57a",
"url": "https://www.last.fm/music/Hauntologist/Hollow",
"playcount": "32",
"@attr": {
"rank": "1"
},
"name": "Hollow"
},
{
"artist": {
"url": "https://www.last.fm/music/dissimulator",
"name": "dissimulator",
"mbid": "6c3b93e1-d3e7-413c-b030-b6ddd918676f"
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/4187cd6dcf48fdc84cfd9cbf2c9f7cca.jpg"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/4187cd6dcf48fdc84cfd9cbf2c9f7cca.jpg"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/4187cd6dcf48fdc84cfd9cbf2c9f7cca.jpg"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/4187cd6dcf48fdc84cfd9cbf2c9f7cca.jpg"
}
],
"mbid": "87de5dc1-e7ec-4011-a9a7-38b875398a58",
"url": "https://www.last.fm/music/dissimulator/Lower+Form+Resistance",
"playcount": "21",
"@attr": {
"rank": "2"
},
"name": "Lower Form Resistance"
},
{
"artist": {
"url": "https://www.last.fm/music/Aesop+Rock",
"name": "Aesop Rock",
"mbid": "aba64937-3334-4c65-90a1-4e6b9d4d7ada"
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/f48ad490ad9259825a1bf155be03f685.jpg"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/f48ad490ad9259825a1bf155be03f685.jpg"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/f48ad490ad9259825a1bf155be03f685.jpg"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/f48ad490ad9259825a1bf155be03f685.jpg"
}
],
"mbid": "8db6e790-7d34-453d-9ac0-f8b6a91b732d",
"url": "https://www.last.fm/music/Aesop+Rock/Integrated+Tech+Solutions",
"playcount": "18",
"@attr": {
"rank": "3"
},
"name": "Integrated Tech Solutions"
},
{
"artist": {
"url": "https://www.last.fm/music/Sepulcher",
"name": "Sepulcher",
"mbid": "48cb1769-b1fe-4730-a088-d83be4dc01ef"
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/3e7b1e98d0d303e60c6d0c8d596d0c33.jpg"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/3e7b1e98d0d303e60c6d0c8d596d0c33.jpg"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/3e7b1e98d0d303e60c6d0c8d596d0c33.jpg"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/3e7b1e98d0d303e60c6d0c8d596d0c33.jpg"
}
],
"mbid": "bc021289-1ec3-4057-864e-5709b11ed5e1",
"url": "https://www.last.fm/music/Sepulcher/Panoptic+Horror",
"playcount": "14",
"@attr": {
"rank": "4"
},
"name": "Panoptic Horror"
},
{
"artist": {
"url": "https://www.last.fm/music/AFI",
"name": "AFI",
"mbid": ""
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/aaab03cb19f0487383cb19957a5d92b7.png"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/aaab03cb19f0487383cb19957a5d92b7.png"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/aaab03cb19f0487383cb19957a5d92b7.png"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/aaab03cb19f0487383cb19957a5d92b7.png"
}
],
"mbid": "0f50ea53-37b4-31ef-aff2-e1d9b8763517",
"url": "https://www.last.fm/music/AFI/Crash+Love",
"playcount": "13",
"@attr": {
"rank": "5"
},
"name": "Crash Love"
},
{
"artist": {
"url": "https://www.last.fm/music/AFI",
"name": "AFI",
"mbid": ""
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/d0a4b04bf955443b83095ac7752afb4b.png"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/d0a4b04bf955443b83095ac7752afb4b.png"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/d0a4b04bf955443b83095ac7752afb4b.png"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/d0a4b04bf955443b83095ac7752afb4b.png"
}
],
"mbid": "1f315994-b360-46c8-9241-303c49ff4741",
"url": "https://www.last.fm/music/AFI/Decemberunderground",
"playcount": "13",
"@attr": {
"rank": "6"
},
"name": "Decemberunderground"
},
{
"artist": {
"url": "https://www.last.fm/music/Defeated+Sanity",
"name": "Defeated Sanity",
"mbid": "47a9a488-a721-4d0e-88bc-2452701fb9c9"
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/71fb8b597d7ae9c0987eb3a88a904c8e.jpg"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/71fb8b597d7ae9c0987eb3a88a904c8e.jpg"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/71fb8b597d7ae9c0987eb3a88a904c8e.jpg"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/71fb8b597d7ae9c0987eb3a88a904c8e.jpg"
}
],
"mbid": "2ce66939-5907-4953-b81d-67b0a736438c",
"url": "https://www.last.fm/music/Defeated+Sanity/Chapters+Of+Repugnance",
"playcount": "12",
"@attr": {
"rank": "7"
},
"name": "Chapters Of Repugnance"
},
{
"artist": {
"url": "https://www.last.fm/music/Unaussprechlichen+Kulten",
"name": "Unaussprechlichen Kulten",
"mbid": "dd4d9439-87ee-455d-a8b3-a96364845233"
},
"image": [
{
"size": "small",
"#text": "https://lastfm.freetls.fastly.net/i/u/34s/725c65aaeb1dbae8033dfec452c6ec55.jpg"
},
{
"size": "medium",
"#text": "https://lastfm.freetls.fastly.net/i/u/64s/725c65aaeb1dbae8033dfec452c6ec55.jpg"
},
{
"size": "large",
"#text": "https://lastfm.freetls.fastly.net/i/u/174s/725c65aaeb1dbae8033dfec452c6ec55.jpg"
},
{
"size": "extralarge",
"#text": "https://lastfm.freetls.fastly.net/i/u/300x300/725c65aaeb1dbae8033dfec452c6ec55.jpg"
}
],
"mbid": "2e9bac0f-33bc-4721-b907-978c0e2c4bb1",
"url": "https://www.last.fm/music/Unaussprechlichen+Kulten/H%C3%A4xan+Sabaoth",
"playcount": "12",
"@attr": {
"rank": "8"
},
"name": "Häxan Sabaoth"
}
]

View file

@ -0,0 +1,44 @@
export default {
"results": [
{
"page": "/posts/2024/i-dont-want-anything-your-ai-generates/",
"visitors": 11057
},
{
"page": "/posts/2024/i-found-the-music-i-love-on-the-internet/",
"visitors": 4535
},
{
"page": "/posts/2023/i-removed-tailwind-from-my-site/",
"visitors": 843
},
{
"page": "/posts/2023/default-apps-2023/",
"visitors": 594
},
{
"page": "/posts/2023/check-in-to-your-personal-site/",
"visitors": 523
},
{
"page": "/posts/2024/your-site-your-home-your-web/",
"visitors": 356
},
{
"page": "/posts/2023/i-dont-want-streaming-music/",
"visitors": 350
},
{
"page": "/posts/2024/you-dont-need-a-framework-for-that/",
"visitors": 271
},
{
"page": "/posts/2023/i-block-ads/",
"visitors": 259
},
{
"page": "/posts/2023/leaning-into-google-services/",
"visitors": 249
}
]
}

View file

@ -0,0 +1,74 @@
export default [
{
"mbid": "",
"url": "https://www.last.fm/music/AFI",
"playcount": "37",
"@attr": {
"rank": "1"
},
"name": "AFI"
},
{
"mbid": "3cb2b7a7-53e6-4b0c-9143-16227fc5d1c3",
"url": "https://www.last.fm/music/Hauntologist",
"playcount": "32",
"@attr": {
"rank": "2"
},
"name": "Hauntologist"
},
{
"mbid": "47a9a488-a721-4d0e-88bc-2452701fb9c9",
"url": "https://www.last.fm/music/Defeated+Sanity",
"playcount": "30",
"@attr": {
"rank": "3"
},
"name": "Defeated Sanity"
},
{
"mbid": "37e9d7b2-7779-41b2-b2eb-3685351caad3",
"url": "https://www.last.fm/music/NoMeansNo",
"playcount": "27",
"@attr": {
"rank": "4"
},
"name": "NoMeansNo"
},
{
"mbid": "aba64937-3334-4c65-90a1-4e6b9d4d7ada",
"url": "https://www.last.fm/music/Aesop+Rock",
"playcount": "26",
"@attr": {
"rank": "5"
},
"name": "Aesop Rock"
},
{
"mbid": "7961aa2c-36d4-4601-a691-981b18143991",
"url": "https://www.last.fm/music/Galvanizer",
"playcount": "24",
"@attr": {
"rank": "6"
},
"name": "Galvanizer"
},
{
"mbid": "48cb1769-b1fe-4730-a088-d83be4dc01ef",
"url": "https://www.last.fm/music/Sepulcher",
"playcount": "24",
"@attr": {
"rank": "7"
},
"name": "Sepulcher"
},
{
"mbid": "6c3b93e1-d3e7-413c-b030-b6ddd918676f",
"url": "https://www.last.fm/music/dissimulator",
"playcount": "23",
"@attr": {
"rank": "8"
},
"name": "dissimulator"
}
]

View file

@ -0,0 +1,20 @@
export default [
{
"title": "How to Make a Killing: Blood, Death and Dollars in American Medicine",
"author": "Tom Mueller",
"image": "https://cd-books.b-cdn.net/vfbaoylm8sxmuifsp6y4ifx3p3o3",
"url": "https://app.thestorygraph.com/books/d02f9eec-ca74-419e-a28b-4f59d0e89e01",
"percentage": "86%",
"dateAdded": "2/5/2024, 12:00:00 AM",
"type": "book"
},
{
"author": "Peter Watts",
"title": "Blindsight",
"image": "https://cd-books.b-cdn.net/2e47qqm3giuw0uiexnbsm9egpqst",
"url": "https://app.thestorygraph.com/books/57fd0912-3e02-4a55-b7a8-86f584ecf373",
"percentage": "41%",
"dateAdded": "1/3/2024, 12:00:00 AM",
"type": "book"
}
]

View file

@ -0,0 +1,49 @@
export default [
{
"title": "This website is personal",
"author": "Frills",
"tags": {
"tech": {
"name": "tech",
"type": "manual",
"created": 1707349350280
},
"share": {
"name": "share",
"type": "manual",
"created": 1707349353085
},
"blogging": {
"name": "blogging",
"type": "manual",
"created": 1707349352274
}
},
"summary": "It is easy to get overwhelmed by the fact that anything you put online can live forever. Even when you evolve and change as a person: your thoughts, your values, your opinions, your past-self still exists in this ephemeral space. That is beautiful and terrifying.\nI've been thinking a lot recently about what I share where on the internet. I sometimes toot consistently and conversationally on Mastodon for a week or so at a time; I post a couple of times a year on Instagram, moreso on stories but even that has petered out a lot; and here on my website I have started holding blog posts up on this pedastal as something that need to be substantial or at least a little bit interesting. Somehow, even here, the lure of professionalism and validation has kicked in.\n\ni want to read your thoughts, your feelings, your perceptions of the world as it's happening around you. romanticize your life; tell me the minute details of your commute—your perception is yours, after all, and i've never experienced it before.\nXandra: Everyo...",
"source_url": "https://frills.dev/blog/070224-this-website-is-personal-girls/",
},
{
"url": "https://read.readwise.io/read/01hp2d8dggjjmgjg8pjxvdwnpz",
"title": "The power to not",
"author": "neatnik.net",
"tags": {
"tech": {
"name": "tech",
"type": "manual",
"created": 1707330519410
},
"share": {
"name": "share",
"type": "manual",
"created": 1707330522792
},
"social media": {
"name": "social media",
"type": "manual",
"created": 1707330521543
}
},
"summary": "The urge to engage can be overwhelming.",
"source_url": "https://notes.neatnik.net/2024/01/the-power-to-not",
}
]

View file

@ -0,0 +1,215 @@
export default [
{
"id": 9521296050,
"watched_at": "2024-02-06T14:41:22.000Z",
"action": "checkin",
"type": "movie",
"movie": {
"title": "The Accountant",
"year": 2016,
"ids": {
"trakt": 203309,
"slug": "the-accountant-2016",
"imdb": "tt2140479",
"tmdb": 302946
},
"tagline": "Calculate your choices.",
"overview": "As a math savant uncooks the books for a new client, the Treasury Department closes in on his activities and the body count starts to rise.",
"released": "2016-10-14",
"runtime": 128,
"country": "us",
"trailer": "https://youtube.com/watch?v=0KHOVlEpMyY",
"homepage": "http://www.accountantmovie.com/",
"status": "released",
"rating": 7.72918,
"votes": 30256,
"comment_count": 62,
"updated_at": "2024-02-07T12:06:14.000Z",
"language": "en",
"genres": [
"drama",
"crime",
"thriller"
],
"certification": "R"
}
},
{
"id": 9518530996,
"watched_at": "2024-02-05T00:58:37.000Z",
"action": "watch",
"type": "movie",
"movie": {
"title": "Apocalypse Now",
"year": 1979,
"ids": {
"trakt": 23,
"slug": "apocalypse-now-1979",
"imdb": "tt0078788",
"tmdb": 28
},
"tagline": "This is the end...",
"overview": "At the height of the Vietnam war, Captain Benjamin Willard is sent on a dangerous mission that, officially, \"does not exist, nor will it ever exist.\" His goal is to locate - and eliminate - a mysterious Green Beret Colonel named Walter Kurtz, who has been leading his personal army on illegal guerrilla missions into enemy territory.",
"released": "1979-08-15",
"runtime": 147,
"country": "us",
"trailer": "https://youtube.com/watch?v=9l-ViOOFH-s",
"homepage": "http://www.apocalypsenow.com",
"status": "released",
"rating": 8.22404,
"votes": 10257,
"comment_count": 25,
"updated_at": "2024-02-08T08:05:34.000Z",
"language": "en",
"genres": [
"war",
"drama"
],
"certification": "R"
}
},
{
"id": 9517495292,
"watched_at": "2024-02-04T18:10:40.000Z",
"action": "watch",
"type": "movie",
"movie": {
"title": "10 Cloverfield Lane",
"year": 2016,
"ids": {
"trakt": 214312,
"slug": "10-cloverfield-lane-2016",
"imdb": "tt1179933",
"tmdb": 333371
},
"tagline": "Monsters come in many forms.",
"overview": "After getting in a car accident, a woman is held in a shelter with two men, who claim the outside world is affected by a widespread chemical attack.",
"released": "2016-03-11",
"runtime": 104,
"country": "us",
"trailer": "https://youtube.com/watch?v=yQy-ANhnUpE",
"homepage": "http://www.10cloverfieldlane.com/",
"status": "released",
"rating": 7.23989,
"votes": 24003,
"comment_count": 101,
"updated_at": "2024-02-08T12:37:37.000Z",
"language": "en",
"genres": [
"science-fiction",
"thriller",
"drama",
"horror"
],
"certification": "PG-13"
}
},
{
"id": 9497461700,
"watched_at": "2024-01-26T20:25:58.000Z",
"action": "watch",
"type": "movie",
"movie": {
"title": "Aquaman",
"year": 2018,
"ids": {
"trakt": 193968,
"slug": "aquaman-2018",
"imdb": "tt1477834",
"tmdb": 297802
},
"tagline": "Home Is Calling",
"overview": "Once home to the most advanced civilization on Earth, Atlantis is now an underwater kingdom ruled by the power-hungry King Orm. With a vast army at his disposal, Orm plans to conquer the remaining oceanic people and then the surface world. Standing in his way is Arthur Curry, Orm's half-human, half-Atlantean brother and true heir to the throne.",
"released": "2018-12-21",
"runtime": 143,
"country": "us",
"trailer": "https://youtube.com/watch?v=2wcj6SrX4zw",
"homepage": "https://www.warnerbros.com/movies/aquaman",
"status": "released",
"rating": 7.22762,
"votes": 30766,
"comment_count": 182,
"updated_at": "2024-02-07T08:07:16.000Z",
"language": "en",
"genres": [
"fantasy",
"superhero",
"adventure",
"action"
],
"certification": "PG-13"
}
},
{
"id": 9491909538,
"watched_at": "2024-01-23T23:57:28.000Z",
"action": "watch",
"type": "movie",
"movie": {
"title": "Aquaman and the Lost Kingdom",
"year": 2023,
"ids": {
"trakt": 423775,
"slug": "aquaman-and-the-lost-kingdom-2023",
"imdb": "tt9663764",
"tmdb": 572802
},
"tagline": "The tide is turning.",
"overview": "Black Manta seeks revenge on Aquaman for his father's death. Wielding the Black Trident's power, he becomes a formidable foe. To defend Atlantis, Aquaman forges an alliance with his imprisoned brother. They must protect the kingdom.",
"released": "2023-12-22",
"runtime": 124,
"country": "us",
"trailer": "https://youtube.com/watch?v=UGc5Tzz19UY",
"homepage": "https://www.aquamanmovie.com",
"status": "released",
"rating": 6.4311,
"votes": 4231,
"comment_count": 67,
"updated_at": "2024-02-08T08:06:27.000Z",
"language": "en",
"genres": [
"adventure",
"action",
"fantasy",
"superhero"
],
"certification": "PG-13"
}
},
{
"id": 9480347272,
"watched_at": "2024-01-18T23:17:09.000Z",
"action": "watch",
"type": "movie",
"movie": {
"title": "The Marvels",
"year": 2023,
"ids": {
"trakt": 454832,
"slug": "the-marvels-2023",
"imdb": "tt10676048",
"tmdb": 609681
},
"tagline": "Higher. Further. Faster. Together.",
"overview": "Carol Danvers, aka Captain Marvel, has reclaimed her identity from the tyrannical Kree and taken revenge on the Supreme Intelligence. But unintended consequences see Carol shouldering the burden of a destabilized universe. When her duties send her to an anomalous wormhole linked to a Kree revolutionary, her powers become entangled with that of Jersey City super-fan Kamala Khan, aka Ms. Marvel, and Carols estranged niece, now S.A.B.E.R. astronaut Captain Monica Rambeau. Together, this unlikely trio must team up and learn to work in concert to save the universe.",
"released": "2023-11-10",
"runtime": 105,
"country": "us",
"trailer": "https://youtube.com/watch?v=uwmDH12MAA4",
"homepage": "https://www.marvel.com/movies/the-marvels",
"status": "released",
"rating": 6.27356,
"votes": 5264,
"comment_count": 132,
"updated_at": "2024-02-08T08:06:36.000Z",
"language": "en",
"genres": [
"superhero",
"action",
"adventure",
"science-fiction"
],
"certification": "PG-13"
}
}
]

View file

@ -0,0 +1,4 @@
export default {
"emoji": "🐯",
"content": "You can choose to be kind.",
}

302
src/_data/json/mocks/tv.js Normal file
View file

@ -0,0 +1,302 @@
export default [
{
"id": 9525629074,
"watched_at": "2024-02-08T15:02:00.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 2,
"number": 2,
"title": "Be My Number One",
"ids": {
"trakt": 11317749,
"tvdb": 10193559,
"imdb": "tt25399544",
"tmdb": 5062519,
"tvrage": null
}
},
"show": {
"title": "Tokyo Vice",
"year": 2022,
"ids": {
"trakt": 156576,
"slug": "tokyo-vice",
"tvdb": 366624,
"imdb": "tt2887954",
"tmdb": 90296,
"tvrage": null
}
}
},
{
"id": 9525349458,
"watched_at": "2024-02-08T15:01:14.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 2,
"number": 1,
"title": "Don't Ever F**king Miss",
"ids": {
"trakt": 11317721,
"tvdb": 10173398,
"imdb": "tt20875414",
"tmdb": 5062421,
"tvrage": null
}
},
"show": {
"title": "Tokyo Vice",
"year": 2022,
"ids": {
"trakt": 156576,
"slug": "tokyo-vice",
"tvdb": 366624,
"imdb": "tt2887954",
"tmdb": 90296,
"tvrage": null
}
}
},
{
"id": 9525348552,
"watched_at": "2024-02-08T15:00:39.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 2,
"number": 2,
"title": "Sword",
"ids": {
"trakt": 11165210,
"tvdb": 10247899,
"imdb": "tt22060126",
"tmdb": 4968609,
"tvrage": null
}
},
"show": {
"title": "Halo",
"year": 2022,
"ids": {
"trakt": 150109,
"slug": "halo",
"tvdb": 366524,
"imdb": "tt2934286",
"tmdb": 52814,
"tvrage": null
}
}
},
{
"id": 9525059250,
"watched_at": "2024-02-08T14:59:52.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 2,
"number": 1,
"title": "Sanctuary",
"ids": {
"trakt": 11165209,
"tvdb": 10170113,
"imdb": "tt22060096",
"tmdb": 4968592,
"tvrage": null
}
},
"show": {
"title": "Halo",
"year": 2022,
"ids": {
"trakt": 150109,
"slug": "halo",
"tvdb": 366524,
"imdb": "tt2934286",
"tmdb": 52814,
"tvrage": null
}
}
},
{
"id": 9524186036,
"watched_at": "2024-02-08T02:52:24.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 1,
"number": 2,
"title": "Second Date",
"ids": {
"trakt": 11055013,
"tvdb": 10155338,
"imdb": "tt21650512",
"tmdb": 4926319,
"tvrage": null
}
},
"show": {
"title": "Mr. & Mrs. Smith",
"year": 2024,
"ids": {
"trakt": 173650,
"slug": "mr-mrs-smith-2024",
"tvdb": 396773,
"imdb": "tt14044212",
"tmdb": 118642,
"tvrage": null
}
}
},
{
"id": 9523446431,
"watched_at": "2024-02-07T15:01:33.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 1,
"number": 1,
"title": "First Date",
"ids": {
"trakt": 4810701,
"tvdb": 8216044,
"imdb": "tt14199656",
"tmdb": 2726037,
"tvrage": null
}
},
"show": {
"title": "Mr. & Mrs. Smith",
"year": 2024,
"ids": {
"trakt": 173650,
"slug": "mr-mrs-smith-2024",
"tvdb": 396773,
"imdb": "tt14044212",
"tmdb": 118642,
"tvrage": null
}
}
},
{
"id": 9523322722,
"watched_at": "2024-02-07T14:59:21.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 1,
"number": 6,
"title": "Beehive",
"ids": {
"trakt": 10934939,
"tvdb": 10068291,
"imdb": "tt21107292",
"tmdb": 4785721,
"tvrage": null
}
},
"show": {
"title": "Criminal Record",
"year": 2024,
"ids": {
"trakt": 196417,
"slug": "criminal-record",
"tvdb": 421495,
"imdb": "tt21088136",
"tmdb": 204490,
"tvrage": null
}
}
},
{
"id": 9520276837,
"watched_at": "2024-02-05T15:00:41.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 49,
"number": 11,
"title": "February 3 - Ayo Edebiri",
"ids": {
"trakt": 11375525,
"tvdb": null,
"imdb": null,
"tmdb": 5099959,
"tvrage": null
}
},
"show": {
"title": "Saturday Night Live",
"year": 1975,
"ids": {
"trakt": 1656,
"slug": "saturday-night-live",
"tvdb": 76177,
"imdb": "tt0072562",
"tmdb": 1667,
"tvrage": null
}
}
},
{
"id": 9519545755,
"watched_at": "2024-02-05T14:53:00.000Z",
"action": "watch",
"type": "episode",
"episode": {
"season": 1,
"number": 8,
"title": "We Shall Live Forever",
"ids": {
"trakt": 883344,
"tvdb": 4460471,
"imdb": "tt2356588",
"tmdb": 876837,
"tvrage": null
}
},
"show": {
"title": "Banshee",
"year": 2013,
"ids": {
"trakt": 41522,
"slug": "banshee",
"tvdb": 259765,
"imdb": "tt2017109",
"tmdb": 41727,
"tvrage": null
}
}
},
{
"id": 9519471765,
"watched_at": "2024-02-05T13:56:28.000Z",
"action": "checkin",
"type": "episode",
"episode": {
"season": 1,
"number": 7,
"title": "Behold a Pale Rider",
"ids": {
"trakt": 883343,
"tvdb": 4460470,
"imdb": "tt2371798",
"tmdb": 876836,
"tvrage": null
}
},
"show": {
"title": "Banshee",
"year": 2013,
"ids": {
"trakt": 41522,
"slug": "banshee",
"tvdb": 259765,
"imdb": "tt2017109",
"tmdb": 41727,
"tvrage": null
}
}
},
]

View file

@ -1,19 +1,10 @@
import EleventyFetch from '@11ty/eleventy-fetch' import EleventyFetch from '@11ty/eleventy-fetch'
import LinksMock from './json/mocks/links.js'
export default async function () { export default async function () {
const API_TOKEN_READWISE = process.env.API_TOKEN_READWISE const API_TOKEN_READWISE = process.env.API_TOKEN_READWISE
const url = 'https://readwise.io/api/v3/list?location=archive' const url = 'https://readwise.io/api/v3/list?location=archive'
const res = EleventyFetch(url, { const formatLinkData = (links) => links.map((link) => {
duration: '1h',
type: 'json',
fetchOptions: {
headers: {
Authorization: `Token ${API_TOKEN_READWISE}`,
},
},
}).catch()
const data = await res
const links = data['results'].map((link) => {
return { return {
title: link['title'], title: link['title'],
url: link['source_url'], url: link['source_url'],
@ -25,5 +16,21 @@ export default async function () {
description: `${link['summary']}<br/><br/>`, description: `${link['summary']}<br/><br/>`,
} }
}) })
return links.filter((link) => link.tags.includes('share'))
if (process.env.ELEVENTY_PRODUCTION) {
const res = EleventyFetch(url, {
duration: '1h',
type: 'json',
fetchOptions: {
headers: {
Authorization: `Token ${API_TOKEN_READWISE}`,
},
},
}).catch()
const data = await res
return formatLinkData(data['results']).filter((link) => link.tags.includes('share'))
} else {
return formatLinkData(LinksMock)
}
} }

View file

@ -1,23 +1,12 @@
import EleventyFetch from '@11ty/eleventy-fetch' import EleventyFetch from '@11ty/eleventy-fetch'
import MoviesMock from './json/mocks/movies.js'
export default async function () { export default async function () {
const TV_KEY = process.env.API_KEY_TRAKT const TV_KEY = process.env.API_KEY_TRAKT
const MOVIEDB_KEY = process.env.API_KEY_MOVIEDB const MOVIEDB_KEY = process.env.API_KEY_MOVIEDB
const url = 'https://api.trakt.tv/users/cdransf/history/movies?page=1&limit=6&extended=full' const url = 'https://api.trakt.tv/users/cdransf/history/movies?page=1&limit=6&extended=full'
const res = EleventyFetch(url, { const formatMovieData = (movies) => movies.map((item) => {
duration: '1h', const movie = {
type: 'json',
fetchOptions: {
headers: {
'Content-Type': 'application/json',
'trakt-api-version': 2,
'trakt-api-key': TV_KEY,
},
},
}).catch()
const data = await res
const movies = data.map((item) => {
return {
title: item['movie']['title'], title: item['movie']['title'],
dateAdded: item['watched_at'], dateAdded: item['watched_at'],
url: `https://trakt.tv/movies/${item['movie']['ids']['slug']}`, url: `https://trakt.tv/movies/${item['movie']['ids']['slug']}`,
@ -26,19 +15,38 @@ export default async function () {
description: `${item['movie']['overview']}<br/><br/>`, description: `${item['movie']['overview']}<br/><br/>`,
type: 'movie', type: 'movie',
} }
if (process.env.ELEVENTY_PRODUCTION === 'false') movie.image = 'https://cd-movies.b-cdn.net'
return movie;
}) })
for (const movie of movies) { if (process.env.ELEVENTY_PRODUCTION) {
const tmdbId = movie['tmdbId'] const res = EleventyFetch(url, {
const tmdbUrl = `https://api.themoviedb.org/3/movie/${tmdbId}?api_key=${MOVIEDB_KEY}`
const tmdbRes = EleventyFetch(tmdbUrl, {
duration: '1h', duration: '1h',
type: 'json', type: 'json',
}) fetchOptions: {
const tmdbData = await tmdbRes headers: {
const posterPath = tmdbData['poster_path'] 'Content-Type': 'application/json',
movie.image = `https://cd-movies.b-cdn.net/t/p/w500${posterPath}` 'trakt-api-version': 2,
} 'trakt-api-key': TV_KEY,
},
},
}).catch()
const data = await res
const movies = formatMovieData(data)
return movies for (const movie of movies) {
const tmdbId = movie['tmdbId']
const tmdbUrl = `https://api.themoviedb.org/3/movie/${tmdbId}?api_key=${MOVIEDB_KEY}`
const tmdbRes = EleventyFetch(tmdbUrl, {
duration: '1h',
type: 'json',
})
const tmdbData = await tmdbRes
const posterPath = tmdbData['poster_path']
movie.image = `https://cd-movies.b-cdn.net/t/p/w500${posterPath}`
}
return movies;
} else {
return formatMovieData(MoviesMock)
}
} }

View file

@ -1,11 +1,16 @@
import EleventyFetch from '@11ty/eleventy-fetch' import EleventyFetch from '@11ty/eleventy-fetch'
import StatusMock from './json/mocks/status.js'
export default async function () { export default async function () {
const url = 'https://api.omg.lol/address/cory/statuses/' const url = 'https://api.omg.lol/address/cory/statuses/'
const res = EleventyFetch(url, { if (process.env.ELEVENTY_PRODUCTION) {
duration: '1h', const res = EleventyFetch(url, {
type: 'json', duration: '1h',
}).catch() type: 'json',
const status = await res }).catch()
return status.response['statuses'][0] const status = await res
return status.response['statuses'][0]
} else {
return StatusMock
}
} }

View file

@ -9,60 +9,64 @@ export default async function () {
const MUSIC_KEY = process.env.API_KEY_LASTFM const MUSIC_KEY = process.env.API_KEY_LASTFM
const LISTENBRAINZ_TOKEN = process.env.LISTENBRAINZ_TOKEN const LISTENBRAINZ_TOKEN = process.env.LISTENBRAINZ_TOKEN
const url = `https://ws.audioscrobbler.com/2.0/?method=user.getrecenttracks&user=coryd_&api_key=${MUSIC_KEY}&format=json&limit=200` const url = `https://ws.audioscrobbler.com/2.0/?method=user.getrecenttracks&user=coryd_&api_key=${MUSIC_KEY}&format=json&limit=200`
const res = EleventyFetch(url, { if (process.env.ELEVENTY_PRODUCTION) {
duration: '1h', const res = EleventyFetch(url, {
type: 'json', duration: '1h',
}).catch() type: 'json',
const data = await res }).catch()
let submissions = [] const data = await res
data['recenttracks']['track'].forEach((track) => { let submissions = []
let artistMbid = track['artist']['mbid']['mbid'] data['recenttracks']['track'].forEach((track) => {
let artistMbid = track['artist']['mbid']['mbid']
// mbid mismatches // mbid mismatches
if (mbidMap(track['artist']['#text']) !== '') artistMbid = mbidMap(track['artist']['#text']) if (mbidMap(track['artist']['#text']) !== '') artistMbid = mbidMap(track['artist']['#text'])
if (track['date']) if (track['date'])
submissions.push({ submissions.push({
track_metadata: { track_metadata: {
track_name: track['name'], track_name: track['name'],
artist_name: track['artist']['#text'], artist_name: track['artist']['#text'],
release_name: track['album']['#text'], release_name: track['album']['#text'],
additional_info: { additional_info: {
submission_client: 'coryd.dev last.fm importer', submission_client: 'coryd.dev last.fm importer',
lastfm_track_mbid: track['mbid'], lastfm_track_mbid: track['mbid'],
lastfm_release_mbid: track['album']['mbid'], lastfm_release_mbid: track['album']['mbid'],
lastfm_artist_mbid: artistMbid, lastfm_artist_mbid: artistMbid,
},
}, },
}, listened_at: track['date']['uts'],
listened_at: track['date']['uts'], })
}) })
})
await fetch('https://api.listenbrainz.org/1/submit-listens', { await fetch('https://api.listenbrainz.org/1/submit-listens', {
method: 'POST', method: 'POST',
headers: { headers: {
Accept: 'application/json', Accept: 'application/json',
Authorization: `Token ${LISTENBRAINZ_TOKEN}`, Authorization: `Token ${LISTENBRAINZ_TOKEN}`,
}, },
body: JSON.stringify({ body: JSON.stringify({
listen_type: 'import', listen_type: 'import',
payload: submissions, payload: submissions,
}), }),
}) })
await fetch('https://api.listenbrainz.org/1/latest-import', { await fetch('https://api.listenbrainz.org/1/latest-import', {
method: 'POST', method: 'POST',
headers: { headers: {
Accept: 'application/json', Accept: 'application/json',
Authorization: `Token ${LISTENBRAINZ_TOKEN}`, Authorization: `Token ${LISTENBRAINZ_TOKEN}`,
}, },
body: JSON.stringify({ body: JSON.stringify({
service: 'lastfm', service: 'lastfm',
ts: submissions[0]['listened_at'], ts: submissions[0]['listened_at'],
}), }),
}) })
return { return {
listenbrainz_submissions: submissions, listenbrainz_submissions: submissions,
}
} else {
return {}
} }
} }

View file

@ -1,84 +1,93 @@
import EleventyFetch from '@11ty/eleventy-fetch' import EleventyFetch from '@11ty/eleventy-fetch'
import TvMock from './json/mocks/tv.js'
export default async function () { export default async function () {
const TV_KEY = process.env.API_KEY_TRAKT const TV_KEY = process.env.API_KEY_TRAKT
const MOVIEDB_KEY = process.env.API_KEY_MOVIEDB const MOVIEDB_KEY = process.env.API_KEY_MOVIEDB
const url = 'https://api.trakt.tv/users/cdransf/history/shows?page=1&limit=75' const url = 'https://api.trakt.tv/users/cdransf/history/shows?page=1&limit=75'
const res = EleventyFetch(url, { let episodes
duration: '1h', const formatEpisodeData = (shows) => {
type: 'json', const episodeData = []
fetchOptions: { const startingEpisodes = []
headers: { const startingSeasons = []
'Content-Type': 'application/json', shows.reverse().forEach((episode) => {
'trakt-api-version': 2, const episodeNumber = episode['episode']['number']
'trakt-api-key': TV_KEY, const seasonNumber = episode['episode']['season']
}, if (!startingEpisodes.find((e) => e.show === episode['show']['title'])) startingEpisodes.push({ show: episode['show']['title'], episode: episodeNumber })
}, if (!startingSeasons.find((e) => e.show === episode['show']['title'])) startingSeasons.push({ show: episode['show']['title'], season: seasonNumber })
}).catch()
const data = await res
const episodeData = []
const startingEpisodes = []
const startingSeasons = []
data.reverse().forEach((episode) => {
const episodeNumber = episode['episode']['number']
const seasonNumber = episode['episode']['season']
if (!startingEpisodes.find((e) => e.show === episode['show']['title'])) { startingEpisodes.push({ show: episode['show']['title'], episode: episodeNumber }) }
if (!startingSeasons.find((e) => e.show === episode['show']['title'])) { startingSeasons.push({ show: episode['show']['title'], season: seasonNumber }) }
if (episodeData.find((e) => e.name === episode?.['show']?.['title'])) { if (episodeData.find((e) => e.name === episode?.['show']?.['title'])) {
// cache the matched episode reference // cache the matched episode reference
const matchedEpisode = episodeData.find((e) => e.name === episode?.['show']?.['title']) const matchedEpisode = episodeData.find((e) => e.name === episode?.['show']?.['title'])
const startingEpisode = startingEpisodes.find((e) => e.show === episode['show']['title'])['episode'] const startingEpisode = startingEpisodes.find((e) => e.show === episode['show']['title'])['episode']
const startingSeason = startingSeasons.find((e) => e.show === episode['show']['title'])['season'] const startingSeason = startingSeasons.find((e) => e.show === episode['show']['title'])['season']
// remove the matched episode from the array // remove the matched episode from the array
episodeData.splice( episodeData.splice(
episodeData.findIndex((e) => e.name === episode['show']['title']), episodeData.findIndex((e) => e.name === episode['show']['title']),
1 1
) )
// push the new episode to the array // push the new episode to the array
episodeData.push({ episodeData.push({
name: matchedEpisode['name'], name: matchedEpisode['name'],
url: `https://trakt.tv/shows/${episode['show']['ids']['slug']}`, url: `https://trakt.tv/shows/${episode['show']['ids']['slug']}`,
subtext: `S${startingSeason}E${startingEpisode} - S${episode['episode']['season']}E${episode['episode']['number']}`, subtext: `S${startingSeason}E${startingEpisode} - S${episode['episode']['season']}E${episode['episode']['number']}`,
startingEpisode, startingEpisode,
startingSeason, startingSeason,
episode: episodeNumber, episode: episodeNumber,
season: seasonNumber, season: seasonNumber,
id: episode['show']['ids']['trakt'], id: episode['show']['ids']['trakt'],
tmdbId: episode['show']['ids']['tmdb'], tmdbId: episode['show']['ids']['tmdb'],
type: 'tv-range', type: 'tv-range',
}) })
} else { } else {
// if an episode with the same show name doesn't exist, push it to the array // if an episode with the same show name doesn't exist, push it to the array
episodeData.push({ episodeData.push({
name: episode['show']['title'], name: episode['show']['title'],
title: episode['episode']['title'], title: episode['episode']['title'],
url: `https://trakt.tv/shows/${episode['show']['ids']['slug']}/seasons/${episode['episode']['season']}/episodes/${episode['episode']['number']}`, url: `https://trakt.tv/shows/${episode['show']['ids']['slug']}/seasons/${episode['episode']['season']}/episodes/${episode['episode']['number']}`,
subtext: `${episode['show']['title']} • S${episode['episode']['season']}E${episode['episode']['number']}`, subtext: `${episode['show']['title']} • S${episode['episode']['season']}E${episode['episode']['number']}`,
episode: episodeNumber, episode: episodeNumber,
season: seasonNumber, season: seasonNumber,
id: episode['show']['ids']['trakt'], id: episode['show']['ids']['trakt'],
tmdbId: episode['show']['ids']['tmdb'], tmdbId: episode['show']['ids']['tmdb'],
type: 'tv', type: 'tv',
}) })
} }
}) if (process.env.ELEVENTY_PRODUCTION === 'false') episodeData[episodeData.length - 1].image = 'https://cd-movies.b-cdn.net'
const episodes = episodeData.reverse()
for (const episode of episodes) {
const tmdbId = episode['tmdbId']
const tmdbUrl = `https://api.themoviedb.org/3/tv/${tmdbId}?api_key=${MOVIEDB_KEY}`
const tmdbRes = EleventyFetch(tmdbUrl, {
duration: '1h',
type: 'json',
}) })
const tmdbData = await tmdbRes
const posterPath = tmdbData['poster_path'] return episodeData.reverse()
episode.image = `https://cd-movies.b-cdn.net/t/p/w500${posterPath}`
} }
return episodes if (process.env.ELEVENTY_PRODUCTION) {
const res = EleventyFetch(url, {
duration: '1h',
type: 'json',
fetchOptions: {
headers: {
'Content-Type': 'application/json',
'trakt-api-version': 2,
'trakt-api-key': TV_KEY,
},
},
}).catch()
const shows = await res
episodes = formatEpisodeData(shows)
for (const episode of episodes) {
const tmdbId = episode['tmdbId']
const tmdbUrl = `https://api.themoviedb.org/3/tv/${tmdbId}?api_key=${MOVIEDB_KEY}`
const tmdbRes = EleventyFetch(tmdbUrl, {
duration: '1h',
type: 'json',
})
const tmdbData = await tmdbRes
const posterPath = tmdbData['poster_path']
episode.image = `https://cd-movies.b-cdn.net/t/p/w500${posterPath}`
}
} else {
episodes = formatEpisodeData(TvMock)
}
return episodes;
} }

View file

@ -3,12 +3,18 @@ import EleventyFetch from '@11ty/eleventy-fetch'
export default async function () { export default async function () {
const KEY_CORYD = process.env.API_KEY_WEBMENTIONS_CORYD_DEV const KEY_CORYD = process.env.API_KEY_WEBMENTIONS_CORYD_DEV
const url = `https://webmention.io/api/mentions.jf2?token=${KEY_CORYD}&per-page=1000` const url = `https://webmention.io/api/mentions.jf2?token=${KEY_CORYD}&per-page=1000`
const res = EleventyFetch(url, { if (process.env.ELEVENTY_PRODUCTION) {
duration: '1h', const res = EleventyFetch(url, {
type: 'json', duration: '1h',
}).catch() type: 'json',
const webmentions = await res }).catch()
return { const webmentions = await res
mentions: webmentions['children'], return {
mentions: webmentions['children'],
}
} else {
return {
mentions: []
}
} }
} }

View file

@ -1,39 +1,41 @@
;(async function () { if (window.location.hostname !== 'localhost') {
const nowPlaying = document.getElementById('now-playing') ;(async function () {
const nowPlaying = document.getElementById('now-playing')
if (nowPlaying) { if (nowPlaying) {
const content = document.getElementById('now-playing-content') const content = document.getElementById('now-playing-content')
const loading = document.getElementById('now-playing-loading') const loading = document.getElementById('now-playing-loading')
const populateNowPlaying = (data) => { const populateNowPlaying = (data) => {
loading.style.display = 'none'
content.innerHTML = data.content
content.classList.remove('hidden')
}
try {
const cache = JSON.parse(localStorage.getItem('now-playing'))
if (cache) populateNowPlaying(cache)
} catch (e) {
/* quiet catch */
}
const data = await fetch('/api/now-playing', {
type: 'json',
})
.then((data) => data.json())
.catch(() => {
loading.style.display = 'none' loading.style.display = 'none'
content.innerHTML = data.content
content.classList.remove('hidden')
}
try {
const cache = JSON.parse(localStorage.getItem('now-playing'))
if (cache) populateNowPlaying(cache)
} catch (e) {
/* quiet catch */
}
const data = await fetch('/api/now-playing', {
type: 'json',
}) })
.then((data) => data.json())
.catch(() => {
loading.style.display = 'none'
})
try { try {
localStorage.setItem('now-playing', JSON.stringify(data)) localStorage.setItem('now-playing', JSON.stringify(data))
} catch (e) { } catch (e) {
/* quiet catch */ /* quiet catch */
}
if (!JSON.parse(localStorage.getItem('now-playing')) && !data) nowPlaying.remove()
populateNowPlaying(data)
} }
})()
if (!JSON.parse(localStorage.getItem('now-playing')) && !data) nowPlaying.remove() }
populateNowPlaying(data)
}
})()