Skip to content

Commit fab7e9e

Browse files
authoredJul 21, 2024··
perf!: rewrite i18n resolving and url normalizing (#319)
1 parent b2d1409 commit fab7e9e

25 files changed

+845
-590
lines changed
 
+23-17
Original file line numberDiff line numberDiff line change
@@ -1,32 +1,38 @@
1-
import { defineEventHandler, getQuery, setHeader } from 'h3'
2-
import { fixSlashes } from 'site-config-stack/urls'
1+
import { appendHeader, defineEventHandler, setHeader } from 'h3'
32
import { useSimpleSitemapRuntimeConfig } from '../utils'
4-
import { buildSitemapIndex } from '../sitemap/builder/sitemap-index'
3+
import { buildSitemapIndex, urlsToIndexXml } from '../sitemap/builder/sitemap-index'
54
import type { SitemapOutputHookCtx } from '../../types'
6-
import { createSitePathResolver, useNitroApp, useSiteConfig } from '#imports'
5+
import { useNitroUrlResolvers } from '..//sitemap/nitro'
6+
import { useNitroApp } from '#imports'
77

88
export default defineEventHandler(async (e) => {
9-
const canonicalQuery = getQuery(e).canonical
10-
const isShowingCanonical = typeof canonicalQuery !== 'undefined' && canonicalQuery !== 'false'
119
const runtimeConfig = useSimpleSitemapRuntimeConfig()
12-
const siteConfig = useSiteConfig(e)
13-
let sitemap = (await buildSitemapIndex({
14-
event: e,
15-
canonicalUrlResolver: createSitePathResolver(e, { canonical: isShowingCanonical || !import.meta.dev, absolute: true, withBase: true }),
16-
relativeBaseUrlResolver: createSitePathResolver(e, { absolute: false, withBase: true }),
17-
fixSlashes: (path: string) => fixSlashes(siteConfig.trailingSlash, path),
18-
}, runtimeConfig))
19-
2010
const nitro = useNitroApp()
11+
const resolvers = useNitroUrlResolvers(e)
12+
const sitemaps = (await buildSitemapIndex(resolvers, runtimeConfig))
13+
14+
// tell the prerender to render the other sitemaps (if we prerender this one)
15+
// this solves the dynamic chunking sitemap issue
16+
if (import.meta.prerender) {
17+
appendHeader(
18+
e,
19+
'x-nitro-prerender',
20+
sitemaps.filter(entry => !!entry._sitemapName)
21+
.map(entry => encodeURIComponent(`/${entry._sitemapName}-sitemap.xml`)).join(', '),
22+
)
23+
}
24+
25+
const indexResolvedCtx = { sitemaps }
26+
await nitro.hooks.callHook('sitemap:index-resolved', indexResolvedCtx)
2127

22-
const ctx: SitemapOutputHookCtx = { sitemap, sitemapName: 'sitemap' }
28+
const output = urlsToIndexXml(indexResolvedCtx.sitemaps, resolvers, runtimeConfig)
29+
const ctx: SitemapOutputHookCtx = { sitemap: output, sitemapName: 'sitemap' }
2330
await nitro.hooks.callHook('sitemap:output', ctx)
24-
sitemap = ctx.sitemap
2531

2632
setHeader(e, 'Content-Type', 'text/xml; charset=UTF-8')
2733
if (runtimeConfig.cacheMaxAgeSeconds)
2834
setHeader(e, 'Cache-Control', `public, max-age=${runtimeConfig.cacheMaxAgeSeconds}, must-revalidate`)
2935
else
3036
setHeader(e, 'Cache-Control', `no-cache, no-store`)
31-
return sitemap
37+
return ctx.sitemap
3238
})

‎src/runtime/nitro/sitemap/builder/sitemap-index.ts

+10-35
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,16 @@
11
import { defu } from 'defu'
2-
import { appendHeader } from 'h3'
32
import type {
43
ModuleRuntimeConfig,
54
NitroUrlResolvers,
65
ResolvedSitemapUrl,
76
SitemapIndexEntry,
87
SitemapUrl,
98
} from '../../../types'
10-
import { normaliseDate, normaliseSitemapUrls } from '../urlset/normalise'
9+
import { normaliseDate } from '../urlset/normalise'
1110
import { globalSitemapSources, resolveSitemapSources } from '../urlset/sources'
12-
import { applyI18nEnhancements } from '../urlset/i18n'
13-
import { filterSitemapUrls } from '../urlset/filter'
1411
import { sortSitemapUrls } from '../urlset/sort'
1512
import { escapeValueForXml, wrapSitemapXml } from './xml'
16-
import { useNitroApp } from '#imports'
13+
import { resolveSitemapEntries } from './sitemap'
1714

1815
export async function buildSitemapIndex(resolvers: NitroUrlResolvers, runtimeConfig: ModuleRuntimeConfig) {
1916
const {
@@ -25,10 +22,6 @@ export async function buildSitemapIndex(resolvers: NitroUrlResolvers, runtimeCon
2522
autoI18n,
2623
isI18nMapped,
2724
sortEntries,
28-
// xls
29-
version,
30-
xsl,
31-
credits,
3225
} = runtimeConfig
3326

3427
if (!sitemaps)
@@ -42,22 +35,13 @@ export async function buildSitemapIndex(resolvers: NitroUrlResolvers, runtimeCon
4235
const chunks: Record<string | number, { urls: SitemapUrl[] }> = {}
4336
if (isChunking) {
4437
const sitemap = sitemaps.chunks
45-
// TODO
4638
// we need to figure out how many entries we're dealing with
4739
const sources = await resolveSitemapSources(await globalSitemapSources())
48-
// we need to generate multiple sitemaps with dynamically generated names
49-
const normalisedUrls = normaliseSitemapUrls(sources.map(e => e.urls).flat(), resolvers)
40+
const normalisedUrls = resolveSitemapEntries(sitemap, sources, { autoI18n, isI18nMapped })
5041
// 2. enhance
51-
let enhancedUrls: ResolvedSitemapUrl[] = normalisedUrls
42+
const enhancedUrls: ResolvedSitemapUrl[] = normalisedUrls
5243
.map(e => defu(e, sitemap.defaults) as ResolvedSitemapUrl)
53-
// TODO enable
54-
if (autoI18n?.locales)
55-
enhancedUrls = applyI18nEnhancements(enhancedUrls, { isI18nMapped, autoI18n, sitemapName: sitemap.sitemapName })
56-
// 3. filtered urls
57-
// TODO make sure include and exclude start with baseURL?
58-
const filteredUrls = filterSitemapUrls(enhancedUrls, { ...sitemap, autoI18n, isMultiSitemap: true })
59-
// 4. sort
60-
const sortedUrls = maybeSort(filteredUrls)
44+
const sortedUrls = maybeSort(enhancedUrls)
6145
// split into the max size which should be 1000
6246
sortedUrls.forEach((url, i) => {
6347
const chunkIndex = Math.floor(i / (defaultSitemapsChunkSize as number))
@@ -74,21 +58,12 @@ export async function buildSitemapIndex(resolvers: NitroUrlResolvers, runtimeCon
7458
}
7559
}
7660

77-
// tell the prerender to render the other sitemaps (if we prerender this one)
78-
// this solves the dynamic chunking sitemap issue
79-
if (import.meta.prerender) {
80-
appendHeader(
81-
resolvers.event,
82-
'x-nitro-prerender',
83-
Object.keys(chunks).map(name => encodeURIComponent(`/${name}-sitemap.xml`)).join(', '),
84-
)
85-
}
86-
8761
const entries: SitemapIndexEntry[] = []
8862
// normalise
8963
for (const name in chunks) {
9064
const sitemap = chunks[name]
9165
const entry: SitemapIndexEntry = {
66+
_sitemapName: name,
9267
sitemap: resolvers.canonicalUrlResolver(`${name}-sitemap.xml`),
9368
}
9469
let lastmod = sitemap.urls
@@ -110,11 +85,11 @@ export async function buildSitemapIndex(resolvers: NitroUrlResolvers, runtimeCon
11085
}))
11186
}
11287

113-
const ctx = { sitemaps: entries }
114-
const nitro = useNitroApp()
115-
await nitro.hooks.callHook('sitemap:index-resolved', ctx)
88+
return entries
89+
}
11690

117-
const sitemapXml = ctx.sitemaps.map(e => [
91+
export function urlsToIndexXml(sitemaps: SitemapIndexEntry[], resolvers: NitroUrlResolvers, { version, xsl, credits }: Pick<ModuleRuntimeConfig, 'version' | 'xsl' | 'credits'>) {
92+
const sitemapXml = sitemaps.map(e => [
11893
' <sitemap>',
11994
` <loc>${escapeValueForXml(e.sitemap)}</loc>`,
12095
// lastmod is optional

‎src/runtime/nitro/sitemap/builder/sitemap.ts

+173-65
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,178 @@
1-
import { defu } from 'defu'
21
import { resolveSitePath } from 'site-config-stack/urls'
3-
import { parseURL, withHttps } from 'ufo'
2+
import { joinURL, withHttps } from 'ufo'
43
import type {
4+
AlternativeEntry, AutoI18nConfig,
55
ModuleRuntimeConfig,
66
NitroUrlResolvers,
77
ResolvedSitemapUrl,
88
SitemapDefinition,
9-
SitemapRenderCtx,
9+
SitemapSourceResolved,
1010
SitemapUrlInput,
1111
} from '../../../types'
12-
import { normaliseSitemapUrls } from '../urlset/normalise'
12+
import { preNormalizeEntry } from '../urlset/normalise'
1313
import { childSitemapSources, globalSitemapSources, resolveSitemapSources } from '../urlset/sources'
14-
import { filterSitemapUrls } from '../urlset/filter'
15-
import { applyI18nEnhancements, normaliseI18nSources } from '../urlset/i18n'
1614
import { sortSitemapUrls } from '../urlset/sort'
17-
import { splitForLocales } from '../../utils'
18-
import { createNitroRouteRuleMatcher } from '../../kit'
15+
import { createPathFilter, logger, splitForLocales } from '../../../utils-pure'
1916
import { handleEntry, wrapSitemapXml } from './xml'
20-
import { useNitroApp } from '#imports'
2117

22-
export async function buildSitemap(sitemap: SitemapDefinition, resolvers: NitroUrlResolvers, runtimeConfig: ModuleRuntimeConfig) {
18+
export interface NormalizedI18n extends ResolvedSitemapUrl {
19+
_pathWithoutPrefix: string
20+
_locale: AutoI18nConfig['locales'][number]
21+
_index?: number
22+
}
23+
24+
export function resolveSitemapEntries(sitemap: SitemapDefinition, sources: SitemapSourceResolved[], runtimeConfig: Pick<ModuleRuntimeConfig, 'autoI18n' | 'isI18nMapped'>): ResolvedSitemapUrl[] {
25+
const {
26+
autoI18n,
27+
isI18nMapped,
28+
} = runtimeConfig
29+
const filterPath = createPathFilter({
30+
include: sitemap.include,
31+
exclude: sitemap.exclude,
32+
})
33+
// 1. normalise
34+
const _urls = sources.flatMap(e => e.urls).map((_e) => {
35+
const e = preNormalizeEntry(_e)
36+
if (!e.loc || !filterPath(e.loc))
37+
return false
38+
return e
39+
}).filter(Boolean) as ResolvedSitemapUrl[]
40+
41+
let validI18nUrlsForTransform: NormalizedI18n[] = []
42+
let warnIncorrectI18nTransformUsage = false
43+
const withoutPrefixPaths: Record<string, NormalizedI18n[]> = {}
44+
if (autoI18n && autoI18n.strategy !== 'no_prefix') {
45+
const localeCodes = autoI18n.locales.map(l => l.code)
46+
validI18nUrlsForTransform = _urls.map((_e, i) => {
47+
if (_e._abs)
48+
return false
49+
const split = splitForLocales(_e.loc, localeCodes)
50+
let localeCode = split[0]
51+
const pathWithoutPrefix = split[1]
52+
if (!localeCode)
53+
localeCode = autoI18n.defaultLocale
54+
const e = _e as NormalizedI18n
55+
e._pathWithoutPrefix = pathWithoutPrefix
56+
const locale = autoI18n.locales.find(l => l.code === localeCode)!
57+
if (!locale)
58+
return false
59+
e._locale = locale
60+
e._index = i
61+
withoutPrefixPaths[pathWithoutPrefix] = withoutPrefixPaths[pathWithoutPrefix] || []
62+
// need to make sure the locale doesn't already exist
63+
if (!withoutPrefixPaths[pathWithoutPrefix].some(e => e._locale.code === locale.code))
64+
withoutPrefixPaths[pathWithoutPrefix].push(e)
65+
return e
66+
}).filter(Boolean) as NormalizedI18n[]
67+
68+
for (const e of validI18nUrlsForTransform) {
69+
// let's try and find other urls that we can use for alternatives
70+
if (!e._i18nTransform && !e.alternatives?.length) {
71+
const alternatives = withoutPrefixPaths[e._pathWithoutPrefix]
72+
.map((u) => {
73+
const entries: AlternativeEntry[] = []
74+
if (u._locale.code === autoI18n.defaultLocale) {
75+
entries.push({
76+
href: u.loc,
77+
hreflang: 'x-default',
78+
})
79+
}
80+
entries.push({
81+
href: u.loc,
82+
hreflang: u._locale.code || autoI18n.defaultLocale,
83+
})
84+
return entries
85+
})
86+
.flat()
87+
.filter(Boolean) as AlternativeEntry[]
88+
if (alternatives.length)
89+
e.alternatives = alternatives
90+
}
91+
else if (e._i18nTransform) {
92+
delete e._i18nTransform
93+
if (autoI18n.strategy === 'no_prefix') {
94+
warnIncorrectI18nTransformUsage = true
95+
}
96+
// keep single entry, just add alternatvies
97+
if (autoI18n.differentDomains) {
98+
e.alternatives = [
99+
{
100+
// apply default locale domain
101+
...autoI18n.locales.find(l => [l.code, l.iso].includes(autoI18n.defaultLocale)),
102+
code: 'x-default',
103+
},
104+
...autoI18n.locales
105+
.filter(l => !!l.domain),
106+
]
107+
.map((locale) => {
108+
return {
109+
hreflang: locale.iso || locale.code,
110+
href: joinURL(withHttps(locale.domain!), e._pathWithoutPrefix),
111+
}
112+
})
113+
}
114+
else {
115+
// need to add urls for all other locales
116+
for (const l of autoI18n.locales) {
117+
let loc = joinURL(`/${l.code}`, e._pathWithoutPrefix)
118+
if (autoI18n.differentDomains || (['prefix_and_default', 'prefix_except_default'].includes(autoI18n.strategy) && l.code === autoI18n.defaultLocale))
119+
loc = e._pathWithoutPrefix
120+
const _sitemap = isI18nMapped ? (l.iso || l.code) : undefined
121+
const newEntry: NormalizedI18n = preNormalizeEntry({
122+
_sitemap,
123+
...e,
124+
_index: undefined,
125+
_key: `${_sitemap || ''}${loc}`,
126+
_locale: l,
127+
loc,
128+
alternatives: [{code: 'x-default'}, ...autoI18n.locales].map((locale) => {
129+
const code = locale.code === 'x-default' ? autoI18n.defaultLocale : locale.code
130+
const isDefault = locale.code === 'x-default' || locale.code === autoI18n.defaultLocale
131+
let href = ''
132+
if (autoI18n.strategy === 'prefix') {
133+
href = joinURL('/', code, e._pathWithoutPrefix)
134+
} else if (['prefix_and_default', 'prefix_except_default'].includes(autoI18n.strategy)) {
135+
if (isDefault) {
136+
// no prefix
137+
href = e._pathWithoutPrefix
138+
} else {
139+
href = joinURL('/', code, e._pathWithoutPrefix)
140+
}
141+
}
142+
const hreflang = locale.iso || locale.code
143+
if (!filterPath(href))
144+
return false
145+
return {
146+
hreflang,
147+
href,
148+
}
149+
}).filter(Boolean),
150+
})
151+
if (e._locale.code === newEntry._locale.code) {
152+
// replace
153+
_urls[e._index] = newEntry
154+
// avoid getting re-replaced
155+
e._index = undefined
156+
} else {
157+
_urls.push(newEntry)
158+
}
159+
}
160+
}
161+
}
162+
if (isI18nMapped) {
163+
e._sitemap = e._sitemap || e._locale.iso || e._locale.code
164+
}
165+
if (e._index)
166+
_urls[e._index] = e
167+
}
168+
}
169+
if (import.meta.dev && warnIncorrectI18nTransformUsage) {
170+
logger.warn('You\'re using _i18nTransform with the `no_prefix` strategy. This will cause issues with the sitemap. Please remove the _i18nTransform flag or change i18n strategy.')
171+
}
172+
return _urls
173+
}
174+
175+
export async function buildSitemapUrls(sitemap: SitemapDefinition, resolvers: NitroUrlResolvers, runtimeConfig: ModuleRuntimeConfig) {
23176
// 0. resolve sources
24177
// 1. normalise
25178
// 2. filter
@@ -38,10 +191,6 @@ export async function buildSitemap(sitemap: SitemapDefinition, resolvers: NitroU
38191
sortEntries,
39192
// chunking
40193
defaultSitemapsChunkSize,
41-
// xls
42-
version,
43-
xsl,
44-
credits,
45194
} = runtimeConfig
46195
const isChunking = typeof sitemaps.chunks !== 'undefined' && !Number.isNaN(Number(sitemap.sitemapName))
47196
function maybeSort(urls: ResolvedSitemapUrl[]) {
@@ -71,65 +220,24 @@ export async function buildSitemap(sitemap: SitemapDefinition, resolvers: NitroU
71220
// always fetch all sitemap data for the primary sitemap
72221
const sources = sitemap.includeAppSources ? await globalSitemapSources() : []
73222
sources.push(...await childSitemapSources(sitemap))
74-
let resolvedSources = await resolveSitemapSources(sources, resolvers.event)
75-
// normalise the sources for i18n
76-
if (autoI18n)
77-
resolvedSources = normaliseI18nSources(resolvedSources, { autoI18n, isI18nMapped, ...sitemap })
78-
// 1. normalise
79-
const normalisedUrls = normaliseSitemapUrls(resolvedSources.map(e => e.urls).flat(), resolvers)
80-
81-
const routeRuleMatcher = createNitroRouteRuleMatcher()
82-
let enhancedUrls: ResolvedSitemapUrl[] = normalisedUrls
83-
// apply defaults
84-
.map(e => defu(e, sitemap.defaults) as ResolvedSitemapUrl)
85-
// apply route rules
86-
.map((e) => {
87-
const path = parseURL(e.loc).pathname
88-
let routeRules = routeRuleMatcher(path)
89-
// apply top-level path without prefix, users can still target the localed path
90-
if (autoI18n?.locales && autoI18n?.strategy !== 'no_prefix') {
91-
// remove the locale path from the prefix, if it exists, need to use regex
92-
const match = splitForLocales(path, autoI18n.locales.map(l => l.code))
93-
const pathWithoutPrefix = match[1]
94-
if (pathWithoutPrefix && pathWithoutPrefix !== path)
95-
routeRules = defu(routeRules, routeRuleMatcher(pathWithoutPrefix))
96-
}
97-
98-
if (routeRules.sitemap === false)
99-
return false
100-
if (typeof routeRules.index !== 'undefined' && !routeRules.index)
101-
return false
102-
const hasRobotsDisabled = Object.entries(routeRules.headers || {})
103-
.some(([name, value]) => name.toLowerCase() === 'x-robots-tag' && value.toLowerCase() === 'noindex')
104-
// check for redirects and headers which aren't indexable
105-
if (routeRules.redirect || hasRobotsDisabled)
106-
return false
223+
const resolvedSources = await resolveSitemapSources(sources, resolvers.event)
107224

108-
return routeRules.sitemap ? defu(e, routeRules.sitemap) as ResolvedSitemapUrl : e
109-
})
110-
.filter(Boolean) as ResolvedSitemapUrl[]
111-
// TODO enable
112-
if (autoI18n?.locales)
113-
enhancedUrls = applyI18nEnhancements(enhancedUrls, { isI18nMapped, autoI18n, ...sitemap })
225+
const enhancedUrls = resolveSitemapEntries(sitemap, resolvedSources, { autoI18n, isI18nMapped })
114226
// 3. filtered urls
115227
// TODO make sure include and exclude start with baseURL?
116-
const filteredUrls = filterSitemapUrls(enhancedUrls, { event: resolvers.event, isMultiSitemap, autoI18n, ...sitemap })
228+
const filteredUrls = enhancedUrls.filter((e) => {
229+
if (isMultiSitemap && e._sitemap && sitemap.sitemapName)
230+
return e._sitemap === sitemap.sitemapName
231+
return true
232+
})
117233
// 4. sort
118234
const sortedUrls = maybeSort(filteredUrls)
119235
// 5. maybe slice for chunked
120236
// if we're rendering a partial sitemap, slice the entries
121-
const slicedUrls = maybeSlice(sortedUrls)
122-
// 6. nitro hooks
123-
const nitro = useNitroApp()
124-
const ctx: SitemapRenderCtx = {
125-
urls: slicedUrls,
126-
sitemapName: sitemap.sitemapName,
127-
}
128-
await nitro.hooks.callHook('sitemap:resolved', ctx)
129-
130-
// final urls
131-
const urls = maybeSort(normaliseSitemapUrls(ctx.urls, resolvers))
237+
return maybeSlice(sortedUrls)
238+
}
132239

240+
export function urlsToXml(urls: ResolvedSitemapUrl[], resolvers: NitroUrlResolvers, { version, xsl, credits }: Pick<ModuleRuntimeConfig, 'version' | 'xsl' | 'credits'>) {
133241
const urlset = urls.map((e) => {
134242
const keys = Object.keys(e).filter(k => !k.startsWith('_'))
135243
return [

‎src/runtime/nitro/sitemap/nitro.ts

+60-11
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,20 @@
11
import { getQuery, setHeader } from 'h3'
22
import type { H3Event } from 'h3'
33
import { fixSlashes } from 'site-config-stack/urls'
4-
import type { ModuleRuntimeConfig, NitroUrlResolvers, SitemapDefinition } from '../../types'
5-
import { buildSitemap } from './builder/sitemap'
6-
import { buildSitemapIndex } from './builder/sitemap-index'
7-
import { createSitePathResolver, useNitroApp, useSiteConfig } from '#imports'
4+
import { defu } from 'defu'
5+
import type {
6+
ModuleRuntimeConfig,
7+
NitroUrlResolvers,
8+
ResolvedSitemapUrl,
9+
SitemapDefinition,
10+
SitemapRenderCtx,
11+
} from '../../types'
12+
import { mergeOnKey, splitForLocales } from '../../utils-pure'
13+
import { createNitroRouteRuleMatcher } from '../kit'
14+
import { buildSitemapUrls, urlsToXml } from './builder/sitemap'
15+
import { normaliseEntry } from './urlset/normalise'
16+
import { sortSitemapUrls } from './urlset/sort'
17+
import { createSitePathResolver, getPathRobotConfig, useNitroApp, useSiteConfig } from '#imports'
818

919
export function useNitroUrlResolvers(e: H3Event): NitroUrlResolvers {
1020
const canonicalQuery = getQuery(e).canonical
@@ -26,20 +36,59 @@ export function useNitroUrlResolvers(e: H3Event): NitroUrlResolvers {
2636
export async function createSitemap(e: H3Event, definition: SitemapDefinition, runtimeConfig: ModuleRuntimeConfig) {
2737
const { sitemapName } = definition
2838
const nitro = useNitroApp()
29-
let sitemap = await (
30-
definition.sitemapName === 'index'
31-
? buildSitemapIndex(useNitroUrlResolvers(e), runtimeConfig)
32-
: buildSitemap(definition, useNitroUrlResolvers(e), runtimeConfig)
33-
)
39+
const resolvers = useNitroUrlResolvers(e)
40+
let sitemapUrls = await buildSitemapUrls(definition, resolvers, runtimeConfig)
41+
42+
const routeRuleMatcher = createNitroRouteRuleMatcher()
43+
const { autoI18n } = runtimeConfig
44+
sitemapUrls = sitemapUrls.map((e) => {
45+
// blocked by nuxt-simple-robots (this is a polyfill if not installed)
46+
if (!getPathRobotConfig(e, { path: e._path.pathname, skipSiteIndexable: true }).indexable)
47+
return false
48+
const path = e._path.pathname
49+
let routeRules = routeRuleMatcher(path)
50+
// apply top-level path without prefix, users can still target the localed path
51+
if (autoI18n?.locales && autoI18n?.strategy !== 'no_prefix') {
52+
// remove the locale path from the prefix, if it exists, need to use regex
53+
const match = splitForLocales(path, autoI18n.locales.map(l => l.code))
54+
const pathWithoutPrefix = match[1]
55+
if (pathWithoutPrefix && pathWithoutPrefix !== path)
56+
routeRules = defu(routeRules, routeRuleMatcher(pathWithoutPrefix))
57+
}
58+
59+
if (routeRules.sitemap === false)
60+
return false
61+
if (typeof routeRules.index !== 'undefined' && !routeRules.index)
62+
return false
63+
const hasRobotsDisabled = Object.entries(routeRules.headers || {})
64+
.some(([name, value]) => name.toLowerCase() === 'x-robots-tag' && value.toLowerCase().includes('noindex'))
65+
// check for redirects and headers which aren't indexable
66+
if (routeRules.redirect || hasRobotsDisabled)
67+
return false
68+
69+
return routeRules.sitemap ? defu(e, routeRules.sitemap) as ResolvedSitemapUrl : e
70+
}).filter(Boolean)
71+
72+
// 6. nitro hooks
73+
const resolvedCtx: SitemapRenderCtx = {
74+
urls: sitemapUrls,
75+
sitemapName: sitemapName,
76+
}
77+
await nitro.hooks.callHook('sitemap:resolved', resolvedCtx)
78+
79+
const maybeSort = (urls: ResolvedSitemapUrl[]) => runtimeConfig.sortEntries ? sortSitemapUrls(urls) : urls
80+
// final urls
81+
const urls = maybeSort(mergeOnKey(resolvedCtx.urls.map(e => normaliseEntry(e, definition.defaults, resolvers)), '_key'))
82+
const sitemap = urlsToXml(urls, resolvers, runtimeConfig)
83+
3484
const ctx = { sitemap, sitemapName }
3585
await nitro.hooks.callHook('sitemap:output', ctx)
36-
sitemap = ctx.sitemap
3786
// need to clone the config object to make it writable
3887
setHeader(e, 'Content-Type', 'text/xml; charset=UTF-8')
3988
if (runtimeConfig.cacheMaxAgeSeconds)
4089
setHeader(e, 'Cache-Control', `public, max-age=${runtimeConfig.cacheMaxAgeSeconds}, must-revalidate`)
4190
else
4291
setHeader(e, 'Cache-Control', `no-cache, no-store`)
4392
e.context._isSitemap = true
44-
return sitemap
93+
return ctx.sitemap
4594
}

‎src/runtime/nitro/sitemap/urlset/filter.ts

-35
This file was deleted.

‎src/runtime/nitro/sitemap/urlset/i18n.ts

-174
This file was deleted.

‎src/runtime/nitro/sitemap/urlset/normalise.ts

+76-72
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,16 @@
1-
import { hasProtocol } from 'ufo'
2-
import { fixSlashes } from 'site-config-stack/urls'
1+
import { hasProtocol, parsePath, parseURL } from 'ufo'
2+
import { defu } from 'defu'
33
import type {
44
AlternativeEntry,
55
NitroUrlResolvers,
66
ResolvedSitemapUrl,
77
SitemapUrl,
8-
SitemapUrlInput,
98
} from '../../../types'
109
import { mergeOnKey } from '../../../utils-pure'
1110

12-
function resolve(s: string | URL, resolvers: NitroUrlResolvers): string
13-
function resolve(s: string | undefined | URL, resolvers: NitroUrlResolvers): string | undefined {
14-
if (typeof s === 'undefined')
11+
function resolve(s: string | URL, resolvers?: NitroUrlResolvers): string
12+
function resolve(s: string | undefined | URL, resolvers?: NitroUrlResolvers): string | undefined {
13+
if (typeof s === 'undefined' || !resolvers)
1514
return s
1615
// convert url to string
1716
s = typeof s === 'string' ? s : s.toString()
@@ -22,80 +21,85 @@ function resolve(s: string | undefined | URL, resolvers: NitroUrlResolvers): str
2221
return resolvers.canonicalUrlResolver(s)
2322
}
2423

25-
export function normaliseSitemapUrls(data: SitemapUrlInput[], resolvers: NitroUrlResolvers): ResolvedSitemapUrl[] {
26-
// make sure we're working with objects
27-
const entries: SitemapUrl[] = data
28-
.map(e => typeof e === 'string' ? { loc: e } : e)
29-
// uniform loc
30-
.map((e) => {
31-
// make fields writable so we can modify them
32-
e = { ...e }
33-
if (e.url) {
34-
e.loc = e.url
35-
delete e.url
36-
}
37-
// we want a uniform loc so we can dedupe using it, remove slashes and only get the path
38-
e.loc = fixSlashes(false, e.loc)
39-
return e
40-
})
41-
.filter(Boolean)
24+
function removeTrailingSlash(s: string) {
25+
// need to account for query strings and hashes
26+
// this assumes the URL is normalised
27+
return s.replace(/\/(\?|#|$)/, '$1')
28+
}
4229

43-
// apply auto alternative lang prefixes, needs to happen before normalization
30+
export function preNormalizeEntry(_e: SitemapUrl | string): ResolvedSitemapUrl {
31+
const e = (typeof _e === 'string' ? { loc: _e } : { ..._e }) as ResolvedSitemapUrl
32+
if (e.url && !e.loc) {
33+
e.loc = e.url
34+
delete e.url
35+
}
36+
// we want a uniform loc so we can dedupe using it, remove slashes and only get the path
37+
e.loc = removeTrailingSlash(e.loc)
38+
e._abs = hasProtocol(e.loc, { acceptRelative: false, strict: false })
39+
try {
40+
e._path = e._abs ? parseURL(e.loc) : parsePath(e.loc)
41+
}
42+
catch (e) {
43+
e._path = null
44+
}
45+
if (e._path?.pathname === '')
46+
e.loc = `${e.loc}/`
47+
if (e._path) {
48+
e._key = `${e._sitemap || ''}${e._path?.pathname || '/'}${e._path.search}`
49+
}
50+
else {
51+
e._key = e.loc
52+
}
53+
return e as ResolvedSitemapUrl
54+
}
4455

45-
function normaliseEntry(e: SitemapUrl): ResolvedSitemapUrl {
46-
if (e.lastmod) {
47-
const date = normaliseDate(e.lastmod)
48-
if (date)
49-
e.lastmod = date
50-
else
51-
delete e.lastmod
52-
}
53-
// make sure it's valid
54-
if (!e.lastmod)
56+
export function normaliseEntry(_e: ResolvedSitemapUrl, defaults: Omit<SitemapUrl, 'loc'>, resolvers?: NitroUrlResolvers): ResolvedSitemapUrl {
57+
const e = defu(_e, defaults) as ResolvedSitemapUrl
58+
if (e.lastmod) {
59+
const date = normaliseDate(e.lastmod)
60+
if (date)
61+
e.lastmod = date
62+
else
5563
delete e.lastmod
64+
}
65+
// make sure it's valid
66+
if (!e.lastmod)
67+
delete e.lastmod
5668

57-
// need to make sure siteURL doesn't have the base on the end
58-
e.loc = resolve(e.loc, resolvers)
69+
// need to make sure siteURL doesn't have the base on the end
70+
e.loc = resolve(e.loc, resolvers)
5971

60-
// correct alternative hrefs
61-
if (e.alternatives) {
62-
e.alternatives = mergeOnKey(e.alternatives.map((e) => {
63-
const a: AlternativeEntry & { key?: string } = { ...e }
64-
// string
65-
if (typeof a.href === 'string')
66-
a.href = resolve(a.href, resolvers)
67-
// URL object
68-
else if (typeof a.href === 'object' && a.href)
69-
a.href = resolve(a.href.href, resolvers)
70-
return a
71-
}), 'hreflang')
72-
}
73-
74-
if (e.images) {
75-
e.images = mergeOnKey(e.images.map((i) => {
76-
i = { ...i }
77-
i.loc = resolve(i.loc, resolvers)
78-
return i
79-
}), 'loc')
80-
}
72+
// correct alternative hrefs
73+
if (e.alternatives) {
74+
e.alternatives = mergeOnKey(e.alternatives.map((e) => {
75+
const a: AlternativeEntry & { key?: string } = { ...e }
76+
// string
77+
if (typeof a.href === 'string')
78+
a.href = resolve(a.href, resolvers)
79+
// URL object
80+
else if (typeof a.href === 'object' && a.href)
81+
a.href = resolve(a.href.href, resolvers)
82+
return a
83+
}), 'hreflang')
84+
}
8185

82-
if (e.videos) {
83-
e.videos = e.videos.map((v) => {
84-
v = { ...v }
85-
if (v.content_loc)
86-
v.content_loc = resolve(v.content_loc, resolvers)
87-
return v
88-
})
89-
}
86+
if (e.images) {
87+
e.images = mergeOnKey(e.images.map((i) => {
88+
i = { ...i }
89+
i.loc = resolve(i.loc, resolvers)
90+
return i
91+
}), 'loc')
92+
}
9093

91-
// @todo normalise image href and src
92-
return e as ResolvedSitemapUrl
94+
if (e.videos) {
95+
e.videos = e.videos.map((v) => {
96+
v = { ...v }
97+
if (v.content_loc)
98+
v.content_loc = resolve(v.content_loc, resolvers)
99+
return v
100+
})
93101
}
94-
return mergeOnKey(
95-
entries.map(normaliseEntry)
96-
.map(e => ({ ...e, _key: `${e._sitemap || ''}${e.loc}` })),
97-
'_key',
98-
)
102+
return e
99103
}
100104

101105
const IS_VALID_W3C_DATE = [

‎src/runtime/types.ts

+19-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import type { FetchOptions } from 'ofetch'
22
import type { H3Event } from 'h3'
3+
import type { ParsedURL } from 'ufo'
34

45
// we need to have the module options within the runtime entry
56
// as we don't want to depend on the module entry as it can cause
@@ -211,12 +212,29 @@ export interface ModuleRuntimeConfig extends Pick<ModuleOptions, 'cacheMaxAgeSec
211212
export interface SitemapIndexEntry {
212213
sitemap: string
213214
lastmod?: string
215+
/**
216+
* @internal
217+
*/
218+
_sitemapName?: string
214219
}
215220

216221
export type FilterInput = (string | RegExp | {
217222
regex: string
218223
})
219-
export type ResolvedSitemapUrl = Omit<SitemapUrl, 'url'> & Required<Pick<SitemapUrl, 'loc'>>
224+
export type ResolvedSitemapUrl = Omit<SitemapUrl, 'url'> & Required<Pick<SitemapUrl, 'loc'>> & {
225+
/**
226+
* @internal
227+
*/
228+
_key: string
229+
/**
230+
* @internal
231+
*/
232+
_path: ParsedURL
233+
/**
234+
* @internal
235+
*/
236+
_abs: boolean
237+
}
220238

221239
export interface SitemapDefinition {
222240
/**

‎src/runtime/utils-pure.ts

+8-1
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,15 @@
11
import { createDefu } from 'defu'
22
import { parseURL, withLeadingSlash } from 'ufo'
33
import { createRouter, toRouteMatcher } from 'radix3'
4+
import { createConsola } from 'consola'
45
import type { FilterInput } from './types'
56

7+
export const logger = createConsola({
8+
defaults: {
9+
tag: '@nuxt/sitemap',
10+
},
11+
})
12+
613
const merger = createDefu((obj, key, value) => {
714
// merge arrays using a set
815
if (Array.isArray(obj[key]) && Array.isArray(value))
@@ -21,7 +28,7 @@ export function mergeOnKey<T, K extends keyof T>(arr: T[], key: K) {
2128
return Object.values(res)
2229
}
2330

24-
export function splitForLocales(path: string, locales: string[]) {
31+
export function splitForLocales(path: string, locales: string[]): [string | null, string] {
2532
// we only want to use the first path segment otherwise we can end up turning "/ending" into "/en/ding"
2633
const prefix = withLeadingSlash(path).split('/')[1]
2734
// make sure prefix is a valid locale

‎test/bench/i18n.bench.ts

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
import { bench, describe } from 'vitest'
2+
import { resolveSitemapEntries } from '../../src/runtime/nitro/sitemap/builder/sitemap'
3+
import type { SitemapSourceResolved } from '#sitemap'
4+
5+
const sources: SitemapSourceResolved[] = [
6+
{
7+
urls: Array.from({ length: 3000 }, (_, i) => ({
8+
loc: `/foo-${i}`,
9+
})),
10+
context: {
11+
name: 'foo',
12+
},
13+
sourceType: 'user',
14+
},
15+
]
16+
17+
describe('i18n', () => {
18+
bench('normaliseI18nSources', () => {
19+
resolveSitemapEntries({
20+
sitemapName: 'sitemap.xml',
21+
}, sources, {
22+
autoI18n: {
23+
locales: [
24+
{ code: 'en', iso: 'en' },
25+
{ code: 'fr', iso: 'fr' },
26+
// add 22 more locales
27+
...Array.from({ length: 22 }, (_, i) => ({
28+
code: `code-${i}`,
29+
iso: `iso-${i}`,
30+
})),
31+
],
32+
strategy: 'prefix',
33+
defaultLocale: 'en',
34+
},
35+
isI18nMapped: true,
36+
})
37+
}, {
38+
iterations: 1000,
39+
})
40+
})

‎test/bench/normalize.bench.ts

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import { bench, describe } from 'vitest'
2+
import { preNormalizeEntry } from '../../src/runtime/nitro/sitemap/urlset/normalise'
3+
import type { SitemapSourceResolved } from '#sitemap'
4+
import { resolveSitemapEntries } from '~/src/runtime/nitro/sitemap/builder/sitemap'
5+
6+
const sources: SitemapSourceResolved[] = [
7+
{
8+
urls: Array.from({ length: 3000 }, (_, i) => ({
9+
loc: `/foo-${i}`,
10+
})),
11+
context: {
12+
name: 'foo',
13+
},
14+
sourceType: 'user',
15+
},
16+
]
17+
18+
describe('normalize', () => {
19+
bench('preNormalizeEntry', () => {
20+
resolveSitemapEntries(sources)
21+
const urls = sources.flatMap(s => s.urls)
22+
urls.map(u => preNormalizeEntry(u))
23+
}, {
24+
iterations: 1000,
25+
})
26+
})

‎test/integration/i18n/domains.test.ts

+2-9
Original file line numberDiff line numberDiff line change
@@ -57,21 +57,14 @@ describe('i18n domains', () => {
5757
"<?xml version="1.0" encoding="UTF-8"?><?xml-stylesheet type="text/xsl" href="/__sitemap__/style.xsl"?>
5858
<urlset xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:video="http://www.google.com/schemas/sitemap-video/1.1" xmlns:xhtml="http://www.w3.org/1999/xhtml" xmlns:image="http://www.google.com/schemas/sitemap-image/1.1" xmlns:news="http://www.google.com/schemas/sitemap-news/0.9" xsi:schemaLocation="http://www.sitemaps.org/schemas/sitemap/0.9 http://www.sitemaps.org/schemas/sitemap/0.9/sitemap.xsd http://www.google.com/schemas/sitemap-image/1.1 http://www.google.com/schemas/sitemap-image/1.1/sitemap-image.xsd" xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">
5959
<url>
60-
<loc>https://fr.nuxtseo.com/fr</loc>
60+
<loc>https://fr.nuxtseo.com/fr/</loc>
6161
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/en" />
6262
<xhtml:link rel="alternate" hreflang="es-ES" href="https://es.nuxtseo.com/es" />
6363
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://fr.nuxtseo.com/fr" />
6464
<xhtml:link rel="alternate" hreflang="x-default" href="https://nuxtseo.com/en" />
6565
</url>
6666
<url>
67-
<loc>https://fr.nuxtseo.com/__sitemap/url</loc>
68-
<changefreq>weekly</changefreq>
69-
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/__sitemap/url" />
70-
<xhtml:link rel="alternate" hreflang="es-ES" href="https://es.nuxtseo.com/__sitemap/url" />
71-
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://fr.nuxtseo.com/__sitemap/url" />
72-
</url>
73-
<url>
74-
<loc>https://fr.nuxtseo.com/fr/test</loc>
67+
<loc>https://fr.nuxtseo.com/fr/test/</loc>
7568
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/en/test" />
7669
<xhtml:link rel="alternate" hreflang="es-ES" href="https://es.nuxtseo.com/es/test" />
7770
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://fr.nuxtseo.com/fr/test" />

‎test/integration/i18n/dynamic-urls.test.ts

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ describe('i18n dynamic urls', () => {
4141
</url>
4242
<url>
4343
<loc>https://nuxtseo.com/english-url</loc>
44+
<xhtml:link rel="alternate" href="https://nuxtseo.com/english-url" hreflang="x-default" />
4445
<xhtml:link rel="alternate" href="https://nuxtseo.com/english-url" hreflang="en" />
4546
</url>
4647
<url>

‎test/integration/i18n/filtering.test.ts

+2
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,8 @@ describe('i18n filtering', () => {
3333
</url>
3434
<url>
3535
<loc>https://nuxtseo.com/no-i18n</loc>
36+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="x-default" />
37+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="en" />
3638
</url>
3739
<url>
3840
<loc>https://nuxtseo.com/en/__sitemap/url</loc>

‎test/integration/i18n/generate.test.ts

+2
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,8 @@ describe('generate', () => {
5050
</url>
5151
<url>
5252
<loc>https://nuxtseo.com/no-i18n</loc>
53+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="x-default" />
54+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="en" />
5355
</url>
5456
<url>
5557
<loc>https://nuxtseo.com/en/test</loc>

‎test/integration/i18n/pages.no-prefix.test.ts

+4-24
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,10 @@ describe('i18n pages with no prefix strategy', () => {
7676
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://nuxtseo.com/offres" />
7777
<xhtml:link rel="alternate" hreflang="x-default" href="https://nuxtseo.com/services" />
7878
</url>
79+
<url>
80+
<loc>https://nuxtseo.com/__sitemap/url</loc>
81+
<changefreq>weekly</changefreq>
82+
</url>
7983
<url>
8084
<loc>https://nuxtseo.com/offres/developement</loc>
8185
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/services/development" />
@@ -100,30 +104,6 @@ describe('i18n pages with no prefix strategy', () => {
100104
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://nuxtseo.com/offres/developement" />
101105
<xhtml:link rel="alternate" hreflang="x-default" href="https://nuxtseo.com/services/development" />
102106
</url>
103-
<url>
104-
<loc>https://nuxtseo.com/en/__sitemap/url</loc>
105-
<changefreq>weekly</changefreq>
106-
<xhtml:link rel="alternate" hreflang="x-default" href="https://nuxtseo.com/" />
107-
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/" />
108-
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://nuxtseo.com/" />
109-
<xhtml:link rel="alternate" hreflang="es-ES" href="https://nuxtseo.com/" />
110-
</url>
111-
<url>
112-
<loc>https://nuxtseo.com/es/__sitemap/url</loc>
113-
<changefreq>weekly</changefreq>
114-
<xhtml:link rel="alternate" hreflang="x-default" href="https://nuxtseo.com/" />
115-
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/" />
116-
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://nuxtseo.com/" />
117-
<xhtml:link rel="alternate" hreflang="es-ES" href="https://nuxtseo.com/" />
118-
</url>
119-
<url>
120-
<loc>https://nuxtseo.com/fr/__sitemap/url</loc>
121-
<changefreq>weekly</changefreq>
122-
<xhtml:link rel="alternate" hreflang="x-default" href="https://nuxtseo.com/" />
123-
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/" />
124-
<xhtml:link rel="alternate" hreflang="fr-FR" href="https://nuxtseo.com/" />
125-
<xhtml:link rel="alternate" hreflang="es-ES" href="https://nuxtseo.com/" />
126-
</url>
127107
<url>
128108
<loc>https://nuxtseo.com/offres/developement/app</loc>
129109
<xhtml:link rel="alternate" hreflang="en-US" href="https://nuxtseo.com/services/development/app" />

‎test/integration/i18n/prefix-and-default.test.ts

+2
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,8 @@ describe('i18n prefix and default', () => {
6666
</url>
6767
<url>
6868
<loc>https://nuxtseo.com/no-i18n</loc>
69+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="x-default" />
70+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="en" />
6971
</url>
7072
<url>
7173
<loc>https://nuxtseo.com/test</loc>

‎test/integration/i18n/prefix-except-default.test.ts

+2
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,8 @@ describe('i18n prefix except default', () => {
6666
</url>
6767
<url>
6868
<loc>https://nuxtseo.com/no-i18n</loc>
69+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="x-default" />
70+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="en" />
6971
</url>
7072
<url>
7173
<loc>https://nuxtseo.com/test</loc>

‎test/integration/i18n/prefix-iso.test.ts

+4
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,8 @@ describe('i18n prefix', () => {
5252
</url>
5353
<url>
5454
<loc>https://nuxtseo.com/extra</loc>
55+
<xhtml:link rel="alternate" href="https://nuxtseo.com/extra" hreflang="x-default" />
56+
<xhtml:link rel="alternate" href="https://nuxtseo.com/extra" hreflang="en" />
5557
</url>
5658
<url>
5759
<loc>https://nuxtseo.com/fr</loc>
@@ -62,6 +64,8 @@ describe('i18n prefix', () => {
6264
</url>
6365
<url>
6466
<loc>https://nuxtseo.com/no-i18n</loc>
67+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="x-default" />
68+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="en" />
6569
</url>
6670
<url>
6771
<loc>https://nuxtseo.com/en/test</loc>

‎test/integration/i18n/prefix-simple.test.ts

+4
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,8 @@ describe('i18n prefix', () => {
4545
</url>
4646
<url>
4747
<loc>https://nuxtseo.com/extra</loc>
48+
<xhtml:link rel="alternate" href="https://nuxtseo.com/extra" hreflang="x-default" />
49+
<xhtml:link rel="alternate" href="https://nuxtseo.com/extra" hreflang="en" />
4850
</url>
4951
<url>
5052
<loc>https://nuxtseo.com/fr</loc>
@@ -55,6 +57,8 @@ describe('i18n prefix', () => {
5557
</url>
5658
<url>
5759
<loc>https://nuxtseo.com/no-i18n</loc>
60+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="x-default" />
61+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n" hreflang="en" />
5862
</url>
5963
<url>
6064
<loc>https://nuxtseo.com/en/test</loc>

‎test/integration/i18n/route-rules.test.ts

+12
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,9 @@ describe('i18n route rules', () => {
6666
<changefreq>daily</changefreq>
6767
<priority>1</priority>
6868
<loc>https://nuxtseo.com/defaults</loc>
69+
<xhtml:link rel="alternate" href="https://nuxtseo.com/defaults" hreflang="x-default" />
70+
<xhtml:link rel="alternate" href="https://nuxtseo.com/defaults" hreflang="en" />
71+
<xhtml:link rel="alternate" href="https://nuxtseo.com/fr/defaults" hreflang="fr" />
6972
</url>
7073
<url>
7174
<loc>https://nuxtseo.com/__sitemap/url</loc>
@@ -79,6 +82,9 @@ describe('i18n route rules', () => {
7982
<changefreq>daily</changefreq>
8083
<priority>1</priority>
8184
<loc>https://nuxtseo.com/fr/defaults</loc>
85+
<xhtml:link rel="alternate" href="https://nuxtseo.com/defaults" hreflang="x-default" />
86+
<xhtml:link rel="alternate" href="https://nuxtseo.com/defaults" hreflang="en" />
87+
<xhtml:link rel="alternate" href="https://nuxtseo.com/fr/defaults" hreflang="fr" />
8288
</url>
8389
<url>
8490
<loc>https://nuxtseo.com/es/__sitemap/url</loc>
@@ -100,11 +106,17 @@ describe('i18n route rules', () => {
100106
<changefreq>daily</changefreq>
101107
<priority>1</priority>
102108
<loc>https://nuxtseo.com/wildcard/defaults/foo</loc>
109+
<xhtml:link rel="alternate" href="https://nuxtseo.com/wildcard/defaults/foo" hreflang="x-default" />
110+
<xhtml:link rel="alternate" href="https://nuxtseo.com/wildcard/defaults/foo" hreflang="en" />
111+
<xhtml:link rel="alternate" href="https://nuxtseo.com/fr/wildcard/defaults/foo" hreflang="fr" />
103112
</url>
104113
<url>
105114
<changefreq>daily</changefreq>
106115
<priority>1</priority>
107116
<loc>https://nuxtseo.com/fr/wildcard/defaults/foo</loc>
117+
<xhtml:link rel="alternate" href="https://nuxtseo.com/wildcard/defaults/foo" hreflang="x-default" />
118+
<xhtml:link rel="alternate" href="https://nuxtseo.com/wildcard/defaults/foo" hreflang="en" />
119+
<xhtml:link rel="alternate" href="https://nuxtseo.com/fr/wildcard/defaults/foo" hreflang="fr" />
108120
</url>
109121
</urlset>"
110122
`)

‎test/integration/i18n/simple-trailing.test.ts

+4
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,8 @@ describe('i18n prefix', () => {
4848
</url>
4949
<url>
5050
<loc>https://nuxtseo.com/extra/</loc>
51+
<xhtml:link rel="alternate" href="https://nuxtseo.com/extra/" hreflang="x-default" />
52+
<xhtml:link rel="alternate" href="https://nuxtseo.com/extra/" hreflang="en" />
5153
</url>
5254
<url>
5355
<loc>https://nuxtseo.com/fr/</loc>
@@ -58,6 +60,8 @@ describe('i18n prefix', () => {
5860
</url>
5961
<url>
6062
<loc>https://nuxtseo.com/no-i18n/</loc>
63+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n/" hreflang="x-default" />
64+
<xhtml:link rel="alternate" href="https://nuxtseo.com/no-i18n/" hreflang="en" />
6165
</url>
6266
<url>
6367
<loc>https://nuxtseo.com/en/test/</loc>

‎test/unit/applyI18nEnhancements.test.ts

-43
This file was deleted.

‎test/unit/i18n.test.ts

+348
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { describe, expect, it } from 'vitest'
22
import { splitPathForI18nLocales } from '../../src/util/i18n'
33
import type { AutoI18nConfig } from '../../src/runtime/types'
4+
import { resolveSitemapEntries } from '../../src/runtime/nitro/sitemap/builder/sitemap'
45

56
const EnFrAutoI18n = {
67
locales: [{
@@ -52,4 +53,351 @@ describe('i18n', () => {
5253
const data2 = splitPathForI18nLocales('/fr/about', { ...EnFrAutoI18n, strategy: 'prefix' })
5354
expect(data2).toMatchInlineSnapshot('"/fr/about"')
5455
})
56+
it('_i18nTransform without prefix', () => {
57+
const urls = resolveSitemapEntries({
58+
sitemapName: 'sitemap.xml',
59+
}, [{
60+
urls: [
61+
{
62+
loc: '/__sitemap/url',
63+
changefreq: 'weekly',
64+
_i18nTransform: true,
65+
},
66+
],
67+
context: {
68+
name: 'foo',
69+
},
70+
sourceType: 'user',
71+
}], {
72+
locales: [{
73+
code: 'en',
74+
iso: 'en-US',
75+
}, {
76+
code: 'fr',
77+
iso: 'fr-FR',
78+
}],
79+
defaultLocale: 'en',
80+
strategy: 'no_prefix',
81+
isI18nMapped: true,
82+
})
83+
expect(urls).toMatchInlineSnapshot(`
84+
[
85+
{
86+
"_abs": false,
87+
"_i18nTransform": true,
88+
"_key": "/__sitemap/url",
89+
"_path": {
90+
"hash": "",
91+
"pathname": "/__sitemap/url",
92+
"search": "",
93+
},
94+
"changefreq": "weekly",
95+
"loc": "/__sitemap/url",
96+
},
97+
]
98+
`)
99+
})
100+
it('_i18nTransform prefix_except_default', () => {
101+
const urls = resolveSitemapEntries({
102+
sitemapName: 'sitemap.xml',
103+
}, [{
104+
urls: [
105+
{
106+
loc: '/__sitemap/url',
107+
changefreq: 'weekly',
108+
_i18nTransform: true,
109+
},
110+
],
111+
context: {
112+
name: 'foo',
113+
},
114+
sourceType: 'user',
115+
}], {
116+
autoI18n: {
117+
locales: [{
118+
code: 'en',
119+
iso: 'en-US',
120+
}, {
121+
code: 'fr',
122+
iso: 'fr-FR',
123+
}],
124+
defaultLocale: 'en',
125+
strategy: 'prefix_except_default',
126+
},
127+
isI18nMapped: true,
128+
})
129+
expect(urls).toMatchInlineSnapshot(`
130+
[
131+
{
132+
"_abs": false,
133+
"_index": undefined,
134+
"_key": "en-US/__sitemap/url",
135+
"_locale": {
136+
"code": "en",
137+
"iso": "en-US",
138+
},
139+
"_path": {
140+
"hash": "",
141+
"pathname": "/__sitemap/url",
142+
"search": "",
143+
},
144+
"_pathWithoutPrefix": "/__sitemap/url",
145+
"_sitemap": "en-US",
146+
"alternatives": [
147+
{
148+
"href": "/__sitemap/url",
149+
"hreflang": "x-default",
150+
},
151+
{
152+
"href": "/__sitemap/url",
153+
"hreflang": "en-US",
154+
},
155+
{
156+
"href": "/fr/__sitemap/url",
157+
"hreflang": "fr-FR",
158+
},
159+
],
160+
"changefreq": "weekly",
161+
"loc": "/__sitemap/url",
162+
},
163+
{
164+
"_abs": false,
165+
"_index": undefined,
166+
"_key": "fr-FR/fr/__sitemap/url",
167+
"_locale": {
168+
"code": "fr",
169+
"iso": "fr-FR",
170+
},
171+
"_path": {
172+
"hash": "",
173+
"pathname": "/fr/__sitemap/url",
174+
"search": "",
175+
},
176+
"_pathWithoutPrefix": "/__sitemap/url",
177+
"_sitemap": "fr-FR",
178+
"alternatives": [
179+
{
180+
"href": "/__sitemap/url",
181+
"hreflang": "x-default",
182+
},
183+
{
184+
"href": "/__sitemap/url",
185+
"hreflang": "en-US",
186+
},
187+
{
188+
"href": "/fr/__sitemap/url",
189+
"hreflang": "fr-FR",
190+
},
191+
],
192+
"changefreq": "weekly",
193+
"loc": "/fr/__sitemap/url",
194+
},
195+
]
196+
`)
197+
})
198+
it('applies alternative links', () => {
199+
const urls = resolveSitemapEntries({
200+
sitemapName: 'sitemap.xml',
201+
}, [{
202+
urls: [],
203+
context: {
204+
name: 'foo',
205+
},
206+
sourceType: 'user',
207+
}, {
208+
urls: [
209+
{
210+
loc: '/en/dynamic/foo',
211+
},
212+
{
213+
loc: '/fr/dynamic/foo',
214+
},
215+
{
216+
loc: 'endless-dungeon', // issue with en being picked up as the locale
217+
_i18nTransform: true,
218+
},
219+
{
220+
loc: 'english-url', // issue with en being picked up as the locale
221+
},
222+
// absolute URL issue
223+
{ loc: 'https://www.somedomain.com/abc/def' },
224+
],
225+
context: {
226+
name: 'foo',
227+
},
228+
sourceType: 'user',
229+
}], {
230+
autoI18n: EnFrAutoI18n,
231+
isI18nMapped: true,
232+
})
233+
expect(urls).toMatchInlineSnapshot(`
234+
[
235+
{
236+
"_abs": false,
237+
"_index": 0,
238+
"_key": "/en/dynamic/foo",
239+
"_locale": {
240+
"code": "en",
241+
"iso": "en-US",
242+
},
243+
"_path": {
244+
"hash": "",
245+
"pathname": "/en/dynamic/foo",
246+
"search": "",
247+
},
248+
"_pathWithoutPrefix": "/dynamic/foo",
249+
"_sitemap": "en-US",
250+
"alternatives": [
251+
{
252+
"href": "/en/dynamic/foo",
253+
"hreflang": "x-default",
254+
},
255+
{
256+
"href": "/en/dynamic/foo",
257+
"hreflang": "en",
258+
},
259+
{
260+
"href": "/fr/dynamic/foo",
261+
"hreflang": "fr",
262+
},
263+
],
264+
"loc": "/en/dynamic/foo",
265+
},
266+
{
267+
"_abs": false,
268+
"_index": 1,
269+
"_key": "/fr/dynamic/foo",
270+
"_locale": {
271+
"code": "fr",
272+
"iso": "fr-FR",
273+
},
274+
"_path": {
275+
"hash": "",
276+
"pathname": "/fr/dynamic/foo",
277+
"search": "",
278+
},
279+
"_pathWithoutPrefix": "/dynamic/foo",
280+
"_sitemap": "fr-FR",
281+
"alternatives": [
282+
{
283+
"href": "/en/dynamic/foo",
284+
"hreflang": "x-default",
285+
},
286+
{
287+
"href": "/en/dynamic/foo",
288+
"hreflang": "en",
289+
},
290+
{
291+
"href": "/fr/dynamic/foo",
292+
"hreflang": "fr",
293+
},
294+
],
295+
"loc": "/fr/dynamic/foo",
296+
},
297+
{
298+
"_abs": false,
299+
"_index": undefined,
300+
"_key": "en-USendless-dungeon",
301+
"_locale": {
302+
"code": "en",
303+
"iso": "en-US",
304+
},
305+
"_path": {
306+
"hash": "",
307+
"pathname": "endless-dungeon",
308+
"search": "",
309+
},
310+
"_pathWithoutPrefix": "endless-dungeon",
311+
"_sitemap": "en-US",
312+
"alternatives": [
313+
{
314+
"href": "endless-dungeon",
315+
"hreflang": "x-default",
316+
},
317+
{
318+
"href": "endless-dungeon",
319+
"hreflang": "en-US",
320+
},
321+
{
322+
"href": "/fr/endless-dungeon",
323+
"hreflang": "fr-FR",
324+
},
325+
],
326+
"loc": "endless-dungeon",
327+
},
328+
{
329+
"_abs": false,
330+
"_index": 3,
331+
"_key": "english-url",
332+
"_locale": {
333+
"code": "en",
334+
"iso": "en-US",
335+
},
336+
"_path": {
337+
"hash": "",
338+
"pathname": "english-url",
339+
"search": "",
340+
},
341+
"_pathWithoutPrefix": "english-url",
342+
"_sitemap": "en-US",
343+
"alternatives": [
344+
{
345+
"href": "english-url",
346+
"hreflang": "x-default",
347+
},
348+
{
349+
"href": "english-url",
350+
"hreflang": "en",
351+
},
352+
],
353+
"loc": "english-url",
354+
},
355+
{
356+
"_abs": true,
357+
"_key": "/abc/def",
358+
"_path": {
359+
"auth": "",
360+
"hash": "",
361+
"host": "www.somedomain.com",
362+
"pathname": "/abc/def",
363+
"protocol": "https:",
364+
"search": "",
365+
Symbol(ufo:protocolRelative): false,
366+
},
367+
"loc": "https://www.somedomain.com/abc/def",
368+
},
369+
{
370+
"_abs": false,
371+
"_index": undefined,
372+
"_key": "fr-FR/fr/endless-dungeon",
373+
"_locale": {
374+
"code": "fr",
375+
"iso": "fr-FR",
376+
},
377+
"_path": {
378+
"hash": "",
379+
"pathname": "/fr/endless-dungeon",
380+
"search": "",
381+
},
382+
"_pathWithoutPrefix": "endless-dungeon",
383+
"_sitemap": "fr-FR",
384+
"alternatives": [
385+
{
386+
"href": "endless-dungeon",
387+
"hreflang": "x-default",
388+
},
389+
{
390+
"href": "endless-dungeon",
391+
"hreflang": "en-US",
392+
},
393+
{
394+
"href": "/fr/endless-dungeon",
395+
"hreflang": "fr-FR",
396+
},
397+
],
398+
"loc": "/fr/endless-dungeon",
399+
},
400+
]
401+
`)
402+
})
55403
})

‎test/unit/normalise.test.ts

+23-103
Original file line numberDiff line numberDiff line change
@@ -1,113 +1,33 @@
11
import { describe, expect, it } from 'vitest'
2-
import { fixSlashes } from 'site-config-stack/urls'
3-
import type { NitroUrlResolvers } from '../../src/runtime/types'
4-
import { normaliseSitemapUrls } from '../../src/runtime/nitro/sitemap/urlset/normalise'
5-
6-
const resolvers = {
7-
fixSlashes: (path: string) => fixSlashes(true, path),
8-
canonicalUrlResolver: (path: string) => fixSlashes(true, path),
9-
relativeBaseUrlResolver: (path: string) => path,
10-
} as NitroUrlResolvers
2+
import { preNormalizeEntry } from '../../src/runtime/nitro/sitemap/urlset/normalise'
113

124
describe('normalise', () => {
135
it('query', async () => {
14-
const normalisedWithoutSlash = await normaliseSitemapUrls([
15-
{ loc: '/query?foo=bar' },
16-
], resolvers)
6+
const normalisedWithoutSlash = preNormalizeEntry({ loc: '/query?foo=bar' })
177
expect(normalisedWithoutSlash).toMatchInlineSnapshot(`
18-
[
19-
{
20-
"_key": "/query/?foo=bar",
21-
"loc": "/query/?foo=bar",
22-
},
23-
]
8+
{
9+
"_abs": false,
10+
"_key": "/query?foo=bar",
11+
"_path": {
12+
"hash": "",
13+
"pathname": "/query",
14+
"search": "?foo=bar",
15+
},
16+
"loc": "/query?foo=bar",
17+
}
2418
`)
25-
const normalisedWithSlash = await normaliseSitemapUrls([
26-
{ loc: '/query/?foo=bar' },
27-
], resolvers)
19+
const normalisedWithSlash = preNormalizeEntry({ loc: '/query/?foo=bar' })
2820
expect(normalisedWithSlash).toMatchInlineSnapshot(`
29-
[
30-
{
31-
"_key": "/query/?foo=bar",
32-
"loc": "/query/?foo=bar",
33-
},
34-
]
35-
`)
36-
})
37-
it('sorting', async () => {
38-
const data = await normaliseSitemapUrls([
39-
{ loc: '/a' },
40-
{ loc: '/b' },
41-
{ loc: '/c' },
42-
{ loc: '/1' },
43-
{ loc: '/2' },
44-
{ loc: '/10' },
45-
], resolvers)
46-
expect(data).toMatchInlineSnapshot(`
47-
[
48-
{
49-
"_key": "/a/",
50-
"loc": "/a/",
51-
},
52-
{
53-
"_key": "/b/",
54-
"loc": "/b/",
55-
},
56-
{
57-
"_key": "/c/",
58-
"loc": "/c/",
59-
},
60-
{
61-
"_key": "/1/",
62-
"loc": "/1/",
63-
},
64-
{
65-
"_key": "/2/",
66-
"loc": "/2/",
67-
},
68-
{
69-
"_key": "/10/",
70-
"loc": "/10/",
71-
},
72-
]
73-
`)
74-
})
75-
it('sorting disabled', async () => {
76-
const data = await normaliseSitemapUrls([
77-
{ loc: '/b' },
78-
{ loc: '/a' },
79-
{ loc: '/c' },
80-
{ loc: '/1' },
81-
{ loc: '/10' },
82-
{ loc: '/2' },
83-
], resolvers)
84-
expect(data).toMatchInlineSnapshot(`
85-
[
86-
{
87-
"_key": "/b/",
88-
"loc": "/b/",
89-
},
90-
{
91-
"_key": "/a/",
92-
"loc": "/a/",
93-
},
94-
{
95-
"_key": "/c/",
96-
"loc": "/c/",
97-
},
98-
{
99-
"_key": "/1/",
100-
"loc": "/1/",
101-
},
102-
{
103-
"_key": "/10/",
104-
"loc": "/10/",
105-
},
106-
{
107-
"_key": "/2/",
108-
"loc": "/2/",
109-
},
110-
]
21+
{
22+
"_abs": false,
23+
"_key": "/query?foo=bar",
24+
"_path": {
25+
"hash": "",
26+
"pathname": "/query",
27+
"search": "?foo=bar",
28+
},
29+
"loc": "/query?foo=bar",
30+
}
11131
`)
11232
})
11333
})

0 commit comments

Comments
 (0)
Please sign in to comment.