2021-08-22 16:35:45 -07:00
import InteropService_Importer_Md from '../../services/interop/InteropService_Importer_Md' ;
import Note from '../../models/Note' ;
2022-03-29 00:13:13 +08:00
import Folder from '../../models/Folder' ;
import * as fs from 'fs-extra' ;
import { createTempDir , setupDatabaseAndSynchronizer , supportDir , switchClient } from '../../testing/test-utils' ;
2021-08-22 16:35:45 -07:00
import { MarkupToHtml } from '@joplin/renderer' ;
2023-12-15 13:28:09 +00:00
import { FolderEntity , NoteEntity , ResourceEntity } from '../database/types' ;
import Resource from '../../models/Resource' ;
2021-08-22 16:35:45 -07:00
2023-02-20 12:02:29 -03:00
describe ( 'InteropService_Importer_Md' , ( ) = > {
2022-03-29 00:13:13 +08:00
let tempDir : string ;
2021-08-22 16:35:45 -07:00
async function importNote ( path : string ) {
2023-11-15 10:33:20 -03:00
const newFolder = await Folder . save ( { title : 'folder' } ) ;
2021-08-22 16:35:45 -07:00
const importer = new InteropService_Importer_Md ( ) ;
2023-11-15 10:33:20 -03:00
await importer . init ( path , {
format : 'md' ,
outputFormat : 'md' ,
path ,
destinationFolder : newFolder ,
destinationFolderId : newFolder.id ,
} ) ;
importer . setMetadata ( { fileExtensions : [ 'md' ] } ) ;
await importer . exec ( { warnings : [ ] } ) ;
2023-12-15 13:28:09 +00:00
const allNotes : NoteEntity [ ] = await Note . all ( ) ;
2023-11-15 10:33:20 -03:00
return allNotes [ 0 ] ;
2021-08-22 16:35:45 -07:00
}
2022-03-29 00:13:13 +08:00
async function importNoteDirectory ( path : string ) {
const importer = new InteropService_Importer_Md ( ) ;
2023-11-15 10:33:20 -03:00
await importer . init ( path , {
format : 'md' ,
outputFormat : 'md' ,
path ,
} ) ;
2022-03-29 00:13:13 +08:00
importer . setMetadata ( { fileExtensions : [ 'md' , 'html' ] } ) ;
2023-11-15 10:33:20 -03:00
return await importer . exec ( { warnings : [ ] } ) ;
2022-03-29 00:13:13 +08:00
}
2022-11-15 10:23:50 +00:00
beforeEach ( async ( ) = > {
2021-08-22 16:35:45 -07:00
await setupDatabaseAndSynchronizer ( 1 ) ;
await switchClient ( 1 ) ;
2022-03-29 00:13:13 +08:00
tempDir = await createTempDir ( ) ;
2021-08-22 16:35:45 -07:00
} ) ;
2022-03-29 00:13:13 +08:00
afterEach ( async ( ) = > {
await fs . remove ( tempDir ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should import linked files and modify tags appropriately' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample.md ` ) ;
const tagNonExistentFile = '![does not exist](does_not_exist.png)' ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 2 ) ;
const inexistentLinkUnchanged = note . body . includes ( tagNonExistentFile ) ;
expect ( inexistentLinkUnchanged ) . toBe ( true ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should only create 1 resource for duplicate links, all tags should be updated' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-duplicate-links.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 1 ) ;
const reg = new RegExp ( items [ 0 ] . id , 'g' ) ;
const matched = note . body . match ( reg ) ;
expect ( matched . length ) . toBe ( 2 ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should import linked files and modify tags appropriately when link is also in alt text' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-link-in-alt-text.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 1 ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should passthrough unchanged if no links present' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-no-links.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 0 ) ;
expect ( note . body ) . toContain ( 'Unidentified vessel travelling at sub warp speed, bearing 235.7. Fluctuations in energy readings from it, Captain. All transporters off.' ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should import linked image with special characters in name' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-special-chars.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 3 ) ;
const noteIds = await Note . linkedNoteIds ( note . body ) ;
expect ( noteIds . length ) . toBe ( 1 ) ;
const spaceSyntaxLeft = note . body . includes ( '<../../photo sample.jpg>' ) ;
expect ( spaceSyntaxLeft ) . toBe ( false ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should import resources and notes for files' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-files.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 3 ) ;
const noteIds = await Note . linkedNoteIds ( note . body ) ;
expect ( noteIds . length ) . toBe ( 1 ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should gracefully handle reference cycles in notes' , async ( ) = > {
2023-11-15 10:33:20 -03:00
await importNoteDirectory ( ` ${ supportDir } /test_notes/md/cycle-reference ` ) ;
const [ noteA , noteB ] = await Note . all ( ) ;
2021-08-22 16:35:45 -07:00
const noteAIds = await Note . linkedNoteIds ( noteA . body ) ;
expect ( noteAIds . length ) . toBe ( 1 ) ;
const noteBIds = await Note . linkedNoteIds ( noteB . body ) ;
expect ( noteBIds . length ) . toBe ( 1 ) ;
expect ( noteAIds [ 0 ] ) . toEqual ( noteB . id ) ;
expect ( noteBIds [ 0 ] ) . toEqual ( noteA . id ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should not import resources from file:// links' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-file-links.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 0 ) ;
expect ( note . body ) . toContain ( '![sample](file://../../photo.jpg)' ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should attach resources that are missing the file extension' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-no-extension.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 1 ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should attach resources that include anchor links' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-anchor-link.md ` ) ;
const itemIds = await Note . linkedItemIds ( note . body ) ;
expect ( itemIds . length ) . toBe ( 1 ) ;
expect ( note . body ) . toContain ( ` [Section 1](:/ ${ itemIds [ 0 ] } #markdown) ` ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should attach resources that include a title' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample-link-title.md ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 3 ) ;
const noteIds = await Note . linkedNoteIds ( note . body ) ;
expect ( noteIds . length ) . toBe ( 1 ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should import notes with html file extension as html' , async ( ) = > {
2021-08-22 16:35:45 -07:00
const note = await importNote ( ` ${ supportDir } /test_notes/md/sample.html ` ) ;
const items = await Note . linkedItems ( note . body ) ;
expect ( items . length ) . toBe ( 3 ) ;
const noteIds = await Note . linkedNoteIds ( note . body ) ;
expect ( noteIds . length ) . toBe ( 1 ) ;
expect ( note . markup_language ) . toBe ( MarkupToHtml . MARKUP_LANGUAGE_HTML ) ;
const preservedAlt = note . body . includes ( 'alt="../../photo.jpg"' ) ;
expect ( preservedAlt ) . toBe ( true ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should import non-empty directory' , async ( ) = > {
2022-03-29 00:13:13 +08:00
await fs . mkdirp ( ` ${ tempDir } /non-empty/non-empty ` ) ;
await fs . writeFile ( ` ${ tempDir } /non-empty/non-empty/sample.md ` , '# Sample' ) ;
await importNoteDirectory ( ` ${ tempDir } /non-empty ` ) ;
const allFolders = await Folder . all ( ) ;
expect ( allFolders . map ( ( f : FolderEntity ) = > f . title ) . indexOf ( 'non-empty' ) ) . toBeGreaterThanOrEqual ( 0 ) ;
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should not import empty directory' , async ( ) = > {
2023-11-15 10:33:20 -03:00
await fs . mkdirp ( ` ${ tempDir } /empty1/empty2 ` ) ;
2022-03-29 00:13:13 +08:00
2023-11-15 10:33:20 -03:00
await importNoteDirectory ( ` ${ tempDir } /empty1 ` ) ;
2022-03-29 00:13:13 +08:00
const allFolders = await Folder . all ( ) ;
2023-11-15 10:33:20 -03:00
expect ( allFolders . map ( ( f : FolderEntity ) = > f . title ) . indexOf ( 'empty1' ) ) . toBe ( 0 ) ;
expect ( allFolders . map ( ( f : FolderEntity ) = > f . title ) . indexOf ( 'empty2' ) ) . toBe ( - 1 ) ;
2022-03-29 00:13:13 +08:00
} ) ;
2023-02-20 12:02:29 -03:00
it ( 'should import directory with non-empty subdirectory' , async ( ) = > {
2022-03-29 00:13:13 +08:00
await fs . mkdirp ( ` ${ tempDir } /non-empty-subdir/non-empty-subdir/subdir-empty ` ) ;
await fs . mkdirp ( ` ${ tempDir } /non-empty-subdir/non-empty-subdir/subdir-non-empty ` ) ;
await fs . writeFile ( ` ${ tempDir } /non-empty-subdir/non-empty-subdir/subdir-non-empty/sample.md ` , '# Sample' ) ;
await importNoteDirectory ( ` ${ tempDir } /non-empty-subdir ` ) ;
const allFolders = await Folder . all ( ) ;
expect ( allFolders . map ( ( f : FolderEntity ) = > f . title ) . indexOf ( 'non-empty-subdir' ) ) . toBeGreaterThanOrEqual ( 0 ) ;
expect ( allFolders . map ( ( f : FolderEntity ) = > f . title ) . indexOf ( 'subdir-empty' ) ) . toBe ( - 1 ) ;
expect ( allFolders . map ( ( f : FolderEntity ) = > f . title ) . indexOf ( 'subdir-non-empty' ) ) . toBeGreaterThanOrEqual ( 0 ) ;
} ) ;
2023-11-15 10:33:20 -03:00
it ( 'should import all files before replacing links' , async ( ) = > {
await fs . mkdirp ( ` ${ tempDir } /links/0/1/2 ` ) ;
await fs . mkdirp ( ` ${ tempDir } /links/Target_folder ` ) ;
await fs . writeFile ( ` ${ tempDir } /links/Target_folder/Targeted_note.md ` , '# Targeted_note' ) ;
await fs . writeFile ( ` ${ tempDir } /links/0/1/2/Note_with_reference_to_another_note.md ` , '# 20\n[Target_folder:Targeted_note](../../../Target_folder/Targeted_note.md)' ) ;
await importNoteDirectory ( ` ${ tempDir } /links ` ) ;
const allFolders = await Folder . all ( ) ;
const allNotes = await Note . all ( ) ;
const targetFolder = allFolders . find ( f = > f . title === 'Target_folder' ) ;
const noteBeingReferenced = allNotes . find ( n = > n . title === 'Targeted_note' ) ;
expect ( noteBeingReferenced . parent_id ) . toBe ( targetFolder . id ) ;
} ) ;
2023-12-15 13:28:09 +00:00
it ( 'should not fail to import file that contains a link to a file that does not exist' , async ( ) = > {
// The first implicit test is that the below call doesn't throw due to the invalid image
const note = await importNote ( ` ${ supportDir } /test_notes/md/invalid-image-link.md ` ) ;
const links = Note . linkedItemIds ( note . body ) ;
expect ( links . length ) . toBe ( 1 ) ;
const resource : ResourceEntity = await Resource . load ( links [ 0 ] ) ;
// The invalid image is imported as-is
expect ( resource . title ) . toBe ( 'invalid-image.jpg' ) ;
} ) ;
2021-08-22 16:35:45 -07:00
} ) ;