@sonordev/site-kit 2.2.8 → 2.5.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/dist/SitemapSync-3ISOAVDT.mjs +4 -0
- package/dist/{SitemapSync-XGKC63TH.mjs.map → SitemapSync-3ISOAVDT.mjs.map} +1 -1
- package/dist/SitemapSync-4Y65CMUK.js +13 -0
- package/dist/{SitemapSync-TPA3GTGU.js.map → SitemapSync-4Y65CMUK.js.map} +1 -1
- package/dist/blog/index.d.mts +31 -3
- package/dist/blog/index.d.ts +31 -3
- package/dist/blog/index.js +194 -10
- package/dist/blog/index.js.map +1 -1
- package/dist/blog/index.mjs +183 -4
- package/dist/blog/index.mjs.map +1 -1
- package/dist/blog/server-ui.d.mts +1 -1
- package/dist/blog/server-ui.d.ts +1 -1
- package/dist/blog/server-ui.js +3 -3
- package/dist/blog/server-ui.mjs +1 -1
- package/dist/blog/server.d.mts +79 -7
- package/dist/blog/server.d.ts +79 -7
- package/dist/blog/server.js +64 -32
- package/dist/blog/server.mjs +1 -1
- package/dist/{chunk-WRCX2NKY.mjs → chunk-2NM6RGAV.mjs} +226 -22
- package/dist/chunk-2NM6RGAV.mjs.map +1 -0
- package/dist/chunk-5B4FABFK.js +28 -0
- package/dist/chunk-5B4FABFK.js.map +1 -0
- package/dist/{chunk-DTVZJPVM.mjs → chunk-5SQ4NRPH.mjs} +9 -2
- package/dist/chunk-5SQ4NRPH.mjs.map +1 -0
- package/dist/chunk-ATG4FJY6.js +76 -0
- package/dist/chunk-ATG4FJY6.js.map +1 -0
- package/dist/{chunk-F4VAOBJM.mjs → chunk-DV2BURIN.mjs} +3 -3
- package/dist/{chunk-F4VAOBJM.mjs.map → chunk-DV2BURIN.mjs.map} +1 -1
- package/dist/{chunk-GQKBGL2W.js → chunk-DZKX3GHL.js} +233 -21
- package/dist/chunk-DZKX3GHL.js.map +1 -0
- package/dist/{chunk-LNMI6OMN.js → chunk-F54HGPDM.js} +137 -4
- package/dist/chunk-F54HGPDM.js.map +1 -0
- package/dist/{chunk-DPO3ZPFK.js → chunk-GVDPTXN3.js} +3 -3
- package/dist/{chunk-DPO3ZPFK.js.map → chunk-GVDPTXN3.js.map} +1 -1
- package/dist/chunk-H23ZT2I2.mjs +67 -0
- package/dist/chunk-H23ZT2I2.mjs.map +1 -0
- package/dist/chunk-H4OBGC43.mjs +26 -0
- package/dist/chunk-H4OBGC43.mjs.map +1 -0
- package/dist/{chunk-UHMM22HX.js → chunk-JNXQX2R6.js} +7 -3
- package/dist/chunk-JNXQX2R6.js.map +1 -0
- package/dist/{chunk-Z6EHHJWU.mjs → chunk-MNOVPHL6.mjs} +230 -35
- package/dist/chunk-MNOVPHL6.mjs.map +1 -0
- package/dist/{chunk-ITPVKQB6.js → chunk-MWE2HRPU.js} +229 -34
- package/dist/chunk-MWE2HRPU.js.map +1 -0
- package/dist/{chunk-AWMEH65F.js → chunk-PAF5IGGF.js} +9 -2
- package/dist/chunk-PAF5IGGF.js.map +1 -0
- package/dist/{chunk-OOZCN7AF.mjs → chunk-T5UU7I4V.mjs} +137 -5
- package/dist/chunk-T5UU7I4V.mjs.map +1 -0
- package/dist/{chunk-PU5ULVL5.mjs → chunk-ZUCVEQZB.mjs} +7 -3
- package/dist/chunk-ZUCVEQZB.mjs.map +1 -0
- package/dist/cli/index.js +352 -78
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/index.mjs +352 -78
- package/dist/cli/index.mjs.map +1 -1
- package/dist/config/index.d.mts +17 -0
- package/dist/config/index.d.ts +17 -0
- package/dist/config/index.js +43 -3
- package/dist/config/index.js.map +1 -1
- package/dist/config/index.mjs +43 -3
- package/dist/config/index.mjs.map +1 -1
- package/dist/forms/index.js +3 -1
- package/dist/forms/index.js.map +1 -1
- package/dist/forms/index.mjs +3 -1
- package/dist/forms/index.mjs.map +1 -1
- package/dist/index.d.mts +2 -2
- package/dist/index.d.ts +2 -2
- package/dist/layout/client.js +2 -2
- package/dist/layout/client.mjs +1 -1
- package/dist/layout/index.d.mts +6 -1
- package/dist/layout/index.d.ts +6 -1
- package/dist/layout/index.js +9 -5
- package/dist/layout/index.js.map +1 -1
- package/dist/layout/index.mjs +8 -4
- package/dist/layout/index.mjs.map +1 -1
- package/dist/llms/contract.d.mts +43 -0
- package/dist/llms/contract.d.ts +43 -0
- package/dist/llms/contract.js +41 -0
- package/dist/llms/contract.js.map +1 -0
- package/dist/llms/contract.mjs +4 -0
- package/dist/llms/contract.mjs.map +1 -0
- package/dist/llms/index.d.mts +67 -5
- package/dist/llms/index.d.ts +67 -5
- package/dist/llms/index.js +154 -36
- package/dist/llms/index.js.map +1 -1
- package/dist/llms/index.mjs +107 -27
- package/dist/llms/index.mjs.map +1 -1
- package/dist/middleware/index.d.mts +13 -1
- package/dist/middleware/index.d.ts +13 -1
- package/dist/middleware/index.js +11 -0
- package/dist/middleware/index.js.map +1 -1
- package/dist/middleware/index.mjs +11 -0
- package/dist/middleware/index.mjs.map +1 -1
- package/dist/{routing-ccNYbFLU.d.ts → routing-C7gmHWm9.d.ts} +1 -1
- package/dist/{routing-ebQln7wH.d.mts → routing-trNzR1Pz.d.mts} +1 -1
- package/dist/seo/client.js +2 -2
- package/dist/seo/client.mjs +1 -1
- package/dist/seo/index.d.mts +19 -4
- package/dist/seo/index.d.ts +19 -4
- package/dist/seo/index.js +51 -16
- package/dist/seo/index.js.map +1 -1
- package/dist/seo/index.mjs +43 -9
- package/dist/seo/index.mjs.map +1 -1
- package/dist/seo/server.d.mts +2 -2
- package/dist/seo/server.d.ts +2 -2
- package/dist/seo/server.js +5 -5
- package/dist/seo/server.mjs +1 -1
- package/dist/sitemap/index.d.mts +8 -1
- package/dist/sitemap/index.d.ts +8 -1
- package/dist/sitemap/index.js +35 -11
- package/dist/sitemap/index.js.map +1 -1
- package/dist/sitemap/index.mjs +34 -10
- package/dist/sitemap/index.mjs.map +1 -1
- package/dist/{types-BxzT7yhf.d.ts → types-0NuBL1Gg.d.ts} +34 -0
- package/dist/{types-DWMpAtGy.d.mts → types-5P5B9RgV.d.mts} +57 -1
- package/dist/{types-DWMpAtGy.d.ts → types-5P5B9RgV.d.ts} +57 -1
- package/dist/{types-CGkyylOa.d.mts → types-DYyIAgQg.d.mts} +2 -0
- package/dist/{types-CGkyylOa.d.ts → types-DYyIAgQg.d.ts} +2 -0
- package/dist/{types-BxzT7yhf.d.mts → types-J7Z_FqmV.d.mts} +34 -0
- package/package.json +15 -1
- package/scripts/postinstall.cjs +67 -0
- package/dist/SitemapSync-TPA3GTGU.js +0 -13
- package/dist/SitemapSync-XGKC63TH.mjs +0 -4
- package/dist/chunk-AWMEH65F.js.map +0 -1
- package/dist/chunk-DTVZJPVM.mjs.map +0 -1
- package/dist/chunk-GQKBGL2W.js.map +0 -1
- package/dist/chunk-ITPVKQB6.js.map +0 -1
- package/dist/chunk-LNMI6OMN.js.map +0 -1
- package/dist/chunk-OOZCN7AF.mjs.map +0 -1
- package/dist/chunk-PU5ULVL5.mjs.map +0 -1
- package/dist/chunk-UHMM22HX.js.map +0 -1
- package/dist/chunk-WRCX2NKY.mjs.map +0 -1
- package/dist/chunk-Z6EHHJWU.mjs.map +0 -1
package/dist/seo/index.mjs
CHANGED
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
export { ManagedNoScripts, ManagedScripts } from '../chunk-L7XPPFSS.mjs';
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
2
|
+
import { pickManagedLlmSchemaForJsonLd } from '../chunk-H23ZT2I2.mjs';
|
|
3
|
+
export { SitemapSync } from '../chunk-ZUCVEQZB.mjs';
|
|
4
|
+
import { getManagedMetadata } from '../chunk-5SQ4NRPH.mjs';
|
|
5
|
+
export { generateSitemap, getABVariant, getManagedMetadata, getManagedMetadataWithAB, getRedirect, getRobotsDirective, isIndexable } from '../chunk-5SQ4NRPH.mjs';
|
|
5
6
|
import { getContentBlock, getSchemaMarkups, getSEOPageData, getEntityEnhancedSchema, getFAQData, getInternalLinks, getEntities } from '../chunk-TUFP3FGI.mjs';
|
|
6
7
|
export { getABTest, getContentBlock, getEntities, getEntityEnhancedSchema, getFAQData, getInternalLinks, getManagedScripts, getPrimaryEntity, getRedirectData, getRobotsData, getSEOPageData, getSchemaMarkups, getSitemapEntries, getVisibilityScore, getVisibilitySummary, recordABImpression, registerSitemap } from '../chunk-TUFP3FGI.mjs';
|
|
7
8
|
import '../chunk-4XPGGLVP.mjs';
|
|
@@ -110,17 +111,27 @@ async function ManagedSchemaAsync({
|
|
|
110
111
|
async function LLMSchemaAsync({
|
|
111
112
|
path
|
|
112
113
|
}) {
|
|
113
|
-
const { page: pageData } = await getSEOPageData(path);
|
|
114
|
-
|
|
114
|
+
const { page: pageData, project } = await getSEOPageData(path);
|
|
115
|
+
const raw = pageData?.managed_llm_schema;
|
|
116
|
+
const picked = pickManagedLlmSchemaForJsonLd(raw);
|
|
117
|
+
if (!picked) {
|
|
115
118
|
return null;
|
|
116
119
|
}
|
|
120
|
+
const siteUrl = (project?.site_url || "").replace(/\/$/, "");
|
|
121
|
+
const websiteId = siteUrl ? `${siteUrl}/#website` : void 0;
|
|
117
122
|
const llmSchema = {
|
|
118
123
|
"@context": "https://schema.org",
|
|
119
124
|
"@type": "WebPage",
|
|
120
|
-
...
|
|
121
|
-
// Add AI-specific metadata hints
|
|
125
|
+
...picked,
|
|
122
126
|
additionalType: "https://sonor.io/ns/LLMOptimizedContent"
|
|
123
127
|
};
|
|
128
|
+
if (siteUrl && websiteId) {
|
|
129
|
+
llmSchema.isPartOf = {
|
|
130
|
+
"@type": "WebSite",
|
|
131
|
+
"@id": websiteId,
|
|
132
|
+
url: siteUrl
|
|
133
|
+
};
|
|
134
|
+
}
|
|
124
135
|
return /* @__PURE__ */ jsx(
|
|
125
136
|
"script",
|
|
126
137
|
{
|
|
@@ -128,7 +139,7 @@ async function LLMSchemaAsync({
|
|
|
128
139
|
"data-llm-optimized": "true",
|
|
129
140
|
async: true,
|
|
130
141
|
dangerouslySetInnerHTML: {
|
|
131
|
-
__html: JSON.stringify(llmSchema
|
|
142
|
+
__html: JSON.stringify(llmSchema)
|
|
132
143
|
}
|
|
133
144
|
}
|
|
134
145
|
);
|
|
@@ -190,6 +201,29 @@ function ManagedSchema(props) {
|
|
|
190
201
|
function LLMSchema(props) {
|
|
191
202
|
return /* @__PURE__ */ jsx(Suspense, { fallback: null, children: /* @__PURE__ */ jsx(LLMSchemaAsync, { ...props }) });
|
|
192
203
|
}
|
|
204
|
+
|
|
205
|
+
// src/seo/groundingSchema.ts
|
|
206
|
+
function createWebSiteOrganizationStub(input) {
|
|
207
|
+
const origin = input.url.replace(/\/$/, "");
|
|
208
|
+
const websiteId = `${origin}/#website`;
|
|
209
|
+
const orgId = `${origin}/#organization`;
|
|
210
|
+
const org = {
|
|
211
|
+
"@type": "Organization",
|
|
212
|
+
"@id": orgId,
|
|
213
|
+
name: input.name,
|
|
214
|
+
url: origin,
|
|
215
|
+
...input.sameAs?.length ? { sameAs: input.sameAs } : {},
|
|
216
|
+
...input.knowsAbout?.length ? { knowsAbout: input.knowsAbout } : {}
|
|
217
|
+
};
|
|
218
|
+
const site = {
|
|
219
|
+
"@type": "WebSite",
|
|
220
|
+
"@id": websiteId,
|
|
221
|
+
url: origin,
|
|
222
|
+
name: input.name,
|
|
223
|
+
publisher: { "@id": orgId }
|
|
224
|
+
};
|
|
225
|
+
return [org, site];
|
|
226
|
+
}
|
|
193
227
|
var faqStyles = `
|
|
194
228
|
.sk-faq-items {
|
|
195
229
|
display: flex;
|
|
@@ -639,6 +673,6 @@ function LocationPageContent(props) {
|
|
|
639
673
|
return /* @__PURE__ */ jsx(Suspense, { fallback: props.fallback ?? null, children: /* @__PURE__ */ jsx(LocationPageContentAsync, { ...props }) });
|
|
640
674
|
}
|
|
641
675
|
|
|
642
|
-
export { LLMSchema, LocationPageContent, ManagedContent, ManagedFAQ, ManagedInternalLinks, ManagedSchema, createBreadcrumbSchema, createSchema, getLocationSection, getManagedContentData, withManagedMetadata };
|
|
676
|
+
export { LLMSchema, LocationPageContent, ManagedContent, ManagedFAQ, ManagedInternalLinks, ManagedSchema, createBreadcrumbSchema, createSchema, createWebSiteOrganizationStub, getLocationSection, getManagedContentData, withManagedMetadata };
|
|
643
677
|
//# sourceMappingURL=index.mjs.map
|
|
644
678
|
//# sourceMappingURL=index.mjs.map
|
package/dist/seo/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/seo/withManagedMetadata.ts","../../src/seo/ManagedSchema.tsx","../../src/seo/ManagedFAQ.tsx","../../src/seo/ManagedInternalLinks.tsx","../../src/seo/ManagedContent.tsx","../../src/seo/LocationPageContent.tsx"],"names":["jsx","Suspense","jsxs","Fragment"],"mappings":"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","file":"index.mjs","sourcesContent":["/**\n * @sonordev/site-kit/seo - withManagedMetadata\n *\n * Higher-order function that wraps Next.js generateMetadata to auto-merge\n * Portal-managed SEO data. Portal values serve as defaults; page-level\n * metadata returned by the caller takes priority.\n *\n * @example Zero-config (Portal manages everything):\n * ```ts\n * export const generateMetadata = withManagedMetadata('/about')\n * ```\n *\n * @example With local overrides:\n * ```ts\n * export const generateMetadata = withManagedMetadata('/about', async () => ({\n * title: 'About Us',\n * }))\n * ```\n *\n * @example Dynamic paths:\n * ```ts\n * export const generateMetadata = withManagedMetadata(\n * async ({ params }) => `/services/${(await params).slug}`\n * )\n * ```\n */\n\nimport type { Metadata, ResolvingMetadata } from 'next'\nimport { getManagedMetadata } from './getManagedMetadata'\n\ntype MetadataArgs = {\n params: Promise<Record<string, string>>\n searchParams?: Promise<Record<string, string>>\n}\n\ntype PathResolver =\n | string\n | ((args: MetadataArgs) => string | Promise<string>)\n\ntype PageMetadataFn = (\n args: MetadataArgs,\n parent: ResolvingMetadata\n) => Promise<Metadata> | Metadata\n\n/**\n * Wrap generateMetadata to auto-merge Portal managed metadata.\n *\n * - `path` can be a static string or a function that derives the path\n * from the page's params.\n * - `pageMetadata` (optional) returns page-specific overrides.\n * Anything it returns wins over Portal values.\n */\nexport function withManagedMetadata(\n path: PathResolver,\n pageMetadata?: PageMetadataFn,\n) {\n return async function generateMetadata(\n args: MetadataArgs,\n parent: ResolvingMetadata,\n ): Promise<Metadata> {\n const resolvedPath =\n typeof path === 'function' ? await path(args) : path\n\n // Fetch managed metadata from Portal (acts as defaults)\n const managed = await getManagedMetadata({ path: resolvedPath })\n\n // If no page-specific function, just return managed\n if (!pageMetadata) {\n return managed\n }\n\n // Get page-specific metadata (overrides managed)\n const pageSpecific = await pageMetadata(args, parent)\n\n // Deep merge: page-specific wins over managed\n return {\n ...managed,\n ...pageSpecific,\n openGraph: {\n ...(managed.openGraph as Record<string, unknown> | undefined),\n ...(pageSpecific.openGraph as Record<string, unknown> | undefined),\n },\n twitter: {\n ...(managed.twitter as Record<string, unknown> | undefined),\n ...(pageSpecific.twitter as Record<string, unknown> | undefined),\n },\n } as Metadata\n }\n}\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getSchemaMarkups, getEntityEnhancedSchema, getSEOPageData } from './server-api'\nimport type { ManagedSchemaProps } from './types'\n\n/**\n * Speakable configuration for schema markup\n */\nexport interface SpeakableSpec {\n /** CSS selectors for speakable content */\n cssSelector?: string[]\n /** XPath selectors for speakable content */\n xpath?: string[]\n}\n\n/**\n * Extended schema props with speakable and entity support\n */\nexport interface EnhancedManagedSchemaProps extends ManagedSchemaProps {\n /** Add speakable specification to page schema */\n speakable?: SpeakableSpec | boolean\n /** Page type for speakable (WebPage or Article) */\n pageType?: 'WebPage' | 'Article'\n /** Page name for speakable schema */\n pageName?: string\n /** Page URL for speakable schema */\n pageUrl?: string\n /** Include entity-enhanced schema from knowledge graph (AI Visibility) */\n includeEntityGraph?: boolean\n}\n\n/**\n * Default speakable selectors for common page elements\n */\nexport const DEFAULT_SPEAKABLE_SELECTORS = [\n 'h1',\n '[data-speakable=\"true\"]',\n '.page-summary',\n '.key-points',\n '.aeo-block[data-speakable=\"true\"]',\n]\n\n/**\n * ManagedSchema - Server Component that injects JSON-LD schema\n * \n * Fetches schema markup from Portal and renders as script tags.\n * Now with speakable support for voice assistants and AI systems.\n * \n * @example\n * ```tsx\n * // app/services/[slug]/page.tsx\n * import { ManagedSchema } from '@sonordev/seo'\n * \n * export default async function ServicePage({ params }) {\n * return (\n * <>\n * <ManagedSchema\n * path={`/services/${params.slug}`}\n * speakable={true}\n * pageName=\"Family Law Services\"\n * pageUrl=\"https://example.com/services/family-law\"\n * />\n * <main>...</main>\n * </>\n * )\n * }\n * ```\n */\nasync function ManagedSchemaAsync({\n path,\n additionalSchemas = [],\n includeTypes,\n excludeTypes,\n speakable,\n pageType = 'WebPage',\n pageName,\n pageUrl,\n includeEntityGraph = true,\n}: EnhancedManagedSchemaProps): Promise<React.ReactElement | null> {\n // Fetch all data in parallel — avoids sequential waterfall that delays streaming\n const [schemas, pageData, entitySchemas] = await Promise.all([\n getSchemaMarkups(path, { includeTypes, excludeTypes }),\n getSEOPageData(path),\n includeEntityGraph\n ? getEntityEnhancedSchema(path).catch(() => [] as object[])\n : Promise.resolve([] as object[]),\n ])\n\n const page = pageData?.page\n const siteUrl = pageData?.project?.site_url || ''\n\n // Combine all schemas: entity-enhanced + managed + page-auto-generated + additional\n const allSchemas = [\n ...entitySchemas,\n ...schemas.map((s: { schema_json?: object }) => s.schema_json),\n // Include auto-generated schema from Signal meta optimization\n ...(page?.managed_schema ? [page.managed_schema] : []),\n ...additionalSchemas,\n ]\n\n // Auto-generate BreadcrumbList if none exists and path has segments\n const hasBreadcrumb = allSchemas.some(\n (s) => s && typeof s === 'object' && (s as Record<string, unknown>)['@type'] === 'BreadcrumbList'\n )\n if (!hasBreadcrumb && path !== '/' && siteUrl) {\n allSchemas.push(createBreadcrumbSchema(siteUrl, path))\n }\n\n // Add speakable schema if requested\n if (speakable && pageName && pageUrl) {\n const speakableSchema = createSpeakableWebPageSchema(\n pageType,\n pageName,\n pageUrl,\n typeof speakable === 'object' ? speakable : undefined\n )\n allSchemas.push(speakableSchema)\n }\n\n // If still empty, try entity-enhanced schema as last-resort fallback\n if (allSchemas.length === 0 && !includeEntityGraph) {\n try {\n const fallbackSchemas = await getEntityEnhancedSchema(path)\n allSchemas.push(...fallbackSchemas)\n } catch {\n // Silent fallback\n }\n }\n\n if (allSchemas.length === 0) {\n return null\n }\n\n // If multiple schemas, wrap in @graph\n const schemaContent = allSchemas.length === 1\n ? allSchemas[0]\n : {\n '@context': 'https://schema.org',\n '@graph': allSchemas\n .filter(s => s && typeof s === 'object')\n .map(s => {\n // Remove @context from individual schemas when in graph\n const { '@context': _, ...rest } = s as Record<string, unknown>\n return rest\n }),\n }\n\n // Add @context if not in graph mode\n const finalSchema = allSchemas.length === 1\n ? { '@context': 'https://schema.org', ...schemaContent as Record<string, unknown> }\n : schemaContent\n\n return (\n <script\n type=\"application/ld+json\"\n async\n dangerouslySetInnerHTML={{\n __html: JSON.stringify(finalSchema, null, 0),\n }}\n />\n )\n}\n\n/**\n * LLMSchema - Server Component that injects LLM-optimized structured data\n * \n * This component renders AI-visibility optimized data that helps LLM crawlers\n * (like ChatGPT, Claude, Perplexity) better understand page content.\n * \n * The schema includes:\n * - Detailed description (100-200 words for context)\n * - Keywords and topics\n * - Target audience\n * - Content relationships\n * \n * @example\n * ```tsx\n * import { LLMSchema } from '@sonordev/seo'\n * \n * export default async function ServicePage({ params }) {\n * return (\n * <>\n * <LLMSchema\n * path={`/services/${params.slug}`}\n * />\n * <main>...</main>\n * </>\n * )\n * }\n * ```\n */\nasync function LLMSchemaAsync({\n path,\n}: {\n projectId?: string\n path: string\n}): Promise<React.ReactElement | null> {\n const { page: pageData } = await getSEOPageData(path)\n\n if (!pageData?.managed_llm_schema) {\n return null\n }\n\n // Render as a special JSON-LD type that LLM crawlers can parse\n const llmSchema = {\n '@context': 'https://schema.org',\n '@type': 'WebPage',\n ...pageData.managed_llm_schema,\n // Add AI-specific metadata hints\n additionalType: 'https://sonor.io/ns/LLMOptimizedContent',\n }\n\n return (\n <script\n type=\"application/ld+json\"\n data-llm-optimized=\"true\"\n async\n dangerouslySetInnerHTML={{\n __html: JSON.stringify(llmSchema, null, 0),\n }}\n />\n )\n}\n\n/**\n * Generate schema for a specific type with managed data\n * \n * Helper to create common schema types\n */\nexport function createSchema(\n type: string,\n data: Record<string, unknown>\n): Record<string, unknown> {\n return {\n '@context': 'https://schema.org',\n '@type': type,\n ...data,\n }\n}\n\n/**\n * Create a WebPage or Article schema with speakable specification\n * \n * This helps voice assistants and AI systems identify key content to read aloud.\n * \n * @see https://schema.org/speakable\n * @see https://developers.google.com/search/docs/appearance/structured-data/speakable\n */\nexport function createSpeakableWebPageSchema(\n type: 'WebPage' | 'Article',\n name: string,\n url: string,\n speakable?: SpeakableSpec\n): Record<string, unknown> {\n const speakableSpec: Record<string, unknown> = {\n '@type': 'SpeakableSpecification',\n }\n\n if (speakable?.cssSelector?.length) {\n speakableSpec.cssSelector = speakable.cssSelector\n } else if (speakable?.xpath?.length) {\n speakableSpec.xpath = speakable.xpath\n } else {\n // Use default selectors\n speakableSpec.cssSelector = DEFAULT_SPEAKABLE_SELECTORS\n }\n\n return {\n '@context': 'https://schema.org',\n '@type': type,\n name,\n url,\n speakable: speakableSpec,\n }\n}\n\n/**\n * Create BreadcrumbList schema from path\n */\nexport function createBreadcrumbSchema(\n baseUrl: string,\n path: string,\n labels?: Record<string, string>\n): Record<string, unknown> {\n const segments = path.split('/').filter(Boolean)\n \n const items = segments.map((segment, index) => {\n const itemPath = '/' + segments.slice(0, index + 1).join('/')\n const label = labels?.[segment] || segment.replace(/-/g, ' ').replace(/\\b\\w/g, l => l.toUpperCase())\n \n return {\n '@type': 'ListItem',\n position: index + 1,\n name: label,\n item: `${baseUrl}${itemPath}`,\n }\n })\n\n // Add home as first item\n items.unshift({\n '@type': 'ListItem',\n position: 0,\n name: 'Home',\n item: baseUrl,\n })\n\n // Re-number positions\n items.forEach((item, index) => {\n item.position = index + 1\n })\n\n return createSchema('BreadcrumbList', {\n itemListElement: items,\n })\n}\n\n/**\n * ManagedSchema — Suspense-wrapped so API fetches never block page streaming.\n * The hero and other page content flush immediately; schema scripts stream in when ready.\n */\nexport function ManagedSchema(props: EnhancedManagedSchemaProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedSchemaAsync {...props} />\n </Suspense>\n )\n}\n\n/**\n * LLMSchema — Suspense-wrapped so API fetches never block page streaming.\n */\nexport function LLMSchema(props: { projectId?: string; path: string }): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <LLMSchemaAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedSchema\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getFAQData } from './server-api'\nimport type { ManagedFAQProps, FAQItem } from './types'\nimport { createSchema } from './ManagedSchema'\n\n/**\n * Inline styles for the accordion FAQ (no external CSS dependency)\n */\nconst faqStyles = `\n.sk-faq-items {\n display: flex;\n flex-direction: column;\n gap: 0;\n}\n.sk-faq-item {\n border-bottom: 1px solid rgba(0,0,0,0.08);\n}\n.sk-faq-item:first-child {\n border-top: 1px solid rgba(0,0,0,0.08);\n}\n.sk-faq-item summary {\n display: flex;\n align-items: center;\n justify-content: space-between;\n cursor: pointer;\n padding: 1.25rem 0;\n font-weight: 600;\n font-size: 1.05rem;\n line-height: 1.5;\n color: inherit;\n list-style: none;\n user-select: none;\n transition: color 0.15s ease;\n}\n.sk-faq-item summary:hover {\n opacity: 0.8;\n}\n.sk-faq-item summary::-webkit-details-marker {\n display: none;\n}\n.sk-faq-item summary::marker {\n display: none;\n content: '';\n}\n.sk-faq-chevron {\n flex-shrink: 0;\n width: 1.25rem;\n height: 1.25rem;\n margin-left: 1rem;\n transition: transform 0.2s ease;\n opacity: 0.5;\n}\n.sk-faq-item[open] .sk-faq-chevron {\n transform: rotate(180deg);\n}\n.sk-faq-answer {\n padding: 0 0 1.25rem 0;\n color: rgba(0,0,0,0.6);\n line-height: 1.75;\n font-size: 0.95rem;\n}\n.sk-faq-answer p {\n margin: 0 0 0.75rem 0;\n}\n.sk-faq-answer p:last-child {\n margin-bottom: 0;\n}\n.sk-faq-answer ul, .sk-faq-answer ol {\n padding-left: 1.5rem;\n margin: 0.5rem 0;\n}\n.sk-faq-answer li {\n margin-bottom: 0.25rem;\n}\n.sk-faq-answer a {\n color: inherit;\n text-decoration: underline;\n text-underline-offset: 2px;\n}\n.sk-faq-title {\n font-size: 1.75rem;\n font-weight: 700;\n margin-bottom: 0.5rem;\n}\n.sk-faq-description {\n color: rgba(0,0,0,0.6);\n margin-bottom: 2rem;\n font-size: 1rem;\n line-height: 1.6;\n}\n\n`\n\n/**\n * Chevron SVG (no icon library dependency)\n */\nfunction ChevronDown() {\n return (\n <svg\n className=\"sk-faq-chevron\"\n xmlns=\"http://www.w3.org/2000/svg\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n >\n <polyline points=\"6 9 12 15 18 9\" />\n </svg>\n )\n}\n\n/**\n * Default FAQ item renderer – uses native <details>/<summary> for toggle\n */\nfunction DefaultFAQItem({ item, index }: { item: FAQItem; index: number }) {\n return (\n <details key={item.id} className=\"sk-faq-item\">\n <summary>\n <span>{item.question}</span>\n <ChevronDown />\n </summary>\n <div\n className=\"sk-faq-answer\"\n dangerouslySetInnerHTML={{ __html: item.answer }}\n />\n </details>\n )\n}\n\n/**\n * Generate FAQ schema from items\n * \n * IMPORTANT: This is the ONLY place FAQ schema (FAQPage) is generated.\n * The CLI setup command does NOT generate FAQ schema - it only extracts/uploads\n * FAQ data to the Portal. This component then dynamically generates the schema\n * from that data, ensuring FAQ changes in Portal automatically update the schema.\n */\nfunction generateFAQSchema(items: FAQItem[]): Record<string, unknown> {\n return createSchema('FAQPage', {\n mainEntity: items\n .filter(item => item.is_visible)\n .map(item => ({\n '@type': 'Question',\n name: item.question,\n acceptedAnswer: {\n '@type': 'Answer',\n text: item.answer,\n },\n })),\n })\n}\n\n/**\n * ManagedFAQ - Server Component that renders FAQ section with schema\n * \n * Fetches FAQ content from Portal and renders with optional schema injection\n * \n * @example\n * ```tsx\n * // app/services/plumbing/page.tsx\n * import { ManagedFAQ } from '@sonordev/seo'\n * \n * export default async function PlumbingPage() {\n * return (\n * <main>\n * <h1>Plumbing Services</h1>\n * <section>\n * <ManagedFAQ\n * path=\"/services/plumbing\"\n * showTitle\n * includeSchema\n * />\n * </section>\n * </main>\n * )\n * }\n * ```\n */\nasync function ManagedFAQAsync({\n path,\n className,\n renderItem,\n includeSchema = true,\n showTitle = true,\n}: ManagedFAQProps): Promise<React.ReactElement | null> {\n const faqData = await getFAQData(path)\n\n if (!faqData || !faqData.items?.length) {\n return null\n }\n\n const visibleItems = faqData.items.filter((item: FAQItem) => item.is_visible)\n \n if (visibleItems.length === 0) {\n return null\n }\n\n // Sort by order\n visibleItems.sort((a: FAQItem, b: FAQItem) => a.order - b.order)\n\n const shouldIncludeSchema = includeSchema && faqData.include_schema\n\n return (\n <>\n {shouldIncludeSchema && (\n <script\n type=\"application/ld+json\"\n dangerouslySetInnerHTML={{\n __html: JSON.stringify(generateFAQSchema(visibleItems), null, 0),\n }}\n />\n )}\n {/* Embedded styles for accordion FAQ (no external CSS file needed) */}\n <style dangerouslySetInnerHTML={{ __html: faqStyles }} />\n <div className={className || 'sk-faq'}>\n {showTitle && faqData.title && (\n <h2 className=\"sk-faq-title\">{faqData.title}</h2>\n )}\n {faqData.description && (\n <p className=\"sk-faq-description\">{faqData.description}</p>\n )}\n <div className=\"sk-faq-items\">\n {visibleItems.map((item: FAQItem, index: number) => \n renderItem \n ? renderItem(item, index)\n : <DefaultFAQItem key={item.id} item={item} index={index} />\n )}\n </div>\n </div>\n </>\n )\n}\n\n/**\n * ManagedFAQ — Suspense-wrapped so the FAQ API fetch never blocks page streaming.\n */\nexport function ManagedFAQ(props: ManagedFAQProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedFAQAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedFAQ\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getInternalLinks } from './server-api'\nimport type { ManagedInternalLinksProps, ManagedLink } from './types'\n\n/**\n * Default link renderer\n */\nfunction DefaultLinkRenderer({ link }: { link: ManagedLink }) {\n const href = link.target_url || link.target_path\n \n return (\n <a \n key={link.id}\n href={href}\n className=\"sk-internal-link\"\n >\n {link.anchor_text}\n </a>\n )\n}\n\n/**\n * ManagedInternalLinks - Server Component for AI-suggested internal links\n * \n * Fetches internal link suggestions from Portal and renders them\n * \n * @example\n * ```tsx\n * // In your article component\n * import { ManagedInternalLinks } from '@sonordev/seo'\n * \n * export default async function BlogPost({ params }) {\n * return (\n * <article>\n * <p>Your content here...</p>\n * \n * <ManagedInternalLinks\n * path={`/blog/${params.slug}`}\n * position=\"bottom\"\n * limit={5}\n * />\n * </article>\n * )\n * }\n * ```\n */\nasync function ManagedInternalLinksAsync({\n path,\n position = 'bottom',\n limit = 5,\n className,\n renderLink,\n}: ManagedInternalLinksProps): Promise<React.ReactElement | null> {\n const links = await getInternalLinks(path, { position, limit })\n\n if (!links.length) {\n return null\n }\n\n const containerClass = className || `sk-internal-links sk-internal-links--${position}`\n\n // Different layouts based on position\n if (position === 'inline') {\n // Inline links are meant to be inserted into content\n return (\n <span className={containerClass}>\n {links.map((link: ManagedLink) => \n renderLink ? renderLink(link) : <DefaultLinkRenderer key={link.id} link={link} />\n )}\n </span>\n )\n }\n\n if (position === 'sidebar') {\n return (\n <aside className={containerClass}>\n <h4 className=\"sk-internal-links-title\">Related Pages</h4>\n <ul className=\"sk-internal-links-list\">\n {links.map((link: ManagedLink) => (\n <li key={link.id}>\n {renderLink ? renderLink(link) : <DefaultLinkRenderer link={link} />}\n </li>\n ))}\n </ul>\n </aside>\n )\n }\n\n if (position === 'related') {\n return (\n <nav className={containerClass} aria-label=\"Related content\">\n <h3 className=\"sk-internal-links-title\">You May Also Like</h3>\n <div className=\"sk-internal-links-grid\">\n {links.map((link: ManagedLink) => (\n <div key={link.id} className=\"sk-internal-link-card\">\n {renderLink ? renderLink(link) : <DefaultLinkRenderer link={link} />}\n {link.context && (\n <p className=\"sk-internal-link-context\">{link.context}</p>\n )}\n </div>\n ))}\n </div>\n </nav>\n )\n }\n\n // Default: bottom position\n return (\n <div className={containerClass}>\n <h4 className=\"sk-internal-links-title\">Related Articles</h4>\n <ul className=\"sk-internal-links-list\">\n {links.map((link: ManagedLink) => (\n <li key={link.id}>\n {renderLink ? renderLink(link) : <DefaultLinkRenderer link={link} />}\n </li>\n ))}\n </ul>\n </div>\n )\n}\n\nexport function ManagedInternalLinks(props: ManagedInternalLinksProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedInternalLinksAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedInternalLinks\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getContentBlock, getEntities, getPrimaryEntity } from './server-api'\nimport type { ManagedContentProps, ManagedContentBlock, SEOEntity } from './types'\n\n/**\n * Parse and render markdown content (basic support)\n * For full markdown, use a proper parser in your custom renderer\n */\nfunction renderMarkdown(content: string): string {\n return content\n // Headers\n .replace(/^### (.*$)/gim, '<h3>$1</h3>')\n .replace(/^## (.*$)/gim, '<h2>$1</h2>')\n .replace(/^# (.*$)/gim, '<h1>$1</h1>')\n // Bold\n .replace(/\\*\\*(.*)\\*\\*/gim, '<strong>$1</strong>')\n // Italic\n .replace(/\\*(.*)\\*/gim, '<em>$1</em>')\n // Links\n .replace(/\\[(.*?)\\]\\((.*?)\\)/gim, '<a href=\"$2\">$1</a>')\n // Paragraphs\n .replace(/\\n\\n/gim, '</p><p>')\n .replace(/^(.+)$/gim, '<p>$1</p>')\n}\n\n/**\n * Inject entity annotations into HTML content\n * Wraps entity mentions with data-sonor-entity attributes for knowledge graph linking\n */\nfunction injectEntityAnnotations(html: string, entities: SEOEntity[]): string {\n if (!entities.length) return html\n \n let annotatedHtml = html\n \n // Sort entities by name length (longest first) to avoid partial matches\n const sortedEntities = [...entities].sort((a, b) => b.name.length - a.name.length)\n \n for (const entity of sortedEntities) {\n // Skip very short names to avoid false positives\n if (entity.name.length < 3) continue\n \n // Create case-insensitive regex for entity name\n // Avoid matching inside HTML tags or existing annotations\n const regex = new RegExp(\n `(?<![\\\\w-])(?<!data-sonor-entity=\")${escapeRegExp(entity.name)}(?![\\\\w-])`,\n 'gi'\n )\n \n // Determine schema type for microdata\n const schemaType = getSchemaTypeForEntity(entity.entity_type)\n \n annotatedHtml = annotatedHtml.replace(regex, (match) => {\n return `<span class=\"aeo-entity aeo-entity-${entity.entity_type}\" ` +\n `data-sonor-entity=\"${entity.id}\" ` +\n `data-sonor-entity-type=\"${entity.entity_type}\" ` +\n `data-sonor-entity-name=\"${entity.name}\" ` +\n `itemscope itemtype=\"https://schema.org/${schemaType}\">` +\n `<span itemprop=\"name\">${match}</span></span>`\n })\n }\n \n return annotatedHtml\n}\n\nfunction escapeRegExp(string: string): string {\n return string.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')\n}\n\nfunction getSchemaTypeForEntity(entityType: string): string {\n const typeMap: Record<string, string> = {\n organization: 'Organization',\n person: 'Person',\n service: 'Service',\n product: 'Product',\n location: 'Place',\n concept: 'Thing',\n credential: 'EducationalOccupationalCredential',\n }\n return typeMap[entityType] || 'Thing'\n}\n\n/**\n * ManagedContent - Server Component for CMS-controlled content blocks\n * \n * Fetches content sections from Portal and renders them\n * Supports HTML, Markdown, JSON, and React component references\n * \n * @example\n * ```tsx\n * // Hero section managed by Portal\n * import { ManagedContent } from '@sonordev/seo'\n * \n * export default async function ServicePage({ params }) {\n * return (\n * <main>\n * <ManagedContent\n * path={`/services/${params.slug}`}\n * section=\"hero\"\n * fallback={<DefaultHero />}\n * />\n *\n * <ManagedContent\n * path={`/services/${params.slug}`}\n * section=\"features\"\n * />\n *\n * <ManagedContent\n * path={`/services/${params.slug}`}\n * section=\"cta\"\n * />\n * </main>\n * )\n * }\n * ```\n */\nasync function ManagedContentAsync({\n path,\n section,\n fallback,\n className,\n components = {},\n injectEntityAnnotations: shouldInjectEntities = false,\n}: ManagedContentProps): Promise<React.ReactElement | null> {\n const block = await getContentBlock(path, section)\n\n if (!block) {\n if (fallback) {\n return <>{fallback}</>\n }\n return null\n }\n\n // Fetch entities if annotation is enabled\n let entities: SEOEntity[] = []\n if (shouldInjectEntities) {\n try {\n entities = await getEntities() as SEOEntity[]\n } catch (err) {\n console.warn('@sonordev/seo: Failed to fetch entities for annotation:', err)\n }\n }\n\n const containerClass = className || `sk-content sk-content--${section}`\n\n // Helper to process HTML with optional entity injection\n const processHtml = (html: string): string => {\n if (shouldInjectEntities && entities.length > 0) {\n return injectEntityAnnotations(html, entities)\n }\n return html\n }\n\n // Handle different content types\n switch (block.content_type) {\n case 'html':\n return (\n <div \n className={containerClass}\n dangerouslySetInnerHTML={{ __html: processHtml(block.content as string) }}\n />\n )\n\n case 'markdown':\n const htmlContent = processHtml(renderMarkdown(block.content as string))\n return (\n <div \n className={containerClass}\n dangerouslySetInnerHTML={{ __html: htmlContent }}\n />\n )\n\n case 'json':\n // JSON content for structured data - render as data attributes or custom handling\n const jsonData = typeof block.content === 'string' \n ? JSON.parse(block.content)\n : block.content\n\n return (\n <div \n className={containerClass}\n data-content={JSON.stringify(jsonData)}\n >\n {/* Render JSON structure - customize based on your needs */}\n {jsonData.title && <h2>{jsonData.title}</h2>}\n {jsonData.subtitle && <p className=\"subtitle\">{jsonData.subtitle}</p>}\n {jsonData.content && <div dangerouslySetInnerHTML={{ __html: jsonData.content }} />}\n {jsonData.items && (\n <ul>\n {jsonData.items.map((item: { text: string } | string, index: number) => (\n <li key={index}>{typeof item === 'string' ? item : item.text}</li>\n ))}\n </ul>\n )}\n </div>\n )\n\n case 'react':\n // React component reference - lookup from provided components map\n const componentData = typeof block.content === 'string'\n ? JSON.parse(block.content)\n : block.content as Record<string, unknown>\n\n const componentName = componentData.component as string\n const componentProps = componentData.props as Record<string, unknown> || {}\n\n const Component = components[componentName]\n \n if (!Component) {\n console.warn(`@sonordev/seo: Component \"${componentName}\" not found in components map`)\n return fallback ? <>{fallback}</> : null\n }\n\n return (\n <div className={containerClass}>\n <Component {...componentProps} />\n </div>\n )\n\n default:\n console.warn(`@sonordev/seo: Unknown content type \"${block.content_type}\"`)\n return fallback ? <>{fallback}</> : null\n }\n}\n\n/**\n * Get content block data without rendering\n * \n * Useful when you need to access the raw data\n */\nexport async function getManagedContentData(\n path: string,\n section: string\n): Promise<ManagedContentBlock | null> {\n return getContentBlock(path, section)\n}\n\nexport function ManagedContent(props: ManagedContentProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedContentAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedContent\n","/**\n * LocationPageContent - Server Component for fetching location page sections\n * \n * Usage:\n * <LocationPageContent \n * projectId=\"uuid\" \n * path=\"/areas/seattle-wa/plumbing\" \n * section=\"hero\" \n * />\n * \n * This component fetches content from seo_location_pages.sections for a given\n * path and section, making it editable in Portal without code deploys.\n */\n\nimport * as React from 'react'\nimport { cache, Suspense } from 'react'\n\n// ============================================\n// Types\n// ============================================\n\nexport interface LocationPageContentProps {\n /** Sonor project ID */\n projectId: string\n /** Page path (e.g., /areas/seattle-wa/plumbing) */\n path: string\n /** Section ID or type to fetch (e.g., \"hero\", \"intro\", \"services\") */\n section: string\n /** Fallback content if section not found */\n fallback?: React.ReactNode\n /** Additional className for wrapper */\n className?: string\n /** Custom renderer for the section data */\n render?: (data: LocationSectionData) => React.ReactNode\n}\n\nexport interface LocationSectionData {\n type: string\n content: any\n}\n\nexport interface HeroSectionContent {\n title: string\n subtitle?: string\n cta_text?: string\n cta_href?: string\n stats?: Array<{ label: string; value: string }>\n background_image?: string\n}\n\nexport interface ServiceGridContent {\n services: Array<{\n title: string\n description: string\n href: string\n icon?: string\n }>\n}\n\nexport interface TextSectionContent {\n heading?: string\n paragraphs: string[]\n html?: string\n}\n\n// ============================================\n// API Config\n// ============================================\n\nfunction getApiConfig() {\n const apiUrl = process.env.SONOR_API_URL || 'https://api.sonor.io'\n return { apiUrl }\n}\n\n// ============================================\n// Cached API Call\n// ============================================\n\n/**\n * Fetch location page section content from Portal\n * Cached with React's cache() for request deduplication\n */\nexport const getLocationSection = cache(async (\n projectId: string,\n path: string,\n section: string\n): Promise<LocationSectionData | null> => {\n const { apiUrl } = getApiConfig()\n \n try {\n const response = await fetch(`${apiUrl}/api/public/seo/location-content`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n project_id: projectId,\n path,\n section,\n }),\n next: { revalidate: 3600 }, // Cache for 1 hour\n })\n \n if (!response.ok) {\n console.error(`LocationPageContent: Failed to fetch section \"${section}\" for path \"${path}\"`)\n return null\n }\n \n const data = await response.json()\n return data\n } catch (error) {\n console.error('LocationPageContent: API error:', error)\n return null\n }\n})\n\n// ============================================\n// Default Renderers\n// ============================================\n\nfunction HeroRenderer({ data, className }: { data: HeroSectionContent; className?: string }) {\n return (\n <section className={className || 'location-hero'}>\n <h1>{data.title}</h1>\n {data.subtitle && <p className=\"subtitle\">{data.subtitle}</p>}\n {data.stats && (\n <div className=\"stats-grid\">\n {data.stats.map((stat, i) => (\n <div key={i} className=\"stat\">\n <span className=\"stat-value\">{stat.value}</span>\n <span className=\"stat-label\">{stat.label}</span>\n </div>\n ))}\n </div>\n )}\n {data.cta_text && data.cta_href && (\n <a href={data.cta_href} className=\"cta-button\">\n {data.cta_text}\n </a>\n )}\n </section>\n )\n}\n\nfunction TextRenderer({ data, className }: { data: TextSectionContent; className?: string }) {\n if (data.html) {\n return (\n <section \n className={className || 'location-text'} \n dangerouslySetInnerHTML={{ __html: data.html }} \n />\n )\n }\n \n return (\n <section className={className || 'location-text'}>\n {data.heading && <h2>{data.heading}</h2>}\n {data.paragraphs.map((p, i) => (\n <p key={i}>{p}</p>\n ))}\n </section>\n )\n}\n\nfunction ServicesGridRenderer({ data, className }: { data: ServiceGridContent; className?: string }) {\n return (\n <section className={className || 'location-services-grid'}>\n <div className=\"services-list\">\n {data.services.map((service, i) => (\n <a key={i} href={service.href} className=\"service-card\">\n {service.icon && <span className=\"service-icon\">{service.icon}</span>}\n <h3>{service.title}</h3>\n <p>{service.description}</p>\n </a>\n ))}\n </div>\n </section>\n )\n}\n\nfunction DefaultRenderer({ data, className }: { data: any; className?: string }) {\n // Fallback: just render as JSON for debugging\n if (process.env.NODE_ENV === 'development') {\n return (\n <section className={className}>\n <pre style={{ fontSize: '12px', background: '#f0f0f0', padding: '1rem' }}>\n {JSON.stringify(data, null, 2)}\n </pre>\n </section>\n )\n }\n \n // In production, try to render content as text/html\n if (typeof data.content === 'string') {\n return (\n <section \n className={className} \n dangerouslySetInnerHTML={{ __html: data.content }} \n />\n )\n }\n \n return null\n}\n\n// ============================================\n// Main Component\n// ============================================\n\n/**\n * LocationPageContent - Server Component\n * \n * Fetches and renders a section of a location page from Portal.\n * Content is fully editable in Portal → SEO → Location Pages.\n */\nasync function LocationPageContentAsync({\n projectId,\n path,\n section,\n fallback = null,\n className,\n render,\n}: LocationPageContentProps): Promise<React.ReactElement | null> {\n const data = await getLocationSection(projectId, path, section)\n \n if (!data) {\n return fallback as React.ReactElement | null\n }\n \n // If custom renderer provided, use it\n if (render) {\n return <>{render(data)}</>\n }\n \n // Use default renderers based on section type\n switch (data.type) {\n case 'hero':\n return <HeroRenderer data={data.content as HeroSectionContent} className={className} />\n \n case 'text':\n case 'intro':\n case 'about':\n case 'about_location':\n return <TextRenderer data={data.content as TextSectionContent} className={className} />\n \n case 'services_grid':\n case 'services':\n return <ServicesGridRenderer data={data.content as ServiceGridContent} className={className} />\n \n default:\n return <DefaultRenderer data={data} className={className} />\n }\n}\n\n/**\n * LocationPageContent — Suspense-wrapped so API fetches never block page streaming.\n */\nexport function LocationPageContent(props: LocationPageContentProps): React.ReactElement {\n return (\n <Suspense fallback={props.fallback as React.ReactElement ?? null}>\n <LocationPageContentAsync {...props} />\n </Suspense>\n )\n}\n\n// Export types for external use\nexport type {\n HeroSectionContent as LocationHeroContent,\n ServiceGridContent as LocationServicesContent,\n TextSectionContent as LocationTextContent,\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../../src/seo/withManagedMetadata.ts","../../src/seo/ManagedSchema.tsx","../../src/seo/groundingSchema.ts","../../src/seo/ManagedFAQ.tsx","../../src/seo/ManagedInternalLinks.tsx","../../src/seo/ManagedContent.tsx","../../src/seo/LocationPageContent.tsx"],"names":["jsx","Suspense","jsxs","Fragment"],"mappings":"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","file":"index.mjs","sourcesContent":["/**\n * @sonordev/site-kit/seo - withManagedMetadata\n *\n * Higher-order function that wraps Next.js generateMetadata to auto-merge\n * Portal-managed SEO data. Portal values serve as defaults; page-level\n * metadata returned by the caller takes priority.\n *\n * @example Zero-config (Portal manages everything):\n * ```ts\n * export const generateMetadata = withManagedMetadata('/about')\n * ```\n *\n * @example With local overrides:\n * ```ts\n * export const generateMetadata = withManagedMetadata('/about', async () => ({\n * title: 'About Us',\n * }))\n * ```\n *\n * @example Dynamic paths:\n * ```ts\n * export const generateMetadata = withManagedMetadata(\n * async ({ params }) => `/services/${(await params).slug}`\n * )\n * ```\n */\n\nimport type { Metadata, ResolvingMetadata } from 'next'\nimport { getManagedMetadata } from './getManagedMetadata'\n\ntype MetadataArgs = {\n params: Promise<Record<string, string>>\n searchParams?: Promise<Record<string, string>>\n}\n\ntype PathResolver =\n | string\n | ((args: MetadataArgs) => string | Promise<string>)\n\ntype PageMetadataFn = (\n args: MetadataArgs,\n parent: ResolvingMetadata\n) => Promise<Metadata> | Metadata\n\n/**\n * Wrap generateMetadata to auto-merge Portal managed metadata.\n *\n * - `path` can be a static string or a function that derives the path\n * from the page's params.\n * - `pageMetadata` (optional) returns page-specific overrides.\n * Anything it returns wins over Portal values.\n */\nexport function withManagedMetadata(\n path: PathResolver,\n pageMetadata?: PageMetadataFn,\n) {\n return async function generateMetadata(\n args: MetadataArgs,\n parent: ResolvingMetadata,\n ): Promise<Metadata> {\n const resolvedPath =\n typeof path === 'function' ? await path(args) : path\n\n // Fetch managed metadata from Portal (acts as defaults)\n const managed = await getManagedMetadata({ path: resolvedPath })\n\n // If no page-specific function, just return managed\n if (!pageMetadata) {\n return managed\n }\n\n // Get page-specific metadata (overrides managed)\n const pageSpecific = await pageMetadata(args, parent)\n\n // Deep merge: page-specific wins over managed\n return {\n ...managed,\n ...pageSpecific,\n openGraph: {\n ...(managed.openGraph as Record<string, unknown> | undefined),\n ...(pageSpecific.openGraph as Record<string, unknown> | undefined),\n },\n twitter: {\n ...(managed.twitter as Record<string, unknown> | undefined),\n ...(pageSpecific.twitter as Record<string, unknown> | undefined),\n },\n } as Metadata\n }\n}\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getSchemaMarkups, getEntityEnhancedSchema, getSEOPageData } from './server-api'\nimport type { ManagedSchemaProps } from './types'\nimport { pickManagedLlmSchemaForJsonLd } from '../llms/contract'\n\n/**\n * Speakable configuration for schema markup\n */\nexport interface SpeakableSpec {\n /** CSS selectors for speakable content */\n cssSelector?: string[]\n /** XPath selectors for speakable content */\n xpath?: string[]\n}\n\n/**\n * Extended schema props with speakable and entity support\n */\nexport interface EnhancedManagedSchemaProps extends ManagedSchemaProps {\n /** Add speakable specification to page schema */\n speakable?: SpeakableSpec | boolean\n /** Page type for speakable (WebPage or Article) */\n pageType?: 'WebPage' | 'Article'\n /** Page name for speakable schema */\n pageName?: string\n /** Page URL for speakable schema */\n pageUrl?: string\n /** Include entity-enhanced schema from knowledge graph (AI Visibility) */\n includeEntityGraph?: boolean\n}\n\n/**\n * Default speakable selectors for common page elements\n */\nexport const DEFAULT_SPEAKABLE_SELECTORS = [\n 'h1',\n '[data-speakable=\"true\"]',\n '.page-summary',\n '.key-points',\n '.aeo-block[data-speakable=\"true\"]',\n]\n\n/**\n * ManagedSchema - Server Component that injects JSON-LD schema\n * \n * Fetches schema markup from Portal and renders as script tags.\n * Now with speakable support for voice assistants and AI systems.\n * \n * @example\n * ```tsx\n * // app/services/[slug]/page.tsx\n * import { ManagedSchema } from '@sonordev/seo'\n * \n * export default async function ServicePage({ params }) {\n * return (\n * <>\n * <ManagedSchema\n * path={`/services/${params.slug}`}\n * speakable={true}\n * pageName=\"Family Law Services\"\n * pageUrl=\"https://example.com/services/family-law\"\n * />\n * <main>...</main>\n * </>\n * )\n * }\n * ```\n */\nasync function ManagedSchemaAsync({\n path,\n additionalSchemas = [],\n includeTypes,\n excludeTypes,\n speakable,\n pageType = 'WebPage',\n pageName,\n pageUrl,\n includeEntityGraph = true,\n}: EnhancedManagedSchemaProps): Promise<React.ReactElement | null> {\n // Fetch all data in parallel — avoids sequential waterfall that delays streaming\n const [schemas, pageData, entitySchemas] = await Promise.all([\n getSchemaMarkups(path, { includeTypes, excludeTypes }),\n getSEOPageData(path),\n includeEntityGraph\n ? getEntityEnhancedSchema(path).catch(() => [] as object[])\n : Promise.resolve([] as object[]),\n ])\n\n const page = pageData?.page\n const siteUrl = pageData?.project?.site_url || ''\n\n // Combine all schemas: entity-enhanced + managed + page-auto-generated + additional\n const allSchemas = [\n ...entitySchemas,\n ...schemas.map((s: { schema_json?: object }) => s.schema_json),\n // Include auto-generated schema from Signal meta optimization\n ...(page?.managed_schema ? [page.managed_schema] : []),\n ...additionalSchemas,\n ]\n\n // Auto-generate BreadcrumbList if none exists and path has segments\n const hasBreadcrumb = allSchemas.some(\n (s) => s && typeof s === 'object' && (s as Record<string, unknown>)['@type'] === 'BreadcrumbList'\n )\n if (!hasBreadcrumb && path !== '/' && siteUrl) {\n allSchemas.push(createBreadcrumbSchema(siteUrl, path))\n }\n\n // Add speakable schema if requested\n if (speakable && pageName && pageUrl) {\n const speakableSchema = createSpeakableWebPageSchema(\n pageType,\n pageName,\n pageUrl,\n typeof speakable === 'object' ? speakable : undefined\n )\n allSchemas.push(speakableSchema)\n }\n\n // If still empty, try entity-enhanced schema as last-resort fallback\n if (allSchemas.length === 0 && !includeEntityGraph) {\n try {\n const fallbackSchemas = await getEntityEnhancedSchema(path)\n allSchemas.push(...fallbackSchemas)\n } catch {\n // Silent fallback\n }\n }\n\n if (allSchemas.length === 0) {\n return null\n }\n\n // If multiple schemas, wrap in @graph\n const schemaContent = allSchemas.length === 1\n ? allSchemas[0]\n : {\n '@context': 'https://schema.org',\n '@graph': allSchemas\n .filter(s => s && typeof s === 'object')\n .map(s => {\n // Remove @context from individual schemas when in graph\n const { '@context': _, ...rest } = s as Record<string, unknown>\n return rest\n }),\n }\n\n // Add @context if not in graph mode\n const finalSchema = allSchemas.length === 1\n ? { '@context': 'https://schema.org', ...schemaContent as Record<string, unknown> }\n : schemaContent\n\n return (\n <script\n type=\"application/ld+json\"\n async\n dangerouslySetInnerHTML={{\n __html: JSON.stringify(finalSchema, null, 0),\n }}\n />\n )\n}\n\n/**\n * LLMSchema - Server Component that injects LLM-optimized structured data\n * \n * This component renders AI-visibility optimized data that helps LLM crawlers\n * (like ChatGPT, Claude, Perplexity) better understand page content.\n * \n * The schema includes:\n * - Detailed description (100-200 words for context)\n * - Keywords and topics\n * - Target audience\n * - Content relationships\n * \n * @example\n * ```tsx\n * import { LLMSchema } from '@sonordev/seo'\n * \n * export default async function ServicePage({ params }) {\n * return (\n * <>\n * <LLMSchema\n * path={`/services/${params.slug}`}\n * />\n * <main>...</main>\n * </>\n * )\n * }\n * ```\n */\nasync function LLMSchemaAsync({\n path,\n}: {\n projectId?: string\n path: string\n}): Promise<React.ReactElement | null> {\n const { page: pageData, project } = await getSEOPageData(path)\n\n const raw = pageData?.managed_llm_schema as Record<string, unknown> | undefined\n const picked = pickManagedLlmSchemaForJsonLd(raw)\n if (!picked) {\n return null\n }\n\n const siteUrl = (project?.site_url || '').replace(/\\/$/, '')\n const websiteId = siteUrl ? `${siteUrl}/#website` : undefined\n\n // Render as a special JSON-LD type that LLM crawlers can parse\n const llmSchema: Record<string, unknown> = {\n '@context': 'https://schema.org',\n '@type': 'WebPage',\n ...picked,\n additionalType: 'https://sonor.io/ns/LLMOptimizedContent',\n }\n\n if (siteUrl && websiteId) {\n llmSchema.isPartOf = {\n '@type': 'WebSite',\n '@id': websiteId,\n url: siteUrl,\n }\n }\n\n return (\n <script\n type=\"application/ld+json\"\n data-llm-optimized=\"true\"\n async\n dangerouslySetInnerHTML={{\n __html: JSON.stringify(llmSchema),\n }}\n />\n )\n}\n\n/**\n * Generate schema for a specific type with managed data\n * \n * Helper to create common schema types\n */\nexport function createSchema(\n type: string,\n data: Record<string, unknown>\n): Record<string, unknown> {\n return {\n '@context': 'https://schema.org',\n '@type': type,\n ...data,\n }\n}\n\n/**\n * Create a WebPage or Article schema with speakable specification\n * \n * This helps voice assistants and AI systems identify key content to read aloud.\n * \n * @see https://schema.org/speakable\n * @see https://developers.google.com/search/docs/appearance/structured-data/speakable\n */\nexport function createSpeakableWebPageSchema(\n type: 'WebPage' | 'Article',\n name: string,\n url: string,\n speakable?: SpeakableSpec\n): Record<string, unknown> {\n const speakableSpec: Record<string, unknown> = {\n '@type': 'SpeakableSpecification',\n }\n\n if (speakable?.cssSelector?.length) {\n speakableSpec.cssSelector = speakable.cssSelector\n } else if (speakable?.xpath?.length) {\n speakableSpec.xpath = speakable.xpath\n } else {\n // Use default selectors\n speakableSpec.cssSelector = DEFAULT_SPEAKABLE_SELECTORS\n }\n\n return {\n '@context': 'https://schema.org',\n '@type': type,\n name,\n url,\n speakable: speakableSpec,\n }\n}\n\n/**\n * Create BreadcrumbList schema from path\n */\nexport function createBreadcrumbSchema(\n baseUrl: string,\n path: string,\n labels?: Record<string, string>\n): Record<string, unknown> {\n const segments = path.split('/').filter(Boolean)\n \n const items = segments.map((segment, index) => {\n const itemPath = '/' + segments.slice(0, index + 1).join('/')\n const label = labels?.[segment] || segment.replace(/-/g, ' ').replace(/\\b\\w/g, l => l.toUpperCase())\n \n return {\n '@type': 'ListItem',\n position: index + 1,\n name: label,\n item: `${baseUrl}${itemPath}`,\n }\n })\n\n // Add home as first item\n items.unshift({\n '@type': 'ListItem',\n position: 0,\n name: 'Home',\n item: baseUrl,\n })\n\n // Re-number positions\n items.forEach((item, index) => {\n item.position = index + 1\n })\n\n return createSchema('BreadcrumbList', {\n itemListElement: items,\n })\n}\n\n/**\n * ManagedSchema — Suspense-wrapped so API fetches never block page streaming.\n * The hero and other page content flush immediately; schema scripts stream in when ready.\n */\nexport function ManagedSchema(props: EnhancedManagedSchemaProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedSchemaAsync {...props} />\n </Suspense>\n )\n}\n\n/**\n * LLMSchema — Suspense-wrapped so API fetches never block page streaming.\n */\nexport function LLMSchema(props: { projectId?: string; path: string }): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <LLMSchemaAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedSchema\n","/**\n * Optional JSON-LD stub when Portal does **not** emit Organization/WebSite.\n * Default product stance: Portal owns entity nodes; only use this when confirmed absent.\n */\n\nexport interface WebSiteOrganizationStubInput {\n name: string\n url: string\n sameAs?: string[]\n knowsAbout?: string[]\n}\n\n/**\n * Minimal WebSite + Organization pair with stable @id anchors for isPartOf links.\n */\nexport function createWebSiteOrganizationStub(\n input: WebSiteOrganizationStubInput,\n): Record<string, unknown>[] {\n const origin = input.url.replace(/\\/$/, '')\n const websiteId = `${origin}/#website`\n const orgId = `${origin}/#organization`\n const org: Record<string, unknown> = {\n '@type': 'Organization',\n '@id': orgId,\n name: input.name,\n url: origin,\n ...(input.sameAs?.length ? { sameAs: input.sameAs } : {}),\n ...(input.knowsAbout?.length ? { knowsAbout: input.knowsAbout } : {}),\n }\n const site: Record<string, unknown> = {\n '@type': 'WebSite',\n '@id': websiteId,\n url: origin,\n name: input.name,\n publisher: { '@id': orgId },\n }\n return [org, site]\n}\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getFAQData } from './server-api'\nimport type { ManagedFAQProps, FAQItem } from './types'\nimport { createSchema } from './ManagedSchema'\n\n/**\n * Inline styles for the accordion FAQ (no external CSS dependency)\n */\nconst faqStyles = `\n.sk-faq-items {\n display: flex;\n flex-direction: column;\n gap: 0;\n}\n.sk-faq-item {\n border-bottom: 1px solid rgba(0,0,0,0.08);\n}\n.sk-faq-item:first-child {\n border-top: 1px solid rgba(0,0,0,0.08);\n}\n.sk-faq-item summary {\n display: flex;\n align-items: center;\n justify-content: space-between;\n cursor: pointer;\n padding: 1.25rem 0;\n font-weight: 600;\n font-size: 1.05rem;\n line-height: 1.5;\n color: inherit;\n list-style: none;\n user-select: none;\n transition: color 0.15s ease;\n}\n.sk-faq-item summary:hover {\n opacity: 0.8;\n}\n.sk-faq-item summary::-webkit-details-marker {\n display: none;\n}\n.sk-faq-item summary::marker {\n display: none;\n content: '';\n}\n.sk-faq-chevron {\n flex-shrink: 0;\n width: 1.25rem;\n height: 1.25rem;\n margin-left: 1rem;\n transition: transform 0.2s ease;\n opacity: 0.5;\n}\n.sk-faq-item[open] .sk-faq-chevron {\n transform: rotate(180deg);\n}\n.sk-faq-answer {\n padding: 0 0 1.25rem 0;\n color: rgba(0,0,0,0.6);\n line-height: 1.75;\n font-size: 0.95rem;\n}\n.sk-faq-answer p {\n margin: 0 0 0.75rem 0;\n}\n.sk-faq-answer p:last-child {\n margin-bottom: 0;\n}\n.sk-faq-answer ul, .sk-faq-answer ol {\n padding-left: 1.5rem;\n margin: 0.5rem 0;\n}\n.sk-faq-answer li {\n margin-bottom: 0.25rem;\n}\n.sk-faq-answer a {\n color: inherit;\n text-decoration: underline;\n text-underline-offset: 2px;\n}\n.sk-faq-title {\n font-size: 1.75rem;\n font-weight: 700;\n margin-bottom: 0.5rem;\n}\n.sk-faq-description {\n color: rgba(0,0,0,0.6);\n margin-bottom: 2rem;\n font-size: 1rem;\n line-height: 1.6;\n}\n\n`\n\n/**\n * Chevron SVG (no icon library dependency)\n */\nfunction ChevronDown() {\n return (\n <svg\n className=\"sk-faq-chevron\"\n xmlns=\"http://www.w3.org/2000/svg\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n >\n <polyline points=\"6 9 12 15 18 9\" />\n </svg>\n )\n}\n\n/**\n * Default FAQ item renderer – uses native <details>/<summary> for toggle\n */\nfunction DefaultFAQItem({ item, index }: { item: FAQItem; index: number }) {\n return (\n <details key={item.id} className=\"sk-faq-item\">\n <summary>\n <span>{item.question}</span>\n <ChevronDown />\n </summary>\n <div\n className=\"sk-faq-answer\"\n dangerouslySetInnerHTML={{ __html: item.answer }}\n />\n </details>\n )\n}\n\n/**\n * Generate FAQ schema from items\n * \n * IMPORTANT: This is the ONLY place FAQ schema (FAQPage) is generated.\n * The CLI setup command does NOT generate FAQ schema - it only extracts/uploads\n * FAQ data to the Portal. This component then dynamically generates the schema\n * from that data, ensuring FAQ changes in Portal automatically update the schema.\n */\nfunction generateFAQSchema(items: FAQItem[]): Record<string, unknown> {\n return createSchema('FAQPage', {\n mainEntity: items\n .filter(item => item.is_visible)\n .map(item => ({\n '@type': 'Question',\n name: item.question,\n acceptedAnswer: {\n '@type': 'Answer',\n text: item.answer,\n },\n })),\n })\n}\n\n/**\n * ManagedFAQ - Server Component that renders FAQ section with schema\n * \n * Fetches FAQ content from Portal and renders with optional schema injection\n * \n * @example\n * ```tsx\n * // app/services/plumbing/page.tsx\n * import { ManagedFAQ } from '@sonordev/seo'\n * \n * export default async function PlumbingPage() {\n * return (\n * <main>\n * <h1>Plumbing Services</h1>\n * <section>\n * <ManagedFAQ\n * path=\"/services/plumbing\"\n * showTitle\n * includeSchema\n * />\n * </section>\n * </main>\n * )\n * }\n * ```\n */\nasync function ManagedFAQAsync({\n path,\n className,\n renderItem,\n includeSchema = true,\n showTitle = true,\n}: ManagedFAQProps): Promise<React.ReactElement | null> {\n const faqData = await getFAQData(path)\n\n if (!faqData || !faqData.items?.length) {\n return null\n }\n\n const visibleItems = faqData.items.filter((item: FAQItem) => item.is_visible)\n \n if (visibleItems.length === 0) {\n return null\n }\n\n // Sort by order\n visibleItems.sort((a: FAQItem, b: FAQItem) => a.order - b.order)\n\n const shouldIncludeSchema = includeSchema && faqData.include_schema\n\n return (\n <>\n {shouldIncludeSchema && (\n <script\n type=\"application/ld+json\"\n dangerouslySetInnerHTML={{\n __html: JSON.stringify(generateFAQSchema(visibleItems), null, 0),\n }}\n />\n )}\n {/* Embedded styles for accordion FAQ (no external CSS file needed) */}\n <style dangerouslySetInnerHTML={{ __html: faqStyles }} />\n <div className={className || 'sk-faq'}>\n {showTitle && faqData.title && (\n <h2 className=\"sk-faq-title\">{faqData.title}</h2>\n )}\n {faqData.description && (\n <p className=\"sk-faq-description\">{faqData.description}</p>\n )}\n <div className=\"sk-faq-items\">\n {visibleItems.map((item: FAQItem, index: number) => \n renderItem \n ? renderItem(item, index)\n : <DefaultFAQItem key={item.id} item={item} index={index} />\n )}\n </div>\n </div>\n </>\n )\n}\n\n/**\n * ManagedFAQ — Suspense-wrapped so the FAQ API fetch never blocks page streaming.\n */\nexport function ManagedFAQ(props: ManagedFAQProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedFAQAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedFAQ\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getInternalLinks } from './server-api'\nimport type { ManagedInternalLinksProps, ManagedLink } from './types'\n\n/**\n * Default link renderer\n */\nfunction DefaultLinkRenderer({ link }: { link: ManagedLink }) {\n const href = link.target_url || link.target_path\n \n return (\n <a \n key={link.id}\n href={href}\n className=\"sk-internal-link\"\n >\n {link.anchor_text}\n </a>\n )\n}\n\n/**\n * ManagedInternalLinks - Server Component for AI-suggested internal links\n * \n * Fetches internal link suggestions from Portal and renders them\n * \n * @example\n * ```tsx\n * // In your article component\n * import { ManagedInternalLinks } from '@sonordev/seo'\n * \n * export default async function BlogPost({ params }) {\n * return (\n * <article>\n * <p>Your content here...</p>\n * \n * <ManagedInternalLinks\n * path={`/blog/${params.slug}`}\n * position=\"bottom\"\n * limit={5}\n * />\n * </article>\n * )\n * }\n * ```\n */\nasync function ManagedInternalLinksAsync({\n path,\n position = 'bottom',\n limit = 5,\n className,\n renderLink,\n}: ManagedInternalLinksProps): Promise<React.ReactElement | null> {\n const links = await getInternalLinks(path, { position, limit })\n\n if (!links.length) {\n return null\n }\n\n const containerClass = className || `sk-internal-links sk-internal-links--${position}`\n\n // Different layouts based on position\n if (position === 'inline') {\n // Inline links are meant to be inserted into content\n return (\n <span className={containerClass}>\n {links.map((link: ManagedLink) => \n renderLink ? renderLink(link) : <DefaultLinkRenderer key={link.id} link={link} />\n )}\n </span>\n )\n }\n\n if (position === 'sidebar') {\n return (\n <aside className={containerClass}>\n <h4 className=\"sk-internal-links-title\">Related Pages</h4>\n <ul className=\"sk-internal-links-list\">\n {links.map((link: ManagedLink) => (\n <li key={link.id}>\n {renderLink ? renderLink(link) : <DefaultLinkRenderer link={link} />}\n </li>\n ))}\n </ul>\n </aside>\n )\n }\n\n if (position === 'related') {\n return (\n <nav className={containerClass} aria-label=\"Related content\">\n <h3 className=\"sk-internal-links-title\">You May Also Like</h3>\n <div className=\"sk-internal-links-grid\">\n {links.map((link: ManagedLink) => (\n <div key={link.id} className=\"sk-internal-link-card\">\n {renderLink ? renderLink(link) : <DefaultLinkRenderer link={link} />}\n {link.context && (\n <p className=\"sk-internal-link-context\">{link.context}</p>\n )}\n </div>\n ))}\n </div>\n </nav>\n )\n }\n\n // Default: bottom position\n return (\n <div className={containerClass}>\n <h4 className=\"sk-internal-links-title\">Related Articles</h4>\n <ul className=\"sk-internal-links-list\">\n {links.map((link: ManagedLink) => (\n <li key={link.id}>\n {renderLink ? renderLink(link) : <DefaultLinkRenderer link={link} />}\n </li>\n ))}\n </ul>\n </div>\n )\n}\n\nexport function ManagedInternalLinks(props: ManagedInternalLinksProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedInternalLinksAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedInternalLinks\n","import * as React from 'react'\nimport { Suspense } from 'react'\nimport { getContentBlock, getEntities, getPrimaryEntity } from './server-api'\nimport type { ManagedContentProps, ManagedContentBlock, SEOEntity } from './types'\n\n/**\n * Parse and render markdown content (basic support)\n * For full markdown, use a proper parser in your custom renderer\n */\nfunction renderMarkdown(content: string): string {\n return content\n // Headers\n .replace(/^### (.*$)/gim, '<h3>$1</h3>')\n .replace(/^## (.*$)/gim, '<h2>$1</h2>')\n .replace(/^# (.*$)/gim, '<h1>$1</h1>')\n // Bold\n .replace(/\\*\\*(.*)\\*\\*/gim, '<strong>$1</strong>')\n // Italic\n .replace(/\\*(.*)\\*/gim, '<em>$1</em>')\n // Links\n .replace(/\\[(.*?)\\]\\((.*?)\\)/gim, '<a href=\"$2\">$1</a>')\n // Paragraphs\n .replace(/\\n\\n/gim, '</p><p>')\n .replace(/^(.+)$/gim, '<p>$1</p>')\n}\n\n/**\n * Inject entity annotations into HTML content\n * Wraps entity mentions with data-sonor-entity attributes for knowledge graph linking\n */\nfunction injectEntityAnnotations(html: string, entities: SEOEntity[]): string {\n if (!entities.length) return html\n \n let annotatedHtml = html\n \n // Sort entities by name length (longest first) to avoid partial matches\n const sortedEntities = [...entities].sort((a, b) => b.name.length - a.name.length)\n \n for (const entity of sortedEntities) {\n // Skip very short names to avoid false positives\n if (entity.name.length < 3) continue\n \n // Create case-insensitive regex for entity name\n // Avoid matching inside HTML tags or existing annotations\n const regex = new RegExp(\n `(?<![\\\\w-])(?<!data-sonor-entity=\")${escapeRegExp(entity.name)}(?![\\\\w-])`,\n 'gi'\n )\n \n // Determine schema type for microdata\n const schemaType = getSchemaTypeForEntity(entity.entity_type)\n \n annotatedHtml = annotatedHtml.replace(regex, (match) => {\n return `<span class=\"aeo-entity aeo-entity-${entity.entity_type}\" ` +\n `data-sonor-entity=\"${entity.id}\" ` +\n `data-sonor-entity-type=\"${entity.entity_type}\" ` +\n `data-sonor-entity-name=\"${entity.name}\" ` +\n `itemscope itemtype=\"https://schema.org/${schemaType}\">` +\n `<span itemprop=\"name\">${match}</span></span>`\n })\n }\n \n return annotatedHtml\n}\n\nfunction escapeRegExp(string: string): string {\n return string.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')\n}\n\nfunction getSchemaTypeForEntity(entityType: string): string {\n const typeMap: Record<string, string> = {\n organization: 'Organization',\n person: 'Person',\n service: 'Service',\n product: 'Product',\n location: 'Place',\n concept: 'Thing',\n credential: 'EducationalOccupationalCredential',\n }\n return typeMap[entityType] || 'Thing'\n}\n\n/**\n * ManagedContent - Server Component for CMS-controlled content blocks\n * \n * Fetches content sections from Portal and renders them\n * Supports HTML, Markdown, JSON, and React component references\n * \n * @example\n * ```tsx\n * // Hero section managed by Portal\n * import { ManagedContent } from '@sonordev/seo'\n * \n * export default async function ServicePage({ params }) {\n * return (\n * <main>\n * <ManagedContent\n * path={`/services/${params.slug}`}\n * section=\"hero\"\n * fallback={<DefaultHero />}\n * />\n *\n * <ManagedContent\n * path={`/services/${params.slug}`}\n * section=\"features\"\n * />\n *\n * <ManagedContent\n * path={`/services/${params.slug}`}\n * section=\"cta\"\n * />\n * </main>\n * )\n * }\n * ```\n */\nasync function ManagedContentAsync({\n path,\n section,\n fallback,\n className,\n components = {},\n injectEntityAnnotations: shouldInjectEntities = false,\n}: ManagedContentProps): Promise<React.ReactElement | null> {\n const block = await getContentBlock(path, section)\n\n if (!block) {\n if (fallback) {\n return <>{fallback}</>\n }\n return null\n }\n\n // Fetch entities if annotation is enabled\n let entities: SEOEntity[] = []\n if (shouldInjectEntities) {\n try {\n entities = await getEntities() as SEOEntity[]\n } catch (err) {\n console.warn('@sonordev/seo: Failed to fetch entities for annotation:', err)\n }\n }\n\n const containerClass = className || `sk-content sk-content--${section}`\n\n // Helper to process HTML with optional entity injection\n const processHtml = (html: string): string => {\n if (shouldInjectEntities && entities.length > 0) {\n return injectEntityAnnotations(html, entities)\n }\n return html\n }\n\n // Handle different content types\n switch (block.content_type) {\n case 'html':\n return (\n <div \n className={containerClass}\n dangerouslySetInnerHTML={{ __html: processHtml(block.content as string) }}\n />\n )\n\n case 'markdown':\n const htmlContent = processHtml(renderMarkdown(block.content as string))\n return (\n <div \n className={containerClass}\n dangerouslySetInnerHTML={{ __html: htmlContent }}\n />\n )\n\n case 'json':\n // JSON content for structured data - render as data attributes or custom handling\n const jsonData = typeof block.content === 'string' \n ? JSON.parse(block.content)\n : block.content\n\n return (\n <div \n className={containerClass}\n data-content={JSON.stringify(jsonData)}\n >\n {/* Render JSON structure - customize based on your needs */}\n {jsonData.title && <h2>{jsonData.title}</h2>}\n {jsonData.subtitle && <p className=\"subtitle\">{jsonData.subtitle}</p>}\n {jsonData.content && <div dangerouslySetInnerHTML={{ __html: jsonData.content }} />}\n {jsonData.items && (\n <ul>\n {jsonData.items.map((item: { text: string } | string, index: number) => (\n <li key={index}>{typeof item === 'string' ? item : item.text}</li>\n ))}\n </ul>\n )}\n </div>\n )\n\n case 'react':\n // React component reference - lookup from provided components map\n const componentData = typeof block.content === 'string'\n ? JSON.parse(block.content)\n : block.content as Record<string, unknown>\n\n const componentName = componentData.component as string\n const componentProps = componentData.props as Record<string, unknown> || {}\n\n const Component = components[componentName]\n \n if (!Component) {\n console.warn(`@sonordev/seo: Component \"${componentName}\" not found in components map`)\n return fallback ? <>{fallback}</> : null\n }\n\n return (\n <div className={containerClass}>\n <Component {...componentProps} />\n </div>\n )\n\n default:\n console.warn(`@sonordev/seo: Unknown content type \"${block.content_type}\"`)\n return fallback ? <>{fallback}</> : null\n }\n}\n\n/**\n * Get content block data without rendering\n * \n * Useful when you need to access the raw data\n */\nexport async function getManagedContentData(\n path: string,\n section: string\n): Promise<ManagedContentBlock | null> {\n return getContentBlock(path, section)\n}\n\nexport function ManagedContent(props: ManagedContentProps): React.ReactElement {\n return (\n <Suspense fallback={null}>\n <ManagedContentAsync {...props} />\n </Suspense>\n )\n}\n\nexport default ManagedContent\n","/**\n * LocationPageContent - Server Component for fetching location page sections\n * \n * Usage:\n * <LocationPageContent \n * projectId=\"uuid\" \n * path=\"/areas/seattle-wa/plumbing\" \n * section=\"hero\" \n * />\n * \n * This component fetches content from seo_location_pages.sections for a given\n * path and section, making it editable in Portal without code deploys.\n */\n\nimport * as React from 'react'\nimport { cache, Suspense } from 'react'\n\n// ============================================\n// Types\n// ============================================\n\nexport interface LocationPageContentProps {\n /** Sonor project ID */\n projectId: string\n /** Page path (e.g., /areas/seattle-wa/plumbing) */\n path: string\n /** Section ID or type to fetch (e.g., \"hero\", \"intro\", \"services\") */\n section: string\n /** Fallback content if section not found */\n fallback?: React.ReactNode\n /** Additional className for wrapper */\n className?: string\n /** Custom renderer for the section data */\n render?: (data: LocationSectionData) => React.ReactNode\n}\n\nexport interface LocationSectionData {\n type: string\n content: any\n}\n\nexport interface HeroSectionContent {\n title: string\n subtitle?: string\n cta_text?: string\n cta_href?: string\n stats?: Array<{ label: string; value: string }>\n background_image?: string\n}\n\nexport interface ServiceGridContent {\n services: Array<{\n title: string\n description: string\n href: string\n icon?: string\n }>\n}\n\nexport interface TextSectionContent {\n heading?: string\n paragraphs: string[]\n html?: string\n}\n\n// ============================================\n// API Config\n// ============================================\n\nfunction getApiConfig() {\n const apiUrl = process.env.SONOR_API_URL || 'https://api.sonor.io'\n return { apiUrl }\n}\n\n// ============================================\n// Cached API Call\n// ============================================\n\n/**\n * Fetch location page section content from Portal\n * Cached with React's cache() for request deduplication\n */\nexport const getLocationSection = cache(async (\n projectId: string,\n path: string,\n section: string\n): Promise<LocationSectionData | null> => {\n const { apiUrl } = getApiConfig()\n \n try {\n const response = await fetch(`${apiUrl}/api/public/seo/location-content`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n project_id: projectId,\n path,\n section,\n }),\n next: { revalidate: 3600 }, // Cache for 1 hour\n })\n \n if (!response.ok) {\n console.error(`LocationPageContent: Failed to fetch section \"${section}\" for path \"${path}\"`)\n return null\n }\n \n const data = await response.json()\n return data\n } catch (error) {\n console.error('LocationPageContent: API error:', error)\n return null\n }\n})\n\n// ============================================\n// Default Renderers\n// ============================================\n\nfunction HeroRenderer({ data, className }: { data: HeroSectionContent; className?: string }) {\n return (\n <section className={className || 'location-hero'}>\n <h1>{data.title}</h1>\n {data.subtitle && <p className=\"subtitle\">{data.subtitle}</p>}\n {data.stats && (\n <div className=\"stats-grid\">\n {data.stats.map((stat, i) => (\n <div key={i} className=\"stat\">\n <span className=\"stat-value\">{stat.value}</span>\n <span className=\"stat-label\">{stat.label}</span>\n </div>\n ))}\n </div>\n )}\n {data.cta_text && data.cta_href && (\n <a href={data.cta_href} className=\"cta-button\">\n {data.cta_text}\n </a>\n )}\n </section>\n )\n}\n\nfunction TextRenderer({ data, className }: { data: TextSectionContent; className?: string }) {\n if (data.html) {\n return (\n <section \n className={className || 'location-text'} \n dangerouslySetInnerHTML={{ __html: data.html }} \n />\n )\n }\n \n return (\n <section className={className || 'location-text'}>\n {data.heading && <h2>{data.heading}</h2>}\n {data.paragraphs.map((p, i) => (\n <p key={i}>{p}</p>\n ))}\n </section>\n )\n}\n\nfunction ServicesGridRenderer({ data, className }: { data: ServiceGridContent; className?: string }) {\n return (\n <section className={className || 'location-services-grid'}>\n <div className=\"services-list\">\n {data.services.map((service, i) => (\n <a key={i} href={service.href} className=\"service-card\">\n {service.icon && <span className=\"service-icon\">{service.icon}</span>}\n <h3>{service.title}</h3>\n <p>{service.description}</p>\n </a>\n ))}\n </div>\n </section>\n )\n}\n\nfunction DefaultRenderer({ data, className }: { data: any; className?: string }) {\n // Fallback: just render as JSON for debugging\n if (process.env.NODE_ENV === 'development') {\n return (\n <section className={className}>\n <pre style={{ fontSize: '12px', background: '#f0f0f0', padding: '1rem' }}>\n {JSON.stringify(data, null, 2)}\n </pre>\n </section>\n )\n }\n \n // In production, try to render content as text/html\n if (typeof data.content === 'string') {\n return (\n <section \n className={className} \n dangerouslySetInnerHTML={{ __html: data.content }} \n />\n )\n }\n \n return null\n}\n\n// ============================================\n// Main Component\n// ============================================\n\n/**\n * LocationPageContent - Server Component\n * \n * Fetches and renders a section of a location page from Portal.\n * Content is fully editable in Portal → SEO → Location Pages.\n */\nasync function LocationPageContentAsync({\n projectId,\n path,\n section,\n fallback = null,\n className,\n render,\n}: LocationPageContentProps): Promise<React.ReactElement | null> {\n const data = await getLocationSection(projectId, path, section)\n \n if (!data) {\n return fallback as React.ReactElement | null\n }\n \n // If custom renderer provided, use it\n if (render) {\n return <>{render(data)}</>\n }\n \n // Use default renderers based on section type\n switch (data.type) {\n case 'hero':\n return <HeroRenderer data={data.content as HeroSectionContent} className={className} />\n \n case 'text':\n case 'intro':\n case 'about':\n case 'about_location':\n return <TextRenderer data={data.content as TextSectionContent} className={className} />\n \n case 'services_grid':\n case 'services':\n return <ServicesGridRenderer data={data.content as ServiceGridContent} className={className} />\n \n default:\n return <DefaultRenderer data={data} className={className} />\n }\n}\n\n/**\n * LocationPageContent — Suspense-wrapped so API fetches never block page streaming.\n */\nexport function LocationPageContent(props: LocationPageContentProps): React.ReactElement {\n return (\n <Suspense fallback={props.fallback as React.ReactElement ?? null}>\n <LocationPageContentAsync {...props} />\n </Suspense>\n )\n}\n\n// Export types for external use\nexport type {\n HeroSectionContent as LocationHeroContent,\n ServiceGridContent as LocationServicesContent,\n TextSectionContent as LocationTextContent,\n}\n"]}
|
package/dist/seo/server.d.mts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
export { b as generateSitemap, g as getManagedMetadata, a as getManagedMetadataWithAB, r as registerLocalSitemap } from '../routing-
|
|
2
|
-
export { A as ABTest, k as ABTestResult, F as FAQItem, l as GetABVariantOptions, G as GetManagedMetadataOptions, n as GetRedirectOptions, r as GetRobotsOptions, t as GetSitemapEntriesOptions, h as ManagedContentBlock, i as ManagedContentProps, d as ManagedFAQData, e as ManagedFAQProps, g as ManagedInternalLinksProps, f as ManagedLink, M as ManagedMetadataResult, m as ManagedRedirect, c as ManagedSchemaProps, o as ManagedScript, p as ManagedScriptsProps, R as RedirectResult, q as RobotsDirective, j as SEOEntity, a as SEOPageData, b as SchemaMarkup, s as SitemapEntry, S as SonorSEOConfig } from '../types-
|
|
1
|
+
export { b as generateSitemap, g as getManagedMetadata, a as getManagedMetadataWithAB, r as registerLocalSitemap } from '../routing-trNzR1Pz.mjs';
|
|
2
|
+
export { A as ABTest, k as ABTestResult, F as FAQItem, l as GetABVariantOptions, G as GetManagedMetadataOptions, n as GetRedirectOptions, r as GetRobotsOptions, t as GetSitemapEntriesOptions, h as ManagedContentBlock, i as ManagedContentProps, d as ManagedFAQData, e as ManagedFAQProps, g as ManagedInternalLinksProps, f as ManagedLink, M as ManagedMetadataResult, m as ManagedRedirect, c as ManagedSchemaProps, o as ManagedScript, p as ManagedScriptsProps, R as RedirectResult, q as RobotsDirective, j as SEOEntity, a as SEOPageData, b as SchemaMarkup, s as SitemapEntry, S as SonorSEOConfig } from '../types-DYyIAgQg.mjs';
|
|
3
3
|
import 'next';
|
|
4
4
|
|
|
5
5
|
/**
|
package/dist/seo/server.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
export { b as generateSitemap, g as getManagedMetadata, a as getManagedMetadataWithAB, r as registerLocalSitemap } from '../routing-
|
|
2
|
-
export { A as ABTest, k as ABTestResult, F as FAQItem, l as GetABVariantOptions, G as GetManagedMetadataOptions, n as GetRedirectOptions, r as GetRobotsOptions, t as GetSitemapEntriesOptions, h as ManagedContentBlock, i as ManagedContentProps, d as ManagedFAQData, e as ManagedFAQProps, g as ManagedInternalLinksProps, f as ManagedLink, M as ManagedMetadataResult, m as ManagedRedirect, c as ManagedSchemaProps, o as ManagedScript, p as ManagedScriptsProps, R as RedirectResult, q as RobotsDirective, j as SEOEntity, a as SEOPageData, b as SchemaMarkup, s as SitemapEntry, S as SonorSEOConfig } from '../types-
|
|
1
|
+
export { b as generateSitemap, g as getManagedMetadata, a as getManagedMetadataWithAB, r as registerLocalSitemap } from '../routing-C7gmHWm9.js';
|
|
2
|
+
export { A as ABTest, k as ABTestResult, F as FAQItem, l as GetABVariantOptions, G as GetManagedMetadataOptions, n as GetRedirectOptions, r as GetRobotsOptions, t as GetSitemapEntriesOptions, h as ManagedContentBlock, i as ManagedContentProps, d as ManagedFAQData, e as ManagedFAQProps, g as ManagedInternalLinksProps, f as ManagedLink, M as ManagedMetadataResult, m as ManagedRedirect, c as ManagedSchemaProps, o as ManagedScript, p as ManagedScriptsProps, R as RedirectResult, q as RobotsDirective, j as SEOEntity, a as SEOPageData, b as SchemaMarkup, s as SitemapEntry, S as SonorSEOConfig } from '../types-DYyIAgQg.js';
|
|
3
3
|
import 'next';
|
|
4
4
|
|
|
5
5
|
/**
|
package/dist/seo/server.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
|
-
var
|
|
3
|
+
var chunkPAF5IGGF_js = require('../chunk-PAF5IGGF.js');
|
|
4
4
|
require('../chunk-VU4GL5ZG.js');
|
|
5
5
|
require('../chunk-ZSMWDLMK.js');
|
|
6
6
|
var react = require('react');
|
|
@@ -185,19 +185,19 @@ react.cache(async (projectId) => {
|
|
|
185
185
|
|
|
186
186
|
Object.defineProperty(exports, "generateSitemap", {
|
|
187
187
|
enumerable: true,
|
|
188
|
-
get: function () { return
|
|
188
|
+
get: function () { return chunkPAF5IGGF_js.generateSitemap; }
|
|
189
189
|
});
|
|
190
190
|
Object.defineProperty(exports, "getManagedMetadata", {
|
|
191
191
|
enumerable: true,
|
|
192
|
-
get: function () { return
|
|
192
|
+
get: function () { return chunkPAF5IGGF_js.getManagedMetadata; }
|
|
193
193
|
});
|
|
194
194
|
Object.defineProperty(exports, "getManagedMetadataWithAB", {
|
|
195
195
|
enumerable: true,
|
|
196
|
-
get: function () { return
|
|
196
|
+
get: function () { return chunkPAF5IGGF_js.getManagedMetadataWithAB; }
|
|
197
197
|
});
|
|
198
198
|
Object.defineProperty(exports, "registerLocalSitemap", {
|
|
199
199
|
enumerable: true,
|
|
200
|
-
get: function () { return
|
|
200
|
+
get: function () { return chunkPAF5IGGF_js.registerLocalSitemap; }
|
|
201
201
|
});
|
|
202
202
|
exports.getABTest = getABTest;
|
|
203
203
|
exports.getContentBlock = getContentBlock;
|
package/dist/seo/server.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export { generateSitemap, getManagedMetadata, getManagedMetadataWithAB, registerLocalSitemap } from '../chunk-
|
|
1
|
+
export { generateSitemap, getManagedMetadata, getManagedMetadataWithAB, registerLocalSitemap } from '../chunk-5SQ4NRPH.mjs';
|
|
2
2
|
import '../chunk-TUFP3FGI.mjs';
|
|
3
3
|
import '../chunk-4XPGGLVP.mjs';
|
|
4
4
|
import { cache } from 'react';
|
package/dist/sitemap/index.d.mts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
import { L as LLMsDataResponse } from '../types-
|
|
1
|
+
import { L as LLMsDataResponse } from '../types-J7Z_FqmV.mjs';
|
|
2
|
+
import '../llms/contract.mjs';
|
|
2
3
|
|
|
3
4
|
/**
|
|
4
5
|
* @sonordev/site-kit - Sitemap Generator
|
|
@@ -74,8 +75,14 @@ interface SitemapConfig {
|
|
|
74
75
|
optimizedLLMsFullTxt?: boolean;
|
|
75
76
|
/** When Portal returns empty services, use this to supply local site data for llms.txt */
|
|
76
77
|
getLocalData?: () => Promise<LLMsDataResponse | null>;
|
|
78
|
+
/** Pass-through to GET /api/public/llms/txt?publicSummaryOnly=true and local llms fallback */
|
|
79
|
+
llmsPublicSummaryOnly?: boolean;
|
|
77
80
|
/** Use Signal AI visibility scores to compute page priority (40% visibility + 30% depth + 30% base) */
|
|
78
81
|
intelligentPriority?: boolean;
|
|
82
|
+
/** When true, append `/llms.txt` to sitemap (low priority). Default false. */
|
|
83
|
+
includeLlmsTxtInSitemap?: boolean;
|
|
84
|
+
/** When true, append `/llms-full.txt` (requires includeLlmsTxtInSitemap or set alone — both supported). */
|
|
85
|
+
includeLlmsFullTxtInSitemap?: boolean;
|
|
79
86
|
}
|
|
80
87
|
interface SyncSitemapOptions {
|
|
81
88
|
/** Trigger Signal AI meta optimization after sync */
|
package/dist/sitemap/index.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
import { L as LLMsDataResponse } from '../types-
|
|
1
|
+
import { L as LLMsDataResponse } from '../types-0NuBL1Gg.js';
|
|
2
|
+
import '../llms/contract.js';
|
|
2
3
|
|
|
3
4
|
/**
|
|
4
5
|
* @sonordev/site-kit - Sitemap Generator
|
|
@@ -74,8 +75,14 @@ interface SitemapConfig {
|
|
|
74
75
|
optimizedLLMsFullTxt?: boolean;
|
|
75
76
|
/** When Portal returns empty services, use this to supply local site data for llms.txt */
|
|
76
77
|
getLocalData?: () => Promise<LLMsDataResponse | null>;
|
|
78
|
+
/** Pass-through to GET /api/public/llms/txt?publicSummaryOnly=true and local llms fallback */
|
|
79
|
+
llmsPublicSummaryOnly?: boolean;
|
|
77
80
|
/** Use Signal AI visibility scores to compute page priority (40% visibility + 30% depth + 30% base) */
|
|
78
81
|
intelligentPriority?: boolean;
|
|
82
|
+
/** When true, append `/llms.txt` to sitemap (low priority). Default false. */
|
|
83
|
+
includeLlmsTxtInSitemap?: boolean;
|
|
84
|
+
/** When true, append `/llms-full.txt` (requires includeLlmsTxtInSitemap or set alone — both supported). */
|
|
85
|
+
includeLlmsFullTxtInSitemap?: boolean;
|
|
79
86
|
}
|
|
80
87
|
interface SyncSitemapOptions {
|
|
81
88
|
/** Trigger Signal AI meta optimization after sync */
|
package/dist/sitemap/index.js
CHANGED
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
|
-
var
|
|
3
|
+
var chunkMWE2HRPU_js = require('../chunk-MWE2HRPU.js');
|
|
4
4
|
var chunkJP5WGL75_js = require('../chunk-JP5WGL75.js');
|
|
5
|
+
require('../chunk-ATG4FJY6.js');
|
|
5
6
|
require('../chunk-ZSMWDLMK.js');
|
|
6
7
|
var fs = require('fs');
|
|
7
8
|
var path = require('path');
|
|
@@ -75,11 +76,7 @@ async function tryGenerateStaticParams(pageFilePath) {
|
|
|
75
76
|
return params;
|
|
76
77
|
}
|
|
77
78
|
}
|
|
78
|
-
} catch
|
|
79
|
-
console.warn(
|
|
80
|
-
`[site-kit] Could not resolve generateStaticParams from ${pageFilePath}:`,
|
|
81
|
-
err.message
|
|
82
|
-
);
|
|
79
|
+
} catch {
|
|
83
80
|
}
|
|
84
81
|
return null;
|
|
85
82
|
}
|
|
@@ -150,8 +147,8 @@ async function discoverPages(appDir, currentPath = "", pages = [], options = {})
|
|
|
150
147
|
}
|
|
151
148
|
}
|
|
152
149
|
} else {
|
|
153
|
-
console.
|
|
154
|
-
`[site-kit]
|
|
150
|
+
console.debug(
|
|
151
|
+
`[site-kit] Dynamic segment "${entry}" at /${currentPath} not resolved via filesystem. If these routes are missing from sitemap, add them via additionalPaths or dynamicRoutes config.`
|
|
155
152
|
);
|
|
156
153
|
}
|
|
157
154
|
continue;
|
|
@@ -262,6 +259,7 @@ async function syncSitemapToPortal(entries, apiUrl, apiKey, options) {
|
|
|
262
259
|
} catch {
|
|
263
260
|
path = entry.url.startsWith("/") ? entry.url : `/${entry.url}`;
|
|
264
261
|
}
|
|
262
|
+
path = path.replace(/^\/{1,2}[a-z0-9.-]+\.[a-z]{2,}(\/|$)/i, "/");
|
|
265
263
|
const normalized = {
|
|
266
264
|
path,
|
|
267
265
|
priority: entry.priority,
|
|
@@ -272,6 +270,13 @@ async function syncSitemapToPortal(entries, apiUrl, apiKey, options) {
|
|
|
272
270
|
normalized.description = entry.description;
|
|
273
271
|
}
|
|
274
272
|
return normalized;
|
|
273
|
+
}).filter((entry) => {
|
|
274
|
+
const p = entry.path;
|
|
275
|
+
if (p.includes("://") || /\/\/[a-z0-9.-]+\.[a-z]{2,}/i.test(p)) {
|
|
276
|
+
console.warn(`[site-kit] Filtering malformed sitemap path: ${p}`);
|
|
277
|
+
return false;
|
|
278
|
+
}
|
|
279
|
+
return true;
|
|
275
280
|
});
|
|
276
281
|
try {
|
|
277
282
|
const body = { entries: normalizedEntries, mode: "full-replace" };
|
|
@@ -442,9 +447,27 @@ function createSitemap(config) {
|
|
|
442
447
|
}
|
|
443
448
|
}
|
|
444
449
|
entries.sort((a, b) => (b.priority ?? 0) - (a.priority ?? 0));
|
|
450
|
+
const pushLlmsUrl = (pathSuffix, priority) => {
|
|
451
|
+
const url = `${normalizedBaseUrl}${pathSuffix}`;
|
|
452
|
+
if (!entries.some((e) => e.url === url)) {
|
|
453
|
+
entries.push({
|
|
454
|
+
url,
|
|
455
|
+
lastModified: /* @__PURE__ */ new Date(),
|
|
456
|
+
changeFrequency: "monthly",
|
|
457
|
+
priority
|
|
458
|
+
});
|
|
459
|
+
}
|
|
460
|
+
};
|
|
461
|
+
if (config.includeLlmsTxtInSitemap) {
|
|
462
|
+
pushLlmsUrl("/llms.txt", 0.35);
|
|
463
|
+
}
|
|
464
|
+
if (config.includeLlmsFullTxtInSitemap) {
|
|
465
|
+
pushLlmsUrl("/llms-full.txt", 0.3);
|
|
466
|
+
}
|
|
445
467
|
console.log(`[site-kit] Generated sitemap with ${entries.length} pages`);
|
|
446
468
|
const isDev = process.env.NODE_ENV === "development";
|
|
447
|
-
|
|
469
|
+
const isProductionBuild = process.env.NEXT_PHASE === "phase-production-build";
|
|
470
|
+
if (!config.disableSync && !isDev && !discoveryFailed && isProductionBuild) {
|
|
448
471
|
const apiUrl = config.apiUrl || process.env.SONOR_API_URL || "https://api.sonor.io";
|
|
449
472
|
const apiKey = config.apiKey || process.env.SONOR_API_KEY;
|
|
450
473
|
if (apiKey) {
|
|
@@ -477,12 +500,13 @@ function createSitemap(config) {
|
|
|
477
500
|
}
|
|
478
501
|
if (config.optimizedLLMsTxt !== false) {
|
|
479
502
|
try {
|
|
480
|
-
const writeResult = await
|
|
503
|
+
const writeResult = await chunkMWE2HRPU_js.writeLLMsTxtToPublic({
|
|
481
504
|
apiUrl,
|
|
482
505
|
apiKey,
|
|
483
506
|
full: config.optimizedLLMsFullTxt ?? false,
|
|
484
507
|
fallbackToLocal: true,
|
|
485
|
-
getLocalData: config.getLocalData
|
|
508
|
+
getLocalData: config.getLocalData,
|
|
509
|
+
publicSummaryOnly: config.llmsPublicSummaryOnly
|
|
486
510
|
});
|
|
487
511
|
if (writeResult.success) {
|
|
488
512
|
console.log(`[site-kit] Wrote llms.txt (optimized=${writeResult.optimized})`);
|