| | |
| | | import path from "path" |
| | | import fs from "fs" |
| | | import { QuartzConfig } from "../cfg" |
| | | import { PerfTimer } from "../perf" |
| | | import { PerfTimer } from "../util/perf" |
| | | import { getStaticResourcesFromPlugins } from "../plugins" |
| | | import { EmitCallback } from "../plugins/types" |
| | | import { ProcessedContent } from "../plugins/vfile" |
| | | import { QUARTZ, slugify } from "../path" |
| | | import { globbyStream } from "globby" |
| | | import { QuartzLogger } from "../util/log" |
| | | import { trace } from "../util/trace" |
| | | import { BuildCtx } from "../util/ctx" |
| | | import { styleText } from "util" |
| | | |
| | | export async function emitContent(output: string, cfg: QuartzConfig, content: ProcessedContent[], verbose: boolean) { |
| | | export async function emitContent(ctx: BuildCtx, content: ProcessedContent[]) { |
| | | const { argv, cfg } = ctx |
| | | const perf = new PerfTimer() |
| | | const log = new QuartzLogger(ctx.argv.verbose) |
| | | |
| | | |
| | | const staticResources = getStaticResourcesFromPlugins(cfg.plugins) |
| | | const emit: EmitCallback = async ({ slug, ext, content }) => { |
| | | const pathToPage = path.join(output, slug + ext) |
| | | const dir = path.dirname(pathToPage) |
| | | await fs.promises.mkdir(dir, { recursive: true }) |
| | | await fs.promises.writeFile(pathToPage, content) |
| | | return pathToPage |
| | | } |
| | | log.start(`Emitting files`) |
| | | |
| | | let emittedFiles = 0 |
| | | for (const emitter of cfg.plugins.emitters) { |
| | | const emitted = await emitter.emit(content, staticResources, emit) |
| | | emittedFiles += emitted.length |
| | | |
| | | if (verbose) { |
| | | for (const file of emitted) { |
| | | console.log(`[emit:${emitter.name}] ${file}`) |
| | | const staticResources = getStaticResourcesFromPlugins(ctx) |
| | | await Promise.all( |
| | | cfg.plugins.emitters.map(async (emitter) => { |
| | | try { |
| | | const emitted = await emitter.emit(ctx, content, staticResources) |
| | | if (Symbol.asyncIterator in emitted) { |
| | | // Async generator case |
| | | for await (const file of emitted) { |
| | | emittedFiles++ |
| | | if (ctx.argv.verbose) { |
| | | console.log(`[emit:${emitter.name}] ${file}`) |
| | | } else { |
| | | log.updateText(`${emitter.name} -> ${styleText("gray", file)}`) |
| | | } |
| | | } |
| | | } else { |
| | | // Array case |
| | | emittedFiles += emitted.length |
| | | for (const file of emitted) { |
| | | if (ctx.argv.verbose) { |
| | | console.log(`[emit:${emitter.name}] ${file}`) |
| | | } else { |
| | | log.updateText(`${emitter.name} -> ${styleText("gray", file)}`) |
| | | } |
| | | } |
| | | } |
| | | } catch (err) { |
| | | trace(`Failed to emit from plugin \`${emitter.name}\``, err as Error) |
| | | } |
| | | } |
| | | } |
| | | }), |
| | | ) |
| | | |
| | | const staticPath = path.join(QUARTZ, "static") |
| | | await fs.promises.cp(staticPath, path.join(output, "static"), { recursive: true }) |
| | | |
| | | // glob all non MD/MDX/HTML files in content folder and copy it over |
| | | const assetsPath = path.join("public", "assets") |
| | | for await (const fp of globbyStream("**", { |
| | | ignore: ["**/*.{md,mdx,html}"], |
| | | cwd: "./content", |
| | | })) { |
| | | const ext = path.extname(fp as string) |
| | | const src = path.join("content", fp as string) |
| | | const dest = path.join(assetsPath, slugify(fp as string) + ext) |
| | | const dir = path.dirname(dest) |
| | | await fs.promises.mkdir(dir, { recursive: true }) // ensure dir exists |
| | | await fs.promises.copyFile(src, dest) |
| | | emittedFiles += 1 |
| | | if (verbose) { |
| | | console.log(`[emit:Assets] ${dest}`) |
| | | } |
| | | } |
| | | |
| | | if (verbose) { |
| | | console.log(`[emit:Static] ${path.join(output, "static", "**")}`) |
| | | console.log(`Emitted ${emittedFiles} files to \`${output}\` in ${perf.timeSince()}`) |
| | | } |
| | | log.end(`Emitted ${emittedFiles} files to \`${argv.output}\` in ${perf.timeSince()}`) |
| | | } |