@autoblogwriter/sdk 3.0.3 → 3.0.5
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/dist/index.cjs +142 -16
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +3 -3
- package/dist/index.d.ts +3 -3
- package/dist/index.js +142 -16
- package/dist/index.js.map +1 -1
- package/dist/{metadata-CE4bus6R.d.cts → metadata-DaBB2mdh.d.cts} +1 -1
- package/dist/{metadata-DprVkArj.d.ts → metadata-Du-JkBih.d.ts} +1 -1
- package/dist/next.d.cts +2 -2
- package/dist/next.d.ts +2 -2
- package/dist/react.cjs +142 -16
- 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 +142 -16
- package/dist/react.js.map +1 -1
- package/dist/revalidate.d.cts +1 -1
- package/dist/revalidate.d.ts +1 -1
- package/dist/{types-HIhu-PBU.d.cts → types-DuDmhtM7.d.cts} +12 -0
- package/dist/{types-HIhu-PBU.d.ts → types-DuDmhtM7.d.ts} +12 -0
- package/package.json +1 -1
package/dist/index.cjs
CHANGED
|
@@ -391,7 +391,11 @@ function escapeHtml(value) {
|
|
|
391
391
|
function renderInline(input) {
|
|
392
392
|
let output = escapeHtml(input);
|
|
393
393
|
output = output.replace(/\*\*(.+?)\*\*/g, "<strong>$1</strong>");
|
|
394
|
+
output = output.replace(/__(.+?)__/g, "<strong>$1</strong>");
|
|
394
395
|
output = output.replace(/\*(.+?)\*/g, "<em>$1</em>");
|
|
396
|
+
output = output.replace(/(?<!\w)_(.+?)_(?!\w)/g, "<em>$1</em>");
|
|
397
|
+
output = output.replace(/~~(.+?)~~/g, "<s>$1</s>");
|
|
398
|
+
output = output.replace(/\+\+(.+?)\+\+/g, "<u>$1</u>");
|
|
395
399
|
output = output.replace(/`([^`]+)`/g, (_, code) => `<code>${code}</code>`);
|
|
396
400
|
output = output.replace(/!\[([^\]]*)\]\(([^)]+)\)/g, (_, alt, src) => {
|
|
397
401
|
const safeSrc = escapeHtml(src);
|
|
@@ -406,14 +410,97 @@ function renderInline(input) {
|
|
|
406
410
|
return output;
|
|
407
411
|
}
|
|
408
412
|
function replaceCodeBlocks(markdown, blocks) {
|
|
409
|
-
return markdown.replace(/```([\s\S]*?)```/g, (_, code) => {
|
|
410
|
-
const
|
|
413
|
+
return markdown.replace(/```(\w*)\n?([\s\S]*?)```/g, (_, lang, code) => {
|
|
414
|
+
const langAttr = lang ? ` class="language-${escapeHtml(lang)}"` : "";
|
|
415
|
+
const index = blocks.push(`<pre><code${langAttr}>${escapeHtml(code.trim())}</code></pre>`) - 1;
|
|
411
416
|
return `${CODE_BLOCK_TOKEN}${index}__`;
|
|
412
417
|
});
|
|
413
418
|
}
|
|
414
419
|
function restoreCodeBlocks(html, blocks) {
|
|
415
420
|
return html.replace(/__AUTOBLOGWRITER_CODE_BLOCK_(\d+)__/g, (_, rawIndex) => blocks[Number(rawIndex)] ?? "");
|
|
416
421
|
}
|
|
422
|
+
function isUnorderedListItem(line) {
|
|
423
|
+
return /^[\t ]*[-*+] /.test(line);
|
|
424
|
+
}
|
|
425
|
+
function isOrderedListItem(line) {
|
|
426
|
+
return /^[\t ]*\d+\. /.test(line);
|
|
427
|
+
}
|
|
428
|
+
function getListItemContent(line) {
|
|
429
|
+
return line.replace(/^[\t ]*(?:[-*+]|\d+\.) /, "");
|
|
430
|
+
}
|
|
431
|
+
function getIndentLevel(line) {
|
|
432
|
+
const match = line.match(/^([\t ]*)/);
|
|
433
|
+
if (!match) return 0;
|
|
434
|
+
const indent = match[1];
|
|
435
|
+
return indent.replace(/\t/g, " ").length;
|
|
436
|
+
}
|
|
437
|
+
function parseListItems(lines, baseIndent) {
|
|
438
|
+
const items = [];
|
|
439
|
+
let i = 0;
|
|
440
|
+
while (i < lines.length) {
|
|
441
|
+
const indent = getIndentLevel(lines[i]);
|
|
442
|
+
if (indent > baseIndent && items.length > 0) {
|
|
443
|
+
const nestedLines = [];
|
|
444
|
+
while (i < lines.length && getIndentLevel(lines[i]) > baseIndent) {
|
|
445
|
+
nestedLines.push(lines[i]);
|
|
446
|
+
i++;
|
|
447
|
+
}
|
|
448
|
+
items[items.length - 1].children = parseListItems(nestedLines, baseIndent + 2);
|
|
449
|
+
} else {
|
|
450
|
+
items.push({ content: getListItemContent(lines[i]), children: [] });
|
|
451
|
+
i++;
|
|
452
|
+
}
|
|
453
|
+
}
|
|
454
|
+
return items;
|
|
455
|
+
}
|
|
456
|
+
function renderListItems(items, tag) {
|
|
457
|
+
const lis = items.map((item) => {
|
|
458
|
+
let li = `<li>${renderInline(item.content)}`;
|
|
459
|
+
if (item.children.length > 0) {
|
|
460
|
+
li += renderListItems(item.children, tag);
|
|
461
|
+
}
|
|
462
|
+
li += "</li>";
|
|
463
|
+
return li;
|
|
464
|
+
});
|
|
465
|
+
return `<${tag}>${lis.join("")}</${tag}>`;
|
|
466
|
+
}
|
|
467
|
+
function renderBlock(block) {
|
|
468
|
+
const headingMatch = block.match(/^(#{1,6})\s+(.*)$/);
|
|
469
|
+
if (headingMatch) {
|
|
470
|
+
const level = headingMatch[1].length;
|
|
471
|
+
const content = renderInline(headingMatch[2]);
|
|
472
|
+
return `<h${level}>${content}</h${level}>`;
|
|
473
|
+
}
|
|
474
|
+
if (/^(?:---+|\*\*\*+|___+)$/.test(block.trim())) {
|
|
475
|
+
return "<hr />";
|
|
476
|
+
}
|
|
477
|
+
const imageMatch = block.match(/^!\[([^\]]*)\]\(([^)]+)\)$/);
|
|
478
|
+
if (imageMatch) {
|
|
479
|
+
const safeSrc = escapeHtml(imageMatch[2]);
|
|
480
|
+
const safeAlt = escapeHtml(imageMatch[1]);
|
|
481
|
+
return `<img src="${safeSrc}" alt="${safeAlt}" />`;
|
|
482
|
+
}
|
|
483
|
+
const lines = block.split("\n");
|
|
484
|
+
if (lines.every((l) => l.startsWith("> ") || l === ">")) {
|
|
485
|
+
const inner = lines.map((l) => l.replace(/^>\s?/, "")).join("\n");
|
|
486
|
+
const innerHtml = inner.split(/\n{2,}/).map((b) => b.trim()).filter(Boolean).map(renderBlock).join("\n");
|
|
487
|
+
return `<blockquote>${innerHtml}</blockquote>`;
|
|
488
|
+
}
|
|
489
|
+
if (lines.every((l) => isUnorderedListItem(l))) {
|
|
490
|
+
const items = parseListItems(lines, 0);
|
|
491
|
+
return renderListItems(items, "ul");
|
|
492
|
+
}
|
|
493
|
+
if (lines.every((l) => isOrderedListItem(l))) {
|
|
494
|
+
const items = parseListItems(lines, 0);
|
|
495
|
+
return renderListItems(items, "ol");
|
|
496
|
+
}
|
|
497
|
+
if (lines.length > 0 && (isUnorderedListItem(lines[0]) || isOrderedListItem(lines[0]))) {
|
|
498
|
+
const isOrdered = isOrderedListItem(lines[0]);
|
|
499
|
+
const items = parseListItems(lines, 0);
|
|
500
|
+
return renderListItems(items, isOrdered ? "ol" : "ul");
|
|
501
|
+
}
|
|
502
|
+
return `<p>${renderInline(block)}</p>`;
|
|
503
|
+
}
|
|
417
504
|
function renderMarkdownToHtml(markdown) {
|
|
418
505
|
if (!markdown) {
|
|
419
506
|
return "";
|
|
@@ -422,22 +509,61 @@ function renderMarkdownToHtml(markdown) {
|
|
|
422
509
|
const withoutCode = replaceCodeBlocks(markdown, codeBlocks);
|
|
423
510
|
const normalized = withoutCode.replace(/\r\n/g, "\n");
|
|
424
511
|
const withHeadingBreaks = normalized.replace(/\n(#{1,6}\s)/g, "\n\n$1").replace(/(#{1,6}\s[^\n]+)\n(?!#|\n)/g, "$1\n\n");
|
|
425
|
-
const
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
if (
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
512
|
+
const rawBlocks = [];
|
|
513
|
+
let currentBlock = [];
|
|
514
|
+
const flushBlock = () => {
|
|
515
|
+
if (currentBlock.length > 0) {
|
|
516
|
+
rawBlocks.push(currentBlock.join("\n").trim());
|
|
517
|
+
currentBlock = [];
|
|
518
|
+
}
|
|
519
|
+
};
|
|
520
|
+
const inputLines = withHeadingBreaks.split("\n");
|
|
521
|
+
let i = 0;
|
|
522
|
+
while (i < inputLines.length) {
|
|
523
|
+
const line = inputLines[i];
|
|
524
|
+
const trimmed = line.trim();
|
|
525
|
+
if (trimmed === "") {
|
|
526
|
+
const prevIsListOrQuote = currentBlock.length > 0 && (isUnorderedListItem(currentBlock[0]) || isOrderedListItem(currentBlock[0]) || currentBlock[0].startsWith("> "));
|
|
527
|
+
if (prevIsListOrQuote) {
|
|
528
|
+
let nextIdx = i + 1;
|
|
529
|
+
while (nextIdx < inputLines.length && inputLines[nextIdx].trim() === "") nextIdx++;
|
|
530
|
+
if (nextIdx < inputLines.length) {
|
|
531
|
+
const nextTrimmed = inputLines[nextIdx].trim();
|
|
532
|
+
const continues = isUnorderedListItem(currentBlock[0]) && isUnorderedListItem(nextTrimmed) || isOrderedListItem(currentBlock[0]) && isOrderedListItem(nextTrimmed) || currentBlock[0].startsWith("> ") && (nextTrimmed.startsWith("> ") || nextTrimmed === ">");
|
|
533
|
+
if (continues) {
|
|
534
|
+
i++;
|
|
535
|
+
continue;
|
|
536
|
+
}
|
|
537
|
+
}
|
|
538
|
+
}
|
|
539
|
+
flushBlock();
|
|
540
|
+
i++;
|
|
541
|
+
continue;
|
|
432
542
|
}
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
543
|
+
if (isUnorderedListItem(trimmed) || isOrderedListItem(trimmed)) {
|
|
544
|
+
if (currentBlock.length > 0 && !isUnorderedListItem(currentBlock[0]) && !isOrderedListItem(currentBlock[0])) {
|
|
545
|
+
flushBlock();
|
|
546
|
+
}
|
|
547
|
+
currentBlock.push(trimmed);
|
|
548
|
+
i++;
|
|
549
|
+
continue;
|
|
438
550
|
}
|
|
439
|
-
|
|
440
|
-
|
|
551
|
+
if (trimmed.startsWith("> ") || trimmed === ">") {
|
|
552
|
+
if (currentBlock.length > 0 && !currentBlock[0].startsWith("> ") && currentBlock[0] !== ">") {
|
|
553
|
+
flushBlock();
|
|
554
|
+
}
|
|
555
|
+
currentBlock.push(trimmed);
|
|
556
|
+
i++;
|
|
557
|
+
continue;
|
|
558
|
+
}
|
|
559
|
+
if (currentBlock.length > 0 && (isUnorderedListItem(currentBlock[0]) || isOrderedListItem(currentBlock[0]) || currentBlock[0].startsWith("> "))) {
|
|
560
|
+
flushBlock();
|
|
561
|
+
}
|
|
562
|
+
currentBlock.push(line);
|
|
563
|
+
i++;
|
|
564
|
+
}
|
|
565
|
+
flushBlock();
|
|
566
|
+
const htmlBlocks = rawBlocks.filter(Boolean).map(renderBlock);
|
|
441
567
|
const html = htmlBlocks.join("\n");
|
|
442
568
|
return restoreCodeBlocks(html, codeBlocks);
|
|
443
569
|
}
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/auth.ts","../src/errors.ts","../src/utils.ts","../src/client.ts","../src/sitemap.ts","../src/robots.ts","../src/render.ts","../src/revalidate.ts","../src/metadata.ts","../src/env.ts"],"sourcesContent":["export { createAutoBlogWriterClient } from \"./client\";\nexport type { AutoBlogWriterClient } from \"./client\";\n\nexport { buildSitemap } from \"./sitemap\";\nexport { buildRobots } from \"./robots\";\nexport { renderMarkdownToHtml } from \"./render\";\nexport { verifyWebhookSignature, createRevalidateRouteHandler } from \"./revalidate\";\nexport { buildNextMetadata, type NextMetadata } from \"./metadata\";\n\nexport { buildAuthHeaders } from \"./auth\";\nexport { DEFAULT_TIMEOUT_MS, resolveClientConfig } from \"./utils\";\nexport { AutoBlogWriterError, ConfigError, ApiError, NotFoundError } from \"./errors\";\nexport { createAutoBlogWriterFromEnv, type AutoBlogWriterEnvConfig } from \"./env\";\n\nexport type {\n AutoBlogWriterClientConfig,\n BlogPost,\n BuildRobotsOptions,\n BuildSitemapOptions,\n MetadataRouteRobots,\n MetadataRouteSitemap,\n PaginatedList,\n PostsResponse,\n SitemapEntry,\n AutoBlogWriterRevalidatePayload,\n RevalidatePathFn,\n RevalidateTagFn,\n FetchRequestOptions,\n FetchNextConfig,\n} from \"./types\";\n","import { AuthMode } from \"./types\";\n\nexport function buildAuthHeaders(apiKey: string, authMode: AuthMode = \"bearer\"): Record<string, string> {\n if (authMode === \"x-api-key\") {\n return { \"x-api-key\": apiKey };\n }\n return { Authorization: `Bearer ${apiKey}` };\n}\n","export class AutoBlogWriterError extends Error {\n public readonly causeError?: unknown;\n\n constructor(message: string, options?: { cause?: unknown }) {\n super(message);\n this.name = new.target.name;\n this.causeError = options?.cause;\n }\n}\n\nexport class ConfigError extends AutoBlogWriterError {}\n\nexport interface ApiErrorDetails {\n status: number;\n code?: string;\n details?: unknown;\n}\n\nexport class ApiError extends AutoBlogWriterError {\n public readonly status: number;\n public readonly code?: string;\n public readonly details?: unknown;\n\n constructor(message: string, info: ApiErrorDetails, options?: { cause?: unknown }) {\n super(message, options);\n this.status = info.status;\n this.code = info.code;\n this.details = info.details;\n }\n}\n\nexport class NotFoundError extends ApiError {\n constructor(message: string, info?: Partial<ApiErrorDetails>) {\n super(message, { status: info?.status ?? 404, code: info?.code, details: info?.details });\n }\n}\n","import { ApiError, ConfigError } from \"./errors\";\nimport { AuthMode, AutoBlogWriterClientConfig, InternalClientConfig } from \"./types\";\n\nexport const DEFAULT_TIMEOUT_MS = 10_000;\n\nexport function normalizeApiUrl(apiUrl: string): string {\n if (!apiUrl) {\n throw new ConfigError(\"apiUrl is required\");\n }\n\n return apiUrl.replace(/\\/$/, \"\");\n}\n\nexport function resolveClientConfig(config: AutoBlogWriterClientConfig): InternalClientConfig {\n if (!config) {\n throw new ConfigError(\"Client configuration is required\");\n }\n\n const apiKey = config.apiKey?.trim();\n if (!apiKey) {\n throw new ConfigError(\"apiKey is required to authenticate with AutoBlogWriter\");\n }\n\n if (!config.workspaceId && !config.workspaceSlug) {\n throw new ConfigError(\"Provide either workspaceId or workspaceSlug\");\n }\n\n const authMode: AuthMode = config.authMode ?? \"bearer\";\n\n return {\n apiKey,\n apiUrl: normalizeApiUrl(config.apiUrl),\n workspaceId: config.workspaceId,\n workspaceSlug: config.workspaceSlug,\n authMode,\n fetch: config.fetch ?? globalThis.fetch.bind(globalThis),\n headers: { ...(config.headers ?? {}) },\n timeoutMs: config.timeoutMs ?? DEFAULT_TIMEOUT_MS,\n };\n}\n\nexport function buildQuery(params: Record<string, string | number | undefined | null>): string {\n const query = new URLSearchParams();\n Object.entries(params).forEach(([key, value]) => {\n if (value === undefined || value === null || value === \"\") {\n return;\n }\n query.set(key, String(value));\n });\n const qs = query.toString();\n return qs ? `?${qs}` : \"\";\n}\n\nexport async function withTimeout<T>(promise: Promise<T>, timeoutMs?: number): Promise<T> {\n if (!timeoutMs) {\n return promise;\n }\n\n let timeoutId: ReturnType<typeof setTimeout>;\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n timeoutId = setTimeout(() => {\n reject(\n new ApiError(`Request timed out after ${timeoutMs}ms`, {\n status: 408,\n }),\n );\n }, timeoutMs);\n });\n\n try {\n return await Promise.race([promise, timeoutPromise]);\n } finally {\n clearTimeout(timeoutId!);\n }\n}\n\nexport function mergeHeaders(\n ...headerObjects: Array<Record<string, string> | undefined>\n): Record<string, string> {\n return headerObjects.reduce<Record<string, string>>((acc, headers) => {\n if (!headers) {\n return acc;\n }\n Object.entries(headers).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return;\n }\n acc[key.toLowerCase()] = value;\n });\n return acc;\n }, {});\n}\n","import { buildAuthHeaders } from \"./auth\";\nimport { ApiError, AutoBlogWriterError, ConfigError } from \"./errors\";\nimport { buildQuery, mergeHeaders, resolveClientConfig, withTimeout } from \"./utils\";\nimport type {\n AutoBlogWriterClientConfig,\n BlogPost,\n FetchRequestOptions,\n PaginatedList,\n PostsResponse,\n} from \"./types\";\n\ninterface RequestOptions {\n path: string;\n method?: string;\n query?: Record<string, string | number | undefined | null>;\n body?: unknown;\n allowNotFound?: boolean;\n headers?: Record<string, string>;\n cache?: RequestCache;\n next?: FetchRequestOptions[\"next\"];\n}\n\nexport interface AutoBlogWriterClient {\n getPosts(params?: { limit?: number; cursor?: string; category?: string } & FetchRequestOptions): Promise<PostsResponse>;\n getPostBySlug(slug: string, options?: FetchRequestOptions): Promise<BlogPost | null>;\n getSitemapEntries(): Promise<Array<{ slug: string; updatedAt: string }>>;\n}\n\nexport function createAutoBlogWriterClient(config: AutoBlogWriterClientConfig): AutoBlogWriterClient {\n const resolved = resolveClientConfig(config);\n const fetchImpl = resolved.fetch;\n const debug =\n typeof process !== \"undefined\" &&\n typeof process.env !== \"undefined\" &&\n process.env.AUTOBLOGWRITER_DEBUG === \"true\";\n const debugLog = (...args: unknown[]) => {\n if (debug) {\n // eslint-disable-next-line no-console\n console.log(\"[autoblogwriter]\", ...args);\n }\n };\n\n function ensureWorkspaceSlug(): string {\n if (!resolved.workspaceSlug) {\n throw new ConfigError(\"workspaceSlug is required to call the AutoBlogWriter public API\");\n }\n return resolved.workspaceSlug;\n }\n\n function buildUrl(path: string, query?: Record<string, string | number | undefined | null>): string {\n const qs = buildQuery(query ?? {});\n return `${resolved.apiUrl}${path}${qs}`;\n }\n\n function unwrapSuccessPayload<T>(payload: unknown): T {\n if (payload && typeof payload === \"object\") {\n const maybePayload = payload as { success?: boolean; data?: T; error?: { message?: string } };\n if (\"success\" in maybePayload) {\n if (maybePayload.success === false) {\n throw new ApiError(\n maybePayload.error?.message ?? \"AutoBlogWriter request failed\",\n { status: 500, details: maybePayload },\n );\n }\n if (maybePayload.success && \"data\" in maybePayload) {\n return maybePayload.data as T;\n }\n }\n }\n return payload as T;\n }\n\n async function request<T>(opts: RequestOptions): Promise<T> {\n const method = opts.method ?? \"GET\";\n const url = buildUrl(opts.path, opts.query);\n\n const authHeaders = buildAuthHeaders(resolved.apiKey, resolved.authMode);\n const baseHeaders = mergeHeaders(resolved.headers, authHeaders, opts.headers);\n\n const init: RequestInit = {\n method,\n headers: baseHeaders,\n };\n\n if (opts.body) {\n init.body = typeof opts.body === \"string\" ? opts.body : JSON.stringify(opts.body);\n init.headers = {\n ...baseHeaders,\n \"content-type\": baseHeaders[\"content-type\"] ?? \"application/json\",\n };\n }\n\n if (opts.cache !== undefined) {\n init.cache = opts.cache;\n }\n\n if (opts.next) {\n (init as RequestInit & { next?: FetchRequestOptions[\"next\"] }).next = opts.next;\n }\n\n const startedAt = Date.now();\n try {\n const response = await withTimeout(fetchImpl(url, init), resolved.timeoutMs);\n debugLog(\"response\", {\n method,\n url,\n status: response.status,\n durationMs: Date.now() - startedAt,\n });\n if (opts.allowNotFound && response.status === 404) {\n return null as T;\n }\n\n const contentType = response.headers.get(\"content-type\") ?? \"\";\n const isJson = contentType.includes(\"application/json\");\n const payload = isJson ? await response.json().catch(() => undefined) : await response.text();\n\n if (!response.ok) {\n if (response.status === 404 && opts.allowNotFound) {\n return null as T;\n }\n\n const errorBody = payload as any;\n const message =\n errorBody?.error?.message ?? errorBody?.message ?? `Request failed with status ${response.status}`;\n throw new ApiError(message, {\n status: response.status,\n code: errorBody?.error?.code ?? errorBody?.code,\n details: errorBody?.error?.details ?? errorBody,\n });\n }\n\n if (isJson) {\n return unwrapSuccessPayload<T>(payload);\n }\n\n return (payload as T) ?? (undefined as T);\n } catch (error) {\n debugLog(\"request failed\", {\n method,\n url,\n durationMs: Date.now() - startedAt,\n error: error instanceof Error ? error.message : error,\n });\n if (error instanceof AutoBlogWriterError) {\n throw error;\n }\n const causeMessage =\n error instanceof Error ? error.message : typeof error === \"string\" ? error : undefined;\n const message = causeMessage\n ? `Network request to AutoBlogWriter failed: ${causeMessage}`\n : \"Network request to AutoBlogWriter failed\";\n throw new ApiError(\n message,\n {\n status: 0,\n details: { url, method },\n },\n { cause: error },\n );\n }\n }\n\n async function fetchBlogPage(\n limit: number,\n page: number,\n cacheOptions?: FetchRequestOptions,\n ): Promise<PaginatedList<BlogPost>> {\n const workspaceSlug = ensureWorkspaceSlug();\n debugLog(\"fetch posts page\", { workspaceSlug, limit, page });\n return request<PaginatedList<BlogPost>>({\n path: `/v1/public/${encodeURIComponent(workspaceSlug)}/blogs`,\n query: { limit, page },\n cache: cacheOptions?.cache,\n next: cacheOptions?.next,\n });\n }\n\n return {\n async getPosts(params) {\n const limit = params?.limit ?? 20;\n const cursorPage = params?.cursor ? Number(params.cursor) : undefined;\n const page = Number.isFinite(cursorPage) && cursorPage! >= 1 ? cursorPage! : 1;\n const data = await request<PaginatedList<BlogPost>>({\n path: `/v1/public/${encodeURIComponent(ensureWorkspaceSlug())}/blogs`,\n query: { limit, page, category: params?.category },\n cache: params?.cache,\n next: params?.next,\n });\n return {\n posts: data.items,\n nextCursor: data.pagination.hasMore ? String(data.pagination.page + 1) : undefined,\n };\n },\n async getPostBySlug(slug: string, options?: FetchRequestOptions) {\n if (!slug) {\n throw new ApiError(\"slug is required\", { status: 400 });\n }\n const workspaceSlug = ensureWorkspaceSlug();\n const post = await request<{ post: BlogPost } | null>({\n path: `/v1/public/${encodeURIComponent(workspaceSlug)}/blogs/${encodeURIComponent(slug)}`,\n allowNotFound: true,\n cache: options?.cache,\n next: options?.next,\n });\n if (post === null) {\n return null;\n }\n return post.post;\n },\n async getSitemapEntries() {\n const entries: Array<{ slug: string; updatedAt: string }> = [];\n const limit = 100;\n let page = 1;\n\n while (true) {\n const pageData = await fetchBlogPage(limit, page);\n pageData.items.forEach((post) => {\n entries.push({ slug: post.slug, updatedAt: post.updatedAt });\n });\n\n if (!pageData.pagination.hasMore) {\n break;\n }\n\n page = pageData.pagination.page + 1;\n }\n\n return entries;\n },\n };\n}\n","import { BuildSitemapOptions, MetadataRouteSitemap, MetadataRouteSitemapEntry } from \"./types\";\n\nfunction normalizeSiteUrl(url: string): string {\n return url.replace(/\\/$/, \"\");\n}\n\nfunction normalizeRoutePrefix(routePrefix?: string): string {\n if (!routePrefix || routePrefix === \"/\") {\n return \"\";\n }\n const withSlash = routePrefix.startsWith(\"/\") ? routePrefix : `/${routePrefix}`;\n return withSlash.endsWith(\"/\") ? withSlash.slice(0, -1) : withSlash;\n}\n\nfunction normalizeSlug(slug: string): string {\n return slug.replace(/^\\/+/, \"\").replace(/\\/+$/, \"\");\n}\n\nexport function buildSitemap(opts: BuildSitemapOptions): MetadataRouteSitemap {\n const siteUrl = normalizeSiteUrl(opts.siteUrl);\n const prefix = normalizeRoutePrefix(opts.routePrefix ?? \"/blog\");\n\n const entries: MetadataRouteSitemapEntry[] = opts.entries.map((entry) => {\n const slug = normalizeSlug(entry.slug);\n const hasSlug = slug.length > 0;\n const slugFragment = hasSlug ? `/${slug}` : \"\";\n const path = prefix ? `${prefix}${slugFragment}` : hasSlug ? `/${slug}` : \"/\";\n return {\n url: `${siteUrl}${path}`,\n lastModified: entry.updatedAt,\n };\n });\n\n return entries;\n}\n","import { BuildRobotsOptions, MetadataRouteRobots } from \"./types\";\n\nfunction normalizeSiteUrl(url: string): string {\n return url.replace(/\\/$/, \"\");\n}\n\nfunction normalizePath(path: string): string {\n if (!path) {\n return \"/sitemap.xml\";\n }\n if (!path.startsWith(\"/\")) {\n return `/${path}`;\n }\n return path;\n}\n\nexport function buildRobots(opts: BuildRobotsOptions): MetadataRouteRobots {\n const siteUrl = normalizeSiteUrl(opts.siteUrl);\n const sitemapPath = normalizePath(opts.sitemapPath ?? \"/sitemap.xml\");\n const sitemapUrl = `${siteUrl}${sitemapPath}`;\n\n return {\n rules: [{ userAgent: \"*\", allow: \"/\" }],\n sitemap: sitemapUrl,\n };\n}\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 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","import type { BlogPost } from \"./types\";\n\nexport interface NextMetadata {\n title?: string;\n description?: string;\n keywords?: string[];\n alternates?: {\n canonical?: string;\n };\n openGraph?: {\n type?: string;\n title?: string;\n description?: string;\n images?: Array<{ url: string }>;\n publishedTime?: string;\n modifiedTime?: string;\n };\n twitter?: {\n card?: string;\n title?: string;\n description?: string;\n images?: string[];\n };\n}\n\nexport function buildNextMetadata(post: BlogPost): NextMetadata {\n const title = post.seo?.title ?? post.title;\n const description = post.seo?.description ?? post.excerpt;\n const keywords = post.seo?.keywords;\n const canonical = post.metadata?.canonicalUrl;\n const ogImageUrl = post.metadata?.ogImageUrl;\n\n const metadata: NextMetadata = {\n title,\n description,\n };\n\n if (keywords && keywords.length > 0) {\n metadata.keywords = keywords;\n }\n\n if (canonical) {\n metadata.alternates = { canonical };\n }\n\n metadata.openGraph = {\n type: \"article\",\n title,\n description,\n publishedTime: post.publishedAt,\n modifiedTime: post.updatedAt,\n };\n\n if (ogImageUrl) {\n metadata.openGraph.images = [{ url: ogImageUrl }];\n }\n\n metadata.twitter = {\n card: ogImageUrl ? \"summary_large_image\" : \"summary\",\n title,\n description,\n };\n\n if (ogImageUrl) {\n metadata.twitter.images = [ogImageUrl];\n }\n\n return metadata;\n}\n","import { createAutoBlogWriterClient, type AutoBlogWriterClient } from \"./client\";\n\nexport interface AutoBlogWriterEnvConfig {\n client: AutoBlogWriterClient;\n workspaceSlug: string;\n workspaceId?: string;\n siteUrl: string;\n revalidateSecret?: string;\n apiUrl: string;\n apiKey: string;\n tags: {\n posts: string;\n post: (slug: string) => string;\n sitemap: string;\n };\n}\n\nlet _instance: AutoBlogWriterEnvConfig | null = null;\n\nfunction requireEnv(name: string): string {\n const value = process.env[name];\n if (!value) {\n throw new Error(`Missing required environment variable: ${name}`);\n }\n return value;\n}\n\nexport function createAutoBlogWriterFromEnv(): AutoBlogWriterEnvConfig {\n if (_instance) {\n return _instance;\n }\n\n const apiUrl = process.env.AUTOBLOGWRITER_API_URL ?? \"https://api.autoblogwriter.app\";\n const apiKey = requireEnv(\"AUTOBLOGWRITER_API_KEY\");\n const workspaceSlug = requireEnv(\"AUTOBLOGWRITER_WORKSPACE_SLUG\");\n const workspaceId = process.env.AUTOBLOGWRITER_WORKSPACE_ID;\n const siteUrl =\n process.env.SITE_URL ?? process.env.NEXT_PUBLIC_SITE_URL ?? \"http://localhost:3000\";\n const revalidateSecret = process.env.AUTOBLOGWRITER_REVALIDATE_SECRET;\n\n const client = createAutoBlogWriterClient({\n apiUrl,\n apiKey,\n workspaceSlug,\n workspaceId,\n });\n\n _instance = {\n client,\n workspaceSlug,\n workspaceId,\n siteUrl,\n revalidateSecret,\n apiUrl,\n apiKey,\n tags: {\n posts: `autoblogwriter:${workspaceSlug}:posts`,\n post: (slug: string) => `autoblogwriter:${workspaceSlug}:post:${slug}`,\n sitemap: `autoblogwriter:${workspaceSlug}:sitemap`,\n },\n };\n\n return _instance;\n}\n"],"mappings":"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","names":["normalizeSiteUrl","crypto"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/auth.ts","../src/errors.ts","../src/utils.ts","../src/client.ts","../src/sitemap.ts","../src/robots.ts","../src/render.ts","../src/revalidate.ts","../src/metadata.ts","../src/env.ts"],"sourcesContent":["export { createAutoBlogWriterClient } from \"./client\";\nexport type { AutoBlogWriterClient } from \"./client\";\n\nexport { buildSitemap } from \"./sitemap\";\nexport { buildRobots } from \"./robots\";\nexport { renderMarkdownToHtml } from \"./render\";\nexport { verifyWebhookSignature, createRevalidateRouteHandler } from \"./revalidate\";\nexport { buildNextMetadata, type NextMetadata } from \"./metadata\";\n\nexport { buildAuthHeaders } from \"./auth\";\nexport { DEFAULT_TIMEOUT_MS, resolveClientConfig } from \"./utils\";\nexport { AutoBlogWriterError, ConfigError, ApiError, NotFoundError } from \"./errors\";\nexport { createAutoBlogWriterFromEnv, type AutoBlogWriterEnvConfig } from \"./env\";\n\nexport type {\n AutoBlogWriterClientConfig,\n BlogPost,\n BuildRobotsOptions,\n BuildSitemapOptions,\n MetadataRouteRobots,\n MetadataRouteSitemap,\n PaginatedList,\n PostsResponse,\n SitemapEntry,\n AutoBlogWriterRevalidatePayload,\n RevalidatePathFn,\n RevalidateTagFn,\n FetchRequestOptions,\n FetchNextConfig,\n} from \"./types\";\n","import { AuthMode } from \"./types\";\n\nexport function buildAuthHeaders(apiKey: string, authMode: AuthMode = \"bearer\"): Record<string, string> {\n if (authMode === \"x-api-key\") {\n return { \"x-api-key\": apiKey };\n }\n return { Authorization: `Bearer ${apiKey}` };\n}\n","export class AutoBlogWriterError extends Error {\n public readonly causeError?: unknown;\n\n constructor(message: string, options?: { cause?: unknown }) {\n super(message);\n this.name = new.target.name;\n this.causeError = options?.cause;\n }\n}\n\nexport class ConfigError extends AutoBlogWriterError {}\n\nexport interface ApiErrorDetails {\n status: number;\n code?: string;\n details?: unknown;\n}\n\nexport class ApiError extends AutoBlogWriterError {\n public readonly status: number;\n public readonly code?: string;\n public readonly details?: unknown;\n\n constructor(message: string, info: ApiErrorDetails, options?: { cause?: unknown }) {\n super(message, options);\n this.status = info.status;\n this.code = info.code;\n this.details = info.details;\n }\n}\n\nexport class NotFoundError extends ApiError {\n constructor(message: string, info?: Partial<ApiErrorDetails>) {\n super(message, { status: info?.status ?? 404, code: info?.code, details: info?.details });\n }\n}\n","import { ApiError, ConfigError } from \"./errors\";\nimport { AuthMode, AutoBlogWriterClientConfig, InternalClientConfig } from \"./types\";\n\nexport const DEFAULT_TIMEOUT_MS = 10_000;\n\nexport function normalizeApiUrl(apiUrl: string): string {\n if (!apiUrl) {\n throw new ConfigError(\"apiUrl is required\");\n }\n\n return apiUrl.replace(/\\/$/, \"\");\n}\n\nexport function resolveClientConfig(config: AutoBlogWriterClientConfig): InternalClientConfig {\n if (!config) {\n throw new ConfigError(\"Client configuration is required\");\n }\n\n const apiKey = config.apiKey?.trim();\n if (!apiKey) {\n throw new ConfigError(\"apiKey is required to authenticate with AutoBlogWriter\");\n }\n\n if (!config.workspaceId && !config.workspaceSlug) {\n throw new ConfigError(\"Provide either workspaceId or workspaceSlug\");\n }\n\n const authMode: AuthMode = config.authMode ?? \"bearer\";\n\n return {\n apiKey,\n apiUrl: normalizeApiUrl(config.apiUrl),\n workspaceId: config.workspaceId,\n workspaceSlug: config.workspaceSlug,\n authMode,\n fetch: config.fetch ?? globalThis.fetch.bind(globalThis),\n headers: { ...(config.headers ?? {}) },\n timeoutMs: config.timeoutMs ?? DEFAULT_TIMEOUT_MS,\n };\n}\n\nexport function buildQuery(params: Record<string, string | number | undefined | null>): string {\n const query = new URLSearchParams();\n Object.entries(params).forEach(([key, value]) => {\n if (value === undefined || value === null || value === \"\") {\n return;\n }\n query.set(key, String(value));\n });\n const qs = query.toString();\n return qs ? `?${qs}` : \"\";\n}\n\nexport async function withTimeout<T>(promise: Promise<T>, timeoutMs?: number): Promise<T> {\n if (!timeoutMs) {\n return promise;\n }\n\n let timeoutId: ReturnType<typeof setTimeout>;\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n timeoutId = setTimeout(() => {\n reject(\n new ApiError(`Request timed out after ${timeoutMs}ms`, {\n status: 408,\n }),\n );\n }, timeoutMs);\n });\n\n try {\n return await Promise.race([promise, timeoutPromise]);\n } finally {\n clearTimeout(timeoutId!);\n }\n}\n\nexport function mergeHeaders(\n ...headerObjects: Array<Record<string, string> | undefined>\n): Record<string, string> {\n return headerObjects.reduce<Record<string, string>>((acc, headers) => {\n if (!headers) {\n return acc;\n }\n Object.entries(headers).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return;\n }\n acc[key.toLowerCase()] = value;\n });\n return acc;\n }, {});\n}\n","import { buildAuthHeaders } from \"./auth\";\nimport { ApiError, AutoBlogWriterError, ConfigError } from \"./errors\";\nimport { buildQuery, mergeHeaders, resolveClientConfig, withTimeout } from \"./utils\";\nimport type {\n AutoBlogWriterClientConfig,\n BlogPost,\n FetchRequestOptions,\n PaginatedList,\n PostsResponse,\n} from \"./types\";\n\ninterface RequestOptions {\n path: string;\n method?: string;\n query?: Record<string, string | number | undefined | null>;\n body?: unknown;\n allowNotFound?: boolean;\n headers?: Record<string, string>;\n cache?: RequestCache;\n next?: FetchRequestOptions[\"next\"];\n}\n\nexport interface AutoBlogWriterClient {\n getPosts(params?: { limit?: number; cursor?: string; category?: string } & FetchRequestOptions): Promise<PostsResponse>;\n getPostBySlug(slug: string, options?: FetchRequestOptions): Promise<BlogPost | null>;\n getSitemapEntries(): Promise<Array<{ slug: string; updatedAt: string }>>;\n}\n\nexport function createAutoBlogWriterClient(config: AutoBlogWriterClientConfig): AutoBlogWriterClient {\n const resolved = resolveClientConfig(config);\n const fetchImpl = resolved.fetch;\n const debug =\n typeof process !== \"undefined\" &&\n typeof process.env !== \"undefined\" &&\n process.env.AUTOBLOGWRITER_DEBUG === \"true\";\n const debugLog = (...args: unknown[]) => {\n if (debug) {\n // eslint-disable-next-line no-console\n console.log(\"[autoblogwriter]\", ...args);\n }\n };\n\n function ensureWorkspaceSlug(): string {\n if (!resolved.workspaceSlug) {\n throw new ConfigError(\"workspaceSlug is required to call the AutoBlogWriter public API\");\n }\n return resolved.workspaceSlug;\n }\n\n function buildUrl(path: string, query?: Record<string, string | number | undefined | null>): string {\n const qs = buildQuery(query ?? {});\n return `${resolved.apiUrl}${path}${qs}`;\n }\n\n function unwrapSuccessPayload<T>(payload: unknown): T {\n if (payload && typeof payload === \"object\") {\n const maybePayload = payload as { success?: boolean; data?: T; error?: { message?: string } };\n if (\"success\" in maybePayload) {\n if (maybePayload.success === false) {\n throw new ApiError(\n maybePayload.error?.message ?? \"AutoBlogWriter request failed\",\n { status: 500, details: maybePayload },\n );\n }\n if (maybePayload.success && \"data\" in maybePayload) {\n return maybePayload.data as T;\n }\n }\n }\n return payload as T;\n }\n\n async function request<T>(opts: RequestOptions): Promise<T> {\n const method = opts.method ?? \"GET\";\n const url = buildUrl(opts.path, opts.query);\n\n const authHeaders = buildAuthHeaders(resolved.apiKey, resolved.authMode);\n const baseHeaders = mergeHeaders(resolved.headers, authHeaders, opts.headers);\n\n const init: RequestInit = {\n method,\n headers: baseHeaders,\n };\n\n if (opts.body) {\n init.body = typeof opts.body === \"string\" ? opts.body : JSON.stringify(opts.body);\n init.headers = {\n ...baseHeaders,\n \"content-type\": baseHeaders[\"content-type\"] ?? \"application/json\",\n };\n }\n\n if (opts.cache !== undefined) {\n init.cache = opts.cache;\n }\n\n if (opts.next) {\n (init as RequestInit & { next?: FetchRequestOptions[\"next\"] }).next = opts.next;\n }\n\n const startedAt = Date.now();\n try {\n const response = await withTimeout(fetchImpl(url, init), resolved.timeoutMs);\n debugLog(\"response\", {\n method,\n url,\n status: response.status,\n durationMs: Date.now() - startedAt,\n });\n if (opts.allowNotFound && response.status === 404) {\n return null as T;\n }\n\n const contentType = response.headers.get(\"content-type\") ?? \"\";\n const isJson = contentType.includes(\"application/json\");\n const payload = isJson ? await response.json().catch(() => undefined) : await response.text();\n\n if (!response.ok) {\n if (response.status === 404 && opts.allowNotFound) {\n return null as T;\n }\n\n const errorBody = payload as any;\n const message =\n errorBody?.error?.message ?? errorBody?.message ?? `Request failed with status ${response.status}`;\n throw new ApiError(message, {\n status: response.status,\n code: errorBody?.error?.code ?? errorBody?.code,\n details: errorBody?.error?.details ?? errorBody,\n });\n }\n\n if (isJson) {\n return unwrapSuccessPayload<T>(payload);\n }\n\n return (payload as T) ?? (undefined as T);\n } catch (error) {\n debugLog(\"request failed\", {\n method,\n url,\n durationMs: Date.now() - startedAt,\n error: error instanceof Error ? error.message : error,\n });\n if (error instanceof AutoBlogWriterError) {\n throw error;\n }\n const causeMessage =\n error instanceof Error ? error.message : typeof error === \"string\" ? error : undefined;\n const message = causeMessage\n ? `Network request to AutoBlogWriter failed: ${causeMessage}`\n : \"Network request to AutoBlogWriter failed\";\n throw new ApiError(\n message,\n {\n status: 0,\n details: { url, method },\n },\n { cause: error },\n );\n }\n }\n\n async function fetchBlogPage(\n limit: number,\n page: number,\n cacheOptions?: FetchRequestOptions,\n ): Promise<PaginatedList<BlogPost>> {\n const workspaceSlug = ensureWorkspaceSlug();\n debugLog(\"fetch posts page\", { workspaceSlug, limit, page });\n return request<PaginatedList<BlogPost>>({\n path: `/v1/public/${encodeURIComponent(workspaceSlug)}/blogs`,\n query: { limit, page },\n cache: cacheOptions?.cache,\n next: cacheOptions?.next,\n });\n }\n\n return {\n async getPosts(params) {\n const limit = params?.limit ?? 20;\n const cursorPage = params?.cursor ? Number(params.cursor) : undefined;\n const page = Number.isFinite(cursorPage) && cursorPage! >= 1 ? cursorPage! : 1;\n const data = await request<PaginatedList<BlogPost>>({\n path: `/v1/public/${encodeURIComponent(ensureWorkspaceSlug())}/blogs`,\n query: { limit, page, category: params?.category },\n cache: params?.cache,\n next: params?.next,\n });\n return {\n posts: data.items,\n nextCursor: data.pagination.hasMore ? String(data.pagination.page + 1) : undefined,\n };\n },\n async getPostBySlug(slug: string, options?: FetchRequestOptions) {\n if (!slug) {\n throw new ApiError(\"slug is required\", { status: 400 });\n }\n const workspaceSlug = ensureWorkspaceSlug();\n const post = await request<{ post: BlogPost } | null>({\n path: `/v1/public/${encodeURIComponent(workspaceSlug)}/blogs/${encodeURIComponent(slug)}`,\n allowNotFound: true,\n cache: options?.cache,\n next: options?.next,\n });\n if (post === null) {\n return null;\n }\n return post.post;\n },\n async getSitemapEntries() {\n const entries: Array<{ slug: string; updatedAt: string }> = [];\n const limit = 100;\n let page = 1;\n\n while (true) {\n const pageData = await fetchBlogPage(limit, page);\n pageData.items.forEach((post) => {\n entries.push({ slug: post.slug, updatedAt: post.updatedAt });\n });\n\n if (!pageData.pagination.hasMore) {\n break;\n }\n\n page = pageData.pagination.page + 1;\n }\n\n return entries;\n },\n };\n}\n","import { BuildSitemapOptions, MetadataRouteSitemap, MetadataRouteSitemapEntry } from \"./types\";\n\nfunction normalizeSiteUrl(url: string): string {\n return url.replace(/\\/$/, \"\");\n}\n\nfunction normalizeRoutePrefix(routePrefix?: string): string {\n if (!routePrefix || routePrefix === \"/\") {\n return \"\";\n }\n const withSlash = routePrefix.startsWith(\"/\") ? routePrefix : `/${routePrefix}`;\n return withSlash.endsWith(\"/\") ? withSlash.slice(0, -1) : withSlash;\n}\n\nfunction normalizeSlug(slug: string): string {\n return slug.replace(/^\\/+/, \"\").replace(/\\/+$/, \"\");\n}\n\nexport function buildSitemap(opts: BuildSitemapOptions): MetadataRouteSitemap {\n const siteUrl = normalizeSiteUrl(opts.siteUrl);\n const prefix = normalizeRoutePrefix(opts.routePrefix ?? \"/blog\");\n\n const entries: MetadataRouteSitemapEntry[] = opts.entries.map((entry) => {\n const slug = normalizeSlug(entry.slug);\n const hasSlug = slug.length > 0;\n const slugFragment = hasSlug ? `/${slug}` : \"\";\n const path = prefix ? `${prefix}${slugFragment}` : hasSlug ? `/${slug}` : \"/\";\n return {\n url: `${siteUrl}${path}`,\n lastModified: entry.updatedAt,\n };\n });\n\n return entries;\n}\n","import { BuildRobotsOptions, MetadataRouteRobots } from \"./types\";\n\nfunction normalizeSiteUrl(url: string): string {\n return url.replace(/\\/$/, \"\");\n}\n\nfunction normalizePath(path: string): string {\n if (!path) {\n return \"/sitemap.xml\";\n }\n if (!path.startsWith(\"/\")) {\n return `/${path}`;\n }\n return path;\n}\n\nexport function buildRobots(opts: BuildRobotsOptions): MetadataRouteRobots {\n const siteUrl = normalizeSiteUrl(opts.siteUrl);\n const sitemapPath = normalizePath(opts.sitemapPath ?? \"/sitemap.xml\");\n const sitemapUrl = `${siteUrl}${sitemapPath}`;\n\n return {\n rules: [{ userAgent: \"*\", allow: \"/\" }],\n sitemap: sitemapUrl,\n };\n}\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 // bold\n output = output.replace(/\\*\\*(.+?)\\*\\*/g, \"<strong>$1</strong>\");\n output = output.replace(/__(.+?)__/g, \"<strong>$1</strong>\");\n // italic\n output = output.replace(/\\*(.+?)\\*/g, \"<em>$1</em>\");\n output = output.replace(/(?<!\\w)_(.+?)_(?!\\w)/g, \"<em>$1</em>\");\n // strikethrough\n output = output.replace(/~~(.+?)~~/g, \"<s>$1</s>\");\n // underline (non-standard but common: ++text++)\n output = output.replace(/\\+\\+(.+?)\\+\\+/g, \"<u>$1</u>\");\n // inline code\n output = output.replace(/`([^`]+)`/g, (_, code) => `<code>${code}</code>`);\n // images (before links)\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 // links\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 // line breaks\n output = output.replace(/\\n/g, \"<br />\");\n return output;\n}\n\nfunction replaceCodeBlocks(markdown: string, blocks: string[]): string {\n return markdown.replace(/```(\\w*)\\n?([\\s\\S]*?)```/g, (_, lang, code) => {\n const langAttr = lang ? ` class=\"language-${escapeHtml(lang)}\"` : \"\";\n const index = blocks.push(`<pre><code${langAttr}>${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\nfunction isUnorderedListItem(line: string): boolean {\n return /^[\\t ]*[-*+] /.test(line);\n}\n\nfunction isOrderedListItem(line: string): boolean {\n return /^[\\t ]*\\d+\\. /.test(line);\n}\n\nfunction getListItemContent(line: string): string {\n return line.replace(/^[\\t ]*(?:[-*+]|\\d+\\.) /, \"\");\n}\n\nfunction getIndentLevel(line: string): number {\n const match = line.match(/^([\\t ]*)/);\n if (!match) return 0;\n const indent = match[1];\n // Count tabs as 2 spaces worth\n return indent.replace(/\\t/g, \" \").length;\n}\n\ninterface ListItem {\n content: string;\n children: ListItem[];\n}\n\nfunction parseListItems(lines: string[], baseIndent: number): ListItem[] {\n const items: ListItem[] = [];\n let i = 0;\n\n while (i < lines.length) {\n const indent = getIndentLevel(lines[i]);\n if (indent > baseIndent && items.length > 0) {\n // Nested: collect all lines at this deeper level\n const nestedLines: string[] = [];\n while (i < lines.length && getIndentLevel(lines[i]) > baseIndent) {\n nestedLines.push(lines[i]);\n i++;\n }\n items[items.length - 1].children = parseListItems(nestedLines, baseIndent + 2);\n } else {\n items.push({ content: getListItemContent(lines[i]), children: [] });\n i++;\n }\n }\n\n return items;\n}\n\nfunction renderListItems(items: ListItem[], tag: \"ul\" | \"ol\"): string {\n const lis = items.map((item) => {\n let li = `<li>${renderInline(item.content)}`;\n if (item.children.length > 0) {\n li += renderListItems(item.children, tag);\n }\n li += \"</li>\";\n return li;\n });\n return `<${tag}>${lis.join(\"\")}</${tag}>`;\n}\n\nfunction renderBlock(block: string): string {\n // Heading\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 // Horizontal rule\n if (/^(?:---+|\\*\\*\\*+|___+)$/.test(block.trim())) {\n return \"<hr />\";\n }\n\n // 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 // Blockquote\n const lines = block.split(\"\\n\");\n if (lines.every((l) => l.startsWith(\"> \") || l === \">\")) {\n const inner = lines.map((l) => l.replace(/^>\\s?/, \"\")).join(\"\\n\");\n // Recursively render blockquote content\n const innerHtml = inner\n .split(/\\n{2,}/)\n .map((b) => b.trim())\n .filter(Boolean)\n .map(renderBlock)\n .join(\"\\n\");\n return `<blockquote>${innerHtml}</blockquote>`;\n }\n\n // Unordered list\n if (lines.every((l) => isUnorderedListItem(l))) {\n const items = parseListItems(lines, 0);\n return renderListItems(items, \"ul\");\n }\n\n // Ordered list\n if (lines.every((l) => isOrderedListItem(l))) {\n const items = parseListItems(lines, 0);\n return renderListItems(items, \"ol\");\n }\n\n // Mixed list lines or lines starting with list markers — try to detect\n if (lines.length > 0 && (isUnorderedListItem(lines[0]) || isOrderedListItem(lines[0]))) {\n const isOrdered = isOrderedListItem(lines[0]);\n const items = parseListItems(lines, 0);\n return renderListItems(items, isOrdered ? \"ol\" : \"ul\");\n }\n\n // Default: paragraph\n return `<p>${renderInline(block)}</p>`;\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\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\n // Split into blocks but keep list items and blockquotes grouped\n const rawBlocks: string[] = [];\n let currentBlock: string[] = [];\n\n const flushBlock = () => {\n if (currentBlock.length > 0) {\n rawBlocks.push(currentBlock.join(\"\\n\").trim());\n currentBlock = [];\n }\n };\n\n const inputLines = withHeadingBreaks.split(\"\\n\");\n let i = 0;\n\n while (i < inputLines.length) {\n const line = inputLines[i];\n const trimmed = line.trim();\n\n // Empty line: flush current block\n if (trimmed === \"\") {\n // But if we're in a list or blockquote, keep going if next line continues it\n const prevIsListOrQuote =\n currentBlock.length > 0 &&\n (isUnorderedListItem(currentBlock[0]) ||\n isOrderedListItem(currentBlock[0]) ||\n currentBlock[0].startsWith(\"> \"));\n\n if (prevIsListOrQuote) {\n // Check if the next non-empty line continues the same structure\n let nextIdx = i + 1;\n while (nextIdx < inputLines.length && inputLines[nextIdx].trim() === \"\") nextIdx++;\n if (nextIdx < inputLines.length) {\n const nextTrimmed = inputLines[nextIdx].trim();\n const continues =\n (isUnorderedListItem(currentBlock[0]) && isUnorderedListItem(nextTrimmed)) ||\n (isOrderedListItem(currentBlock[0]) && isOrderedListItem(nextTrimmed)) ||\n (currentBlock[0].startsWith(\"> \") && (nextTrimmed.startsWith(\"> \") || nextTrimmed === \">\"));\n if (continues) {\n i++;\n continue;\n }\n }\n }\n flushBlock();\n i++;\n continue;\n }\n\n // List items: group together\n if (isUnorderedListItem(trimmed) || isOrderedListItem(trimmed)) {\n // If current block is not a list, flush first\n if (\n currentBlock.length > 0 &&\n !isUnorderedListItem(currentBlock[0]) &&\n !isOrderedListItem(currentBlock[0])\n ) {\n flushBlock();\n }\n currentBlock.push(trimmed);\n i++;\n continue;\n }\n\n // Blockquote lines: group together\n if (trimmed.startsWith(\"> \") || trimmed === \">\") {\n if (currentBlock.length > 0 && !currentBlock[0].startsWith(\"> \") && currentBlock[0] !== \">\") {\n flushBlock();\n }\n currentBlock.push(trimmed);\n i++;\n continue;\n }\n\n // Regular line\n if (\n currentBlock.length > 0 &&\n (isUnorderedListItem(currentBlock[0]) ||\n isOrderedListItem(currentBlock[0]) ||\n currentBlock[0].startsWith(\"> \"))\n ) {\n flushBlock();\n }\n currentBlock.push(line);\n i++;\n }\n flushBlock();\n\n const htmlBlocks = rawBlocks.filter(Boolean).map(renderBlock);\n const html = htmlBlocks.join(\"\\n\");\n return restoreCodeBlocks(html, codeBlocks);\n}\n","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","import type { BlogPost } from \"./types\";\n\nexport interface NextMetadata {\n title?: string;\n description?: string;\n keywords?: string[];\n alternates?: {\n canonical?: string;\n };\n openGraph?: {\n type?: string;\n title?: string;\n description?: string;\n images?: Array<{ url: string }>;\n publishedTime?: string;\n modifiedTime?: string;\n };\n twitter?: {\n card?: string;\n title?: string;\n description?: string;\n images?: string[];\n };\n}\n\nexport function buildNextMetadata(post: BlogPost): NextMetadata {\n const title = post.seo?.title ?? post.title;\n const description = post.seo?.description ?? post.excerpt;\n const keywords = post.seo?.keywords;\n const canonical = post.metadata?.canonicalUrl;\n const ogImageUrl = post.metadata?.ogImageUrl;\n\n const metadata: NextMetadata = {\n title,\n description,\n };\n\n if (keywords && keywords.length > 0) {\n metadata.keywords = keywords;\n }\n\n if (canonical) {\n metadata.alternates = { canonical };\n }\n\n metadata.openGraph = {\n type: \"article\",\n title,\n description,\n publishedTime: post.publishedAt,\n modifiedTime: post.updatedAt,\n };\n\n if (ogImageUrl) {\n metadata.openGraph.images = [{ url: ogImageUrl }];\n }\n\n metadata.twitter = {\n card: ogImageUrl ? \"summary_large_image\" : \"summary\",\n title,\n description,\n };\n\n if (ogImageUrl) {\n metadata.twitter.images = [ogImageUrl];\n }\n\n return metadata;\n}\n","import { createAutoBlogWriterClient, type AutoBlogWriterClient } from \"./client\";\n\nexport interface AutoBlogWriterEnvConfig {\n client: AutoBlogWriterClient;\n workspaceSlug: string;\n workspaceId?: string;\n siteUrl: string;\n revalidateSecret?: string;\n apiUrl: string;\n apiKey: string;\n tags: {\n posts: string;\n post: (slug: string) => string;\n sitemap: string;\n };\n}\n\nlet _instance: AutoBlogWriterEnvConfig | null = null;\n\nfunction requireEnv(name: string): string {\n const value = process.env[name];\n if (!value) {\n throw new Error(`Missing required environment variable: ${name}`);\n }\n return value;\n}\n\nexport function createAutoBlogWriterFromEnv(): AutoBlogWriterEnvConfig {\n if (_instance) {\n return _instance;\n }\n\n const apiUrl = process.env.AUTOBLOGWRITER_API_URL ?? \"https://api.autoblogwriter.app\";\n const apiKey = requireEnv(\"AUTOBLOGWRITER_API_KEY\");\n const workspaceSlug = requireEnv(\"AUTOBLOGWRITER_WORKSPACE_SLUG\");\n const workspaceId = process.env.AUTOBLOGWRITER_WORKSPACE_ID;\n const siteUrl =\n process.env.SITE_URL ?? process.env.NEXT_PUBLIC_SITE_URL ?? \"http://localhost:3000\";\n const revalidateSecret = process.env.AUTOBLOGWRITER_REVALIDATE_SECRET;\n\n const client = createAutoBlogWriterClient({\n apiUrl,\n apiKey,\n workspaceSlug,\n workspaceId,\n });\n\n _instance = {\n client,\n workspaceSlug,\n workspaceId,\n siteUrl,\n revalidateSecret,\n apiUrl,\n apiKey,\n tags: {\n posts: `autoblogwriter:${workspaceSlug}:posts`,\n post: (slug: string) => `autoblogwriter:${workspaceSlug}:post:${slug}`,\n sitemap: `autoblogwriter:${workspaceSlug}:sitemap`,\n },\n };\n\n return _instance;\n}\n"],"mappings":"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","names":["normalizeSiteUrl","crypto"]}
|
package/dist/index.d.cts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
import { A as AutoBlogWriterClientConfig, F as FetchRequestOptions, P as PostsResponse, B as BlogPost, a as BuildSitemapOptions, M as MetadataRouteSitemap, b as BuildRobotsOptions, c as MetadataRouteRobots, d as AuthMode, I as InternalClientConfig } from './types-
|
|
2
|
-
export { f as AutoBlogWriterRevalidatePayload, h as FetchNextConfig, e as PaginatedList, R as RevalidatePathFn, g as RevalidateTagFn, S as SitemapEntry } from './types-
|
|
1
|
+
import { A as AutoBlogWriterClientConfig, F as FetchRequestOptions, P as PostsResponse, B as BlogPost, a as BuildSitemapOptions, M as MetadataRouteSitemap, b as BuildRobotsOptions, c as MetadataRouteRobots, d as AuthMode, I as InternalClientConfig } from './types-DuDmhtM7.cjs';
|
|
2
|
+
export { f as AutoBlogWriterRevalidatePayload, h as FetchNextConfig, e as PaginatedList, R as RevalidatePathFn, g as RevalidateTagFn, S as SitemapEntry } from './types-DuDmhtM7.cjs';
|
|
3
3
|
export { createRevalidateRouteHandler, verifyWebhookSignature } from './revalidate.cjs';
|
|
4
|
-
export { N as NextMetadata, b as buildNextMetadata } from './metadata-
|
|
4
|
+
export { N as NextMetadata, b as buildNextMetadata } from './metadata-DaBB2mdh.cjs';
|
|
5
5
|
|
|
6
6
|
interface AutoBlogWriterClient {
|
|
7
7
|
getPosts(params?: {
|
package/dist/index.d.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
import { A as AutoBlogWriterClientConfig, F as FetchRequestOptions, P as PostsResponse, B as BlogPost, a as BuildSitemapOptions, M as MetadataRouteSitemap, b as BuildRobotsOptions, c as MetadataRouteRobots, d as AuthMode, I as InternalClientConfig } from './types-
|
|
2
|
-
export { f as AutoBlogWriterRevalidatePayload, h as FetchNextConfig, e as PaginatedList, R as RevalidatePathFn, g as RevalidateTagFn, S as SitemapEntry } from './types-
|
|
1
|
+
import { A as AutoBlogWriterClientConfig, F as FetchRequestOptions, P as PostsResponse, B as BlogPost, a as BuildSitemapOptions, M as MetadataRouteSitemap, b as BuildRobotsOptions, c as MetadataRouteRobots, d as AuthMode, I as InternalClientConfig } from './types-DuDmhtM7.js';
|
|
2
|
+
export { f as AutoBlogWriterRevalidatePayload, h as FetchNextConfig, e as PaginatedList, R as RevalidatePathFn, g as RevalidateTagFn, S as SitemapEntry } from './types-DuDmhtM7.js';
|
|
3
3
|
export { createRevalidateRouteHandler, verifyWebhookSignature } from './revalidate.js';
|
|
4
|
-
export { N as NextMetadata, b as buildNextMetadata } from './metadata-
|
|
4
|
+
export { N as NextMetadata, b as buildNextMetadata } from './metadata-Du-JkBih.js';
|
|
5
5
|
|
|
6
6
|
interface AutoBlogWriterClient {
|
|
7
7
|
getPosts(params?: {
|
package/dist/index.js
CHANGED
|
@@ -341,7 +341,11 @@ function escapeHtml(value) {
|
|
|
341
341
|
function renderInline(input) {
|
|
342
342
|
let output = escapeHtml(input);
|
|
343
343
|
output = output.replace(/\*\*(.+?)\*\*/g, "<strong>$1</strong>");
|
|
344
|
+
output = output.replace(/__(.+?)__/g, "<strong>$1</strong>");
|
|
344
345
|
output = output.replace(/\*(.+?)\*/g, "<em>$1</em>");
|
|
346
|
+
output = output.replace(/(?<!\w)_(.+?)_(?!\w)/g, "<em>$1</em>");
|
|
347
|
+
output = output.replace(/~~(.+?)~~/g, "<s>$1</s>");
|
|
348
|
+
output = output.replace(/\+\+(.+?)\+\+/g, "<u>$1</u>");
|
|
345
349
|
output = output.replace(/`([^`]+)`/g, (_, code) => `<code>${code}</code>`);
|
|
346
350
|
output = output.replace(/!\[([^\]]*)\]\(([^)]+)\)/g, (_, alt, src) => {
|
|
347
351
|
const safeSrc = escapeHtml(src);
|
|
@@ -356,14 +360,97 @@ function renderInline(input) {
|
|
|
356
360
|
return output;
|
|
357
361
|
}
|
|
358
362
|
function replaceCodeBlocks(markdown, blocks) {
|
|
359
|
-
return markdown.replace(/```([\s\S]*?)```/g, (_, code) => {
|
|
360
|
-
const
|
|
363
|
+
return markdown.replace(/```(\w*)\n?([\s\S]*?)```/g, (_, lang, code) => {
|
|
364
|
+
const langAttr = lang ? ` class="language-${escapeHtml(lang)}"` : "";
|
|
365
|
+
const index = blocks.push(`<pre><code${langAttr}>${escapeHtml(code.trim())}</code></pre>`) - 1;
|
|
361
366
|
return `${CODE_BLOCK_TOKEN}${index}__`;
|
|
362
367
|
});
|
|
363
368
|
}
|
|
364
369
|
function restoreCodeBlocks(html, blocks) {
|
|
365
370
|
return html.replace(/__AUTOBLOGWRITER_CODE_BLOCK_(\d+)__/g, (_, rawIndex) => blocks[Number(rawIndex)] ?? "");
|
|
366
371
|
}
|
|
372
|
+
function isUnorderedListItem(line) {
|
|
373
|
+
return /^[\t ]*[-*+] /.test(line);
|
|
374
|
+
}
|
|
375
|
+
function isOrderedListItem(line) {
|
|
376
|
+
return /^[\t ]*\d+\. /.test(line);
|
|
377
|
+
}
|
|
378
|
+
function getListItemContent(line) {
|
|
379
|
+
return line.replace(/^[\t ]*(?:[-*+]|\d+\.) /, "");
|
|
380
|
+
}
|
|
381
|
+
function getIndentLevel(line) {
|
|
382
|
+
const match = line.match(/^([\t ]*)/);
|
|
383
|
+
if (!match) return 0;
|
|
384
|
+
const indent = match[1];
|
|
385
|
+
return indent.replace(/\t/g, " ").length;
|
|
386
|
+
}
|
|
387
|
+
function parseListItems(lines, baseIndent) {
|
|
388
|
+
const items = [];
|
|
389
|
+
let i = 0;
|
|
390
|
+
while (i < lines.length) {
|
|
391
|
+
const indent = getIndentLevel(lines[i]);
|
|
392
|
+
if (indent > baseIndent && items.length > 0) {
|
|
393
|
+
const nestedLines = [];
|
|
394
|
+
while (i < lines.length && getIndentLevel(lines[i]) > baseIndent) {
|
|
395
|
+
nestedLines.push(lines[i]);
|
|
396
|
+
i++;
|
|
397
|
+
}
|
|
398
|
+
items[items.length - 1].children = parseListItems(nestedLines, baseIndent + 2);
|
|
399
|
+
} else {
|
|
400
|
+
items.push({ content: getListItemContent(lines[i]), children: [] });
|
|
401
|
+
i++;
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
return items;
|
|
405
|
+
}
|
|
406
|
+
function renderListItems(items, tag) {
|
|
407
|
+
const lis = items.map((item) => {
|
|
408
|
+
let li = `<li>${renderInline(item.content)}`;
|
|
409
|
+
if (item.children.length > 0) {
|
|
410
|
+
li += renderListItems(item.children, tag);
|
|
411
|
+
}
|
|
412
|
+
li += "</li>";
|
|
413
|
+
return li;
|
|
414
|
+
});
|
|
415
|
+
return `<${tag}>${lis.join("")}</${tag}>`;
|
|
416
|
+
}
|
|
417
|
+
function renderBlock(block) {
|
|
418
|
+
const headingMatch = block.match(/^(#{1,6})\s+(.*)$/);
|
|
419
|
+
if (headingMatch) {
|
|
420
|
+
const level = headingMatch[1].length;
|
|
421
|
+
const content = renderInline(headingMatch[2]);
|
|
422
|
+
return `<h${level}>${content}</h${level}>`;
|
|
423
|
+
}
|
|
424
|
+
if (/^(?:---+|\*\*\*+|___+)$/.test(block.trim())) {
|
|
425
|
+
return "<hr />";
|
|
426
|
+
}
|
|
427
|
+
const imageMatch = block.match(/^!\[([^\]]*)\]\(([^)]+)\)$/);
|
|
428
|
+
if (imageMatch) {
|
|
429
|
+
const safeSrc = escapeHtml(imageMatch[2]);
|
|
430
|
+
const safeAlt = escapeHtml(imageMatch[1]);
|
|
431
|
+
return `<img src="${safeSrc}" alt="${safeAlt}" />`;
|
|
432
|
+
}
|
|
433
|
+
const lines = block.split("\n");
|
|
434
|
+
if (lines.every((l) => l.startsWith("> ") || l === ">")) {
|
|
435
|
+
const inner = lines.map((l) => l.replace(/^>\s?/, "")).join("\n");
|
|
436
|
+
const innerHtml = inner.split(/\n{2,}/).map((b) => b.trim()).filter(Boolean).map(renderBlock).join("\n");
|
|
437
|
+
return `<blockquote>${innerHtml}</blockquote>`;
|
|
438
|
+
}
|
|
439
|
+
if (lines.every((l) => isUnorderedListItem(l))) {
|
|
440
|
+
const items = parseListItems(lines, 0);
|
|
441
|
+
return renderListItems(items, "ul");
|
|
442
|
+
}
|
|
443
|
+
if (lines.every((l) => isOrderedListItem(l))) {
|
|
444
|
+
const items = parseListItems(lines, 0);
|
|
445
|
+
return renderListItems(items, "ol");
|
|
446
|
+
}
|
|
447
|
+
if (lines.length > 0 && (isUnorderedListItem(lines[0]) || isOrderedListItem(lines[0]))) {
|
|
448
|
+
const isOrdered = isOrderedListItem(lines[0]);
|
|
449
|
+
const items = parseListItems(lines, 0);
|
|
450
|
+
return renderListItems(items, isOrdered ? "ol" : "ul");
|
|
451
|
+
}
|
|
452
|
+
return `<p>${renderInline(block)}</p>`;
|
|
453
|
+
}
|
|
367
454
|
function renderMarkdownToHtml(markdown) {
|
|
368
455
|
if (!markdown) {
|
|
369
456
|
return "";
|
|
@@ -372,22 +459,61 @@ function renderMarkdownToHtml(markdown) {
|
|
|
372
459
|
const withoutCode = replaceCodeBlocks(markdown, codeBlocks);
|
|
373
460
|
const normalized = withoutCode.replace(/\r\n/g, "\n");
|
|
374
461
|
const withHeadingBreaks = normalized.replace(/\n(#{1,6}\s)/g, "\n\n$1").replace(/(#{1,6}\s[^\n]+)\n(?!#|\n)/g, "$1\n\n");
|
|
375
|
-
const
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
if (
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
462
|
+
const rawBlocks = [];
|
|
463
|
+
let currentBlock = [];
|
|
464
|
+
const flushBlock = () => {
|
|
465
|
+
if (currentBlock.length > 0) {
|
|
466
|
+
rawBlocks.push(currentBlock.join("\n").trim());
|
|
467
|
+
currentBlock = [];
|
|
468
|
+
}
|
|
469
|
+
};
|
|
470
|
+
const inputLines = withHeadingBreaks.split("\n");
|
|
471
|
+
let i = 0;
|
|
472
|
+
while (i < inputLines.length) {
|
|
473
|
+
const line = inputLines[i];
|
|
474
|
+
const trimmed = line.trim();
|
|
475
|
+
if (trimmed === "") {
|
|
476
|
+
const prevIsListOrQuote = currentBlock.length > 0 && (isUnorderedListItem(currentBlock[0]) || isOrderedListItem(currentBlock[0]) || currentBlock[0].startsWith("> "));
|
|
477
|
+
if (prevIsListOrQuote) {
|
|
478
|
+
let nextIdx = i + 1;
|
|
479
|
+
while (nextIdx < inputLines.length && inputLines[nextIdx].trim() === "") nextIdx++;
|
|
480
|
+
if (nextIdx < inputLines.length) {
|
|
481
|
+
const nextTrimmed = inputLines[nextIdx].trim();
|
|
482
|
+
const continues = isUnorderedListItem(currentBlock[0]) && isUnorderedListItem(nextTrimmed) || isOrderedListItem(currentBlock[0]) && isOrderedListItem(nextTrimmed) || currentBlock[0].startsWith("> ") && (nextTrimmed.startsWith("> ") || nextTrimmed === ">");
|
|
483
|
+
if (continues) {
|
|
484
|
+
i++;
|
|
485
|
+
continue;
|
|
486
|
+
}
|
|
487
|
+
}
|
|
488
|
+
}
|
|
489
|
+
flushBlock();
|
|
490
|
+
i++;
|
|
491
|
+
continue;
|
|
382
492
|
}
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
493
|
+
if (isUnorderedListItem(trimmed) || isOrderedListItem(trimmed)) {
|
|
494
|
+
if (currentBlock.length > 0 && !isUnorderedListItem(currentBlock[0]) && !isOrderedListItem(currentBlock[0])) {
|
|
495
|
+
flushBlock();
|
|
496
|
+
}
|
|
497
|
+
currentBlock.push(trimmed);
|
|
498
|
+
i++;
|
|
499
|
+
continue;
|
|
388
500
|
}
|
|
389
|
-
|
|
390
|
-
|
|
501
|
+
if (trimmed.startsWith("> ") || trimmed === ">") {
|
|
502
|
+
if (currentBlock.length > 0 && !currentBlock[0].startsWith("> ") && currentBlock[0] !== ">") {
|
|
503
|
+
flushBlock();
|
|
504
|
+
}
|
|
505
|
+
currentBlock.push(trimmed);
|
|
506
|
+
i++;
|
|
507
|
+
continue;
|
|
508
|
+
}
|
|
509
|
+
if (currentBlock.length > 0 && (isUnorderedListItem(currentBlock[0]) || isOrderedListItem(currentBlock[0]) || currentBlock[0].startsWith("> "))) {
|
|
510
|
+
flushBlock();
|
|
511
|
+
}
|
|
512
|
+
currentBlock.push(line);
|
|
513
|
+
i++;
|
|
514
|
+
}
|
|
515
|
+
flushBlock();
|
|
516
|
+
const htmlBlocks = rawBlocks.filter(Boolean).map(renderBlock);
|
|
391
517
|
const html = htmlBlocks.join("\n");
|
|
392
518
|
return restoreCodeBlocks(html, codeBlocks);
|
|
393
519
|
}
|