UNPKG

@redpanda-data/docs-extensions-and-macros

Version:

Antora extensions and macros developed for Redpanda documentation.

164 lines (148 loc) 6.33 kB
'use strict'; const File = require('vinyl') const { posix: path } = require('path') const ENCODED_SPACE_RX = /%20/g module.exports.register = function () { this.once('contextStarted', () => { const { publishFiles: produceRedirectsDelegate } = this.getFunctions() this.replaceFunctions({ produceRedirects (playbook, aliases) { if ('findBy' in aliases) aliases = aliases.findBy({ family: 'alias' }) // @deprecated remove in Antora 4 if (!(aliases = aliases.filter((it) => it.pub.url !== it.rel.pub.url)).length) return [] let siteUrl = playbook.site.url if (siteUrl) siteUrl = stripTrailingSlash(siteUrl, '') const directoryRedirects = (playbook.urls.htmlExtensionStyle || 'default') !== 'default' switch (playbook.urls.redirectFacility) { case 'gitlab': return createNetlifyRedirects(aliases, extractUrlPath(siteUrl), !directoryRedirects, false) case 'httpd': return createHttpdHtaccess(aliases, extractUrlPath(siteUrl), directoryRedirects) case 'netlify': return createNetlifyRedirects(aliases, extractUrlPath(siteUrl), !directoryRedirects) case 'nginx': return createNginxRewriteConf(aliases, extractUrlPath(siteUrl)) case 'static': return populateStaticRedirectFiles( aliases.filter((it) => it.out), siteUrl ) default: return unpublish(aliases) } } }) }) } function createStaticRedirectContents (file, siteUrl) { const targetUrl = file.rel.pub.url let linkTag let to = targetUrl.charAt() === '/' ? computeRelativeUrlPath(file.pub.url, targetUrl) : undefined let toText = to if (to) { if (siteUrl && siteUrl.charAt() !== '/') { linkTag = `<link rel="canonical" href="${(toText = siteUrl + targetUrl)}">\n` } } else { linkTag = `<link rel="canonical" href="${(toText = to = targetUrl)}">\n` } return `<!DOCTYPE html> <meta charset="utf-8"> ${linkTag || ''}<script>location="${to}"</script> <meta http-equiv="refresh" content="0; url=${to}"> <meta name="robots" content="noindex"> <title>Redirect Notice</title> <h1>Redirect Notice</h1> <p>The page you requested has been relocated to <a href="${to}">${toText}</a>.</p>` } function extractUrlPath (url) { if (url) { if (url.charAt() === '/') return url const urlPath = new URL(url).pathname return urlPath === '/' ? '' : urlPath } else { return '' } } function createHttpdHtaccess (files, urlPath, directoryRedirects = false) { const rules = files.reduce((accum, file) => { delete file.out let fromUrl = file.pub.url fromUrl = ~fromUrl.indexOf('%20') ? `'${urlPath}${fromUrl.replace(ENCODED_SPACE_RX, ' ')}'` : urlPath + fromUrl let toUrl = file.rel.pub.url toUrl = ~toUrl.indexOf('%20') ? `'${urlPath}${toUrl.replace(ENCODED_SPACE_RX, ' ')}'` : urlPath + toUrl // see https://httpd.apache.org/docs/current/en/mod/mod_alias.html#redirect // NOTE: redirect rule for directory prefix does not require trailing slash if (file.pub.splat) { accum.push(`Redirect 302 ${fromUrl} ${stripTrailingSlash(toUrl)}`) } else if (directoryRedirects) { accum.push(`RedirectMatch 301 ^${regexpEscape(fromUrl)}$ ${stripTrailingSlash(toUrl)}`) } else { accum.push(`Redirect 301 ${fromUrl} ${toUrl}`) } return accum }, []) return [new File({ contents: Buffer.from(rules.join('\n') + '\n'), out: { path: '.htaccess' } })] } // NOTE: a trailing slash on the pathname will be ignored // see https://docs.netlify.com/routing/redirects/redirect-options/#trailing-slash // however, we keep it when generating the rules for clarity function createNetlifyRedirects (files, urlPath, addDirectoryRedirects = false, useForceFlag = true) { const rules = files.reduce((accum, file) => { delete file.out const fromUrl = urlPath + file.pub.url const toUrl = urlPath + file.rel.pub.url const forceFlag = useForceFlag ? '!' : '' if (file.pub.splat) { accum.push(`${fromUrl}/* ${ensureTrailingSlash(toUrl)}:splat 302${forceFlag}`) } else { accum.push(`${fromUrl} ${toUrl} 301${forceFlag}`) if (addDirectoryRedirects && fromUrl.endsWith('/index.html')) { accum.push(`${fromUrl.substr(0, fromUrl.length - 10)} ${toUrl} 301${forceFlag}`) } } return accum }, []) return [new File({ contents: Buffer.from(rules.join('\n') + '\n'), out: { path: '_redirects' } })] } function createNginxRewriteConf (files, urlPath) { const rules = files.map((file) => { delete file.out let fromUrl = file.pub.url fromUrl = ~fromUrl.indexOf('%20') ? `'${urlPath}${fromUrl.replace(ENCODED_SPACE_RX, ' ')}'` : urlPath + fromUrl let toUrl = file.rel.pub.url toUrl = ~toUrl.indexOf('%20') ? `'${urlPath}${toUrl.replace(ENCODED_SPACE_RX, ' ')}'` : urlPath + toUrl if (file.pub.splat) { const toUrlWithTrailingSlash = ensureTrailingSlash(toUrl) return `location ^~ ${fromUrl}/ { rewrite ^${regexpEscape(fromUrl)}/(.*)$ ${toUrlWithTrailingSlash}$1 redirect; }` } else { return `location = ${fromUrl} { return 301 ${toUrl}; }` } }) return [new File({ contents: Buffer.from(rules.join('\n').trim() + '\n'), out: { path: '.etc/nginx/rewrite.conf' } })] } function populateStaticRedirectFiles(files, siteUrl) { for (const file of files) { const content = createStaticRedirectContents(file, siteUrl) + '\n'; file.contents = Buffer.from(content); } return [] } function unpublish (files) { files.forEach((file) => delete file.out) return [] } function computeRelativeUrlPath (from, to) { if (to === from) return to.charAt(to.length - 1) === '/' ? './' : path.basename(to) return (path.relative(path.dirname(from + '.'), to) || '.') + (to.charAt(to.length - 1) === '/' ? '/' : '') } function ensureTrailingSlash (str) { return str.charAt(str.length - 1) === '/' ? str : str + '/' } function stripTrailingSlash (str, root = '/') { if (str === '/') return root const lastIdx = str.length - 1 return str.charAt(lastIdx) === '/' ? str.substr(0, lastIdx) : str } function regexpEscape (str) { return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&') // don't escape "-" since it's meaningless in a literal }