feat: mock when running locally
This commit is contained in:
parent
12f4929dd3
commit
6757a60134
22 changed files with 1342 additions and 300 deletions
|
@ -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();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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 {}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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'))
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
}
|
||||||
|
|
258
src/_data/json/mocks/albums.js
Normal file
258
src/_data/json/mocks/albums.js
Normal 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"
|
||||||
|
}
|
||||||
|
]
|
44
src/_data/json/mocks/analytics.js
Normal file
44
src/_data/json/mocks/analytics.js
Normal 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
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
74
src/_data/json/mocks/artists.js
Normal file
74
src/_data/json/mocks/artists.js
Normal 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"
|
||||||
|
}
|
||||||
|
]
|
20
src/_data/json/mocks/books.js
Normal file
20
src/_data/json/mocks/books.js
Normal 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"
|
||||||
|
}
|
||||||
|
]
|
49
src/_data/json/mocks/links.js
Normal file
49
src/_data/json/mocks/links.js
Normal 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",
|
||||||
|
}
|
||||||
|
]
|
215
src/_data/json/mocks/movies.js
Normal file
215
src/_data/json/mocks/movies.js
Normal 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 Carol’s 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"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
4
src/_data/json/mocks/status.js
Normal file
4
src/_data/json/mocks/status.js
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
export default {
|
||||||
|
"emoji": "🐯",
|
||||||
|
"content": "You can choose to be kind.",
|
||||||
|
}
|
302
src/_data/json/mocks/tv.js
Normal file
302
src/_data/json/mocks/tv.js
Normal 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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
]
|
|
@ -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)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
151
src/_data/tv.js
151
src/_data/tv.js
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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: []
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
|
||||||
}
|
|
||||||
})()
|
|
||||||
|
|
Reference in a new issue