renovate/lib/modules/platform/github/massage-markdown-links.ts

72 lines
2.5 KiB
TypeScript

import type { Content } from 'mdast';
import remark from 'remark';
import type { Plugin, Transformer } from 'unified';
import { logger } from '../../../logger';
import { regEx } from '../../../util/regex';
interface UrlMatch {
start: number;
end: number;
replaceTo: string;
}
//according to https://github.com/dead-claudia/github-limits
const urlRegex =
/(?:https?:)?(?:\/\/)?(?:www\.)?(?<!api\.)(?:to)?github\.com\/[-a-z0-9]+\/[-_a-z0-9.]+\/(?:discussions|issues|pull)\/[0-9]+(?:#[-_a-z0-9]+)?/i; // TODO #12872 (?<!re) after text not matching
function massageLink(input: string): string {
return input.replace(regEx(/(?:to)?github\.com/i), 'togithub.com');
}
function collectLinkPosition(input: string, matches: UrlMatch[]): Plugin {
const transformer = (tree: Content): void => {
const startOffset: number = tree.position?.start.offset ?? 0;
const endOffset: number = tree.position?.end.offset ?? 0;
if (tree.type === 'link') {
const substr = input.slice(startOffset, endOffset);
const url: string = tree.url;
const offset: number = startOffset + substr.lastIndexOf(url);
if (urlRegex.test(url)) {
matches.push({
start: offset,
end: offset + url.length,
replaceTo: massageLink(url),
});
}
} else if (tree.type === 'text') {
const globalUrlReg = new RegExp(urlRegex, 'gi');
const urlMatches = [...tree.value.matchAll(globalUrlReg)];
for (const match of urlMatches) {
const [url] = match;
const start = startOffset + (match.index ?? 0);
const end = start + url.length;
const newUrl = massageLink(url);
matches.push({ start, end, replaceTo: `[${url}](${newUrl})` });
}
} else if ('children' in tree) {
tree.children.forEach((child: Content) => {
transformer(child);
});
}
};
return () => transformer as Transformer;
}
export function massageMarkdownLinks(content: string): string {
try {
const rightSpaces = content.replace(content.trimEnd(), '');
const matches: UrlMatch[] = [];
remark().use(collectLinkPosition(content, matches)).processSync(content);
const result = matches.reduceRight((acc, { start, end, replaceTo }) => {
const leftPart = acc.slice(0, start);
const rightPart = acc.slice(end);
return leftPart + replaceTo + rightPart;
}, content);
return result.trimEnd() + rightSpaces;
} catch (err) /* istanbul ignore next */ {
logger.warn({ err }, `Unable to massage markdown text`);
return content;
}
}