@autoblogwriter/sdk 2.1.1 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +56 -27
- package/dist/index.cjs +34 -34
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +13 -13
- package/dist/index.d.ts +13 -13
- package/dist/index.js +31 -31
- package/dist/index.js.map +1 -1
- package/dist/{metadata-DUQ3-Hhw.d.ts → metadata-CeyU-t7o.d.ts} +1 -1
- package/dist/{metadata-Ihd3IqKu.d.cts → metadata-D1acl8ZW.d.cts} +1 -1
- package/dist/next.cjs +33 -33
- package/dist/next.cjs.map +1 -1
- package/dist/next.d.cts +2 -2
- package/dist/next.d.ts +2 -2
- package/dist/next.js +33 -33
- package/dist/next.js.map +1 -1
- package/dist/react.cjs +2 -2
- package/dist/react.cjs.map +1 -1
- package/dist/react.d.cts +1 -1
- package/dist/react.d.ts +1 -1
- package/dist/react.js +2 -2
- package/dist/react.js.map +1 -1
- package/dist/revalidate.cjs +5 -5
- package/dist/revalidate.cjs.map +1 -1
- package/dist/revalidate.d.cts +3 -3
- package/dist/revalidate.d.ts +3 -3
- package/dist/revalidate.js +5 -5
- package/dist/revalidate.js.map +1 -1
- package/dist/{types-CCDRs0sO.d.cts → types-ClZVNiGd.d.cts} +4 -4
- package/dist/{types-CCDRs0sO.d.ts → types-ClZVNiGd.d.ts} +4 -4
- package/package.json +1 -1
package/dist/react.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/react/index.ts","../src/render.ts","../src/react/Markdown.tsx","../src/react/BlogPost.tsx","../src/react/BlogPostList.tsx"],"sourcesContent":["export { Markdown, type MarkdownProps } from \"./Markdown\";\nexport { BlogPost, type BlogPostProps } from \"./BlogPost\";\nexport { BlogPostList, type BlogPostListProps } from \"./BlogPostList\";\n","const CODE_BLOCK_TOKEN = \"__BLOGAUTO_CODE_BLOCK_\";\n\nfunction escapeHtml(value: string): string {\n return value\n .replace(/&/g, \"&\")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\")\n .replace(/\"/g, \""\")\n .replace(/'/g, \"'\");\n}\n\nfunction renderInline(input: string): string {\n let output = escapeHtml(input);\n output = output.replace(/\\*\\*(.+?)\\*\\*/g, \"<strong>$1</strong>\");\n output = output.replace(/\\*(.+?)\\*/g, \"<em>$1</em>\");\n output = output.replace(/`([^`]+)`/g, (_, code) => `<code>${code}</code>`);\n output = output.replace(/!\\[([^\\]]*)\\]\\(([^)]+)\\)/g, (_, alt, src) => {\n const safeSrc = escapeHtml(src);\n const safeAlt = escapeHtml(alt);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n });\n output = output.replace(/\\[(.+?)\\]\\((.+?)\\)/g, (_, label, href) => {\n const safeHref = escapeHtml(href);\n return `<a href=\"${safeHref}\" target=\"_blank\" rel=\"noreferrer\">${label}</a>`;\n });\n output = output.replace(/\\n/g, \"<br />\");\n return output;\n}\n\nfunction replaceCodeBlocks(markdown: string, blocks: string[]): string {\n return markdown.replace(/```([\\s\\S]*?)```/g, (_, code) => {\n const index = blocks.push(`<pre><code>${escapeHtml(code.trim())}</code></pre>`) - 1;\n return `${CODE_BLOCK_TOKEN}${index}__`;\n });\n}\n\nfunction restoreCodeBlocks(html: string, blocks: string[]): string {\n return html.replace(/__BLOGAUTO_CODE_BLOCK_(\\d+)__/g, (_, rawIndex) => blocks[Number(rawIndex)] ?? \"\");\n}\n\nexport function renderMarkdownToHtml(markdown: string): string {\n if (!markdown) {\n return \"\";\n }\n\n const codeBlocks: string[] = [];\n const withoutCode = replaceCodeBlocks(markdown, codeBlocks);\n const normalized = withoutCode.replace(/\\r\\n/g, \"\\n\");\n // Insert blank lines around heading lines so they become separate blocks\n const withHeadingBreaks = normalized\n .replace(/\\n(#{1,6}\\s)/g, \"\\n\\n$1\")\n .replace(/(#{1,6}\\s[^\\n]+)\\n(?!#|\\n)/g, \"$1\\n\\n\");\n const blocks = withHeadingBreaks\n .split(/\\n{2,}/)\n .map((block) => block.trim())\n .filter(Boolean);\n\n const htmlBlocks = blocks.map((block) => {\n const headingMatch = block.match(/^(#{1,6})\\s+(.*)$/);\n if (headingMatch) {\n const level = headingMatch[1].length;\n const content = renderInline(headingMatch[2]);\n return `<h${level}>${content}</h${level}>`;\n }\n\n // Check if block is a standalone image\n const imageMatch = block.match(/^!\\[([^\\]]*)\\]\\(([^)]+)\\)$/);\n if (imageMatch) {\n const safeSrc = escapeHtml(imageMatch[2]);\n const safeAlt = escapeHtml(imageMatch[1]);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n }\n\n return `<p>${renderInline(block)}</p>`;\n });\n\n const html = htmlBlocks.join(\"\\n\");\n return restoreCodeBlocks(html, codeBlocks);\n}\n","import { renderMarkdownToHtml } from \"../render\";\n\nexport interface MarkdownProps {\n source?: string | null;\n className?: string;\n}\n\nexport function Markdown({ source, className = \"ba-markdown\" }: MarkdownProps) {\n if (!source) {\n return null;\n }\n\n return (\n <div\n className={className}\n dangerouslySetInnerHTML={{ __html: renderMarkdownToHtml(source) }}\n />\n );\n}\n","import type { BlogPost as BlogPostType } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\nexport interface BlogPostProps {\n post: BlogPostType;\n showTitle?: boolean;\n showDate?: boolean;\n className?: string;\n renderContent?: (content: string) => React.ReactNode;\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPost({\n post,\n showTitle = true,\n showDate = true,\n className = \"ba-post\",\n renderContent,\n}: BlogPostProps) {\n const readingTime = post.metadata?.readingTimeMinutes;\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article className={className}>\n {showTitle && <h1 className=\"ba-post-title\">{post.title}</h1>}\n {(showDate || readingTime) && (\n <p className=\"ba-post-meta\">\n {showDate && (\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n )}\n {showDate && readingTime && <span className=\"ba-post-meta-sep\"> · </span>}\n {readingTime && (\n <span className=\"ba-post-reading-time\">{readingTime} min read</span>\n )}\n </p>\n )}\n {renderContent ? renderContent(post.content) : <Markdown source={post.content} />}\n </article>\n );\n}\n","import type { BlogPost } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype LinkComponent = React.ComponentType<any>;\n\nexport interface BlogPostListProps {\n posts: BlogPost[];\n title?: string;\n routePrefix?: string;\n linkComponent?: LinkComponent;\n className?: string;\n renderCard?: (post: BlogPost, href: string) => React.ReactNode;\n}\n\nfunction DefaultLink({\n href,\n className,\n children,\n}: {\n href: string;\n className?: string;\n children: React.ReactNode;\n}) {\n return (\n <a href={href} className={className}>\n {children}\n </a>\n );\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPostList({\n posts,\n title = \"Latest posts\",\n routePrefix = \"/blog\",\n linkComponent,\n className = \"ba-listing\",\n renderCard,\n}: BlogPostListProps) {\n const Link = linkComponent ?? DefaultLink;\n const prefix = routePrefix.replace(/\\/$/, \"\");\n\n return (\n <section className={className}>\n {title && <h1 className=\"ba-listing-title\">{title}</h1>}\n <div className=\"ba-posts\">\n {posts.map((post) => {\n const href = `${prefix}/${post.slug}`;\n\n if (renderCard) {\n return <div key={post.id}>{renderCard(post, href)}</div>;\n }\n\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article key={post.id} className=\"ba-post-card\">\n <h2 className=\"ba-post-card-title\">\n <Link href={href} className=\"ba-post-card-link\">\n {post.title}\n </Link>\n </h2>\n <p className=\"ba-post-card-meta\">\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n </p>\n <div className=\"ba-post-card-excerpt\">\n <Markdown source={post.excerpt ?? post.content.slice(0, 180)} />\n </div>\n </article>\n );\n })}\n </div>\n </section>\n );\n}\n"],"mappings":"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","names":["import_jsx_runtime","import_jsx_runtime","formatDate"]}
|
|
1
|
+
{"version":3,"sources":["../src/react/index.ts","../src/render.ts","../src/react/Markdown.tsx","../src/react/BlogPost.tsx","../src/react/BlogPostList.tsx"],"sourcesContent":["export { Markdown, type MarkdownProps } from \"./Markdown\";\nexport { BlogPost, type BlogPostProps } from \"./BlogPost\";\nexport { BlogPostList, type BlogPostListProps } from \"./BlogPostList\";\n","const CODE_BLOCK_TOKEN = \"__AUTOBLOGWRITER_CODE_BLOCK_\";\n\nfunction escapeHtml(value: string): string {\n return value\n .replace(/&/g, \"&\")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\")\n .replace(/\"/g, \""\")\n .replace(/'/g, \"'\");\n}\n\nfunction renderInline(input: string): string {\n let output = escapeHtml(input);\n output = output.replace(/\\*\\*(.+?)\\*\\*/g, \"<strong>$1</strong>\");\n output = output.replace(/\\*(.+?)\\*/g, \"<em>$1</em>\");\n output = output.replace(/`([^`]+)`/g, (_, code) => `<code>${code}</code>`);\n output = output.replace(/!\\[([^\\]]*)\\]\\(([^)]+)\\)/g, (_, alt, src) => {\n const safeSrc = escapeHtml(src);\n const safeAlt = escapeHtml(alt);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n });\n output = output.replace(/\\[(.+?)\\]\\((.+?)\\)/g, (_, label, href) => {\n const safeHref = escapeHtml(href);\n return `<a href=\"${safeHref}\" target=\"_blank\" rel=\"noreferrer\">${label}</a>`;\n });\n output = output.replace(/\\n/g, \"<br />\");\n return output;\n}\n\nfunction replaceCodeBlocks(markdown: string, blocks: string[]): string {\n return markdown.replace(/```([\\s\\S]*?)```/g, (_, code) => {\n const index = blocks.push(`<pre><code>${escapeHtml(code.trim())}</code></pre>`) - 1;\n return `${CODE_BLOCK_TOKEN}${index}__`;\n });\n}\n\nfunction restoreCodeBlocks(html: string, blocks: string[]): string {\n return html.replace(/__AUTOBLOGWRITER_CODE_BLOCK_(\\d+)__/g, (_, rawIndex) => blocks[Number(rawIndex)] ?? \"\");\n}\n\nexport function renderMarkdownToHtml(markdown: string): string {\n if (!markdown) {\n return \"\";\n }\n\n const codeBlocks: string[] = [];\n const withoutCode = replaceCodeBlocks(markdown, codeBlocks);\n const normalized = withoutCode.replace(/\\r\\n/g, \"\\n\");\n // Insert blank lines around heading lines so they become separate blocks\n const withHeadingBreaks = normalized\n .replace(/\\n(#{1,6}\\s)/g, \"\\n\\n$1\")\n .replace(/(#{1,6}\\s[^\\n]+)\\n(?!#|\\n)/g, \"$1\\n\\n\");\n const blocks = withHeadingBreaks\n .split(/\\n{2,}/)\n .map((block) => block.trim())\n .filter(Boolean);\n\n const htmlBlocks = blocks.map((block) => {\n const headingMatch = block.match(/^(#{1,6})\\s+(.*)$/);\n if (headingMatch) {\n const level = headingMatch[1].length;\n const content = renderInline(headingMatch[2]);\n return `<h${level}>${content}</h${level}>`;\n }\n\n // Check if block is a standalone image\n const imageMatch = block.match(/^!\\[([^\\]]*)\\]\\(([^)]+)\\)$/);\n if (imageMatch) {\n const safeSrc = escapeHtml(imageMatch[2]);\n const safeAlt = escapeHtml(imageMatch[1]);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n }\n\n return `<p>${renderInline(block)}</p>`;\n });\n\n const html = htmlBlocks.join(\"\\n\");\n return restoreCodeBlocks(html, codeBlocks);\n}\n","import { renderMarkdownToHtml } from \"../render\";\n\nexport interface MarkdownProps {\n source?: string | null;\n className?: string;\n}\n\nexport function Markdown({ source, className = \"ba-markdown\" }: MarkdownProps) {\n if (!source) {\n return null;\n }\n\n return (\n <div\n className={className}\n dangerouslySetInnerHTML={{ __html: renderMarkdownToHtml(source) }}\n />\n );\n}\n","import type { BlogPost as BlogPostType } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\nexport interface BlogPostProps {\n post: BlogPostType;\n showTitle?: boolean;\n showDate?: boolean;\n className?: string;\n renderContent?: (content: string) => React.ReactNode;\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPost({\n post,\n showTitle = true,\n showDate = true,\n className = \"ba-post\",\n renderContent,\n}: BlogPostProps) {\n const readingTime = post.metadata?.readingTimeMinutes;\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article className={className}>\n {showTitle && <h1 className=\"ba-post-title\">{post.title}</h1>}\n {(showDate || readingTime) && (\n <p className=\"ba-post-meta\">\n {showDate && (\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n )}\n {showDate && readingTime && <span className=\"ba-post-meta-sep\"> · </span>}\n {readingTime && (\n <span className=\"ba-post-reading-time\">{readingTime} min read</span>\n )}\n </p>\n )}\n {renderContent ? renderContent(post.content) : <Markdown source={post.content} />}\n </article>\n );\n}\n","import type { BlogPost } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype LinkComponent = React.ComponentType<any>;\n\nexport interface BlogPostListProps {\n posts: BlogPost[];\n title?: string;\n routePrefix?: string;\n linkComponent?: LinkComponent;\n className?: string;\n renderCard?: (post: BlogPost, href: string) => React.ReactNode;\n}\n\nfunction DefaultLink({\n href,\n className,\n children,\n}: {\n href: string;\n className?: string;\n children: React.ReactNode;\n}) {\n return (\n <a href={href} className={className}>\n {children}\n </a>\n );\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPostList({\n posts,\n title = \"Latest posts\",\n routePrefix = \"/blog\",\n linkComponent,\n className = \"ba-listing\",\n renderCard,\n}: BlogPostListProps) {\n const Link = linkComponent ?? DefaultLink;\n const prefix = routePrefix.replace(/\\/$/, \"\");\n\n return (\n <section className={className}>\n {title && <h1 className=\"ba-listing-title\">{title}</h1>}\n <div className=\"ba-posts\">\n {posts.map((post) => {\n const href = `${prefix}/${post.slug}`;\n\n if (renderCard) {\n return <div key={post.id}>{renderCard(post, href)}</div>;\n }\n\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article key={post.id} className=\"ba-post-card\">\n <h2 className=\"ba-post-card-title\">\n <Link href={href} className=\"ba-post-card-link\">\n {post.title}\n </Link>\n </h2>\n <p className=\"ba-post-card-meta\">\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n </p>\n <div className=\"ba-post-card-excerpt\">\n <Markdown source={post.excerpt ?? post.content.slice(0, 180)} />\n </div>\n </article>\n );\n })}\n </div>\n </section>\n );\n}\n"],"mappings":"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","names":["import_jsx_runtime","import_jsx_runtime","formatDate"]}
|
package/dist/react.d.cts
CHANGED
package/dist/react.d.ts
CHANGED
package/dist/react.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
// src/render.ts
|
|
2
|
-
var CODE_BLOCK_TOKEN = "
|
|
2
|
+
var CODE_BLOCK_TOKEN = "__AUTOBLOGWRITER_CODE_BLOCK_";
|
|
3
3
|
function escapeHtml(value) {
|
|
4
4
|
return value.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">").replace(/"/g, """).replace(/'/g, "'");
|
|
5
5
|
}
|
|
@@ -27,7 +27,7 @@ function replaceCodeBlocks(markdown, blocks) {
|
|
|
27
27
|
});
|
|
28
28
|
}
|
|
29
29
|
function restoreCodeBlocks(html, blocks) {
|
|
30
|
-
return html.replace(/
|
|
30
|
+
return html.replace(/__AUTOBLOGWRITER_CODE_BLOCK_(\d+)__/g, (_, rawIndex) => blocks[Number(rawIndex)] ?? "");
|
|
31
31
|
}
|
|
32
32
|
function renderMarkdownToHtml(markdown) {
|
|
33
33
|
if (!markdown) {
|
package/dist/react.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/render.ts","../src/react/Markdown.tsx","../src/react/BlogPost.tsx","../src/react/BlogPostList.tsx"],"sourcesContent":["const CODE_BLOCK_TOKEN = \"__BLOGAUTO_CODE_BLOCK_\";\n\nfunction escapeHtml(value: string): string {\n return value\n .replace(/&/g, \"&\")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\")\n .replace(/\"/g, \""\")\n .replace(/'/g, \"'\");\n}\n\nfunction renderInline(input: string): string {\n let output = escapeHtml(input);\n output = output.replace(/\\*\\*(.+?)\\*\\*/g, \"<strong>$1</strong>\");\n output = output.replace(/\\*(.+?)\\*/g, \"<em>$1</em>\");\n output = output.replace(/`([^`]+)`/g, (_, code) => `<code>${code}</code>`);\n output = output.replace(/!\\[([^\\]]*)\\]\\(([^)]+)\\)/g, (_, alt, src) => {\n const safeSrc = escapeHtml(src);\n const safeAlt = escapeHtml(alt);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n });\n output = output.replace(/\\[(.+?)\\]\\((.+?)\\)/g, (_, label, href) => {\n const safeHref = escapeHtml(href);\n return `<a href=\"${safeHref}\" target=\"_blank\" rel=\"noreferrer\">${label}</a>`;\n });\n output = output.replace(/\\n/g, \"<br />\");\n return output;\n}\n\nfunction replaceCodeBlocks(markdown: string, blocks: string[]): string {\n return markdown.replace(/```([\\s\\S]*?)```/g, (_, code) => {\n const index = blocks.push(`<pre><code>${escapeHtml(code.trim())}</code></pre>`) - 1;\n return `${CODE_BLOCK_TOKEN}${index}__`;\n });\n}\n\nfunction restoreCodeBlocks(html: string, blocks: string[]): string {\n return html.replace(/__BLOGAUTO_CODE_BLOCK_(\\d+)__/g, (_, rawIndex) => blocks[Number(rawIndex)] ?? \"\");\n}\n\nexport function renderMarkdownToHtml(markdown: string): string {\n if (!markdown) {\n return \"\";\n }\n\n const codeBlocks: string[] = [];\n const withoutCode = replaceCodeBlocks(markdown, codeBlocks);\n const normalized = withoutCode.replace(/\\r\\n/g, \"\\n\");\n // Insert blank lines around heading lines so they become separate blocks\n const withHeadingBreaks = normalized\n .replace(/\\n(#{1,6}\\s)/g, \"\\n\\n$1\")\n .replace(/(#{1,6}\\s[^\\n]+)\\n(?!#|\\n)/g, \"$1\\n\\n\");\n const blocks = withHeadingBreaks\n .split(/\\n{2,}/)\n .map((block) => block.trim())\n .filter(Boolean);\n\n const htmlBlocks = blocks.map((block) => {\n const headingMatch = block.match(/^(#{1,6})\\s+(.*)$/);\n if (headingMatch) {\n const level = headingMatch[1].length;\n const content = renderInline(headingMatch[2]);\n return `<h${level}>${content}</h${level}>`;\n }\n\n // Check if block is a standalone image\n const imageMatch = block.match(/^!\\[([^\\]]*)\\]\\(([^)]+)\\)$/);\n if (imageMatch) {\n const safeSrc = escapeHtml(imageMatch[2]);\n const safeAlt = escapeHtml(imageMatch[1]);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n }\n\n return `<p>${renderInline(block)}</p>`;\n });\n\n const html = htmlBlocks.join(\"\\n\");\n return restoreCodeBlocks(html, codeBlocks);\n}\n","import { renderMarkdownToHtml } from \"../render\";\n\nexport interface MarkdownProps {\n source?: string | null;\n className?: string;\n}\n\nexport function Markdown({ source, className = \"ba-markdown\" }: MarkdownProps) {\n if (!source) {\n return null;\n }\n\n return (\n <div\n className={className}\n dangerouslySetInnerHTML={{ __html: renderMarkdownToHtml(source) }}\n />\n );\n}\n","import type { BlogPost as BlogPostType } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\nexport interface BlogPostProps {\n post: BlogPostType;\n showTitle?: boolean;\n showDate?: boolean;\n className?: string;\n renderContent?: (content: string) => React.ReactNode;\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPost({\n post,\n showTitle = true,\n showDate = true,\n className = \"ba-post\",\n renderContent,\n}: BlogPostProps) {\n const readingTime = post.metadata?.readingTimeMinutes;\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article className={className}>\n {showTitle && <h1 className=\"ba-post-title\">{post.title}</h1>}\n {(showDate || readingTime) && (\n <p className=\"ba-post-meta\">\n {showDate && (\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n )}\n {showDate && readingTime && <span className=\"ba-post-meta-sep\"> · </span>}\n {readingTime && (\n <span className=\"ba-post-reading-time\">{readingTime} min read</span>\n )}\n </p>\n )}\n {renderContent ? renderContent(post.content) : <Markdown source={post.content} />}\n </article>\n );\n}\n","import type { BlogPost } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype LinkComponent = React.ComponentType<any>;\n\nexport interface BlogPostListProps {\n posts: BlogPost[];\n title?: string;\n routePrefix?: string;\n linkComponent?: LinkComponent;\n className?: string;\n renderCard?: (post: BlogPost, href: string) => React.ReactNode;\n}\n\nfunction DefaultLink({\n href,\n className,\n children,\n}: {\n href: string;\n className?: string;\n children: React.ReactNode;\n}) {\n return (\n <a href={href} className={className}>\n {children}\n </a>\n );\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPostList({\n posts,\n title = \"Latest posts\",\n routePrefix = \"/blog\",\n linkComponent,\n className = \"ba-listing\",\n renderCard,\n}: BlogPostListProps) {\n const Link = linkComponent ?? DefaultLink;\n const prefix = routePrefix.replace(/\\/$/, \"\");\n\n return (\n <section className={className}>\n {title && <h1 className=\"ba-listing-title\">{title}</h1>}\n <div className=\"ba-posts\">\n {posts.map((post) => {\n const href = `${prefix}/${post.slug}`;\n\n if (renderCard) {\n return <div key={post.id}>{renderCard(post, href)}</div>;\n }\n\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article key={post.id} className=\"ba-post-card\">\n <h2 className=\"ba-post-card-title\">\n <Link href={href} className=\"ba-post-card-link\">\n {post.title}\n </Link>\n </h2>\n <p className=\"ba-post-card-meta\">\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n </p>\n <div className=\"ba-post-card-excerpt\">\n <Markdown source={post.excerpt ?? post.content.slice(0, 180)} />\n </div>\n </article>\n );\n })}\n </div>\n </section>\n );\n}\n"],"mappings":"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","names":["jsx","jsx","jsxs","formatDate"]}
|
|
1
|
+
{"version":3,"sources":["../src/render.ts","../src/react/Markdown.tsx","../src/react/BlogPost.tsx","../src/react/BlogPostList.tsx"],"sourcesContent":["const CODE_BLOCK_TOKEN = \"__AUTOBLOGWRITER_CODE_BLOCK_\";\n\nfunction escapeHtml(value: string): string {\n return value\n .replace(/&/g, \"&\")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\")\n .replace(/\"/g, \""\")\n .replace(/'/g, \"'\");\n}\n\nfunction renderInline(input: string): string {\n let output = escapeHtml(input);\n output = output.replace(/\\*\\*(.+?)\\*\\*/g, \"<strong>$1</strong>\");\n output = output.replace(/\\*(.+?)\\*/g, \"<em>$1</em>\");\n output = output.replace(/`([^`]+)`/g, (_, code) => `<code>${code}</code>`);\n output = output.replace(/!\\[([^\\]]*)\\]\\(([^)]+)\\)/g, (_, alt, src) => {\n const safeSrc = escapeHtml(src);\n const safeAlt = escapeHtml(alt);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n });\n output = output.replace(/\\[(.+?)\\]\\((.+?)\\)/g, (_, label, href) => {\n const safeHref = escapeHtml(href);\n return `<a href=\"${safeHref}\" target=\"_blank\" rel=\"noreferrer\">${label}</a>`;\n });\n output = output.replace(/\\n/g, \"<br />\");\n return output;\n}\n\nfunction replaceCodeBlocks(markdown: string, blocks: string[]): string {\n return markdown.replace(/```([\\s\\S]*?)```/g, (_, code) => {\n const index = blocks.push(`<pre><code>${escapeHtml(code.trim())}</code></pre>`) - 1;\n return `${CODE_BLOCK_TOKEN}${index}__`;\n });\n}\n\nfunction restoreCodeBlocks(html: string, blocks: string[]): string {\n return html.replace(/__AUTOBLOGWRITER_CODE_BLOCK_(\\d+)__/g, (_, rawIndex) => blocks[Number(rawIndex)] ?? \"\");\n}\n\nexport function renderMarkdownToHtml(markdown: string): string {\n if (!markdown) {\n return \"\";\n }\n\n const codeBlocks: string[] = [];\n const withoutCode = replaceCodeBlocks(markdown, codeBlocks);\n const normalized = withoutCode.replace(/\\r\\n/g, \"\\n\");\n // Insert blank lines around heading lines so they become separate blocks\n const withHeadingBreaks = normalized\n .replace(/\\n(#{1,6}\\s)/g, \"\\n\\n$1\")\n .replace(/(#{1,6}\\s[^\\n]+)\\n(?!#|\\n)/g, \"$1\\n\\n\");\n const blocks = withHeadingBreaks\n .split(/\\n{2,}/)\n .map((block) => block.trim())\n .filter(Boolean);\n\n const htmlBlocks = blocks.map((block) => {\n const headingMatch = block.match(/^(#{1,6})\\s+(.*)$/);\n if (headingMatch) {\n const level = headingMatch[1].length;\n const content = renderInline(headingMatch[2]);\n return `<h${level}>${content}</h${level}>`;\n }\n\n // Check if block is a standalone image\n const imageMatch = block.match(/^!\\[([^\\]]*)\\]\\(([^)]+)\\)$/);\n if (imageMatch) {\n const safeSrc = escapeHtml(imageMatch[2]);\n const safeAlt = escapeHtml(imageMatch[1]);\n return `<img src=\"${safeSrc}\" alt=\"${safeAlt}\" />`;\n }\n\n return `<p>${renderInline(block)}</p>`;\n });\n\n const html = htmlBlocks.join(\"\\n\");\n return restoreCodeBlocks(html, codeBlocks);\n}\n","import { renderMarkdownToHtml } from \"../render\";\n\nexport interface MarkdownProps {\n source?: string | null;\n className?: string;\n}\n\nexport function Markdown({ source, className = \"ba-markdown\" }: MarkdownProps) {\n if (!source) {\n return null;\n }\n\n return (\n <div\n className={className}\n dangerouslySetInnerHTML={{ __html: renderMarkdownToHtml(source) }}\n />\n );\n}\n","import type { BlogPost as BlogPostType } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\nexport interface BlogPostProps {\n post: BlogPostType;\n showTitle?: boolean;\n showDate?: boolean;\n className?: string;\n renderContent?: (content: string) => React.ReactNode;\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPost({\n post,\n showTitle = true,\n showDate = true,\n className = \"ba-post\",\n renderContent,\n}: BlogPostProps) {\n const readingTime = post.metadata?.readingTimeMinutes;\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article className={className}>\n {showTitle && <h1 className=\"ba-post-title\">{post.title}</h1>}\n {(showDate || readingTime) && (\n <p className=\"ba-post-meta\">\n {showDate && (\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n )}\n {showDate && readingTime && <span className=\"ba-post-meta-sep\"> · </span>}\n {readingTime && (\n <span className=\"ba-post-reading-time\">{readingTime} min read</span>\n )}\n </p>\n )}\n {renderContent ? renderContent(post.content) : <Markdown source={post.content} />}\n </article>\n );\n}\n","import type { BlogPost } from \"../types\";\nimport { Markdown } from \"./Markdown\";\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype LinkComponent = React.ComponentType<any>;\n\nexport interface BlogPostListProps {\n posts: BlogPost[];\n title?: string;\n routePrefix?: string;\n linkComponent?: LinkComponent;\n className?: string;\n renderCard?: (post: BlogPost, href: string) => React.ReactNode;\n}\n\nfunction DefaultLink({\n href,\n className,\n children,\n}: {\n href: string;\n className?: string;\n children: React.ReactNode;\n}) {\n return (\n <a href={href} className={className}>\n {children}\n </a>\n );\n}\n\nfunction formatDate(dateStr: string): string {\n return new Date(dateStr).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n });\n}\n\nexport function BlogPostList({\n posts,\n title = \"Latest posts\",\n routePrefix = \"/blog\",\n linkComponent,\n className = \"ba-listing\",\n renderCard,\n}: BlogPostListProps) {\n const Link = linkComponent ?? DefaultLink;\n const prefix = routePrefix.replace(/\\/$/, \"\");\n\n return (\n <section className={className}>\n {title && <h1 className=\"ba-listing-title\">{title}</h1>}\n <div className=\"ba-posts\">\n {posts.map((post) => {\n const href = `${prefix}/${post.slug}`;\n\n if (renderCard) {\n return <div key={post.id}>{renderCard(post, href)}</div>;\n }\n\n const dateStr = post.publishedAt ?? post.updatedAt;\n\n return (\n <article key={post.id} className=\"ba-post-card\">\n <h2 className=\"ba-post-card-title\">\n <Link href={href} className=\"ba-post-card-link\">\n {post.title}\n </Link>\n </h2>\n <p className=\"ba-post-card-meta\">\n <time dateTime={dateStr}>{formatDate(dateStr)}</time>\n </p>\n <div className=\"ba-post-card-excerpt\">\n <Markdown source={post.excerpt ?? post.content.slice(0, 180)} />\n </div>\n </article>\n );\n })}\n </div>\n </section>\n );\n}\n"],"mappings":"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","names":["jsx","jsx","jsxs","formatDate"]}
|
package/dist/revalidate.cjs
CHANGED
|
@@ -87,10 +87,10 @@ function defaultPaths(payload) {
|
|
|
87
87
|
function defaultTags(payload) {
|
|
88
88
|
const tags = /* @__PURE__ */ new Set();
|
|
89
89
|
if (payload.workspaceSlug) {
|
|
90
|
-
tags.add(`
|
|
91
|
-
tags.add(`
|
|
90
|
+
tags.add(`autoblogwriter:${payload.workspaceSlug}:sitemap`);
|
|
91
|
+
tags.add(`autoblogwriter:${payload.workspaceSlug}:posts`);
|
|
92
92
|
if (payload.postSlug) {
|
|
93
|
-
tags.add(`
|
|
93
|
+
tags.add(`autoblogwriter:${payload.workspaceSlug}:post:${payload.postSlug}`);
|
|
94
94
|
}
|
|
95
95
|
}
|
|
96
96
|
return Array.from(tags);
|
|
@@ -119,8 +119,8 @@ function createRevalidateRouteHandler(options) {
|
|
|
119
119
|
const allowedSkewMs = Math.max(1, options.allowedSkewSeconds ?? 300) * 1e3;
|
|
120
120
|
const pathBuilder = options.revalidatePaths ?? defaultPaths;
|
|
121
121
|
const tagBuilder = options.revalidateTags ?? defaultTags;
|
|
122
|
-
return async function
|
|
123
|
-
const signature = request.headers.get("x-
|
|
122
|
+
return async function autoBlogWriterRevalidateHandler(request) {
|
|
123
|
+
const signature = request.headers.get("x-autoblogwriter-signature");
|
|
124
124
|
const receivedAt = Date.now();
|
|
125
125
|
const rawBody = await request.text();
|
|
126
126
|
const isValid = verifyWebhookSignature({
|
package/dist/revalidate.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/revalidate.ts"],"sourcesContent":["import crypto from \"crypto\";\nimport type {
|
|
1
|
+
{"version":3,"sources":["../src/revalidate.ts"],"sourcesContent":["import crypto from \"crypto\";\nimport type { AutoBlogWriterRevalidatePayload, RevalidatePathFn, RevalidateTagFn } from \"./types\";\n\nexport interface VerifyWebhookSignatureOptions {\n rawBody: string | Buffer;\n signature: string | null;\n secret: string;\n}\n\nexport interface CreateRevalidateRouteHandlerOptions {\n secret: string;\n allowedSkewSeconds?: number;\n revalidatePath?: RevalidatePathFn;\n revalidateTag?: RevalidateTagFn;\n revalidatePaths?: (payload: AutoBlogWriterRevalidatePayload) => string[];\n revalidateTags?: (payload: AutoBlogWriterRevalidatePayload) => string[];\n}\n\ninterface JsonResponseInit extends ResponseInit {\n status?: number;\n}\n\nfunction jsonResponse(body: unknown, init?: JsonResponseInit): Response {\n const headers = new Headers(init?.headers);\n headers.set(\"content-type\", \"application/json\");\n return new Response(JSON.stringify(body), {\n ...init,\n headers,\n });\n}\n\nfunction normalizeSignature(signature: string | null): Buffer | null {\n if (!signature) {\n return null;\n }\n const trimmed = signature.trim();\n if (!trimmed) {\n return null;\n }\n\n const withoutPrefix = trimmed.startsWith(\"sha256=\") ? trimmed.slice(7) : trimmed;\n if (!withoutPrefix) {\n return null;\n }\n\n try {\n return Buffer.from(withoutPrefix, \"hex\");\n } catch {\n return null;\n }\n}\n\nexport function verifyWebhookSignature(opts: VerifyWebhookSignatureOptions): boolean {\n const { rawBody, signature, secret } = opts;\n if (!secret) {\n throw new Error(\"Secret is required to verify webhook signatures\");\n }\n const providedSignature = normalizeSignature(signature);\n if (!providedSignature) {\n return false;\n }\n\n const bodyBuffer = typeof rawBody === \"string\" ? Buffer.from(rawBody) : rawBody;\n const expected = crypto.createHmac(\"sha256\", secret).update(bodyBuffer).digest();\n\n if (providedSignature.length !== expected.length) {\n return false;\n }\n\n return crypto.timingSafeEqual(providedSignature, expected);\n}\n\nfunction defaultPaths(payload: AutoBlogWriterRevalidatePayload): string[] {\n const paths = new Set<string>([\"/sitemap.xml\", \"/robots.txt\", \"/blog\"]);\n if (payload.postSlug) {\n paths.add(`/blog/${payload.postSlug}`);\n }\n return Array.from(paths);\n}\n\nfunction defaultTags(payload: AutoBlogWriterRevalidatePayload): string[] {\n const tags = new Set<string>();\n if (payload.workspaceSlug) {\n tags.add(`autoblogwriter:${payload.workspaceSlug}:sitemap`);\n tags.add(`autoblogwriter:${payload.workspaceSlug}:posts`);\n if (payload.postSlug) {\n tags.add(`autoblogwriter:${payload.workspaceSlug}:post:${payload.postSlug}`);\n }\n }\n return Array.from(tags);\n}\n\nfunction dedupe(values: string[] | undefined): string[] {\n if (!values || values.length === 0) {\n return [];\n }\n return Array.from(\n new Set(\n values\n .filter((value) => typeof value === \"string\" && value.trim().length > 0)\n .map((value) => value.trim()),\n ),\n );\n}\n\nfunction determineRouteType(path: string): \"route\" | \"page\" {\n const lower = path.toLowerCase();\n if (lower.endsWith(\".xml\") || lower.endsWith(\".txt\")) {\n return \"route\";\n }\n return \"page\";\n}\n\nexport function createRevalidateRouteHandler(\n options: CreateRevalidateRouteHandlerOptions,\n): (request: Request) => Promise<Response> {\n if (!options.secret) {\n throw new Error(\"secret is required for createRevalidateRouteHandler\");\n }\n\n const allowedSkewMs = Math.max(1, options.allowedSkewSeconds ?? 300) * 1000;\n const pathBuilder = options.revalidatePaths ?? defaultPaths;\n const tagBuilder = options.revalidateTags ?? defaultTags;\n\n return async function autoBlogWriterRevalidateHandler(request: Request): Promise<Response> {\n const signature = request.headers.get(\"x-autoblogwriter-signature\");\n const receivedAt = Date.now();\n\n const rawBody = await request.text();\n const isValid = verifyWebhookSignature({\n rawBody,\n signature,\n secret: options.secret,\n });\n\n if (!isValid) {\n return jsonResponse({ error: \"Invalid webhook signature\" }, { status: 401 });\n }\n\n let payload: AutoBlogWriterRevalidatePayload;\n try {\n payload = JSON.parse(rawBody) as AutoBlogWriterRevalidatePayload;\n } catch {\n return jsonResponse({ error: \"Invalid JSON payload\" }, { status: 400 });\n }\n\n if (!payload.workspaceSlug || typeof payload.workspaceSlug !== \"string\") {\n return jsonResponse({ error: \"workspaceSlug is required\" }, { status: 400 });\n }\n\n if (!payload.ts || typeof payload.ts !== \"string\") {\n return jsonResponse({ error: \"Timestamp is required\" }, { status: 400 });\n }\n\n const payloadTs = Date.parse(payload.ts);\n if (Number.isNaN(payloadTs)) {\n return jsonResponse({ error: \"Invalid timestamp\" }, { status: 400 });\n }\n\n if (Math.abs(receivedAt - payloadTs) > allowedSkewMs) {\n return jsonResponse({ error: \"Webhook timestamp is outside allowed skew\" }, { status: 409 });\n }\n\n const paths = dedupe(pathBuilder(payload));\n const tags = dedupe(tagBuilder(payload));\n\n try {\n if (options.revalidatePath && paths.length > 0) {\n await Promise.all(\n paths.map((path) => options.revalidatePath!(path, determineRouteType(path))),\n );\n }\n\n if (options.revalidateTag && tags.length > 0) {\n await Promise.all(tags.map((tag) => options.revalidateTag!(tag)));\n }\n } catch (error) {\n return jsonResponse(\n { error: \"Failed to revalidate cache\", details: (error as Error).message },\n { status: 500 },\n );\n }\n\n return jsonResponse({\n ok: true,\n event: payload.event ?? \"post.published\",\n revalidated: {\n paths,\n tags,\n },\n });\n };\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,oBAAmB;AAsBnB,SAAS,aAAa,MAAe,MAAmC;AACtE,QAAM,UAAU,IAAI,QAAQ,MAAM,OAAO;AACzC,UAAQ,IAAI,gBAAgB,kBAAkB;AAC9C,SAAO,IAAI,SAAS,KAAK,UAAU,IAAI,GAAG;AAAA,IACxC,GAAG;AAAA,IACH;AAAA,EACF,CAAC;AACH;AAEA,SAAS,mBAAmB,WAAyC;AACnE,MAAI,CAAC,WAAW;AACd,WAAO;AAAA,EACT;AACA,QAAM,UAAU,UAAU,KAAK;AAC/B,MAAI,CAAC,SAAS;AACZ,WAAO;AAAA,EACT;AAEA,QAAM,gBAAgB,QAAQ,WAAW,SAAS,IAAI,QAAQ,MAAM,CAAC,IAAI;AACzE,MAAI,CAAC,eAAe;AAClB,WAAO;AAAA,EACT;AAEA,MAAI;AACF,WAAO,OAAO,KAAK,eAAe,KAAK;AAAA,EACzC,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEO,SAAS,uBAAuB,MAA8C;AACnF,QAAM,EAAE,SAAS,WAAW,OAAO,IAAI;AACvC,MAAI,CAAC,QAAQ;AACX,UAAM,IAAI,MAAM,iDAAiD;AAAA,EACnE;AACA,QAAM,oBAAoB,mBAAmB,SAAS;AACtD,MAAI,CAAC,mBAAmB;AACtB,WAAO;AAAA,EACT;AAEA,QAAM,aAAa,OAAO,YAAY,WAAW,OAAO,KAAK,OAAO,IAAI;AACxE,QAAM,WAAW,cAAAA,QAAO,WAAW,UAAU,MAAM,EAAE,OAAO,UAAU,EAAE,OAAO;AAE/E,MAAI,kBAAkB,WAAW,SAAS,QAAQ;AAChD,WAAO;AAAA,EACT;AAEA,SAAO,cAAAA,QAAO,gBAAgB,mBAAmB,QAAQ;AAC3D;AAEA,SAAS,aAAa,SAAoD;AACxE,QAAM,QAAQ,oBAAI,IAAY,CAAC,gBAAgB,eAAe,OAAO,CAAC;AACtE,MAAI,QAAQ,UAAU;AACpB,UAAM,IAAI,SAAS,QAAQ,QAAQ,EAAE;AAAA,EACvC;AACA,SAAO,MAAM,KAAK,KAAK;AACzB;AAEA,SAAS,YAAY,SAAoD;AACvE,QAAM,OAAO,oBAAI,IAAY;AAC7B,MAAI,QAAQ,eAAe;AACzB,SAAK,IAAI,kBAAkB,QAAQ,aAAa,UAAU;AAC1D,SAAK,IAAI,kBAAkB,QAAQ,aAAa,QAAQ;AACxD,QAAI,QAAQ,UAAU;AACpB,WAAK,IAAI,kBAAkB,QAAQ,aAAa,SAAS,QAAQ,QAAQ,EAAE;AAAA,IAC7E;AAAA,EACF;AACA,SAAO,MAAM,KAAK,IAAI;AACxB;AAEA,SAAS,OAAO,QAAwC;AACtD,MAAI,CAAC,UAAU,OAAO,WAAW,GAAG;AAClC,WAAO,CAAC;AAAA,EACV;AACA,SAAO,MAAM;AAAA,IACX,IAAI;AAAA,MACF,OACG,OAAO,CAAC,UAAU,OAAO,UAAU,YAAY,MAAM,KAAK,EAAE,SAAS,CAAC,EACtE,IAAI,CAAC,UAAU,MAAM,KAAK,CAAC;AAAA,IAChC;AAAA,EACF;AACF;AAEA,SAAS,mBAAmB,MAAgC;AAC1D,QAAM,QAAQ,KAAK,YAAY;AAC/B,MAAI,MAAM,SAAS,MAAM,KAAK,MAAM,SAAS,MAAM,GAAG;AACpD,WAAO;AAAA,EACT;AACA,SAAO;AACT;AAEO,SAAS,6BACd,SACyC;AACzC,MAAI,CAAC,QAAQ,QAAQ;AACnB,UAAM,IAAI,MAAM,qDAAqD;AAAA,EACvE;AAEA,QAAM,gBAAgB,KAAK,IAAI,GAAG,QAAQ,sBAAsB,GAAG,IAAI;AACvE,QAAM,cAAc,QAAQ,mBAAmB;AAC/C,QAAM,aAAa,QAAQ,kBAAkB;AAE7C,SAAO,eAAe,gCAAgC,SAAqC;AACzF,UAAM,YAAY,QAAQ,QAAQ,IAAI,4BAA4B;AAClE,UAAM,aAAa,KAAK,IAAI;AAE5B,UAAM,UAAU,MAAM,QAAQ,KAAK;AACnC,UAAM,UAAU,uBAAuB;AAAA,MACrC;AAAA,MACA;AAAA,MACA,QAAQ,QAAQ;AAAA,IAClB,CAAC;AAED,QAAI,CAAC,SAAS;AACZ,aAAO,aAAa,EAAE,OAAO,4BAA4B,GAAG,EAAE,QAAQ,IAAI,CAAC;AAAA,IAC7E;AAEA,QAAI;AACJ,QAAI;AACF,gBAAU,KAAK,MAAM,OAAO;AAAA,IAC9B,QAAQ;AACN,aAAO,aAAa,EAAE,OAAO,uBAAuB,GAAG,EAAE,QAAQ,IAAI,CAAC;AAAA,IACxE;AAEA,QAAI,CAAC,QAAQ,iBAAiB,OAAO,QAAQ,kBAAkB,UAAU;AACvE,aAAO,aAAa,EAAE,OAAO,4BAA4B,GAAG,EAAE,QAAQ,IAAI,CAAC;AAAA,IAC7E;AAEA,QAAI,CAAC,QAAQ,MAAM,OAAO,QAAQ,OAAO,UAAU;AACjD,aAAO,aAAa,EAAE,OAAO,wBAAwB,GAAG,EAAE,QAAQ,IAAI,CAAC;AAAA,IACzE;AAEA,UAAM,YAAY,KAAK,MAAM,QAAQ,EAAE;AACvC,QAAI,OAAO,MAAM,SAAS,GAAG;AAC3B,aAAO,aAAa,EAAE,OAAO,oBAAoB,GAAG,EAAE,QAAQ,IAAI,CAAC;AAAA,IACrE;AAEA,QAAI,KAAK,IAAI,aAAa,SAAS,IAAI,eAAe;AACpD,aAAO,aAAa,EAAE,OAAO,4CAA4C,GAAG,EAAE,QAAQ,IAAI,CAAC;AAAA,IAC7F;AAEA,UAAM,QAAQ,OAAO,YAAY,OAAO,CAAC;AACzC,UAAM,OAAO,OAAO,WAAW,OAAO,CAAC;AAEvC,QAAI;AACF,UAAI,QAAQ,kBAAkB,MAAM,SAAS,GAAG;AAC9C,cAAM,QAAQ;AAAA,UACZ,MAAM,IAAI,CAAC,SAAS,QAAQ,eAAgB,MAAM,mBAAmB,IAAI,CAAC,CAAC;AAAA,QAC7E;AAAA,MACF;AAEA,UAAI,QAAQ,iBAAiB,KAAK,SAAS,GAAG;AAC5C,cAAM,QAAQ,IAAI,KAAK,IAAI,CAAC,QAAQ,QAAQ,cAAe,GAAG,CAAC,CAAC;AAAA,MAClE;AAAA,IACF,SAAS,OAAO;AACd,aAAO;AAAA,QACL,EAAE,OAAO,8BAA8B,SAAU,MAAgB,QAAQ;AAAA,QACzE,EAAE,QAAQ,IAAI;AAAA,MAChB;AAAA,IACF;AAEA,WAAO,aAAa;AAAA,MAClB,IAAI;AAAA,MACJ,OAAO,QAAQ,SAAS;AAAA,MACxB,aAAa;AAAA,QACX;AAAA,QACA;AAAA,MACF;AAAA,IACF,CAAC;AAAA,EACH;AACF;","names":["crypto"]}
|
package/dist/revalidate.d.cts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { R as RevalidatePathFn, g as RevalidateTagFn, f as
|
|
1
|
+
import { R as RevalidatePathFn, g as RevalidateTagFn, f as AutoBlogWriterRevalidatePayload } from './types-ClZVNiGd.cjs';
|
|
2
2
|
|
|
3
3
|
interface VerifyWebhookSignatureOptions {
|
|
4
4
|
rawBody: string | Buffer;
|
|
@@ -10,8 +10,8 @@ interface CreateRevalidateRouteHandlerOptions {
|
|
|
10
10
|
allowedSkewSeconds?: number;
|
|
11
11
|
revalidatePath?: RevalidatePathFn;
|
|
12
12
|
revalidateTag?: RevalidateTagFn;
|
|
13
|
-
revalidatePaths?: (payload:
|
|
14
|
-
revalidateTags?: (payload:
|
|
13
|
+
revalidatePaths?: (payload: AutoBlogWriterRevalidatePayload) => string[];
|
|
14
|
+
revalidateTags?: (payload: AutoBlogWriterRevalidatePayload) => string[];
|
|
15
15
|
}
|
|
16
16
|
declare function verifyWebhookSignature(opts: VerifyWebhookSignatureOptions): boolean;
|
|
17
17
|
declare function createRevalidateRouteHandler(options: CreateRevalidateRouteHandlerOptions): (request: Request) => Promise<Response>;
|
package/dist/revalidate.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { R as RevalidatePathFn, g as RevalidateTagFn, f as
|
|
1
|
+
import { R as RevalidatePathFn, g as RevalidateTagFn, f as AutoBlogWriterRevalidatePayload } from './types-ClZVNiGd.js';
|
|
2
2
|
|
|
3
3
|
interface VerifyWebhookSignatureOptions {
|
|
4
4
|
rawBody: string | Buffer;
|
|
@@ -10,8 +10,8 @@ interface CreateRevalidateRouteHandlerOptions {
|
|
|
10
10
|
allowedSkewSeconds?: number;
|
|
11
11
|
revalidatePath?: RevalidatePathFn;
|
|
12
12
|
revalidateTag?: RevalidateTagFn;
|
|
13
|
-
revalidatePaths?: (payload:
|
|
14
|
-
revalidateTags?: (payload:
|
|
13
|
+
revalidatePaths?: (payload: AutoBlogWriterRevalidatePayload) => string[];
|
|
14
|
+
revalidateTags?: (payload: AutoBlogWriterRevalidatePayload) => string[];
|
|
15
15
|
}
|
|
16
16
|
declare function verifyWebhookSignature(opts: VerifyWebhookSignatureOptions): boolean;
|
|
17
17
|
declare function createRevalidateRouteHandler(options: CreateRevalidateRouteHandlerOptions): (request: Request) => Promise<Response>;
|
package/dist/revalidate.js
CHANGED
|
@@ -52,10 +52,10 @@ function defaultPaths(payload) {
|
|
|
52
52
|
function defaultTags(payload) {
|
|
53
53
|
const tags = /* @__PURE__ */ new Set();
|
|
54
54
|
if (payload.workspaceSlug) {
|
|
55
|
-
tags.add(`
|
|
56
|
-
tags.add(`
|
|
55
|
+
tags.add(`autoblogwriter:${payload.workspaceSlug}:sitemap`);
|
|
56
|
+
tags.add(`autoblogwriter:${payload.workspaceSlug}:posts`);
|
|
57
57
|
if (payload.postSlug) {
|
|
58
|
-
tags.add(`
|
|
58
|
+
tags.add(`autoblogwriter:${payload.workspaceSlug}:post:${payload.postSlug}`);
|
|
59
59
|
}
|
|
60
60
|
}
|
|
61
61
|
return Array.from(tags);
|
|
@@ -84,8 +84,8 @@ function createRevalidateRouteHandler(options) {
|
|
|
84
84
|
const allowedSkewMs = Math.max(1, options.allowedSkewSeconds ?? 300) * 1e3;
|
|
85
85
|
const pathBuilder = options.revalidatePaths ?? defaultPaths;
|
|
86
86
|
const tagBuilder = options.revalidateTags ?? defaultTags;
|
|
87
|
-
return async function
|
|
88
|
-
const signature = request.headers.get("x-
|
|
87
|
+
return async function autoBlogWriterRevalidateHandler(request) {
|
|
88
|
+
const signature = request.headers.get("x-autoblogwriter-signature");
|
|
89
89
|
const receivedAt = Date.now();
|
|
90
90
|
const rawBody = await request.text();
|
|
91
91
|
const isValid = verifyWebhookSignature({
|
package/dist/revalidate.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/revalidate.ts"],"sourcesContent":["import crypto from \"crypto\";\nimport type {
|
|
1
|
+
{"version":3,"sources":["../src/revalidate.ts"],"sourcesContent":["import crypto from \"crypto\";\nimport type { AutoBlogWriterRevalidatePayload, RevalidatePathFn, RevalidateTagFn } from \"./types\";\n\nexport interface VerifyWebhookSignatureOptions {\n rawBody: string | Buffer;\n signature: string | null;\n secret: string;\n}\n\nexport interface CreateRevalidateRouteHandlerOptions {\n secret: string;\n allowedSkewSeconds?: number;\n revalidatePath?: RevalidatePathFn;\n revalidateTag?: RevalidateTagFn;\n revalidatePaths?: (payload: AutoBlogWriterRevalidatePayload) => string[];\n revalidateTags?: (payload: AutoBlogWriterRevalidatePayload) => string[];\n}\n\ninterface JsonResponseInit extends ResponseInit {\n status?: number;\n}\n\nfunction jsonResponse(body: unknown, init?: JsonResponseInit): Response {\n const headers = new Headers(init?.headers);\n headers.set(\"content-type\", \"application/json\");\n return new Response(JSON.stringify(body), {\n ...init,\n headers,\n });\n}\n\nfunction normalizeSignature(signature: string | null): Buffer | null {\n if (!signature) {\n return null;\n }\n const trimmed = signature.trim();\n if (!trimmed) {\n return null;\n }\n\n const withoutPrefix = trimmed.startsWith(\"sha256=\") ? trimmed.slice(7) : trimmed;\n if (!withoutPrefix) {\n return null;\n }\n\n try {\n return Buffer.from(withoutPrefix, \"hex\");\n } catch {\n return null;\n }\n}\n\nexport function verifyWebhookSignature(opts: VerifyWebhookSignatureOptions): boolean {\n const { rawBody, signature, secret } = opts;\n if (!secret) {\n throw new Error(\"Secret is required to verify webhook signatures\");\n }\n const providedSignature = normalizeSignature(signature);\n if (!providedSignature) {\n return false;\n }\n\n const bodyBuffer = typeof rawBody === \"string\" ? Buffer.from(rawBody) : rawBody;\n const expected = crypto.createHmac(\"sha256\", secret).update(bodyBuffer).digest();\n\n if (providedSignature.length !== expected.length) {\n return false;\n }\n\n return crypto.timingSafeEqual(providedSignature, expected);\n}\n\nfunction defaultPaths(payload: AutoBlogWriterRevalidatePayload): string[] {\n const paths = new Set<string>([\"/sitemap.xml\", \"/robots.txt\", \"/blog\"]);\n if (payload.postSlug) {\n paths.add(`/blog/${payload.postSlug}`);\n }\n return Array.from(paths);\n}\n\nfunction defaultTags(payload: AutoBlogWriterRevalidatePayload): string[] {\n const tags = new Set<string>();\n if (payload.workspaceSlug) {\n tags.add(`autoblogwriter:${payload.workspaceSlug}:sitemap`);\n tags.add(`autoblogwriter:${payload.workspaceSlug}:posts`);\n if (payload.postSlug) {\n tags.add(`autoblogwriter:${payload.workspaceSlug}:post:${payload.postSlug}`);\n }\n }\n return Array.from(tags);\n}\n\nfunction dedupe(values: string[] | undefined): string[] {\n if (!values || values.length === 0) {\n return [];\n }\n return Array.from(\n new Set(\n values\n .filter((value) => typeof value === \"string\" && value.trim().length > 0)\n .map((value) => value.trim()),\n ),\n );\n}\n\nfunction determineRouteType(path: string): \"route\" | \"page\" {\n const lower = path.toLowerCase();\n if (lower.endsWith(\".xml\") || lower.endsWith(\".txt\")) {\n return \"route\";\n }\n return \"page\";\n}\n\nexport function createRevalidateRouteHandler(\n options: CreateRevalidateRouteHandlerOptions,\n): (request: Request) => Promise<Response> {\n if (!options.secret) {\n throw new Error(\"secret is required for createRevalidateRouteHandler\");\n }\n\n const allowedSkewMs = Math.max(1, options.allowedSkewSeconds ?? 300) * 1000;\n const pathBuilder = options.revalidatePaths ?? defaultPaths;\n const tagBuilder = options.revalidateTags ?? defaultTags;\n\n return async function autoBlogWriterRevalidateHandler(request: Request): Promise<Response> {\n const signature = request.headers.get(\"x-autoblogwriter-signature\");\n const receivedAt = Date.now();\n\n const rawBody = await request.text();\n const isValid = verifyWebhookSignature({\n rawBody,\n signature,\n secret: options.secret,\n });\n\n if (!isValid) {\n return jsonResponse({ error: \"Invalid webhook signature\" }, { status: 401 });\n }\n\n let payload: AutoBlogWriterRevalidatePayload;\n try {\n payload = JSON.parse(rawBody) as AutoBlogWriterRevalidatePayload;\n } catch {\n return jsonResponse({ error: \"Invalid JSON payload\" }, { status: 400 });\n }\n\n if (!payload.workspaceSlug || typeof payload.workspaceSlug !== \"string\") {\n return jsonResponse({ error: \"workspaceSlug is required\" }, { status: 400 });\n }\n\n if (!payload.ts || typeof payload.ts !== \"string\") {\n return jsonResponse({ error: \"Timestamp is required\" }, { status: 400 });\n }\n\n const payloadTs = Date.parse(payload.ts);\n if (Number.isNaN(payloadTs)) {\n return jsonResponse({ error: \"Invalid timestamp\" }, { status: 400 });\n }\n\n if (Math.abs(receivedAt - payloadTs) > allowedSkewMs) {\n return jsonResponse({ error: \"Webhook timestamp is outside allowed skew\" }, { status: 409 });\n }\n\n const paths = dedupe(pathBuilder(payload));\n const tags = dedupe(tagBuilder(payload));\n\n try {\n if (options.revalidatePath && paths.length > 0) {\n await Promise.all(\n paths.map((path) => options.revalidatePath!(path, determineRouteType(path))),\n );\n }\n\n if (options.revalidateTag && tags.length > 0) {\n await Promise.all(tags.map((tag) => options.revalidateTag!(tag)));\n }\n } catch (error) {\n return jsonResponse(\n { error: \"Failed to revalidate cache\", details: (error as Error).message },\n { status: 500 },\n );\n }\n\n return jsonResponse({\n ok: true,\n event: payload.event ?? \"post.published\",\n revalidated: {\n paths,\n tags,\n },\n });\n };\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
type AuthMode = "bearer" | "x-api-key";
|
|
2
|
-
interface
|
|
2
|
+
interface AutoBlogWriterClientConfig {
|
|
3
3
|
apiUrl: string;
|
|
4
4
|
apiKey: string;
|
|
5
5
|
workspaceId?: string;
|
|
@@ -9,7 +9,7 @@ interface BlogAutoClientConfig {
|
|
|
9
9
|
headers?: Record<string, string>;
|
|
10
10
|
timeoutMs?: number;
|
|
11
11
|
}
|
|
12
|
-
interface InternalClientConfig extends Required<Pick<
|
|
12
|
+
interface InternalClientConfig extends Required<Pick<AutoBlogWriterClientConfig, "apiUrl" | "apiKey">> {
|
|
13
13
|
workspaceId?: string;
|
|
14
14
|
workspaceSlug?: string;
|
|
15
15
|
authMode: AuthMode;
|
|
@@ -79,7 +79,7 @@ interface BuildRobotsOptions {
|
|
|
79
79
|
siteUrl: string;
|
|
80
80
|
sitemapPath?: string;
|
|
81
81
|
}
|
|
82
|
-
interface
|
|
82
|
+
interface AutoBlogWriterRevalidatePayload {
|
|
83
83
|
workspaceSlug: string;
|
|
84
84
|
postSlug?: string | null;
|
|
85
85
|
event: string;
|
|
@@ -97,4 +97,4 @@ interface FetchRequestOptions {
|
|
|
97
97
|
next?: FetchNextConfig;
|
|
98
98
|
}
|
|
99
99
|
|
|
100
|
-
export type {
|
|
100
|
+
export type { AutoBlogWriterClientConfig as A, BlogPost as B, FetchRequestOptions as F, InternalClientConfig as I, MetadataRouteSitemap as M, PostsResponse as P, RevalidatePathFn as R, SitemapEntry as S, BuildSitemapOptions as a, BuildRobotsOptions as b, MetadataRouteRobots as c, AuthMode as d, PaginatedList as e, AutoBlogWriterRevalidatePayload as f, RevalidateTagFn as g, FetchNextConfig as h };
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
type AuthMode = "bearer" | "x-api-key";
|
|
2
|
-
interface
|
|
2
|
+
interface AutoBlogWriterClientConfig {
|
|
3
3
|
apiUrl: string;
|
|
4
4
|
apiKey: string;
|
|
5
5
|
workspaceId?: string;
|
|
@@ -9,7 +9,7 @@ interface BlogAutoClientConfig {
|
|
|
9
9
|
headers?: Record<string, string>;
|
|
10
10
|
timeoutMs?: number;
|
|
11
11
|
}
|
|
12
|
-
interface InternalClientConfig extends Required<Pick<
|
|
12
|
+
interface InternalClientConfig extends Required<Pick<AutoBlogWriterClientConfig, "apiUrl" | "apiKey">> {
|
|
13
13
|
workspaceId?: string;
|
|
14
14
|
workspaceSlug?: string;
|
|
15
15
|
authMode: AuthMode;
|
|
@@ -79,7 +79,7 @@ interface BuildRobotsOptions {
|
|
|
79
79
|
siteUrl: string;
|
|
80
80
|
sitemapPath?: string;
|
|
81
81
|
}
|
|
82
|
-
interface
|
|
82
|
+
interface AutoBlogWriterRevalidatePayload {
|
|
83
83
|
workspaceSlug: string;
|
|
84
84
|
postSlug?: string | null;
|
|
85
85
|
event: string;
|
|
@@ -97,4 +97,4 @@ interface FetchRequestOptions {
|
|
|
97
97
|
next?: FetchNextConfig;
|
|
98
98
|
}
|
|
99
99
|
|
|
100
|
-
export type {
|
|
100
|
+
export type { AutoBlogWriterClientConfig as A, BlogPost as B, FetchRequestOptions as F, InternalClientConfig as I, MetadataRouteSitemap as M, PostsResponse as P, RevalidatePathFn as R, SitemapEntry as S, BuildSitemapOptions as a, BuildRobotsOptions as b, MetadataRouteRobots as c, AuthMode as d, PaginatedList as e, AutoBlogWriterRevalidatePayload as f, RevalidateTagFn as g, FetchNextConfig as h };
|
package/package.json
CHANGED