mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-02 12:47:41 +02:00
199 lines
8.6 KiB
TypeScript
199 lines
8.6 KiB
TypeScript
import InteropService_Importer_Md from '../../services/interop/InteropService_Importer_Md';
|
|
import Note from '../../models/Note';
|
|
import Folder from '../../models/Folder';
|
|
import * as fs from 'fs-extra';
|
|
import { createTempDir, setupDatabaseAndSynchronizer, supportDir, switchClient } from '../../testing/test-utils';
|
|
import { MarkupToHtml } from '@joplin/renderer';
|
|
import { FolderEntity, NoteEntity, ResourceEntity } from '../database/types';
|
|
import Resource from '../../models/Resource';
|
|
|
|
|
|
describe('InteropService_Importer_Md', () => {
|
|
let tempDir: string;
|
|
async function importNote(path: string) {
|
|
const newFolder = await Folder.save({ title: 'folder' });
|
|
const importer = new InteropService_Importer_Md();
|
|
await importer.init(path, {
|
|
format: 'md',
|
|
outputFormat: 'md',
|
|
path,
|
|
destinationFolder: newFolder,
|
|
destinationFolderId: newFolder.id,
|
|
});
|
|
importer.setMetadata({ fileExtensions: ['md'] });
|
|
await importer.exec({ warnings: [] });
|
|
const allNotes: NoteEntity[] = await Note.all();
|
|
return allNotes[0];
|
|
}
|
|
async function importNoteDirectory(path: string) {
|
|
const importer = new InteropService_Importer_Md();
|
|
await importer.init(path, {
|
|
format: 'md',
|
|
outputFormat: 'md',
|
|
path,
|
|
});
|
|
importer.setMetadata({ fileExtensions: ['md', 'html'] });
|
|
return await importer.exec({ warnings: [] });
|
|
}
|
|
beforeEach(async () => {
|
|
await setupDatabaseAndSynchronizer(1);
|
|
await switchClient(1);
|
|
tempDir = await createTempDir();
|
|
});
|
|
afterEach(async () => {
|
|
await fs.remove(tempDir);
|
|
});
|
|
it('should import linked files and modify tags appropriately', async () => {
|
|
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);
|
|
});
|
|
it('should only create 1 resource for duplicate links, all tags should be updated', async () => {
|
|
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);
|
|
});
|
|
it('should import linked files and modify tags appropriately when link is also in alt text', async () => {
|
|
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);
|
|
});
|
|
it('should passthrough unchanged if no links present', async () => {
|
|
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.');
|
|
});
|
|
it('should import linked image with special characters in name', async () => {
|
|
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);
|
|
});
|
|
it('should import resources and notes for files', async () => {
|
|
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);
|
|
});
|
|
it('should gracefully handle reference cycles in notes', async () => {
|
|
await importNoteDirectory(`${supportDir}/test_notes/md/cycle-reference`);
|
|
const [noteA, noteB] = await Note.all();
|
|
|
|
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);
|
|
});
|
|
it('should not import resources from file:// links', async () => {
|
|
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)');
|
|
});
|
|
it('should attach resources that are missing the file extension', async () => {
|
|
const note = await importNote(`${supportDir}/test_notes/md/sample-no-extension.md`);
|
|
|
|
const items = await Note.linkedItems(note.body);
|
|
expect(items.length).toBe(1);
|
|
});
|
|
it('should attach resources that include anchor links', async () => {
|
|
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)`);
|
|
});
|
|
it('should attach resources that include a title', async () => {
|
|
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);
|
|
});
|
|
it('should import notes with html file extension as html', async () => {
|
|
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);
|
|
});
|
|
it('should import non-empty directory', async () => {
|
|
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);
|
|
});
|
|
it('should not import empty directory', async () => {
|
|
await fs.mkdirp(`${tempDir}/empty1/empty2`);
|
|
|
|
await importNoteDirectory(`${tempDir}/empty1`);
|
|
const allFolders = await Folder.all();
|
|
expect(allFolders.map((f: FolderEntity) => f.title).indexOf('empty1')).toBe(0);
|
|
expect(allFolders.map((f: FolderEntity) => f.title).indexOf('empty2')).toBe(-1);
|
|
});
|
|
it('should import directory with non-empty subdirectory', async () => {
|
|
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);
|
|
});
|
|
|
|
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);
|
|
});
|
|
|
|
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');
|
|
});
|
|
});
|