import { BadRequestException, Injectable, Logger } from '@nestjs/common'; import { PageRepo } from '@docmost/db/repos/page/page.repo'; import { MultipartFile } from '@fastify/multipart'; import { sanitize } from 'sanitize-filename-ts'; import * as path from 'path'; import { htmlToJson, tiptapExtensions, } from '../../collaboration/collaboration.util'; import { InjectKysely } from 'nestjs-kysely'; import { KyselyDB } from '@docmost/db/types/kysely.types'; import { generateSlugId } from '../../common/helpers'; import { generateJitteredKeyBetween } from 'fractional-indexing-jittered'; import { markdownToHtml } from './utils/marked.utils'; import { TiptapTransformer } from '@hocuspocus/transformer'; import * as Y from 'yjs'; @Injectable() export class ImportService { private readonly logger = new Logger(ImportService.name); constructor( private readonly pageRepo: PageRepo, @InjectKysely() private readonly db: KyselyDB, ) {} async importPage( filePromise: Promise, userId: string, spaceId: string, workspaceId: string, ): Promise { const file = await filePromise; const fileBuffer = await file.toBuffer(); const fileName = sanitize(file.filename).slice(0, 255).split('.')[0]; const fileExtension = path.extname(file.filename).toLowerCase(); const fileContent = fileBuffer.toString(); let prosemirrorState = null; let createdPage = null; try { if (fileExtension.endsWith('.md')) { prosemirrorState = await this.processMarkdown(fileContent); } else if (fileExtension.endsWith('.html')) { prosemirrorState = await this.processHTML(fileContent); } } catch (err) { const message = 'Error processing file content'; this.logger.error(message, err); throw new BadRequestException(message); } if (!prosemirrorState) { const message = 'Failed to create ProseMirror state'; this.logger.error(message); throw new BadRequestException(message); } const { title, prosemirrorJson } = this.extractTitleAndRemoveHeading(prosemirrorState); const pageTitle = title || fileName; if (prosemirrorJson) { try { const pagePosition = await this.getNewPagePosition(spaceId); createdPage = await this.pageRepo.insertPage({ slugId: generateSlugId(), title: pageTitle, content: prosemirrorJson, ydoc: await this.createYdoc(prosemirrorJson), position: pagePosition, spaceId: spaceId, creatorId: userId, workspaceId: workspaceId, lastUpdatedById: userId, }); this.logger.debug( `Successfully imported "${title}${fileExtension}. ID: ${createdPage.id} - SlugId: ${createdPage.slugId}"`, ); } catch (err) { const message = 'Failed to create imported page'; this.logger.error(message, err); throw new BadRequestException(message); } } return createdPage; } async processMarkdown(markdownInput: string): Promise { try { const html = await markdownToHtml(markdownInput); return this.processHTML(html); } catch (err) { throw err; } } async processHTML(htmlInput: string): Promise { try { return htmlToJson(htmlInput); } catch (err) { throw err; } } async createYdoc(prosemirrorJson: any): Promise { if (prosemirrorJson) { this.logger.debug(`Converting prosemirror json state to ydoc`); const ydoc = TiptapTransformer.toYdoc( prosemirrorJson, 'default', tiptapExtensions, ); Y.encodeStateAsUpdate(ydoc); return Buffer.from(Y.encodeStateAsUpdate(ydoc)); } return null; } extractTitleAndRemoveHeading(prosemirrorState: any) { let title = null; if ( prosemirrorState?.content?.length > 0 && prosemirrorState.content[0].type === 'heading' && prosemirrorState.content[0].attrs?.level === 1 ) { title = prosemirrorState.content[0].content[0].text; // remove h1 header node from state prosemirrorState.content.shift(); } return { title, prosemirrorJson: prosemirrorState }; } async getNewPagePosition(spaceId: string): Promise { const lastPage = await this.db .selectFrom('pages') .select(['id', 'position']) .where('spaceId', '=', spaceId) .orderBy('position', 'desc') .limit(1) .where('parentPageId', 'is', null) .executeTakeFirst(); if (lastPage) { return generateJitteredKeyBetween(lastPage.position, null); } else { return generateJitteredKeyBetween(null, null); } } }