From 13c3b712c7ba45d0081f459fc06f142216a4ec59 Mon Sep 17 00:00:00 2001 From: Nate Moore Date: Tue, 30 Jan 2024 17:16:31 -0600 Subject: [PATCH] Improves YAML error handling consistency (#9882) * fix: improves yaml error handling consistency * chore: add changeset --- .changeset/twelve-humans-remain.md | 5 +++ packages/astro/src/content/utils.ts | 36 +++++++++++++------ .../content-entry-type.ts | 7 ++-- .../astro/src/vite-plugin-markdown/index.ts | 33 ++--------------- 4 files changed, 37 insertions(+), 44 deletions(-) create mode 100644 .changeset/twelve-humans-remain.md diff --git a/.changeset/twelve-humans-remain.md b/.changeset/twelve-humans-remain.md new file mode 100644 index 0000000000..afbcc025a7 --- /dev/null +++ b/.changeset/twelve-humans-remain.md @@ -0,0 +1,5 @@ +--- +"astro": patch +--- + +Improves handling of YAML parsing errors diff --git a/packages/astro/src/content/utils.ts b/packages/astro/src/content/utils.ts index 3515a43c72..21d5cc0eae 100644 --- a/packages/astro/src/content/utils.ts +++ b/packages/astro/src/content/utils.ts @@ -14,7 +14,8 @@ import type { } from '../@types/astro.js'; import { AstroError, AstroErrorData } from '../core/errors/index.js'; -import { formatYAMLException, isYAMLException } from '../core/errors/utils.js'; +import { MarkdownError } from '../core/errors/index.js'; +import { isYAMLException } from '../core/errors/utils.js'; import { CONTENT_FLAGS, CONTENT_TYPES_FILE } from './consts.js'; import { errorMap } from './error-map.js'; import { createImage } from './runtime-assets.js'; @@ -287,18 +288,31 @@ function getYAMLErrorLine(rawData: string | undefined, objectKey: string) { return numNewlinesBeforeKey; } -export function parseFrontmatter(fileContents: string) { + +export function safeParseFrontmatter(source: string, id?: string) { try { - // `matter` is empty string on cache results - // clear cache to prevent this - (matter as any).clearCache(); - return matter(fileContents); - } catch (e) { - if (isYAMLException(e)) { - throw formatYAMLException(e); - } else { - throw e; + return matter(source); + } catch (err: any) { + const markdownError = new MarkdownError({ + name: 'MarkdownError', + message: err.message, + stack: err.stack, + location: id ? { + file: id, + } : undefined, + }); + + if (isYAMLException(err)) { + markdownError.setLocation({ + file: id, + line: err.mark.line, + column: err.mark.column, + }); + + markdownError.setMessage(err.reason); } + + throw markdownError; } } diff --git a/packages/astro/src/vite-plugin-markdown/content-entry-type.ts b/packages/astro/src/vite-plugin-markdown/content-entry-type.ts index dc32969747..9e8690033c 100644 --- a/packages/astro/src/vite-plugin-markdown/content-entry-type.ts +++ b/packages/astro/src/vite-plugin-markdown/content-entry-type.ts @@ -1,10 +1,11 @@ +import { fileURLToPath } from 'node:url'; import type { ContentEntryType } from '../@types/astro.js'; -import { parseFrontmatter } from '../content/utils.js'; +import { safeParseFrontmatter } from '../content/utils.js'; export const markdownContentEntryType: ContentEntryType = { extensions: ['.md'], - async getEntryInfo({ contents }: { contents: string }) { - const parsed = parseFrontmatter(contents); + async getEntryInfo({ contents, fileUrl }: { contents: string, fileUrl: URL }) { + const parsed = safeParseFrontmatter(contents, fileURLToPath(fileUrl)); return { data: parsed.data, body: parsed.content, diff --git a/packages/astro/src/vite-plugin-markdown/index.ts b/packages/astro/src/vite-plugin-markdown/index.ts index cc138149b1..445b69d310 100644 --- a/packages/astro/src/vite-plugin-markdown/index.ts +++ b/packages/astro/src/vite-plugin-markdown/index.ts @@ -3,14 +3,14 @@ import { InvalidAstroDataError, type MarkdownProcessor, } from '@astrojs/markdown-remark'; -import matter from 'gray-matter'; import fs from 'node:fs'; import path from 'node:path'; import { fileURLToPath, pathToFileURL } from 'node:url'; import type { Plugin } from 'vite'; import { normalizePath } from 'vite'; import type { AstroSettings } from '../@types/astro.js'; -import { AstroError, AstroErrorData, MarkdownError } from '../core/errors/index.js'; +import { AstroError, AstroErrorData } from '../core/errors/index.js'; +import { safeParseFrontmatter } from '../content/utils.js'; import type { Logger } from '../core/logger/core.js'; import { isMarkdownFile } from '../core/util.js'; import { shorthash } from '../runtime/server/shorthash.js'; @@ -23,33 +23,6 @@ interface AstroPluginOptions { logger: Logger; } -function safeMatter(source: string, id: string) { - try { - return matter(source); - } catch (err: any) { - const markdownError = new MarkdownError({ - name: 'MarkdownError', - message: err.message, - stack: err.stack, - location: { - file: id, - }, - }); - - if (err.name === 'YAMLException') { - markdownError.setLocation({ - file: id, - line: err.mark.line, - column: err.mark.column, - }); - - markdownError.setMessage(err.reason); - } - - throw markdownError; - } -} - const astroServerRuntimeModulePath = normalizePath( fileURLToPath(new URL('../runtime/server/index.js', import.meta.url)) ); @@ -75,7 +48,7 @@ export default function markdown({ settings, logger }: AstroPluginOptions): Plug if (isMarkdownFile(id)) { const { fileId, fileUrl } = getFileInfo(id, settings.config); const rawFile = await fs.promises.readFile(fileId, 'utf-8'); - const raw = safeMatter(rawFile, id); + const raw = safeParseFrontmatter(rawFile, id); const fileURL = pathToFileURL(fileId);