From be35fa6cf63f9e803f015c10133bb0a62d4a5b18 Mon Sep 17 00:00:00 2001 From: Manuel Leduc Date: Tue, 14 Jan 2025 10:58:06 +0100 Subject: [PATCH] CRISTAL-406: Selecting an internal link still displays a full url - Fix a problem with external link on Nextcloud and File System backends - Add more checks to prevent issues in case of bad syntax --- .../src/parseInternalLinks.ts | 68 ++++++++++++++----- .../src/filesystemModelReferenceParser.ts | 3 + .../src/nextcloudModelReferenceParser.ts | 3 + 3 files changed, 58 insertions(+), 16 deletions(-) diff --git a/core/markdown/markdown-default/src/parseInternalLinks.ts b/core/markdown/markdown-default/src/parseInternalLinks.ts index 9e3d058c1..738dbe51b 100644 --- a/core/markdown/markdown-default/src/parseInternalLinks.ts +++ b/core/markdown/markdown-default/src/parseInternalLinks.ts @@ -87,6 +87,53 @@ function parseStringForInternalLinks( return tokens; } +function referenceToURL( + remoteURLSerializer: RemoteURLSerializer, + modelReferenceParser: ModelReferenceParser, + reference: string, +) { + return remoteURLSerializer.serialize(modelReferenceParser.parse(reference)); +} + +function parseReference( + v: InternalLink, + remoteURLSerializer: RemoteURLSerializer, + modelReferenceParser: ModelReferenceParser, +) { + const { text, reference } = v; + let href: string; + try { + href = + referenceToURL(remoteURLSerializer, modelReferenceParser, reference) ?? + ""; + } catch { + // Prevent the parser from failing in case of bad reference (e.g., an url in an internal link). + href = reference; + } + return { text, reference, href }; +} + +function handleLink( + v: InternalLink, + remoteURLSerializer: RemoteURLSerializer, + modelReferenceParser: ModelReferenceParser, + state: StateCore, +) { + const { text, reference, href } = parseReference( + v, + remoteURLSerializer, + modelReferenceParser, + ); + + const openToken = new state.Token("link_open", "a", 1); + openToken.attrSet("href", href ?? ""); + openToken.attrPush(["class", "internal-link"]); + const contentToken = new state.Token("text", "", 0); + contentToken.content = text ?? reference; + const closeToken = new state.Token("link_close", "a", -1); + return [openToken, contentToken, closeToken]; +} + export function parseInternalLinks( modelReferenceParser: ModelReferenceParser, remoteURLSerializer: RemoteURLSerializer, @@ -100,29 +147,18 @@ export function parseInternalLinks( // We replace the content of the current block node only if at least a link has been found. if (hasLink(internalTokens)) { blockToken.content = ""; - // TODO: reduce the number of statements in the following method and reactivate the disabled eslint - // rule. - // eslint-disable-next-line max-statements blockToken.children = internalTokens.flatMap((v) => { if (typeof v == "string") { const token = new state.Token("text", "span", 0); token.content = v; return [token]; } else { - const { text, reference } = v; - - const openToken = new state.Token("link_open", "a", 1); - openToken.attrSet( - "href", - remoteURLSerializer.serialize( - modelReferenceParser.parse(reference), - ) ?? "", + return handleLink( + v, + remoteURLSerializer, + modelReferenceParser, + state, ); - openToken.attrPush(["class", "internal-link"]); - const contentToken = new state.Token("text", "", 0); - contentToken.content = text || reference; - const closeToken = new state.Token("link_close", "a", -1); - return [openToken, contentToken, closeToken]; } }); } diff --git a/core/model/model-reference/model-reference-filesystem/src/filesystemModelReferenceParser.ts b/core/model/model-reference/model-reference-filesystem/src/filesystemModelReferenceParser.ts index 567140753..f3ecbca49 100644 --- a/core/model/model-reference/model-reference-filesystem/src/filesystemModelReferenceParser.ts +++ b/core/model/model-reference/model-reference-filesystem/src/filesystemModelReferenceParser.ts @@ -30,6 +30,9 @@ import { injectable } from "inversify"; @injectable() export class FileSystemModelReferenceParser implements ModelReferenceParser { parse(reference: string): EntityReference { + if (/^https?:\/\//.test(reference)) { + throw new Error(`[${reference}] is not a valid entity reference`); + } const segments = reference.split(/(?