chore: clean up config + file structure

This commit is contained in:
Cory Dransfeldt 2023-12-10 20:37:18 -08:00
parent d1f8ecd88c
commit b6fb54ab98
No known key found for this signature in database
9 changed files with 300 additions and 312 deletions

View file

@ -3,100 +3,38 @@ const tablerIcons = require('eleventy-plugin-tabler-icons')
const pluginUnfurl = require('eleventy-plugin-unfurl') const pluginUnfurl = require('eleventy-plugin-unfurl')
const pluginFilesMinifier = require('@sherby/eleventy-plugin-files-minifier') const pluginFilesMinifier = require('@sherby/eleventy-plugin-files-minifier')
const schema = require('@quasibit/eleventy-plugin-schema') const schema = require('@quasibit/eleventy-plugin-schema')
const { eleventyImagePlugin } = require('@11ty/eleventy-img')
const pluginRss = require('@11ty/eleventy-plugin-rss') const pluginRss = require('@11ty/eleventy-plugin-rss')
const outdent = require('outdent')
const Image = require('@11ty/eleventy-img')
const embedYouTube = require('eleventy-plugin-youtube-embed') const embedYouTube = require('eleventy-plugin-youtube-embed')
const markdownIt = require('markdown-it') const markdownIt = require('markdown-it')
const markdownItAnchor = require('markdown-it-anchor') const markdownItAnchor = require('markdown-it-anchor')
const markdownItFootnote = require('markdown-it-footnote') const markdownItFootnote = require('markdown-it-footnote')
const filters = require('./config/filters.js')
const dateFilters = require('./config/dateFilters.js') const filters = require('./config/filters/index.js')
const mediaFilters = require('./config/mediaFilters.js')
const feedFilters = require('./config/feedFilters.js')
const CleanCSS = require('clean-css') const CleanCSS = require('clean-css')
const now = String(Date.now())
const { execSync } = require('child_process') const { execSync } = require('child_process')
const tagAliases = require('./src/_data/json/tag-aliases.json') const tagAliases = require('./src/_data/json/tag-aliases.json')
// load .env // load .env
require('dotenv-flow').config() require('dotenv-flow').config()
const imageShortcode = async ( /**
src, * @param {import("@11ty/eleventy/src/UserConfig")} eleventyConfig
alt, */
className = undefined,
loading = 'lazy',
widths = [75, 150, 300, 600, 900, 1200],
formats = ['webp', 'jpeg'],
sizes = '100vw'
) => {
const imageMetadata = await Image(src, {
widths: [...widths, null],
formats: [...formats, null],
outputDir: './_site/assets/img/cache/',
urlPath: '/assets/img/cache/',
})
const stringifyAttributes = (attributeMap) => { const packageVersion = require('./package.json').version
return Object.entries(attributeMap)
.map(([attribute, value]) => {
if (typeof value === 'undefined') return ''
return `${attribute}="${value}"`
})
.join(' ')
}
const sourceHtmlString = Object.values(imageMetadata) // module import shortcodes
.map((images) => { const { img } = require('./config/shortcodes/index.js')
const { sourceType } = images[0]
const sourceAttributes = stringifyAttributes({
type: sourceType,
srcset: images.map((image) => image.srcset).join(', '),
sizes,
})
return `<source ${sourceAttributes}>`
})
.join('\n')
const getLargestImage = (format) => {
const images = imageMetadata[format]
return images[images.length - 1]
}
const largestUnoptimizedImg = getLargestImage(formats[0])
const imgAttributes = stringifyAttributes({
src: largestUnoptimizedImg.url,
width: largestUnoptimizedImg.width,
height: largestUnoptimizedImg.height,
alt,
loading,
decoding: 'async',
})
const imgHtmlString = `<img ${imgAttributes}>`
const pictureAttributes = stringifyAttributes({
class: className,
})
const picture = `<picture ${pictureAttributes}>
${sourceHtmlString}
${imgHtmlString}
</picture>`
return outdent`${picture}`
}
module.exports = function (eleventyConfig) { module.exports = function (eleventyConfig) {
// plugins
eleventyConfig.addPlugin(syntaxHighlight) eleventyConfig.addPlugin(syntaxHighlight)
eleventyConfig.addPlugin(tablerIcons) eleventyConfig.addPlugin(tablerIcons)
eleventyConfig.addPlugin(pluginUnfurl) eleventyConfig.addPlugin(pluginUnfurl)
eleventyConfig.addPlugin(pluginFilesMinifier) eleventyConfig.addPlugin(pluginFilesMinifier)
eleventyConfig.addPlugin(schema) eleventyConfig.addPlugin(schema)
eleventyConfig.addPlugin(eleventyImagePlugin)
eleventyConfig.addPlugin(embedYouTube, { eleventyConfig.addPlugin(embedYouTube, {
modestBranding: true, modestBranding: true,
lite: { lite: {
@ -120,7 +58,7 @@ module.exports = function (eleventyConfig) {
eleventyConfig.addPassthroughCopy('_redirects') eleventyConfig.addPassthroughCopy('_redirects')
// shortcodes // shortcodes
eleventyConfig.addShortcode('version', () => now) eleventyConfig.addShortcode('version', () => packageVersion)
// enable merging of tags // enable merging of tags
eleventyConfig.setDataDeepMerge(true) eleventyConfig.setDataDeepMerge(true)
@ -181,41 +119,20 @@ module.exports = function (eleventyConfig) {
md.use(markdownItFootnote) md.use(markdownItFootnote)
eleventyConfig.setLibrary('md', md) eleventyConfig.setLibrary('md', md)
// markdown filter
eleventyConfig.addLiquidFilter('markdown', (content) => { eleventyConfig.addLiquidFilter('markdown', (content) => {
if (!content) return if (!content) return
return md.render(content) return md.render(content)
}) })
// filters
Object.keys(filters).forEach((filterName) => { Object.keys(filters).forEach((filterName) => {
eleventyConfig.addLiquidFilter(filterName, filters[filterName]) eleventyConfig.addLiquidFilter(filterName, filters[filterName])
}) })
// date filters
Object.keys(dateFilters).forEach((filterName) => {
eleventyConfig.addLiquidFilter(filterName, dateFilters[filterName])
})
// media filters
Object.keys(mediaFilters).forEach((filterName) => {
eleventyConfig.addLiquidFilter(filterName, mediaFilters[filterName])
})
// feed filters
Object.keys(feedFilters).forEach((filterName) => {
eleventyConfig.addLiquidFilter(filterName, feedFilters[filterName])
})
// css filters
eleventyConfig.addFilter('cssmin', (code) => new CleanCSS({}).minify(code).styles)
// rss filters
eleventyConfig.addLiquidFilter('dateToRfc822', pluginRss.dateToRfc822) eleventyConfig.addLiquidFilter('dateToRfc822', pluginRss.dateToRfc822)
eleventyConfig.addLiquidFilter('absoluteUrl', pluginRss.absoluteUrl) eleventyConfig.addLiquidFilter('absoluteUrl', pluginRss.absoluteUrl)
// image shortcode eleventyConfig.addFilter('cssmin', (code) => new CleanCSS({}).minify(code).styles)
eleventyConfig.addShortcode('image', imageShortcode)
eleventyConfig.addShortcode('image', img)
eleventyConfig.on('eleventy.after', () => { eleventyConfig.on('eleventy.after', () => {
execSync(`npx pagefind --site _site --glob "**/*.html"`, { encoding: 'utf-8' }) execSync(`npx pagefind --site _site --glob "**/*.html"`, { encoding: 'utf-8' })

View file

@ -1,44 +0,0 @@
const { DateTime } = require('luxon')
module.exports = {
readableDate: (date) => {
return DateTime.fromISO(date).toFormat('LLLL d, yyyy')
},
toDateTime: (date) => {
const formatted = DateTime.fromISO(date)
const trail = (number) => {
return parseInt(number, 10) < 10 ? `0${number}` : number
}
return `${formatted.year}-${trail(formatted.month)}-${trail(formatted.day)} ${trail(
formatted.hour
)}:${trail(formatted.minute)}`
},
toDateTimeFromUnix: (date) => {
const formatted = DateTime.fromSeconds(parseInt(date, 10))
const trail = (number) => {
return parseInt(number, 10) < 10 ? `0${number}` : number
}
return `${trail(formatted.month)}.${trail(formatted.day)}.${formatted.year} ${trail(
formatted.hour
)}:${trail(formatted.minute)}`
},
isoDateOnly: (date) => {
let d = new Date(date)
let month = '' + (d.getMonth() + 1)
let day = '' + d.getDate()
let year = d.getFullYear()
if (month.length < 2) month = '0' + month
if (day.length < 2) day = '0' + day
return [month, day, year].join('.')
},
stringToDate: (string) => {
if (!string) return
return new Date(string)
},
}

View file

@ -1,31 +0,0 @@
const markdownIt = require('markdown-it')
const { URL } = require('url')
const BASE_URL = 'https://coryd.dev'
module.exports = {
normalizeEntries: (entries) => {
const md = markdownIt({ html: true, linkify: true })
const posts = []
entries.forEach((entry) => {
const dateKey = Object.keys(entry).find((key) => key.includes('date'))
const date = new Date(entry[dateKey])
let excerpt = ''
// set the entry excerpt
if (entry.description) excerpt = entry.description
if (entry.data?.post_excerpt) excerpt = md.render(entry.data.post_excerpt)
// if there's a valid entry return a normalized object
if (entry)
posts.push({
title: entry.data?.title || entry.title,
url: entry.url.includes('http') ? entry.url : new URL(entry.url, BASE_URL).toString(),
content: entry.description,
date,
excerpt,
})
})
return posts
},
}

View file

@ -1,102 +0,0 @@
const marked = require('marked')
const sanitizeHTML = require('sanitize-html')
const utmPattern = /[?&](utm_[^&=]+=[^&#]*)/gi
const BASE_URL = 'https://coryd.dev'
module.exports = {
trim: (string, limit) => {
return string.length <= limit ? string : `${string.slice(0, limit)}...`
},
stripIndex: (path) => {
return path.replace('/index.html', '/')
},
mdToHtml: (content) => {
return marked.parse(content)
},
btoa: (string) => {
return btoa(string)
},
dashLower: (string) => string.replace(/\s+/g, '-').toLowerCase(),
encodeAmp: (string) => {
if (!string) return
const pattern = /&(?!(?:[a-zA-Z]+|#[0-9]+|#x[0-9a-fA-F]+);)/g
const replacement = '&amp;'
return string.replace(pattern, replacement)
},
stripUtm: (string) => {
if (!string) return
return string.replace(utmPattern, '')
},
getPostImage: (image) => {
if (image && image !== '') return image
return `${BASE_URL}/assets/img/social-card.jpg`
},
getPopularPosts: (posts, analytics) => {
return posts
.filter((post) => {
if (analytics.find((p) => p.page === post.url)) return true
})
.sort((a, b) => {
const visitors = (page) => analytics.filter((p) => p.page === page.url).pop().visitors
return visitors(b) - visitors(a)
})
},
tagLookup: (url, tagMap) => {
if (!url) return
if (url.includes('thestorygraph.com')) return '#Books #NowReading #TheStoryGraph'
if (url.includes('trakt.tv')) return '#Movies #Watching #Trakt'
return tagMap[url] || ''
},
webmentionsByUrl: (webmentions, url) => {
const allowedTypes = ['mention-of', 'in-reply-to', 'like-of', 'repost-of']
const data = {
'like-of': [],
'repost-of': [],
'in-reply-to': [],
'mention-of': [],
'link-to': [],
}
const hasRequiredReplyFields = (entry) => {
const { author, published, content } = entry
return author.name && author.photo && published && content
}
const hasRequiredMentionFields = (entry) => {
const { name, url } = entry
return name && url
}
const filtered =
webmentions
.filter((entry) => entry['wm-target'] === `${BASE_URL}${url}`)
.filter((entry) => allowedTypes.includes(entry['wm-property'])) || []
filtered.forEach((m) => {
if (data[m['wm-property']]) {
const isReply = m['wm-property'] === 'in-reply-to'
const isMention = m['wm-property'] === 'mention-of'
const isValidReply = (isReply || isMention) && hasRequiredReplyFields(m)
if (isReply || isMention) {
if (isValidReply) {
m.sanitized = sanitizeHTML(m.content.html)
data[m['wm-property']].unshift(m)
}
if (isMention && hasRequiredMentionFields(m)) data['link-to'].push(m)
return
}
data[m['wm-property']].unshift(m)
}
})
data['in-reply-to'] = [...data['in-reply-to'], ...data['mention-of']]
data['in-reply-to'].sort((a, b) =>
a.published > b.published ? 1 : b.published > a.published ? -1 : 0
)
return data
},
}

210
config/filters/index.js Normal file
View file

@ -0,0 +1,210 @@
const { DateTime } = require('luxon')
const markdownIt = require('markdown-it')
const { URL } = require('url')
const marked = require('marked')
const sanitizeHTML = require('sanitize-html')
const utmPattern = /[?&](utm_[^&=]+=[^&#]*)/gi
const BASE_URL = 'https://coryd.dev'
module.exports = {
// general
trim: (string, limit) => {
return string.length <= limit ? string : `${string.slice(0, limit)}...`
},
stripIndex: (path) => {
return path.replace('/index.html', '/')
},
mdToHtml: (content) => {
return marked.parse(content)
},
btoa: (string) => {
return btoa(string)
},
dashLower: (string) => string.replace(/\s+/g, '-').toLowerCase(),
encodeAmp: (string) => {
if (!string) return
const pattern = /&(?!(?:[a-zA-Z]+|#[0-9]+|#x[0-9a-fA-F]+);)/g
const replacement = '&amp;'
return string.replace(pattern, replacement)
},
stripUtm: (string) => {
if (!string) return
return string.replace(utmPattern, '')
},
getPostImage: (image) => {
if (image && image !== '') return image
return `${BASE_URL}/assets/img/social-card.jpg`
},
getPopularPosts: (posts, analytics) => {
return posts
.filter((post) => {
if (analytics.find((p) => p.page === post.url)) return true
})
.sort((a, b) => {
const visitors = (page) => analytics.filter((p) => p.page === page.url).pop().visitors
return visitors(b) - visitors(a)
})
},
tagLookup: (url, tagMap) => {
if (!url) return
if (url.includes('thestorygraph.com')) return '#Books #NowReading #TheStoryGraph'
if (url.includes('trakt.tv')) return '#Movies #Watching #Trakt'
return tagMap[url] || ''
},
webmentionsByUrl: (webmentions, url) => {
const allowedTypes = ['mention-of', 'in-reply-to', 'like-of', 'repost-of']
const data = {
'like-of': [],
'repost-of': [],
'in-reply-to': [],
'mention-of': [],
'link-to': [],
}
const hasRequiredReplyFields = (entry) => {
const { author, published, content } = entry
return author.name && author.photo && published && content
}
const hasRequiredMentionFields = (entry) => {
const { name, url } = entry
return name && url
}
const filtered =
webmentions
.filter((entry) => entry['wm-target'] === `${BASE_URL}${url}`)
.filter((entry) => allowedTypes.includes(entry['wm-property'])) || []
filtered.forEach((m) => {
if (data[m['wm-property']]) {
const isReply = m['wm-property'] === 'in-reply-to'
const isMention = m['wm-property'] === 'mention-of'
const isValidReply = (isReply || isMention) && hasRequiredReplyFields(m)
if (isReply || isMention) {
if (isValidReply) {
m.sanitized = sanitizeHTML(m.content.html)
data[m['wm-property']].unshift(m)
}
if (isMention && hasRequiredMentionFields(m)) data['link-to'].push(m)
return
}
data[m['wm-property']].unshift(m)
}
})
data['in-reply-to'] = [...data['in-reply-to'], ...data['mention-of']]
data['in-reply-to'].sort((a, b) =>
a.published > b.published ? 1 : b.published > a.published ? -1 : 0
)
return data
},
// dates
readableDate: (date) => {
return DateTime.fromISO(date).toFormat('LLLL d, yyyy')
},
toDateTime: (date) => {
const formatted = DateTime.fromISO(date)
const trail = (number) => {
return parseInt(number, 10) < 10 ? `0${number}` : number
}
return `${formatted.year}-${trail(formatted.month)}-${trail(formatted.day)} ${trail(
formatted.hour
)}:${trail(formatted.minute)}`
},
toDateTimeFromUnix: (date) => {
const formatted = DateTime.fromSeconds(parseInt(date, 10))
const trail = (number) => {
return parseInt(number, 10) < 10 ? `0${number}` : number
}
return `${trail(formatted.month)}.${trail(formatted.day)}.${formatted.year} ${trail(
formatted.hour
)}:${trail(formatted.minute)}`
},
isoDateOnly: (date) => {
let d = new Date(date)
let month = '' + (d.getMonth() + 1)
let day = '' + d.getDate()
let year = d.getFullYear()
if (month.length < 2) month = '0' + month
if (day.length < 2) day = '0' + day
return [month, day, year].join('.')
},
stringToDate: (string) => {
if (!string) return
return new Date(string)
},
// feeds
normalizeEntries: (entries) => {
const md = markdownIt({ html: true, linkify: true })
const posts = []
entries.forEach((entry) => {
const dateKey = Object.keys(entry).find((key) => key.includes('date'))
const date = new Date(entry[dateKey])
let excerpt = ''
// set the entry excerpt
if (entry.description) excerpt = entry.description
if (entry.data?.post_excerpt) excerpt = md.render(entry.data.post_excerpt)
// if there's a valid entry return a normalized object
if (entry)
posts.push({
title: entry.data?.title || entry.title,
url: entry.url.includes('http') ? entry.url : new URL(entry.url, BASE_URL).toString(),
content: entry.description,
date,
excerpt,
})
})
return posts
},
// media
normalizeMedia: (media) =>
media.map((item) => {
let normalized = {
image: item['image'],
url: item['url'],
}
if (item.type === 'album') {
normalized['title'] = item['title']
normalized['alt'] = `${item['title']} by ${item['artist']}`
normalized['subtext'] = `${item['artist']}`
}
if (item.type === 'artist') {
normalized['title'] = item['title']
normalized['alt'] = `${item['plays']} plays of ${item['title']}`
normalized['subtext'] = `${item['plays']} plays`
}
if (item.type === 'movie') normalized['alt'] = item['title']
if (item.type === 'book') {
normalized['alt'] = `${item['title']} by ${item['author']}`
normalized['subtext'] = `${item['percentage']} finished`
normalized['percentage'] = item['percentage']
}
if (item.type === 'tv') {
normalized['title'] = item['title']
normalized['alt'] = `${item['title']} from ${item['name']}`
normalized['subtext'] = item['subtext']
}
if (item.type === 'tv-range') {
normalized['title'] = item['name']
normalized['alt'] = `${item['subtext']} from ${item['name']}`
normalized['subtext'] = item['subtext']
}
return normalized
}),
}

View file

@ -1,36 +0,0 @@
module.exports = {
normalizeMedia: (media) =>
media.map((item) => {
let normalized = {
image: item['image'],
url: item['url'],
}
if (item.type === 'album') {
normalized['title'] = item['title']
normalized['alt'] = `${item['title']} by ${item['artist']}`
normalized['subtext'] = `${item['artist']}`
}
if (item.type === 'artist') {
normalized['title'] = item['title']
normalized['alt'] = `${item['plays']} plays of ${item['title']}`
normalized['subtext'] = `${item['plays']} plays`
}
if (item.type === 'movie') normalized['alt'] = item['title']
if (item.type === 'book') {
normalized['alt'] = `${item['title']} by ${item['author']}`
normalized['subtext'] = `${item['percentage']} finished`
normalized['percentage'] = item['percentage']
}
if (item.type === 'tv') {
normalized['title'] = item['title']
normalized['alt'] = `${item['title']} from ${item['name']}`
normalized['subtext'] = item['subtext']
}
if (item.type === 'tv-range') {
normalized['title'] = item['name']
normalized['alt'] = `${item['subtext']} from ${item['name']}`
normalized['subtext'] = item['subtext']
}
return normalized
}),
}

View file

@ -0,0 +1,70 @@
const outdent = require('outdent')
const Image = require('@11ty/eleventy-img')
const img = async (
src,
alt,
className = undefined,
loading = 'lazy',
widths = [75, 150, 300, 600, 900, 1200],
formats = ['webp', 'jpeg'],
sizes = '100vw'
) => {
const imageMetadata = await Image(src, {
widths: [...widths, null],
formats: [...formats, null],
outputDir: './_site/assets/img/cache/',
urlPath: '/assets/img/cache/',
})
const stringifyAttributes = (attributeMap) => {
return Object.entries(attributeMap)
.map(([attribute, value]) => {
if (typeof value === 'undefined') return ''
return `${attribute}="${value}"`
})
.join(' ')
}
const sourceHtmlString = Object.values(imageMetadata)
.map((images) => {
const { sourceType } = images[0]
const sourceAttributes = stringifyAttributes({
type: sourceType,
srcset: images.map((image) => image.srcset).join(', '),
sizes,
})
return `<source ${sourceAttributes}>`
})
.join('\n')
const getLargestImage = (format) => {
const images = imageMetadata[format]
return images[images.length - 1]
}
const largestUnoptimizedImg = getLargestImage(formats[0])
const imgAttributes = stringifyAttributes({
src: largestUnoptimizedImg.url,
width: largestUnoptimizedImg.width,
height: largestUnoptimizedImg.height,
alt,
loading,
decoding: 'async',
})
const imgHtmlString = `<img ${imgAttributes}>`
const pictureAttributes = stringifyAttributes({
class: className,
})
const picture = `<picture ${pictureAttributes}>
${sourceHtmlString}
${imgHtmlString}
</picture>`
return outdent`${picture}`
}
module.exports = img

View file

@ -0,0 +1,4 @@
const img = require('./img')
module.exports = {
img,
}

View file

@ -1,6 +1,6 @@
{ {
"name": "coryd.dev", "name": "coryd.dev",
"version": "2.5.1", "version": "2.5.3",
"description": "The source for my personal site, blog and portfolio. Built using 11ty and hosted on Netlify.", "description": "The source for my personal site, blog and portfolio. Built using 11ty and hosted on Netlify.",
"main": "index.html", "main": "index.html",
"scripts": { "scripts": {