feat: a tiny bit of work to rip out letterboxd

This commit is contained in:
Cory Dransfeldt 2023-10-02 09:20:07 -07:00
parent 2f0c3d7ac0
commit 681f26e9ea
8 changed files with 80 additions and 53 deletions

1
.env
View file

@ -1,5 +1,6 @@
API_KEY_LASTFM= API_KEY_LASTFM=
API_KEY_TRAKT= API_KEY_TRAKT=
API_KEY_MOVIEDB=
API_KEY_WEBMENTIONS_CORYD_DEV= API_KEY_WEBMENTIONS_CORYD_DEV=
ACCESS_TOKEN_MATTER= ACCESS_TOKEN_MATTER=
SITE_ID_CLICKY= SITE_ID_CLICKY=

View file

@ -45,7 +45,7 @@ module.exports = {
tagLookup: (url, tagMap) => { tagLookup: (url, tagMap) => {
if (!url) return if (!url) return
if (url.includes('https://goodreads.com')) return '#Books #Reading' if (url.includes('https://goodreads.com')) return '#Books #Reading'
if (url.includes('https://letterboxd.com')) return '#Movies #Letterboxd' if (url.includes('https://trakt.tv')) return '#Movies #Trakt'
return tagMap[url] || '' return tagMap[url] || ''
}, },
webmentionsByUrl: (webmentions, url) => { webmentionsByUrl: (webmentions, url) => {

View file

@ -15,11 +15,7 @@ module.exports = {
normalized['alt'] = `${item['plays']} plays of ${item['title']}` normalized['alt'] = `${item['plays']} plays of ${item['title']}`
normalized['subtext'] = `${item['plays']} plays` normalized['subtext'] = `${item['plays']} plays`
} }
if (item.type === 'book') normalized['alt'] = item['title'] if (item.type === 'book' || item.type === 'movie') normalized['alt'] = item['title']
if (item.type === 'movie') {
normalized['title'] = item['title']
normalized['alt'] = `${item['title']} - ${item['description']}`
}
if (item.type === 'tv') { if (item.type === 'tv') {
normalized['title'] = item['title'] normalized['title'] = item['title']
normalized['alt'] = `${item['title']} from ${item['name']}` normalized['alt'] = `${item['title']} from ${item['name']}`

View file

@ -1,32 +1,43 @@
const Parser = require('rss-parser') const EleventyFetch = require('@11ty/eleventy-fetch')
const { AssetCache } = require('@11ty/eleventy-fetch')
module.exports = async function () { module.exports = async function () {
const parser = new Parser() const TV_KEY = process.env.API_KEY_TRAKT
const url = 'https://letterboxd.com/cdme/rss' const MOVIEDB_KEY = process.env.API_KEY_MOVIEDB
const asset = new AssetCache('movies_data') const url = 'https://api.trakt.tv/users/cdransf/history/movies?page=1&limit=30'
if (asset.isCacheValid('1h')) return await asset.getCachedValue() const res = EleventyFetch(url, {
const res = await parser.parseURL(url).catch((error) => { duration: '1h',
console.log(error.message) 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'],
dateAdded: item['watched_at'],
url: `https://trakt.tv/movies/${item['movie']['ids']['slug']}`,
id: item['movie']['ids']['trakt'],
tmdbId: item['movie']['ids']['tmdb'],
type: 'movie',
}
}) })
const movies = res.items
.map((item) => { for (const movie of movies) {
const images = item['content']?.match(/<img [^>]*src="[^"]*"[^>]*>/gm) || [] const tmdbId = movie['tmdbId']
return { const tmdbUrl = `https://api.themoviedb.org/3/movie/${tmdbId}?api_key=${MOVIEDB_KEY}`
title: item['title'], const tmdbRes = EleventyFetch(tmdbUrl, {
date: item['pubDate'], duration: '1h',
description: item['contentSnippet'], type: 'json',
image: images.length
? images
.map((image) => image.replace(/.*src="([^"]*)".*/, '$1'))[0]
.replace('https://a.ltrbxd.com', 'https://movies.coryd.dev')
: 'https://cdn.coryd.dev/movies/missing-movie.jpg',
url: item['link'],
id: item['guid'],
type: 'movie',
}
}) })
.filter((movie) => !movie.url.includes('/list/')) const tmdbData = await tmdbRes
await asset.save(movies, 'json') const posterPath = tmdbData['poster_path']
movie.image = `https://movies.coryd.dev/t/p/w500${posterPath}`
}
return movies return movies
} }

View file

@ -19,7 +19,6 @@ module.exports = async function () {
url: 'https://open.spotify.com/user/mdh0acvmvfsbunzt6ywnq2tg3', url: 'https://open.spotify.com/user/mdh0acvmvfsbunzt6ywnq2tg3',
icon: 'brand-spotify', icon: 'brand-spotify',
}, },
{ name: 'Letterboxd', url: 'https://letterboxd.com/cdme', icon: 'brand-letterboxd' },
{ name: 'Trakt', url: 'https://trakt.tv/users/cdransf', icon: 'device-tv' }, { name: 'Trakt', url: 'https://trakt.tv/users/cdransf', icon: 'device-tv' },
{ name: 'Goodreads', url: 'https://www.goodreads.com/cdransf', icon: 'books' }, { name: 'Goodreads', url: 'https://www.goodreads.com/cdransf', icon: 'books' },
{ name: 'Buy Me a Coffee', url: 'https://www.buymeacoffee.com/cory', icon: 'cup' }, { name: 'Buy Me a Coffee', url: 'https://www.buymeacoffee.com/cory', icon: 'cup' },

View file

@ -2,6 +2,7 @@ const EleventyFetch = require('@11ty/eleventy-fetch')
module.exports = async function () { module.exports = 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 url = 'https://api.trakt.tv/users/cdransf/history/shows?page=1&limit=30' const url = 'https://api.trakt.tv/users/cdransf/history/shows?page=1&limit=30'
const res = EleventyFetch(url, { const res = EleventyFetch(url, {
duration: '1h', duration: '1h',
@ -15,59 +16,66 @@ module.exports = async function () {
}, },
}).catch() }).catch()
const data = await res const data = await res
const episodes = [] const episodeData = []
data.reverse().forEach((episode) => { data.reverse().forEach((episode) => {
const episodeNumber = episode['episode']['number'] const episodeNumber = episode['episode']['number']
const seasonNumber = episode['episode']['season'] const seasonNumber = episode['episode']['season']
if (episodes.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 = episodes.find((e) => e.name === episode?.['show']?.['title']) const matchedEpisode = episodeData.find((e) => e.name === episode?.['show']?.['title'])
// remove the matched episode from the array // remove the matched episode from the array
episodes.splice( episodeData.splice(
episodes.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
episodes.push({ episodeData.push({
name: matchedEpisode['name'], name: matchedEpisode['name'],
title: matchedEpisode['title'], title: matchedEpisode['title'],
url: `https://trakt.tv/shows/${episode['show']['ids']['slug']}`, url: `https://trakt.tv/shows/${episode['show']['ids']['slug']}`,
subtext: `S${matchedEpisode['startingSeason'] || matchedEpisode['season']}E${ subtext: `S${matchedEpisode['startingSeason'] || matchedEpisode['season']}E${
matchedEpisode['startingEpisode'] || matchedEpisode['episode'] matchedEpisode['startingEpisode'] || matchedEpisode['episode']
} - S${episode['episode']['season']}E${episode['episode']['number']}`, } - S${episode['episode']['season']}E${episode['episode']['number']}`,
image:
`https://cdn.coryd.dev/tv/${matchedEpisode['name']
.replace(':', '')
.replace(/\s+/g, '-')
.toLowerCase()}.jpg` || 'https://cdn.coryd.dev/tv/missing-tv.jpg',
startingEpisode: matchedEpisode['episode'], startingEpisode: matchedEpisode['episode'],
startingSeason: matchedEpisode['season'], startingSeason: matchedEpisode['season'],
episode: episodeNumber, episode: episodeNumber,
season: seasonNumber, season: seasonNumber,
id: episode['show']['ids']['trakt'],
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
episodes.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']}`,
image:
`https://cdn.coryd.dev/tv/${episode['show']['title']
.replace(':', '')
.replace(/\s+/g, '-')
.toLowerCase()}.jpg` || 'https://cdn.coryd.dev/tv/missing-tv.jpg',
episode: episodeNumber, episode: episodeNumber,
season: seasonNumber, season: seasonNumber,
id: episode['show']['ids']['trakt'],
tmdbId: episode['show']['ids']['tmdb'],
type: 'tv', type: 'tv',
}) })
} }
}) })
// return a reverse sorted array of episodes to match the watch order const episodes = episodeData.reverse()
return episodes.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']
episode.image = `https://movies.coryd.dev/t/p/w500${posterPath}`
}
return episodes
} }

13
src/feeds/movies.liquid Normal file
View file

@ -0,0 +1,13 @@
---
layout: null
eleventyExcludeFromCollections: true
permalink: /feeds/movies
---
{% render "partials/feeds/rss.liquid"
permalink:"/feeds/movies"
title:"Movies • Cory Dransfeldt"
description:"Movies I've watched recently."
data:movies
updated:movies[0].dateAdded
site:site
%}

View file

@ -83,7 +83,6 @@ Software and services that I use for work and my own enjoyment.
- [Spotify](https://spotify.com) - [Spotify](https://spotify.com)
- [Last.fm](https://last.fm) - [Last.fm](https://last.fm)
- [Trakt](https://trakt.tv) - [Trakt](https://trakt.tv)
- [Letterboxd](https://letterboxd.com)
- [Goodreads](https://goodreads.com) - [Goodreads](https://goodreads.com)
- [Slack](http://slack.com) - [Slack](http://slack.com)
- [Discord](http://discord.com) - [Discord](http://discord.com)