-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathgenerate-posts-images.mjs
133 lines (111 loc) · 3.14 KB
/
generate-posts-images.mjs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
// @ts-check
import fs from 'fs'
import path from 'path'
import playwright from 'playwright'
import * as url from 'url'
const rootDir = url.fileURLToPath(new URL('..', import.meta.url))
const DEV = process.env.NODE_ENV === 'development'
console.log('Preparing Playwright to generate posts images...')
const browser = await playwright.chromium.launch()
const context = await browser.newContext(playwright.devices['Desktop Chrome'])
const page = await context.newPage()
page.on('requestfailed', (request) => {
throw new Error(
`Failed to generate post image due to request failure: ${
request.failure()?.errorText
}`
)
})
/** @typedef {() => void} NoOp */
/** @type {NoOp[]} */
const operations = []
for (const {
name,
url,
getScreenshotPath,
} of await getPostImagesToScreenshot()) {
await page.goto(url, { waitUntil: 'networkidle' })
const screenshot = await page.screenshot()
const screenshotPath = getScreenshotPath(name)
const blogPostPath = new URL(url).pathname
operations.push(() => {
saveScreenshot(screenshot, screenshotPath)
console.log(
'Saved post image for %s in path %s',
blogPostPath,
stripRootDir(screenshotPath)
)
})
console.log(`Prepared image of post ${blogPostPath}`)
}
for (const operation of operations) {
operation()
}
await context.close()
await browser.close()
/**
* @typedef {object} PostImage
* @property {string} url
* @property {string} name
*/
async function fetchPostImages(/** @type {string} */ url) {
const response = await fetch(url)
const json = response.json()
return /** @type {PostImage[]} */ json
}
/**
* @typedef {PostImage} PostImageToScrenshot
* @property {(name: string) => name} getScrenshotPath
*/
export async function getPostImagesToScreenshot() {
const [englishPostImages, portuguesePostImages] = await Promise.all([
fetchPostImages('http://localhost:4321/posts/images.json'),
fetchPostImages('http://localhost:4321/pt/posts/images.json'),
])
return [
...englishPostImages.map((/** @type {PostImage} */ postImage) => ({
...postImage,
getScreenshotPath: () => {
if (DEV) {
return path.join(rootDir, 'screenshots', `${postImage.name}.png`)
}
return path.join(
rootDir,
'public',
'posts',
postImage.name,
'image.png'
)
},
})),
...portuguesePostImages.map((/** @type {PostImage} */ postImage) => ({
...postImage,
getScreenshotPath: () => {
if (DEV) {
return path.join(rootDir, 'screenshots', `${postImage.name}.pt.png`)
}
return path.join(
rootDir,
'public',
'pt',
'posts',
postImage.name,
'image.png'
)
},
})),
]
}
/** @type {(screenshot: Buffer, filePath: string) => void} */
function saveScreenshot(screenshot, filePath) {
if (!fs.existsSync(path.dirname(filePath))) {
fs.mkdirSync(path.dirname(filePath), { recursive: true })
}
fs.writeFileSync(filePath, screenshot)
}
/**
* @type {(filePath: string) => string}
*/
function stripRootDir(filePath) {
return filePath.replace(rootDir, '')
}