feat: a tiny bit of work to rip out letterboxd
This commit is contained in:
parent
2f0c3d7ac0
commit
681f26e9ea
8 changed files with 80 additions and 53 deletions
1
.env
1
.env
|
@ -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=
|
||||||
|
|
|
@ -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) => {
|
||||||
|
|
|
@ -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']}`
|
||||||
|
|
|
@ -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: {
|
||||||
const movies = res.items
|
headers: {
|
||||||
.map((item) => {
|
'Content-Type': 'application/json',
|
||||||
const images = item['content']?.match(/<img [^>]*src="[^"]*"[^>]*>/gm) || []
|
'trakt-api-version': 2,
|
||||||
|
'trakt-api-key': TV_KEY,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}).catch()
|
||||||
|
const data = await res
|
||||||
|
const movies = data.map((item) => {
|
||||||
return {
|
return {
|
||||||
title: item['title'],
|
title: item['movie']['title'],
|
||||||
date: item['pubDate'],
|
dateAdded: item['watched_at'],
|
||||||
description: item['contentSnippet'],
|
url: `https://trakt.tv/movies/${item['movie']['ids']['slug']}`,
|
||||||
image: images.length
|
id: item['movie']['ids']['trakt'],
|
||||||
? images
|
tmdbId: item['movie']['ids']['tmdb'],
|
||||||
.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',
|
type: 'movie',
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.filter((movie) => !movie.url.includes('/list/'))
|
|
||||||
await asset.save(movies, 'json')
|
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://movies.coryd.dev/t/p/w500${posterPath}`
|
||||||
|
}
|
||||||
|
|
||||||
return movies
|
return movies
|
||||||
}
|
}
|
||||||
|
|
|
@ -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' },
|
||||||
|
|
|
@ -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
13
src/feeds/movies.liquid
Normal 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
|
||||||
|
%}
|
|
@ -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)
|
||||||
|
|
Reference in a new issue