platform/services/github/pod-github/src/sync/guest.ts
Dakshesh Jain aaf2c14e7a
Some checks are pending
CI / build (push) Waiting to run
CI / svelte-check (push) Blocked by required conditions
CI / formatting (push) Blocked by required conditions
CI / test (push) Blocked by required conditions
CI / uitest (push) Waiting to run
CI / uitest-pg (push) Waiting to run
CI / uitest-qms (push) Waiting to run
CI / docker-build (push) Blocked by required conditions
CI / dist-build (push) Blocked by required conditions
fix(github): broken images in issue description (#7534)
Signed-off-by: Dakshesh Jain <dakshesh.jain14@gmail.com>
2025-01-05 22:43:47 +07:00

109 lines
3.3 KiB
TypeScript

//
// Copyright © 2023 Hardcore Engineering Inc.
//
import { Branding, generateId, TxOperations, WorkspaceIdWithUrl } from '@hcengineering/core'
import { MarkupMarkType, MarkupNode, MarkupNodeType, traverseMarkupNode } from '@hcengineering/text'
import { getPublicLink } from '@hcengineering/server-guest-resources'
import { Task } from '@hcengineering/task'
import { generateToken } from '@hcengineering/server-token'
const githubLinkText = process.env.LINK_TEXT ?? 'Huly&reg;:'
const githubLinkTextOld = 'View in Huly'
export function hasHulyLinkText (text: string): boolean {
return text.includes(githubLinkText) || text.includes(githubLinkTextOld)
}
export function hasHulyLink (href: string, guestLink: string): boolean {
return href.includes(guestLink)
}
export async function stripGuestLink (markdown: MarkupNode): Promise<void> {
const toRemove: MarkupNode[] = []
traverseMarkupNode(markdown, (node) => {
if (node.content === undefined) {
return
}
const oldLength = node.content.length
node.content = node.content.filter((it) => it.type !== MarkupNodeType.subLink)
// sub is an inline node hence tiptap wraps it with a paragraph
// so we need to remove the parent paragraph node if it is empty
if (node.content.length !== oldLength && node.type === MarkupNodeType.paragraph) {
toRemove.push(node)
}
})
// traverse nodes once again and remove empty parent node
traverseMarkupNode(markdown, (node) => {
if (node.content === undefined) {
return
}
node.content = node.content.filter((it) => !toRemove.includes(it))
})
}
export async function appendGuestLink (
client: TxOperations,
doc: Task,
markdown: MarkupNode,
workspace: WorkspaceIdWithUrl,
branding: Branding | null
): Promise<void> {
const publicLink = await getPublicLink(doc, client, workspace, false, branding)
await stripGuestLink(markdown)
appendGuestLinkToModel(markdown, publicLink, doc.identifier)
appendGuestLinkToImage(markdown, workspace)
}
export function appendGuestLinkToModel (markdown: MarkupNode, publicLink: string, identifier: string): void {
markdown.content = [
...(markdown.content ?? []),
{
type: MarkupNodeType.paragraph,
content: [
{
type: MarkupNodeType.subLink,
content: [
{
type: MarkupNodeType.text,
text: githubLinkText.trim() + ' <b>' + identifier + '</b>',
marks: [{ type: MarkupMarkType.link, attrs: { href: publicLink, _target: '_blank' } }]
}
]
}
]
}
]
}
function findImageTags (node: MarkupNode): MarkupNode[] {
if (node.type === MarkupNodeType.paragraph) {
return node.content?.flatMap(findImageTags) ?? []
}
if (node.type === MarkupNodeType.image) {
return [node]
}
return []
}
export function appendGuestLinkToImage (markdown: MarkupNode, workspace: WorkspaceIdWithUrl): void {
const imageTags: MarkupNode[] = markdown.content?.flatMap(findImageTags) ?? []
if (imageTags.length === 0) {
return
}
const id = generateId()
const token = generateToken(id, workspace, { linkId: id, guest: 'true' })
for (const imageTag of imageTags) {
const src = imageTag.attrs?.src
if (src !== undefined && imageTag.attrs !== undefined) {
imageTag.attrs.token = token
}
}
}